306 lines
8.3 KiB
C++
306 lines
8.3 KiB
C++
// -*- tab-width: 4; Mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*-
|
|
|
|
/*
|
|
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/>.
|
|
*/
|
|
|
|
/**
|
|
The strategy for roll/pitch autotune is to give the user a AUTOTUNE
|
|
flight mode which behaves just like FBWA, but does automatic
|
|
tuning.
|
|
|
|
While the user is flying in AUTOTUNE the gains are saved every 10
|
|
seconds, but the saved gains are not the current gains, instead it
|
|
saves the gains from 10s ago. When the user exits AUTOTUNE the
|
|
gains are restored from 10s ago.
|
|
|
|
This allows the user to fly as much as they want in AUTOTUNE mode,
|
|
and if they are ever unhappy they just exit the mode. If they stay
|
|
in AUTOTUNE for more than 10s then their gains will have changed.
|
|
|
|
Using this approach users don't need any special switches, they
|
|
just need to be able to enter and exit AUTOTUNE mode
|
|
*/
|
|
|
|
#include <AP_HAL.h>
|
|
#include <AP_Common.h>
|
|
#include <AP_Math.h>
|
|
#include "AP_AutoTune.h"
|
|
|
|
extern const AP_HAL::HAL& hal;
|
|
|
|
// time in milliseconds between autotune saves
|
|
#define AUTOTUNE_SAVE_PERIOD 10000U
|
|
|
|
// how much time we have to overshoot for to reduce gains
|
|
#define AUTOTUNE_OVERSHOOT_TIME 100
|
|
|
|
// how much time we have to undershoot for to increase gains
|
|
#define AUTOTUNE_UNDERSHOOT_TIME 200
|
|
|
|
// step size for increasing gains
|
|
#define AUTOTUNE_INCREASE_STEP 0.05f
|
|
|
|
// step size for decreasing gains
|
|
#define AUTOTUNE_DECREASE_STEP 0.1f
|
|
|
|
// rate ranges
|
|
#define AUTOTUNE_MAX_RATE 90
|
|
#define AUTOTUNE_DEF_RATE 60
|
|
#define AUTOTUNE_MIN_RATE 30
|
|
|
|
// I ranges
|
|
#define AUTOTUNE_MAX_I 0.3
|
|
#define AUTOTUNE_DEF_I 0.1
|
|
#define AUTOTUNE_MIN_I 0.05
|
|
|
|
// D ranges
|
|
#define AUTOTUNE_MAX_D 0.15
|
|
#define AUTOTUNE_MIN_D 0.05
|
|
|
|
// min/max P gains
|
|
#define AUTOTUNE_MAX_P 4.0f
|
|
#define AUTOTUNE_MIN_P 0.3f
|
|
|
|
// tau ranges
|
|
#define AUTOTUNE_MAX_TAU 0.7
|
|
#define AUTOTUNE_MIN_TAU 0.2
|
|
|
|
#define AUTOTUNE_MIN_IMAX 2000
|
|
#define AUTOTUNE_MAX_IMAX 4000
|
|
|
|
// constructor
|
|
AP_AutoTune::AP_AutoTune(ATGains &_gains, ATType _type, DataFlash_Class &_dataflash) :
|
|
current(_gains),
|
|
type(_type),
|
|
dataflash(_dataflash)
|
|
{}
|
|
|
|
#if CONFIG_HAL_BOARD == HAL_BOARD_AVR_SITL
|
|
#include <stdio.h>
|
|
# define Debug(fmt, args ...) do {::printf("%s:%d: " fmt "\n", __FUNCTION__, __LINE__, ## args); } while(0)
|
|
#else
|
|
# define Debug(fmt, args ...)
|
|
#endif
|
|
|
|
/*
|
|
start an autotune session
|
|
*/
|
|
void AP_AutoTune::start(void)
|
|
{
|
|
running = true;
|
|
state = DEMAND_UNSATURATED;
|
|
uint32_t now = hal.scheduler->millis();
|
|
|
|
state_enter_ms = now;
|
|
last_save_ms = now;
|
|
|
|
last_save = current;
|
|
restore = current;
|
|
|
|
/*
|
|
set some reasonable values for I and D if the user hasn't set
|
|
them at all
|
|
*/
|
|
if (current.rmax < AUTOTUNE_MIN_RATE || current.rmax > AUTOTUNE_MAX_RATE) {
|
|
current.rmax.set(AUTOTUNE_DEF_RATE);
|
|
}
|
|
|
|
current.D = constrain_float(current.D, AUTOTUNE_MIN_D, AUTOTUNE_MAX_D);
|
|
current.tau = constrain_float(current.tau, AUTOTUNE_MIN_TAU, AUTOTUNE_MAX_TAU);
|
|
current.imax = constrain_float(current.imax, AUTOTUNE_MIN_IMAX, AUTOTUNE_MAX_IMAX);
|
|
|
|
// force a 45 degree phase
|
|
current.I = current.D / current.tau;
|
|
|
|
next_save = current;
|
|
|
|
Debug("START P -> %.3f\n", current.P.get());
|
|
}
|
|
|
|
/*
|
|
called when we change state to see if we should change gains
|
|
*/
|
|
void AP_AutoTune::stop(void)
|
|
{
|
|
running = false;
|
|
save_gains(restore);
|
|
}
|
|
|
|
|
|
/*
|
|
one update cycle of the autotuner
|
|
*/
|
|
void AP_AutoTune::update(float desired_rate, float achieved_rate, float servo_out)
|
|
{
|
|
if (!running) {
|
|
return;
|
|
}
|
|
check_save();
|
|
|
|
// see what state we are in
|
|
ATState new_state;
|
|
float abs_desired_rate = fabsf(desired_rate);
|
|
uint32_t now = hal.scheduler->millis();
|
|
|
|
if (fabsf(servo_out) >= 45) {
|
|
// we have saturated the servo demand (not including
|
|
// integrator), we cannot get any information that would allow
|
|
// us to increase the gain
|
|
saturated_surfaces = true;
|
|
}
|
|
|
|
if (abs_desired_rate < 0.8f * current.rmax) {
|
|
// we are not demanding max rate
|
|
new_state = DEMAND_UNSATURATED;
|
|
} else if (fabsf(achieved_rate) > abs_desired_rate) {
|
|
new_state = desired_rate > 0 ? DEMAND_OVER_POS : DEMAND_OVER_NEG;
|
|
} else {
|
|
new_state = desired_rate > 0 ? DEMAND_UNDER_POS : DEMAND_UNDER_NEG;
|
|
}
|
|
if (new_state != state) {
|
|
check_state_exit(now - state_enter_ms);
|
|
state = new_state;
|
|
state_enter_ms = now;
|
|
saturated_surfaces = false;
|
|
}
|
|
if (state != DEMAND_UNSATURATED) {
|
|
write_log(servo_out, desired_rate, achieved_rate);
|
|
}
|
|
}
|
|
|
|
/*
|
|
called when we change state to see if we should change gains
|
|
*/
|
|
void AP_AutoTune::check_state_exit(uint32_t state_time_ms)
|
|
{
|
|
switch (state) {
|
|
case DEMAND_UNSATURATED:
|
|
break;
|
|
case DEMAND_UNDER_POS:
|
|
case DEMAND_UNDER_NEG:
|
|
// we increase P if we have not saturated the surfaces during
|
|
// this state, and we have
|
|
if (state_time_ms >= AUTOTUNE_UNDERSHOOT_TIME && !saturated_surfaces) {
|
|
current.P += AUTOTUNE_INCREASE_STEP;
|
|
if (current.P > AUTOTUNE_MAX_P) {
|
|
current.P = AUTOTUNE_MAX_P;
|
|
}
|
|
Debug("UNDER P -> %.3f\n", current.P.get());
|
|
}
|
|
break;
|
|
case DEMAND_OVER_POS:
|
|
case DEMAND_OVER_NEG:
|
|
if (state_time_ms >= AUTOTUNE_OVERSHOOT_TIME) {
|
|
current.P -= AUTOTUNE_DECREASE_STEP;
|
|
if (current.P < AUTOTUNE_MIN_P) {
|
|
current.P = AUTOTUNE_MIN_P;
|
|
}
|
|
Debug("OVER P -> %.3f\n", current.P.get());
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
see if we should save new values
|
|
*/
|
|
void AP_AutoTune::check_save(void)
|
|
{
|
|
if (hal.scheduler->millis() - last_save_ms < AUTOTUNE_SAVE_PERIOD) {
|
|
return;
|
|
}
|
|
|
|
// save the next_save values, which are the autotune value from
|
|
// the last save period. This means the pilot has
|
|
// AUTOTUNE_SAVE_PERIOD milliseconds to decide they don't like the
|
|
// gains and switch out of autotune
|
|
ATGains tmp = current;
|
|
|
|
save_gains(next_save);
|
|
Debug("SAVE P -> %.3f\n", current.P.get());
|
|
|
|
// restore our current gains
|
|
current = tmp;
|
|
|
|
// if the pilot exits autotune they get these saved values
|
|
restore = next_save;
|
|
|
|
// the next values to save will be the ones we are flying now
|
|
next_save = current;
|
|
last_save_ms = hal.scheduler->millis();
|
|
}
|
|
|
|
/*
|
|
save a set of gains
|
|
*/
|
|
void AP_AutoTune::save_gains(const ATGains &v)
|
|
{
|
|
current = last_save;
|
|
current.tau.set_and_save_ifchanged(v.tau);
|
|
current.P.set_and_save_ifchanged(v.P);
|
|
current.I.set_and_save_ifchanged(v.I);
|
|
current.D.set_and_save_ifchanged(v.D);
|
|
current.rmax.set_and_save_ifchanged(v.rmax);
|
|
current.imax.set_and_save_ifchanged(v.imax);
|
|
last_save = current;
|
|
}
|
|
|
|
#define LOG_MSG_ATRP 211
|
|
|
|
struct PACKED log_ATRP {
|
|
LOG_PACKET_HEADER;
|
|
uint32_t timestamp;
|
|
uint8_t type;
|
|
uint8_t state;
|
|
int16_t servo;
|
|
float demanded;
|
|
float achieved;
|
|
float P;
|
|
};
|
|
|
|
static const struct LogStructure at_log_structures[] PROGMEM = {
|
|
{ LOG_MSG_ATRP, sizeof(log_ATRP),
|
|
"ATRP", "IBBcfff", "TimeMS,Type,State,Servo,Demanded,Achieved,P" },
|
|
};
|
|
|
|
void AP_AutoTune::write_log_headers(void)
|
|
{
|
|
if (!logging_started) {
|
|
logging_started = true;
|
|
dataflash.AddLogFormats(at_log_structures, 1);
|
|
}
|
|
}
|
|
|
|
void AP_AutoTune::write_log(float servo, float demanded, float achieved)
|
|
{
|
|
if (!dataflash.logging_started()) {
|
|
return;
|
|
}
|
|
|
|
write_log_headers();
|
|
|
|
struct log_ATRP pkt = {
|
|
LOG_PACKET_HEADER_INIT(LOG_MSG_ATRP),
|
|
timestamp : hal.scheduler->millis(),
|
|
type : type,
|
|
state : (uint8_t)state,
|
|
servo : (int16_t)(servo*100),
|
|
demanded : demanded,
|
|
achieved : achieved,
|
|
P : current.P.get()
|
|
};
|
|
dataflash.WriteBlock(&pkt, sizeof(pkt));
|
|
}
|