| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217 |
- /*
- Simple DirectMedia Layer
- Copyright (C) 1997-2014 Sam Lantinga <[email protected]>
- This software is provided 'as-is', without any express or implied
- warranty. In no event will the authors be held liable for any damages
- arising from the use of this software.
- Permission is granted to anyone to use this software for any purpose,
- including commercial applications, and to alter it and redistribute it
- freely, subject to the following restrictions:
- 1. The origin of this software must not be misrepresented; you must not
- claim that you wrote the original software. If you use this software
- in a product, an acknowledgment in the product documentation would be
- appreciated but is not required.
- 2. Altered source versions must be plainly marked as such, and must not be
- misrepresented as being the original software.
- 3. This notice may not be removed or altered from any source distribution.
- */
- #include "../../SDL_internal.h"
- /* An implementation of semaphores using mutexes and condition variables */
- #include "SDL_timer.h"
- #include "SDL_thread.h"
- #include "SDL_systhread_c.h"
- #if SDL_THREADS_DISABLED
- SDL_sem *
- SDL_CreateSemaphore(Uint32 initial_value)
- {
- SDL_SetError("SDL not built with thread support");
- return (SDL_sem *) 0;
- }
- void
- SDL_DestroySemaphore(SDL_sem * sem)
- {
- }
- int
- SDL_SemTryWait(SDL_sem * sem)
- {
- return SDL_SetError("SDL not built with thread support");
- }
- int
- SDL_SemWaitTimeout(SDL_sem * sem, Uint32 timeout)
- {
- return SDL_SetError("SDL not built with thread support");
- }
- int
- SDL_SemWait(SDL_sem * sem)
- {
- return SDL_SetError("SDL not built with thread support");
- }
- Uint32
- SDL_SemValue(SDL_sem * sem)
- {
- return 0;
- }
- int
- SDL_SemPost(SDL_sem * sem)
- {
- return SDL_SetError("SDL not built with thread support");
- }
- #else
- struct SDL_semaphore
- {
- Uint32 count;
- Uint32 waiters_count;
- SDL_mutex *count_lock;
- SDL_cond *count_nonzero;
- };
- SDL_sem *
- SDL_CreateSemaphore(Uint32 initial_value)
- {
- SDL_sem *sem;
- sem = (SDL_sem *) SDL_malloc(sizeof(*sem));
- if (!sem) {
- SDL_OutOfMemory();
- return NULL;
- }
- sem->count = initial_value;
- sem->waiters_count = 0;
- sem->count_lock = SDL_CreateMutex();
- sem->count_nonzero = SDL_CreateCond();
- if (!sem->count_lock || !sem->count_nonzero) {
- SDL_DestroySemaphore(sem);
- return NULL;
- }
- return sem;
- }
- /* WARNING:
- You cannot call this function when another thread is using the semaphore.
- */
- void
- SDL_DestroySemaphore(SDL_sem * sem)
- {
- if (sem) {
- sem->count = 0xFFFFFFFF;
- while (sem->waiters_count > 0) {
- SDL_CondSignal(sem->count_nonzero);
- SDL_Delay(10);
- }
- SDL_DestroyCond(sem->count_nonzero);
- if (sem->count_lock) {
- SDL_LockMutex(sem->count_lock);
- SDL_UnlockMutex(sem->count_lock);
- SDL_DestroyMutex(sem->count_lock);
- }
- SDL_free(sem);
- }
- }
- int
- SDL_SemTryWait(SDL_sem * sem)
- {
- int retval;
- if (!sem) {
- return SDL_SetError("Passed a NULL semaphore");
- }
- retval = SDL_MUTEX_TIMEDOUT;
- SDL_LockMutex(sem->count_lock);
- if (sem->count > 0) {
- --sem->count;
- retval = 0;
- }
- SDL_UnlockMutex(sem->count_lock);
- return retval;
- }
- int
- SDL_SemWaitTimeout(SDL_sem * sem, Uint32 timeout)
- {
- int retval;
- if (!sem) {
- return SDL_SetError("Passed a NULL semaphore");
- }
- /* A timeout of 0 is an easy case */
- if (timeout == 0) {
- return SDL_SemTryWait(sem);
- }
- SDL_LockMutex(sem->count_lock);
- ++sem->waiters_count;
- retval = 0;
- while ((sem->count == 0) && (retval != SDL_MUTEX_TIMEDOUT)) {
- retval = SDL_CondWaitTimeout(sem->count_nonzero,
- sem->count_lock, timeout);
- }
- --sem->waiters_count;
- if (retval == 0) {
- --sem->count;
- }
- SDL_UnlockMutex(sem->count_lock);
- return retval;
- }
- int
- SDL_SemWait(SDL_sem * sem)
- {
- return SDL_SemWaitTimeout(sem, SDL_MUTEX_MAXWAIT);
- }
- Uint32
- SDL_SemValue(SDL_sem * sem)
- {
- Uint32 value;
- value = 0;
- if (sem) {
- SDL_LockMutex(sem->count_lock);
- value = sem->count;
- SDL_UnlockMutex(sem->count_lock);
- }
- return value;
- }
- int
- SDL_SemPost(SDL_sem * sem)
- {
- if (!sem) {
- return SDL_SetError("Passed a NULL semaphore");
- }
- SDL_LockMutex(sem->count_lock);
- if (sem->waiters_count > 0) {
- SDL_CondSignal(sem->count_nonzero);
- }
- ++sem->count;
- SDL_UnlockMutex(sem->count_lock);
- return 0;
- }
- #endif /* SDL_THREADS_DISABLED */
- /* vi: set ts=4 sw=4 expandtab: */
|