2019-10-16 20:36:42 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
* 2018 Freie Universität Berlin
|
|
|
|
* 2020 Inria
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @ingroup sys_event
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Event Loop Thread implementation
|
|
|
|
*
|
|
|
|
* @author Hauke Petersen <hauke.petersen@fu-berlin.de>
|
|
|
|
* @author Kaspar Schleiser <kaspar@schleiser.de>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
|
2020-11-18 20:48:16 +01:00
|
|
|
#include "architecture.h"
|
2019-10-16 20:36:42 +02:00
|
|
|
#include "thread.h"
|
|
|
|
#include "event.h"
|
|
|
|
#include "event/thread.h"
|
|
|
|
|
2020-11-18 20:48:16 +01:00
|
|
|
struct event_queue_and_size {
|
|
|
|
event_queue_t *q;
|
|
|
|
size_t q_numof;
|
|
|
|
};
|
|
|
|
|
|
|
|
static void *_handler_thread(void *tagged_ptr)
|
2019-10-16 20:36:42 +02:00
|
|
|
{
|
2020-11-18 20:48:16 +01:00
|
|
|
event_queue_t *qs = ptrtag_ptr(tagged_ptr);
|
|
|
|
/* number of queues is encoded in lower pointer bits */
|
|
|
|
size_t n = ptrtag_tag(tagged_ptr) + 1;
|
|
|
|
event_queues_claim(qs, n);
|
|
|
|
/* start event loop */
|
|
|
|
event_loop_multi(qs, n);
|
2019-10-16 20:36:42 +02:00
|
|
|
|
|
|
|
/* should be never reached */
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-11-18 20:48:16 +01:00
|
|
|
void event_thread_init_multi(event_queue_t *queues, size_t queues_numof,
|
|
|
|
char *stack, size_t stack_size, unsigned priority)
|
2019-10-16 20:36:42 +02:00
|
|
|
{
|
|
|
|
/* For the auto_init use case, this will be called before main gets
|
|
|
|
* started. main might already use the queues, so they need to be
|
|
|
|
* initialized at that point.
|
|
|
|
*
|
|
|
|
* They will be claimed within the handler thread.
|
|
|
|
*/
|
2020-11-18 20:48:16 +01:00
|
|
|
event_queues_init_detached(queues, queues_numof);
|
2019-10-16 20:36:42 +02:00
|
|
|
|
2020-11-18 20:48:16 +01:00
|
|
|
void *tagged_ptr = ptrtag(queues, queues_numof - 1);
|
|
|
|
|
2021-09-24 16:20:35 +02:00
|
|
|
thread_create(stack, stack_size, priority, THREAD_CREATE_STACKTEST,
|
|
|
|
_handler_thread, tagged_ptr, "event");
|
2019-10-16 20:36:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef EVENT_THREAD_STACKSIZE_DEFAULT
|
|
|
|
# ifdef ISR_STACKSIZE
|
|
|
|
# define EVENT_THREAD_STACKSIZE_DEFAULT ISR_STACKSIZE
|
|
|
|
# else
|
|
|
|
# define EVENT_THREAD_STACKSIZE_DEFAULT THREAD_STACKSIZE_SMALL
|
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef EVENT_THREAD_HIGHEST_STACKSIZE
|
|
|
|
#define EVENT_THREAD_HIGHEST_STACKSIZE EVENT_THREAD_STACKSIZE_DEFAULT
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef EVENT_THREAD_HIGHEST_PRIO
|
|
|
|
#define EVENT_THREAD_HIGHEST_PRIO (0)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef EVENT_THREAD_MEDIUM_STACKSIZE
|
|
|
|
#define EVENT_THREAD_MEDIUM_STACKSIZE EVENT_THREAD_STACKSIZE_DEFAULT
|
|
|
|
#endif
|
|
|
|
#ifndef EVENT_THREAD_MEDIUM_PRIO
|
|
|
|
#define EVENT_THREAD_MEDIUM_PRIO (THREAD_PRIORITY_MAIN - 1)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef EVENT_THREAD_LOWEST_STACKSIZE
|
|
|
|
#define EVENT_THREAD_LOWEST_STACKSIZE EVENT_THREAD_STACKSIZE_DEFAULT
|
|
|
|
#endif
|
|
|
|
#ifndef EVENT_THREAD_LOWEST_PRIO
|
|
|
|
#define EVENT_THREAD_LOWEST_PRIO (THREAD_PRIORITY_IDLE - 1)
|
|
|
|
#endif
|
|
|
|
|
2020-11-18 20:48:16 +01:00
|
|
|
/* rely on compiler / linker to garbage collect unused stacks */
|
|
|
|
static char WORD_ALIGNED _evq_highest_stack[EVENT_THREAD_HIGHEST_STACKSIZE];
|
|
|
|
static char WORD_ALIGNED _evq_medium_stack[EVENT_THREAD_MEDIUM_STACKSIZE];
|
|
|
|
static char WORD_ALIGNED _evq_lowest_stack[EVENT_THREAD_LOWEST_STACKSIZE];
|
2019-10-16 20:36:42 +02:00
|
|
|
|
2020-11-18 20:48:16 +01:00
|
|
|
event_queue_t event_thread_queues[EVENT_QUEUE_PRIO_NUMOF];
|
2019-10-16 20:36:42 +02:00
|
|
|
|
|
|
|
void auto_init_event_thread(void)
|
|
|
|
{
|
2020-11-18 20:48:16 +01:00
|
|
|
if (IS_USED(MODULE_EVENT_THREAD_HIGHEST)) {
|
|
|
|
/* In order to allow highest priority events to preempt all others,
|
|
|
|
* high priority events must be run in their own thread. This thread
|
|
|
|
* can preempt than preempt the other event thread(s). */
|
|
|
|
event_thread_init(EVENT_PRIO_HIGHEST,
|
|
|
|
_evq_highest_stack, sizeof(_evq_highest_stack),
|
|
|
|
EVENT_THREAD_HIGHEST_PRIO);
|
|
|
|
}
|
|
|
|
if (IS_USED(MODULE_EVENT_THREAD_MEDIUM)) {
|
|
|
|
/* In order to allow medium priority events to preempt low priority
|
|
|
|
* events, we need to move the low priority events into their own
|
|
|
|
* thread. The always existing medium priority event thread can then
|
|
|
|
* preempt the lowest priority event thread. */
|
|
|
|
event_thread_init(EVENT_PRIO_LOWEST,
|
|
|
|
_evq_lowest_stack, sizeof(_evq_lowest_stack),
|
|
|
|
EVENT_THREAD_LOWEST_PRIO);
|
|
|
|
}
|
|
|
|
|
|
|
|
event_queue_t *qs = EVENT_PRIO_MEDIUM;
|
|
|
|
size_t qs_numof = 1;
|
|
|
|
if (!IS_USED(MODULE_EVENT_THREAD_HIGHEST)) {
|
|
|
|
qs = EVENT_PRIO_HIGHEST;
|
|
|
|
qs_numof = 2;
|
|
|
|
}
|
|
|
|
if (!IS_USED(MODULE_EVENT_THREAD_MEDIUM)) {
|
|
|
|
qs_numof++;
|
2019-10-16 20:36:42 +02:00
|
|
|
}
|
2020-11-18 20:48:16 +01:00
|
|
|
event_thread_init_multi(qs, qs_numof,
|
|
|
|
_evq_medium_stack, sizeof(_evq_medium_stack),
|
|
|
|
EVENT_THREAD_MEDIUM_PRIO);
|
2019-10-16 20:36:42 +02:00
|
|
|
}
|