X-Git-Url: http://git.silcnet.org/gitweb/?a=blobdiff_plain;f=lib%2Fsilcutil%2Fwin32%2Fsilcwin32thread.c;h=fa35683dbe4b19b5ebd96de2d9f8d52c45d1ca2a;hb=e7b6c157b80152bf9fb9266e6bdd93f9fb0db776;hp=ca2a211207c9baf3feec507151d2eb5f4d99a63d;hpb=382d15d447b7a95390decfa783836ae4fe255b3d;p=silc.git diff --git a/lib/silcutil/win32/silcwin32thread.c b/lib/silcutil/win32/silcwin32thread.c index ca2a2112..fa35683d 100644 --- a/lib/silcutil/win32/silcwin32thread.c +++ b/lib/silcutil/win32/silcwin32thread.c @@ -4,13 +4,12 @@ Author: Pekka Riikonen - Copyright (C) 2001 Pekka Riikonen + Copyright (C) 2001 - 2007 Pekka Riikonen This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by - the Free Software Foundation; either version 2 of the License, or - (at your option) any later version. - + the Free Software Foundation; version 2 of the License. + This program 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 @@ -19,7 +18,9 @@ */ /* $Id$ */ -#include "silcincludes.h" +#include "silc.h" + +/**************************** SILC Thread API *******************************/ #ifdef SILC_THREADS @@ -28,11 +29,9 @@ typedef struct { HANDLE thread; SilcThreadStart start_func; void *context; - bool waitable; + SilcBool waitable; } *SilcWin32Thread; -static DWORD silc_thread_tls; - /* Actual routine that is called by WIN32 when the thread is created. We will call the start_func from here. When this returns the thread is destroyed. */ @@ -40,16 +39,22 @@ static DWORD silc_thread_tls; unsigned __stdcall silc_thread_win32_start(void *context) { SilcWin32Thread thread = (SilcWin32Thread)context; + SilcTls tls; + + tls = silc_thread_tls_init(); + if (tls) + tls->platform_context = thread; - TlsSetValue(silc_thread_tls, context); silc_thread_exit(thread->start_func(thread->context)); + silc_free(tls); + return 0; } #endif SilcThread silc_thread_create(SilcThreadStart start_func, void *context, - bool waitable) + SilcBool waitable) { #ifdef SILC_THREADS SilcWin32Thread thread; @@ -58,13 +63,18 @@ SilcThread silc_thread_create(SilcThreadStart start_func, void *context, SILC_LOG_DEBUG(("Creating new thread")); thread = silc_calloc(1, sizeof(*thread)); + if (!thread) + return NULL; thread->start_func = start_func; thread->context = context; thread->waitable = waitable; - thread->thread = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE)silc_thread_win32_start, (void *)thread, 0, &id); + thread->thread = + _beginthreadex(NULL, 0, (LPTHREAD_START_ROUTINE)silc_thread_win32_start, + (void *)thread, 0, &id); if (!thread->thread) { SILC_LOG_ERROR(("Could not create new thread")); + silc_set_errno_reason(SILC_ERR, "Could not create new thread"); silc_free(thread); return NULL; } @@ -80,36 +90,37 @@ SilcThread silc_thread_create(SilcThreadStart start_func, void *context, void silc_thread_exit(void *exit_value) { #ifdef SILC_THREADS - SilcWin32Thread thread = TlsGetValue(silc_thread_tls); - + SilcTls tls = silc_thread_get_tls(); + SilcWin32Thread thread = tls->platform_context; + if (thread) { /* If the thread is waitable the memory is freed only in silc_thread_wait by another thread. If not waitable, free it now. */ - if (!thread->waitable) { - TerminateThread(thread->thread, 0); + if (!thread->waitable) silc_free(thread); - } - - TlsSetValue(silc_thread_tls, NULL); } - ExitThread(0); + + _endthreadex(0); #endif } SilcThread silc_thread_self(void) { #ifdef SILC_THREADS - SilcWin32Thread self = TlsGetValue(silc_thread_tls); + SilcTls tls = silc_thread_get_tls(); + SilcWin32Thread self = tls->platform_context; if (!self) { - /* This should only happen for the main thread! */ + /* This should only happen for the main thread. */ HANDLE handle = GetCurrentThread (); HANDLE process = GetCurrentProcess (); self = silc_calloc(1, sizeof(*self)); - DuplicateHandle(process, handle, process, - &self->thread, 0, FALSE, - DUPLICATE_SAME_ACCESS); - TlsSetValue(silc_thread_tls, self); + if (self) { + DuplicateHandle(process, handle, process, + &self->thread, 0, FALSE, + DUPLICATE_SAME_ACCESS); + tls->platform_context = self; + } } return (SilcThread)self; @@ -118,7 +129,7 @@ SilcThread silc_thread_self(void) #endif } -bool silc_thread_wait(SilcThread thread, void **exit_value) +SilcBool silc_thread_wait(SilcThread thread, void **exit_value) { #ifdef SILC_THREADS SilcWin32Thread self = (SilcWin32Thread)thread; @@ -130,12 +141,9 @@ bool silc_thread_wait(SilcThread thread, void **exit_value) /* The thread is waitable thus we will free all memory after the WaitForSingleObject returns, the thread is destroyed after that. */ + WaitForSingleObject(self->thread, INFINITE); + CloseHandle(self->thread); - /* 2 sec timeout, otherwise we would run to infinite loop some cases.. */ - if (WaitForSingleObject(self->thread, 2000) == WAIT_TIMEOUT) - TerminateThread(self->thread, 0); - - silc_free(self); if (exit_value) *exit_value = NULL; @@ -144,3 +152,311 @@ bool silc_thread_wait(SilcThread thread, void **exit_value) return FALSE; #endif } + +void silc_thread_yield(void) +{ +#ifdef SILC_THREADS + SleepEx (0,0); +#endif /* SILC_THREADS */ +} + + +/***************************** SILC Mutex API *******************************/ + +/* SILC Mutex structure */ +struct SilcMutexStruct { +#ifdef SILC_THREADS + CRITICAL_SECTION mutex; +#endif /* SILC_THREADS */ + unsigned int locked : 1; +}; + +SilcBool silc_mutex_alloc(SilcMutex *mutex) +{ +#ifdef SILC_THREADS + *mutex = silc_calloc(1, sizeof(**mutex)); + if (!(*mutex)) + return FALSE; + InitializeCriticalSection(&((*mutex)->mutex)); + return TRUE; +#else + return FALSE; +#endif /* SILC_THREADS */ +} + +void silc_mutex_free(SilcMutex mutex) +{ +#ifdef SILC_THREADS + if (mutex) { + DeleteCriticalSection(&mutex->mutex); + silc_free(mutex); + } +#endif /* SILC_THREADS */ +} + +void silc_mutex_lock(SilcMutex mutex) +{ +#ifdef SILC_THREADS + if (mutex) { + EnterCriticalSection(&mutex->mutex); + SILC_ASSERT(mutex->locked == FALSE); + mutex->locked = TRUE; + } +#endif /* SILC_THREADS */ +} + +void silc_mutex_unlock(SilcMutex mutex) +{ +#ifdef SILC_THREADS + if (mutex) { + SILC_ASSERT(mutex->locked == TRUE); + mutex->locked = FALSE; + LeaveCriticalSection(&mutex->mutex); + } +#endif /* SILC_THREADS */ +} + +void silc_mutex_assert_locked(SilcMutex mutex) +{ +#ifdef SILC_THREADS + if (mutex) + SILC_ASSERT(mutex->locked); +#endif /* SILC_THREADS */ +} + + +/***************************** SILC Rwlock API ******************************/ + +/* SILC read/write lock structure */ +struct SilcRwLockStruct { +#ifdef SILC_THREADS + SilcMutex mutex; + SilcCond cond; +#endif /* SILC_THREADS */ + unsigned int readers : 31; + unsigned int locked : 1; +}; + +SilcBool silc_rwlock_alloc(SilcRwLock *rwlock) +{ +#ifdef SILC_THREADS + *rwlock = silc_calloc(1, sizeof(**rwlock)); + if (!(*rwlock)) + return FALSE; + if (!silc_mutex_alloc(&(*rwlock)->mutex)) { + silc_free(*rwlock); + return FALSE; + } + if (!silc_cond_alloc(&(*rwlock)->cond)) { + silc_mutex_free((*rwlock)->mutex); + silc_free(*rwlock); + return FALSE; + } + return TRUE; +#else + return FALSE; +#endif /* SILC_THREADS */ +} + +void silc_rwlock_free(SilcRwLock rwlock) +{ +#ifdef SILC_THREADS + if (rwlock) { + silc_mutex_free(rwlock->mutex); + silc_cond_free(rwlock->cond); + silc_free(rwlock); + } +#endif /* SILC_THREADS */ +} + +void silc_rwlock_rdlock(SilcRwLock rwlock) +{ +#ifdef SILC_THREADS + if (rwlock) { + silc_mutex_lock(rwlock->mutex); + rwlock->readers++; + silc_mutex_unlock(rwlock->mutex); + } +#endif /* SILC_THREADS */ +} + +void silc_rwlock_wrlock(SilcRwLock rwlock) +{ +#ifdef SILC_THREADS + if (rwlock) { + silc_mutex_lock(rwlock->mutex); + while (rwlock->readers > 0) + silc_cond_wait(rwlock->cond, rwlock->mutex); + rwlock->locked = TRUE; + } +#endif /* SILC_THREADS */ +} + +void silc_rwlock_unlock(SilcRwLock rwlock) +{ +#ifdef SILC_THREADS + if (rwlock) { + if (rwlock->locked) { + /* Unlock writer */ + rwlock->locked = FALSE; + silc_mutex_unlock(rwlock->mutex); + return; + } + + /* Unlock reader */ + silc_mutex_lock(rwlock->mutex); + rwlock->readers--; + silc_cond_broadcast(rwlock->cond); + silc_mutex_unlock(rwlock->mutex); + } +#endif /* SILC_THREADS */ +} + + +/**************************** SILC Cond API ******************************/ + +/* SILC Conditional Variable context */ +struct SilcCondStruct { +#ifdef SILC_THREADS + HANDLE event; +#endif /* SILC_THREADS*/ + unsigned int waiters : 23; + unsigned int signal : 1; +}; + +SilcBool silc_cond_alloc(SilcCond *cond) +{ +#ifdef SILC_THREADS + *cond = silc_calloc(1, sizeof(**cond)); + if (*cond == NULL) + return FALSE; + (*cond)->event = CreateEvent(NULL, TRUE, FALSE, NULL); + return TRUE; +#else + return FALSE; +#endif /* SILC_THREADS*/ +} + +void silc_cond_free(SilcCond cond) +{ +#ifdef SILC_THREADS + CloseHandle(cond->event); + silc_free(cond); +#endif /* SILC_THREADS*/ +} + +void silc_cond_signal(SilcCond cond) +{ +#ifdef SILC_THREADS + cond->signal = TRUE; + SetEvent(cond->event); +#endif /* SILC_THREADS*/ +} + +void silc_cond_broadcast(SilcCond cond) +{ +#ifdef SILC_THREADS + cond->signal = TRUE; + SetEvent(cond->event); +#endif /* SILC_THREADS*/ +} + +void silc_cond_wait(SilcCond cond, SilcMutex mutex) +{ +#ifdef SILC_THREADS + silc_cond_timedwait(cond, mutex, 0); +#endif /* SILC_THREADS*/ +} + +SilcBool silc_cond_timedwait(SilcCond cond, SilcMutex mutex, + int timeout) +{ +#ifdef SILC_THREADS + DWORD ret, t = INFINITE; + + if (timeout) + t = timeout; + + while (TRUE) { + cond->waiters++; + silc_mutex_unlock(mutex); + + ret = WaitForSingleObject(cond->event, t); + + silc_mutex_lock(mutex); + cond->waiters--; + + if (ret != WAIT_OBJECT_0) + return FALSE; + + if (cond->signal) { + cond->signal = FALSE; + ResetEvent(cond->event); + break; + } + } +#endif /* SILC_THREADS*/ + return TRUE; +} + +/************************** Thread-local Storage ****************************/ + +#ifdef SILC_THREADS + +static DWORD silc_tls; +SilcBool silc_tls_set = FALSE; + +SilcTls silc_thread_tls_init(void) +{ + SilcTls tls; + + if (!silc_tls_set) { + silc_tls = TlsAlloc(); + if (silc_tls == TLS_OUT_OF_INDEXES) { + SILC_LOG_ERROR(("Error creating Thread-local storage")); + return NULL; + } + + silc_tls_set = TRUE; + } + + if (silc_thread_get_tls()) + return silc_thread_get_tls(); + + /* Allocate Tls for the thread */ + tls = silc_calloc(1, sizeof(*tls)); + if (!tls) { + SILC_LOG_ERROR(("Error allocating Thread-local storage")); + return NULL; + } + + TlsSetValue(silc_tls, tls); + return tls; +} + +SilcTls silc_thread_get_tls(void) +{ + return (SilcTls)TlsGetValue(silc_tls); +} + +#else + +SilcTlsStruct tls; +SilcTls tls_ptr = NULL; + +SilcTls silc_thread_tls_init(void) +{ + if (silc_thread_get_tls()) + return silc_thread_get_tls(); + + tls_ptr = &tls; + memset(tls_ptr, 0, sizeof(*tls_ptr)); + return tls_ptr; +} + +SilcTls silc_thread_get_tls(void) +{ + return tls_ptr; +} + +#endif /* SILC_THREADS */