1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 09:52:45 +01:00
RIOT/core/include/cond.h
2021-08-13 19:50:38 +02:00

216 lines
6.8 KiB
C

/*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @defgroup core_sync_cond Condition Variable
* @ingroup core_sync
* @brief Condition variable for thread synchronization
*
* This file contains a condition variable with Mesa-style semantics.
*
* Condition variable solve the following problem. Suppose that a thread should
* sleep until a certain condition comes true. Condition variables provide a
* primitive whereby a thread can go to sleep by calling cond_wait(). Then,
* when the condition comes true in a thread or interrupt context, cond_signal()
* can be called, to wake up the thread.
*
* "Mesa-style semantics" means that, when cond_signal() is called, the
* sleeping thread becomes runnable, but may not be scheduled immediately. In
* contrast, "Hoare-style semantics" means that when cond_signal() is called,
* the sleeping thread is awakened and immediately scheduled. The condition
* variable in this file implements Mesa-style semantics, as is used by other
* standard implementations, such as pthreads.
*
* To avoid races, condition variables are used with mutexes. When a thread is
* put to sleep with cond_wait, it atomically unlocks the provided mutex and
* then goes to sleep. When it is awakened with cond_signal, it reacquires the
* mutex.
*
* As a rule of thumb, every condition variable should have a corresponding
* mutex, and that mutex should be held whenever performing any operation with
* the condition variable. There are exceptions to this rule, where it is
* appropriate to call cond_signal or cond_broadcast without the mutex held
* (for example, if you know that no thread will call cond_wait concurrently).
* It is safe to call cond_signal or cond_broadcast in interrupt context.
*
* However, the programmer should be aware of the following situation that
* could arise with Mesa-style condition variables: the condition may become
* true, making the sleeping thread runnable, but the condition may become
* false again before the thread is scheduled. To handle this case, the
* condition variable should be used in a while loop as follows:
*
* ```
* mutex_lock(&lock);
* while (condition_is_not_true) {
* cond_wait(&cond, &lock);
* }
* // do work while condition is true.
* mutex_unlock(&lock);
* ```
*
* When used in this way, the thread checks, once it has has awakened, whether
* the condition is actually true, and goes to sleep again if it is not. This
* is the standard way to use Mesa-style condition variables.
*
* Example: Suppose we want to implement a bounded queue, such as a Unix-style
* pipe between two threads. When data is written to the pipe, it is appended
* to a queue, and the writing thread blocks if the queue is full. When data
* is read from the pipe, it is removed from the queue; if the queue is empty,
* the reading thread blocks until it is not empty. If the pipe is closed by
* the sender, waiting reading threads wake up.
*
* Here is a sketch of how to implement such a structure with condition
* variables. For simplicity, messages are single bytes. We assume a FIFO data
* structure queue_t. We assume it is unsafe to add to the queue if it is full,
* or remove from the queue if it is empty.
*
* ```
* typedef struct pipe {
* queue_t queue;
* cond_t read_cond;
* cond_t write_cond;
* mutex_t lock;
* bool closed;
* } pipe_t;
*
* void pipe_init(pipe_t* pipe) {
* queue_init(&pipe->queue);
* cond_init(&pipe->read_cond);
* cond_init(&pipe->write_cond);
* mutex_init(&pipe->lock);
* pipe->closed = false;
* }
*
* void pipe_write(pipe_t* pipe, char c) {
* mutex_lock(&pipe->lock);
* while (queue_length(&pipe->queue) == MAX_QUEUE_LENGTH && !pipe->closed) {
* cond_wait(&pipe->write_cond, &pipe->lock);
* }
* if (pipe->closed) {
* mutex_unlock(&pipe->lock);
* return 0;
* }
* add_to_queue(&pipe->queue, c);
* cond_signal(&pipe->read_cond);
* mutex_unlock(&pipe->lock);
* return 1;
* }
*
* void pipe_close(pipe_t* pipe) {
* mutex_lock(&pipe->lock);
* pipe->closed = true;
* cond_broadcast(&pipe->read_cond);
* cond_broadcast(&pipe->write_cond);
* mutex_unlock(&pipe->lock);
* }
*
* int pipe_read(pipe_t* pipe, char* buf) {
* mutex_lock(&pipe->lock);
* while (queue_length(&pipe->queue) == 0 && !pipe->closed) {
* cond_wait(&pipe->read_cond, &pipe->lock);
* }
* if (pipe->closed) {
* mutex_unlock(&pipe->lock);
* return 0;
* }
* *buf = remove_from_queue(&pipe->queue);
* cond_signal(&pipe->write_cond);
* mutex_unlock(&pipe->lock);
* return 1;
* }
* ```
*
* Note that this could actually be written with a single condition variable.
* However, the example includes two for didactic reasons.
*
* @{
*
* @file
* @brief Condition variable for thread synchronization
*
* @author Sam Kumar <samkumar@berkeley.edu>
*/
#ifndef COND_H
#define COND_H
#include <stdbool.h>
#include <stddef.h>
#include "list.h"
#include "mutex.h"
#ifdef __cplusplus
extern "C" {
#endif
/**
* @brief Condition variable structure. Must never be modified by the user.
*/
typedef struct {
/**
* @brief The process waiting queue of the condition variable.
*
* @internal
*/
list_node_t queue;
} cond_t;
/**
* @brief Static initializer for cond_t.
*
* @note This initializer is preferable to cond_init().
*/
#define COND_INIT { { NULL } }
/**
* @brief Initializes a condition variable.
*
* @details For initialization of variables use COND_INIT instead.
* Only use the function call for dynamically allocated condition
* variables.
*
* @param[in] cond Pre-allocated condition structure. Must not be NULL.
*/
void cond_init(cond_t *cond);
/**
* @brief Waits on a condition.
*
* @param[in] cond Condition variable to wait on.
* @param[in] mutex Mutex object held by the current thread.
*/
void cond_wait(cond_t *cond, mutex_t *mutex);
/**
* @brief Wakes up one thread waiting on the condition variable.
*
* @details The thread is marked as runnable and will only be scheduled later
* at the scheduler's whim, so the thread should re-check the condition and wait
* again if it is not fulfilled.
*
* @param[in] cond Condition variable to signal.
*/
void cond_signal(cond_t *cond);
/**
* @brief Wakes up all threads waiting on the condition variable.
*
* @details The threads are marked as runnable and will only be scheduled later
* at the scheduler's whim, so they should re-check the condition and wait again
* if it is not fulfilled.
*
* @param[in] cond Condition variable to broadcast.
*/
void cond_broadcast(cond_t *cond);
#ifdef __cplusplus
}
#endif
#endif /* COND_H */
/** @} */