|  | /* | 
|  | Copyright (c) 2011 mingw-w64 project | 
|  |  | 
|  | Permission is hereby granted, free of charge, to any person obtaining a | 
|  | copy of this software and associated documentation files (the "Software"), | 
|  | to deal in the Software without restriction, including without limitation | 
|  | the rights to use, copy, modify, merge, publish, distribute, sublicense, | 
|  | and/or sell copies of the Software, and to permit persons to whom the | 
|  | Software is furnished to do so, subject to the following conditions: | 
|  |  | 
|  | The above copyright notice and this permission notice shall be included in | 
|  | all copies or substantial portions of the Software. | 
|  |  | 
|  | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
|  | IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
|  | FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | 
|  | AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
|  | LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | 
|  | FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER | 
|  | DEALINGS IN THE SOFTWARE. | 
|  | */ | 
|  |  | 
|  | #include <windows.h> | 
|  | #include <stdio.h> | 
|  | #include <malloc.h> | 
|  | #include "pthread.h" | 
|  | #include "thread.h" | 
|  | #include "misc.h" | 
|  | #include "semaphore.h" | 
|  | #include "sem.h" | 
|  | #include "mutex.h" | 
|  | #include "ref.h" | 
|  | #include "spinlock.h" | 
|  |  | 
|  | int do_sema_b_wait_intern (HANDLE sema, int nointerrupt, DWORD timeout); | 
|  |  | 
|  | static int | 
|  | sem_result (int res) | 
|  | { | 
|  | if (res != 0) { | 
|  | errno = res; | 
|  | return -1; | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int | 
|  | sem_init (sem_t *sem, int pshared, unsigned int value) | 
|  | { | 
|  | _sem_t *sv; | 
|  |  | 
|  | if (!sem || value > (unsigned int)SEM_VALUE_MAX) | 
|  | return sem_result (EINVAL); | 
|  | if (pshared != PTHREAD_PROCESS_PRIVATE) | 
|  | return sem_result (EPERM); | 
|  |  | 
|  | if (!(sv = (sem_t) calloc (1,sizeof (*sv)))) | 
|  | return sem_result (ENOMEM); | 
|  |  | 
|  | sv->value = value; | 
|  | if (pthread_mutex_init (&sv->vlock, NULL) != 0) | 
|  | { | 
|  | free (sv); | 
|  | return sem_result (ENOSPC); | 
|  | } | 
|  | if ((sv->s = CreateSemaphore (NULL, 0, SEM_VALUE_MAX, NULL)) == NULL) | 
|  | { | 
|  | pthread_mutex_destroy (&sv->vlock); | 
|  | free (sv); | 
|  | return sem_result (ENOSPC); | 
|  | } | 
|  |  | 
|  | sv->valid = LIFE_SEM; | 
|  | *sem = sv; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int | 
|  | sem_destroy (sem_t *sem) | 
|  | { | 
|  | int r; | 
|  | _sem_t *sv = NULL; | 
|  |  | 
|  | if (!sem || (sv = *sem) == NULL) | 
|  | return sem_result (EINVAL); | 
|  | if ((r = pthread_mutex_lock (&sv->vlock)) != 0) | 
|  | return sem_result (r); | 
|  |  | 
|  | if (sv->value < 0) | 
|  | { | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  | return sem_result (EBUSY); | 
|  | } | 
|  | if (!CloseHandle (sv->s)) | 
|  | { | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  | return sem_result (EINVAL); | 
|  | } | 
|  | *sem = NULL; | 
|  | sv->value = SEM_VALUE_MAX; | 
|  | pthread_mutex_unlock(&sv->vlock); | 
|  | Sleep (0); | 
|  | while (pthread_mutex_destroy (&sv->vlock) == EBUSY) | 
|  | Sleep (0); | 
|  | sv->valid = DEAD_SEM; | 
|  | free (sv); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int | 
|  | sem_std_enter (sem_t *sem,_sem_t **svp, int do_test) | 
|  | { | 
|  | int r; | 
|  | _sem_t *sv; | 
|  |  | 
|  | if (do_test) | 
|  | pthread_testcancel (); | 
|  | if (!sem) | 
|  | return sem_result (EINVAL); | 
|  | sv = *sem; | 
|  | if (sv == NULL) | 
|  | return sem_result (EINVAL); | 
|  |  | 
|  | if ((r = pthread_mutex_lock (&sv->vlock)) != 0) | 
|  | return sem_result (r); | 
|  |  | 
|  | if (*sem == NULL) | 
|  | { | 
|  | pthread_mutex_unlock(&sv->vlock); | 
|  | return sem_result (EINVAL); | 
|  | } | 
|  | *svp = sv; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int | 
|  | sem_trywait (sem_t *sem) | 
|  | { | 
|  | _sem_t *sv; | 
|  |  | 
|  | if (sem_std_enter (sem, &sv, 0) != 0) | 
|  | return -1; | 
|  | if (sv->value <= 0) | 
|  | { | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  | return sem_result (EAGAIN); | 
|  | } | 
|  | sv->value--; | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | struct sSemTimedWait | 
|  | { | 
|  | sem_t *p; | 
|  | int *ret; | 
|  | }; | 
|  |  | 
|  | static void | 
|  | clean_wait_sem (void *s) | 
|  | { | 
|  | struct sSemTimedWait *p = (struct sSemTimedWait *) s; | 
|  | _sem_t *sv = NULL; | 
|  |  | 
|  | if (sem_std_enter (p->p, &sv, 0) != 0) | 
|  | return; | 
|  |  | 
|  | if (WaitForSingleObject (sv->s, 0) != WAIT_OBJECT_0) | 
|  | InterlockedIncrement (&sv->value); | 
|  | else if (p->ret) | 
|  | p->ret[0] = 0; | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  | } | 
|  |  | 
|  | int | 
|  | sem_wait (sem_t *sem) | 
|  | { | 
|  | long cur_v; | 
|  | int ret = 0; | 
|  | _sem_t *sv; | 
|  | HANDLE semh; | 
|  | struct sSemTimedWait arg; | 
|  |  | 
|  | if (sem_std_enter (sem, &sv, 1) != 0) | 
|  | return -1; | 
|  |  | 
|  | arg.ret = NULL; | 
|  | arg.p = sem; | 
|  | InterlockedDecrement (&sv->value); | 
|  | cur_v = sv->value; | 
|  | semh = sv->s; | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  |  | 
|  | if (cur_v >= 0) | 
|  | return 0; | 
|  | else | 
|  | { | 
|  | pthread_cleanup_push (clean_wait_sem, (void *) &arg); | 
|  | ret = do_sema_b_wait_intern (semh, 2, INFINITE); | 
|  | pthread_cleanup_pop (ret); | 
|  | } | 
|  |  | 
|  | if (!ret) | 
|  | return 0; | 
|  |  | 
|  | return sem_result (ret); | 
|  | } | 
|  |  | 
|  | int | 
|  | sem_timedwait (sem_t *sem, const struct timespec *t) | 
|  | { | 
|  | int cur_v, ret = 0; | 
|  | DWORD dwr; | 
|  | HANDLE semh; | 
|  | _sem_t *sv; | 
|  | struct sSemTimedWait arg; | 
|  |  | 
|  | if (!t) | 
|  | return sem_wait (sem); | 
|  | dwr = dwMilliSecs(_pthread_rel_time_in_ms (t)); | 
|  |  | 
|  | if (sem_std_enter (sem, &sv, 1) != 0) | 
|  | return -1; | 
|  |  | 
|  | arg.ret = &ret; | 
|  | arg.p = sem; | 
|  | InterlockedDecrement (&sv->value); | 
|  | cur_v = sv->value; | 
|  | semh = sv->s; | 
|  | pthread_mutex_unlock(&sv->vlock); | 
|  |  | 
|  | if (cur_v >= 0) | 
|  | return 0; | 
|  | else | 
|  | { | 
|  | pthread_cleanup_push (clean_wait_sem, (void *) &arg); | 
|  | ret = do_sema_b_wait_intern (semh, 2, dwr); | 
|  | pthread_cleanup_pop (ret); | 
|  | } | 
|  |  | 
|  | if (!ret) | 
|  | return 0; | 
|  | return sem_result (ret); | 
|  | } | 
|  |  | 
|  | int | 
|  | sem_post (sem_t *sem) | 
|  | { | 
|  | _sem_t *sv;; | 
|  |  | 
|  | if (sem_std_enter (sem, &sv, 0) != 0) | 
|  | return -1; | 
|  |  | 
|  | if (sv->value >= SEM_VALUE_MAX) | 
|  | { | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  | return sem_result (ERANGE); | 
|  | } | 
|  | InterlockedIncrement (&sv->value); | 
|  | if (sv->value > 0 || ReleaseSemaphore (sv->s, 1, NULL)) | 
|  | { | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  | return 0; | 
|  | } | 
|  | InterlockedDecrement (&sv->value); | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  |  | 
|  | return sem_result (EINVAL); | 
|  | } | 
|  |  | 
|  | int | 
|  | sem_post_multiple (sem_t *sem, int count) | 
|  | { | 
|  | int waiters_count; | 
|  | _sem_t *sv;; | 
|  |  | 
|  | if (count <= 0) | 
|  | return sem_result (EINVAL); | 
|  | if (sem_std_enter (sem, &sv, 0) != 0) | 
|  | return -1; | 
|  |  | 
|  | if (sv->value > (SEM_VALUE_MAX - count)) | 
|  | { | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  | return sem_result (ERANGE); | 
|  | } | 
|  | waiters_count = -sv->value; | 
|  | sv->value += count; | 
|  | /*InterlockedExchangeAdd((long*)&sv->value, (long) count);*/ | 
|  | if (waiters_count <= 0 | 
|  | || ReleaseSemaphore (sv->s, | 
|  | (waiters_count < count ? waiters_count | 
|  | : count), NULL)) | 
|  | { | 
|  | pthread_mutex_unlock(&sv->vlock); | 
|  | return 0; | 
|  | } | 
|  | /*InterlockedExchangeAdd((long*)&sv->value, -((long) count));*/ | 
|  | sv->value -= count; | 
|  | pthread_mutex_unlock(&sv->vlock); | 
|  | return sem_result (EINVAL); | 
|  | } | 
|  |  | 
|  | sem_t * | 
|  | sem_open (const char *name, int oflag, mode_t mode, unsigned int value) | 
|  | { | 
|  | sem_result (ENOSYS); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | int | 
|  | sem_close (sem_t *sem) | 
|  | { | 
|  | return sem_result (ENOSYS); | 
|  | } | 
|  |  | 
|  | int | 
|  | sem_unlink (const char *name) | 
|  | { | 
|  | return sem_result (ENOSYS); | 
|  | } | 
|  |  | 
|  | int | 
|  | sem_getvalue (sem_t *sem, int *sval) | 
|  | { | 
|  | _sem_t *sv; | 
|  | int r; | 
|  |  | 
|  | if (!sval) | 
|  | return sem_result (EINVAL); | 
|  |  | 
|  | if (!sem || (sv = *sem) == NULL) | 
|  | return sem_result (EINVAL); | 
|  |  | 
|  | if ((r = pthread_mutex_lock (&sv->vlock)) != 0) | 
|  | return sem_result (r); | 
|  | if (*sem == NULL) | 
|  | { | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  | return sem_result (EINVAL); | 
|  | } | 
|  |  | 
|  | *sval = (int) sv->value; | 
|  | pthread_mutex_unlock (&sv->vlock); | 
|  | return 0; | 
|  | } |