/* * POSIX compatible implementation of barriers. * * Copyright (C) 2014 Freie Universität Berlin * * This file is subject to the terms and conditions of the GNU Lesser General * Public License. See the file LICENSE in the top level directory for more * details. */ /** * @ingroup pthread * @{ * @file * @brief Synchronization barriers. * @author René Kijewski * @} */ #include "sched.h" #include "pthread.h" #define ENABLE_DEBUG (0) #include "debug.h" int pthread_barrier_init(pthread_barrier_t *barrier, const pthread_barrierattr_t *attr, unsigned int count) { (void) attr; barrier->next = NULL; mutex_init(&barrier->mutex); barrier->count = count; return 0; } int pthread_barrier_destroy(pthread_barrier_t *barrier) { barrier->count = -1; return 0; } int pthread_barrier_wait(pthread_barrier_t *barrier) { /* Idea: the count is decreased by every thread that waits on the barrier. * If the value is bigger than zero afterwards, then the thread has to wait * to be woken up. Once the value reaches zero, everyone gets woken up. */ mutex_lock(&barrier->mutex); DEBUG("%s: hit a synchronization barrier. pid=%u\n", active_thread->name, active_thread->pid); if (--barrier->count > 0) { /* need to wait for further threads */ DEBUG("%s: waiting for %u threads. pid=%u\n", active_thread->name, barrier->count, active_thread->pid); pthread_barrier_waiting_node_t node; node.pid = thread_pid; node.next = barrier->next; node.cont = 0; barrier->next = &node; mutex_unlock(&barrier->mutex); while (1) { /* The mutex is reacquired before checking if we should continue, * so that the waiting thread don't accidentially run before the * wake up loop has ended. Otherwise the thread could run into the * the barrier again before `barrier->count` was reset. */ mutex_lock(&barrier->mutex); if (node.cont) { break; } mutex_unlock_and_sleep(&barrier->mutex); } } else { /* all threads have arrived, wake everybody up */ DEBUG("%s: waking every other thread up. pid=%u\n", active_thread->name, active_thread->pid); int count = 1; /* Count number of woken up threads. * The first thread is the current thread. */ pthread_barrier_waiting_node_t *next; for (next = barrier->next; next; next = next->next) { ++count; next->cont = 1; sched_set_status((tcb_t *) sched_threads[next->pid], STATUS_PENDING); } barrier->next = NULL; barrier->count = count; } mutex_unlock(&barrier->mutex); return 0; }