There are at least two things ‘spurious wakeup’ could mean:
- A thread blocked in
pthread_cond_wait
can return from the call even though no call topthread_call_signal
orpthread_cond_broadcast
on the condition occurred. - A thread blocked in
pthread_cond_wait
returns because of a call topthread_cond_signal
orpthread_cond_broadcast
, however after reacquiring the mutex the underlying predicate is found to no longer be true.
But the latter case can occur even if the condition variable implementation does not allow the former case. Consider a producer consumer queue, and three threads.
- Thread 1 has just dequeued an element and released the mutex, and the queue is now empty. The thread is doing whatever it does with the element it acquired on some CPU.
- Thread 2 attempts to dequeue an element, but finds the queue to be empty when checked under the mutex, calls
pthread_cond_wait
, and blocks in the call awaiting signal/broadcast. - Thread 3 obtains the mutex, inserts a new element into the queue, notifies the condition variable, and releases the lock.
- In response to the notification from thread 3, thread 2, which was waiting on the condition, is scheduled to run.
- However before thread 2 manages to get on the CPU and grab the queue lock, thread 1 completes its current task, and returns to the queue for more work. It obtains the queue lock, checks the predicate, and finds that there is work in the queue. It proceeds to dequeue the item that thread 3 inserted, releases the lock, and does whatever it does with the item that thread 3 enqueued.
- Thread 2 now gets on a CPU and obtains the lock, but when it checks the predicate, it finds that the queue is empty. Thread 1 ‘stole’ the item, so the wakeup appears to be spurious. Thread 2 needs to wait on the condition again.
So since you already always need to check the predicate under a loop, it makes no difference if the underlying condition variables can have other sorts of spurious wakeups.