2020-08-19 17:20:22 +02:00
|
|
|
/*
|
|
|
|
* Copyright (C) 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 pkg_openwsn
|
|
|
|
* @{
|
|
|
|
*
|
|
|
|
* @file
|
|
|
|
* @brief Event based implementation of OpenWSN scheduler
|
|
|
|
*
|
|
|
|
* @author Francisco Molina <francois-xavier.molina@inria.fr>
|
|
|
|
*
|
|
|
|
* @}
|
|
|
|
*/
|
|
|
|
#include "event.h"
|
|
|
|
#include "event/callback.h"
|
|
|
|
#include "memarray.h"
|
|
|
|
#include "periph/pm.h"
|
|
|
|
|
|
|
|
#include "scheduler.h"
|
|
|
|
#include "debugpins.h"
|
|
|
|
#include "leds.h"
|
|
|
|
|
|
|
|
#ifndef LOG_LEVEL
|
|
|
|
#define LOG_LEVEL LOG_ERROR
|
|
|
|
#endif
|
|
|
|
#include "log.h"
|
|
|
|
|
|
|
|
static scheduler_vars_t _scheduler_vars;
|
|
|
|
/* event queues for every priority */
|
|
|
|
static event_queue_t _queues[TASKPRIO_MAX];
|
|
|
|
|
|
|
|
#if SCHEDULER_DEBUG_ENABLE
|
|
|
|
scheduler_dbg_t scheduler_dbg;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
void scheduler_init(void)
|
|
|
|
{
|
|
|
|
#if SCHEDULER_DEBUG_ENABLE
|
|
|
|
memset(&scheduler_dbg, 0, sizeof(scheduler_dbg_t));
|
|
|
|
#endif
|
|
|
|
memset(&_scheduler_vars, 0, sizeof(scheduler_vars_t));
|
|
|
|
memarray_init(&_scheduler_vars.memarray, &_scheduler_vars.task_buff,
|
|
|
|
sizeof(event_callback_t), TASK_LIST_DEPTH);
|
|
|
|
|
|
|
|
event_queues_init_detached(_queues, TASKPRIO_MAX);
|
|
|
|
}
|
|
|
|
|
|
|
|
void scheduler_start(unsigned state)
|
|
|
|
{
|
|
|
|
irq_restore(state);
|
|
|
|
|
|
|
|
/* claim all queues */
|
|
|
|
event_queues_claim(_queues, TASKPRIO_MAX);
|
|
|
|
|
|
|
|
/* wait for events */
|
|
|
|
event_t *event;
|
|
|
|
while ((event = event_wait_multi(_queues, TASKPRIO_MAX))) {
|
2020-10-07 10:03:21 +02:00
|
|
|
debugpins_task_set();
|
2020-08-19 17:20:22 +02:00
|
|
|
event->handler(event);
|
|
|
|
/* remove from task list */
|
|
|
|
memarray_free(&_scheduler_vars.memarray, event);
|
|
|
|
#if SCHEDULER_DEBUG_ENABLE
|
|
|
|
scheduler_dbg.numTasksCur--;
|
|
|
|
#endif
|
2020-10-07 10:03:21 +02:00
|
|
|
debugpins_task_clr();
|
2020-08-19 17:20:22 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void scheduler_push_task(task_cbt cb, task_prio_t prio)
|
|
|
|
{
|
|
|
|
unsigned state = irq_disable();
|
|
|
|
/* get a free event from the queue */
|
|
|
|
event_callback_t *event = memarray_calloc(&_scheduler_vars.memarray);
|
|
|
|
|
|
|
|
if (!event) {
|
|
|
|
/* task list has overflown. This should never happen! */
|
|
|
|
LOG_ERROR("[openos/scheduler]: critical, task list overflow\n");
|
|
|
|
leds_error_blink();
|
|
|
|
pm_off();
|
|
|
|
}
|
|
|
|
|
|
|
|
event_callback_init(event, (void (*)(void *)) cb, NULL);
|
|
|
|
event_post(&_queues[prio], (event_t *)event);
|
|
|
|
|
|
|
|
#if SCHEDULER_DEBUG_ENABLE
|
|
|
|
scheduler_dbg.numTasksCur++;
|
|
|
|
if (scheduler_dbg.numTasksCur > scheduler_dbg.numTasksMax) {
|
|
|
|
scheduler_dbg.numTasksMax = scheduler_dbg.numTasksCur;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
irq_restore(state);
|
|
|
|
}
|