mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-18 14:48:28 -04:00
ae4bcf0f6f
this adds an extra budget of time per loop when we are not achieving scheduled tasks at a rate of at least 1/8 of the desired rate. This fixes an issue where a vehicle can become uncontrollable if the user asks for a SCHED_LOOP_RATE which is not achievable. As these events happen we add extra loop budget until we are able to run all tasks. We drop the extra time when the CPU pressure eases.
368 lines
12 KiB
C++
368 lines
12 KiB
C++
/*
|
|
This program is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/*
|
|
* main loop scheduler for APM
|
|
* Author: Andrew Tridgell, January 2013
|
|
*
|
|
*/
|
|
#include "AP_Scheduler.h"
|
|
|
|
#include <AP_HAL/AP_HAL.h>
|
|
#include <AP_Param/AP_Param.h>
|
|
#include <AP_Vehicle/AP_Vehicle.h>
|
|
#include <AP_Logger/AP_Logger.h>
|
|
#include <AP_InertialSensor/AP_InertialSensor.h>
|
|
#include <AP_InternalError/AP_InternalError.h>
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
|
|
#include <SITL/SITL.h>
|
|
#endif
|
|
#include <stdio.h>
|
|
|
|
#if APM_BUILD_TYPE(APM_BUILD_ArduCopter) || APM_BUILD_TYPE(APM_BUILD_ArduSub)
|
|
#define SCHEDULER_DEFAULT_LOOP_RATE 400
|
|
#else
|
|
#define SCHEDULER_DEFAULT_LOOP_RATE 50
|
|
#endif
|
|
|
|
#define debug(level, fmt, args...) do { if ((level) <= _debug.get()) { hal.console->printf(fmt, ##args); }} while (0)
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
const AP_Param::GroupInfo AP_Scheduler::var_info[] = {
|
|
// @Param: DEBUG
|
|
// @DisplayName: Scheduler debug level
|
|
// @Description: Set to non-zero to enable scheduler debug messages. When set to show "Slips" the scheduler will display a message whenever a scheduled task is delayed due to too much CPU load. When set to ShowOverruns the scheduled will display a message whenever a task takes longer than the limit promised in the task table.
|
|
// @Values: 0:Disabled,2:ShowSlips,3:ShowOverruns
|
|
// @User: Advanced
|
|
AP_GROUPINFO("DEBUG", 0, AP_Scheduler, _debug, 0),
|
|
|
|
// @Param: LOOP_RATE
|
|
// @DisplayName: Scheduling main loop rate
|
|
// @Description: This controls the rate of the main control loop in Hz. This should only be changed by developers. This only takes effect on restart. Values over 400 are considered highly experimental.
|
|
// @Values: 50:50Hz,100:100Hz,200:200Hz,250:250Hz,300:300Hz,400:400Hz
|
|
// @RebootRequired: True
|
|
// @User: Advanced
|
|
AP_GROUPINFO("LOOP_RATE", 1, AP_Scheduler, _loop_rate_hz, SCHEDULER_DEFAULT_LOOP_RATE),
|
|
|
|
AP_GROUPEND
|
|
};
|
|
|
|
// constructor
|
|
AP_Scheduler::AP_Scheduler(scheduler_fastloop_fn_t fastloop_fn) :
|
|
_fastloop_fn(fastloop_fn)
|
|
{
|
|
if (_singleton) {
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
|
|
AP_HAL::panic("Too many schedulers");
|
|
#endif
|
|
return;
|
|
}
|
|
_singleton = this;
|
|
|
|
AP_Param::setup_object_defaults(this, var_info);
|
|
|
|
// only allow 50 to 2000 Hz
|
|
if (_loop_rate_hz < 50) {
|
|
_loop_rate_hz.set(50);
|
|
} else if (_loop_rate_hz > 2000) {
|
|
_loop_rate_hz.set(2000);
|
|
}
|
|
_last_loop_time_s = 1.0 / _loop_rate_hz;
|
|
}
|
|
|
|
/*
|
|
* Get the AP_Scheduler singleton
|
|
*/
|
|
AP_Scheduler *AP_Scheduler::_singleton;
|
|
AP_Scheduler *AP_Scheduler::get_singleton()
|
|
{
|
|
return _singleton;
|
|
}
|
|
|
|
// initialise the scheduler
|
|
void AP_Scheduler::init(const AP_Scheduler::Task *tasks, uint8_t num_tasks, uint32_t log_performance_bit)
|
|
{
|
|
_tasks = tasks;
|
|
_num_tasks = num_tasks;
|
|
_last_run = new uint16_t[_num_tasks];
|
|
memset(_last_run, 0, sizeof(_last_run[0]) * _num_tasks);
|
|
_tick_counter = 0;
|
|
|
|
// setup initial performance counters
|
|
perf_info.set_loop_rate(get_loop_rate_hz());
|
|
perf_info.reset();
|
|
|
|
_log_performance_bit = log_performance_bit;
|
|
}
|
|
|
|
// one tick has passed
|
|
void AP_Scheduler::tick(void)
|
|
{
|
|
_tick_counter++;
|
|
}
|
|
|
|
/*
|
|
run one tick
|
|
this will run as many scheduler tasks as we can in the specified time
|
|
*/
|
|
void AP_Scheduler::run(uint32_t time_available)
|
|
{
|
|
uint32_t run_started_usec = AP_HAL::micros();
|
|
uint32_t now = run_started_usec;
|
|
|
|
if (_debug > 1 && _perf_counters == nullptr) {
|
|
_perf_counters = new AP_HAL::Util::perf_counter_t[_num_tasks];
|
|
if (_perf_counters != nullptr) {
|
|
for (uint8_t i=0; i<_num_tasks; i++) {
|
|
_perf_counters[i] = hal.util->perf_alloc(AP_HAL::Util::PC_ELAPSED, _tasks[i].name);
|
|
}
|
|
}
|
|
}
|
|
|
|
for (uint8_t i=0; i<_num_tasks; i++) {
|
|
uint32_t dt = _tick_counter - _last_run[i];
|
|
uint32_t interval_ticks = _loop_rate_hz / _tasks[i].rate_hz;
|
|
if (interval_ticks < 1) {
|
|
interval_ticks = 1;
|
|
}
|
|
if (dt < interval_ticks) {
|
|
// this task is not yet scheduled to run again
|
|
continue;
|
|
}
|
|
// this task is due to run. Do we have enough time to run it?
|
|
_task_time_allowed = _tasks[i].max_time_micros;
|
|
|
|
if (dt >= interval_ticks*2) {
|
|
// we've slipped a whole run of this task!
|
|
debug(2, "Scheduler slip task[%u-%s] (%u/%u/%u)\n",
|
|
(unsigned)i,
|
|
_tasks[i].name,
|
|
(unsigned)dt,
|
|
(unsigned)interval_ticks,
|
|
(unsigned)_task_time_allowed);
|
|
}
|
|
|
|
if (dt >= interval_ticks*max_task_slowdown) {
|
|
// we are going beyond the maximum slowdown factor for a
|
|
// task. This will trigger increasing the time budget
|
|
task_not_achieved++;
|
|
}
|
|
|
|
if (_task_time_allowed > time_available) {
|
|
// not enough time to run this task. Continue loop -
|
|
// maybe another task will fit into time remaining
|
|
continue;
|
|
}
|
|
|
|
// run it
|
|
_task_time_started = now;
|
|
hal.util->persistent_data.scheduler_task = i;
|
|
if (_debug > 1 && _perf_counters && _perf_counters[i]) {
|
|
hal.util->perf_begin(_perf_counters[i]);
|
|
}
|
|
_tasks[i].function();
|
|
if (_debug > 1 && _perf_counters && _perf_counters[i]) {
|
|
hal.util->perf_end(_perf_counters[i]);
|
|
}
|
|
hal.util->persistent_data.scheduler_task = -1;
|
|
|
|
// record the tick counter when we ran. This drives
|
|
// when we next run the event
|
|
_last_run[i] = _tick_counter;
|
|
|
|
// work out how long the event actually took
|
|
now = AP_HAL::micros();
|
|
uint32_t time_taken = now - _task_time_started;
|
|
|
|
if (time_taken > _task_time_allowed) {
|
|
// the event overran!
|
|
debug(3, "Scheduler overrun task[%u-%s] (%u/%u)\n",
|
|
(unsigned)i,
|
|
_tasks[i].name,
|
|
(unsigned)time_taken,
|
|
(unsigned)_task_time_allowed);
|
|
}
|
|
if (time_taken >= time_available) {
|
|
time_available = 0;
|
|
break;
|
|
}
|
|
time_available -= time_taken;
|
|
}
|
|
|
|
// update number of spare microseconds
|
|
_spare_micros += time_available;
|
|
|
|
_spare_ticks++;
|
|
if (_spare_ticks == 32) {
|
|
_spare_ticks /= 2;
|
|
_spare_micros /= 2;
|
|
}
|
|
}
|
|
|
|
/*
|
|
return number of micros until the current task reaches its deadline
|
|
*/
|
|
uint16_t AP_Scheduler::time_available_usec(void)
|
|
{
|
|
uint32_t dt = AP_HAL::micros() - _task_time_started;
|
|
if (dt > _task_time_allowed) {
|
|
return 0;
|
|
}
|
|
return _task_time_allowed - dt;
|
|
}
|
|
|
|
/*
|
|
calculate load average as a number from 0 to 1
|
|
*/
|
|
float AP_Scheduler::load_average()
|
|
{
|
|
if (_spare_ticks == 0) {
|
|
return 0.0f;
|
|
}
|
|
const uint32_t loop_us = get_loop_period_us();
|
|
const uint32_t used_time = loop_us - (_spare_micros/_spare_ticks);
|
|
return used_time / (float)loop_us;
|
|
}
|
|
|
|
void AP_Scheduler::loop()
|
|
{
|
|
// wait for an INS sample
|
|
hal.util->persistent_data.scheduler_task = -3;
|
|
AP::ins().wait_for_sample();
|
|
hal.util->persistent_data.scheduler_task = -1;
|
|
|
|
const uint32_t sample_time_us = AP_HAL::micros();
|
|
|
|
if (_loop_timer_start_us == 0) {
|
|
_loop_timer_start_us = sample_time_us;
|
|
_last_loop_time_s = get_loop_period_s();
|
|
} else {
|
|
_last_loop_time_s = (sample_time_us - _loop_timer_start_us) * 1.0e-6;
|
|
}
|
|
|
|
// Execute the fast loop
|
|
// ---------------------
|
|
if (_fastloop_fn) {
|
|
hal.util->persistent_data.scheduler_task = -2;
|
|
_fastloop_fn();
|
|
hal.util->persistent_data.scheduler_task = -1;
|
|
}
|
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
|
|
{
|
|
/*
|
|
for testing low CPU conditions we can add an optional delay in SITL
|
|
*/
|
|
auto *sitl = AP::sitl();
|
|
uint32_t loop_delay_us = sitl->loop_delay.get();
|
|
hal.scheduler->delay_microseconds(loop_delay_us);
|
|
}
|
|
#endif
|
|
|
|
// tell the scheduler one tick has passed
|
|
tick();
|
|
|
|
// run all the tasks that are due to run. Note that we only
|
|
// have to call this once per loop, as the tasks are scheduled
|
|
// in multiples of the main loop tick. So if they don't run on
|
|
// the first call to the scheduler they won't run on a later
|
|
// call until scheduler.tick() is called again
|
|
const uint32_t loop_us = get_loop_period_us();
|
|
uint32_t now = AP_HAL::micros();
|
|
uint32_t time_available = 0;
|
|
if (now - sample_time_us < loop_us) {
|
|
// get remaining time available for this loop
|
|
time_available = loop_us - (now - sample_time_us);
|
|
}
|
|
|
|
// add in extra loop time determined by not achieving scheduler tasks
|
|
time_available += extra_loop_us;
|
|
|
|
// run the tasks
|
|
run(time_available);
|
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL
|
|
// move result of AP_HAL::micros() forward:
|
|
hal.scheduler->delay_microseconds(1);
|
|
#endif
|
|
|
|
if (task_not_achieved > 0) {
|
|
// add some extra time to the budget
|
|
extra_loop_us = MIN(extra_loop_us+100U, 5000U);
|
|
task_not_achieved = 0;
|
|
task_all_achieved = 0;
|
|
} else if (extra_loop_us > 0) {
|
|
task_all_achieved++;
|
|
if (task_all_achieved > 50) {
|
|
// we have gone through 50 loops without a task taking too
|
|
// long. CPU pressure has eased, so drop the extra time we're
|
|
// giving each loop
|
|
task_all_achieved = 0;
|
|
// we are achieving all tasks, slowly lower the extra loop time
|
|
extra_loop_us = MAX(0U, extra_loop_us-50U);
|
|
}
|
|
}
|
|
|
|
// check loop time
|
|
perf_info.check_loop_time(sample_time_us - _loop_timer_start_us);
|
|
|
|
_loop_timer_start_us = sample_time_us;
|
|
}
|
|
|
|
void AP_Scheduler::update_logging()
|
|
{
|
|
if (debug_flags()) {
|
|
perf_info.update_logging();
|
|
}
|
|
if (_log_performance_bit != (uint32_t)-1 &&
|
|
AP::logger().should_log(_log_performance_bit)) {
|
|
Log_Write_Performance();
|
|
}
|
|
perf_info.set_loop_rate(get_loop_rate_hz());
|
|
perf_info.reset();
|
|
}
|
|
|
|
// Write a performance monitoring packet
|
|
void AP_Scheduler::Log_Write_Performance()
|
|
{
|
|
const AP_HAL::Util::PersistentData &pd = hal.util->persistent_data;
|
|
struct log_Performance pkt = {
|
|
LOG_PACKET_HEADER_INIT(LOG_PERFORMANCE_MSG),
|
|
time_us : AP_HAL::micros64(),
|
|
num_long_running : perf_info.get_num_long_running(),
|
|
num_loops : perf_info.get_num_loops(),
|
|
max_time : perf_info.get_max_time(),
|
|
mem_avail : hal.util->available_memory(),
|
|
load : (uint16_t)(load_average() * 1000),
|
|
internal_errors : AP::internalerror().errors(),
|
|
internal_error_count : AP::internalerror().count(),
|
|
spi_count : pd.spi_count,
|
|
i2c_count : pd.i2c_count,
|
|
i2c_isr_count : pd.i2c_isr_count,
|
|
extra_loop_us : extra_loop_us,
|
|
};
|
|
AP::logger().WriteCriticalBlock(&pkt, sizeof(pkt));
|
|
}
|
|
|
|
namespace AP {
|
|
|
|
AP_Scheduler &scheduler()
|
|
{
|
|
return *AP_Scheduler::get_singleton();
|
|
}
|
|
|
|
};
|