1 /*****************************************************************************/
2 // Copyright 2002-2008 Adobe Systems Incorporated
3 // All Rights Reserved.
4 //
5 // NOTICE: Adobe permits you to use, modify, and distribute this file in
6 // accordance with the terms of the Adobe license agreement accompanying it.
7 /*****************************************************************************/
8
9 /* $Id: //mondo/dng_sdk_1_4/dng_sdk/source/dng_pthread.cpp#2 $ */
10 /* $DateTime: 2012/07/31 22:04:34 $ */
11 /* $Change: 840853 $ */
12 /* $Author: tknoll $ */
13
14 #include "dng_pthread.h"
15
16 /*****************************************************************************/
17
18 #if qDNGThreadSafe
19
20 /*****************************************************************************/
21
22 #include "dng_assertions.h"
23
24 /*****************************************************************************/
25
26 #if qWinOS
27
28 #pragma warning(disable : 4786)
29
30 // Nothing in this file requires Unicode,
31 // However, CreateSemaphore has a path parameter
32 // (which is NULL always in this code) and thus
33 // does not work on Win98 if UNICODE is defined.
34 // So we force it off here.
35
36 #undef UNICODE
37 #undef _UNICODE
38
39 #include <windows.h>
40 #include <process.h>
41 #include <errno.h>
42 #include <memory>
43 #include <new>
44 #include <map>
45
46 #else
47
48 #include <sys/time.h>
49
50 #endif
51
52 /*****************************************************************************/
53
54 #if qWinOS
55
56 /*****************************************************************************/
57
58 namespace {
59 struct waiter {
60 struct waiter *prev;
61 struct waiter *next;
62 HANDLE semaphore;
63 bool chosen_by_signal;
64 };
65 }
66
67 /*****************************************************************************/
68
69 struct dng_pthread_mutex_impl
70 {
71 CRITICAL_SECTION lock;
72
dng_pthread_mutex_impldng_pthread_mutex_impl73 dng_pthread_mutex_impl() { ::InitializeCriticalSection(&lock); }
~dng_pthread_mutex_impldng_pthread_mutex_impl74 ~dng_pthread_mutex_impl() { ::DeleteCriticalSection(&lock); }
Lockdng_pthread_mutex_impl75 void Lock() { ::EnterCriticalSection(&lock); }
Unlockdng_pthread_mutex_impl76 void Unlock() { ::LeaveCriticalSection(&lock); }
77 private:
operator =dng_pthread_mutex_impl78 dng_pthread_mutex_impl &operator=(const dng_pthread_mutex_impl &) { }
dng_pthread_mutex_impldng_pthread_mutex_impl79 dng_pthread_mutex_impl(const dng_pthread_mutex_impl &) { }
80 };
81
82 /*****************************************************************************/
83
84 struct dng_pthread_cond_impl
85 {
86 dng_pthread_mutex_impl lock; // Mutual exclusion on next two variables
87 waiter *head_waiter; // List of threads waiting on this condition
88 waiter *tail_waiter; // Used to get FIFO, rather than LIFO, behavior for pthread_cond_signal
89 unsigned int broadcast_generation; // Used as sort of a separator on broadcasts
90 // saves having to walk the waiters list setting
91 // each one's "chosen_by_signal" flag while the condition is locked
92
dng_pthread_cond_impldng_pthread_cond_impl93 dng_pthread_cond_impl() : head_waiter(NULL), tail_waiter(NULL), broadcast_generation(0) { }
~dng_pthread_cond_impldng_pthread_cond_impl94 ~dng_pthread_cond_impl() { } ;
95
96 // Non copyable
97 private:
operator =dng_pthread_cond_impl98 dng_pthread_cond_impl &operator=(const dng_pthread_cond_impl &) { }
dng_pthread_cond_impldng_pthread_cond_impl99 dng_pthread_cond_impl(const dng_pthread_cond_impl &) { }
100
101 };
102
103 /*****************************************************************************/
104
105 namespace
106 {
107
108 struct ScopedLock
109 {
110 dng_pthread_mutex_impl *mutex;
111
ScopedLock__anon60426a8c0211::ScopedLock112 ScopedLock(dng_pthread_mutex_impl *arg) : mutex(arg)
113 {
114 mutex->Lock();
115 }
ScopedLock__anon60426a8c0211::ScopedLock116 ScopedLock(dng_pthread_mutex_impl &arg) : mutex(&arg)
117 {
118 mutex->Lock();
119 }
~ScopedLock__anon60426a8c0211::ScopedLock120 ~ScopedLock()
121 {
122 mutex->Unlock();
123 }
124 private:
operator =__anon60426a8c0211::ScopedLock125 ScopedLock &operator=(const ScopedLock &) { }
ScopedLock__anon60426a8c0211::ScopedLock126 ScopedLock(const ScopedLock &) { }
127 };
128
129 dng_pthread_mutex_impl validationLock;
130
ValidateMutex(dng_pthread_mutex_t * mutex)131 void ValidateMutex(dng_pthread_mutex_t *mutex)
132 {
133 if (*mutex != DNG_PTHREAD_MUTEX_INITIALIZER)
134 return;
135
136 ScopedLock lock(validationLock);
137
138 if (*mutex == DNG_PTHREAD_MUTEX_INITIALIZER)
139 dng_pthread_mutex_init(mutex, NULL);
140 }
141
ValidateCond(dng_pthread_cond_t * cond)142 void ValidateCond(dng_pthread_cond_t *cond)
143 {
144 if (*cond != DNG_PTHREAD_COND_INITIALIZER)
145 return;
146
147 ScopedLock lock(validationLock);
148
149 if (*cond == DNG_PTHREAD_COND_INITIALIZER)
150 dng_pthread_cond_init(cond, NULL);
151 }
152
153 DWORD thread_wait_sema_TLS_index;
154 bool thread_wait_sema_inited = false;
155 dng_pthread_once_t once_thread_TLS = DNG_PTHREAD_ONCE_INIT;
156
init_thread_TLS()157 void init_thread_TLS()
158 {
159 thread_wait_sema_TLS_index = ::TlsAlloc();
160 thread_wait_sema_inited = true;
161 }
162
finalize_thread_TLS()163 void finalize_thread_TLS()
164 {
165 if (thread_wait_sema_inited)
166 {
167 ::TlsFree(thread_wait_sema_TLS_index);
168 thread_wait_sema_inited = false;
169 }
170 }
171
172 dng_pthread_mutex_impl primaryHandleMapLock;
173
174 typedef std::map<DWORD, std::pair<HANDLE, void **> > ThreadMapType;
175
176 // A map to make sure handles are freed and to allow returning a pointer sized result
177 // even on 64-bit Windows.
178 ThreadMapType primaryHandleMap;
179
GetThreadSemaphore()180 HANDLE GetThreadSemaphore()
181 {
182 dng_pthread_once(&once_thread_TLS, init_thread_TLS);
183
184 HANDLE semaphore = ::TlsGetValue(thread_wait_sema_TLS_index);
185 if (semaphore == NULL)
186 {
187 semaphore = ::CreateSemaphore(NULL, 0, 1, NULL);
188 ::TlsSetValue(thread_wait_sema_TLS_index, semaphore);
189 }
190
191 return semaphore;
192 }
193
FreeThreadSemaphore()194 void FreeThreadSemaphore()
195 {
196 if (thread_wait_sema_inited)
197 {
198 HANDLE semaphore = (HANDLE)::TlsGetValue(thread_wait_sema_TLS_index);
199
200 if (semaphore != NULL)
201 {
202 ::TlsSetValue(thread_wait_sema_TLS_index, NULL);
203 ::CloseHandle(semaphore);
204 }
205 }
206 }
207
208 struct trampoline_args
209 {
210 void *(*func)(void *);
211 void *arg;
212 };
213
214 // This trampoline takes care of the return type being different
215 // between pthreads thread funcs and Windows C lib thread funcs
trampoline(void * arg_arg)216 unsigned __stdcall trampoline(void *arg_arg)
217 {
218 trampoline_args *args_ptr = (trampoline_args *)arg_arg;
219 trampoline_args args = *args_ptr;
220
221 delete args_ptr;
222
223 GetThreadSemaphore();
224
225 void *result = args.func(args.arg);
226
227 {
228 ScopedLock lockMap(primaryHandleMapLock);
229
230 ThreadMapType::iterator iter = primaryHandleMap.find(pthread_self());
231 if (iter != primaryHandleMap.end())
232 *iter->second.second = result;
233 }
234
235 FreeThreadSemaphore();
236
237 return S_OK;
238 }
239
240 }
241
242 /*****************************************************************************/
243
244 extern "C" {
245
246 /*****************************************************************************/
247
248 struct dng_pthread_attr_impl
249 {
250 size_t stacksize;
251 };
252
253 /*****************************************************************************/
254
dng_pthread_attr_init(pthread_attr_t * attr)255 int dng_pthread_attr_init(pthread_attr_t *attr)
256 {
257 dng_pthread_attr_impl *newAttrs;
258
259 newAttrs = new (std::nothrow) dng_pthread_attr_impl;
260 if (newAttrs == NULL)
261 return -1; // ENOMEM;
262
263 newAttrs->stacksize = 0;
264
265 *attr = newAttrs;
266
267 return 0;
268 }
269
270 /*****************************************************************************/
271
dng_pthread_attr_destroy(pthread_attr_t * attr)272 int dng_pthread_attr_destroy(pthread_attr_t *attr)
273 {
274 if (*attr == NULL)
275 return -1; // EINVAL
276
277 delete *attr;
278
279 *attr = NULL;
280
281 return 0;
282 }
283
284 /*****************************************************************************/
285
dng_pthread_attr_setstacksize(dng_pthread_attr_t * attr,size_t stacksize)286 int dng_pthread_attr_setstacksize(dng_pthread_attr_t *attr, size_t stacksize)
287 {
288 if (attr == NULL || (*attr) == NULL)
289 return -1; // EINVAL
290
291 (*attr)->stacksize = stacksize;
292
293 return 0;
294 }
295
296 /*****************************************************************************/
297
dng_pthread_attr_getstacksize(const dng_pthread_attr_t * attr,size_t * stacksize)298 int dng_pthread_attr_getstacksize(const dng_pthread_attr_t *attr, size_t *stacksize)
299 {
300 if (attr == NULL || (*attr) == NULL || stacksize == NULL)
301 return -1; // EINVAL
302
303 *stacksize = (*attr)->stacksize;
304
305 return 0;
306 }
307
308 /*****************************************************************************/
309
dng_pthread_create(dng_pthread_t * thread,const pthread_attr_t * attrs,void * (* func)(void *),void * arg)310 int dng_pthread_create(dng_pthread_t *thread, const pthread_attr_t *attrs, void * (*func)(void *), void *arg)
311 {
312 try
313 {
314 uintptr_t result;
315 unsigned threadID;
316 std::auto_ptr<trampoline_args> args(new (std::nothrow) trampoline_args);
317 std::auto_ptr<void *> resultHolder(new (std::nothrow) (void *));
318
319 if (args.get() == NULL || resultHolder.get () == NULL)
320 return -1; // ENOMEM
321
322 args->func = func;
323 args->arg = arg;
324
325 size_t stacksize = 0;
326
327 if (attrs != NULL)
328 dng_pthread_attr_getstacksize (attrs, &stacksize);
329
330 {
331 ScopedLock lockMap(primaryHandleMapLock);
332
333 result = _beginthreadex(NULL, (unsigned)stacksize, trampoline, args.get(), 0, &threadID);
334 if (result == NULL)
335 return -1; // ENOMEM
336 args.release();
337
338 std::pair<DWORD, std::pair<HANDLE, void **> > newMapEntry(threadID,
339 std::pair<HANDLE, void **>((HANDLE)result, resultHolder.get ()));
340 std::pair<ThreadMapType::iterator, bool> insertion = primaryHandleMap.insert(newMapEntry);
341
342 // If there is a handle open on the thread, its ID should not be reused so assert that an insertion was made.
343 DNG_ASSERT(insertion.second, "pthread emulation logic error");
344 }
345
346
347 resultHolder.release ();
348
349 *thread = (dng_pthread_t)threadID;
350 return 0;
351 }
352 catch (const std::bad_alloc &)
353 {
354 return -1;
355 }
356 }
357
358 /*****************************************************************************/
359
dng_pthread_detach(dng_pthread_t thread)360 int dng_pthread_detach(dng_pthread_t thread)
361 {
362 HANDLE primaryHandle;
363 void **resultHolder = NULL;
364
365 {
366 ScopedLock lockMap(primaryHandleMapLock);
367
368 ThreadMapType::iterator iter = primaryHandleMap.find(thread);
369 if (iter == primaryHandleMap.end())
370 return -1;
371
372 primaryHandle = iter->second.first;
373
374 // A join is waiting on the thread.
375 if (primaryHandle == NULL)
376 return -1;
377
378 resultHolder = iter->second.second;
379
380 primaryHandleMap.erase(iter);
381 }
382
383 delete resultHolder;
384
385 if (!::CloseHandle(primaryHandle))
386 return -1;
387
388 return 0;
389 }
390
391 /*****************************************************************************/
392
dng_pthread_join(dng_pthread_t thread,void ** result)393 int dng_pthread_join(dng_pthread_t thread, void **result)
394 {
395 bool found = false;
396 HANDLE primaryHandle = NULL;
397 void **resultHolder = NULL;
398
399 ThreadMapType::iterator iter;
400
401 {
402 ScopedLock lockMap(primaryHandleMapLock);
403
404 iter = primaryHandleMap.find(thread);
405 found = iter != primaryHandleMap.end();
406 if (found)
407 {
408 primaryHandle = iter->second.first;
409 resultHolder = iter->second.second;
410
411 // Set HANDLE to NULL to force any later join or detach to fail.
412 iter->second.first = NULL;
413 }
414 }
415
416 // This case can happens when joining a thread not created with pthread_create,
417 // which is a bad idea, but it gets mapped to doing the join, but always returns NULL.
418 if (!found)
419 primaryHandle = ::OpenThread(SYNCHRONIZE|THREAD_QUERY_INFORMATION, FALSE, thread);
420
421 if (primaryHandle == NULL)
422 return -1;
423
424 DWORD err;
425 if (::WaitForSingleObject(primaryHandle, INFINITE) != WAIT_OBJECT_0)
426 {
427 err = ::GetLastError();
428 return -1;
429 }
430
431 {
432 ScopedLock lockMap(primaryHandleMapLock);
433
434 if (iter != primaryHandleMap.end())
435 primaryHandleMap.erase(iter);
436 }
437
438 ::CloseHandle(primaryHandle);
439 if (result != NULL && resultHolder != NULL)
440 *result = *resultHolder;
441
442 delete resultHolder;
443
444 return 0;
445 }
446
447 /*****************************************************************************/
448
dng_pthread_self()449 dng_pthread_t dng_pthread_self()
450 {
451 return (dng_pthread_t)::GetCurrentThreadId();
452 }
453
454 /*****************************************************************************/
455
dng_pthread_exit(void * result)456 void dng_pthread_exit(void *result)
457 {
458 {
459 ScopedLock lockMap(primaryHandleMapLock);
460
461 ThreadMapType::iterator iter = primaryHandleMap.find(pthread_self());
462 if (iter != primaryHandleMap.end())
463 *iter->second.second = result;
464 }
465
466 FreeThreadSemaphore();
467
468 _endthreadex(S_OK);
469 }
470
471 /*****************************************************************************/
472
dng_pthread_mutex_init(dng_pthread_mutex_t * mutex,void *)473 int dng_pthread_mutex_init(dng_pthread_mutex_t *mutex, void * /* attrs */)
474 {
475 dng_pthread_mutex_t result;
476 try {
477 result = new(dng_pthread_mutex_impl);
478 } catch (const std::bad_alloc &)
479 {
480 return -1;
481 }
482
483 if (result == NULL)
484 return -1;
485 *mutex = result;
486 return 0;
487 }
488
489 /*****************************************************************************/
490
dng_pthread_mutex_destroy(dng_pthread_mutex_t * mutex)491 int dng_pthread_mutex_destroy(dng_pthread_mutex_t *mutex)
492 {
493 if (*mutex == DNG_PTHREAD_MUTEX_INITIALIZER)
494 {
495 *mutex = NULL;
496 return 0;
497 }
498
499 delete *mutex;
500 *mutex = NULL;
501 return 0;
502 }
503
504 /*****************************************************************************/
505
dng_pthread_cond_init(dng_pthread_cond_t * cond,void *)506 int dng_pthread_cond_init(dng_pthread_cond_t *cond, void * /* attrs */)
507 {
508 dng_pthread_cond_t result;
509 try {
510 result = new(dng_pthread_cond_impl);
511 } catch (const std::bad_alloc &)
512 {
513 return -1;
514 }
515
516 if (result == NULL)
517 return -1;
518 *cond = result;
519 return 0;
520 }
521
522 /*****************************************************************************/
523
dng_pthread_cond_destroy(dng_pthread_cond_t * cond)524 int dng_pthread_cond_destroy(dng_pthread_cond_t *cond)
525 {
526 if (*cond == DNG_PTHREAD_COND_INITIALIZER)
527 {
528 *cond = NULL;
529 return 0;
530 }
531
532 delete *cond;
533 *cond = NULL;
534 return 0;
535 }
536
537 /*****************************************************************************/
538
dng_pthread_mutexattr_init(dng_pthread_mutexattr_t * mutexattr)539 int dng_pthread_mutexattr_init(dng_pthread_mutexattr_t* mutexattr)
540 {
541 return 0;
542 }
543
544 /*****************************************************************************/
545
dng_pthread_mutexattr_settype(dng_pthread_mutexattr_t * mutexattr,int type)546 int dng_pthread_mutexattr_settype(dng_pthread_mutexattr_t* mutexattr, int type)
547 {
548 return 0;
549 }
550
551 /*****************************************************************************/
552
dng_pthread_mutex_lock(dng_pthread_mutex_t * mutex)553 int dng_pthread_mutex_lock(dng_pthread_mutex_t *mutex)
554 {
555 ValidateMutex(mutex);
556 (*mutex)->Lock();
557 return 0;
558 }
559
560 /*****************************************************************************/
561
dng_pthread_mutex_unlock(dng_pthread_mutex_t * mutex)562 int dng_pthread_mutex_unlock(dng_pthread_mutex_t *mutex)
563 {
564 ValidateMutex(mutex);
565 (*mutex)->Unlock();
566 return 0;
567 }
568
569 /*****************************************************************************/
570
cond_wait_internal(dng_pthread_cond_t * cond,dng_pthread_mutex_t * mutex,int timeout_milliseconds)571 static int cond_wait_internal(dng_pthread_cond_t *cond, dng_pthread_mutex_t *mutex, int timeout_milliseconds)
572 {
573 dng_pthread_cond_impl &real_cond = **cond;
574 dng_pthread_mutex_impl &real_mutex = **mutex;
575
576 waiter this_wait;
577 HANDLE semaphore = GetThreadSemaphore();
578 int my_generation; // The broadcast generation this waiter is in
579
580 {
581 this_wait.next = NULL;
582 this_wait.semaphore = semaphore;
583 this_wait.chosen_by_signal = 0;
584
585 ScopedLock lock1(real_cond.lock);
586
587 // Add this waiter to the end of the list.
588 this_wait.prev = real_cond.tail_waiter;
589 if (real_cond.tail_waiter != NULL)
590 real_cond.tail_waiter->next = &this_wait;
591 real_cond.tail_waiter = &this_wait;
592
593 // If the list was empty, set the head of the list to this waiter.
594 if (real_cond.head_waiter == NULL)
595 real_cond.head_waiter = &this_wait;
596
597 // Note which broadcast generation this waiter belongs to.
598 my_generation = real_cond.broadcast_generation;
599 }
600
601 real_mutex.Unlock();
602
603 DWORD result = ::WaitForSingleObject(semaphore, timeout_milliseconds);
604
605 if (result == WAIT_TIMEOUT)
606 {
607 // If the wait timed out, this thread is likely still on the waiters list
608 // of the condition. However, there is a race in that the thread may have been
609 // signaled or broadcast between when WaitForSingleObject decided
610 // we had timed out and this code running.
611
612 bool mustConsumeSemaphore = false;
613 {
614 ScopedLock lock2(real_cond.lock);
615
616 bool chosen_by_signal = this_wait.chosen_by_signal;
617 bool chosen_by_broadcast = my_generation != real_cond.broadcast_generation;
618
619 if (chosen_by_signal || chosen_by_broadcast)
620 mustConsumeSemaphore = true;
621 else
622 {
623 // Still on waiters list. Remove this waiter from list.
624 if (this_wait.next != NULL)
625 this_wait.next->prev = this_wait.prev;
626 else
627 real_cond.tail_waiter = this_wait.prev;
628
629 if (this_wait.prev != NULL)
630 this_wait.prev->next = this_wait.next;
631 else
632 real_cond.head_waiter = this_wait.next;
633 }
634 }
635
636 if (mustConsumeSemaphore)
637 {
638 ::WaitForSingleObject(semaphore, INFINITE);
639 result = WAIT_OBJECT_0;
640 }
641 }
642 else
643 DNG_ASSERT (result == WAIT_OBJECT_0, "pthread emulation logic error");
644
645 // reacquire the mutex
646 real_mutex.Lock();
647
648 return (result == WAIT_TIMEOUT) ? DNG_ETIMEDOUT : 0;
649 }
650
651 /*****************************************************************************/
652
dng_pthread_cond_wait(dng_pthread_cond_t * cond,dng_pthread_mutex_t * mutex)653 int dng_pthread_cond_wait(dng_pthread_cond_t *cond, dng_pthread_mutex_t *mutex)
654 {
655 ValidateCond(cond);
656
657 return cond_wait_internal(cond, mutex, INFINITE);
658 }
659
660 /*****************************************************************************/
661
dng_pthread_cond_timedwait(dng_pthread_cond_t * cond,dng_pthread_mutex_t * mutex,struct dng_timespec * latest_time)662 int dng_pthread_cond_timedwait(dng_pthread_cond_t *cond, dng_pthread_mutex_t *mutex, struct dng_timespec *latest_time)
663 {
664 ValidateCond(cond);
665
666 struct dng_timespec sys_timespec;
667
668 dng_pthread_now (&sys_timespec);
669
670 __int64 sys_time = (__int64)sys_timespec.tv_sec * 1000000000 + sys_timespec.tv_nsec;
671 __int64 lock_time = (__int64)latest_time->tv_sec * 1000000000 + latest_time->tv_nsec;
672
673 int wait_millisecs = (int)((lock_time - sys_time + 500000) / 1000000);
674
675 if (wait_millisecs < 0)
676 wait_millisecs = 0;
677
678 return cond_wait_internal(cond, mutex, wait_millisecs);
679 }
680
681 /*****************************************************************************/
682
dng_pthread_cond_signal(dng_pthread_cond_t * cond)683 int dng_pthread_cond_signal(dng_pthread_cond_t *cond)
684 {
685 ValidateCond(cond);
686
687 waiter *first;
688 dng_pthread_cond_impl &real_cond = **cond;
689
690 {
691 ScopedLock lock(real_cond.lock);
692
693 first = real_cond.head_waiter;
694 if (first != NULL)
695 {
696 if (first->next != NULL)
697 first->next->prev = NULL;
698 else
699 real_cond.tail_waiter = NULL; // Or first->prev, which is always NULL in this case
700
701 first->chosen_by_signal = true;
702
703 real_cond.head_waiter = first->next;
704 }
705 }
706
707 if (first != NULL)
708 ::ReleaseSemaphore(first->semaphore, 1, NULL);
709
710 return 0;
711 }
712
713 /*****************************************************************************/
714
dng_pthread_cond_broadcast(dng_pthread_cond_t * cond)715 int dng_pthread_cond_broadcast(dng_pthread_cond_t *cond)
716 {
717 ValidateCond(cond);
718
719 waiter *first;
720 dng_pthread_cond_impl &real_cond = **cond;
721
722 {
723 ScopedLock lock(real_cond.lock);
724
725 first = real_cond.head_waiter;
726 real_cond.head_waiter = NULL;
727 real_cond.tail_waiter = NULL;
728
729 real_cond.broadcast_generation++;
730 }
731
732 while (first != NULL)
733 {
734 waiter *next = first->next;
735 ::ReleaseSemaphore(first->semaphore, 1, NULL);
736 first = next;
737 }
738
739 return 0;
740 }
741
742 /*****************************************************************************/
743
dng_pthread_once(dng_pthread_once_t * once,void (* init_func)())744 int dng_pthread_once(dng_pthread_once_t *once, void (*init_func)())
745 {
746 if (once == NULL || init_func == NULL)
747 return EINVAL;
748
749 if (once->inited)
750 return 0;
751
752 if (::InterlockedIncrement(&once->semaphore) == 0)
753 {
754 init_func();
755 once->inited = 1;
756 }
757 else
758 {
759 while (!once->inited)
760 Sleep(0);
761 }
762
763 return 0;
764 }
765
766 /*****************************************************************************/
767
dng_pthread_key_create(dng_pthread_key_t * key,void (* destructor)(void *))768 int dng_pthread_key_create(dng_pthread_key_t * key, void (*destructor) (void *))
769 {
770 if (destructor != NULL)
771 return -1;
772
773 DWORD result = ::TlsAlloc();
774 if (result == TLS_OUT_OF_INDEXES)
775 return -1;
776 *key = (unsigned long)result;
777 return 0;
778 }
779
780 /*****************************************************************************/
781
dng_pthread_key_delete(dng_pthread_key_t key)782 int dng_pthread_key_delete(dng_pthread_key_t key)
783 {
784 if (::TlsFree((DWORD)key))
785 return 0;
786 return -1;
787 }
788
789 /*****************************************************************************/
790
dng_pthread_setspecific(dng_pthread_key_t key,const void * value)791 int dng_pthread_setspecific(dng_pthread_key_t key, const void *value)
792 {
793 if (::TlsSetValue((DWORD)key, const_cast<void *>(value)))
794 return 0;
795 return -1;
796 }
797
798 /*****************************************************************************/
799
dng_pthread_getspecific(dng_pthread_key_t key)800 void *dng_pthread_getspecific(dng_pthread_key_t key)
801 {
802 return ::TlsGetValue((DWORD)key);
803 }
804
805 /*****************************************************************************/
806
807 namespace {
808 struct rw_waiter {
809 struct rw_waiter *prev;
810 struct rw_waiter *next;
811 HANDLE semaphore;
812 bool is_writer;
813 };
814 }
815
816 struct dng_pthread_rwlock_impl
817 {
818 dng_pthread_mutex_impl mutex;
819
820 rw_waiter *head_waiter;
821 rw_waiter *tail_waiter;
822
823 unsigned long readers_active;
824 unsigned long writers_waiting;
825 bool writer_active;
826
827 dng_pthread_cond_impl read_wait;
828 dng_pthread_cond_impl write_wait;
829
dng_pthread_rwlock_impldng_pthread_rwlock_impl830 dng_pthread_rwlock_impl ()
831 : mutex ()
832 , head_waiter (NULL)
833 , tail_waiter (NULL)
834 , readers_active (0)
835 , writers_waiting (0)
836 , read_wait ()
837 , write_wait ()
838 , writer_active (false)
839 {
840 }
841
~dng_pthread_rwlock_impldng_pthread_rwlock_impl842 ~dng_pthread_rwlock_impl ()
843 {
844 }
845
WakeHeadWaiterdng_pthread_rwlock_impl846 void WakeHeadWaiter ()
847 {
848 HANDLE semaphore = head_waiter->semaphore;
849
850 head_waiter = head_waiter->next;
851 if (head_waiter == NULL)
852 tail_waiter = NULL;
853
854 ::ReleaseSemaphore(semaphore, 1, NULL);
855 }
856
857 };
858
859 /*****************************************************************************/
860
dng_pthread_rwlock_init(dng_pthread_rwlock_t * rwlock,const pthread_rwlockattr_t * attrs)861 int dng_pthread_rwlock_init(dng_pthread_rwlock_t *rwlock, const pthread_rwlockattr_t *attrs)
862 {
863 dng_pthread_rwlock_impl *newRWLock;
864
865 newRWLock = new (std::nothrow) dng_pthread_rwlock_impl;
866 if (newRWLock == NULL)
867 return -1; // ENOMEM;
868
869 *rwlock = newRWLock;
870
871 return 0;
872 }
873
874 /*****************************************************************************/
875
dng_pthread_rwlock_destroy(dng_pthread_rwlock_t * rwlock)876 int dng_pthread_rwlock_destroy(dng_pthread_rwlock_t *rwlock)
877 {
878 dng_pthread_rwlock_impl &real_rwlock = **rwlock;
879
880 {
881 ScopedLock lock (real_rwlock.mutex);
882
883 if (real_rwlock.head_waiter != NULL ||
884 real_rwlock.readers_active != 0 ||
885 real_rwlock.writers_waiting != 0 ||
886 real_rwlock.writer_active)
887 return -1; // EBUSY
888 }
889
890 delete *rwlock;
891 *rwlock = NULL;
892 return 0;
893 }
894
895 /*****************************************************************************/
896
897 #define CHECK_RWLOCK_STATE(real_rwlock) \
898 DNG_ASSERT (!real_rwlock.writer_active || real_rwlock.readers_active == 0, "dng_pthread_rwlock_t logic error")
899
900 /*****************************************************************************/
901
dng_pthread_rwlock_rdlock(dng_pthread_rwlock_t * rwlock)902 int dng_pthread_rwlock_rdlock(dng_pthread_rwlock_t *rwlock)
903 {
904 dng_pthread_rwlock_impl &real_rwlock = **rwlock;
905
906 struct rw_waiter this_wait;
907 bool doWait = false;;
908 int result = 0;
909 HANDLE semaphore=NULL;
910
911 {
912
913 ScopedLock lock (real_rwlock.mutex);
914
915 CHECK_RWLOCK_STATE (real_rwlock);
916
917 if (real_rwlock.writers_waiting > 0 || real_rwlock.writer_active)
918 {
919 semaphore = GetThreadSemaphore();
920
921 this_wait.next = NULL;
922 this_wait.semaphore = semaphore;
923 this_wait.is_writer = false;
924
925 // Add this waiter to the end of the list.
926 this_wait.prev = real_rwlock.tail_waiter;
927 if (real_rwlock.tail_waiter != NULL)
928 real_rwlock.tail_waiter->next = &this_wait;
929 real_rwlock.tail_waiter = &this_wait;
930
931 // If the list was empty, set the head of the list to this waiter.
932 if (real_rwlock.head_waiter == NULL)
933 real_rwlock.head_waiter = &this_wait;
934
935 doWait = true;
936 }
937 else
938 real_rwlock.readers_active++;
939 }
940
941 if (result == 0 && doWait)
942 result = (WaitForSingleObject(semaphore, INFINITE) == WAIT_OBJECT_0) ? 0 : -1;
943
944 return result;
945 }
946
947 /*****************************************************************************/
948
dng_pthread_rwlock_tryrdlock(dng_pthread_rwlock_t * rwlock)949 int dng_pthread_rwlock_tryrdlock(dng_pthread_rwlock_t *rwlock)
950 {
951 dng_pthread_rwlock_impl &real_rwlock = **rwlock;
952
953 ScopedLock lock (real_rwlock.mutex);
954
955 CHECK_RWLOCK_STATE (real_rwlock);
956
957 if (real_rwlock.writers_waiting == 0 && !real_rwlock.writer_active)
958 {
959 real_rwlock.readers_active++;
960 return 0;
961 }
962
963 return -1;
964 }
965
966 /*****************************************************************************/
967
dng_pthread_rwlock_trywrlock(dng_pthread_rwlock_t * rwlock)968 int dng_pthread_rwlock_trywrlock(dng_pthread_rwlock_t *rwlock)
969 {
970 dng_pthread_rwlock_impl &real_rwlock = **rwlock;
971
972 ScopedLock lock (real_rwlock.mutex);
973
974 CHECK_RWLOCK_STATE (real_rwlock);
975
976 if (real_rwlock.readers_active == 0 &&
977 real_rwlock.writers_waiting == 0 &&
978 !real_rwlock.writer_active)
979 {
980 real_rwlock.writer_active = true;
981 return 0;
982 }
983
984 return -1;
985 }
986
987 /*****************************************************************************/
988
dng_pthread_rwlock_unlock(dng_pthread_rwlock_t * rwlock)989 int dng_pthread_rwlock_unlock(dng_pthread_rwlock_t *rwlock)
990 {
991 dng_pthread_rwlock_impl &real_rwlock = **rwlock;
992
993 int result = 0;
994
995 ScopedLock lock (real_rwlock.mutex);
996
997 CHECK_RWLOCK_STATE (real_rwlock);
998
999 if (real_rwlock.readers_active > 0)
1000 --real_rwlock.readers_active;
1001 else
1002 real_rwlock.writer_active = false;
1003
1004 while (real_rwlock.head_waiter != NULL)
1005 {
1006 if (real_rwlock.head_waiter->is_writer)
1007 {
1008 if (real_rwlock.readers_active == 0)
1009 {
1010 real_rwlock.writers_waiting--;
1011 real_rwlock.writer_active = true;
1012 real_rwlock.WakeHeadWaiter ();
1013 }
1014
1015 break;
1016 }
1017 else
1018 {
1019 ++real_rwlock.readers_active;
1020 real_rwlock.WakeHeadWaiter ();
1021 }
1022 }
1023
1024 return result;
1025 }
1026
1027 /*****************************************************************************/
1028
dng_pthread_rwlock_wrlock(dng_pthread_rwlock_t * rwlock)1029 int dng_pthread_rwlock_wrlock(dng_pthread_rwlock_t *rwlock)
1030 {
1031 dng_pthread_rwlock_impl &real_rwlock = **rwlock;
1032
1033 int result = 0;
1034 struct rw_waiter this_wait;
1035 HANDLE semaphore=NULL;
1036 bool doWait = false;
1037
1038 {
1039 ScopedLock lock (real_rwlock.mutex);
1040
1041 CHECK_RWLOCK_STATE (real_rwlock);
1042
1043 if (real_rwlock.readers_active ||
1044 real_rwlock.writers_waiting ||
1045 real_rwlock.writer_active)
1046 {
1047 semaphore = GetThreadSemaphore();
1048
1049 this_wait.next = NULL;
1050 this_wait.semaphore = semaphore;
1051 this_wait.is_writer = true;
1052
1053 // Add this waiter to the end of the list.
1054 this_wait.prev = real_rwlock.tail_waiter;
1055 if (real_rwlock.tail_waiter != NULL)
1056 real_rwlock.tail_waiter->next = &this_wait;
1057 real_rwlock.tail_waiter = &this_wait;
1058
1059 // If the list was empty, set the head of the list to this waiter.
1060 if (real_rwlock.head_waiter == NULL)
1061 real_rwlock.head_waiter = &this_wait;
1062
1063 real_rwlock.writers_waiting++;
1064
1065 doWait = true;
1066 }
1067 else
1068 real_rwlock.writer_active = true;
1069 }
1070
1071 if (result == 0 && doWait)
1072 result = (WaitForSingleObject(semaphore, INFINITE) == WAIT_OBJECT_0) ? 0 : -1;
1073
1074 return result;
1075 }
1076
1077 /*****************************************************************************/
1078
dng_pthread_disassociate()1079 void dng_pthread_disassociate()
1080 {
1081 FreeThreadSemaphore();
1082 }
1083
dng_pthread_terminate()1084 void dng_pthread_terminate()
1085 {
1086 finalize_thread_TLS();
1087 }
1088
1089 /*****************************************************************************/
1090
1091 } // extern "C"
1092
1093 /*****************************************************************************/
1094
1095 #endif
1096
1097 /*****************************************************************************/
1098
dng_pthread_now(struct timespec * now)1099 int dng_pthread_now (struct timespec *now)
1100 {
1101
1102 if (now == NULL)
1103 return -1; // EINVAL
1104
1105 #if qWinOS
1106
1107 FILETIME ft;
1108 ::GetSystemTimeAsFileTime(&ft);
1109
1110 __int64 sys_time = ((__int64)ft.dwHighDateTime << 32) + ft.dwLowDateTime;
1111
1112 #define SecsFrom1601To1970 11644473600
1113
1114 sys_time -= SecsFrom1601To1970 * 10000000LL;
1115
1116 sys_time *= 100; // Convert from 100ns to 1ns units
1117
1118 now->tv_sec = (long)(sys_time / 1000000000);
1119 now->tv_nsec = (long)(sys_time % 1000000000);
1120
1121 #else
1122
1123 struct timeval tv;
1124
1125 if (gettimeofday (&tv, NULL) != 0)
1126 return errno;
1127
1128 now->tv_sec = tv.tv_sec;
1129 now->tv_nsec = tv.tv_usec * 1000;
1130
1131 #endif
1132
1133 return 0;
1134
1135 }
1136
1137 /*****************************************************************************/
1138
1139 #endif // qDNGThreadSafe
1140
1141 /*****************************************************************************/
1142