| /* |
| * Header used to adapt pthread-based POSIX code to Windows API threads. |
| * |
| * Copyright (C) 2009 Andrzej K. Haczewski <ahaczewski@gmail.com> |
| */ |
| |
| #ifndef PTHREAD_H |
| #define PTHREAD_H |
| |
| #ifndef WIN32_LEAN_AND_MEAN |
| #define WIN32_LEAN_AND_MEAN |
| #endif |
| |
| #include <windows.h> |
| |
| /* |
| * Defines that adapt Windows API threads to pthreads API |
| */ |
| #define pthread_mutex_t CRITICAL_SECTION |
| |
| static inline int return_0(int i) { |
| return 0; |
| } |
| #define pthread_mutex_init(a,b) return_0((InitializeCriticalSection((a)), 0)) |
| #define pthread_mutex_destroy(a) DeleteCriticalSection((a)) |
| #define pthread_mutex_lock EnterCriticalSection |
| #define pthread_mutex_unlock LeaveCriticalSection |
| |
| typedef int pthread_mutexattr_t; |
| #define pthread_mutexattr_init(a) (*(a) = 0) |
| #define pthread_mutexattr_destroy(a) do {} while (0) |
| #define pthread_mutexattr_settype(a, t) 0 |
| #define PTHREAD_MUTEX_RECURSIVE 0 |
| |
| /* |
| * Implement simple condition variable for Windows threads, based on ACE |
| * implementation. |
| * |
| * See original implementation: http://bit.ly/1vkDjo |
| * ACE homepage: http://www.cse.wustl.edu/~schmidt/ACE.html |
| * See also: http://www.cse.wustl.edu/~schmidt/win32-cv-1.html |
| */ |
| typedef struct { |
| LONG waiters; |
| int was_broadcast; |
| CRITICAL_SECTION waiters_lock; |
| HANDLE sema; |
| HANDLE continue_broadcast; |
| } pthread_cond_t; |
| |
| extern int pthread_cond_init(pthread_cond_t *cond, const void *unused); |
| extern int pthread_cond_destroy(pthread_cond_t *cond); |
| extern int pthread_cond_wait(pthread_cond_t *cond, CRITICAL_SECTION *mutex); |
| extern int pthread_cond_signal(pthread_cond_t *cond); |
| extern int pthread_cond_broadcast(pthread_cond_t *cond); |
| |
| /* |
| * Simple thread creation implementation using pthread API |
| */ |
| typedef struct { |
| HANDLE handle; |
| void *(*start_routine)(void*); |
| void *arg; |
| DWORD tid; |
| } pthread_t; |
| |
| extern int pthread_create(pthread_t *thread, const void *unused, |
| void *(*start_routine)(void*), void *arg); |
| |
| /* |
| * To avoid the need of copying a struct, we use small macro wrapper to pass |
| * pointer to win32_pthread_join instead. |
| */ |
| #define pthread_join(a, b) win32_pthread_join(&(a), (b)) |
| |
| extern int win32_pthread_join(pthread_t *thread, void **value_ptr); |
| |
| #define pthread_equal(t1, t2) ((t1).tid == (t2).tid) |
| extern pthread_t pthread_self(void); |
| |
| static inline void NORETURN pthread_exit(void *ret) |
| { |
| ExitThread((DWORD)(intptr_t)ret); |
| } |
| |
| typedef DWORD pthread_key_t; |
| static inline int pthread_key_create(pthread_key_t *keyp, void (*destructor)(void *value)) |
| { |
| return (*keyp = TlsAlloc()) == TLS_OUT_OF_INDEXES ? EAGAIN : 0; |
| } |
| |
| static inline int pthread_key_delete(pthread_key_t key) |
| { |
| return TlsFree(key) ? 0 : EINVAL; |
| } |
| |
| static inline int pthread_setspecific(pthread_key_t key, const void *value) |
| { |
| return TlsSetValue(key, (void *)value) ? 0 : EINVAL; |
| } |
| |
| static inline void *pthread_getspecific(pthread_key_t key) |
| { |
| return TlsGetValue(key); |
| } |
| |
| #ifndef __MINGW64_VERSION_MAJOR |
| static inline int pthread_sigmask(int how, const sigset_t *set, sigset_t *oset) |
| { |
| return 0; |
| } |
| #endif |
| |
| #endif /* PTHREAD_H */ |