sched: Place common global variables into a struct
Some architectures do better if the global variables accessed from the timer dispatch code are in a struct. Signed-off-by: Kevin O'Connor <kevin@koconnor.net>
This commit is contained in:
parent
86fee2d517
commit
c015c0708f
76
src/sched.c
76
src/sched.c
|
@ -1,6 +1,6 @@
|
|||
// Basic scheduling functions and startup/shutdown code.
|
||||
//
|
||||
// Copyright (C) 2016,2017 Kevin O'Connor <kevin@koconnor.net>
|
||||
// Copyright (C) 2016-2021 Kevin O'Connor <kevin@koconnor.net>
|
||||
//
|
||||
// This file may be distributed under the terms of the GNU GPLv3 license.
|
||||
|
||||
|
@ -15,14 +15,19 @@
|
|||
#include "sched.h" // sched_check_periodic
|
||||
#include "stepper.h" // stepper_event
|
||||
|
||||
static struct timer periodic_timer, sentinel_timer, deleted_timer;
|
||||
|
||||
static struct {
|
||||
struct timer *timer_list;
|
||||
int8_t tasks_status;
|
||||
uint8_t shutdown_status, shutdown_reason;
|
||||
} SchedStatus = { .timer_list = &periodic_timer };
|
||||
|
||||
|
||||
/****************************************************************
|
||||
* Timers
|
||||
****************************************************************/
|
||||
|
||||
static struct timer periodic_timer, *timer_list = &periodic_timer;
|
||||
static struct timer sentinel_timer, deleted_timer;
|
||||
|
||||
// The periodic_timer simplifies the timer code by ensuring there is
|
||||
// always a timer on the timer list and that there is always a timer
|
||||
// not far in the future.
|
||||
|
@ -62,7 +67,7 @@ static struct timer sentinel_timer = {
|
|||
static void __always_inline
|
||||
insert_timer(struct timer *t, uint32_t waketime)
|
||||
{
|
||||
struct timer *prev, *pos = timer_list;
|
||||
struct timer *prev, *pos = SchedStatus.timer_list;
|
||||
for (;;) {
|
||||
prev = pos;
|
||||
if (CONFIG_MACH_AVR)
|
||||
|
@ -82,9 +87,9 @@ sched_add_timer(struct timer *add)
|
|||
{
|
||||
uint32_t waketime = add->waketime;
|
||||
irqstatus_t flag = irq_save();
|
||||
if (unlikely(timer_is_before(waketime, timer_list->waketime))) {
|
||||
if (unlikely(timer_is_before(waketime, SchedStatus.timer_list->waketime))) {
|
||||
// This timer is before all other scheduled timers
|
||||
struct timer *tl = timer_list;
|
||||
struct timer *tl = SchedStatus.timer_list;
|
||||
if (timer_is_before(waketime, timer_read_time()))
|
||||
try_shutdown("Timer too close");
|
||||
if (tl == &deleted_timer)
|
||||
|
@ -93,7 +98,7 @@ sched_add_timer(struct timer *add)
|
|||
add->next = tl;
|
||||
deleted_timer.waketime = waketime;
|
||||
deleted_timer.next = add;
|
||||
timer_list = &deleted_timer;
|
||||
SchedStatus.timer_list = &deleted_timer;
|
||||
timer_kick();
|
||||
} else {
|
||||
insert_timer(add, waketime);
|
||||
|
@ -117,15 +122,15 @@ void
|
|||
sched_del_timer(struct timer *del)
|
||||
{
|
||||
irqstatus_t flag = irq_save();
|
||||
if (timer_list == del) {
|
||||
if (SchedStatus.timer_list == del) {
|
||||
// Deleting the next active timer - replace with deleted_timer
|
||||
deleted_timer.waketime = del->waketime;
|
||||
deleted_timer.next = del->next;
|
||||
timer_list = &deleted_timer;
|
||||
SchedStatus.timer_list = &deleted_timer;
|
||||
} else {
|
||||
// Find and remove from timer list (if present)
|
||||
struct timer *pos;
|
||||
for (pos = timer_list; pos->next; pos = pos->next) {
|
||||
for (pos = SchedStatus.timer_list; pos->next; pos = pos->next) {
|
||||
if (pos->next == del) {
|
||||
pos->next = del->next;
|
||||
break;
|
||||
|
@ -140,7 +145,7 @@ unsigned int
|
|||
sched_timer_dispatch(void)
|
||||
{
|
||||
// Invoke timer callback
|
||||
struct timer *t = timer_list;
|
||||
struct timer *t = SchedStatus.timer_list;
|
||||
uint_fast8_t res;
|
||||
uint32_t updated_waketime;
|
||||
if (CONFIG_INLINE_STEPPER_HACK && likely(!t->func)) {
|
||||
|
@ -155,10 +160,10 @@ sched_timer_dispatch(void)
|
|||
unsigned int next_waketime = updated_waketime;
|
||||
if (unlikely(res == SF_DONE)) {
|
||||
next_waketime = t->next->waketime;
|
||||
timer_list = t->next;
|
||||
SchedStatus.timer_list = t->next;
|
||||
} else if (!timer_is_before(updated_waketime, t->next->waketime)) {
|
||||
next_waketime = t->next->waketime;
|
||||
timer_list = t->next;
|
||||
SchedStatus.timer_list = t->next;
|
||||
insert_timer(t, updated_waketime);
|
||||
}
|
||||
|
||||
|
@ -169,7 +174,7 @@ sched_timer_dispatch(void)
|
|||
void
|
||||
sched_timer_reset(void)
|
||||
{
|
||||
timer_list = &deleted_timer;
|
||||
SchedStatus.timer_list = &deleted_timer;
|
||||
deleted_timer.waketime = periodic_timer.waketime;
|
||||
deleted_timer.next = &periodic_timer;
|
||||
periodic_timer.next = &sentinel_timer;
|
||||
|
@ -181,8 +186,6 @@ sched_timer_reset(void)
|
|||
* Tasks
|
||||
****************************************************************/
|
||||
|
||||
static int_fast8_t tasks_status;
|
||||
|
||||
#define TS_IDLE -1
|
||||
#define TS_REQUESTED 0
|
||||
#define TS_RUNNING 1
|
||||
|
@ -191,14 +194,14 @@ static int_fast8_t tasks_status;
|
|||
void
|
||||
sched_wake_tasks(void)
|
||||
{
|
||||
tasks_status = TS_REQUESTED;
|
||||
SchedStatus.tasks_status = TS_REQUESTED;
|
||||
}
|
||||
|
||||
// Check if tasks need to be run
|
||||
uint8_t
|
||||
sched_tasks_busy(void)
|
||||
{
|
||||
return tasks_status >= TS_REQUESTED;
|
||||
return SchedStatus.tasks_status >= TS_REQUESTED;
|
||||
}
|
||||
|
||||
// Note that a task is ready to run
|
||||
|
@ -227,20 +230,20 @@ run_tasks(void)
|
|||
for (;;) {
|
||||
// Check if can sleep
|
||||
irq_poll();
|
||||
if (tasks_status != TS_REQUESTED) {
|
||||
if (SchedStatus.tasks_status != TS_REQUESTED) {
|
||||
start -= timer_read_time();
|
||||
irq_disable();
|
||||
if (tasks_status != TS_REQUESTED) {
|
||||
if (SchedStatus.tasks_status != TS_REQUESTED) {
|
||||
// Sleep processor (only run timers) until tasks woken
|
||||
tasks_status = TS_IDLE;
|
||||
SchedStatus.tasks_status = TS_IDLE;
|
||||
do {
|
||||
irq_wait();
|
||||
} while (tasks_status != TS_REQUESTED);
|
||||
} while (SchedStatus.tasks_status != TS_REQUESTED);
|
||||
}
|
||||
irq_enable();
|
||||
start += timer_read_time();
|
||||
}
|
||||
tasks_status = TS_RUNNING;
|
||||
SchedStatus.tasks_status = TS_RUNNING;
|
||||
|
||||
// Run all tasks
|
||||
extern void ctr_run_taskfuncs(void);
|
||||
|
@ -258,25 +261,23 @@ run_tasks(void)
|
|||
* Shutdown processing
|
||||
****************************************************************/
|
||||
|
||||
static uint_fast8_t shutdown_status, shutdown_reason;
|
||||
|
||||
// Return true if the machine is in an emergency stop state
|
||||
uint8_t
|
||||
sched_is_shutdown(void)
|
||||
{
|
||||
return !!shutdown_status;
|
||||
return !!SchedStatus.shutdown_status;
|
||||
}
|
||||
|
||||
// Transition out of shutdown state
|
||||
void
|
||||
sched_clear_shutdown(void)
|
||||
{
|
||||
if (!shutdown_status)
|
||||
if (!SchedStatus.shutdown_status)
|
||||
shutdown("Shutdown cleared when not shutdown");
|
||||
if (shutdown_status == 2)
|
||||
if (SchedStatus.shutdown_status == 2)
|
||||
// Ignore attempt to clear shutdown if still processing shutdown
|
||||
return;
|
||||
shutdown_status = 0;
|
||||
SchedStatus.shutdown_status = 0;
|
||||
}
|
||||
|
||||
// Invoke all shutdown functions (as declared by DECL_SHUTDOWN)
|
||||
|
@ -285,30 +286,31 @@ run_shutdown(int reason)
|
|||
{
|
||||
irq_disable();
|
||||
uint32_t cur = timer_read_time();
|
||||
if (!shutdown_status)
|
||||
shutdown_reason = reason;
|
||||
shutdown_status = 2;
|
||||
if (!SchedStatus.shutdown_status)
|
||||
SchedStatus.shutdown_reason = reason;
|
||||
SchedStatus.shutdown_status = 2;
|
||||
sched_timer_reset();
|
||||
extern void ctr_run_shutdownfuncs(void);
|
||||
ctr_run_shutdownfuncs();
|
||||
shutdown_status = 1;
|
||||
SchedStatus.shutdown_status = 1;
|
||||
irq_enable();
|
||||
|
||||
sendf("shutdown clock=%u static_string_id=%hu", cur, shutdown_reason);
|
||||
sendf("shutdown clock=%u static_string_id=%hu", cur
|
||||
, SchedStatus.shutdown_reason);
|
||||
}
|
||||
|
||||
// Report the last shutdown reason code
|
||||
void
|
||||
sched_report_shutdown(void)
|
||||
{
|
||||
sendf("is_shutdown static_string_id=%hu", shutdown_reason);
|
||||
sendf("is_shutdown static_string_id=%hu", SchedStatus.shutdown_reason);
|
||||
}
|
||||
|
||||
// Shutdown the machine if not already in the process of shutting down
|
||||
void __always_inline
|
||||
sched_try_shutdown(uint_fast8_t reason)
|
||||
{
|
||||
if (!shutdown_status)
|
||||
if (!SchedStatus.shutdown_status)
|
||||
sched_shutdown(reason);
|
||||
}
|
||||
|
||||
|
|
Loading…
Reference in New Issue