• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * C11 <threads.h> emulation library
3  *
4  * (C) Copyright yohhoy 2012.
5  * Distributed under the Boost Software License, Version 1.0.
6  *
7  * Permission is hereby granted, free of charge, to any person or organization
8  * obtaining a copy of the software and accompanying documentation covered by
9  * this license (the "Software") to use, reproduce, display, distribute,
10  * execute, and transmit the Software, and to prepare [[derivative work]]s of the
11  * Software, and to permit third-parties to whom the Software is furnished to
12  * do so, all subject to the following:
13  *
14  * The copyright notices in the Software and this entire statement, including
15  * the above license grant, this restriction and the following disclaimer,
16  * must be included in all copies of the Software, in whole or in part, and
17  * all derivative works of the Software, unless such copies or derivative
18  * works are solely in the form of machine-executable object code generated by
19  * a source language processor.
20  *
21  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
22  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
23  * FITNESS FOR A PARTICULAR PURPOSE, TITLE AND NON-INFRINGEMENT. IN NO EVENT
24  * SHALL THE COPYRIGHT HOLDERS OR ANYONE DISTRIBUTING THE SOFTWARE BE LIABLE
25  * FOR ANY DAMAGES OR OTHER LIABILITY, WHETHER IN CONTRACT, TORT OR OTHERWISE,
26  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
27  * DEALINGS IN THE SOFTWARE.
28  */
29 #ifndef assert
30 #include <assert.h>
31 #endif
32 #include <limits.h>
33 #include <errno.h>
34 #include <process.h>  // MSVCRT
35 #include <stdlib.h>
36 
37 /*
38 Configuration macro:
39 
40   EMULATED_THREADS_USE_NATIVE_CALL_ONCE
41     Use native WindowsAPI one-time initialization function.
42     (requires WinVista or later)
43     Otherwise emulate by mtx_trylock() + *busy loop* for WinXP.
44 
45   EMULATED_THREADS_USE_NATIVE_CV
46     Use native WindowsAPI condition variable object.
47     (requires WinVista or later)
48     Otherwise use emulated implementation for WinXP.
49 
50   EMULATED_THREADS_TSS_DTOR_SLOTNUM
51     Max registerable TSS dtor number.
52 */
53 
54 // XXX: Retain XP compatibility
55 #if 0
56 #if _WIN32_WINNT >= 0x0600
57 // Prefer native WindowsAPI on newer environment.
58 #if !defined(__MINGW32__)
59 #define EMULATED_THREADS_USE_NATIVE_CALL_ONCE
60 #endif
61 #define EMULATED_THREADS_USE_NATIVE_CV
62 #endif
63 #endif
64 #define EMULATED_THREADS_TSS_DTOR_SLOTNUM 64  // see TLS_MINIMUM_AVAILABLE
65 
66 
67 #include <windows.h>
68 
69 // check configuration
70 #if defined(EMULATED_THREADS_USE_NATIVE_CALL_ONCE) && (_WIN32_WINNT < 0x0600)
71 #error EMULATED_THREADS_USE_NATIVE_CALL_ONCE requires _WIN32_WINNT>=0x0600
72 #endif
73 
74 #if defined(EMULATED_THREADS_USE_NATIVE_CV) && (_WIN32_WINNT < 0x0600)
75 #error EMULATED_THREADS_USE_NATIVE_CV requires _WIN32_WINNT>=0x0600
76 #endif
77 
78 /* Visual Studio 2015 and later */
79 #ifdef _MSC_VER
80 #define HAVE_TIMESPEC_GET
81 #endif
82 
83 /*---------------------------- macros ----------------------------*/
84 #ifdef EMULATED_THREADS_USE_NATIVE_CALL_ONCE
85 #define ONCE_FLAG_INIT INIT_ONCE_STATIC_INIT
86 #else
87 #define ONCE_FLAG_INIT {0}
88 #endif
89 #define TSS_DTOR_ITERATIONS 1
90 
91 // FIXME: temporary non-standard hack to ease transition
92 #define _MTX_INITIALIZER_NP {(PCRITICAL_SECTION_DEBUG)-1, -1, 0, 0, 0, 0}
93 
94 /*---------------------------- types ----------------------------*/
95 typedef struct cnd_t {
96 #ifdef EMULATED_THREADS_USE_NATIVE_CV
97     CONDITION_VARIABLE condvar;
98 #else
99     int blocked;
100     int gone;
101     int to_unblock;
102     HANDLE sem_queue;
103     HANDLE sem_gate;
104     CRITICAL_SECTION monitor;
105 #endif
106 } cnd_t;
107 
108 typedef HANDLE thrd_t;
109 
110 typedef DWORD tss_t;
111 
112 typedef CRITICAL_SECTION mtx_t;
113 
114 #ifdef EMULATED_THREADS_USE_NATIVE_CALL_ONCE
115 typedef INIT_ONCE once_flag;
116 #else
117 typedef struct once_flag_t {
118     volatile LONG status;
119 } once_flag;
120 #endif
121 
122 
123 static inline void * tss_get(tss_t key);
124 static inline void thrd_yield(void);
125 static inline int mtx_trylock(mtx_t *mtx);
126 static inline int mtx_lock(mtx_t *mtx);
127 static inline int mtx_unlock(mtx_t *mtx);
128 
129 /*
130 Implementation limits:
131   - Conditionally emulation for "Initialization functions"
132     (see EMULATED_THREADS_USE_NATIVE_CALL_ONCE macro)
133   - Emulated `mtx_timelock()' with mtx_trylock() + *busy loop*
134 */
135 static void impl_tss_dtor_invoke(void);  // forward decl.
136 
137 struct impl_thrd_param {
138     thrd_start_t func;
139     void *arg;
140 };
141 
impl_thrd_routine(void * p)142 static unsigned __stdcall impl_thrd_routine(void *p)
143 {
144     struct impl_thrd_param pack;
145     int code;
146     memcpy(&pack, p, sizeof(struct impl_thrd_param));
147     free(p);
148     code = pack.func(pack.arg);
149     impl_tss_dtor_invoke();
150     return (unsigned)code;
151 }
152 
impl_timespec2msec(const struct timespec * ts)153 static DWORD impl_timespec2msec(const struct timespec *ts)
154 {
155     return (DWORD)((ts->tv_sec * 1000U) + (ts->tv_nsec / 1000000L));
156 }
157 
158 #ifdef EMULATED_THREADS_USE_NATIVE_CALL_ONCE
159 struct impl_call_once_param { void (*func)(void); };
impl_call_once_callback(PINIT_ONCE InitOnce,PVOID Parameter,PVOID * Context)160 static BOOL CALLBACK impl_call_once_callback(PINIT_ONCE InitOnce, PVOID Parameter, PVOID *Context)
161 {
162     struct impl_call_once_param *param = (struct impl_call_once_param*)Parameter;
163     (param->func)();
164     ((void)InitOnce); ((void)Context);  // suppress warning
165     return TRUE;
166 }
167 #endif  // ifdef EMULATED_THREADS_USE_NATIVE_CALL_ONCE
168 
169 #ifndef EMULATED_THREADS_USE_NATIVE_CV
170 /*
171 Note:
172   The implementation of condition variable is ported from Boost.Interprocess
173   See http://www.boost.org/boost/interprocess/sync/windows/condition.hpp
174 */
impl_cond_do_signal(cnd_t * cond,int broadcast)175 static void impl_cond_do_signal(cnd_t *cond, int broadcast)
176 {
177     int nsignal = 0;
178 
179     EnterCriticalSection(&cond->monitor);
180     if (cond->to_unblock != 0) {
181         if (cond->blocked == 0) {
182             LeaveCriticalSection(&cond->monitor);
183             return;
184         }
185         if (broadcast) {
186             cond->to_unblock += nsignal = cond->blocked;
187             cond->blocked = 0;
188         } else {
189             nsignal = 1;
190             cond->to_unblock++;
191             cond->blocked--;
192         }
193     } else if (cond->blocked > cond->gone) {
194         WaitForSingleObject(cond->sem_gate, INFINITE);
195         if (cond->gone != 0) {
196             cond->blocked -= cond->gone;
197             cond->gone = 0;
198         }
199         if (broadcast) {
200             nsignal = cond->to_unblock = cond->blocked;
201             cond->blocked = 0;
202         } else {
203             nsignal = cond->to_unblock = 1;
204             cond->blocked--;
205         }
206     }
207     LeaveCriticalSection(&cond->monitor);
208 
209     if (0 < nsignal)
210         ReleaseSemaphore(cond->sem_queue, nsignal, NULL);
211 }
212 
impl_cond_do_wait(cnd_t * cond,mtx_t * mtx,const struct timespec * ts)213 static int impl_cond_do_wait(cnd_t *cond, mtx_t *mtx, const struct timespec *ts)
214 {
215     int nleft = 0;
216     int ngone = 0;
217     int timeout = 0;
218     DWORD w;
219 
220     WaitForSingleObject(cond->sem_gate, INFINITE);
221     cond->blocked++;
222     ReleaseSemaphore(cond->sem_gate, 1, NULL);
223 
224     mtx_unlock(mtx);
225 
226     w = WaitForSingleObject(cond->sem_queue, ts ? impl_timespec2msec(ts) : INFINITE);
227     timeout = (w == WAIT_TIMEOUT);
228 
229     EnterCriticalSection(&cond->monitor);
230     if ((nleft = cond->to_unblock) != 0) {
231         if (timeout) {
232             if (cond->blocked != 0) {
233                 cond->blocked--;
234             } else {
235                 cond->gone++;
236             }
237         }
238         if (--cond->to_unblock == 0) {
239             if (cond->blocked != 0) {
240                 ReleaseSemaphore(cond->sem_gate, 1, NULL);
241                 nleft = 0;
242             }
243             else if ((ngone = cond->gone) != 0) {
244                 cond->gone = 0;
245             }
246         }
247     } else if (++cond->gone == INT_MAX/2) {
248         WaitForSingleObject(cond->sem_gate, INFINITE);
249         cond->blocked -= cond->gone;
250         ReleaseSemaphore(cond->sem_gate, 1, NULL);
251         cond->gone = 0;
252     }
253     LeaveCriticalSection(&cond->monitor);
254 
255     if (nleft == 1) {
256         while (ngone--)
257             WaitForSingleObject(cond->sem_queue, INFINITE);
258         ReleaseSemaphore(cond->sem_gate, 1, NULL);
259     }
260 
261     mtx_lock(mtx);
262     return timeout ? thrd_busy : thrd_success;
263 }
264 #endif  // ifndef EMULATED_THREADS_USE_NATIVE_CV
265 
266 static struct impl_tss_dtor_entry {
267     tss_t key;
268     tss_dtor_t dtor;
269 } impl_tss_dtor_tbl[EMULATED_THREADS_TSS_DTOR_SLOTNUM];
270 
impl_tss_dtor_register(tss_t key,tss_dtor_t dtor)271 static int impl_tss_dtor_register(tss_t key, tss_dtor_t dtor)
272 {
273     int i;
274     for (i = 0; i < EMULATED_THREADS_TSS_DTOR_SLOTNUM; i++) {
275         if (!impl_tss_dtor_tbl[i].dtor)
276             break;
277     }
278     if (i == EMULATED_THREADS_TSS_DTOR_SLOTNUM)
279         return 1;
280     impl_tss_dtor_tbl[i].key = key;
281     impl_tss_dtor_tbl[i].dtor = dtor;
282     return 0;
283 }
284 
impl_tss_dtor_invoke()285 static void impl_tss_dtor_invoke()
286 {
287     int i;
288     for (i = 0; i < EMULATED_THREADS_TSS_DTOR_SLOTNUM; i++) {
289         if (impl_tss_dtor_tbl[i].dtor) {
290             void* val = tss_get(impl_tss_dtor_tbl[i].key);
291             if (val)
292                 (impl_tss_dtor_tbl[i].dtor)(val);
293         }
294     }
295 }
296 
297 
298 /*--------------- 7.25.2 Initialization functions ---------------*/
299 // 7.25.2.1
300 static inline void
call_once(once_flag * flag,void (* func)(void))301 call_once(once_flag *flag, void (*func)(void))
302 {
303     assert(flag && func);
304 #ifdef EMULATED_THREADS_USE_NATIVE_CALL_ONCE
305     {
306     struct impl_call_once_param param;
307     param.func = func;
308     InitOnceExecuteOnce(flag, impl_call_once_callback, (PVOID)&param, NULL);
309     }
310 #else
311     if (InterlockedCompareExchange(&flag->status, 1, 0) == 0) {
312         (func)();
313         InterlockedExchange(&flag->status, 2);
314     } else {
315         while (flag->status == 1) {
316             // busy loop!
317             thrd_yield();
318         }
319     }
320 #endif
321 }
322 
323 
324 /*------------- 7.25.3 Condition variable functions -------------*/
325 // 7.25.3.1
326 static inline int
cnd_broadcast(cnd_t * cond)327 cnd_broadcast(cnd_t *cond)
328 {
329     if (!cond) return thrd_error;
330 #ifdef EMULATED_THREADS_USE_NATIVE_CV
331     WakeAllConditionVariable(&cond->condvar);
332 #else
333     impl_cond_do_signal(cond, 1);
334 #endif
335     return thrd_success;
336 }
337 
338 // 7.25.3.2
339 static inline void
cnd_destroy(cnd_t * cond)340 cnd_destroy(cnd_t *cond)
341 {
342     assert(cond);
343 #ifdef EMULATED_THREADS_USE_NATIVE_CV
344     // do nothing
345 #else
346     CloseHandle(cond->sem_queue);
347     CloseHandle(cond->sem_gate);
348     DeleteCriticalSection(&cond->monitor);
349 #endif
350 }
351 
352 // 7.25.3.3
353 static inline int
cnd_init(cnd_t * cond)354 cnd_init(cnd_t *cond)
355 {
356     if (!cond) return thrd_error;
357 #ifdef EMULATED_THREADS_USE_NATIVE_CV
358     InitializeConditionVariable(&cond->condvar);
359 #else
360     cond->blocked = 0;
361     cond->gone = 0;
362     cond->to_unblock = 0;
363     cond->sem_queue = CreateSemaphore(NULL, 0, LONG_MAX, NULL);
364     cond->sem_gate = CreateSemaphore(NULL, 1, 1, NULL);
365     InitializeCriticalSection(&cond->monitor);
366 #endif
367     return thrd_success;
368 }
369 
370 // 7.25.3.4
371 static inline int
cnd_signal(cnd_t * cond)372 cnd_signal(cnd_t *cond)
373 {
374     if (!cond) return thrd_error;
375 #ifdef EMULATED_THREADS_USE_NATIVE_CV
376     WakeConditionVariable(&cond->condvar);
377 #else
378     impl_cond_do_signal(cond, 0);
379 #endif
380     return thrd_success;
381 }
382 
383 // 7.25.3.5
384 static inline int
cnd_timedwait(cnd_t * cond,mtx_t * mtx,const struct timespec * abs_time)385 cnd_timedwait(cnd_t *cond, mtx_t *mtx, const struct timespec *abs_time)
386 {
387     if (!cond || !mtx || !abs_time) return thrd_error;
388 #ifdef EMULATED_THREADS_USE_NATIVE_CV
389     if (SleepConditionVariableCS(&cond->condvar, mtx, impl_timespec2msec(abs_time)))
390         return thrd_success;
391     return (GetLastError() == ERROR_TIMEOUT) ? thrd_busy : thrd_error;
392 #else
393     return impl_cond_do_wait(cond, mtx, abs_time);
394 #endif
395 }
396 
397 // 7.25.3.6
398 static inline int
cnd_wait(cnd_t * cond,mtx_t * mtx)399 cnd_wait(cnd_t *cond, mtx_t *mtx)
400 {
401     if (!cond || !mtx) return thrd_error;
402 #ifdef EMULATED_THREADS_USE_NATIVE_CV
403     SleepConditionVariableCS(&cond->condvar, mtx, INFINITE);
404 #else
405     impl_cond_do_wait(cond, mtx, NULL);
406 #endif
407     return thrd_success;
408 }
409 
410 
411 /*-------------------- 7.25.4 Mutex functions --------------------*/
412 // 7.25.4.1
413 static inline void
mtx_destroy(mtx_t * mtx)414 mtx_destroy(mtx_t *mtx)
415 {
416     assert(mtx);
417     DeleteCriticalSection(mtx);
418 }
419 
420 // 7.25.4.2
421 static inline int
mtx_init(mtx_t * mtx,int type)422 mtx_init(mtx_t *mtx, int type)
423 {
424     if (!mtx) return thrd_error;
425     if (type != mtx_plain && type != mtx_timed && type != mtx_try
426       && type != (mtx_plain|mtx_recursive)
427       && type != (mtx_timed|mtx_recursive)
428       && type != (mtx_try|mtx_recursive))
429         return thrd_error;
430     InitializeCriticalSection(mtx);
431     return thrd_success;
432 }
433 
434 // 7.25.4.3
435 static inline int
mtx_lock(mtx_t * mtx)436 mtx_lock(mtx_t *mtx)
437 {
438     if (!mtx) return thrd_error;
439     EnterCriticalSection(mtx);
440     return thrd_success;
441 }
442 
443 // 7.25.4.4
444 static inline int
mtx_timedlock(mtx_t * mtx,const struct timespec * ts)445 mtx_timedlock(mtx_t *mtx, const struct timespec *ts)
446 {
447     time_t expire, now;
448     if (!mtx || !ts) return thrd_error;
449     expire = time(NULL);
450     expire += ts->tv_sec;
451     while (mtx_trylock(mtx) != thrd_success) {
452         now = time(NULL);
453         if (expire < now)
454             return thrd_busy;
455         // busy loop!
456         thrd_yield();
457     }
458     return thrd_success;
459 }
460 
461 // 7.25.4.5
462 static inline int
mtx_trylock(mtx_t * mtx)463 mtx_trylock(mtx_t *mtx)
464 {
465     if (!mtx) return thrd_error;
466     return TryEnterCriticalSection(mtx) ? thrd_success : thrd_busy;
467 }
468 
469 // 7.25.4.6
470 static inline int
mtx_unlock(mtx_t * mtx)471 mtx_unlock(mtx_t *mtx)
472 {
473     if (!mtx) return thrd_error;
474     LeaveCriticalSection(mtx);
475     return thrd_success;
476 }
477 
478 
479 /*------------------- 7.25.5 Thread functions -------------------*/
480 // 7.25.5.1
481 static inline int
thrd_create(thrd_t * thr,thrd_start_t func,void * arg)482 thrd_create(thrd_t *thr, thrd_start_t func, void *arg)
483 {
484     struct impl_thrd_param *pack;
485     uintptr_t handle;
486     if (!thr) return thrd_error;
487     pack = (struct impl_thrd_param *)malloc(sizeof(struct impl_thrd_param));
488     if (!pack) return thrd_nomem;
489     pack->func = func;
490     pack->arg = arg;
491     handle = _beginthreadex(NULL, 0, impl_thrd_routine, pack, 0, NULL);
492     if (handle == 0) {
493         if (errno == EAGAIN || errno == EACCES)
494             return thrd_nomem;
495         return thrd_error;
496     }
497     *thr = (thrd_t)handle;
498     return thrd_success;
499 }
500 
501 #if 0
502 // 7.25.5.2
503 static inline thrd_t
504 thrd_current(void)
505 {
506     HANDLE hCurrentThread;
507     BOOL bRet;
508 
509     /* GetCurrentThread() returns a pseudo-handle, which we need
510      * to pass to DuplicateHandle(). Only the resulting handle can be used
511      * from other threads.
512      *
513      * Note that neither handle can be compared to the one by thread_create.
514      * Only the thread IDs - as returned by GetThreadId() and GetCurrentThreadId()
515      * can be compared directly.
516      *
517      * Other potential solutions would be:
518      * - define thrd_t as a thread Ids, but this would mean we'd need to OpenThread for many operations
519      * - use malloc'ed memory for thrd_t. This would imply using TLS for current thread.
520      *
521      * Neither is particularly nice.
522      *
523      * Life would be much easier if C11 threads had different abstractions for
524      * threads and thread IDs, just like C++11 threads does...
525      */
526 
527     bRet = DuplicateHandle(GetCurrentProcess(), // source process (pseudo) handle
528                            GetCurrentThread(), // source (pseudo) handle
529                            GetCurrentProcess(), // target process
530                            &hCurrentThread, // target handle
531                            0,
532                            FALSE,
533                            DUPLICATE_SAME_ACCESS);
534     assert(bRet);
535     if (!bRet) {
536 	hCurrentThread = GetCurrentThread();
537     }
538     return hCurrentThread;
539 }
540 #endif
541 
542 // 7.25.5.3
543 static inline int
thrd_detach(thrd_t thr)544 thrd_detach(thrd_t thr)
545 {
546     CloseHandle(thr);
547     return thrd_success;
548 }
549 
550 // 7.25.5.4
551 static inline int
thrd_equal(thrd_t thr0,thrd_t thr1)552 thrd_equal(thrd_t thr0, thrd_t thr1)
553 {
554     return GetThreadId(thr0) == GetThreadId(thr1);
555 }
556 
557 // 7.25.5.5
558 static inline void
thrd_exit(int res)559 thrd_exit(int res)
560 {
561     impl_tss_dtor_invoke();
562     _endthreadex((unsigned)res);
563 }
564 
565 // 7.25.5.6
566 static inline int
thrd_join(thrd_t thr,int * res)567 thrd_join(thrd_t thr, int *res)
568 {
569     DWORD w, code;
570     w = WaitForSingleObject(thr, INFINITE);
571     if (w != WAIT_OBJECT_0)
572         return thrd_error;
573     if (res) {
574         if (!GetExitCodeThread(thr, &code)) {
575             CloseHandle(thr);
576             return thrd_error;
577         }
578         *res = (int)code;
579     }
580     CloseHandle(thr);
581     return thrd_success;
582 }
583 
584 // 7.25.5.7
585 static inline void
thrd_sleep(const struct timespec * time_point,struct timespec * remaining)586 thrd_sleep(const struct timespec *time_point, struct timespec *remaining)
587 {
588     assert(time_point);
589     assert(!remaining); /* not implemented */
590     Sleep(impl_timespec2msec(time_point));
591 }
592 
593 // 7.25.5.8
594 static inline void
thrd_yield(void)595 thrd_yield(void)
596 {
597     SwitchToThread();
598 }
599 
600 
601 /*----------- 7.25.6 Thread-specific storage functions -----------*/
602 // 7.25.6.1
603 static inline int
tss_create(tss_t * key,tss_dtor_t dtor)604 tss_create(tss_t *key, tss_dtor_t dtor)
605 {
606     if (!key) return thrd_error;
607     *key = TlsAlloc();
608     if (dtor) {
609         if (impl_tss_dtor_register(*key, dtor)) {
610             TlsFree(*key);
611             return thrd_error;
612         }
613     }
614     return (*key != 0xFFFFFFFF) ? thrd_success : thrd_error;
615 }
616 
617 // 7.25.6.2
618 static inline void
tss_delete(tss_t key)619 tss_delete(tss_t key)
620 {
621     TlsFree(key);
622 }
623 
624 // 7.25.6.3
625 static inline void *
tss_get(tss_t key)626 tss_get(tss_t key)
627 {
628     return TlsGetValue(key);
629 }
630 
631 // 7.25.6.4
632 static inline int
tss_set(tss_t key,void * val)633 tss_set(tss_t key, void *val)
634 {
635     return TlsSetValue(key, val) ? thrd_success : thrd_error;
636 }
637 
638 
639 /*-------------------- 7.25.7 Time functions --------------------*/
640 // 7.25.6.1
641 #ifndef HAVE_TIMESPEC_GET
642 static inline int
timespec_get(struct timespec * ts,int base)643 timespec_get(struct timespec *ts, int base)
644 {
645     if (!ts) return 0;
646     if (base == TIME_UTC) {
647         ts->tv_sec = time(NULL);
648         ts->tv_nsec = 0;
649         return base;
650     }
651     return 0;
652 }
653 #endif
654