AP_Motors: modifications based on suggested changes

This commit is contained in:
Bill Geyer 2021-11-11 12:09:26 -05:00 committed by Bill Geyer
parent a8d74c6414
commit afaff18bd0
2 changed files with 90 additions and 66 deletions

View File

@ -125,7 +125,16 @@ const AP_Param::GroupInfo AP_MotorsHeli_RSC::var_info[] = {
// @User: Standard // @User: Standard
AP_GROUPINFO("THRCRV_100", 12, AP_MotorsHeli_RSC, _thrcrv[4], AP_MOTORS_HELI_RSC_THRCRV_100_DEFAULT), AP_GROUPINFO("THRCRV_100", 12, AP_MotorsHeli_RSC, _thrcrv[4], AP_MOTORS_HELI_RSC_THRCRV_100_DEFAULT),
// Indices 13 thru 17 have been re-assigned and should not be used in the future // Indices 13 thru 16 have been re-assigned and should not be used in the future
// @Param: GOV_RANGE
// @DisplayName: Governor Operational Range
// @Description: RPM range +/- governor rpm reference setting where governor is operational. If speed sensor fails or rpm falls outside of this range, the governor will disengage and return to throttle curve. Recommended range is 100
// @Range: 50 200
// @Units: RPM
// @Increment: 10
// @User: Standard
AP_GROUPINFO("GOV_RANGE", 17, AP_MotorsHeli_RSC, _governor_range, AP_MOTORS_HELI_RSC_GOVERNOR_RANGE_DEFAULT),
#if CONFIG_HAL_BOARD == HAL_BOARD_SITL #if CONFIG_HAL_BOARD == HAL_BOARD_SITL
// @Param: AROT_PCT // @Param: AROT_PCT
@ -138,39 +147,39 @@ const AP_Param::GroupInfo AP_MotorsHeli_RSC::var_info[] = {
AP_GROUPINFO("AROT_PCT", 18, AP_MotorsHeli_RSC, _ext_gov_arot_pct, 0), AP_GROUPINFO("AROT_PCT", 18, AP_MotorsHeli_RSC, _ext_gov_arot_pct, 0),
#endif #endif
// @Param: GOV_CLDWN // @Param: CLDWN_TIME
// @DisplayName: AutoThrottle Cooldown Time // @DisplayName: Cooldown Time
// @Description: Will provide a fast idle for engine cooldown by raising the Ground Idle speed setting by 50% for the number of seconds the timer is set for. A setting of zero disables the fast idle. This feature will only apply after the governor and autothrottle have been engaged (throttle switch on and rotor rpm at least 100% of normal speed). At any time during fast idle, disarming will shut the engine down. // @Description: Will provide a fast idle for engine cooldown by raising the Ground Idle speed setting by 50% for the number of seconds the timer is set for. A setting of zero disables the fast idle. This feature will only apply after the runup complete has been declared. This will not extend the time before ground idle is declared, which triggers engine shutdown for autonomous landings.
// @Range: 0 120 // @Range: 0 120
// @Units: s // @Units: s
// @Increment: 1 // @Increment: 1
// @User: Standard // @User: Standard
AP_GROUPINFO("GOV_CLDWN", 19, AP_MotorsHeli_RSC, _autothrottle_cooldown, 0), AP_GROUPINFO("CLDWN_TIME", 19, AP_MotorsHeli_RSC, _cooldown_time, 0),
// @Param: GOV_COMP // @Param: GOV_COMP
// @DisplayName: Governor Torque Compensator // @DisplayName: Governor Torque Compensator
// @Description: Adjusts the autothrottle governor torque compensator that determines how fast the governor will adjust the base torque reference to compensate for changes in density altitude. If Rrpm is low or high by more than 2-5 rpm, increase this setting by 1% at a time until the governor speed matches your Rrpm setting. Setting the compensator too high can result in surging and throttle "hunting". Do not make large adjustments at one time // @Description: Adjusts the autothrottle governor torque compensator that determines how fast the governor will adjust the base torque reference to compensate for changes in density altitude. If RPM is low or high by more than 2-5 RPM, increase this setting by 1% at a time until the governor speed matches your RPM setting. Setting the compensator too high can result in surging and throttle "hunting". Do not make large adjustments at one time
// @Range: 0 70 // @Range: 0 70
// @Units: % // @Units: %
// @Increment: 1 // @Increment: 0.1
// @User: Standard // @User: Standard
AP_GROUPINFO("GOV_COMP", 20, AP_MotorsHeli_RSC, _governor_compensator, 25), AP_GROUPINFO("GOV_COMP", 20, AP_MotorsHeli_RSC, _governor_compensator, 25),
// @Param: GOV_DROOP // @Param: GOV_DROOP
// @DisplayName: Governor Droop Compensator // @DisplayName: Governor Droop Compensator
// @Description: AutoThrottle governor droop response under load, normal settings of 0-50%. Higher value is quicker response to large speed changes due to load but may cause surging. Adjust this to be as aggressive as possible without getting surging or Rrpm over-run when the governor responds to large load changes on the rotor system // @Description: AutoThrottle governor droop response under load, normal settings of 0-50%. Higher value is quicker response to large speed changes due to load but may cause surging. Adjust this to be as aggressive as possible without getting surging or RPM over-run when the governor responds to large load changes on the rotor system
// @Range: 0 100 // @Range: 0 100
// @Units: % // @Units: %
// @Increment: 1 // @Increment: 0.1
// @User: Standard // @User: Standard
AP_GROUPINFO("GOV_DROOP", 21, AP_MotorsHeli_RSC, _governor_droop_response, 25), AP_GROUPINFO("GOV_DROOP", 21, AP_MotorsHeli_RSC, _governor_droop_response, 25),
// @Param: GOV_FF // @Param: GOV_FF
// @DisplayName: Governor Feedforward // @DisplayName: Governor Feedforward
// @Description: Feedforward governor gain to throttle response during sudden loading/unloading of the rotor system. If Rrpm drops excessively during full collective climb with the droop response set correctly, increase the governor feedforward. // @Description: Feedforward governor gain to throttle response during sudden loading/unloading of the rotor system. If RPM drops excessively during full collective climb with the droop response set correctly, increase the governor feedforward.
// @Range: 0 100 // @Range: 0 100
// @Units: % // @Units: %
// @Increment: 1 // @Increment: 0.1
// @User: Standard // @User: Standard
AP_GROUPINFO("GOV_FF", 22, AP_MotorsHeli_RSC, _governor_ff, 50), AP_GROUPINFO("GOV_FF", 22, AP_MotorsHeli_RSC, _governor_ff, 50),
@ -185,7 +194,7 @@ const AP_Param::GroupInfo AP_MotorsHeli_RSC::var_info[] = {
// @Param: GOV_TORQUE // @Param: GOV_TORQUE
// @DisplayName: Governor Torque Limiter // @DisplayName: Governor Torque Limiter
// @Description: Adjusts the engine's percentage of torque rise on autothrottle during ramp-up to governor speed. The torque rise will determine how fast the rotor speed will ramp up when rotor speed reaches 50% of the rotor rpm setting. The sequence of events engaging the governor is as follows: Throttle ramp time will engage the clutch and start the main rotor turning. The collective should be at feather pitch and the throttle curve set to provide at least 50% of normal Rrpm at feather pitch. The autothrottle torque limiter will automatically activate and start accelerating the main rotor. If the autothrottle consistently fails to accelerate the main rotor during ramp-in due to engine tune or other factors, then increase the torque limiter setting. NOTE: throttle ramp time and throttle curve should be tuned using MODE Throttle Curve before using MODE AutoThrottle // @Description: Adjusts the engine's percentage of torque rise on autothrottle during ramp-up to governor speed. The torque rise will determine how fast the rotor speed will ramp up when rotor speed reaches 50% of the rotor RPM setting. The sequence of events engaging the governor is as follows: Throttle ramp time will engage the clutch and start the main rotor turning. The collective should be at flat pitch and the throttle curve set to provide at least 50% of normal RPM at flat pitch. The autothrottle torque limiter will automatically activate and start accelerating the main rotor. If the autothrottle consistently fails to accelerate the main rotor during ramp-in due to engine tune or other factors, then increase the torque limiter setting. NOTE: throttle ramp time and throttle curve should be tuned using RSC_MODE Throttle Curve before using RSC_MODE AutoThrottle
// @Range: 10 60 // @Range: 10 60
// @Units: % // @Units: %
// @Increment: 1 // @Increment: 1
@ -223,9 +232,15 @@ void AP_MotorsHeli_RSC::set_throttle_curve()
// output - update value to send to ESC/Servo // output - update value to send to ESC/Servo
void AP_MotorsHeli_RSC::output(RotorControlState state) void AP_MotorsHeli_RSC::output(RotorControlState state)
{ {
// _rotor_rpm available to the RSC output // _rotor_RPM available to the RSC output
const AP_RPM *rpm = AP_RPM::get_singleton(); const AP_RPM *rpm = AP_RPM::get_singleton();
if (rpm != nullptr) {
if (!rpm->get_rpm(0, _rotor_rpm)) { if (!rpm->get_rpm(0, _rotor_rpm)) {
// No valid RPM data
_rotor_rpm = -1;
}
} else {
// No RPM because pointer is null
_rotor_rpm = -1; _rotor_rpm = -1;
} }
@ -261,21 +276,21 @@ void AP_MotorsHeli_RSC::output(RotorControlState state)
// set rotor control speed to engine idle and ensure governor is reset, if used // set rotor control speed to engine idle and ensure governor is reset, if used
governor_reset(); governor_reset();
_autothrottle = false;
_governor_fault = false; _governor_fault = false;
if (!_autothrottle) {
if (_in_autorotation) { if (_in_autorotation) {
// if in autorotation and using an external governor, set the output to tell the governor to use bailout ramp // if in autorotation and using an external governor, set the output to tell the governor to use bailout ramp
_control_output = constrain_float( _rsc_arot_bailout_pct/100.0f , 0.0f, 0.4f); _control_output = constrain_float( _rsc_arot_bailout_pct/100.0f , 0.0f, 0.4f);
} else { } else {
// set rotor control speed to idle speed parameter, this happens instantly and ignores ramping // set rotor control speed to idle speed parameter, this happens instantly and ignores ramping
_control_output = get_idle_output(); if (_cooldown_time > 0) {
_control_output = get_idle_output() * 1.5f;
_fast_idle_timer += dt;
if (_fast_idle_timer > (float)_cooldown_time) {
_fast_idle_timer = 0.0f;
} }
} else { } else {
_control_output = get_idle_output() * 1.5f; _control_output = get_idle_output();
_autothrottle_fast_idle += dt;
if (_autothrottle_fast_idle > _autothrottle_cooldown) {
_autothrottle = false;
_autothrottle_fast_idle = 0.0f;
} }
} }
break; break;
@ -427,64 +442,71 @@ float AP_MotorsHeli_RSC::calculate_throttlecurve(float collective_in)
void AP_MotorsHeli_RSC::autothrottle_run() void AP_MotorsHeli_RSC::autothrottle_run()
{ {
float throttlecurve = calculate_throttlecurve(_collective_in); float throttlecurve = calculate_throttlecurve(_collective_in);
float const torque_ref_error_rpm = 2.0f;
// if the desired governor RPM is zero, use the throttle curve only and exit
if (_governor_rpm == 0) {
_control_output = get_idle_output() + (_rotor_ramp_output * (throttlecurve - get_idle_output()));
return;
}
// autothrottle main power loop with governor // autothrottle main power loop with governor
if (_governor_engage && !_governor_fault) { if (_governor_engage && !_governor_fault) {
// calculate droop - difference between actual and desired speed // calculate droop - difference between actual and desired speed
float governor_droop = (_governor_rpm - _rotor_rpm) * get_governor_droop_response(); float governor_droop = ((float)_governor_rpm - _rotor_rpm) * _governor_droop_response * 0.0001f;
_governor_output = governor_droop + ((throttlecurve - _governor_torque_reference) * get_governor_ff()); _governor_output = governor_droop + ((throttlecurve - _governor_torque_reference) * _governor_ff * 0.01);
if (_rotor_rpm < (_governor_rpm - 2.0f)) { if (_rotor_rpm < ((float)_governor_rpm - torque_ref_error_rpm)) {
_governor_torque_reference += get_governor_compensator(); // torque compensator _governor_torque_reference += get_governor_compensator(); // torque compensator
} else if (_rotor_rpm > (_governor_rpm + 2.0f)) { } else if (_rotor_rpm > ((float)_governor_rpm + torque_ref_error_rpm)) {
_governor_torque_reference -= get_governor_compensator(); _governor_torque_reference -= get_governor_compensator();
} }
// throttle output uses droop + torque compensation to maintain proper rotor speed // throttle output uses droop + torque compensation to maintain proper rotor speed
_control_output = constrain_float((_governor_torque_reference + _governor_output), (get_idle_output() * 1.5f), 1.0f); _control_output = constrain_float((_governor_torque_reference + _governor_output), (get_idle_output() * 1.5f), 1.0f);
// governor and speed sensor fault detection - must maintain Rrpm -3/+2% // governor and speed sensor fault detection - must maintain RPM within governor range
// speed fault detector will allow a fault to persist for 200 contiguous governor updates // speed fault detector will allow a fault to persist for 200 contiguous governor updates
// this is failsafe for bad speed sensor or severely mis-adjusted governor // this is failsafe for bad speed sensor or severely mis-adjusted governor
if ((_rotor_rpm <= (_governor_rpm * 0.97f)) || (_rotor_rpm >= (_governor_rpm * 1.02f))) { if ((_rotor_rpm <= (_governor_rpm - _governor_range)) || (_rotor_rpm >= (_governor_rpm + _governor_range))) {
_governor_fault_count += 1.0f; _governor_fault_count++;
if (_governor_fault_count > 200.0f) { if (_governor_fault_count > 200) {
governor_reset(); governor_reset();
_governor_fault = true; _governor_fault = true;
if (_rotor_rpm >= (_governor_rpm * 1.02f)) { if (_rotor_rpm >= (_governor_rpm + _governor_range)) {
gcs().send_text(MAV_SEVERITY_WARNING, "Governor Fault: Rotor Overspeed"); gcs().send_text(MAV_SEVERITY_WARNING, "Governor Fault: Rotor Overspeed");
} else { } else {
gcs().send_text(MAV_SEVERITY_WARNING, "Governor Fault: Rotor Underspeed"); gcs().send_text(MAV_SEVERITY_WARNING, "Governor Fault: Rotor Underspeed");
} }
} }
} else { } else {
_governor_fault_count = 0.0f; // reset fault count if the fault doesn't persist _governor_fault_count = 0; // reset fault count if the fault doesn't persist
} }
} else if (!_governor_engage && !_governor_fault) { } else if (!_governor_engage && !_governor_fault) {
// torque rise limiter accelerates rotor to the reference speed
// this limits the max torque rise the governor could call for from the main power loop
if (_rotor_rpm >= (_governor_rpm * 0.5f)) {
float torque_limit = (get_governor_torque() * get_governor_torque());
_governor_output = (_rotor_rpm / _governor_rpm) * torque_limit;
_control_output = constrain_float(get_idle_output() + (_rotor_ramp_output * (throttlecurve + _governor_output - get_idle_output())), 0.0f, 1.0f);
_governor_torque_reference = _control_output; // increment torque setting to be passed to main power loop
if (_rotor_rpm >= (_governor_rpm - 2.0f)) {
_governor_engage = true;
_autothrottle = true;
gcs().send_text(MAV_SEVERITY_CRITICAL, "Governor Engaged");
}
} else {
// temporary use of throttle curve and ramp timer to accelerate rotor to governor min torque rise speed
_control_output = get_idle_output() + (_rotor_ramp_output * (throttlecurve - get_idle_output()));
}
} else {
// failsafe - if governor has faulted use throttle curve
_control_output = get_idle_output() + (_rotor_ramp_output * (throttlecurve - get_idle_output()));
}
// if governor is not engaged and rotor is overspeeding by more than 2% due to misconfigured // if governor is not engaged and rotor is overspeeding by more than 2% due to misconfigured
// throttle curve or stuck throttle, set a fault and governor will not operate // throttle curve or stuck throttle, set a fault and governor will not operate
if (!_governor_engage && !_governor_fault && (_rotor_rpm > (_governor_rpm * 1.02f))) { if (_rotor_rpm > (_governor_rpm + _governor_range)) {
_governor_fault = true; _governor_fault = true;
governor_reset(); governor_reset();
gcs().send_text(MAV_SEVERITY_WARNING, "Governor Fault: Rotor Overspeed"); gcs().send_text(MAV_SEVERITY_WARNING, "Governor Fault: Rotor Overspeed");
_governor_output = 0.0f;
// torque rise limiter accelerates rotor to the reference speed
// this limits the max torque rise the governor could call for from the main power loop
} else if (_rotor_rpm >= (_governor_rpm * 0.5f)) {
float torque_limit = (get_governor_torque() * get_governor_torque());
_governor_output = (_rotor_rpm / (float)_governor_rpm) * torque_limit;
if (_rotor_rpm >= ((float)_governor_rpm - torque_ref_error_rpm)) {
_governor_engage = true;
_autothrottle = true;
gcs().send_text(MAV_SEVERITY_NOTICE, "Governor Engaged");
}
} else {
// temporary use of throttle curve and ramp timer to accelerate rotor to governor min torque rise speed
_governor_output = 0.0f;
}
_control_output = constrain_float(get_idle_output() + (_rotor_ramp_output * (throttlecurve + _governor_output - get_idle_output())), 0.0f, 1.0f);
_governor_torque_reference = _control_output; // increment torque setting to be passed to main power loop
} else {
// failsafe - if governor has faulted use throttle curve
_control_output = get_idle_output() + (_rotor_ramp_output * (throttlecurve - get_idle_output()));
} }
} }
@ -494,4 +516,5 @@ void AP_MotorsHeli_RSC::governor_reset()
_governor_output = 0.0f; _governor_output = 0.0f;
_governor_torque_reference = 0.0f; _governor_torque_reference = 0.0f;
_governor_engage = false; _governor_engage = false;
_governor_fault_count = 0; // reset fault count when governor reset
} }

View File

@ -27,6 +27,9 @@
#define AP_MOTORS_HELI_RSC_THRCRV_75_DEFAULT 50 #define AP_MOTORS_HELI_RSC_THRCRV_75_DEFAULT 50
#define AP_MOTORS_HELI_RSC_THRCRV_100_DEFAULT 100 #define AP_MOTORS_HELI_RSC_THRCRV_100_DEFAULT 100
// RSC governor defaults
#define AP_MOTORS_HELI_RSC_GOVERNOR_RANGE_DEFAULT 100
// rotor controller states // rotor controller states
enum RotorControlState { enum RotorControlState {
ROTOR_CONTROL_STOP = 0, ROTOR_CONTROL_STOP = 0,
@ -150,6 +153,9 @@ private:
bool _in_autorotation; // true if vehicle is currently in an autorotation bool _in_autorotation; // true if vehicle is currently in an autorotation
int16_t _rsc_arot_bailout_pct; // the throttle percentage sent to the external governor to signal that autorotation bailout ramp should be used int16_t _rsc_arot_bailout_pct; // the throttle percentage sent to the external governor to signal that autorotation bailout ramp should be used
bool _spooldown_complete; // flag for determining if spooldown is complete bool _spooldown_complete; // flag for determining if spooldown is complete
float _fast_idle_timer; // cooldown timer variable
uint8_t _governor_fault_count; // variable for tracking governor speed sensor faults
float _governor_torque_reference; // governor reference for load calculations
// update_rotor_ramp - slews rotor output scalar between 0 and 1, outputs float scalar to _rotor_ramp_output // update_rotor_ramp - slews rotor output scalar between 0 and 1, outputs float scalar to _rotor_ramp_output
void update_rotor_ramp(float rotor_ramp_input, float dt); void update_rotor_ramp(float rotor_ramp_input, float dt);
@ -166,22 +172,17 @@ private:
// parameters // parameters
AP_Int16 _power_slewrate; // throttle slew rate (percentage per second) AP_Int16 _power_slewrate; // throttle slew rate (percentage per second)
AP_Int16 _thrcrv[5]; // throttle value sent to throttle servo at 0, 25, 50, 75 and 100 percent collective AP_Int16 _thrcrv[5]; // throttle value sent to throttle servo at 0, 25, 50, 75 and 100 percent collective
AP_Float _governor_rpm; // governor reference for speed calculations AP_Int16 _governor_rpm; // governor reference for speed calculations
AP_Float _governor_torque; // governor torque rise setting AP_Float _governor_torque; // governor torque rise setting
AP_Float _governor_torque_reference; // governor reference for load calculations
AP_Float _governor_compensator; // governor torque compensator variable AP_Float _governor_compensator; // governor torque compensator variable
AP_Float _governor_droop_response; // governor response to droop under load AP_Float _governor_droop_response; // governor response to droop under load
AP_Float _governor_ff; // governor feedforward variable AP_Float _governor_ff; // governor feedforward variable
AP_Float _governor_fault_count; // variable for tracking governor speed sensor faults AP_Float _governor_range; // RPM range +/- governor rpm reference setting where governor is operational
AP_Float _autothrottle_fast_idle; // autothrottle variable for cooldown AP_Int16 _cooldown_time; // cooldown time to provide a fast idle
AP_Float _autothrottle_cooldown; // autothrottle cooldown timer to provide a fast idle
// parameter accessors to allow conversions // parameter accessors to allow conversions
float get_critical_speed() const { return _critical_speed * 0.01; } float get_critical_speed() const { return _critical_speed * 0.01; }
float get_idle_output() { return _idle_output * 0.01; } float get_idle_output() const { return _idle_output * 0.01; }
float get_governor_torque() { return _governor_torque * 0.01; } float get_governor_torque() const { return _governor_torque * 0.01; }
float get_governor_compensator() { return _governor_compensator * 0.000001; } float get_governor_compensator() const { return _governor_compensator * 0.000001; }
float get_governor_droop_response() { return _governor_droop_response * 0.0001f; }
float get_governor_ff() { return _governor_ff * 0.01; }
}; };