Use lightweight mutex

This commit is contained in:
Francisco Javier Trujillo Mata 2021-11-23 15:27:41 +01:00 committed by Sam Lantinga
parent 0341ec93fe
commit 996f30b0e4

View file

@ -27,30 +27,37 @@
#include "SDL_thread.h" #include "SDL_thread.h"
#include "SDL_systhread_c.h" #include "SDL_systhread_c.h"
#include <pspthreadman.h>
#include <pspkerror.h>
#define SCE_KERNEL_MUTEX_ATTR_RECURSIVE 0x0200U
struct SDL_mutex struct SDL_mutex
{ {
int recursive; SceLwMutexWorkarea lock;
SDL_threadID owner;
SDL_sem *sem;
}; };
/* Create a mutex */ /* Create a mutex */
SDL_mutex * SDL_mutex *
SDL_CreateMutex(void) SDL_CreateMutex(void)
{ {
SDL_mutex *mutex; SDL_mutex *mutex = NULL;
SceInt32 res = 0;
/* Allocate mutex memory */ /* Allocate mutex memory */
mutex = (SDL_mutex *) SDL_malloc(sizeof(*mutex)); mutex = (SDL_mutex *) SDL_malloc(sizeof(*mutex));
if (mutex) { if (mutex) {
/* Create the mutex semaphore, with initial value 1 */
mutex->sem = SDL_CreateSemaphore(1); res = sceKernelCreateLwMutex(
mutex->recursive = 0; &mutex->lock,
mutex->owner = 0; "SDL mutex",
if (!mutex->sem) { SCE_KERNEL_MUTEX_ATTR_RECURSIVE,
SDL_free(mutex); 0,
mutex = NULL; NULL
);
if (res < 0) {
SDL_SetError("Error trying to create mutex: %x", res);
} }
} else { } else {
SDL_OutOfMemory(); SDL_OutOfMemory();
@ -63,37 +70,56 @@ void
SDL_DestroyMutex(SDL_mutex * mutex) SDL_DestroyMutex(SDL_mutex * mutex)
{ {
if (mutex) { if (mutex) {
if (mutex->sem) { sceKernelDeleteLwMutex(&mutex->lock);
SDL_DestroySemaphore(mutex->sem);
}
SDL_free(mutex); SDL_free(mutex);
} }
} }
/* Lock the semaphore */ /* Try to lock the mutex */
int
SDL_TryLockMutex(SDL_mutex * mutex)
{
#if SDL_THREADS_DISABLED
return 0;
#else
SceInt32 res = 0;
if (mutex == NULL) {
return SDL_SetError("Passed a NULL mutex");
}
res = sceKernelTryLockLwMutex(&mutex->lock, 1);
switch (res) {
case SCE_KERNEL_ERROR_OK:
return 0;
break;
case SCE_KERNEL_ERROR_WAIT_TIMEOUT:
return SDL_MUTEX_TIMEDOUT;
break;
default:
return SDL_SetError("Error trying to lock mutex: %x", res);
break;
}
return -1;
#endif /* SDL_THREADS_DISABLED */
}
/* Lock the mutex */
int int
SDL_mutexP(SDL_mutex * mutex) SDL_mutexP(SDL_mutex * mutex)
{ {
#if SDL_THREADS_DISABLED #if SDL_THREADS_DISABLED
return 0; return 0;
#else #else
SDL_threadID this_thread; SceInt32 res = 0;
if (mutex == NULL) { if (mutex == NULL) {
return SDL_SetError("Passed a NULL mutex"); return SDL_SetError("Passed a NULL mutex");
} }
this_thread = SDL_ThreadID(); res = sceKernelLockLwMutex(&mutex->lock, 1, NULL);
if (mutex->owner == this_thread) { if (res != SCE_KERNEL_ERROR_OK) {
++mutex->recursive; return SDL_SetError("Error trying to lock mutex: %x", res);
} else {
/* The order of operations is important.
We set the locking thread id after we obtain the lock
so unlocks from other threads will fail.
*/
SDL_SemWait(mutex->sem);
mutex->owner = this_thread;
mutex->recursive = 0;
} }
return 0; return 0;
@ -107,30 +133,20 @@ SDL_mutexV(SDL_mutex * mutex)
#if SDL_THREADS_DISABLED #if SDL_THREADS_DISABLED
return 0; return 0;
#else #else
SceInt32 res = 0;
if (mutex == NULL) { if (mutex == NULL) {
return SDL_SetError("Passed a NULL mutex"); return SDL_SetError("Passed a NULL mutex");
} }
/* If we don't own the mutex, we can't unlock it */ res = sceKernelUnlockLwMutex(&mutex->lock, 1);
if (SDL_ThreadID() != mutex->owner) { if (res != 0) {
return SDL_SetError("mutex not owned by this thread"); return SDL_SetError("Error trying to unlock mutex: %x", res);
} }
if (mutex->recursive) {
--mutex->recursive;
} else {
/* The order of operations is important.
First reset the owner so another thread doesn't lock
the mutex and set the ownership before we reset it,
then release the lock semaphore.
*/
mutex->owner = 0;
SDL_SemPost(mutex->sem);
}
return 0; return 0;
#endif /* SDL_THREADS_DISABLED */ #endif /* SDL_THREADS_DISABLED */
} }
#endif /* SDL_THREAD_PSP */ #endif /* SDL_THREAD_PSP */
/* vi: set ts=4 sw=4 expandtab: */ /* vi: set ts=4 sw=4 expandtab: */