-
Notifications
You must be signed in to change notification settings - Fork 44
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
bkleiner
committed
Dec 10, 2023
1 parent
5ef1e42
commit 4dcf518
Showing
18 changed files
with
377 additions
and
117 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -91,4 +91,6 @@ void looptime_update() { | |
if (flags.arm_state) { | ||
state.armtime += state.looptime; | ||
} | ||
|
||
state.loop_counter++; | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,179 @@ | ||
#include "core/scheduler.h" | ||
|
||
#include <stdbool.h> | ||
#include <string.h> | ||
|
||
#include "core/debug.h" | ||
#include "core/looptime.h" | ||
#include "driver/time.h" | ||
#include "flight/control.h" | ||
#include "io/usb_configurator.h" | ||
#include "tasks.h" | ||
#include "util/cbor_helper.h" | ||
|
||
#define TASK_AVERAGE_SAMPLES 32 | ||
#define TASK_RUNTIME_REDUCTION 0.75 | ||
#define TASK_RUNTIME_MARGIN 1.25 | ||
#define TASK_RUNTIME_BUFFER 10 | ||
|
||
static FAST_RAM uint32_t task_queue_size = 0; | ||
static FAST_RAM task_t *task_queue[TASK_MAX]; | ||
static FAST_RAM task_t *active_task = NULL; | ||
|
||
static bool task_queue_contains(task_t *task) { | ||
for (uint32_t i = 0; i < task_queue_size; i++) { | ||
if (task_queue[i] == task) { | ||
return true; | ||
} | ||
} | ||
return false; | ||
} | ||
|
||
static bool task_queue_push(task_t *task) { | ||
if (task_queue_size >= TASK_MAX || task_queue_contains(task)) { | ||
return false; | ||
} | ||
for (uint32_t i = 0; i < TASK_MAX; i++) { | ||
if (task_queue[i] != NULL && task_queue[i]->priority <= task->priority) { | ||
continue; | ||
} | ||
|
||
memcpy(task_queue + i + 1, task_queue + i, (task_queue_size - i) * sizeof(task_t *)); | ||
task_queue[i] = task; | ||
task_queue_size++; | ||
return true; | ||
} | ||
return false; | ||
} | ||
|
||
static inline bool task_should_run(const uint32_t start_cycles, uint8_t task_mask, task_t *task) { | ||
if ((task_mask & task->mask) == 0) { | ||
// task shall not run in this firmware state | ||
return false; | ||
} | ||
|
||
const int32_t time_left = US_TO_CYCLES(state.looptime_autodetect - TASK_RUNTIME_BUFFER) - (time_cycles() - start_cycles); | ||
if (task->priority != TASK_PRIORITY_REALTIME && task->runtime_worst > time_left) { | ||
// we dont have any time left this loop and task is not realtime | ||
task->runtime_worst *= TASK_RUNTIME_REDUCTION; | ||
return false; | ||
} | ||
|
||
return true; | ||
} | ||
|
||
static inline void task_run(task_t *task) { | ||
const volatile uint32_t start = time_cycles(); | ||
|
||
task->flags = 0; | ||
active_task = task; | ||
task->func(); | ||
active_task = NULL; | ||
|
||
const volatile uint32_t time_taken = time_cycles() - start; | ||
task->runtime_current = time_taken; | ||
|
||
if (task->flags & TASK_FLAG_SKIP_STATS) { | ||
return; | ||
} | ||
|
||
if (time_taken < task->runtime_min) { | ||
task->runtime_min = time_taken; | ||
} | ||
|
||
task->runtime_avg_sum -= task->runtime_avg; | ||
task->runtime_avg_sum += time_taken; | ||
task->runtime_avg = task->runtime_avg_sum / TASK_AVERAGE_SAMPLES; | ||
|
||
if (time_taken > task->runtime_max) { | ||
task->runtime_max = time_taken; | ||
} | ||
|
||
if (task->runtime_worst < (task->runtime_avg * TASK_RUNTIME_MARGIN)) { | ||
task->runtime_worst = task->runtime_avg * TASK_RUNTIME_MARGIN; | ||
} | ||
} | ||
|
||
static inline uint8_t scheduler_task_mask() { | ||
uint8_t task_mask = TASK_MASK_DEFAULT; | ||
if (flags.in_air || flags.arm_state) { | ||
task_mask |= TASK_MASK_IN_AIR; | ||
} else { | ||
task_mask |= TASK_MASK_ON_GROUND; | ||
} | ||
return task_mask; | ||
} | ||
|
||
void task_reset_runtime() { | ||
if (active_task != NULL) { | ||
active_task->flags |= TASK_FLAG_SKIP_STATS; | ||
} | ||
looptime_reset(); | ||
} | ||
|
||
void scheduler_init() { | ||
looptime_init(); | ||
|
||
for (uint32_t i = 0; i < TASK_MAX; i++) { | ||
task_queue_push(&tasks[i]); | ||
} | ||
} | ||
|
||
void scheduler_run() { | ||
looptime_reset(); | ||
|
||
while (1) { | ||
const volatile uint32_t cycles = time_cycles(); | ||
const uint8_t task_mask = scheduler_task_mask(); | ||
for (uint32_t i = 0; i < task_queue_size; i++) { | ||
task_t *task = task_queue[i]; | ||
if (task_should_run(cycles, task_mask, task)) { | ||
task_run(task); | ||
} | ||
} | ||
|
||
looptime_update(); | ||
} | ||
} | ||
|
||
#ifdef DEBUG | ||
|
||
#define ENCODE_CYCLES(val) \ | ||
{ \ | ||
const uint32_t us = CYCLES_TO_US(val); \ | ||
CBOR_CHECK_ERROR(res = cbor_encode_uint32_t(enc, &us)); \ | ||
} | ||
|
||
cbor_result_t cbor_encode_task_stats(cbor_value_t *enc) { | ||
CBOR_CHECK_ERROR(cbor_result_t res = cbor_encode_array_indefinite(enc)); | ||
|
||
for (uint32_t i = 0; i < TASK_MAX; i++) { | ||
CBOR_CHECK_ERROR(res = cbor_encode_map_indefinite(enc)); | ||
|
||
CBOR_CHECK_ERROR(res = cbor_encode_str(enc, "name")); | ||
CBOR_CHECK_ERROR(res = cbor_encode_str(enc, tasks[i].name)); | ||
|
||
CBOR_CHECK_ERROR(res = cbor_encode_str(enc, "current")); | ||
ENCODE_CYCLES(tasks[i].runtime_current) | ||
|
||
CBOR_CHECK_ERROR(res = cbor_encode_str(enc, "min")); | ||
ENCODE_CYCLES(tasks[i].runtime_min) | ||
|
||
CBOR_CHECK_ERROR(res = cbor_encode_str(enc, "avg")); | ||
ENCODE_CYCLES(tasks[i].runtime_avg) | ||
|
||
CBOR_CHECK_ERROR(res = cbor_encode_str(enc, "max")); | ||
ENCODE_CYCLES(tasks[i].runtime_max) | ||
|
||
CBOR_CHECK_ERROR(res = cbor_encode_str(enc, "worst")); | ||
ENCODE_CYCLES(tasks[i].runtime_worst) | ||
|
||
CBOR_CHECK_ERROR(res = cbor_encode_end_indefinite(enc)); | ||
} | ||
|
||
CBOR_CHECK_ERROR(res = cbor_encode_end_indefinite(enc)); | ||
|
||
return res; | ||
} | ||
|
||
#endif |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
#pragma once | ||
|
||
#include <stdint.h> | ||
|
||
#include <cbor.h> | ||
|
||
void scheduler_init(); | ||
void scheduler_run(); | ||
|
||
void task_reset_runtime(); | ||
|
||
cbor_result_t cbor_encode_task_stats(cbor_value_t *enc); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,53 @@ | ||
#include "tasks.h" | ||
|
||
#include <stddef.h> | ||
|
||
#include "driver/serial.h" | ||
#include "driver/usb.h" | ||
#include "flight/control.h" | ||
#include "flight/gestures.h" | ||
#include "flight/imu.h" | ||
#include "flight/sixaxis.h" | ||
#include "io/blackbox.h" | ||
#include "io/buzzer.h" | ||
#include "io/led.h" | ||
#include "io/usb_configurator.h" | ||
#include "io/vbat.h" | ||
#include "io/vtx.h" | ||
#include "osd/render.h" | ||
#include "profile.h" | ||
#include "project.h" | ||
#include "rx/rx.h" | ||
|
||
void util_task() { | ||
// battery low logic | ||
vbat_calc(); | ||
|
||
// check gestures | ||
gestures(); | ||
|
||
// handle led commands | ||
led_update(); | ||
|
||
#if (RGB_LED_NUMBER > 0) | ||
// RGB led control | ||
rgb_led_lvc(); | ||
#ifdef RGB_LED_DMA | ||
rgb_dma_start(); | ||
#endif | ||
#endif | ||
|
||
buzzer_update(); | ||
} | ||
|
||
FAST_RAM task_t tasks[TASK_MAX] = { | ||
[TASK_GYRO] = CREATE_TASK("GYRO", TASK_MASK_ALWAYS, TASK_PRIORITY_REALTIME, sixaxis_read), | ||
[TASK_IMU] = CREATE_TASK("IMU", TASK_MASK_ALWAYS, TASK_PRIORITY_REALTIME, imu_calc), | ||
[TASK_PID] = CREATE_TASK("PID", TASK_MASK_ALWAYS, TASK_PRIORITY_REALTIME, control), | ||
[TASK_RX] = CREATE_TASK("RX", TASK_MASK_ALWAYS, TASK_PRIORITY_HIGH, rx_update), | ||
[TASK_UTIL] = CREATE_TASK("UTIL", TASK_MASK_ALWAYS, TASK_PRIORITY_HIGH, util_task), | ||
[TASK_BLACKBOX] = CREATE_TASK("BLACKBOX", TASK_MASK_ALWAYS, TASK_PRIORITY_MEDIUM, blackbox_update), | ||
[TASK_OSD] = CREATE_TASK("OSD", TASK_MASK_ALWAYS, TASK_PRIORITY_MEDIUM, osd_display), | ||
[TASK_VTX] = CREATE_TASK("VTX", TASK_MASK_ON_GROUND, TASK_PRIORITY_LOW, vtx_update), | ||
[TASK_USB] = CREATE_TASK("USB", TASK_MASK_ON_GROUND, TASK_PRIORITY_LOW, usb_configurator), | ||
}; |
Oops, something went wrong.