threads: Windows mutex and rwlock implementation
Change-Id: I087bd5884eca9f232f4b5a2a6463b06b38a488e7
This commit is contained in:
		 Nicolas Williams
					Nicolas Williams
				
			
				
					committed by
					
						 Jeffrey Altman
						Jeffrey Altman
					
				
			
			
				
	
			
			
			 Jeffrey Altman
						Jeffrey Altman
					
				
			
						parent
						
							1dcfceb090
						
					
				
				
					commit
					1f53a40827
				
			| @@ -108,6 +108,143 @@ | |||||||
| #define HEIMDAL_getspecific(k) pthread_getspecific(k) | #define HEIMDAL_getspecific(k) pthread_getspecific(k) | ||||||
| #define HEIMDAL_key_delete(k) pthread_key_delete(k) | #define HEIMDAL_key_delete(k) pthread_key_delete(k) | ||||||
|  |  | ||||||
|  | #elif defined(_WIN32) | ||||||
|  |  | ||||||
|  | typedef struct heim_mutex { | ||||||
|  |     HANDLE	h; | ||||||
|  | } heim_mutex_t; | ||||||
|  |  | ||||||
|  | static inline void | ||||||
|  | heim_mutex_init(heim_mutex_t *m) | ||||||
|  | { | ||||||
|  |     m->h = CreateSemaphore(NULL, 1, 1, NULL); | ||||||
|  |     if (m->h == INVALID_HANDLE_VALUE) | ||||||
|  | 	abort(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static inline void | ||||||
|  | heim_mutex_lock(heim_mutex_t *m) | ||||||
|  | { | ||||||
|  |     HANDLE h, new_h; | ||||||
|  |     int created = 0; | ||||||
|  |  | ||||||
|  |     h = InterlockedCompareExchangePointer(&m->h, m->h, m->h); | ||||||
|  |     if (h == INVALID_HANDLE_VALUE || h == NULL) { | ||||||
|  | 	created = 1; | ||||||
|  | 	new_h = CreateSemaphore(NULL, 0, 1, NULL); | ||||||
|  | 	if (new_h == INVALID_HANDLE_VALUE) | ||||||
|  | 	    abort(); | ||||||
|  | 	if (InterlockedCompareExchangePointer(&m->h, new_h, h) != h) { | ||||||
|  | 	    created = 0; | ||||||
|  | 	    CloseHandle(new_h); | ||||||
|  | 	} | ||||||
|  |     } | ||||||
|  |     if (!created) | ||||||
|  | 	WaitForSingleObject(m->h, INFINITE); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static inline void | ||||||
|  | heim_mutex_unlock(heim_mutex_t *m) | ||||||
|  | { | ||||||
|  |     if (ReleaseSemaphore(m->h, 1, NULL) == FALSE) | ||||||
|  | 	abort(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static inline void | ||||||
|  | heim_mutex_destroy(heim_mutex_t *m) | ||||||
|  | { | ||||||
|  |     HANDLE h; | ||||||
|  |  | ||||||
|  |     h = InterlockedCompareExchangePointer(&m->h, INVALID_HANDLE_VALUE, m->h); | ||||||
|  |     if (h != INVALID_HANDLE_VALUE) | ||||||
|  | 	CloseHandle(h); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #define HEIMDAL_MUTEX heim_mutex_t | ||||||
|  | #define HEIMDAL_MUTEX_INITIALIZER { INVALID_HANDLE_VALUE } | ||||||
|  | #define HEIMDAL_MUTEX_init(m) heim_mutex_init((m)) | ||||||
|  | #define HEIMDAL_MUTEX_lock(m) heim_mutex_lock((m)) | ||||||
|  | #define HEIMDAL_MUTEX_unlock(m) heim_mutex_unlock((m)) | ||||||
|  | #define HEIMDAL_MUTEX_destroy(m) heim_mutex_destroy((m)) | ||||||
|  |  | ||||||
|  | typedef struct heim_rwlock { | ||||||
|  |     SRWLOCK lock; | ||||||
|  |     int     exclusive; | ||||||
|  | } heim_rwlock_t; | ||||||
|  |  | ||||||
|  | static inline void | ||||||
|  | heim_rwlock_init(heim_rwlock_t *l) | ||||||
|  | { | ||||||
|  |     InitializeSRWLock(&l->lock); | ||||||
|  |     l->exclusive = 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static inline void | ||||||
|  | heim_rwlock_rdlock(heim_rwlock_t *l) | ||||||
|  | { | ||||||
|  |     AcquireSRWLockShared(&l->lock); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static inline void | ||||||
|  | heim_rwlock_wrlock(heim_rwlock_t *l) | ||||||
|  | { | ||||||
|  |     AcquireSRWLockExclusive(&l->lock); | ||||||
|  |     l->exclusive = 1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static inline BOOLEAN | ||||||
|  | heim_rwlock_tryrdlock(heim_rwlock_t *l) | ||||||
|  | { | ||||||
|  |     BOOLEAN bLocked; | ||||||
|  |  | ||||||
|  |     bLocked = TryAcquireSRWLockShared(&l->lock); | ||||||
|  |  | ||||||
|  |     return bLocked; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static inline BOOLEAN | ||||||
|  | heim_rwlock_trywrlock(heim_rwlock_t *l) | ||||||
|  | { | ||||||
|  |     BOOLEAN bLocked; | ||||||
|  |  | ||||||
|  |     bLocked = TryAcquireSRWLockExclusive(&l->lock); | ||||||
|  |     if (bLocked) | ||||||
|  | 	l->exclusive = 1; | ||||||
|  |  | ||||||
|  |     return bLocked; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static inline void | ||||||
|  | heim_rwlock_unlock(heim_rwlock_t *l) | ||||||
|  | { | ||||||
|  |     if (l->exclusive) { | ||||||
|  | 	l->exclusive = 0; | ||||||
|  | 	ReleaseSRWLockExclusive(&(l)->lock); | ||||||
|  |     } else { | ||||||
|  | 	ReleaseSRWLockShared(&(l)->lock); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | static inline void | ||||||
|  | heim_rwlock_destroy(heim_rwlock_t *l) | ||||||
|  | { | ||||||
|  |     /* SRW locks cannot be destroyed so re-initialize */ | ||||||
|  |     InitializeSRWLock(&l->lock); | ||||||
|  |     l->exclusive = 0; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | #define HEIMDAL_RWLOCK heim_rwlock_t | ||||||
|  | #define HEIMDAL_RWLOCK_INITIALIZER {SRWLOCK_INIT, 0} | ||||||
|  | #define	HEIMDAL_RWLOCK_init(l) heim_rwlock_init((l)) | ||||||
|  | #define	HEIMDAL_RWLOCK_rdlock(l) heim_rwlock_rdlock((l)) | ||||||
|  | #define	HEIMDAL_RWLOCK_wrlock(l) heim_rwlock_wrlock((l)) | ||||||
|  | #define	HEIMDAL_RWLOCK_tryrdlock(l) heim_rwlock_tryrdlock((l)) | ||||||
|  | #define	HEIMDAL_RWLOCK_trywrlock(l) heim_rwlock_trywrlock((l)) | ||||||
|  | #define	HEIMDAL_RWLOCK_unlock(l) heim_rwlock_unlock((l)) | ||||||
|  | #define	HEIMDAL_RWLOCK_destroy(l) heim_rwlock_destroy((l)) | ||||||
|  |  | ||||||
|  | #define HEIMDAL_internal_thread_key 1 | ||||||
|  |  | ||||||
| #elif defined(HEIMDAL_DEBUG_THREADS) | #elif defined(HEIMDAL_DEBUG_THREADS) | ||||||
|  |  | ||||||
| /* no threads support, just do consistency checks */ | /* no threads support, just do consistency checks */ | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user