100 lines
		
	
	
	
		
			2.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			100 lines
		
	
	
	
		
			2.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/*
 | 
						|
 * 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
 | 
						|
 | 
						|
#define pthread_cond_t CONDITION_VARIABLE
 | 
						|
 | 
						|
#define pthread_cond_init(a,b) InitializeConditionVariable((a))
 | 
						|
#define pthread_cond_destroy(a) do {} while (0)
 | 
						|
#define pthread_cond_wait(a,b) return_0(SleepConditionVariableCS((a), (b), INFINITE))
 | 
						|
#define pthread_cond_signal WakeConditionVariable
 | 
						|
#define pthread_cond_broadcast WakeAllConditionVariable
 | 
						|
 | 
						|
/*
 | 
						|
 * 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 */
 |