2014-04-16 22:45:42 +02:00
|
|
|
/*
|
|
|
|
* POSIX compatible implementation of barriers.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2014 Freie Universität Berlin
|
|
|
|
*
|
2014-05-14 09:49:09 +02:00
|
|
|
* This file is subject to the terms and conditions of the GNU Lesser General
|
2014-04-16 22:45:42 +02:00
|
|
|
* Public License. See the file LICENSE in the top level directory for more
|
|
|
|
* details.
|
|
|
|
*/
|
|
|
|
|
2014-03-04 01:43:58 +01:00
|
|
|
/**
|
2014-04-16 22:45:42 +02:00
|
|
|
* @ingroup pthread
|
|
|
|
* @{
|
|
|
|
* @file
|
|
|
|
* @brief Synchronization barriers.
|
|
|
|
* @author René Kijewski <kijewski@inf.fu-berlin.de>
|
|
|
|
* @}
|
|
|
|
*/
|
2014-03-04 01:43:58 +01:00
|
|
|
|
|
|
|
#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",
|
2014-04-10 22:28:35 +02:00
|
|
|
sched_active_thread->name, sched_active_thread->pid);
|
2014-03-04 01:43:58 +01:00
|
|
|
|
|
|
|
if (--barrier->count > 0) {
|
|
|
|
/* need to wait for further threads */
|
|
|
|
|
|
|
|
DEBUG("%s: waiting for %u threads. pid=%u\n",
|
2014-04-10 22:28:35 +02:00
|
|
|
sched_active_thread->name, barrier->count, sched_active_thread->pid);
|
2014-03-04 01:43:58 +01:00
|
|
|
|
|
|
|
pthread_barrier_waiting_node_t node;
|
2014-04-10 22:28:35 +02:00
|
|
|
node.pid = sched_active_pid;
|
2014-03-04 01:43:58 +01:00
|
|
|
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",
|
2014-04-10 22:28:35 +02:00
|
|
|
sched_active_thread->name, sched_active_thread->pid);
|
2014-03-04 01:43:58 +01:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|