1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2024-12-29 04:50:03 +01:00
RIOT/sys/include/event.h
2020-02-13 16:18:32 +01:00

281 lines
8.0 KiB
C

/*
* Copyright (C) 2017 Inria
* 2017 Kaspar Schleiser <kaspar@schleiser.de>
* 2018-2019 Freie Universität Berlin
*
* 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 sys_event Event Queue
* @ingroup sys
* @brief Provides an Event loop
*
* This module offers an event queue framework like libevent or libuev.
*
* An event queue is basically a FIFO queue of events, with some functions to
* efficiently and safely handle adding and getting events to / from such a
* queue.
*
* An event queue is bound to a thread, but any thread or ISR can put events
* into a queue. In most cases, the owning thread of a queue is set during the
* queue's initialization. But it is also possible to initialize a queue in a
* detached state from a different context and to set the owning thread
* at a later point of time using the event_queue_claim() function.
*
* An event is a structure containing a pointer to an event handler. It can be
* extended to provide context or arguments to the handler. It can also be
* embedded into existing structures (see examples).
*
* Compared to msg or mbox, this some fundamental differences:
*
* 1. events are "sender allocated". Unlike msg_send(), event_post() never
* blocks or fails.
* 2. events contain everything necessary to handle them, thus a thread
* processing the events of an event queue doesn't need to be changed in
* order to support new event types.
* 3. events can be safely used (and actually perform best) when used within
* one thread, e.g., in order to create a state-machine like process flow.
* This is not (easily) possible using msg queues, as they might fill up.
* 4. an event can only be queued in one event queue at the same time.
* Notifying many queues using only one event object is not possible with
* this implementation.
*
* At the core, event_wait() uses thread flags to implement waiting for events
* to be queued. Thus event queues can be used safely and efficiently in combination
* with thread flags and msg queues.
*
* Examples:
*
* ~~~~~~~~~~~~~~~~~~~~~~~~ {.c}
* // simple event handler
* static void handler(event_t *event)
* {
* printf("triggered 0x%08x\n", (unsigned)event);
* }
*
* static event_t event = { .handler = handler };
* static event_queue_t queue;
*
* int main(void)
* {
* event_queue_init(&queue);
* event_loop(&queue);
* }
*
* [...] event_post(&queue, &event);
*
* // example for event extended event struct
* typedef struct {
* event_t super;
* const char *text;
* } custom_event_t;
*
* static void custom_handler(event_t *event)
* {
* custom_event_t *custom_event = (custom_event_t *)event;
* printf("triggered custom event with text: \"%s\"\n", custom_event->text);
* }
*
* static custom_event_t custom_event = { .super.handler = custom_handler, .text = "CUSTOM EVENT" };
*
* [...] event_post(&queue, &custom_event)
* ~~~~~~~~~~~~~~~~~~~~~~~~
*
* @{
*
* @file
* @brief Event API
*
* @author Kaspar Schleiser <kaspar@schleiser.de>
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
*/
#ifndef EVENT_H
#define EVENT_H
#include <stdint.h>
#include "irq.h"
#include "thread_flags.h"
#include "clist.h"
#ifdef __cplusplus
extern "C" {
#endif
#ifndef THREAD_FLAG_EVENT
/**
* @brief Thread flag use to notify available events in an event queue
*/
#define THREAD_FLAG_EVENT (0x1)
#endif
/**
* @brief event_queue_t static initializer
*/
#define EVENT_QUEUE_INIT { .waiter = (thread_t *)sched_active_thread }
/**
* @brief static initializer for detached event queues
*/
#define EVENT_QUEUE_INIT_DETACHED { .waiter = NULL }
/**
* @brief event structure forward declaration
*/
typedef struct event event_t;
/**
* @brief event handler type definition
*/
typedef void (*event_handler_t)(event_t *);
/**
* @brief event structure
*/
struct event {
clist_node_t list_node; /**< event queue list entry */
event_handler_t handler; /**< pointer to event handler function */
};
/**
* @brief event queue structure
*/
typedef struct {
clist_node_t event_list; /**< list of queued events */
thread_t *waiter; /**< thread ownning event queue */
} event_queue_t;
/**
* @brief Initialize an event queue
*
* This will set the calling thread as owner of @p queue.
*
* @param[out] queue event queue object to initialize
*/
void event_queue_init(event_queue_t *queue);
/**
* @brief Initialize an event queue not binding it to a thread
*
* @param[out] queue event queue object to initialize
*/
void event_queue_init_detached(event_queue_t *queue);
/**
* @brief Bind an event queue to the calling thread
*
* This function must only be called once and only if the given queue is not
* yet bound to a thread.
*
* @pre (queue->waiter == NULL)
*
* @param[out] queue event queue object to bind to a thread
*/
void event_queue_claim(event_queue_t *queue);
/**
* @brief Queue an event
*
* The given event will be posted on the given @p queue. If the event is already
* queued when calling this function, the event will not be touched and remain
* in the previous position on the queue. So reposting an event while it is
* already on the queue will have no effect.
*
* @param[in] queue event queue to queue event in
* @param[in] event event to queue in event queue
*/
void event_post(event_queue_t *queue, event_t *event);
/**
* @brief Cancel a queued event
*
* This will remove a queued event from an event queue.
*
* @note Due to the underlying list implementation, this will run in O(n).
*
* @param[in] queue event queue to remove event from
* @param[in] event event to remove from queue
*/
void event_cancel(event_queue_t *queue, event_t *event);
/**
* @brief Get next event from event queue, non-blocking
*
* In order to handle an event retrieved using this function,
* call event->handler(event).
*
* @param[in] queue event queue to get event from
*
* @returns pointer to next event
* @returns NULL if no event available
*/
event_t *event_get(event_queue_t *queue);
/**
* @brief Get next event from event queue, blocking
*
* This function will block until an event becomes available.
*
* In order to handle an event retrieved using this function,
* call event->handler(event).
*
* @note There can only be a single waiter on a queue!
*
* @param[in] queue event queue to get event from
*
* @returns pointer to next event
*/
event_t *event_wait(event_queue_t *queue);
#if defined(MODULE_XTIMER) || defined(DOXYGEN)
/**
* @brief Get next event from event queue, blocking until timeout expires
*
* @param[in] queue queue to query for an event
* @param[in] timeout maximum time to wait for an event to be posted in us
*
* @return pointer to next event if event was taken from the queue
* @return NULL if timeout expired before an event was posted
*/
event_t *event_wait_timeout(event_queue_t *queue, uint32_t timeout);
/**
* @brief Get next event from event queue, blocking until timeout expires
*
* @param[in] queue queue to query for an event
* @param[in] timeout maximum time to wait for an event to be posted in us
*
* @return pointer to next event if event was taken from the queue
* @return NULL if timeout expired before an event was posted
*/
event_t *event_wait_timeout64(event_queue_t *queue, uint64_t timeout);
#endif
/**
* @brief Simple event loop
*
* This function will forever sit in a loop, waiting for events to be queued
* and executing their handlers.
*
* It is pretty much defined as:
*
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~ {.c}
* while ((event = event_wait(queue))) {
* event->handler(event);
* }
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
* @param[in] queue event queue to process
*/
void event_loop(event_queue_t *queue);
#ifdef __cplusplus
}
#endif
#endif /* EVENT_H */
/** @} */