2 * Copyright (c) 1995 John Birrell <jb@cimlogic.com.au>.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. All advertising materials mentioning features or use of this software
14 * must display the following acknowledgement:
15 * This product includes software developed by John Birrell.
16 * 4. Neither the name of the author nor the names of any co-contributors
17 * may be used to endorse or promote products derived from this software
18 * without specific prior written permission.
20 * THIS SOFTWARE IS PROVIDED BY JOHN BIRRELL AND CONTRIBUTORS ``AS IS'' AND
21 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
24 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
39 #include "pthread_private.h"
44 static inline pthread_t cond_queue_deq(pthread_cond_t);
45 static inline void cond_queue_remove(pthread_cond_t, pthread_t);
46 static inline void cond_queue_enq(pthread_cond_t, pthread_t);
48 /* Reinitialize a condition variable to defaults. */
50 _cond_reinit(pthread_cond_t * cond)
56 else if (*cond == NULL)
57 ret = pthread_cond_init(cond, NULL);
60 * Initialize the condition variable structure:
62 TAILQ_INIT(&(*cond)->c_queue);
63 (*cond)->c_flags = COND_FLAGS_INITED;
64 (*cond)->c_type = COND_TYPE_FAST;
65 (*cond)->c_mutex = NULL;
66 memset(&(*cond)->lock, 0, sizeof((*cond)->lock));
72 pthread_cond_init(pthread_cond_t * cond, const pthread_condattr_t * cond_attr)
74 enum pthread_cond_type type;
82 * Check if a pointer to a condition variable attribute
83 * structure was passed by the caller:
85 if (cond_attr != NULL && *cond_attr != NULL) {
86 /* Default to a fast condition variable: */
87 type = (*cond_attr)->c_type;
89 /* Default to a fast condition variable: */
90 type = COND_TYPE_FAST;
93 /* Process according to condition variable type: */
95 /* Fast condition variable: */
97 /* Nothing to do here. */
100 /* Trap invalid condition variable types: */
102 /* Return an invalid argument error: */
107 /* Check for no errors: */
109 if ((pcond = (pthread_cond_t)
110 malloc(sizeof(struct pthread_cond))) == NULL) {
114 * Initialise the condition variable
117 TAILQ_INIT(&pcond->c_queue);
118 pcond->c_flags |= COND_FLAGS_INITED;
119 pcond->c_type = type;
120 pcond->c_mutex = NULL;
121 memset(&pcond->lock,0,sizeof(pcond->lock));
126 /* Return the completion status: */
131 pthread_cond_destroy(pthread_cond_t * cond)
135 if (cond == NULL || *cond == NULL)
138 /* Lock the condition variable structure: */
139 _SPINLOCK(&(*cond)->lock);
142 * Free the memory allocated for the condition
143 * variable structure:
148 * NULL the caller's pointer now that the condition
149 * variable has been destroyed:
153 /* Return the completion status: */
158 pthread_cond_wait(pthread_cond_t * cond, pthread_mutex_t * mutex)
163 _thread_enter_cancellation_point();
169 * If the condition variable is statically initialized,
170 * perform the dynamic initialization:
172 else if (*cond != NULL ||
173 (rval = pthread_cond_init(cond,NULL)) == 0) {
175 _thread_enter_cancellation_point();
177 /* Lock the condition variable structure: */
178 _SPINLOCK(&(*cond)->lock);
181 * If the condvar was statically allocated, properly
182 * initialize the tail queue.
184 if (((*cond)->c_flags & COND_FLAGS_INITED) == 0) {
185 TAILQ_INIT(&(*cond)->c_queue);
186 (*cond)->c_flags |= COND_FLAGS_INITED;
189 /* Process according to condition variable type: */
190 switch ((*cond)->c_type) {
191 /* Fast condition variable: */
193 if ((mutex == NULL) || (((*cond)->c_mutex != NULL) &&
194 ((*cond)->c_mutex != *mutex))) {
195 /* Unlock the condition variable structure: */
196 _SPINUNLOCK(&(*cond)->lock);
198 /* Return invalid argument error: */
201 /* Reset the timeout and interrupted flags: */
202 _thread_run->timeout = 0;
203 _thread_run->interrupted = 0;
206 * Queue the running thread for the condition
209 cond_queue_enq(*cond, _thread_run);
211 /* Remember the mutex that is being used: */
212 (*cond)->c_mutex = *mutex;
215 _thread_run->wakeup_time.tv_sec = -1;
217 /* Unlock the mutex: */
218 if ((rval = _mutex_cv_unlock(mutex)) != 0) {
220 * Cannot unlock the mutex, so remove
221 * the running thread from the condition
224 cond_queue_remove(*cond, _thread_run);
226 /* Check for no more waiters: */
227 if (TAILQ_FIRST(&(*cond)->c_queue) ==
229 (*cond)->c_mutex = NULL;
231 /* Unlock the condition variable structure: */
232 _SPINUNLOCK(&(*cond)->lock);
236 * Schedule the next thread and unlock
237 * the condition variable structure:
239 _thread_kern_sched_state_unlock(PS_COND_WAIT,
240 &(*cond)->lock, __FILE__, __LINE__);
242 if (_thread_run->interrupted != 0) {
244 * Remember that this thread
250 * Lock the condition variable
251 * while removing the thread.
253 _SPINLOCK(&(*cond)->lock);
255 cond_queue_remove(*cond,
258 /* Check for no more waiters: */
259 if (TAILQ_FIRST(&(*cond)->c_queue) == NULL)
260 (*cond)->c_mutex = NULL;
262 _SPINUNLOCK(&(*cond)->lock);
266 * Note that even though this thread may have
267 * been canceled, POSIX requires that the mutex
268 * be reaquired prior to cancellation.
270 rval = _mutex_cv_lock(mutex);
275 /* Trap invalid condition variable types: */
277 /* Unlock the condition variable structure: */
278 _SPINUNLOCK(&(*cond)->lock);
280 /* Return an invalid argument error: */
285 if (interrupted != 0) {
286 if (_thread_run->continuation != NULL)
287 _thread_run->continuation((void *) _thread_run);
291 _thread_leave_cancellation_point();
294 _thread_leave_cancellation_point();
296 /* Return the completion status: */
301 pthread_cond_timedwait(pthread_cond_t * cond, pthread_mutex_t * mutex,
302 const struct timespec * abstime)
307 _thread_enter_cancellation_point();
309 if (cond == NULL || abstime == NULL)
312 if (abstime->tv_sec < 0 ||
313 abstime->tv_nsec < 0 || abstime->tv_nsec >= 1000000000) {
315 _thread_leave_cancellation_point();
320 * If the condition variable is statically initialized,
321 * perform the dynamic initialization:
324 (rval = pthread_cond_init(cond,NULL)) == 0) {
326 _thread_enter_cancellation_point();
328 /* Lock the condition variable structure: */
329 _SPINLOCK(&(*cond)->lock);
332 * If the condvar was statically allocated, properly
333 * initialize the tail queue.
335 if (((*cond)->c_flags & COND_FLAGS_INITED) == 0) {
336 TAILQ_INIT(&(*cond)->c_queue);
337 (*cond)->c_flags |= COND_FLAGS_INITED;
340 /* Process according to condition variable type: */
341 switch ((*cond)->c_type) {
342 /* Fast condition variable: */
344 if ((mutex == NULL) || (((*cond)->c_mutex != NULL) &&
345 ((*cond)->c_mutex != *mutex))) {
346 /* Return invalid argument error: */
349 /* Unlock the condition variable structure: */
350 _SPINUNLOCK(&(*cond)->lock);
352 /* Set the wakeup time: */
353 _thread_run->wakeup_time.tv_sec =
355 _thread_run->wakeup_time.tv_nsec =
358 /* Reset the timeout and interrupted flags: */
359 _thread_run->timeout = 0;
360 _thread_run->interrupted = 0;
363 * Queue the running thread for the condition
366 cond_queue_enq(*cond, _thread_run);
368 /* Remember the mutex that is being used: */
369 (*cond)->c_mutex = *mutex;
371 /* Unlock the mutex: */
372 if ((rval = _mutex_cv_unlock(mutex)) != 0) {
374 * Cannot unlock the mutex, so remove
375 * the running thread from the condition
378 cond_queue_remove(*cond, _thread_run);
380 /* Check for no more waiters: */
381 if (TAILQ_FIRST(&(*cond)->c_queue) == NULL)
382 (*cond)->c_mutex = NULL;
384 /* Unlock the condition variable structure: */
385 _SPINUNLOCK(&(*cond)->lock);
388 * Schedule the next thread and unlock
389 * the condition variable structure:
391 _thread_kern_sched_state_unlock(PS_COND_WAIT,
392 &(*cond)->lock, __FILE__, __LINE__);
395 * Check if the wait timedout or was
396 * interrupted (canceled):
398 if ((_thread_run->timeout == 0) &&
399 (_thread_run->interrupted == 0)) {
400 /* Lock the mutex: */
401 rval = _mutex_cv_lock(mutex);
405 * Remember if this thread was
408 interrupted = _thread_run->interrupted;
410 /* Lock the condition variable structure: */
411 _SPINLOCK(&(*cond)->lock);
414 * The wait timed out; remove
415 * the thread from the condition
418 cond_queue_remove(*cond,
421 /* Check for no more waiters: */
422 if (TAILQ_FIRST(&(*cond)->c_queue) == NULL)
423 (*cond)->c_mutex = NULL;
425 /* Unock the condition variable structure: */
426 _SPINUNLOCK(&(*cond)->lock);
428 /* Return a timeout error: */
432 * Lock the mutex and ignore any
433 * errors. Note that even though
434 * this thread may have been
435 * canceled, POSIX requires that
436 * the mutex be reaquired prior
439 (void)_mutex_cv_lock(mutex);
445 /* Trap invalid condition variable types: */
447 /* Unlock the condition variable structure: */
448 _SPINUNLOCK(&(*cond)->lock);
450 /* Return an invalid argument error: */
455 if (interrupted != 0) {
456 if (_thread_run->continuation != NULL)
457 _thread_run->continuation((void *) _thread_run);
461 _thread_leave_cancellation_point();
464 _thread_leave_cancellation_point();
466 /* Return the completion status: */
471 pthread_cond_signal(pthread_cond_t * cond)
476 if (cond == NULL || *cond == NULL)
480 * Defer signals to protect the scheduling queues
481 * from access by the signal handler:
483 _thread_kern_sig_defer();
485 /* Lock the condition variable structure: */
486 _SPINLOCK(&(*cond)->lock);
488 /* Process according to condition variable type: */
489 switch ((*cond)->c_type) {
490 /* Fast condition variable: */
492 if ((pthread = cond_queue_deq(*cond)) != NULL)
493 /* Allow the thread to run: */
494 PTHREAD_NEW_STATE(pthread,PS_RUNNING);
496 /* Check for no more waiters: */
497 if (TAILQ_FIRST(&(*cond)->c_queue) == NULL)
498 (*cond)->c_mutex = NULL;
501 /* Trap invalid condition variable types: */
503 /* Return an invalid argument error: */
508 /* Unlock the condition variable structure: */
509 _SPINUNLOCK(&(*cond)->lock);
512 * Undefer and handle pending signals, yielding if
515 _thread_kern_sig_undefer();
518 /* Return the completion status: */
523 pthread_cond_broadcast(pthread_cond_t * cond)
528 if (cond == NULL || *cond == NULL)
532 * Defer signals to protect the scheduling queues
533 * from access by the signal handler:
535 _thread_kern_sig_defer();
537 /* Lock the condition variable structure: */
538 _SPINLOCK(&(*cond)->lock);
540 /* Process according to condition variable type: */
541 switch ((*cond)->c_type) {
542 /* Fast condition variable: */
545 * Enter a loop to bring all threads off the
548 while ((pthread = cond_queue_deq(*cond)) != NULL) {
549 PTHREAD_NEW_STATE(pthread,PS_RUNNING);
552 /* There are no more waiting threads: */
553 (*cond)->c_mutex = NULL;
556 /* Trap invalid condition variable types: */
558 /* Return an invalid argument error: */
563 /* Unlock the condition variable structure: */
564 _SPINUNLOCK(&(*cond)->lock);
567 * Undefer and handle pending signals, yielding if
570 _thread_kern_sig_undefer();
573 /* Return the completion status: */
578 * Dequeue a waiting thread from the head of a condition queue in
579 * descending priority order.
581 static inline pthread_t
582 cond_queue_deq(pthread_cond_t cond)
586 while ((pthread = TAILQ_FIRST(&cond->c_queue)) != NULL) {
587 TAILQ_REMOVE(&cond->c_queue, pthread, qe);
588 pthread->flags &= ~PTHREAD_FLAGS_IN_CONDQ;
589 if ((pthread->timeout == 0) && (pthread->interrupted == 0))
591 * Only exit the loop when we find a thread
592 * that hasn't timed out or been canceled;
593 * those threads are already running and don't
594 * need their run state changed.
603 * Remove a waiting thread from a condition queue in descending priority
607 cond_queue_remove(pthread_cond_t cond, pthread_t pthread)
610 * Because pthread_cond_timedwait() can timeout as well
611 * as be signaled by another thread, it is necessary to
612 * guard against removing the thread from the queue if
613 * it isn't in the queue.
615 if (pthread->flags & PTHREAD_FLAGS_IN_CONDQ) {
616 TAILQ_REMOVE(&cond->c_queue, pthread, qe);
617 pthread->flags &= ~PTHREAD_FLAGS_IN_CONDQ;
622 * Enqueue a waiting thread to a condition queue in descending priority
626 cond_queue_enq(pthread_cond_t cond, pthread_t pthread)
628 pthread_t tid = TAILQ_LAST(&cond->c_queue, cond_head);
631 * For the common case of all threads having equal priority,
632 * we perform a quick check against the priority of the thread
633 * at the tail of the queue.
635 if ((tid == NULL) || (pthread->active_priority <= tid->active_priority))
636 TAILQ_INSERT_TAIL(&cond->c_queue, pthread, qe);
638 tid = TAILQ_FIRST(&cond->c_queue);
639 while (pthread->active_priority <= tid->active_priority)
640 tid = TAILQ_NEXT(tid, qe);
641 TAILQ_INSERT_BEFORE(tid, pthread, qe);
643 pthread->flags |= PTHREAD_FLAGS_IN_CONDQ;