#pragma once #include #include #include #include #include class AP_AutoTune { public: struct ATGains { AP_Float tau; AP_Int16 rmax_pos; AP_Int16 rmax_neg; float FF, P, I, D, IMAX; }; enum ATType { AUTOTUNE_ROLL = 0, AUTOTUNE_PITCH = 1 }; struct PACKED log_ATRP { LOG_PACKET_HEADER; uint64_t time_us; uint8_t type; uint8_t state; int16_t servo; float demanded; float achieved; float P; }; // constructor AP_AutoTune(ATGains &_gains, ATType type, const AP_Vehicle::FixedWing &parms, AC_PID &rpid); // called when autotune mode is entered void start(void); // called to stop autotune and restore gains when user leaves // autotune void stop(void); // update called whenever autotune mode is active. This is // called at the main loop rate void update(AP_Logger::PID_Info &pid_info, float scaler); // are we running? bool running; private: // the current gains ATGains ¤t; AC_PID &rpid; // what type of autotune is this ATType type; const AP_Vehicle::FixedWing &aparm; // values to restore if we leave autotune mode ATGains restore; // values we last saved ATGains last_save; // values to save on the next save event ATGains next_save; // time when we last saved uint32_t last_save_ms; // the demanded/achieved state enum class ATState {IDLE, DEMAND_POS, DEMAND_NEG}; ATState state; // the demanded/achieved state enum class Action {NONE, LOW_RATE, SHORT, RAISE_PD, LOWER_PD}; Action action; // when we entered the current state uint32_t state_enter_ms; void check_save(void); void check_state_exit(uint32_t state_time_ms); void save_gains(const ATGains &v); void save_float_if_changed(AP_Float &v, float value); void save_int16_if_changed(AP_Int16 &v, int16_t value); // get gains with PID components ATGains get_gains(const ATGains ¤t); void set_gains(const ATGains &v); // update rmax and tau towards target void update_rmax(); // 5 point mode filter for FF estimate ModeFilterFloat_Size5 ff_filter; LowPassFilterFloat actuator_filter; LowPassFilterFloat rate_filter; float max_actuator; float min_actuator; float max_rate; float min_rate; float max_target; float min_target; float max_P; float max_D; float min_Dmod; float FF_single; };