2015-07-05 21:00:13 +00:00
|
|
|
/* Copyright (c) 2013-2015 Jeffrey Pfau
|
|
|
|
*
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#ifndef SCE_THREADING_H
|
|
|
|
#define SCE_THREADING_H
|
|
|
|
|
|
|
|
#include <psp2/kernel/threadmgr.h>
|
|
|
|
|
|
|
|
typedef SceUID Thread;
|
2024-04-21 10:01:33 +00:00
|
|
|
typedef SceKernelLwMutexWork Mutex;
|
2015-07-05 21:00:13 +00:00
|
|
|
typedef struct {
|
|
|
|
Mutex mutex;
|
|
|
|
SceUID semaphore;
|
|
|
|
int waiting;
|
|
|
|
} Condition;
|
|
|
|
#define THREAD_ENTRY int
|
|
|
|
typedef THREAD_ENTRY (*ThreadEntry)(void*);
|
2023-05-10 05:21:55 +00:00
|
|
|
#define THREAD_EXIT(RES) return RES
|
2015-07-05 21:00:13 +00:00
|
|
|
|
|
|
|
static inline int MutexInit(Mutex* mutex) {
|
2024-04-21 10:01:33 +00:00
|
|
|
return sceKernelCreateLwMutex(mutex, "mutex", 0, 0, 0);
|
2015-07-05 21:00:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline int MutexDeinit(Mutex* mutex) {
|
2024-04-21 10:01:33 +00:00
|
|
|
return sceKernelDeleteLwMutex(mutex);
|
2015-07-05 21:00:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline int MutexLock(Mutex* mutex) {
|
2024-04-21 10:01:33 +00:00
|
|
|
return sceKernelLockLwMutex(mutex, 1, 0);
|
2015-07-05 21:00:13 +00:00
|
|
|
}
|
|
|
|
|
2015-11-13 06:50:09 +00:00
|
|
|
static inline int MutexTryLock(Mutex* mutex) {
|
2024-04-21 10:01:33 +00:00
|
|
|
return sceKernelTryLockLwMutex(mutex, 1);
|
2015-11-13 06:50:09 +00:00
|
|
|
}
|
|
|
|
|
2015-07-05 21:00:13 +00:00
|
|
|
static inline int MutexUnlock(Mutex* mutex) {
|
2024-04-21 10:01:33 +00:00
|
|
|
return sceKernelUnlockLwMutex(mutex, 1);
|
2015-07-05 21:00:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline int ConditionInit(Condition* cond) {
|
|
|
|
int res = MutexInit(&cond->mutex);
|
|
|
|
if (res < 0) {
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
cond->semaphore = sceKernelCreateSema("SceCondSema", 0, 0, 1, 0);
|
|
|
|
if (cond->semaphore < 0) {
|
|
|
|
MutexDeinit(&cond->mutex);
|
|
|
|
res = cond->semaphore;
|
|
|
|
}
|
|
|
|
cond->waiting = 0;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int ConditionDeinit(Condition* cond) {
|
|
|
|
MutexDeinit(&cond->mutex);
|
|
|
|
return sceKernelDeleteSema(cond->semaphore);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int ConditionWait(Condition* cond, Mutex* mutex) {
|
2015-07-06 04:22:54 +00:00
|
|
|
int ret = MutexLock(&cond->mutex);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
2015-07-05 21:00:13 +00:00
|
|
|
++cond->waiting;
|
|
|
|
MutexUnlock(mutex);
|
|
|
|
MutexUnlock(&cond->mutex);
|
2015-07-06 04:22:54 +00:00
|
|
|
ret = sceKernelWaitSema(cond->semaphore, 1, 0);
|
|
|
|
if (ret < 0) {
|
|
|
|
printf("Premature wakeup: %08X", ret);
|
|
|
|
}
|
2015-07-05 21:00:13 +00:00
|
|
|
MutexLock(mutex);
|
2015-07-06 04:22:54 +00:00
|
|
|
return ret;
|
2015-07-05 21:00:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline int ConditionWaitTimed(Condition* cond, Mutex* mutex, int32_t timeoutMs) {
|
2015-07-06 04:22:54 +00:00
|
|
|
int ret = MutexLock(&cond->mutex);
|
|
|
|
if (ret < 0) {
|
|
|
|
return ret;
|
|
|
|
}
|
2015-07-05 21:00:13 +00:00
|
|
|
++cond->waiting;
|
|
|
|
MutexUnlock(mutex);
|
|
|
|
MutexUnlock(&cond->mutex);
|
|
|
|
SceUInt timeout = 0;
|
|
|
|
if (timeoutMs > 0) {
|
|
|
|
timeout = timeoutMs;
|
|
|
|
}
|
2015-07-06 04:22:54 +00:00
|
|
|
ret = sceKernelWaitSema(cond->semaphore, 1, &timeout);
|
|
|
|
if (ret < 0) {
|
|
|
|
printf("Premature wakeup: %08X", ret);
|
|
|
|
}
|
2015-07-05 21:00:13 +00:00
|
|
|
MutexLock(mutex);
|
2015-07-06 01:19:26 +00:00
|
|
|
return ret;
|
2015-07-05 21:00:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline int ConditionWake(Condition* cond) {
|
|
|
|
MutexLock(&cond->mutex);
|
|
|
|
if (cond->waiting) {
|
|
|
|
--cond->waiting;
|
|
|
|
sceKernelSignalSema(cond->semaphore, 1);
|
|
|
|
}
|
|
|
|
MutexUnlock(&cond->mutex);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct SceThreadEntryArgs {
|
|
|
|
void* context;
|
|
|
|
ThreadEntry entry;
|
|
|
|
};
|
|
|
|
|
|
|
|
static inline int _sceThreadEntry(SceSize args, void* argp) {
|
|
|
|
UNUSED(args);
|
|
|
|
struct SceThreadEntryArgs* arg = argp;
|
|
|
|
return arg->entry(arg->context);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int ThreadCreate(Thread* thread, ThreadEntry entry, void* context) {
|
2016-07-24 20:40:45 +00:00
|
|
|
Thread id = sceKernelCreateThread("SceThread", _sceThreadEntry, 0x10000100, 0x10000, 0, 0, 0);
|
2015-07-05 21:00:13 +00:00
|
|
|
if (id < 0) {
|
2015-07-26 23:13:18 +00:00
|
|
|
*thread = 0;
|
2015-07-05 21:00:13 +00:00
|
|
|
return id;
|
|
|
|
}
|
|
|
|
*thread = id;
|
|
|
|
struct SceThreadEntryArgs args = { context, entry };
|
|
|
|
sceKernelStartThread(id, sizeof(args), &args);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-05-26 21:26:35 +00:00
|
|
|
static inline int ThreadJoin(Thread* thread) {
|
|
|
|
int res = sceKernelWaitThreadEnd(*thread, 0, 0);
|
2016-08-10 06:35:38 +00:00
|
|
|
if (res < 0) {
|
|
|
|
return res;
|
|
|
|
}
|
2019-05-26 21:26:35 +00:00
|
|
|
return sceKernelDeleteThread(*thread);
|
2015-07-05 21:00:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline int ThreadSetName(const char* name) {
|
|
|
|
UNUSED(name);
|
|
|
|
return -1;
|
|
|
|
}
|
2020-07-12 23:57:09 +00:00
|
|
|
|
2022-08-08 20:05:32 +00:00
|
|
|
#if (__STDC_VERSION__ < 201112L) || (__STDC_NO_THREADS__ == 1)
|
|
|
|
typedef int ThreadLocal;
|
|
|
|
|
2020-07-12 23:57:09 +00:00
|
|
|
static inline void ThreadLocalInitKey(ThreadLocal* key) {
|
|
|
|
static int base = 0x90;
|
|
|
|
*key = __atomic_fetch_add(&base, 1, __ATOMIC_SEQ_CST);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void ThreadLocalSetKey(ThreadLocal key, void* value) {
|
|
|
|
void** tls = sceKernelGetTLSAddr(key);
|
|
|
|
*tls = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void* ThreadLocalGetValue(ThreadLocal key) {
|
|
|
|
void** tls = sceKernelGetTLSAddr(key);
|
|
|
|
return *tls;
|
|
|
|
}
|
2015-07-05 21:00:13 +00:00
|
|
|
#endif
|
2022-08-08 20:05:32 +00:00
|
|
|
#endif
|