AP_Scheduler: allow specification of Scheduler table priorities

This commit is contained in:
Peter Barker 2021-11-14 14:18:09 +11:00 committed by Andrew Tridgell
parent 9a47a85c0d
commit e1310b2082
3 changed files with 141 additions and 24 deletions

View File

@ -107,16 +107,17 @@ void AP_Scheduler::init(const AP_Scheduler::Task *tasks, uint8_t num_tasks, uint
} }
_last_loop_time_s = 1.0 / _loop_rate_hz; _last_loop_time_s = 1.0 / _loop_rate_hz;
_vehicle_tasks = tasks;
_num_vehicle_tasks = num_tasks;
AP_Vehicle* vehicle = AP::vehicle(); AP_Vehicle* vehicle = AP::vehicle();
if (vehicle != nullptr) { if (vehicle != nullptr) {
vehicle->get_common_scheduler_tasks(_common_tasks, _num_tasks); vehicle->get_common_scheduler_tasks(_common_tasks, _num_common_tasks);
} }
_num_tasks += num_tasks;
_tasks = tasks; _num_tasks = _num_vehicle_tasks + _num_common_tasks;
_num_unshared_tasks = num_tasks;
_last_run = new uint16_t[_num_tasks]; _last_run = new uint16_t[_num_tasks];
memset(_last_run, 0, sizeof(_last_run[0]) * _num_tasks);
_tick_counter = 0; _tick_counter = 0;
// setup initial performance counters // setup initial performance counters
@ -128,6 +129,25 @@ void AP_Scheduler::init(const AP_Scheduler::Task *tasks, uint8_t num_tasks, uint
} }
_log_performance_bit = log_performance_bit; _log_performance_bit = log_performance_bit;
// sanity check the task lists to ensure the priorities are
// never decrease
uint8_t old = 0;
for (uint8_t i=0; i<_num_common_tasks; i++) {
if (_common_tasks[i].priority < old){
INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control);
break;
}
old = _common_tasks[i].priority;
}
old = 0;
for (uint8_t i=0; i<_num_vehicle_tasks; i++) {
if (_vehicle_tasks[i].priority < old) {
INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control);
break;
}
old = _vehicle_tasks[i].priority;
}
} }
// one tick has passed // one tick has passed
@ -157,8 +177,40 @@ void AP_Scheduler::run(uint32_t time_available)
uint32_t run_started_usec = AP_HAL::micros(); uint32_t run_started_usec = AP_HAL::micros();
uint32_t now = run_started_usec; uint32_t now = run_started_usec;
uint8_t vehicle_tasks_offset = 0;
uint8_t common_tasks_offset = 0;
for (uint8_t i=0; i<_num_tasks; i++) { for (uint8_t i=0; i<_num_tasks; i++) {
const AP_Scheduler::Task& task = (i < _num_unshared_tasks) ? _tasks[i] : _common_tasks[i - _num_unshared_tasks]; // determine which of the common task / vehicle task to run
bool run_vehicle_task = false;
if (vehicle_tasks_offset < _num_vehicle_tasks &&
common_tasks_offset < _num_common_tasks) {
// still have entries on both lists; compare the
// priorities. In case of a tie the vehicle-specific
// entry wins.
const Task &vehicle_task = _vehicle_tasks[vehicle_tasks_offset];
const Task &common_task = _common_tasks[common_tasks_offset];
if (vehicle_task.priority <= common_task.priority) {
run_vehicle_task = true;
}
} else if (vehicle_tasks_offset < _num_vehicle_tasks) {
// out of common tasks to run
run_vehicle_task = true;
} else if (common_tasks_offset < _num_common_tasks) {
// out of vehicle tasks to run
run_vehicle_task = false;
} else {
// this is an error; the outside loop should have terminated
INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control);
break;
}
const AP_Scheduler::Task &task = run_vehicle_task ? _vehicle_tasks[vehicle_tasks_offset] : _common_tasks[common_tasks_offset];
if (run_vehicle_task) {
vehicle_tasks_offset++;
} else {
common_tasks_offset++;
}
const uint16_t dt = _tick_counter - _last_run[i]; const uint16_t dt = _tick_counter - _last_run[i];
// we allow 0 to mean loop rate // we allow 0 to mean loop rate
@ -417,9 +469,54 @@ void AP_Scheduler::task_info(ExpandingString &str)
} }
} }
uint8_t vehicle_tasks_offset = 0;
uint8_t common_tasks_offset = 0;
for (uint8_t i = 0; i < _num_tasks + 1; i++) { for (uint8_t i = 0; i < _num_tasks + 1; i++) {
const char* task_name = (i < _num_unshared_tasks) ? _tasks[i].name : i == _num_tasks ? "fast_loop" : _common_tasks[i - _num_unshared_tasks].name; const AP::PerfInfo::TaskInfo* ti;
const AP::PerfInfo::TaskInfo* ti = perf_info.get_task_info(i); const char *task_name;
if (i == 0) {
// put the fast-loop entry at the top of the list - it's the last task in perf_info
ti = perf_info.get_task_info(_num_tasks);
task_name = "fast_loop";
} else {
ti = perf_info.get_task_info(i - 1);
// now find the task name:
// determine which of the common task / vehicle task to run
bool run_vehicle_task = false;
if (vehicle_tasks_offset < _num_vehicle_tasks &&
common_tasks_offset < _num_common_tasks) {
// still have entries on both lists; compare the
// priorities. In case of a tie the vehicle-specific
// entry wins.
const Task &vehicle_task = _vehicle_tasks[vehicle_tasks_offset];
const Task &common_task = _common_tasks[common_tasks_offset];
if (vehicle_task.priority <= common_task.priority) {
run_vehicle_task = true;
}
} else if (vehicle_tasks_offset < _num_vehicle_tasks) {
// out of common tasks to run
run_vehicle_task = true;
} else if (common_tasks_offset < _num_common_tasks) {
// out of vehicle tasks to run
run_vehicle_task = false;
} else {
// this is an error; the outside loop should have terminated
INTERNAL_ERROR(AP_InternalError::error_t::flow_of_control);
return;
}
if (run_vehicle_task) {
task_name = _vehicle_tasks[vehicle_tasks_offset++].name;
} else {
task_name = _common_tasks[common_tasks_offset++].name;
}
// the loop counter i is adjusted here because we emit the
// fast-loop entry first but it appears last in the
// perf_info list
}
uint16_t avg = 0; uint16_t avg = 0;
float pct = 0.0f; float pct = 0.0f;

View File

@ -40,11 +40,12 @@
/* /*
useful macro for creating scheduler task table useful macro for creating scheduler task table
*/ */
#define SCHED_TASK_CLASS(classname, classptr, func, _rate_hz, _max_time_micros) { \ #define SCHED_TASK_CLASS(classname, classptr, func, _rate_hz, _max_time_micros, _priority) { \
.function = FUNCTOR_BIND(classptr, &classname::func, void),\ .function = FUNCTOR_BIND(classptr, &classname::func, void),\
AP_SCHEDULER_NAME_INITIALIZER(classname, func)\ AP_SCHEDULER_NAME_INITIALIZER(classname, func)\
.rate_hz = _rate_hz,\ .rate_hz = _rate_hz,\
.max_time_micros = _max_time_micros\ .max_time_micros = _max_time_micros, \
.priority = _priority \
} }
/* /*
@ -81,6 +82,7 @@ public:
const char *name; const char *name;
float rate_hz; float rate_hz;
uint16_t max_time_micros; uint16_t max_time_micros;
uint8_t priority; // task priority
}; };
enum class Options : uint8_t { enum class Options : uint8_t {
@ -189,18 +191,17 @@ private:
// calculated loop period in seconds // calculated loop period in seconds
float _loop_period_s; float _loop_period_s;
// progmem list of tasks to run // list of tasks to run
const struct Task *_tasks; const struct Task *_vehicle_tasks;
uint8_t _num_vehicle_tasks;
// progmem list of common tasks to run // list of common tasks to run
const struct Task *_common_tasks; const struct Task *_common_tasks;
uint8_t _num_common_tasks;
// total number of tasks in _tasks and _common_tasks list // total number of tasks in _tasks and _common_tasks list
uint8_t _num_tasks; uint8_t _num_tasks;
// number of tasks in _tasks list
uint8_t _num_unshared_tasks;
// number of 'ticks' that have passed (number of times that // number of 'ticks' that have passed (number of times that
// tick() has been called // tick() has been called
uint16_t _tick_counter; uint16_t _tick_counter;

View File

@ -38,17 +38,36 @@ private:
static AP_BoardConfig board_config; static AP_BoardConfig board_config;
static SchedTest schedtest; static SchedTest schedtest;
#define SCHED_TASK(func, _interval_ticks, _max_time_micros) SCHED_TASK_CLASS(SchedTest, &schedtest, func, _interval_ticks, _max_time_micros) #define SCHED_TASK(func, _interval_ticks, _max_time_micros, _priority) SCHED_TASK_CLASS(SchedTest, &schedtest, func, _interval_ticks, _max_time_micros, _priority)
/* /*
scheduler table - all regular tasks are listed here, along with how scheduler table - all regular tasks should be listed here.
often they should be called (in 20ms units) and the maximum time
they are expected to take (in microseconds) All entries in this table must be ordered by priority.
This table is interleaved with the table in AP_Vehicle to determine
the order in which tasks are run. Convenience methods SCHED_TASK
and SCHED_TASK_CLASS are provided to build entries in this structure:
SCHED_TASK arguments:
- name of static function to call
- rate (in Hertz) at which the function should be called
- expected time (in MicroSeconds) that the function should take to run
- priority (0 through 255, lower number meaning higher priority)
SCHED_TASK_CLASS arguments:
- class name of method to be called
- instance on which to call the method
- method to call on that instance
- rate (in Hertz) at which the method should be called
- expected time (in MicroSeconds) that the method should take to run
- priority (0 through 255, lower number meaning higher priority)
*/ */
const AP_Scheduler::Task SchedTest::scheduler_tasks[] = { const AP_Scheduler::Task SchedTest::scheduler_tasks[] = {
SCHED_TASK(ins_update, 50, 1000), SCHED_TASK(ins_update, 50, 1000, 3),
SCHED_TASK(one_hz_print, 1, 1000), SCHED_TASK(one_hz_print, 1, 1000, 6),
SCHED_TASK(five_second_call, 0.2, 1800), SCHED_TASK(five_second_call, 0.2, 1800, 9),
}; };