How can I implement a binary semaphore using the POSIX counting semaphore API? I am using an unnamed semaphore and need to limit its count to 1. I believe I can't use a mutex because I need to be able to unlock from another thread.
1 Answer
If you actually want a semaphore that "absorbs" multiple posts without allowing multiple waits to succeed, and especially if you want to be strict about that, POSIX semaphores are not a good underlying promitive to use to implement it. The right set of primitives to implement it on top of is a mutex, a condition variable, and a bool protected by the mutex. When changing the bool from 0 to 1, you signal the condition variable.
With that said, what you're asking for is something of a smell; it inherently has ambiguous orderings. For example if threads A and B both post the semaphore one after another, and threads X and Y are both just starting to wait, it's possible with your non-counting semaphore that either both waits succeed or that only one does, depending on the order of execution: ABXY or AXBY (or other comparable permutation). Thus, the pattern is likely erroneous unless either there's only one thread that could possibly psot at any given time (in which case, why would it post more than once? maybe this is a non-issue) or ability to post is controlled by holding some sort of lock (again, in which case why would it post more than once?). So if you don't have a design flaw here, it's likely that just using a counting semaphore but not posting it more than once gives the behavior you want.
If that's not the case, then there's probably some other data associated with the semaphore that's not properly synchronized, and you're trying to use the semaphore like a condition variable for it. If that's the case, just put a proper mutex and condition variable around it and use them, and forget the semaphore.
One comment for addressing your specific situation:
I believe I can't use a mutex because I need to be able to unlock from another thread.
This becomes a non-issue if you use a combination of mutex and condition variable, because you don't keep the mutex locked while working. Instead, the fact that the combined system is in-use is part of the state protected by the mutex (e.g. the above-mentioned bool) and any thread that can obtain the mutex can change it (to return it to a released state).
1as thevaluetosem_init? If you’re concerned that users could still increase the value by callingsem_postwithout firstsem_waiting, you could either wrap these calls into functions that guard against this, or use a mutex + a condition variable instead.p->mutexbut then when I call post (from another thread) I get blocked by it.pthread_cond_wait(&p->cvar, &p->mutex);unlocks the mutex while the condition variable is waiting. When the thread wakes up it grabs the mutex again.pthread_cond_waitinherently unlocks the mutex and re-acquires it as part of finishing the wait This is not a convenience; it's an inherent requirement for a condition variable to fulfill its purpose. If the caller had to unlock before callingpthread_cond_wait, the state protected by the mutex would be checked without synchronization (data race) and could change between the check and the wait, thereby causing the waiter never to get woken.