2017-10-22 13:54:43 +02:00
|
|
|
/*
|
2019-05-02 11:22:56 +02:00
|
|
|
* Copyright (C) 2017 Inria
|
|
|
|
* 2017 Kaspar Schleiser <kaspar@schleiser.de>
|
2018-05-24 14:16:53 +02:00
|
|
|
* 2018 Freie Universität Berlin
|
2017-10-22 13:54:43 +02:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2018-05-24 14:16:53 +02:00
|
|
|
/**
|
|
|
|
* @ingroup sys_event
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Event loop implementation
|
|
|
|
*
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2017-10-22 13:54:43 +02:00
|
|
|
#include <assert.h>
|
2023-10-06 18:21:34 +02:00
|
|
|
#include <stdbool.h>
|
2017-10-22 13:54:43 +02:00
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "event.h"
|
|
|
|
#include "clist.h"
|
2024-10-17 11:03:25 +02:00
|
|
|
#include "mutex.h"
|
2017-10-22 13:54:43 +02:00
|
|
|
#include "thread.h"
|
|
|
|
|
2021-01-18 17:37:02 +01:00
|
|
|
#if IS_USED(MODULE_XTIMER)
|
2019-04-25 13:45:15 +02:00
|
|
|
#include "xtimer.h"
|
|
|
|
#endif
|
|
|
|
|
2017-10-22 13:54:43 +02:00
|
|
|
void event_post(event_queue_t *queue, event_t *event)
|
|
|
|
{
|
2018-05-24 14:16:53 +02:00
|
|
|
assert(queue && event);
|
2017-10-22 13:54:43 +02:00
|
|
|
|
|
|
|
unsigned state = irq_disable();
|
2018-04-19 16:41:13 +02:00
|
|
|
if (!event->list_node.next) {
|
|
|
|
clist_rpush(&queue->event_list, &event->list_node);
|
|
|
|
}
|
2018-05-24 14:16:53 +02:00
|
|
|
thread_t *waiter = queue->waiter;
|
2017-10-22 13:54:43 +02:00
|
|
|
irq_restore(state);
|
|
|
|
|
2018-05-24 14:16:53 +02:00
|
|
|
if (waiter) {
|
|
|
|
thread_flags_set(waiter, THREAD_FLAG_EVENT);
|
|
|
|
}
|
2017-10-22 13:54:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void event_cancel(event_queue_t *queue, event_t *event)
|
|
|
|
{
|
|
|
|
assert(queue);
|
|
|
|
assert(event);
|
|
|
|
|
|
|
|
unsigned state = irq_disable();
|
|
|
|
clist_remove(&queue->event_list, &event->list_node);
|
|
|
|
event->list_node.next = NULL;
|
|
|
|
irq_restore(state);
|
|
|
|
}
|
|
|
|
|
2023-10-06 18:21:34 +02:00
|
|
|
bool event_is_queued(const event_queue_t *queue, const event_t *event)
|
|
|
|
{
|
|
|
|
assert(queue);
|
|
|
|
assert(event);
|
|
|
|
|
|
|
|
unsigned state = irq_disable();
|
|
|
|
bool result = clist_find(&queue->event_list, &event->list_node);
|
|
|
|
irq_restore(state);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2017-10-22 13:54:43 +02:00
|
|
|
event_t *event_get(event_queue_t *queue)
|
|
|
|
{
|
|
|
|
unsigned state = irq_disable();
|
|
|
|
event_t *result = (event_t *) clist_lpop(&queue->event_list);
|
|
|
|
irq_restore(state);
|
2018-05-24 14:16:53 +02:00
|
|
|
|
2017-10-22 13:54:43 +02:00
|
|
|
if (result) {
|
|
|
|
result->list_node.next = NULL;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-08-03 11:58:27 +02:00
|
|
|
event_t *event_wait_multi(event_queue_t *queues, size_t n_queues)
|
2017-10-22 13:54:43 +02:00
|
|
|
{
|
2020-08-03 11:58:27 +02:00
|
|
|
assert(queues && n_queues);
|
2021-03-03 22:28:31 +01:00
|
|
|
event_t *result = NULL;
|
2018-05-24 14:16:53 +02:00
|
|
|
|
|
|
|
do {
|
|
|
|
unsigned state = irq_disable();
|
2020-08-03 11:58:27 +02:00
|
|
|
for (size_t i = 0; i < n_queues; i++) {
|
2022-04-14 17:10:24 +02:00
|
|
|
assert(queues[i].waiter);
|
2020-08-03 11:58:27 +02:00
|
|
|
result = container_of(clist_lpop(&queues[i].event_list),
|
|
|
|
event_t, list_node);
|
|
|
|
if (result) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2018-05-24 14:16:53 +02:00
|
|
|
irq_restore(state);
|
|
|
|
if (result == NULL) {
|
|
|
|
thread_flags_wait_any(THREAD_FLAG_EVENT);
|
|
|
|
}
|
|
|
|
} while (result == NULL);
|
|
|
|
|
2017-10-22 13:54:43 +02:00
|
|
|
result->list_node.next = NULL;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-01-18 17:37:02 +01:00
|
|
|
#if IS_USED(MODULE_XTIMER) || IS_USED(MODULE_ZTIMER)
|
|
|
|
static event_t *_wait_timeout(event_queue_t *queue)
|
2019-04-25 13:45:15 +02:00
|
|
|
{
|
|
|
|
assert(queue);
|
2022-04-14 17:10:24 +02:00
|
|
|
assert(queue->waiter);
|
2019-04-25 13:45:15 +02:00
|
|
|
event_t *result;
|
|
|
|
thread_flags_t flags = 0;
|
|
|
|
|
|
|
|
do {
|
|
|
|
result = event_get(queue);
|
|
|
|
if (result == NULL) {
|
2021-01-18 17:37:02 +01:00
|
|
|
flags = thread_flags_wait_any(THREAD_FLAG_EVENT |
|
|
|
|
THREAD_FLAG_TIMEOUT);
|
2019-04-25 13:45:15 +02:00
|
|
|
}
|
|
|
|
} while ((result == NULL) && (flags & THREAD_FLAG_EVENT));
|
|
|
|
|
2021-01-18 17:37:02 +01:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if IS_USED(MODULE_XTIMER)
|
|
|
|
static event_t *_wait_timeout_xtimer(event_queue_t *queue, xtimer_t *timer)
|
|
|
|
{
|
|
|
|
event_t *result = _wait_timeout(queue);
|
2019-04-25 13:45:15 +02:00
|
|
|
if (result) {
|
2020-02-13 16:03:25 +01:00
|
|
|
xtimer_remove(timer);
|
2019-04-25 13:45:15 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2020-02-13 16:03:25 +01:00
|
|
|
|
|
|
|
event_t *event_wait_timeout(event_queue_t *queue, uint32_t timeout)
|
|
|
|
{
|
|
|
|
xtimer_t timer;
|
|
|
|
|
2021-01-18 17:37:02 +01:00
|
|
|
thread_flags_clear(THREAD_FLAG_TIMEOUT);
|
2020-02-13 16:03:25 +01:00
|
|
|
xtimer_set_timeout_flag(&timer, timeout);
|
2021-01-18 17:37:02 +01:00
|
|
|
return _wait_timeout_xtimer(queue, &timer);
|
2020-02-13 16:03:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
event_t *event_wait_timeout64(event_queue_t *queue, uint64_t timeout)
|
|
|
|
{
|
|
|
|
xtimer_t timer;
|
|
|
|
|
2021-01-18 17:37:02 +01:00
|
|
|
thread_flags_clear(THREAD_FLAG_TIMEOUT);
|
2020-02-13 16:03:25 +01:00
|
|
|
xtimer_set_timeout_flag64(&timer, timeout);
|
2021-01-18 17:37:02 +01:00
|
|
|
return _wait_timeout_xtimer(queue, &timer);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if IS_USED(MODULE_ZTIMER)
|
|
|
|
event_t *event_wait_timeout_ztimer(event_queue_t *queue,
|
|
|
|
ztimer_clock_t *clock, uint32_t timeout)
|
|
|
|
{
|
|
|
|
ztimer_t timer;
|
|
|
|
event_t *result;
|
|
|
|
|
|
|
|
thread_flags_clear(THREAD_FLAG_TIMEOUT);
|
|
|
|
ztimer_set_timeout_flag(clock, &timer, timeout);
|
|
|
|
result = _wait_timeout(queue);
|
|
|
|
if (result) {
|
|
|
|
ztimer_remove(clock, &timer);
|
|
|
|
}
|
|
|
|
return result;
|
2020-02-13 16:03:25 +01:00
|
|
|
}
|
2019-04-25 13:45:15 +02:00
|
|
|
#endif
|
2024-10-17 11:03:25 +02:00
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
event_t ev;
|
|
|
|
mutex_t synced;
|
|
|
|
} sync_ev_t;
|
|
|
|
|
|
|
|
static void sync_ev_handler(event_t *ev)
|
|
|
|
{
|
|
|
|
sync_ev_t *sync_ev = (sync_ev_t *)ev;
|
|
|
|
mutex_unlock(&sync_ev->synced);
|
|
|
|
}
|
|
|
|
|
|
|
|
void event_sync(event_queue_t *queue)
|
|
|
|
{
|
|
|
|
/* if we're on the queue, this would block forever */
|
|
|
|
assert(!queue->waiter || queue->waiter->pid != thread_getpid());
|
|
|
|
|
|
|
|
sync_ev_t sync_ev = {
|
|
|
|
.ev.handler = sync_ev_handler,
|
|
|
|
.synced = MUTEX_INIT_LOCKED,
|
|
|
|
};
|
|
|
|
event_post(queue, &sync_ev.ev);
|
|
|
|
mutex_lock(&sync_ev.synced);
|
|
|
|
}
|