110 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			110 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|   | /*
 | ||
|  |  * Public domain | ||
|  |  * pthread.h compatibility shim | ||
|  |  */ | ||
|  | 
 | ||
|  | #ifndef LIBCRYPTOCOMPAT_PTHREAD_H
 | ||
|  | #define LIBCRYPTOCOMPAT_PTHREAD_H
 | ||
|  | 
 | ||
|  | #ifdef _WIN32
 | ||
|  | 
 | ||
|  | #include <malloc.h>
 | ||
|  | #include <stdlib.h>
 | ||
|  | #include <windows.h>
 | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Static once initialization values. | ||
|  |  */ | ||
|  | #define PTHREAD_ONCE_INIT   { INIT_ONCE_STATIC_INIT }
 | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Static mutex initialization values. | ||
|  |  */ | ||
|  | #define PTHREAD_MUTEX_INITIALIZER	{ .lock = NULL }
 | ||
|  | 
 | ||
|  | /*
 | ||
|  |  * Once definitions. | ||
|  |  */ | ||
|  | struct pthread_once { | ||
|  | 	INIT_ONCE once; | ||
|  | }; | ||
|  | typedef struct pthread_once pthread_once_t; | ||
|  | 
 | ||
|  | static inline BOOL CALLBACK | ||
|  | _pthread_once_win32_cb(PINIT_ONCE once, PVOID param, PVOID *context) | ||
|  | { | ||
|  | 	void (*cb) (void) = param; | ||
|  | 	cb(); | ||
|  | 	return TRUE; | ||
|  | } | ||
|  | 
 | ||
|  | static inline int | ||
|  | pthread_once(pthread_once_t *once, void (*cb) (void)) | ||
|  | { | ||
|  | 	BOOL rc = InitOnceExecuteOnce(&once->once, _pthread_once_win32_cb, cb, NULL); | ||
|  | 	if (rc == 0) | ||
|  | 		return -1; | ||
|  | 	else | ||
|  | 		return 0; | ||
|  | } | ||
|  | 
 | ||
|  | typedef DWORD pthread_t; | ||
|  | 
 | ||
|  | static inline pthread_t | ||
|  | pthread_self(void) | ||
|  | { | ||
|  | 	return GetCurrentThreadId(); | ||
|  | } | ||
|  | 
 | ||
|  | static inline int | ||
|  | pthread_equal(pthread_t t1, pthread_t t2) | ||
|  | { | ||
|  | 	return t1 == t2; | ||
|  | } | ||
|  | 
 | ||
|  | struct pthread_mutex { | ||
|  | 	volatile LPCRITICAL_SECTION lock; | ||
|  | }; | ||
|  | typedef struct pthread_mutex pthread_mutex_t; | ||
|  | typedef void pthread_mutexattr_t; | ||
|  | 
 | ||
|  | static inline int | ||
|  | pthread_mutex_init(pthread_mutex_t *mutex, const pthread_mutexattr_t *attr) | ||
|  | { | ||
|  | 	if ((mutex->lock = malloc(sizeof(CRITICAL_SECTION))) == NULL) | ||
|  | 		exit(ENOMEM); | ||
|  | 	InitializeCriticalSection(mutex->lock); | ||
|  | 	return 0; | ||
|  | } | ||
|  | 
 | ||
|  | static inline int | ||
|  | pthread_mutex_lock(pthread_mutex_t *mutex) | ||
|  | { | ||
|  | 	if (mutex->lock == NULL) { | ||
|  | 		LPCRITICAL_SECTION lcs; | ||
|  | 
 | ||
|  | 		if ((lcs = malloc(sizeof(CRITICAL_SECTION))) == NULL) | ||
|  | 			exit(ENOMEM); | ||
|  | 		InitializeCriticalSection(lcs); | ||
|  | 		if (InterlockedCompareExchangePointer((PVOID*)&mutex->lock, (PVOID)lcs, NULL) != NULL) { | ||
|  | 			DeleteCriticalSection(lcs); | ||
|  | 			free(lcs); | ||
|  | 		} | ||
|  | 	} | ||
|  | 	EnterCriticalSection(mutex->lock); | ||
|  | 	return 0; | ||
|  | } | ||
|  | 
 | ||
|  | static inline int | ||
|  | pthread_mutex_unlock(pthread_mutex_t *mutex) | ||
|  | { | ||
|  | 	LeaveCriticalSection(mutex->lock); | ||
|  | 	return 0; | ||
|  | } | ||
|  | 
 | ||
|  | #else
 | ||
|  | #include_next <pthread.h>
 | ||
|  | #endif
 | ||
|  | 
 | ||
|  | #endif
 |