2020-10-19 23:51:05 -03:00
/*
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/>.
*/
/*
support for autotune of helicopters . Based on original autotune code from ArduCopter , written by Leonard Hall
Converted to a library by Andrew Tridgell , and rewritten to include helicopters by Bill Geyer
*/
2023-12-07 20:03:15 -04:00
# include "AC_AutoTune_config.h"
# if AC_AUTOTUNE_ENABLED
2021-01-17 22:28:20 -04:00
# include "AC_AutoTune_Heli.h"
2022-03-03 23:29:45 -04:00
# include <AP_Logger/AP_Logger.h>
2022-11-08 06:10:56 -04:00
# include <GCS_MAVLink/GCS.h>
2022-03-03 23:29:45 -04:00
2021-12-28 00:23:19 -04:00
# define AUTOTUNE_TESTING_STEP_TIMEOUT_MS 5000U // timeout for tuning mode's testing step
2020-10-19 23:51:05 -03:00
# define AUTOTUNE_RD_STEP 0.0005f // minimum increment when increasing/decreasing Rate D term
# define AUTOTUNE_RP_STEP 0.005f // minimum increment when increasing/decreasing Rate P term
# define AUTOTUNE_SP_STEP 0.05f // minimum increment when increasing/decreasing Stab P term
# define AUTOTUNE_PI_RATIO_FOR_TESTING 0.1f // I is set 10x smaller than P during testing
2021-12-27 23:52:15 -04:00
# define AUTOTUNE_PI_RATIO_FINAL 1.0f // I is set 1x P after testing
# define AUTOTUNE_YAW_PI_RATIO_FINAL 0.1f // I is set 1x P after testing
2020-10-19 23:51:05 -03:00
# define AUTOTUNE_RD_MAX 0.020f // maximum Rate D value
2022-01-07 14:37:02 -04:00
# define AUTOTUNE_RP_MIN 0.02f // minimum Rate P value
2021-01-18 00:16:08 -04:00
# define AUTOTUNE_RP_MAX 0.3f // maximum Rate P value
# define AUTOTUNE_SP_MAX 10.0f // maximum Stab P value
# define AUTOTUNE_SP_MIN 3.0f // maximum Stab P value
# define AUTOTUNE_RFF_MAX 0.5f // maximum Stab P value
# define AUTOTUNE_RFF_MIN 0.025f // maximum Stab P value
2021-12-27 00:53:50 -04:00
# define AUTOTUNE_RD_BACKOFF 1.0f // Rate D gains are reduced to 50% of their maximum value discovered during tuning
# define AUTOTUNE_RP_BACKOFF 1.0f // Rate P gains are reduced to 97.5% of their maximum value discovered during tuning
2021-12-27 23:52:15 -04:00
# define AUTOTUNE_SP_BACKOFF 1.0f // Stab P gains are reduced to 90% of their maximum value discovered during tuning
2021-12-27 00:53:50 -04:00
# define AUTOTUNE_ACCEL_RP_BACKOFF 1.0f // back off from maximum acceleration
# define AUTOTUNE_ACCEL_Y_BACKOFF 1.0f // back off from maximum acceleration
2021-12-27 23:52:15 -04:00
2022-02-01 18:03:38 -04:00
# define AUTOTUNE_HELI_TARGET_ANGLE_RLLPIT_CD 1500 // target roll/pitch angle during AUTOTUNE FeedForward rate test
2021-12-27 23:52:15 -04:00
# define AUTOTUNE_HELI_TARGET_RATE_RLLPIT_CDS 5000 // target roll/pitch rate during AUTOTUNE FeedForward rate test
# define AUTOTUNE_FFI_RATIO_FOR_TESTING 0.5f // I is set 2x smaller than VFF during testing
# define AUTOTUNE_FFI_RATIO_FINAL 0.5f // I is set 0.5x VFF after testing
2021-12-27 00:53:50 -04:00
# define AUTOTUNE_RP_ACCEL_MIN 20000.0f // Minimum acceleration for Roll and Pitch
# define AUTOTUNE_Y_ACCEL_MIN 10000.0f // Minimum acceleration for Yaw
2021-01-18 00:16:08 -04:00
# define AUTOTUNE_SEQ_BITMASK_VFF 1
# define AUTOTUNE_SEQ_BITMASK_RATE_D 2
# define AUTOTUNE_SEQ_BITMASK_ANGLE_P 4
# define AUTOTUNE_SEQ_BITMASK_MAX_GAIN 8
2022-05-08 00:28:50 -03:00
# define AUTOTUNE_SEQ_BITMASK_TUNE_CHECK 16
2021-01-18 00:16:08 -04:00
2024-02-29 02:42:39 -04:00
// angle limits preserved from previous behaviour as Multi changed:
# define AUTOTUNE_ANGLE_TARGET_MAX_RP_CD 2000 // target angle during TESTING_RATE step that will cause us to move to next step
# define AUTOTUNE_ANGLE_TARGET_MIN_RP_CD 1000 // minimum target angle during TESTING_RATE step that will cause us to move to next step
# define AUTOTUNE_ANGLE_TARGET_MAX_Y_CD 3000 // target angle during TESTING_RATE step that will cause us to move to next step
# define AUTOTUNE_ANGLE_TARGET_MIN_Y_CD 500 // target angle during TESTING_RATE step that will cause us to move to next step
# define AUTOTUNE_ANGLE_MAX_RP_CD 3000 // maximum allowable angle in degrees during testing
# define AUTOTUNE_ANGLE_NEG_RPY_CD 1000 // maximum allowable angle in degrees during testing
2021-01-18 00:16:08 -04:00
const AP_Param : : GroupInfo AC_AutoTune_Heli : : var_info [ ] = {
2021-10-01 17:35:52 -03:00
// @Param: AXES
// @DisplayName: Autotune axis bitmask
// @Description: 1-byte bitmap of axes to autotune
// @Bitmask: 0:Roll,1:Pitch,2:Yaw
// @User: Standard
2022-02-06 23:45:24 -04:00
AP_GROUPINFO ( " AXES " , 1 , AC_AutoTune_Heli , axis_bitmask , 1 ) ,
2021-01-18 00:16:08 -04:00
// @Param: SEQ
// @DisplayName: AutoTune Sequence Bitmask
2022-05-08 00:28:50 -03:00
// @Description: 2-byte bitmask to select what tuning should be performed. Max gain automatically performed if Rate D is selected. Values: 7:All,1:VFF Only,2:Rate D/Rate P Only(incl max gain),4:Angle P Only,8:Max Gain Only,16:Tune Check,3:VFF and Rate D/Rate P(incl max gain),5:VFF and Angle P,6:Rate D/Rate P(incl max gain) and angle P
// @Bitmask: 0:VFF,1:Rate D/Rate P(incl max gain),2:Angle P,3:Max Gain Only,4:Tune Check
2021-01-18 00:16:08 -04:00
// @User: Standard
2022-01-23 00:36:34 -04:00
AP_GROUPINFO ( " SEQ " , 2 , AC_AutoTune_Heli , seq_bitmask , 3 ) ,
2021-01-18 00:16:08 -04:00
2021-12-28 00:23:19 -04:00
// @Param: FRQ_MIN
2021-01-18 00:16:08 -04:00
// @DisplayName: AutoTune minimum sweep frequency
// @Description: Defines the start frequency for sweeps and dwells
// @Range: 10 30
// @User: Standard
2021-12-28 00:23:19 -04:00
AP_GROUPINFO ( " FRQ_MIN " , 3 , AC_AutoTune_Heli , min_sweep_freq , 10.0f ) ,
2021-01-18 00:16:08 -04:00
2021-12-28 00:23:19 -04:00
// @Param: FRQ_MAX
2021-01-18 00:16:08 -04:00
// @DisplayName: AutoTune maximum sweep frequency
// @Description: Defines the end frequency for sweeps and dwells
// @Range: 50 120
// @User: Standard
2021-12-28 00:23:19 -04:00
AP_GROUPINFO ( " FRQ_MAX " , 4 , AC_AutoTune_Heli , max_sweep_freq , 70.0f ) ,
2021-01-18 00:16:08 -04:00
2021-12-28 00:23:19 -04:00
// @Param: GN_MAX
2021-01-18 00:16:08 -04:00
// @DisplayName: AutoTune maximum response gain
// @Description: Defines the response gain (output/input) to tune
// @Range: 1 2.5
// @User: Standard
2022-02-06 23:45:24 -04:00
AP_GROUPINFO ( " GN_MAX " , 5 , AC_AutoTune_Heli , max_resp_gain , 1.0f ) ,
2021-01-18 00:16:08 -04:00
2021-09-12 17:12:15 -03:00
// @Param: VELXY_P
// @DisplayName: AutoTune velocity xy P gain
// @Description: Velocity xy P gain used to hold position during Max Gain, Rate P, and Rate D frequency sweeps
// @Range: 0 1
// @User: Standard
2021-10-01 17:35:52 -03:00
AP_GROUPINFO ( " VELXY_P " , 6 , AC_AutoTune_Heli , vel_hold_gain , 0.1f ) ,
2021-09-12 17:12:15 -03:00
2024-04-01 00:41:59 -03:00
// @Param: ACC_MAX
// @DisplayName: AutoTune maximum allowable angular acceleration
// @Description: maximum angular acceleration in deg/s/s allowed during autotune maneuvers
// @Range: 1 4000
// @User: Standard
2024-06-07 00:23:58 -03:00
AP_GROUPINFO ( " ACC_MAX " , 7 , AC_AutoTune_Heli , accel_max , 0.0f ) ,
2024-04-01 00:41:59 -03:00
// @Param: RAT_MAX
// @DisplayName: Autotune maximum allowable angular rate
// @Description: maximum angular rate in deg/s allowed during autotune maneuvers
// @Range: 0 500
// @User: Standard
2024-06-07 00:23:58 -03:00
AP_GROUPINFO ( " RAT_MAX " , 8 , AC_AutoTune_Heli , rate_max , 0.0f ) ,
2024-04-01 00:41:59 -03:00
2021-01-18 00:16:08 -04:00
AP_GROUPEND
} ;
// constructor
AC_AutoTune_Heli : : AC_AutoTune_Heli ( )
2020-10-19 23:51:05 -03:00
{
2021-01-18 00:16:08 -04:00
tune_seq [ 0 ] = TUNE_COMPLETE ;
AP_Param : : setup_object_defaults ( this , var_info ) ;
}
2020-10-19 23:51:05 -03:00
2021-12-09 23:42:02 -04:00
// initialize tests for each tune type
2021-01-18 00:16:08 -04:00
void AC_AutoTune_Heli : : test_init ( )
{
2024-04-08 00:34:47 -03:00
AC_AutoTune_FreqResp : : ResponseType resp_type = AC_AutoTune_FreqResp : : ResponseType : : RATE ;
2024-04-11 00:30:36 -03:00
FreqRespCalcType calc_type = RATE ;
FreqRespInput freq_resp_input = TARGET ;
2024-05-19 23:36:38 -03:00
float freq_resp_amplitude = 5.0f ; // amplitude in deg
float filter_freq = 10.0f ;
2022-01-16 19:13:59 -04:00
switch ( tune_type ) {
case RFF_UP :
2024-04-12 00:21:16 -03:00
if ( ! is_positive ( next_test_freq ) ) {
2024-04-16 00:18:32 -03:00
start_freq = 0.25f * M_2PI ;
2024-04-12 00:21:16 -03:00
} else {
start_freq = next_test_freq ;
}
stop_freq = start_freq ;
2024-05-19 23:36:38 -03:00
filter_freq = start_freq ;
2024-04-04 22:42:18 -03:00
attitude_control - > bf_feedforward ( false ) ;
2024-06-07 00:23:58 -03:00
2024-04-11 00:30:36 -03:00
// variables needed to initialize frequency response object and test method
2024-04-08 00:34:47 -03:00
resp_type = AC_AutoTune_FreqResp : : ResponseType : : RATE ;
2024-04-11 00:30:36 -03:00
calc_type = RATE ;
freq_resp_input = TARGET ;
2024-04-08 00:34:47 -03:00
pre_calc_cycles = 1.0f ;
num_dwell_cycles = 3 ;
2022-01-16 19:13:59 -04:00
break ;
case MAX_GAINS :
2024-04-13 02:04:33 -03:00
// initialize start frequency for sweep
if ( ! is_positive ( next_test_freq ) ) {
start_freq = min_sweep_freq ;
stop_freq = max_sweep_freq ;
sweep_complete = true ;
} else {
start_freq = next_test_freq ;
stop_freq = start_freq ;
test_accel_max = 0.0f ;
2020-10-19 23:51:05 -03:00
}
2024-05-19 23:36:38 -03:00
filter_freq = start_freq ;
2024-04-13 02:04:33 -03:00
2024-04-01 00:41:59 -03:00
attitude_control - > bf_feedforward ( false ) ;
2024-04-11 00:30:36 -03:00
// variables needed to initialize frequency response object and test method
2024-04-08 00:34:47 -03:00
resp_type = AC_AutoTune_FreqResp : : ResponseType : : RATE ;
2024-04-11 00:30:36 -03:00
calc_type = RATE ;
freq_resp_input = MOTOR ;
2024-04-08 00:34:47 -03:00
pre_calc_cycles = 6.25f ;
num_dwell_cycles = 6 ;
break ;
case RP_UP :
case RD_UP :
// initialize start frequency
2024-04-12 02:09:33 -03:00
if ( ! is_positive ( next_test_freq ) ) {
2024-04-09 00:28:52 -03:00
// continue using frequency where testing left off with RD_UP completed
2024-04-12 02:09:33 -03:00
if ( curr_data . phase > 150.0f & & curr_data . phase < 180.0f & & tune_type = = RP_UP ) {
start_freq = curr_data . freq ;
2024-04-08 00:34:47 -03:00
// start with freq found for sweep where phase was 180 deg
} else if ( ! is_zero ( sweep_tgt . ph180 . freq ) ) {
2024-04-16 00:18:32 -03:00
start_freq = sweep_tgt . ph180 . freq ;
2024-04-08 00:34:47 -03:00
// otherwise start at min freq to step up in dwell frequency until phase > 160 deg
} else {
2024-04-12 02:09:33 -03:00
start_freq = min_sweep_freq ;
2024-04-08 00:34:47 -03:00
}
2024-04-12 02:09:33 -03:00
} else {
start_freq = next_test_freq ;
2020-10-19 23:51:05 -03:00
}
2024-04-12 02:09:33 -03:00
stop_freq = start_freq ;
2024-05-19 23:36:38 -03:00
filter_freq = start_freq ;
2024-04-12 02:09:33 -03:00
2024-04-08 00:34:47 -03:00
attitude_control - > bf_feedforward ( false ) ;
2024-04-11 00:30:36 -03:00
// variables needed to initialize frequency response object and test method
2024-04-08 00:34:47 -03:00
resp_type = AC_AutoTune_FreqResp : : ResponseType : : RATE ;
2024-04-11 00:30:36 -03:00
calc_type = RATE ;
freq_resp_input = TARGET ;
2024-04-08 00:34:47 -03:00
pre_calc_cycles = 6.25f ;
num_dwell_cycles = 6 ;
2022-01-16 19:13:59 -04:00
break ;
case SP_UP :
2024-04-12 23:26:56 -03:00
// initialize start frequency for sweep
if ( ! is_positive ( next_test_freq ) ) {
start_freq = min_sweep_freq ;
stop_freq = max_sweep_freq ;
sweep_complete = true ;
} else {
start_freq = next_test_freq ;
stop_freq = start_freq ;
test_accel_max = 0.0f ;
2021-01-18 00:16:08 -04:00
}
2024-05-19 23:36:38 -03:00
filter_freq = start_freq ;
2022-05-03 00:35:50 -03:00
attitude_control - > bf_feedforward ( false ) ;
2021-01-18 00:16:08 -04:00
2024-04-11 00:30:36 -03:00
// variables needed to initialize frequency response object and test method
2024-04-08 00:34:47 -03:00
resp_type = AC_AutoTune_FreqResp : : ResponseType : : ANGLE ;
2024-04-11 00:30:36 -03:00
calc_type = DRB ;
freq_resp_input = TARGET ;
2024-04-08 00:34:47 -03:00
pre_calc_cycles = 6.25f ;
num_dwell_cycles = 6 ;
2022-05-08 00:28:50 -03:00
break ;
case TUNE_CHECK :
// initialize start frequency
2024-04-12 23:26:56 -03:00
start_freq = min_sweep_freq ;
stop_freq = max_sweep_freq ;
test_accel_max = 0.0f ;
2024-05-19 23:36:38 -03:00
filter_freq = start_freq ;
2024-04-12 23:26:56 -03:00
2024-04-11 00:30:36 -03:00
// variables needed to initialize frequency response object and test method
2024-04-08 00:34:47 -03:00
resp_type = AC_AutoTune_FreqResp : : ResponseType : : ANGLE ;
2024-04-11 00:30:36 -03:00
calc_type = ANGLE ;
freq_resp_input = TARGET ;
2022-01-16 19:13:59 -04:00
break ;
default :
break ;
2020-10-19 23:51:05 -03:00
}
2021-10-06 00:50:02 -03:00
2024-04-08 00:34:47 -03:00
if ( ! is_equal ( start_freq , stop_freq ) ) {
input_type = AC_AutoTune_FreqResp : : InputType : : SWEEP ;
} else {
input_type = AC_AutoTune_FreqResp : : InputType : : DWELL ;
}
2024-04-11 00:30:36 -03:00
2024-04-08 00:34:47 -03:00
// initialize dwell test method
2024-05-19 23:36:38 -03:00
dwell_test_init ( start_freq , stop_freq , freq_resp_amplitude , filter_freq , freq_resp_input , calc_type , resp_type , input_type ) ;
2024-04-08 00:34:47 -03:00
2020-10-19 23:51:05 -03:00
start_angles = Vector3f ( roll_cd , pitch_cd , desired_yaw_cd ) ; // heli specific
}
2021-12-09 23:42:02 -04:00
// run tests for each tune type
2020-10-19 23:51:05 -03:00
void AC_AutoTune_Heli : : test_run ( AxisType test_axis , const float dir_sign )
{
2022-01-21 14:44:36 -04:00
// if tune complete or beyond frequency range or no max allowed gains then exit testing
if ( tune_type = = TUNE_COMPLETE | |
2022-04-14 00:41:58 -03:00
( ( tune_type = = RP_UP | | tune_type = = RD_UP ) & & ( max_rate_p . max_allowed < = 0.0f | | max_rate_d . max_allowed < = 0.0f ) ) | |
2022-01-21 14:44:36 -04:00
( ( tune_type = = MAX_GAINS | | tune_type = = RP_UP | | tune_type = = RD_UP | | tune_type = = SP_UP ) & & exceeded_freq_range ( start_freq ) ) ) {
2022-04-10 00:17:28 -03:00
load_gains ( GAIN_ORIGINAL ) ;
2022-01-21 14:44:36 -04:00
attitude_control - > use_sqrt_controller ( true ) ;
get_poshold_attitude ( roll_cd , pitch_cd , desired_yaw_cd ) ;
// hold level attitude
attitude_control - > input_euler_angle_roll_pitch_yaw ( roll_cd , pitch_cd , desired_yaw_cd , true ) ;
2022-04-14 00:41:58 -03:00
if ( ( tune_type = = RP_UP | | tune_type = = RD_UP ) & & ( max_rate_p . max_allowed < = 0.0f | | max_rate_d . max_allowed < = 0.0f ) ) {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: Max Gain Determination Failed " ) ;
2022-04-10 00:17:28 -03:00
mode = FAILED ;
2023-07-13 21:58:04 -03:00
LOGGER_WRITE_EVENT ( LogEvent : : AUTOTUNE_FAILED ) ;
2022-04-10 00:17:28 -03:00
update_gcs ( AUTOTUNE_MESSAGE_FAILED ) ;
2022-01-21 14:44:36 -04:00
} else if ( ( tune_type = = MAX_GAINS | | tune_type = = RP_UP | | tune_type = = RD_UP | | tune_type = = SP_UP ) & & exceeded_freq_range ( start_freq ) ) {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: Exceeded frequency range " ) ;
2022-04-10 00:17:28 -03:00
mode = FAILED ;
2023-07-13 21:58:04 -03:00
LOGGER_WRITE_EVENT ( LogEvent : : AUTOTUNE_FAILED ) ;
2022-04-10 00:17:28 -03:00
update_gcs ( AUTOTUNE_MESSAGE_FAILED ) ;
2022-01-21 14:44:36 -04:00
} else if ( tune_type = = TUNE_COMPLETE ) {
counter = AUTOTUNE_SUCCESS_COUNT ;
step = UPDATE_GAINS ;
}
return ;
}
2022-01-16 19:13:59 -04:00
2024-04-13 02:04:33 -03:00
dwell_test_run ( curr_data ) ;
2024-04-11 00:30:36 -03:00
2020-10-19 23:51:05 -03:00
}
2021-12-09 23:42:02 -04:00
// heli specific gcs announcements
2020-10-19 23:51:05 -03:00
void AC_AutoTune_Heli : : do_gcs_announcements ( )
{
const uint32_t now = AP_HAL : : millis ( ) ;
if ( now - announce_time < AUTOTUNE_ANNOUNCE_INTERVAL_MS ) {
return ;
}
2022-01-07 14:37:02 -04:00
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: %s %s " , axis_string ( ) , type_string ( ) ) ;
2022-01-07 14:37:02 -04:00
send_step_string ( ) ;
switch ( tune_type ) {
2024-04-04 22:42:18 -03:00
case RFF_UP :
2022-01-07 14:37:02 -04:00
case RD_UP :
case RP_UP :
case MAX_GAINS :
case SP_UP :
2022-05-08 00:28:50 -03:00
case TUNE_CHECK :
2022-01-07 14:37:02 -04:00
if ( is_equal ( start_freq , stop_freq ) ) {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: Dwell " ) ;
2022-01-07 14:37:02 -04:00
} else {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: Sweep " ) ;
2022-01-07 14:37:02 -04:00
if ( settle_time = = 0 ) {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: freq=%f gain=%f phase=%f " , ( double ) ( curr_test . freq ) , ( double ) ( curr_test . gain ) , ( double ) ( curr_test . phase ) ) ;
2022-01-07 14:37:02 -04:00
}
}
break ;
default :
break ;
}
announce_time = now ;
}
// send post test updates to user
void AC_AutoTune_Heli : : do_post_test_gcs_announcements ( ) {
2020-10-19 23:51:05 -03:00
float tune_rp = 0.0f ;
float tune_rd = 0.0f ;
float tune_rff = 0.0f ;
float tune_sp = 0.0f ;
float tune_accel = 0.0f ;
2022-01-07 14:37:02 -04:00
2020-10-19 23:51:05 -03:00
switch ( axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2020-10-19 23:51:05 -03:00
tune_rp = tune_roll_rp ;
tune_rd = tune_roll_rd ;
tune_rff = tune_roll_rff ;
tune_sp = tune_roll_sp ;
tune_accel = tune_roll_accel ;
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2020-10-19 23:51:05 -03:00
tune_rp = tune_pitch_rp ;
tune_rd = tune_pitch_rd ;
tune_rff = tune_pitch_rff ;
tune_sp = tune_pitch_sp ;
tune_accel = tune_pitch_accel ;
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2020-10-19 23:51:05 -03:00
tune_rp = tune_yaw_rp ;
tune_rd = tune_yaw_rd ;
tune_rff = tune_yaw_rff ;
tune_sp = tune_yaw_sp ;
tune_accel = tune_yaw_accel ;
break ;
}
2022-01-07 14:37:02 -04:00
if ( step = = UPDATE_GAINS ) {
switch ( tune_type ) {
case RFF_UP :
case RP_UP :
case RD_UP :
case SP_UP :
case MAX_GAINS :
if ( is_equal ( start_freq , stop_freq ) ) {
2022-02-10 23:27:08 -04:00
// announce results of dwell
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: freq=%f gain=%f " , ( double ) ( curr_data . freq ) , ( double ) ( curr_data . gain ) ) ;
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: ph=%f " , ( double ) ( curr_data . phase ) ) ;
2022-02-10 23:27:08 -04:00
if ( tune_type = = RP_UP ) {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: rate_p=%f " , ( double ) ( tune_rp ) ) ;
2022-02-10 23:27:08 -04:00
} else if ( tune_type = = RD_UP ) {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: rate_d=%f " , ( double ) ( tune_rd ) ) ;
2024-04-04 22:42:18 -03:00
} else if ( tune_type = = RFF_UP ) {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: rate_ff=%f " , ( double ) ( tune_rff ) ) ;
2022-02-10 23:27:08 -04:00
} else if ( tune_type = = SP_UP ) {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: angle_p=%f tune_accel=%f max_accel=%f " , ( double ) ( tune_sp ) , ( double ) ( tune_accel ) , ( double ) ( test_accel_max ) ) ;
2022-02-10 23:27:08 -04:00
}
2022-01-07 14:37:02 -04:00
} else {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: max_freq=%f max_gain=%f " , ( double ) ( sweep_tgt . maxgain . freq ) , ( double ) ( sweep_tgt . maxgain . gain ) ) ;
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: ph180_freq=%f ph180_gain=%f " , ( double ) ( sweep_tgt . ph180 . freq ) , ( double ) ( sweep_tgt . ph180 . gain ) ) ;
2022-01-07 14:37:02 -04:00
}
break ;
default :
break ;
2020-10-19 23:51:05 -03:00
}
}
}
2021-12-22 10:45:38 -04:00
// backup_gains_and_initialise - store current gains as originals
// called before tuning starts to backup original gains
void AC_AutoTune_Heli : : backup_gains_and_initialise ( )
{
AC_AutoTune : : backup_gains_and_initialise ( ) ;
2021-12-22 23:37:31 -04:00
// initializes dwell test sequence for rate_p_up and rate_d_up tests for tradheli
2024-04-13 02:04:33 -03:00
next_test_freq = 0.0f ;
2021-12-22 23:37:31 -04:00
start_freq = 0.0f ;
stop_freq = 0.0f ;
2021-12-22 10:45:38 -04:00
orig_bf_feedforward = attitude_control - > get_bf_feedforward ( ) ;
// backup original pids and initialise tuned pid values
orig_roll_rp = attitude_control - > get_rate_roll_pid ( ) . kP ( ) ;
orig_roll_ri = attitude_control - > get_rate_roll_pid ( ) . kI ( ) ;
orig_roll_rd = attitude_control - > get_rate_roll_pid ( ) . kD ( ) ;
orig_roll_rff = attitude_control - > get_rate_roll_pid ( ) . ff ( ) ;
orig_roll_fltt = attitude_control - > get_rate_roll_pid ( ) . filt_T_hz ( ) ;
orig_roll_smax = attitude_control - > get_rate_roll_pid ( ) . slew_limit ( ) ;
orig_roll_sp = attitude_control - > get_angle_roll_p ( ) . kP ( ) ;
orig_roll_accel = attitude_control - > get_accel_roll_max_cdss ( ) ;
2024-05-19 23:36:38 -03:00
orig_roll_rate = attitude_control - > get_ang_vel_roll_max_degs ( ) ;
2021-12-22 10:45:38 -04:00
tune_roll_rp = attitude_control - > get_rate_roll_pid ( ) . kP ( ) ;
tune_roll_rd = attitude_control - > get_rate_roll_pid ( ) . kD ( ) ;
tune_roll_rff = attitude_control - > get_rate_roll_pid ( ) . ff ( ) ;
tune_roll_sp = attitude_control - > get_angle_roll_p ( ) . kP ( ) ;
tune_roll_accel = attitude_control - > get_accel_roll_max_cdss ( ) ;
orig_pitch_rp = attitude_control - > get_rate_pitch_pid ( ) . kP ( ) ;
orig_pitch_ri = attitude_control - > get_rate_pitch_pid ( ) . kI ( ) ;
orig_pitch_rd = attitude_control - > get_rate_pitch_pid ( ) . kD ( ) ;
orig_pitch_rff = attitude_control - > get_rate_pitch_pid ( ) . ff ( ) ;
orig_pitch_fltt = attitude_control - > get_rate_pitch_pid ( ) . filt_T_hz ( ) ;
orig_pitch_smax = attitude_control - > get_rate_pitch_pid ( ) . slew_limit ( ) ;
orig_pitch_sp = attitude_control - > get_angle_pitch_p ( ) . kP ( ) ;
orig_pitch_accel = attitude_control - > get_accel_pitch_max_cdss ( ) ;
2024-05-19 23:36:38 -03:00
orig_pitch_rate = attitude_control - > get_ang_vel_pitch_max_degs ( ) ;
2021-12-22 10:45:38 -04:00
tune_pitch_rp = attitude_control - > get_rate_pitch_pid ( ) . kP ( ) ;
tune_pitch_rd = attitude_control - > get_rate_pitch_pid ( ) . kD ( ) ;
tune_pitch_rff = attitude_control - > get_rate_pitch_pid ( ) . ff ( ) ;
tune_pitch_sp = attitude_control - > get_angle_pitch_p ( ) . kP ( ) ;
tune_pitch_accel = attitude_control - > get_accel_pitch_max_cdss ( ) ;
orig_yaw_rp = attitude_control - > get_rate_yaw_pid ( ) . kP ( ) ;
orig_yaw_ri = attitude_control - > get_rate_yaw_pid ( ) . kI ( ) ;
orig_yaw_rd = attitude_control - > get_rate_yaw_pid ( ) . kD ( ) ;
orig_yaw_rff = attitude_control - > get_rate_yaw_pid ( ) . ff ( ) ;
orig_yaw_fltt = attitude_control - > get_rate_yaw_pid ( ) . filt_T_hz ( ) ;
orig_yaw_smax = attitude_control - > get_rate_yaw_pid ( ) . slew_limit ( ) ;
orig_yaw_rLPF = attitude_control - > get_rate_yaw_pid ( ) . filt_E_hz ( ) ;
orig_yaw_accel = attitude_control - > get_accel_yaw_max_cdss ( ) ;
orig_yaw_sp = attitude_control - > get_angle_yaw_p ( ) . kP ( ) ;
2024-05-19 23:36:38 -03:00
orig_yaw_rate = attitude_control - > get_ang_vel_yaw_max_degs ( ) ;
2021-12-22 10:45:38 -04:00
tune_yaw_rp = attitude_control - > get_rate_yaw_pid ( ) . kP ( ) ;
tune_yaw_rd = attitude_control - > get_rate_yaw_pid ( ) . kD ( ) ;
tune_yaw_rff = attitude_control - > get_rate_yaw_pid ( ) . ff ( ) ;
tune_yaw_rLPF = attitude_control - > get_rate_yaw_pid ( ) . filt_E_hz ( ) ;
tune_yaw_sp = attitude_control - > get_angle_yaw_p ( ) . kP ( ) ;
tune_yaw_accel = attitude_control - > get_accel_yaw_max_cdss ( ) ;
2023-07-13 21:58:04 -03:00
LOGGER_WRITE_EVENT ( LogEvent : : AUTOTUNE_INITIALISED ) ;
2021-12-22 10:45:38 -04:00
}
// load_orig_gains - set gains to their original values
// called by stop and failed functions
void AC_AutoTune_Heli : : load_orig_gains ( )
{
attitude_control - > bf_feedforward ( orig_bf_feedforward ) ;
if ( roll_enabled ( ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : ROLL , orig_roll_rp , orig_roll_ri , orig_roll_rd , orig_roll_rff , orig_roll_sp , orig_roll_accel , orig_roll_fltt , 0.0f , orig_roll_smax , orig_roll_rate ) ;
2021-12-22 10:45:38 -04:00
}
if ( pitch_enabled ( ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : PITCH , orig_pitch_rp , orig_pitch_ri , orig_pitch_rd , orig_pitch_rff , orig_pitch_sp , orig_pitch_accel , orig_pitch_fltt , 0.0f , orig_pitch_smax , orig_pitch_rate ) ;
2021-12-22 10:45:38 -04:00
}
if ( yaw_enabled ( ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : YAW , orig_yaw_rp , orig_yaw_ri , orig_yaw_rd , orig_yaw_rff , orig_yaw_sp , orig_yaw_accel , orig_yaw_fltt , orig_yaw_rLPF , orig_yaw_smax , orig_yaw_rate ) ;
2021-12-22 10:45:38 -04:00
}
}
// load_tuned_gains - load tuned gains
void AC_AutoTune_Heli : : load_tuned_gains ( )
{
if ( ! attitude_control - > get_bf_feedforward ( ) ) {
attitude_control - > bf_feedforward ( true ) ;
attitude_control - > set_accel_roll_max_cdss ( 0.0f ) ;
attitude_control - > set_accel_pitch_max_cdss ( 0.0f ) ;
}
if ( roll_enabled ( ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : ROLL , tune_roll_rp , tune_roll_rff * AUTOTUNE_FFI_RATIO_FINAL , tune_roll_rd , tune_roll_rff , tune_roll_sp , tune_roll_accel , orig_roll_fltt , 0.0f , orig_roll_smax , orig_roll_rate ) ;
2021-12-22 10:45:38 -04:00
}
if ( pitch_enabled ( ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : PITCH , tune_pitch_rp , tune_pitch_rff * AUTOTUNE_FFI_RATIO_FINAL , tune_pitch_rd , tune_pitch_rff , tune_pitch_sp , tune_pitch_accel , orig_pitch_fltt , 0.0f , orig_pitch_smax , orig_pitch_rate ) ;
2021-12-22 10:45:38 -04:00
}
if ( yaw_enabled ( ) ) {
if ( ! is_zero ( tune_yaw_rp ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : YAW , tune_yaw_rp , tune_yaw_rp * AUTOTUNE_YAW_PI_RATIO_FINAL , tune_yaw_rd , tune_yaw_rff , tune_yaw_sp , tune_yaw_accel , orig_yaw_fltt , tune_yaw_rLPF , orig_yaw_smax , orig_yaw_rate ) ;
2021-12-22 10:45:38 -04:00
}
}
}
// load_intra_test_gains - gains used between tests
// called during testing mode's update-gains step to set gains ahead of return-to-level step
void AC_AutoTune_Heli : : load_intra_test_gains ( )
{
// we are restarting tuning so reset gains to tuning-start gains (i.e. low I term)
// sanity check the gains
attitude_control - > bf_feedforward ( true ) ;
if ( roll_enabled ( ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : ROLL , orig_roll_rp , orig_roll_rff * AUTOTUNE_FFI_RATIO_FOR_TESTING , orig_roll_rd , orig_roll_rff , orig_roll_sp , orig_roll_accel , orig_roll_fltt , 0.0f , orig_roll_smax , orig_roll_rate ) ;
2021-12-22 10:45:38 -04:00
}
if ( pitch_enabled ( ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : PITCH , orig_pitch_rp , orig_pitch_rff * AUTOTUNE_FFI_RATIO_FOR_TESTING , orig_pitch_rd , orig_pitch_rff , orig_pitch_sp , orig_pitch_accel , orig_pitch_fltt , 0.0f , orig_pitch_smax , orig_pitch_rate ) ;
2021-12-22 10:45:38 -04:00
}
if ( yaw_enabled ( ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : YAW , orig_yaw_rp , orig_yaw_rp * AUTOTUNE_PI_RATIO_FOR_TESTING , orig_yaw_rd , orig_yaw_rff , orig_yaw_sp , orig_yaw_accel , orig_yaw_fltt , orig_yaw_rLPF , orig_yaw_smax , orig_yaw_rate ) ;
2021-12-22 10:45:38 -04:00
}
}
2020-10-19 23:51:05 -03:00
// load_test_gains - load the to-be-tested gains for a single axis
// called by control_attitude() just before it beings testing a gain (i.e. just before it twitches)
void AC_AutoTune_Heli : : load_test_gains ( )
{
2024-06-07 00:23:58 -03:00
float rate_p , rate_i , rate_d , rate_test_max , accel_test_max ;
2020-10-19 23:51:05 -03:00
switch ( axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2024-06-07 00:23:58 -03:00
if ( tune_type = = TUNE_CHECK ) {
rate_test_max = orig_roll_rate ;
accel_test_max = tune_roll_accel ;
} else {
// have attitude controller use accel and rate limit parameter
rate_test_max = rate_max ;
accel_test_max = accel_max ;
}
2022-05-20 23:45:12 -03:00
if ( tune_type = = SP_UP | | tune_type = = TUNE_CHECK ) {
rate_i = tune_roll_rff * AUTOTUNE_FFI_RATIO_FINAL ;
2020-10-19 23:51:05 -03:00
} else {
2021-01-18 00:16:08 -04:00
// freeze integrator to hold trim by making i term small during rate controller tuning
2022-03-31 17:52:40 -03:00
rate_i = 0.01f * orig_roll_ri ;
2020-10-19 23:51:05 -03:00
}
2021-12-11 00:43:40 -04:00
if ( tune_type = = MAX_GAINS & & ! is_zero ( tune_roll_rff ) ) {
2022-03-31 17:52:40 -03:00
rate_p = 0.0f ;
rate_d = 0.0f ;
2021-12-11 00:43:40 -04:00
} else {
2022-03-31 17:52:40 -03:00
rate_p = tune_roll_rp ;
rate_d = tune_roll_rd ;
2021-12-11 00:43:40 -04:00
}
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : ROLL , rate_p , rate_i , rate_d , tune_roll_rff , tune_roll_sp , accel_test_max , orig_roll_fltt , 0.0f , 0.0f , rate_test_max ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2024-06-07 00:23:58 -03:00
if ( tune_type = = TUNE_CHECK ) {
rate_test_max = orig_pitch_rate ;
accel_test_max = tune_pitch_accel ;
} else {
// have attitude controller use accel and rate limit parameter
rate_test_max = rate_max ;
accel_test_max = accel_max ;
}
2022-05-20 23:45:12 -03:00
if ( tune_type = = SP_UP | | tune_type = = TUNE_CHECK ) {
rate_i = tune_pitch_rff * AUTOTUNE_FFI_RATIO_FINAL ;
2020-10-19 23:51:05 -03:00
} else {
2021-01-18 00:16:08 -04:00
// freeze integrator to hold trim by making i term small during rate controller tuning
2022-03-31 17:52:40 -03:00
rate_i = 0.01f * orig_pitch_ri ;
2020-10-19 23:51:05 -03:00
}
2021-12-11 00:43:40 -04:00
if ( tune_type = = MAX_GAINS & & ! is_zero ( tune_pitch_rff ) ) {
2022-03-31 17:52:40 -03:00
rate_p = 0.0f ;
rate_d = 0.0f ;
2021-12-11 00:43:40 -04:00
} else {
2022-03-31 17:52:40 -03:00
rate_p = tune_pitch_rp ;
rate_d = tune_pitch_rd ;
2021-12-11 00:43:40 -04:00
}
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : PITCH , rate_p , rate_i , rate_d , tune_pitch_rff , tune_pitch_sp , accel_test_max , orig_pitch_fltt , 0.0f , 0.0f , rate_test_max ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2024-06-07 00:23:58 -03:00
if ( tune_type = = TUNE_CHECK ) {
rate_test_max = orig_yaw_rate ;
accel_test_max = tune_yaw_accel ;
} else {
// have attitude controller use accel and rate limit parameter
rate_test_max = rate_max ;
accel_test_max = accel_max ;
}
2022-05-20 23:45:12 -03:00
if ( tune_type = = SP_UP | | tune_type = = TUNE_CHECK ) {
rate_i = tune_yaw_rp * AUTOTUNE_YAW_PI_RATIO_FINAL ;
} else {
// freeze integrator to hold trim by making i term small during rate controller tuning
rate_i = 0.01f * orig_yaw_ri ;
}
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : YAW , tune_yaw_rp , rate_i , tune_yaw_rd , tune_yaw_rff , tune_yaw_sp , accel_test_max , orig_yaw_fltt , tune_yaw_rLPF , 0.0f , rate_test_max ) ;
2022-03-31 17:52:40 -03:00
break ;
}
}
// load gains
2024-05-19 23:36:38 -03:00
void AC_AutoTune_Heli : : load_gain_set ( AxisType s_axis , float rate_p , float rate_i , float rate_d , float rate_ff , float angle_p , float max_accel , float rate_fltt , float rate_flte , float smax , float max_rate )
2022-03-31 17:52:40 -03:00
{
2022-06-18 00:21:39 -03:00
switch ( s_axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2024-07-30 08:37:50 -03:00
attitude_control - > get_rate_roll_pid ( ) . set_kP ( rate_p ) ;
attitude_control - > get_rate_roll_pid ( ) . set_kI ( rate_i ) ;
attitude_control - > get_rate_roll_pid ( ) . set_kD ( rate_d ) ;
attitude_control - > get_rate_roll_pid ( ) . set_ff ( rate_ff ) ;
attitude_control - > get_rate_roll_pid ( ) . set_filt_T_hz ( rate_fltt ) ;
attitude_control - > get_rate_roll_pid ( ) . set_slew_limit ( smax ) ;
attitude_control - > get_angle_roll_p ( ) . set_kP ( angle_p ) ;
2022-03-31 17:52:40 -03:00
attitude_control - > set_accel_roll_max_cdss ( max_accel ) ;
2024-05-19 23:36:38 -03:00
attitude_control - > set_ang_vel_roll_max_degs ( max_rate ) ;
2022-03-31 17:52:40 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2024-07-30 08:37:50 -03:00
attitude_control - > get_rate_pitch_pid ( ) . set_kP ( rate_p ) ;
attitude_control - > get_rate_pitch_pid ( ) . set_kI ( rate_i ) ;
attitude_control - > get_rate_pitch_pid ( ) . set_kD ( rate_d ) ;
attitude_control - > get_rate_pitch_pid ( ) . set_ff ( rate_ff ) ;
attitude_control - > get_rate_pitch_pid ( ) . set_filt_T_hz ( rate_fltt ) ;
attitude_control - > get_rate_pitch_pid ( ) . set_slew_limit ( smax ) ;
attitude_control - > get_angle_pitch_p ( ) . set_kP ( angle_p ) ;
2022-03-31 17:52:40 -03:00
attitude_control - > set_accel_pitch_max_cdss ( max_accel ) ;
2024-05-19 23:36:38 -03:00
attitude_control - > set_ang_vel_pitch_max_degs ( max_rate ) ;
2022-03-31 17:52:40 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2024-07-30 08:37:50 -03:00
attitude_control - > get_rate_yaw_pid ( ) . set_kP ( rate_p ) ;
attitude_control - > get_rate_yaw_pid ( ) . set_kI ( rate_i ) ;
attitude_control - > get_rate_yaw_pid ( ) . set_kD ( rate_d ) ;
attitude_control - > get_rate_yaw_pid ( ) . set_ff ( rate_ff ) ;
attitude_control - > get_rate_yaw_pid ( ) . set_filt_T_hz ( rate_fltt ) ;
attitude_control - > get_rate_yaw_pid ( ) . set_slew_limit ( smax ) ;
attitude_control - > get_rate_yaw_pid ( ) . set_filt_E_hz ( rate_flte ) ;
attitude_control - > get_angle_yaw_p ( ) . set_kP ( angle_p ) ;
2022-03-31 17:52:40 -03:00
attitude_control - > set_accel_yaw_max_cdss ( max_accel ) ;
2024-05-19 23:36:38 -03:00
attitude_control - > set_ang_vel_yaw_max_degs ( max_rate ) ;
2020-10-19 23:51:05 -03:00
break ;
}
}
// save_tuning_gains - save the final tuned gains for each axis
// save discovered gains to eeprom if autotuner is enabled (i.e. switch is in the high position)
void AC_AutoTune_Heli : : save_tuning_gains ( )
{
2021-12-11 00:43:40 -04:00
// see if we successfully completed tuning of at least one axis
if ( axes_completed = = 0 ) {
return ;
}
2020-10-19 23:51:05 -03:00
2021-12-11 00:43:40 -04:00
if ( ! attitude_control - > get_bf_feedforward ( ) ) {
attitude_control - > bf_feedforward_save ( true ) ;
attitude_control - > save_accel_roll_max_cdss ( 0.0f ) ;
attitude_control - > save_accel_pitch_max_cdss ( 0.0f ) ;
}
2020-10-19 23:51:05 -03:00
// sanity check the rate P values
2021-01-18 00:16:08 -04:00
if ( ( axes_completed & AUTOTUNE_AXIS_BITMASK_ROLL ) & & roll_enabled ( ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : ROLL , tune_roll_rp , tune_roll_rff * AUTOTUNE_FFI_RATIO_FINAL , tune_roll_rd , tune_roll_rff , tune_roll_sp , tune_roll_accel , orig_roll_fltt , 0.0f , orig_roll_smax , orig_roll_rate ) ;
2022-03-31 17:52:40 -03:00
// save rate roll gains
2020-10-19 23:51:05 -03:00
attitude_control - > get_rate_roll_pid ( ) . save_gains ( ) ;
2022-03-31 17:52:40 -03:00
// save stabilize roll
2021-12-11 00:43:40 -04:00
attitude_control - > get_angle_roll_p ( ) . save_gains ( ) ;
2020-10-19 23:51:05 -03:00
// resave pids to originals in case the autotune is run again
2021-12-11 00:43:40 -04:00
orig_roll_rp = attitude_control - > get_rate_roll_pid ( ) . kP ( ) ;
2020-10-19 23:51:05 -03:00
orig_roll_ri = attitude_control - > get_rate_roll_pid ( ) . kI ( ) ;
2021-12-11 00:43:40 -04:00
orig_roll_rd = attitude_control - > get_rate_roll_pid ( ) . kD ( ) ;
orig_roll_rff = attitude_control - > get_rate_roll_pid ( ) . ff ( ) ;
orig_roll_sp = attitude_control - > get_angle_roll_p ( ) . kP ( ) ;
orig_roll_accel = attitude_control - > get_accel_roll_max_cdss ( ) ;
2020-10-19 23:51:05 -03:00
}
2021-01-18 00:16:08 -04:00
if ( ( axes_completed & AUTOTUNE_AXIS_BITMASK_PITCH ) & & pitch_enabled ( ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : PITCH , tune_pitch_rp , tune_pitch_rff * AUTOTUNE_FFI_RATIO_FINAL , tune_pitch_rd , tune_pitch_rff , tune_pitch_sp , tune_pitch_accel , orig_pitch_fltt , 0.0f , orig_pitch_smax , orig_pitch_rate ) ;
2022-03-31 17:52:40 -03:00
// save rate pitch gains
2020-10-19 23:51:05 -03:00
attitude_control - > get_rate_pitch_pid ( ) . save_gains ( ) ;
2022-03-31 17:52:40 -03:00
// save stabilize pitch
2021-12-11 00:43:40 -04:00
attitude_control - > get_angle_pitch_p ( ) . save_gains ( ) ;
2020-10-19 23:51:05 -03:00
// resave pids to originals in case the autotune is run again
2021-12-11 00:43:40 -04:00
orig_pitch_rp = attitude_control - > get_rate_pitch_pid ( ) . kP ( ) ;
2020-10-19 23:51:05 -03:00
orig_pitch_ri = attitude_control - > get_rate_pitch_pid ( ) . kI ( ) ;
2021-12-11 00:43:40 -04:00
orig_pitch_rd = attitude_control - > get_rate_pitch_pid ( ) . kD ( ) ;
orig_pitch_rff = attitude_control - > get_rate_pitch_pid ( ) . ff ( ) ;
orig_pitch_sp = attitude_control - > get_angle_pitch_p ( ) . kP ( ) ;
orig_pitch_accel = attitude_control - > get_accel_pitch_max_cdss ( ) ;
2020-10-19 23:51:05 -03:00
}
if ( ( axes_completed & AUTOTUNE_AXIS_BITMASK_YAW ) & & yaw_enabled ( ) & & ! is_zero ( tune_yaw_rp ) ) {
2024-07-25 01:07:26 -03:00
load_gain_set ( AxisType : : YAW , tune_yaw_rp , tune_yaw_rp * AUTOTUNE_YAW_PI_RATIO_FINAL , tune_yaw_rd , tune_yaw_rff , tune_yaw_sp , tune_yaw_accel , orig_yaw_fltt , orig_yaw_rLPF , orig_yaw_smax , orig_yaw_rate ) ;
2022-03-31 17:52:40 -03:00
// save rate yaw gains
2020-10-19 23:51:05 -03:00
attitude_control - > get_rate_yaw_pid ( ) . save_gains ( ) ;
2022-03-31 17:52:40 -03:00
// save stabilize yaw
2021-12-11 00:43:40 -04:00
attitude_control - > get_angle_yaw_p ( ) . save_gains ( ) ;
2020-10-19 23:51:05 -03:00
// resave pids to originals in case the autotune is run again
2021-12-11 00:43:40 -04:00
orig_yaw_rp = attitude_control - > get_rate_yaw_pid ( ) . kP ( ) ;
orig_yaw_ri = attitude_control - > get_rate_yaw_pid ( ) . kI ( ) ;
2020-10-19 23:51:05 -03:00
orig_yaw_rd = attitude_control - > get_rate_yaw_pid ( ) . kD ( ) ;
orig_yaw_rff = attitude_control - > get_rate_yaw_pid ( ) . ff ( ) ;
2021-12-11 00:43:40 -04:00
orig_yaw_rLPF = attitude_control - > get_rate_yaw_pid ( ) . filt_E_hz ( ) ;
orig_yaw_sp = attitude_control - > get_angle_yaw_p ( ) . kP ( ) ;
orig_yaw_accel = attitude_control - > get_accel_yaw_max_cdss ( ) ;
2020-10-19 23:51:05 -03:00
}
// update GCS and log save gains event
update_gcs ( AUTOTUNE_MESSAGE_SAVED_GAINS ) ;
2023-07-13 21:58:04 -03:00
LOGGER_WRITE_EVENT ( LogEvent : : AUTOTUNE_SAVEDGAINS ) ;
2020-10-19 23:51:05 -03:00
reset ( ) ;
}
2022-02-04 19:30:10 -04:00
// report final gains for a given axis to GCS
void AC_AutoTune_Heli : : report_final_gains ( AxisType test_axis ) const
{
switch ( test_axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2022-02-04 19:30:10 -04:00
report_axis_gains ( " Roll " , tune_roll_rp , tune_roll_rff * AUTOTUNE_FFI_RATIO_FINAL , tune_roll_rd , tune_roll_rff , tune_roll_sp , tune_roll_accel ) ;
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2022-02-04 19:30:10 -04:00
report_axis_gains ( " Pitch " , tune_pitch_rp , tune_pitch_rff * AUTOTUNE_FFI_RATIO_FINAL , tune_pitch_rd , tune_pitch_rff , tune_pitch_sp , tune_pitch_accel ) ;
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2022-02-04 19:30:10 -04:00
report_axis_gains ( " Yaw " , tune_yaw_rp , tune_yaw_rp * AUTOTUNE_YAW_PI_RATIO_FINAL , tune_yaw_rd , tune_yaw_rff , tune_yaw_sp , tune_yaw_accel ) ;
break ;
}
}
2023-10-11 04:41:49 -03:00
// report gain formatting helper
2022-02-04 19:30:10 -04:00
void AC_AutoTune_Heli : : report_axis_gains ( const char * axis_string , float rate_P , float rate_I , float rate_D , float rate_ff , float angle_P , float max_accel ) const
{
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_NOTICE , " AutoTune: %s complete " , axis_string ) ;
GCS_SEND_TEXT ( MAV_SEVERITY_NOTICE , " AutoTune: %s Rate: P:%0.4f, I:%0.4f, D:%0.5f, FF:%0.4f " , axis_string , rate_P , rate_I , rate_D , rate_ff ) ;
GCS_SEND_TEXT ( MAV_SEVERITY_NOTICE , " AutoTune: %s Angle P:%0.2f, Max Accel:%0.0f " , axis_string , angle_P , max_accel ) ;
2022-02-04 19:30:10 -04:00
}
2024-05-19 23:36:38 -03:00
void AC_AutoTune_Heli : : dwell_test_init ( float start_frq , float stop_frq , float amplitude , float filt_freq , FreqRespInput freq_resp_input , FreqRespCalcType calc_type , AC_AutoTune_FreqResp : : ResponseType resp_type , AC_AutoTune_FreqResp : : InputType waveform_input_type )
2021-12-13 00:02:40 -04:00
{
2024-04-11 00:30:36 -03:00
test_input_type = waveform_input_type ;
test_freq_resp_input = freq_resp_input ;
test_calc_type = calc_type ;
test_start_freq = start_frq ;
2024-05-19 23:36:38 -03:00
//target attitude magnitude
2024-06-16 12:41:52 -03:00
tgt_attitude = radians ( amplitude ) ;
2024-04-11 00:30:36 -03:00
// initialize frequency response object
if ( test_input_type = = AC_AutoTune_FreqResp : : InputType : : SWEEP ) {
step_time_limit_ms = sweep_time_ms + 500 ;
reset_sweep_variables ( ) ;
curr_test . gain = 0.0f ;
curr_test . phase = 0.0f ;
chirp_input . init ( 0.001f * sweep_time_ms , start_frq / M_2PI , stop_frq / M_2PI , 0.0f , 0.0001f * sweep_time_ms , 0.0f ) ;
} else {
if ( ! is_zero ( start_frq ) ) {
// time limit set by adding the pre calc cycles with the dwell cycles. 500 ms added to account for settling with buffer.
2024-06-22 14:52:20 -03:00
step_time_limit_ms = ( uint32_t ) ( 2000 + ( ( float ) num_dwell_cycles + pre_calc_cycles + 2.0f ) * 1000.0f * M_2PI / start_frq ) ;
2024-04-11 00:30:36 -03:00
}
chirp_input . init ( 0.001f * step_time_limit_ms , start_frq / M_2PI , stop_frq / M_2PI , 0.0f , 0.0001f * step_time_limit_ms , 0.0f ) ;
}
2024-06-22 14:52:20 -03:00
freqresp_tgt . init ( test_input_type , resp_type , num_dwell_cycles ) ;
freqresp_mtr . init ( test_input_type , resp_type , num_dwell_cycles ) ;
2024-04-11 00:30:36 -03:00
2022-03-05 23:16:50 -04:00
dwell_start_time_ms = 0.0f ;
settle_time = 200 ;
2021-12-13 00:02:40 -04:00
rotation_rate_filt . set_cutoff_frequency ( filt_freq ) ;
command_filt . set_cutoff_frequency ( filt_freq ) ;
target_rate_filt . set_cutoff_frequency ( filt_freq ) ;
2022-03-05 23:16:50 -04:00
2022-03-20 00:49:46 -03:00
rotation_rate_filt . reset ( 0 ) ;
command_filt . reset ( 0 ) ;
target_rate_filt . reset ( 0 ) ;
rotation_rate = 0.0f ;
command_out = 0.0f ;
filt_target_rate = 0.0f ;
2021-12-13 00:02:40 -04:00
2022-01-01 23:33:52 -04:00
// filter at lower frequency to remove steady state
2024-05-19 23:36:38 -03:00
filt_command_reading . set_cutoff_frequency ( 0.2f * filt_freq ) ;
filt_gyro_reading . set_cutoff_frequency ( 0.05f * filt_freq ) ;
filt_tgt_rate_reading . set_cutoff_frequency ( 0.05f * filt_freq ) ;
filt_att_fdbk_from_velxy_cd . set_cutoff_frequency ( 0.2f * filt_freq ) ;
2022-01-01 23:33:52 -04:00
2024-04-20 11:36:24 -03:00
curr_test_mtr = { } ;
curr_test_tgt = { } ;
2024-04-08 00:34:47 -03:00
cycle_complete_tgt = false ;
cycle_complete_mtr = false ;
2024-04-20 11:36:24 -03:00
sweep_complete = false ;
2022-03-07 01:23:41 -04:00
2021-12-13 00:02:40 -04:00
}
2024-04-13 02:04:33 -03:00
void AC_AutoTune_Heli : : dwell_test_run ( sweep_info & test_data )
2021-12-13 00:02:40 -04:00
{
float gyro_reading = 0.0f ;
float command_reading = 0.0f ;
float tgt_rate_reading = 0.0f ;
const uint32_t now = AP_HAL : : millis ( ) ;
2023-03-06 05:39:47 -04:00
float target_angle_cd = 0.0f ;
2024-04-11 00:30:36 -03:00
float dwell_freq = test_start_freq ;
2021-12-13 00:02:40 -04:00
float cycle_time_ms = 0 ;
if ( ! is_zero ( dwell_freq ) ) {
2022-03-06 20:32:12 -04:00
cycle_time_ms = 1000.0f * M_2PI / dwell_freq ;
}
// body frame calculation of velocity
2021-12-13 00:02:40 -04:00
Vector3f velocity_ned , velocity_bf ;
if ( ahrs_view - > get_velocity_NED ( velocity_ned ) ) {
velocity_bf . x = velocity_ned . x * ahrs_view - > cos_yaw ( ) + velocity_ned . y * ahrs_view - > sin_yaw ( ) ;
velocity_bf . y = - velocity_ned . x * ahrs_view - > sin_yaw ( ) + velocity_ned . y * ahrs_view - > cos_yaw ( ) ;
}
if ( settle_time = = 0 ) {
2024-06-16 12:41:52 -03:00
target_angle_cd = - chirp_input . update ( ( now - dwell_start_time_ms ) * 0.001 , degrees ( tgt_attitude ) * 100.0f ) ;
2024-04-01 00:41:59 -03:00
dwell_freq = chirp_input . get_frequency_rads ( ) ;
2022-03-06 20:32:12 -04:00
const Vector2f att_fdbk {
- 5730.0f * vel_hold_gain * velocity_bf . y ,
5730.0f * vel_hold_gain * velocity_bf . x
} ;
2022-01-01 23:33:52 -04:00
filt_att_fdbk_from_velxy_cd . apply ( att_fdbk , AP : : scheduler ( ) . get_loop_period_s ( ) ) ;
2021-12-13 00:02:40 -04:00
} else {
2024-04-01 00:41:59 -03:00
target_angle_cd = 0.0f ;
2024-07-03 06:34:05 -03:00
trim_yaw_tgt_reading_cd = ( float ) attitude_control - > get_att_target_euler_cd ( ) . z ;
trim_yaw_heading_reading_cd = ( float ) ahrs_view - > yaw_sensor ;
2021-12-13 00:02:40 -04:00
dwell_start_time_ms = now ;
2022-01-01 23:33:52 -04:00
filt_att_fdbk_from_velxy_cd . reset ( Vector2f ( 0.0f , 0.0f ) ) ;
2022-03-06 20:32:12 -04:00
settle_time - - ;
2021-12-13 00:02:40 -04:00
}
2024-04-01 00:41:59 -03:00
const Vector2f trim_angle_cd {
constrain_float ( filt_att_fdbk_from_velxy_cd . get ( ) . x , - 2000.0f , 2000.0f ) ,
constrain_float ( filt_att_fdbk_from_velxy_cd . get ( ) . y , - 2000.0f , 2000.0f )
} ;
switch ( axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2024-04-01 00:41:59 -03:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw ( target_angle_cd + trim_angle_cd . x , trim_angle_cd . y , 0.0f ) ;
command_reading = motors - > get_roll ( ) ;
2024-04-11 00:30:36 -03:00
if ( test_calc_type = = DRB ) {
2024-06-16 12:41:52 -03:00
tgt_rate_reading = radians ( target_angle_cd * 0.01f ) ;
gyro_reading = radians ( ( ( float ) ahrs_view - > roll_sensor + trim_angle_cd . x - target_angle_cd ) * 0.01f ) ;
2024-04-11 00:30:36 -03:00
} else if ( test_calc_type = = RATE ) {
2022-03-06 20:32:12 -04:00
tgt_rate_reading = attitude_control - > rate_bf_targets ( ) . x ;
2024-04-01 00:41:59 -03:00
gyro_reading = ahrs_view - > get_gyro ( ) . x ;
} else {
2024-06-16 12:41:52 -03:00
tgt_rate_reading = radians ( ( float ) attitude_control - > get_att_target_euler_cd ( ) . x * 0.01f ) ;
gyro_reading = radians ( ( float ) ahrs_view - > roll_sensor * 0.01f ) ;
2024-04-01 00:41:59 -03:00
}
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2024-04-01 00:41:59 -03:00
attitude_control - > input_euler_angle_roll_pitch_euler_rate_yaw ( trim_angle_cd . x , target_angle_cd + trim_angle_cd . y , 0.0f ) ;
command_reading = motors - > get_pitch ( ) ;
2024-04-11 00:30:36 -03:00
if ( test_calc_type = = DRB ) {
2024-06-16 12:41:52 -03:00
tgt_rate_reading = radians ( target_angle_cd * 0.01f ) ;
gyro_reading = radians ( ( ( float ) ahrs_view - > pitch_sensor + trim_angle_cd . y - target_angle_cd ) * 0.01f ) ;
2024-04-11 00:30:36 -03:00
} else if ( test_calc_type = = RATE ) {
2022-03-06 20:32:12 -04:00
tgt_rate_reading = attitude_control - > rate_bf_targets ( ) . y ;
2024-04-01 00:41:59 -03:00
gyro_reading = ahrs_view - > get_gyro ( ) . y ;
} else {
2024-06-16 12:41:52 -03:00
tgt_rate_reading = radians ( ( float ) attitude_control - > get_att_target_euler_cd ( ) . y * 0.01f ) ;
gyro_reading = radians ( ( float ) ahrs_view - > pitch_sensor * 0.01f ) ;
2021-12-13 00:02:40 -04:00
}
2024-04-01 00:41:59 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2024-07-03 06:34:05 -03:00
attitude_control - > input_euler_angle_roll_pitch_yaw ( trim_angle_cd . x , trim_angle_cd . y , wrap_180_cd ( trim_yaw_tgt_reading_cd + target_angle_cd ) , false ) ;
2024-04-01 00:41:59 -03:00
command_reading = motors - > get_yaw ( ) ;
2024-04-11 00:30:36 -03:00
if ( test_calc_type = = DRB ) {
2024-06-16 12:41:52 -03:00
tgt_rate_reading = radians ( target_angle_cd * 0.01f ) ;
2024-07-03 06:34:05 -03:00
gyro_reading = radians ( ( wrap_180_cd ( ( float ) ahrs_view - > yaw_sensor - trim_yaw_heading_reading_cd - target_angle_cd ) ) * 0.01f ) ;
2024-04-11 00:30:36 -03:00
} else if ( test_calc_type = = RATE ) {
2024-04-01 00:41:59 -03:00
tgt_rate_reading = attitude_control - > rate_bf_targets ( ) . z ;
gyro_reading = ahrs_view - > get_gyro ( ) . z ;
} else {
2024-07-03 06:34:05 -03:00
tgt_rate_reading = radians ( ( wrap_180_cd ( ( float ) attitude_control - > get_att_target_euler_cd ( ) . z - trim_yaw_tgt_reading_cd ) ) * 0.01f ) ;
gyro_reading = radians ( ( wrap_180_cd ( ( float ) ahrs_view - > yaw_sensor - trim_yaw_heading_reading_cd ) ) * 0.01f ) ;
2021-12-13 00:02:40 -04:00
}
2024-04-01 00:41:59 -03:00
break ;
2021-12-13 00:02:40 -04:00
}
if ( settle_time = = 0 ) {
2022-01-01 23:33:52 -04:00
filt_command_reading . apply ( command_reading , AP : : scheduler ( ) . get_loop_period_s ( ) ) ;
filt_gyro_reading . apply ( gyro_reading , AP : : scheduler ( ) . get_loop_period_s ( ) ) ;
filt_tgt_rate_reading . apply ( tgt_rate_reading , AP : : scheduler ( ) . get_loop_period_s ( ) ) ;
2021-12-13 00:02:40 -04:00
} else {
2022-01-01 23:33:52 -04:00
filt_command_reading . reset ( command_reading ) ;
filt_gyro_reading . reset ( gyro_reading ) ;
filt_tgt_rate_reading . reset ( tgt_rate_reading ) ;
2021-12-13 00:02:40 -04:00
}
// looks at gain and phase of input rate to output rate
2022-01-01 23:33:52 -04:00
rotation_rate = rotation_rate_filt . apply ( ( gyro_reading - filt_gyro_reading . get ( ) ) ,
2022-03-06 20:32:12 -04:00
AP : : scheduler ( ) . get_loop_period_s ( ) ) ;
2022-01-01 23:33:52 -04:00
filt_target_rate = target_rate_filt . apply ( ( tgt_rate_reading - filt_tgt_rate_reading . get ( ) ) ,
2022-03-06 20:32:12 -04:00
AP : : scheduler ( ) . get_loop_period_s ( ) ) ;
2022-01-01 23:33:52 -04:00
command_out = command_filt . apply ( ( command_reading - filt_command_reading . get ( ) ) ,
2022-03-06 20:32:12 -04:00
AP : : scheduler ( ) . get_loop_period_s ( ) ) ;
2021-12-13 00:02:40 -04:00
2024-04-07 00:55:51 -03:00
float dwell_gain_mtr = 0.0f ;
float dwell_phase_mtr = 0.0f ;
float dwell_gain_tgt = 0.0f ;
float dwell_phase_tgt = 0.0f ;
2022-03-06 20:32:12 -04:00
// wait for dwell to start before determining gain and phase
2024-04-11 00:30:36 -03:00
if ( ( float ) ( now - dwell_start_time_ms ) > pre_calc_cycles * cycle_time_ms | | ( test_input_type = = AC_AutoTune_FreqResp : : InputType : : SWEEP & & settle_time = = 0 ) ) {
2024-04-07 00:55:51 -03:00
freqresp_mtr . update ( command_out , command_out , rotation_rate , dwell_freq ) ;
freqresp_tgt . update ( command_out , filt_target_rate , rotation_rate , dwell_freq ) ;
if ( freqresp_mtr . is_cycle_complete ( ) ) {
2024-04-11 00:30:36 -03:00
if ( test_input_type = = AC_AutoTune_FreqResp : : InputType : : SWEEP ) {
2024-04-20 11:36:24 -03:00
if ( is_zero ( curr_test_mtr . freq ) & & freqresp_mtr . get_freq ( ) < test_start_freq ) {
// don't set data since captured frequency is below the start frequency
} else {
curr_test_mtr . freq = freqresp_mtr . get_freq ( ) ;
curr_test_mtr . gain = freqresp_mtr . get_gain ( ) ;
curr_test_mtr . phase = freqresp_mtr . get_phase ( ) ;
}
2024-04-07 00:55:51 -03:00
// reset cycle_complete to allow indication of next cycle
freqresp_mtr . reset_cycle_complete ( ) ;
# if HAL_LOGGING_ENABLED
// log sweep data
Log_AutoTuneSweep ( ) ;
# endif
} else {
dwell_gain_mtr = freqresp_mtr . get_gain ( ) ;
dwell_phase_mtr = freqresp_mtr . get_phase ( ) ;
2024-04-08 00:34:47 -03:00
cycle_complete_mtr = true ;
2024-04-07 00:55:51 -03:00
}
2021-12-13 00:02:40 -04:00
}
2022-03-06 20:32:12 -04:00
2024-04-07 00:55:51 -03:00
if ( freqresp_tgt . is_cycle_complete ( ) ) {
2024-04-11 00:30:36 -03:00
if ( test_input_type = = AC_AutoTune_FreqResp : : InputType : : SWEEP ) {
2024-04-20 11:36:24 -03:00
if ( is_zero ( curr_test_tgt . freq ) & & freqresp_tgt . get_freq ( ) < test_start_freq ) {
// don't set data since captured frequency is below the start frequency
} else {
curr_test_tgt . freq = freqresp_tgt . get_freq ( ) ;
curr_test_tgt . gain = freqresp_tgt . get_gain ( ) ;
curr_test_tgt . phase = freqresp_tgt . get_phase ( ) ;
if ( test_calc_type = = DRB ) { test_accel_max = freqresp_tgt . get_accel_max ( ) ; }
}
2021-12-13 00:02:40 -04:00
// reset cycle_complete to allow indication of next cycle
2024-04-07 00:55:51 -03:00
freqresp_tgt . reset_cycle_complete ( ) ;
2023-07-13 21:58:04 -03:00
# if HAL_LOGGING_ENABLED
2021-12-13 00:02:40 -04:00
// log sweep data
Log_AutoTuneSweep ( ) ;
2023-07-13 21:58:04 -03:00
# endif
2021-12-13 00:02:40 -04:00
} else {
2024-04-07 00:55:51 -03:00
dwell_gain_tgt = freqresp_tgt . get_gain ( ) ;
dwell_phase_tgt = freqresp_tgt . get_phase ( ) ;
2024-04-11 00:30:36 -03:00
if ( test_calc_type = = DRB ) { test_accel_max = freqresp_tgt . get_accel_max ( ) ; }
2024-04-08 00:34:47 -03:00
cycle_complete_tgt = true ;
2024-04-07 00:55:51 -03:00
}
}
2024-04-11 00:30:36 -03:00
if ( test_freq_resp_input = = TARGET ) {
if ( test_input_type = = AC_AutoTune_FreqResp : : InputType : : SWEEP ) {
2024-04-07 00:55:51 -03:00
curr_test = curr_test_tgt ;
} else {
2024-04-12 00:21:16 -03:00
test_data . freq = test_start_freq ;
test_data . gain = dwell_gain_tgt ;
test_data . phase = dwell_phase_tgt ;
2024-04-07 00:55:51 -03:00
}
} else {
2024-04-11 00:30:36 -03:00
if ( test_input_type = = AC_AutoTune_FreqResp : : InputType : : SWEEP ) {
2024-04-07 00:55:51 -03:00
curr_test = curr_test_mtr ;
} else {
2024-04-12 00:21:16 -03:00
test_data . freq = test_start_freq ;
test_data . gain = dwell_gain_mtr ;
test_data . phase = dwell_phase_mtr ;
2021-12-13 00:02:40 -04:00
}
}
}
// set sweep data if a frequency sweep is being conducted
2024-04-11 00:30:36 -03:00
if ( test_input_type = = AC_AutoTune_FreqResp : : InputType : : SWEEP & & ( float ) ( now - dwell_start_time_ms ) > 2.5f * cycle_time_ms ) {
2021-12-13 00:02:40 -04:00
// track sweep phase to prevent capturing 180 deg and 270 deg data after phase has wrapped.
2024-04-07 00:55:51 -03:00
if ( curr_test_tgt . phase > 180.0f & & sweep_tgt . progress = = 0 ) {
sweep_tgt . progress = 1 ;
} else if ( curr_test_tgt . phase > 270.0f & & sweep_tgt . progress = = 1 ) {
sweep_tgt . progress = 2 ;
}
if ( curr_test_tgt . phase < = 160.0f & & curr_test_tgt . phase > = 150.0f & & sweep_tgt . progress = = 0 ) {
sweep_tgt . ph180 = curr_test_tgt ;
}
if ( curr_test_tgt . phase < = 250.0f & & curr_test_tgt . phase > = 240.0f & & sweep_tgt . progress = = 1 ) {
sweep_tgt . ph270 = curr_test_tgt ;
}
if ( curr_test_tgt . gain > sweep_tgt . maxgain . gain ) {
sweep_tgt . maxgain = curr_test_tgt ;
}
// Determine sweep info for motor input to response output
if ( curr_test_mtr . phase > 180.0f & & sweep_mtr . progress = = 0 ) {
sweep_mtr . progress = 1 ;
} else if ( curr_test_mtr . phase > 270.0f & & sweep_mtr . progress = = 1 ) {
sweep_mtr . progress = 2 ;
2021-12-13 00:02:40 -04:00
}
2024-04-07 00:55:51 -03:00
if ( curr_test_mtr . phase < = 160.0f & & curr_test_mtr . phase > = 150.0f & & sweep_mtr . progress = = 0 ) {
sweep_mtr . ph180 = curr_test_mtr ;
2021-12-13 00:02:40 -04:00
}
2024-04-07 00:55:51 -03:00
if ( curr_test_mtr . phase < = 250.0f & & curr_test_mtr . phase > = 240.0f & & sweep_mtr . progress = = 1 ) {
sweep_mtr . ph270 = curr_test_mtr ;
2021-12-13 00:02:40 -04:00
}
2024-04-07 00:55:51 -03:00
if ( curr_test_mtr . gain > sweep_mtr . maxgain . gain ) {
sweep_mtr . maxgain = curr_test_mtr ;
2021-12-13 00:02:40 -04:00
}
2024-04-07 00:55:51 -03:00
2021-12-13 00:02:40 -04:00
if ( now - step_start_time_ms > = sweep_time_ms + 200 ) {
// we have passed the maximum stop time
2024-04-20 11:36:24 -03:00
sweep_complete = true ;
2021-12-13 00:02:40 -04:00
step = UPDATE_GAINS ;
}
} else {
2024-04-07 00:55:51 -03:00
if ( now - step_start_time_ms > = step_time_limit_ms | | ( freqresp_tgt . is_cycle_complete ( ) & & freqresp_mtr . is_cycle_complete ( ) ) ) {
2024-04-08 00:34:47 -03:00
if ( now - step_start_time_ms > = step_time_limit_ms ) {
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: Step time limit exceeded " ) ;
2024-04-08 00:34:47 -03:00
}
cycle_complete_tgt = false ;
cycle_complete_tgt = false ;
2021-12-13 00:02:40 -04:00
// we have passed the maximum stop time
step = UPDATE_GAINS ;
}
}
}
2021-01-17 22:28:20 -04:00
// update gains for the rate p up tune type
2020-10-19 23:51:05 -03:00
void AC_AutoTune_Heli : : updating_rate_p_up_all ( AxisType test_axis )
{
switch ( test_axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2024-04-12 02:09:33 -03:00
updating_rate_p_up ( tune_roll_rp , curr_data , next_test_freq , max_rate_p ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2024-04-12 02:09:33 -03:00
updating_rate_p_up ( tune_pitch_rp , curr_data , next_test_freq , max_rate_p ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2024-04-12 02:09:33 -03:00
updating_rate_p_up ( tune_yaw_rp , curr_data , next_test_freq , max_rate_p ) ;
2020-10-19 23:51:05 -03:00
break ;
}
}
2021-01-17 22:28:20 -04:00
// update gains for the rate d up tune type
2020-10-19 23:51:05 -03:00
void AC_AutoTune_Heli : : updating_rate_d_up_all ( AxisType test_axis )
{
switch ( test_axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2024-04-12 02:09:33 -03:00
updating_rate_d_up ( tune_roll_rd , curr_data , next_test_freq , max_rate_d ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2024-04-12 02:09:33 -03:00
updating_rate_d_up ( tune_pitch_rd , curr_data , next_test_freq , max_rate_d ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2024-04-12 02:09:33 -03:00
updating_rate_d_up ( tune_yaw_rd , curr_data , next_test_freq , max_rate_d ) ;
2020-10-19 23:51:05 -03:00
break ;
}
}
2021-01-17 22:28:20 -04:00
// update gains for the rate ff up tune type
2020-10-19 23:51:05 -03:00
void AC_AutoTune_Heli : : updating_rate_ff_up_all ( AxisType test_axis )
{
switch ( test_axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2024-04-12 00:21:16 -03:00
updating_rate_ff_up ( tune_roll_rff , curr_data , next_test_freq ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2024-04-12 00:21:16 -03:00
updating_rate_ff_up ( tune_pitch_rff , curr_data , next_test_freq ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2024-04-12 00:21:16 -03:00
updating_rate_ff_up ( tune_yaw_rff , curr_data , next_test_freq ) ;
2021-01-18 00:16:08 -04:00
// TODO make FF updating routine determine when to set rff gain to zero based on A/C response
if ( tune_yaw_rff < = AUTOTUNE_RFF_MIN & & counter = = AUTOTUNE_SUCCESS_COUNT ) {
tune_yaw_rff = 0.0f ;
}
2020-10-19 23:51:05 -03:00
break ;
}
}
2021-01-17 22:28:20 -04:00
// update gains for the angle p up tune type
2020-10-19 23:51:05 -03:00
void AC_AutoTune_Heli : : updating_angle_p_up_all ( AxisType test_axis )
{
2022-05-03 00:35:50 -03:00
attitude_control - > bf_feedforward ( orig_bf_feedforward ) ;
2024-04-12 23:26:56 -03:00
// sweep doesn't require gain update so return immediately after setting next test freq
2024-04-13 02:04:33 -03:00
// determine next_test_freq for dwell testing
if ( sweep_complete & & input_type = = AC_AutoTune_FreqResp : : InputType : : SWEEP ) {
// if a max gain frequency was found then set the start of the dwells to that freq otherwise start at min frequency
if ( ! is_zero ( sweep_tgt . maxgain . freq ) ) {
2024-04-20 11:36:24 -03:00
next_test_freq = constrain_float ( sweep_tgt . maxgain . freq , min_sweep_freq , max_sweep_freq ) ;
2024-05-19 23:36:38 -03:00
freq_max = next_test_freq ;
sp_prev_gain = sweep_tgt . maxgain . gain ;
phase_max = sweep_tgt . maxgain . phase ;
found_max_gain_freq = true ;
2024-04-13 02:04:33 -03:00
} else {
next_test_freq = min_sweep_freq ;
2024-04-12 23:26:56 -03:00
}
return ;
}
2020-10-19 23:51:05 -03:00
switch ( test_axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2024-04-12 23:26:56 -03:00
updating_angle_p_up ( tune_roll_sp , curr_data , next_test_freq ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2024-04-12 23:26:56 -03:00
updating_angle_p_up ( tune_pitch_sp , curr_data , next_test_freq ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2024-04-12 23:26:56 -03:00
updating_angle_p_up ( tune_yaw_sp , curr_data , next_test_freq ) ;
2020-10-19 23:51:05 -03:00
break ;
}
}
2021-01-17 22:28:20 -04:00
// update gains for the max gain tune type
2020-10-19 23:51:05 -03:00
void AC_AutoTune_Heli : : updating_max_gains_all ( AxisType test_axis )
{
2024-04-13 02:04:33 -03:00
// sweep doesn't require gain update so return immediately after setting next test freq
// determine next_test_freq for dwell testing
if ( sweep_complete & & input_type = = AC_AutoTune_FreqResp : : InputType : : SWEEP ) {
// if a max gain frequency was found then set the start of the dwells to that freq otherwise start at min frequency
if ( ! is_zero ( sweep_mtr . ph180 . freq ) ) {
2024-04-20 11:36:24 -03:00
next_test_freq = constrain_float ( sweep_mtr . ph180 . freq , min_sweep_freq , max_sweep_freq ) ;
2024-04-13 02:04:33 -03:00
reset_maxgains_update_gain_variables ( ) ;
} else {
next_test_freq = min_sweep_freq ;
}
return ;
}
2020-10-19 23:51:05 -03:00
switch ( test_axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2024-04-13 02:04:33 -03:00
updating_max_gains ( curr_data , next_test_freq , max_rate_p , max_rate_d , tune_roll_rp , tune_roll_rd ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2024-04-13 02:04:33 -03:00
updating_max_gains ( curr_data , next_test_freq , max_rate_p , max_rate_d , tune_pitch_rp , tune_pitch_rd ) ;
2020-10-19 23:51:05 -03:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2024-04-13 02:04:33 -03:00
updating_max_gains ( curr_data , next_test_freq , max_rate_p , max_rate_d , tune_yaw_rp , tune_yaw_rd ) ;
2021-01-18 00:16:08 -04:00
// rate P and rate D can be non zero for yaw and need to be included in the max allowed gain
if ( ! is_zero ( max_rate_p . max_allowed ) & & counter = = AUTOTUNE_SUCCESS_COUNT ) {
max_rate_p . max_allowed + = tune_yaw_rp ;
}
if ( ! is_zero ( max_rate_d . max_allowed ) & & counter = = AUTOTUNE_SUCCESS_COUNT ) {
max_rate_d . max_allowed + = tune_yaw_rd ;
}
2020-10-19 23:51:05 -03:00
break ;
}
}
2021-12-27 00:53:50 -04:00
// set gains post tune for the tune type
void AC_AutoTune_Heli : : set_gains_post_tune ( AxisType test_axis )
{
switch ( tune_type ) {
case RD_UP :
switch ( test_axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2021-12-27 00:53:50 -04:00
tune_roll_rd = MAX ( 0.0f , tune_roll_rd * AUTOTUNE_RD_BACKOFF ) ;
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2021-12-27 00:53:50 -04:00
tune_pitch_rd = MAX ( 0.0f , tune_pitch_rd * AUTOTUNE_RD_BACKOFF ) ;
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2022-01-07 14:37:02 -04:00
tune_yaw_rd = MAX ( 0.0f , tune_yaw_rd * AUTOTUNE_RD_BACKOFF ) ;
2021-12-27 00:53:50 -04:00
break ;
}
break ;
case RP_UP :
switch ( test_axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2022-01-07 14:37:02 -04:00
tune_roll_rp = MAX ( 0.0f , tune_roll_rp * AUTOTUNE_RP_BACKOFF ) ;
2021-12-27 00:53:50 -04:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2022-01-07 14:37:02 -04:00
tune_pitch_rp = MAX ( 0.0f , tune_pitch_rp * AUTOTUNE_RP_BACKOFF ) ;
2021-12-27 00:53:50 -04:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2021-12-27 00:53:50 -04:00
tune_yaw_rp = MAX ( AUTOTUNE_RP_MIN , tune_yaw_rp * AUTOTUNE_RP_BACKOFF ) ;
break ;
}
break ;
case SP_UP :
switch ( test_axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2021-12-27 00:53:50 -04:00
tune_roll_sp = MAX ( AUTOTUNE_SP_MIN , tune_roll_sp * AUTOTUNE_SP_BACKOFF ) ;
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2021-12-27 00:53:50 -04:00
tune_pitch_sp = MAX ( AUTOTUNE_SP_MIN , tune_pitch_sp * AUTOTUNE_SP_BACKOFF ) ;
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2021-12-27 00:53:50 -04:00
tune_yaw_sp = MAX ( AUTOTUNE_SP_MIN , tune_yaw_sp * AUTOTUNE_SP_BACKOFF ) ;
break ;
}
break ;
case RFF_UP :
2022-01-07 14:37:02 -04:00
break ;
default :
2021-12-27 00:53:50 -04:00
break ;
}
}
2020-10-19 23:51:05 -03:00
// updating_rate_ff_up - adjust FF to ensure the target is reached
2023-10-11 04:41:49 -03:00
// FF is adjusted until rate requested is achieved
2024-04-12 00:21:16 -03:00
void AC_AutoTune_Heli : : updating_rate_ff_up ( float & tune_ff , sweep_info & test_data , float & next_freq )
2020-10-19 23:51:05 -03:00
{
2024-05-19 23:36:38 -03:00
float tune_tgt = 0.95 ;
float tune_tol = 0.025 ;
2024-04-12 00:21:16 -03:00
next_freq = test_data . freq ;
2024-05-19 23:36:38 -03:00
// handle axes where FF gain is initially zero
if ( test_data . gain < tune_tgt - tune_tol & & ! is_positive ( tune_ff ) ) {
tune_ff = 0.03f ;
return ;
}
if ( test_data . gain < tune_tgt - 0.2 | | test_data . gain > tune_tgt + 0.2 ) {
tune_ff = tune_ff * constrain_float ( tune_tgt / test_data . gain , 0.75 , 1.25 ) ; //keep changes less than 25%
} else if ( test_data . gain < tune_tgt - 0.1 | | test_data . gain > tune_tgt + 0.1 ) {
if ( test_data . gain < tune_tgt - 0.1 ) {
tune_ff * = 1.05 ;
} else {
tune_ff * = 0.95 ;
}
} else if ( test_data . gain < tune_tgt - tune_tol | | test_data . gain > tune_tgt + tune_tol ) {
if ( test_data . gain < tune_tgt - tune_tol ) {
tune_ff * = 1.02 ;
} else {
tune_ff * = 0.98 ;
2021-01-18 00:16:08 -04:00
}
2024-05-19 23:36:38 -03:00
} else if ( test_data . gain > = tune_tgt - tune_tol & & test_data . gain < = tune_tgt + tune_tol ) {
counter = AUTOTUNE_SUCCESS_COUNT ;
// reset next_freq for next test
next_freq = 0.0f ;
tune_ff = constrain_float ( tune_ff , 0.0f , 1.0f ) ;
2024-04-04 22:42:18 -03:00
}
2020-10-19 23:51:05 -03:00
}
2024-06-17 00:28:31 -03:00
// updating_rate_p_up - uses maximum allowable gain determined from max_gain test to determine rate p gain that does not
// exceed max response gain. A phase of 161 deg is used to conduct the tuning as this phase is where analytically
// max gain to 6db gain margin is determined for a unity feedback controller.
2024-04-12 02:09:33 -03:00
void AC_AutoTune_Heli : : updating_rate_p_up ( float & tune_p , sweep_info & test_data , float & next_freq , max_gain_data & max_gain_p )
2020-10-19 23:51:05 -03:00
{
2024-04-16 00:18:32 -03:00
float test_freq_incr = 0.25f * M_2PI ;
2024-04-12 02:09:33 -03:00
next_freq = test_data . freq ;
2021-01-18 00:16:08 -04:00
2024-04-16 00:18:32 -03:00
sweep_info data_at_ph161 ;
float sugg_freq ;
if ( freq_search_for_phase ( test_data , 161.0f , test_freq_incr , data_at_ph161 , sugg_freq ) ) {
if ( data_at_ph161 . gain < max_resp_gain & & tune_p < 0.6f * max_gain_p . max_allowed ) {
2024-04-12 02:09:33 -03:00
tune_p + = 0.05f * max_gain_p . max_allowed ;
2024-04-16 00:18:32 -03:00
next_freq = data_at_ph161 . freq ;
2024-04-12 02:09:33 -03:00
} else {
counter = AUTOTUNE_SUCCESS_COUNT ;
// reset next_freq for next test
next_freq = 0.0f ;
tune_p - = 0.05f * max_gain_p . max_allowed ;
tune_p = constrain_float ( tune_p , 0.0f , 0.6f * max_gain_p . max_allowed ) ;
}
2024-04-16 00:18:32 -03:00
} else {
next_freq = sugg_freq ;
2021-01-18 00:16:08 -04:00
}
2020-10-19 23:51:05 -03:00
}
2024-06-17 00:28:31 -03:00
// updating_rate_d_up - uses maximum allowable gain determined from max_gain test to determine rate d gain where the response
// gain is at a minimum. A phase of 161 deg is used to conduct the tuning as this phase is where analytically
// max gain to 6db gain margin is determined for a unity feedback controller.
2024-04-12 02:09:33 -03:00
void AC_AutoTune_Heli : : updating_rate_d_up ( float & tune_d , sweep_info & test_data , float & next_freq , max_gain_data & max_gain_d )
2020-10-19 23:51:05 -03:00
{
2024-04-16 00:18:32 -03:00
float test_freq_incr = 0.25f * M_2PI ; // set for 1/4 hz increments
2024-04-12 02:09:33 -03:00
next_freq = test_data . freq ;
2020-10-19 23:51:05 -03:00
2024-04-16 00:18:32 -03:00
sweep_info data_at_ph161 ;
float sugg_freq ;
if ( freq_search_for_phase ( test_data , 161.0f , test_freq_incr , data_at_ph161 , sugg_freq ) ) {
if ( ( data_at_ph161 . gain < rd_prev_gain | | is_zero ( rd_prev_gain ) ) & & tune_d < 0.6f * max_gain_d . max_allowed ) {
tune_d + = 0.05f * max_gain_d . max_allowed ;
rd_prev_gain = data_at_ph161 . gain ;
next_freq = data_at_ph161 . freq ;
} else {
counter = AUTOTUNE_SUCCESS_COUNT ;
// reset next freq and rd_prev_gain for next test
next_freq = 0 ;
rd_prev_gain = 0.0f ;
tune_d - = 0.05f * max_gain_d . max_allowed ;
tune_d = constrain_float ( tune_d , 0.0f , 0.6f * max_gain_d . max_allowed ) ;
}
} else {
next_freq = sugg_freq ;
2021-01-18 00:16:08 -04:00
}
2020-10-19 23:51:05 -03:00
}
2024-06-17 00:28:31 -03:00
// updating_angle_p_up - determines maximum angle p gain for pitch and roll. This is accomplished by determining the frequency
// for the maximum response gain that is the disturbance rejection peak.
2024-04-12 23:26:56 -03:00
void AC_AutoTune_Heli : : updating_angle_p_up ( float & tune_p , sweep_info & test_data , float & next_freq )
2021-01-18 00:16:08 -04:00
{
2024-04-16 00:18:32 -03:00
float test_freq_incr = 0.5f * M_2PI ;
2021-01-18 00:16:08 -04:00
float gain_incr = 0.5f ;
2024-04-12 23:26:56 -03:00
if ( is_zero ( test_data . phase ) ) {
// bad test point. increase slightly in hope of getting better result
next_freq + = 0.5f * test_freq_incr ;
return ;
2021-01-18 00:16:08 -04:00
}
2024-04-12 23:26:56 -03:00
if ( ! found_max_gain_freq ) {
if ( test_data . gain > max_resp_gain & & tune_p > AUTOTUNE_SP_MIN ) {
2021-01-18 00:16:08 -04:00
// exceeded max response gain already, reduce tuning gain to remain under max response gain
tune_p - = gain_incr ;
// force counter to stay on frequency
2024-04-12 23:26:56 -03:00
next_freq = test_data . freq ;
return ;
} else if ( test_data . gain > max_resp_gain & & tune_p < = AUTOTUNE_SP_MIN ) {
2021-01-18 00:16:08 -04:00
// exceeded max response gain at the minimum allowable tuning gain. terminate testing.
tune_p = AUTOTUNE_SP_MIN ;
counter = AUTOTUNE_SUCCESS_COUNT ;
2023-07-13 21:58:04 -03:00
LOGGER_WRITE_EVENT ( LogEvent : : AUTOTUNE_REACHED_LIMIT ) ;
2024-04-12 23:26:56 -03:00
} else if ( test_data . gain > sp_prev_gain ) {
freq_max = test_data . freq ;
phase_max = test_data . phase ;
sp_prev_gain = test_data . gain ;
next_freq = test_data . freq + test_freq_incr ;
return ;
// Gain is expected to continue decreasing past gain peak. declare max gain freq found and refine search.
2024-05-19 23:36:38 -03:00
} else if ( test_data . gain < 0.95f * sp_prev_gain ) {
2024-04-12 23:26:56 -03:00
found_max_gain_freq = true ;
next_freq = freq_max + 0.5 * test_freq_incr ;
return ;
} else {
next_freq = test_data . freq + test_freq_incr ;
return ;
2021-01-18 00:16:08 -04:00
}
2024-04-12 23:26:56 -03:00
}
// refine peak
if ( ! found_peak ) {
// look at frequency above max gain freq found
if ( test_data . freq > freq_max & & test_data . gain > sp_prev_gain ) {
// found max at frequency greater than initial max gain frequency
found_peak = true ;
} else if ( test_data . freq > freq_max & & test_data . gain < sp_prev_gain ) {
// look at frequency below initial max gain frequency
next_freq = test_data . freq - 0.5 * test_freq_incr ;
return ;
} else if ( test_data . freq < freq_max & & test_data . gain > sp_prev_gain ) {
// found max at frequency less than initial max gain frequency
found_peak = true ;
2021-01-18 00:16:08 -04:00
} else {
2024-04-12 23:26:56 -03:00
found_peak = true ;
test_data . freq = freq_max ;
test_data . gain = sp_prev_gain ;
2021-01-18 00:16:08 -04:00
}
2024-04-12 23:26:56 -03:00
sp_prev_gain = test_data . gain ;
2021-01-18 00:16:08 -04:00
}
2024-04-12 23:26:56 -03:00
// start increasing gain
if ( found_max_gain_freq & & found_peak ) {
if ( test_data . gain < max_resp_gain & & tune_p < AUTOTUNE_SP_MAX ) {
tune_p + = gain_incr ;
next_freq = test_data . freq ;
if ( tune_p > = AUTOTUNE_SP_MAX ) {
tune_p = AUTOTUNE_SP_MAX ;
counter = AUTOTUNE_SUCCESS_COUNT ;
LOGGER_WRITE_EVENT ( LogEvent : : AUTOTUNE_REACHED_LIMIT ) ;
2021-01-18 00:16:08 -04:00
}
2024-04-12 23:26:56 -03:00
sp_prev_gain = test_data . gain ;
} else if ( test_data . gain > 1.1f * max_resp_gain & & tune_p > AUTOTUNE_SP_MIN ) {
2021-12-13 23:26:57 -04:00
tune_p - = gain_incr ;
2021-01-18 00:16:08 -04:00
} else {
// adjust tuning gain so max response gain is not exceeded
2024-04-12 23:26:56 -03:00
if ( sp_prev_gain < max_resp_gain & & test_data . gain > max_resp_gain ) {
float adj_factor = ( max_resp_gain - test_data . gain ) / ( test_data . gain - sp_prev_gain ) ;
2021-01-18 00:16:08 -04:00
tune_p = tune_p + gain_incr * adj_factor ;
}
counter = AUTOTUNE_SUCCESS_COUNT ;
}
}
if ( counter = = AUTOTUNE_SUCCESS_COUNT ) {
2024-04-12 23:26:56 -03:00
next_freq = 0.0f ; //initializes next test that uses dwell test
sweep_complete = false ;
2022-01-16 19:13:59 -04:00
reset_sweep_variables ( ) ;
2021-01-18 00:16:08 -04:00
}
}
2024-06-17 00:28:31 -03:00
// updating_max_gains: use dwells at increasing frequency to determine gain at which instability will occur. This uses the frequency
// response of motor class input to rate response to determine the max allowable gain for rate P gain. A phase of 161 deg is used to
// determine analytically the max gain to 6db gain margin for a unity feedback controller. Since acceleration can be more noisy, the
// response of the motor class input to rate response to determine the max allowable gain for rate D gain. A phase of 251 deg is used
// to determine analytically the max gain to 6db gain margin for a unity feedback controller.
2024-04-13 02:04:33 -03:00
void AC_AutoTune_Heli : : updating_max_gains ( sweep_info & test_data , float & next_freq , max_gain_data & max_gain_p , max_gain_data & max_gain_d , float & tune_p , float & tune_d )
2020-10-19 23:51:05 -03:00
{
2024-04-16 00:18:32 -03:00
float test_freq_incr = 0.5f * M_2PI ;
next_freq = test_data . freq ;
2021-01-18 00:16:08 -04:00
2024-04-16 00:18:32 -03:00
sweep_info data_at_phase ;
float sugg_freq ;
2024-04-13 02:04:33 -03:00
if ( ! found_max_p ) {
2024-04-16 00:18:32 -03:00
if ( freq_search_for_phase ( test_data , 161.0f , test_freq_incr , data_at_phase , sugg_freq ) ) {
max_gain_p . freq = data_at_phase . freq ;
max_gain_p . gain = data_at_phase . gain ;
max_gain_p . phase = data_at_phase . phase ;
2020-10-19 23:51:05 -03:00
max_gain_p . max_allowed = powf ( 10.0f , - 1 * ( log10f ( max_gain_p . gain ) * 20.0f + 2.42 ) / 20.0f ) ;
2021-01-18 00:16:08 -04:00
// limit max gain allowed to be no more than 2x the max p gain limit to keep initial gains bounded
max_gain_p . max_allowed = constrain_float ( max_gain_p . max_allowed , 0.0f , 2.0f * AUTOTUNE_RP_MAX ) ;
found_max_p = true ;
2024-04-07 00:55:51 -03:00
if ( ! is_zero ( sweep_mtr . ph270 . freq ) ) {
2024-04-16 00:18:32 -03:00
next_freq = sweep_mtr . ph270 . freq ;
} else {
next_freq = data_at_phase . freq ;
2021-01-18 00:16:08 -04:00
}
2024-04-16 00:18:32 -03:00
} else {
next_freq = sugg_freq ;
2021-01-18 00:16:08 -04:00
}
2024-04-13 02:04:33 -03:00
} else if ( ! found_max_d ) {
2024-04-16 00:18:32 -03:00
if ( freq_search_for_phase ( test_data , 251.0f , test_freq_incr , data_at_phase , sugg_freq ) ) {
max_gain_d . freq = data_at_phase . freq ;
max_gain_d . gain = data_at_phase . gain ;
max_gain_d . phase = data_at_phase . phase ;
2020-10-19 23:51:05 -03:00
max_gain_d . max_allowed = powf ( 10.0f , - 1 * ( log10f ( max_gain_d . freq * max_gain_d . gain ) * 20.0f + 2.42 ) / 20.0f ) ;
2021-01-18 00:16:08 -04:00
// limit max gain allowed to be no more than 2x the max d gain limit to keep initial gains bounded
max_gain_d . max_allowed = constrain_float ( max_gain_d . max_allowed , 0.0f , 2.0f * AUTOTUNE_RD_MAX ) ;
found_max_d = true ;
2024-04-16 00:18:32 -03:00
} else {
next_freq = sugg_freq ;
2020-10-19 23:51:05 -03:00
}
2024-04-13 02:04:33 -03:00
}
2024-04-16 00:18:32 -03:00
if ( found_max_p & & found_max_d ) {
2024-04-13 02:04:33 -03:00
counter = AUTOTUNE_SUCCESS_COUNT ;
// reset variables for next test
next_freq = 0.0f ; //initializes next test that uses dwell test
2024-08-07 00:17:19 -03:00
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: Max rate P freq=%f gain=%f " , ( double ) ( max_rate_p . freq ) , ( double ) ( max_rate_p . gain ) ) ;
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: ph=%f rate_p=%f " , ( double ) ( max_rate_p . phase ) , ( double ) ( max_rate_p . max_allowed ) ) ;
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: Max Rate D freq=%f gain=%f " , ( double ) ( max_rate_d . freq ) , ( double ) ( max_rate_d . gain ) ) ;
GCS_SEND_TEXT ( MAV_SEVERITY_INFO , " AutoTune: ph=%f rate_d=%f " , ( double ) ( max_rate_d . phase ) , ( double ) ( max_rate_d . max_allowed ) ) ;
2022-01-07 14:37:02 -04:00
}
2020-10-19 23:51:05 -03:00
}
2024-02-29 02:42:39 -04:00
float AC_AutoTune_Heli : : target_angle_max_rp_cd ( ) const
{
return AUTOTUNE_ANGLE_TARGET_MAX_RP_CD ;
}
float AC_AutoTune_Heli : : target_angle_max_y_cd ( ) const
{
return AUTOTUNE_ANGLE_TARGET_MAX_Y_CD ;
}
float AC_AutoTune_Heli : : target_angle_min_rp_cd ( ) const
{
return AUTOTUNE_ANGLE_TARGET_MIN_RP_CD ;
}
float AC_AutoTune_Heli : : target_angle_min_y_cd ( ) const
{
return AUTOTUNE_ANGLE_TARGET_MIN_Y_CD ;
}
float AC_AutoTune_Heli : : angle_lim_max_rp_cd ( ) const
{
return AUTOTUNE_ANGLE_MAX_RP_CD ;
}
float AC_AutoTune_Heli : : angle_lim_neg_rpy_cd ( ) const
{
return AUTOTUNE_ANGLE_NEG_RPY_CD ;
}
2024-04-16 00:18:32 -03:00
// freq_search_for_phase: general search strategy for specified phase. interpolation done once specified phase has been bounded.
bool AC_AutoTune_Heli : : freq_search_for_phase ( sweep_info test , float desired_phase , float freq_incr , sweep_info & est_data , float & new_freq )
{
new_freq = test . freq ;
float phase_delta = 20.0f ; // delta from desired phase below and above which full steps are taken
if ( is_zero ( test . phase ) ) {
// bad test point. increase slightly in hope of getting better result
new_freq + = 0.1f * freq_incr ;
return false ;
}
// test to see if desired phase is bounded with a 0.5 freq_incr delta in freq
float freq_delta = fabsf ( prev_test . freq - test . freq ) ;
if ( test . phase > desired_phase & & prev_test . phase < desired_phase & & freq_delta < 0.75f * freq_incr & & is_positive ( prev_test . freq ) ) {
est_data . freq = linear_interpolate ( prev_test . freq , test . freq , desired_phase , prev_test . phase , test . phase ) ;
est_data . gain = linear_interpolate ( prev_test . gain , test . gain , desired_phase , prev_test . phase , test . phase ) ;
est_data . phase = desired_phase ;
prev_test = { } ;
return true ;
} else if ( test . phase < desired_phase & & prev_test . phase > desired_phase & & freq_delta < 0.75f * freq_incr & & is_positive ( prev_test . freq ) ) {
est_data . freq = linear_interpolate ( test . freq , prev_test . freq , desired_phase , test . phase , prev_test . phase ) ;
est_data . gain = linear_interpolate ( test . gain , prev_test . gain , desired_phase , test . phase , prev_test . phase ) ;
est_data . phase = desired_phase ;
prev_test = { } ;
return true ;
}
if ( test . phase < desired_phase - phase_delta ) {
new_freq + = freq_incr ;
} else if ( test . phase > desired_phase + phase_delta ) {
new_freq - = freq_incr ;
} else if ( test . phase > = desired_phase - phase_delta & & test . phase < desired_phase ) {
new_freq + = 0.5f * freq_incr ;
} else if ( test . phase < = desired_phase + phase_delta & & test . phase > = desired_phase ) {
new_freq - = 0.5f * freq_incr ;
}
prev_test = test ;
return false ;
}
2023-07-13 21:58:04 -03:00
# if HAL_LOGGING_ENABLED
2021-12-06 00:27:42 -04:00
// log autotune summary data
2020-10-19 23:51:05 -03:00
void AC_AutoTune_Heli : : Log_AutoTune ( )
{
2022-01-07 14:37:02 -04:00
switch ( axis ) {
2024-07-25 01:07:26 -03:00
case AxisType : : ROLL :
2024-04-13 02:04:33 -03:00
Log_Write_AutoTune ( axis , tune_type , curr_data . freq , curr_data . gain , curr_data . phase , tune_roll_rff , tune_roll_rp , tune_roll_rd , tune_roll_sp , test_accel_max ) ;
2022-01-07 14:37:02 -04:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : PITCH :
2024-04-13 02:04:33 -03:00
Log_Write_AutoTune ( axis , tune_type , curr_data . freq , curr_data . gain , curr_data . phase , tune_pitch_rff , tune_pitch_rp , tune_pitch_rd , tune_pitch_sp , test_accel_max ) ;
2022-01-07 14:37:02 -04:00
break ;
2024-07-25 01:07:26 -03:00
case AxisType : : YAW :
case AxisType : : YAW_D :
2024-04-13 02:04:33 -03:00
Log_Write_AutoTune ( axis , tune_type , curr_data . freq , curr_data . gain , curr_data . phase , tune_yaw_rff , tune_yaw_rp , tune_yaw_rd , tune_yaw_sp , test_accel_max ) ;
2022-01-07 14:37:02 -04:00
break ;
}
2021-01-18 00:16:08 -04:00
}
2021-12-06 00:27:42 -04:00
// log autotune time history results for command, angular rate, and attitude
2021-01-18 00:16:08 -04:00
void AC_AutoTune_Heli : : Log_AutoTuneDetails ( )
{
2024-04-20 11:36:24 -03:00
if ( tune_type = = SP_UP | | tune_type = = TUNE_CHECK ) {
2021-01-18 00:16:08 -04:00
Log_Write_AutoTuneDetails ( command_out , 0.0f , 0.0f , filt_target_rate , rotation_rate ) ;
2020-10-19 23:51:05 -03:00
} else {
2021-01-18 00:16:08 -04:00
Log_Write_AutoTuneDetails ( command_out , filt_target_rate , rotation_rate , 0.0f , 0.0f ) ;
2020-10-19 23:51:05 -03:00
}
}
2021-12-06 00:27:42 -04:00
// log autotune frequency response data
2021-01-18 00:16:08 -04:00
void AC_AutoTune_Heli : : Log_AutoTuneSweep ( )
2020-10-19 23:51:05 -03:00
{
2024-04-07 00:55:51 -03:00
Log_Write_AutoTuneSweep ( curr_test_mtr . freq , curr_test_mtr . gain , curr_test_mtr . phase , curr_test_tgt . freq , curr_test_tgt . gain , curr_test_tgt . phase ) ;
2020-10-19 23:51:05 -03:00
}
2021-01-18 00:16:08 -04:00
// @LoggerMessage: ATNH
// @Description: Heli AutoTune
// @Vehicles: Copter
2020-10-19 23:51:05 -03:00
// @Field: TimeUS: Time since system startup
// @Field: Axis: which axis is currently being tuned
// @Field: TuneStep: step in autotune process
// @Field: Freq: target dwell frequency
// @Field: Gain: measured gain of dwell
// @Field: Phase: measured phase of dwell
// @Field: RFF: new rate gain FF term
// @Field: RP: new rate gain P term
// @Field: RD: new rate gain D term
// @Field: SP: new angle P term
2021-01-18 00:16:08 -04:00
// @Field: ACC: new max accel term
2020-10-19 23:51:05 -03:00
2021-12-06 00:27:42 -04:00
// Write an Autotune summary data packet
2024-07-25 01:07:26 -03:00
void AC_AutoTune_Heli : : Log_Write_AutoTune ( AxisType _axis , uint8_t tune_step , float dwell_freq , float meas_gain , float meas_phase , float new_gain_rff , float new_gain_rp , float new_gain_rd , float new_gain_sp , float max_accel )
2020-10-19 23:51:05 -03:00
{
AP : : logger ( ) . Write (
2021-01-18 00:16:08 -04:00
" ATNH " ,
" TimeUS,Axis,TuneStep,Freq,Gain,Phase,RFF,RP,RD,SP,ACC " ,
" s--E-d----- " ,
" F--000----- " ,
" QBBffffffff " ,
2020-10-19 23:51:05 -03:00
AP_HAL : : micros64 ( ) ,
2024-07-25 01:07:26 -03:00
( uint8_t ) axis ,
2020-10-19 23:51:05 -03:00
tune_step ,
dwell_freq ,
meas_gain ,
meas_phase ,
new_gain_rff ,
new_gain_rp ,
new_gain_rd ,
2021-01-18 00:16:08 -04:00
new_gain_sp ,
max_accel ) ;
2020-10-19 23:51:05 -03:00
}
2021-12-06 00:27:42 -04:00
// Write an Autotune detailed data packet
2021-01-18 00:16:08 -04:00
void AC_AutoTune_Heli : : Log_Write_AutoTuneDetails ( float motor_cmd , float tgt_rate_rads , float rate_rads , float tgt_ang_rad , float ang_rad )
2020-10-19 23:51:05 -03:00
{
2021-01-18 00:16:08 -04:00
// @LoggerMessage: ATDH
// @Description: Heli AutoTune data packet
// @Vehicles: Copter
2020-10-19 23:51:05 -03:00
// @Field: TimeUS: Time since system startup
// @Field: Cmd: current motor command
// @Field: TRate: current target angular rate
// @Field: Rate: current angular rate
2021-01-18 00:16:08 -04:00
// @Field: TAng: current target angle
// @Field: Ang: current angle
2020-10-19 23:51:05 -03:00
AP : : logger ( ) . WriteStreaming (
2021-01-18 00:16:08 -04:00
" ATDH " ,
" TimeUS,Cmd,TRate,Rate,TAng,Ang " ,
" s-kkdd " ,
" F00000 " ,
" Qfffff " ,
AP_HAL : : micros64 ( ) ,
motor_cmd ,
tgt_rate_rads * 57.3 ,
rate_rads * 57.3f ,
tgt_ang_rad * 57.3 ,
ang_rad * 57.3f ) ;
}
2021-12-06 00:27:42 -04:00
// Write an Autotune frequency response data packet
2024-04-07 00:55:51 -03:00
void AC_AutoTune_Heli : : Log_Write_AutoTuneSweep ( float freq_mtr , float gain_mtr , float phase_mtr , float freq_tgt , float gain_tgt , float phase_tgt )
2021-01-18 00:16:08 -04:00
{
// @LoggerMessage: ATSH
// @Description: Heli AutoTune Sweep packet
// @Vehicles: Copter
// @Field: TimeUS: Time since system startup
2024-04-07 00:55:51 -03:00
// @Field: freq_m: current frequency for motor input to rate
// @Field: gain_m: current response gain for motor input to rate
// @Field: phase_m: current response phase for motor input to rate
// @Field: freq_t: current frequency for target rate to rate
// @Field: gain_t: current response gain for target rate to rate
// @Field: phase_t: current response phase for target rate to rate
2021-01-18 00:16:08 -04:00
AP : : logger ( ) . WriteStreaming (
" ATSH " ,
2024-04-07 00:55:51 -03:00
" TimeUS,freq_m,gain_m,phase_m,freq_t,gain_t,phase_t " ,
" sE-dE-d " ,
" F000000 " ,
" Qffffff " ,
2020-10-19 23:51:05 -03:00
AP_HAL : : micros64 ( ) ,
2024-04-07 00:55:51 -03:00
freq_mtr ,
gain_mtr ,
phase_mtr ,
freq_tgt ,
gain_tgt ,
phase_tgt ) ;
2020-10-19 23:51:05 -03:00
}
2023-07-13 21:58:04 -03:00
# endif // HAL_LOGGING_ENABLED
2020-10-19 23:51:05 -03:00
2023-10-11 04:41:49 -03:00
// reset the test variables for each vehicle
2021-12-22 23:37:31 -04:00
void AC_AutoTune_Heli : : reset_vehicle_test_variables ( )
{
// reset dwell test variables if sweep was interrupted in order to restart sweep
if ( ! is_equal ( start_freq , stop_freq ) ) {
start_freq = 0.0f ;
stop_freq = 0.0f ;
2024-04-12 23:26:56 -03:00
next_test_freq = 0.0f ;
sweep_complete = false ;
2021-12-22 23:37:31 -04:00
}
}
2022-01-07 14:37:02 -04:00
// reset the update gain variables for heli
void AC_AutoTune_Heli : : reset_update_gain_variables ( )
{
2024-04-12 00:21:16 -03:00
// reset max gain variables
2022-01-16 19:13:59 -04:00
reset_maxgains_update_gain_variables ( ) ;
// reset rd_up variables
rd_prev_gain = 0.0f ;
// reset sp_up variables
phase_max = 0.0f ;
2024-04-12 23:26:56 -03:00
freq_max = 0.0f ;
2022-01-16 19:13:59 -04:00
sp_prev_gain = 0.0f ;
2024-04-12 23:26:56 -03:00
found_max_gain_freq = false ;
found_peak = false ;
2022-01-16 19:13:59 -04:00
}
// reset the max_gains update gain variables
void AC_AutoTune_Heli : : reset_maxgains_update_gain_variables ( )
{
2022-02-08 19:46:44 -04:00
max_rate_p = { } ;
max_rate_d = { } ;
2024-04-13 02:04:33 -03:00
2022-01-07 14:37:02 -04:00
found_max_p = false ;
found_max_d = false ;
}
2022-01-16 19:13:59 -04:00
// reset the max_gains update gain variables
void AC_AutoTune_Heli : : reset_sweep_variables ( )
{
2024-04-07 00:55:51 -03:00
sweep_tgt . ph180 = { } ;
sweep_tgt . ph270 = { } ;
sweep_tgt . maxgain = { } ;
sweep_tgt . progress = 0 ;
sweep_mtr . ph180 = { } ;
sweep_mtr . ph270 = { } ;
sweep_mtr . maxgain = { } ;
sweep_mtr . progress = 0 ;
2022-02-08 19:31:51 -04:00
2022-01-16 19:13:59 -04:00
}
2021-12-06 00:27:42 -04:00
// set the tuning test sequence
2021-01-18 00:16:08 -04:00
void AC_AutoTune_Heli : : set_tune_sequence ( )
{
uint8_t seq_cnt = 0 ;
if ( seq_bitmask & AUTOTUNE_SEQ_BITMASK_VFF ) {
tune_seq [ seq_cnt ] = RFF_UP ;
seq_cnt + + ;
}
if ( seq_bitmask & AUTOTUNE_SEQ_BITMASK_RATE_D ) {
tune_seq [ seq_cnt ] = MAX_GAINS ;
seq_cnt + + ;
tune_seq [ seq_cnt ] = RD_UP ;
seq_cnt + + ;
tune_seq [ seq_cnt ] = RP_UP ;
seq_cnt + + ;
}
if ( seq_bitmask & AUTOTUNE_SEQ_BITMASK_ANGLE_P ) {
tune_seq [ seq_cnt ] = SP_UP ;
seq_cnt + + ;
}
if ( seq_bitmask & AUTOTUNE_SEQ_BITMASK_MAX_GAIN & & ! ( seq_bitmask & AUTOTUNE_SEQ_BITMASK_RATE_D ) ) {
tune_seq [ seq_cnt ] = MAX_GAINS ;
seq_cnt + + ;
}
2022-05-08 00:28:50 -03:00
if ( seq_bitmask & AUTOTUNE_SEQ_BITMASK_TUNE_CHECK ) {
tune_seq [ seq_cnt ] = TUNE_CHECK ;
seq_cnt + + ;
}
2021-01-18 00:16:08 -04:00
tune_seq [ seq_cnt ] = TUNE_COMPLETE ;
}
2021-12-28 00:23:19 -04:00
// get_testing_step_timeout_ms accessor
uint32_t AC_AutoTune_Heli : : get_testing_step_timeout_ms ( ) const
{
return AUTOTUNE_TESTING_STEP_TIMEOUT_MS ;
}
2022-01-21 14:44:36 -04:00
// exceeded_freq_range - ensures tuning remains inside frequency range
bool AC_AutoTune_Heli : : exceeded_freq_range ( float frequency )
{
bool ret = false ;
if ( frequency < min_sweep_freq | | frequency > max_sweep_freq ) {
ret = true ;
}
return ret ;
}
2023-12-07 20:03:15 -04:00
# endif // AC_AUTOTUNE_ENABLED