2 * Copyright (C) 2009 Andrzej K. Haczewski <ahaczewski@gmail.com>
4 * DISCLAMER: The implementation is Git-specific, it is subset of original
5 * Pthreads API, without lots of other features that Git doesn't use.
6 * Git also makes sure that the passed arguments are valid, so there's
7 * no need for double-checking.
10 #include "../../git-compat-util.h"
16 static unsigned __stdcall
win32_start_routine(void *arg
)
18 pthread_t
*thread
= arg
;
19 thread
->arg
= thread
->start_routine(thread
->arg
);
23 int pthread_create(pthread_t
*thread
, const void *unused
,
24 void *(*start_routine
)(void*), void *arg
)
27 thread
->start_routine
= start_routine
;
28 thread
->handle
= (HANDLE
)
29 _beginthreadex(NULL
, 0, win32_start_routine
, thread
, 0, NULL
);
37 int win32_pthread_join(pthread_t
*thread
, void **value_ptr
)
39 DWORD result
= WaitForSingleObject(thread
->handle
, INFINITE
);
43 *value_ptr
= thread
->arg
;
48 return err_win_to_posix(GetLastError());
52 int pthread_cond_init(pthread_cond_t
*cond
, const void *unused
)
55 cond
->was_broadcast
= 0;
56 InitializeCriticalSection(&cond
->waiters_lock
);
58 cond
->sema
= CreateSemaphore(NULL
, 0, LONG_MAX
, NULL
);
60 die("CreateSemaphore() failed");
62 cond
->continue_broadcast
= CreateEvent(NULL
, /* security */
63 FALSE
, /* auto-reset */
64 FALSE
, /* not signaled */
66 if (!cond
->continue_broadcast
)
67 die("CreateEvent() failed");
72 int pthread_cond_destroy(pthread_cond_t
*cond
)
74 CloseHandle(cond
->sema
);
75 CloseHandle(cond
->continue_broadcast
);
76 DeleteCriticalSection(&cond
->waiters_lock
);
80 int pthread_cond_wait(pthread_cond_t
*cond
, CRITICAL_SECTION
*mutex
)
84 EnterCriticalSection(&cond
->waiters_lock
);
86 LeaveCriticalSection(&cond
->waiters_lock
);
89 * Unlock external mutex and wait for signal.
90 * NOTE: we've held mutex locked long enough to increment
91 * waiters count above, so there's no problem with
92 * leaving mutex unlocked before we wait on semaphore.
94 LeaveCriticalSection(mutex
);
96 /* let's wait - ignore return value */
97 WaitForSingleObject(cond
->sema
, INFINITE
);
100 * Decrease waiters count. If we are the last waiter, then we must
101 * notify the broadcasting thread that it can continue.
102 * But if we continued due to cond_signal, we do not have to do that
103 * because the signaling thread knows that only one waiter continued.
105 EnterCriticalSection(&cond
->waiters_lock
);
107 last_waiter
= cond
->was_broadcast
&& cond
->waiters
== 0;
108 LeaveCriticalSection(&cond
->waiters_lock
);
112 * cond_broadcast was issued while mutex was held. This means
113 * that all other waiters have continued, but are contending
114 * for the mutex at the end of this function because the
115 * broadcasting thread did not leave cond_broadcast, yet.
116 * (This is so that it can be sure that each waiter has
117 * consumed exactly one slice of the semaphor.)
118 * The last waiter must tell the broadcasting thread that it
121 SetEvent(cond
->continue_broadcast
);
123 * Now we go on to contend with all other waiters for
124 * the mutex. Auf in den Kampf!
127 /* lock external mutex again */
128 EnterCriticalSection(mutex
);
134 * IMPORTANT: This implementation requires that pthread_cond_signal
135 * is called while the mutex is held that is used in the corresponding
136 * pthread_cond_wait calls!
138 int pthread_cond_signal(pthread_cond_t
*cond
)
142 EnterCriticalSection(&cond
->waiters_lock
);
143 have_waiters
= cond
->waiters
> 0;
144 LeaveCriticalSection(&cond
->waiters_lock
);
147 * Signal only when there are waiters
150 return ReleaseSemaphore(cond
->sema
, 1, NULL
) ?
151 0 : err_win_to_posix(GetLastError());
157 * DOUBLY IMPORTANT: This implementation requires that pthread_cond_broadcast
158 * is called while the mutex is held that is used in the corresponding
159 * pthread_cond_wait calls!
161 int pthread_cond_broadcast(pthread_cond_t
*cond
)
163 EnterCriticalSection(&cond
->waiters_lock
);
165 if ((cond
->was_broadcast
= cond
->waiters
> 0)) {
166 /* wake up all waiters */
167 ReleaseSemaphore(cond
->sema
, cond
->waiters
, NULL
);
168 LeaveCriticalSection(&cond
->waiters_lock
);
170 * At this point all waiters continue. Each one takes its
171 * slice of the semaphor. Now it's our turn to wait: Since
172 * the external mutex is held, no thread can leave cond_wait,
173 * yet. For this reason, we can be sure that no thread gets
174 * a chance to eat *more* than one slice. OTOH, it means
175 * that the last waiter must send us a wake-up.
177 WaitForSingleObject(cond
->continue_broadcast
, INFINITE
);
179 * Since the external mutex is held, no thread can enter
180 * cond_wait, and, hence, it is safe to reset this flag
181 * without cond->waiters_lock held.
183 cond
->was_broadcast
= 0;
185 LeaveCriticalSection(&cond
->waiters_lock
);