2001-04-26 16:45:43 +00:00
|
|
|
/*
|
|
|
|
SDL - Simple DirectMedia Layer
|
2006-02-01 06:32:25 +00:00
|
|
|
Copyright (C) 1997-2006 Sam Lantinga
|
2001-04-26 16:45:43 +00:00
|
|
|
|
|
|
|
This library is free software; you can redistribute it and/or
|
2006-02-01 06:32:25 +00:00
|
|
|
modify it under the terms of the GNU Lesser General Public
|
2001-04-26 16:45:43 +00:00
|
|
|
License as published by the Free Software Foundation; either
|
2006-02-01 06:32:25 +00:00
|
|
|
version 2.1 of the License, or (at your option) any later version.
|
2001-04-26 16:45:43 +00:00
|
|
|
|
|
|
|
This library is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
2006-02-01 06:32:25 +00:00
|
|
|
Lesser General Public License for more details.
|
2001-04-26 16:45:43 +00:00
|
|
|
|
2006-02-01 06:32:25 +00:00
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
|
|
License along with this library; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
2001-04-26 16:45:43 +00:00
|
|
|
|
|
|
|
Sam Lantinga
|
2001-12-14 12:38:15 +00:00
|
|
|
slouken@libsdl.org
|
2001-04-26 16:45:43 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* Semaphore functions using the Win32 API */
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <windows.h>
|
|
|
|
|
|
|
|
#include "SDL_error.h"
|
|
|
|
#include "SDL_thread.h"
|
|
|
|
#if defined(_WIN32_WCE) && (_WIN32_WCE < 300)
|
2001-05-23 23:35:10 +00:00
|
|
|
#include "win_ce_semaphore.h"
|
|
|
|
#endif
|
2001-04-26 16:45:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
struct SDL_semaphore {
|
2001-05-23 23:35:10 +00:00
|
|
|
#if defined(_WIN32_WCE) && (_WIN32_WCE < 300)
|
|
|
|
SYNCHHANDLE id;
|
|
|
|
#else
|
2001-04-26 16:45:43 +00:00
|
|
|
HANDLE id;
|
2001-05-23 23:35:10 +00:00
|
|
|
#endif
|
2001-04-26 16:45:43 +00:00
|
|
|
Uint32 volatile count;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/* Create a semaphore */
|
|
|
|
SDL_sem *SDL_CreateSemaphore(Uint32 initial_value)
|
|
|
|
{
|
|
|
|
SDL_sem *sem;
|
|
|
|
|
|
|
|
/* Allocate sem memory */
|
|
|
|
sem = (SDL_sem *)malloc(sizeof(*sem));
|
|
|
|
if ( sem ) {
|
|
|
|
/* Create the semaphore, with max value 32K */
|
2001-05-23 23:35:10 +00:00
|
|
|
#if defined(_WIN32_WCE) && (_WIN32_WCE < 300)
|
|
|
|
sem->id = CreateSemaphoreCE(NULL, initial_value, 32*1024, NULL);
|
|
|
|
#else
|
2001-04-26 16:45:43 +00:00
|
|
|
sem->id = CreateSemaphore(NULL, initial_value, 32*1024, NULL);
|
2001-05-23 23:35:10 +00:00
|
|
|
#endif
|
2001-04-26 16:45:43 +00:00
|
|
|
sem->count = initial_value;
|
|
|
|
if ( ! sem->id ) {
|
|
|
|
SDL_SetError("Couldn't create semaphore");
|
|
|
|
free(sem);
|
|
|
|
sem = NULL;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
SDL_OutOfMemory();
|
|
|
|
}
|
|
|
|
return(sem);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Free the semaphore */
|
|
|
|
void SDL_DestroySemaphore(SDL_sem *sem)
|
|
|
|
{
|
|
|
|
if ( sem ) {
|
|
|
|
if ( sem->id ) {
|
2001-05-23 23:35:10 +00:00
|
|
|
#if defined(_WIN32_WCE) && (_WIN32_WCE < 300)
|
|
|
|
CloseSynchHandle(sem->id);
|
|
|
|
#else
|
2001-04-26 16:45:43 +00:00
|
|
|
CloseHandle(sem->id);
|
2001-05-23 23:35:10 +00:00
|
|
|
#endif
|
2001-04-26 16:45:43 +00:00
|
|
|
sem->id = 0;
|
|
|
|
}
|
|
|
|
free(sem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int SDL_SemWaitTimeout(SDL_sem *sem, Uint32 timeout)
|
|
|
|
{
|
|
|
|
int retval;
|
|
|
|
DWORD dwMilliseconds;
|
|
|
|
|
|
|
|
if ( ! sem ) {
|
|
|
|
SDL_SetError("Passed a NULL sem");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( timeout == SDL_MUTEX_MAXWAIT ) {
|
|
|
|
dwMilliseconds = INFINITE;
|
|
|
|
} else {
|
|
|
|
dwMilliseconds = (DWORD)timeout;
|
|
|
|
}
|
2001-05-23 23:35:10 +00:00
|
|
|
#if defined(_WIN32_WCE) && (_WIN32_WCE < 300)
|
|
|
|
switch (WaitForSemaphoreCE(sem->id, dwMilliseconds)) {
|
|
|
|
#else
|
2001-04-26 16:45:43 +00:00
|
|
|
switch (WaitForSingleObject(sem->id, dwMilliseconds)) {
|
2001-05-23 23:35:10 +00:00
|
|
|
#endif
|
2001-04-26 16:45:43 +00:00
|
|
|
case WAIT_OBJECT_0:
|
|
|
|
--sem->count;
|
|
|
|
retval = 0;
|
|
|
|
break;
|
|
|
|
case WAIT_TIMEOUT:
|
|
|
|
retval = SDL_MUTEX_TIMEDOUT;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SDL_SetError("WaitForSingleObject() failed");
|
|
|
|
retval = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SDL_SemTryWait(SDL_sem *sem)
|
|
|
|
{
|
|
|
|
return SDL_SemWaitTimeout(sem, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
int SDL_SemWait(SDL_sem *sem)
|
|
|
|
{
|
|
|
|
return SDL_SemWaitTimeout(sem, SDL_MUTEX_MAXWAIT);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Returns the current count of the semaphore */
|
|
|
|
Uint32 SDL_SemValue(SDL_sem *sem)
|
|
|
|
{
|
|
|
|
if ( ! sem ) {
|
|
|
|
SDL_SetError("Passed a NULL sem");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return sem->count;
|
|
|
|
}
|
|
|
|
|
|
|
|
int SDL_SemPost(SDL_sem *sem)
|
|
|
|
{
|
|
|
|
if ( ! sem ) {
|
|
|
|
SDL_SetError("Passed a NULL sem");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
/* Increase the counter in the first place, because
|
|
|
|
* after a successful release the semaphore may
|
|
|
|
* immediately get destroyed by another thread which
|
|
|
|
* is waiting for this semaphore.
|
|
|
|
*/
|
|
|
|
++sem->count;
|
2001-05-23 23:35:10 +00:00
|
|
|
#if defined(_WIN32_WCE) && (_WIN32_WCE < 300)
|
|
|
|
if ( ReleaseSemaphoreCE(sem->id, 1, NULL) == FALSE ) {
|
|
|
|
#else
|
2001-04-26 16:45:43 +00:00
|
|
|
if ( ReleaseSemaphore(sem->id, 1, NULL) == FALSE ) {
|
2001-05-23 23:35:10 +00:00
|
|
|
#endif
|
2001-04-26 16:45:43 +00:00
|
|
|
--sem->count; /* restore */
|
|
|
|
SDL_SetError("ReleaseSemaphore() failed");
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|