AP_AHRS: added AHRS_OPTIONS parameter

the first option is to disable DCM fallback on fixed wing. This is
suitable in environments with a high likelyhood of GPS interference
This commit is contained in:
Paul Riseborough 2023-03-22 08:40:48 +11:00 committed by Andrew Tridgell
parent 5fce4f5f6d
commit 7601a02e98
2 changed files with 58 additions and 25 deletions

View File

@ -178,6 +178,13 @@ const AP_Param::GroupInfo AP_AHRS::var_info[] = {
// index 17 // index 17
// @Param: OPTIONS
// @DisplayName: Optional AHRS behaviour
// @Description: This controls optional AHRS behaviour. Setting DisableDCMFallbackFW will change the AHRS behaviour for fixed wing aircraft in fly-forward flight to not fall back to DCM when the EKF stops navigating. Setting DisableDCMFallbackVTOL will change the AHRS behaviour for fixed wing aircraft in non fly-forward (VTOL) flight to not fall back to DCM when the EKF stops navigating.
// @Bitmask: 0:DisableDCMFallbackFW, 1:DisableDCMFallbackVTOL
// @User: Advanced
AP_GROUPINFO("OPTIONS", 18, AP_AHRS, _options, 0),
AP_GROUPEND AP_GROUPEND
}; };
@ -1890,21 +1897,11 @@ AP_AHRS::EKFType AP_AHRS::_active_EKF_type(void) const
} }
#if AP_AHRS_DCM_ENABLED #if AP_AHRS_DCM_ENABLED
/* // Handle fallback for fixed wing planes (including VTOL's) and ground vehicles.
fixed wing and rover will fall back to DCM if the EKF doesn't
have GPS. This is the safest option as DCM is very robust. Note
that we also check the filter status when fly_forward is false
and we are disarmed. This is to ensure that the arming checks do
wait for good GPS position on fixed wing and rover
*/
if (ret != EKFType::DCM && if (ret != EKFType::DCM &&
(_vehicle_class == VehicleClass::FIXED_WING || (_vehicle_class == VehicleClass::FIXED_WING ||
_vehicle_class == VehicleClass::GROUND)) { _vehicle_class == VehicleClass::GROUND)) {
if (!dcm.yaw_source_available() && !fly_forward) {
// if we don't have a DCM yaw source available and we are
// in a non-fly-forward mode then we are best off using the EKF
return ret;
}
bool should_use_gps = true; bool should_use_gps = true;
nav_filter_status filt_state; nav_filter_status filt_state;
#if HAL_NAVEKF2_AVAILABLE #if HAL_NAVEKF2_AVAILABLE
@ -1930,26 +1927,53 @@ AP_AHRS::EKFType AP_AHRS::_active_EKF_type(void) const
should_use_gps = true; should_use_gps = true;
} }
#endif #endif
// Handle fallback for the case where the DCM or EKF is unable to provide attitude or height data.
const bool can_use_dcm = dcm.yaw_source_available() || fly_forward;
const bool can_use_ekf = filt_state.flags.attitude && filt_state.flags.vert_vel && filt_state.flags.vert_pos;
if (!can_use_dcm && can_use_ekf) {
// no choice - continue to use EKF
return ret;
} else if (!can_use_ekf) {
// No choice - we have to use DCM
return EKFType::DCM;
}
const bool disable_dcm_fallback = fly_forward?
option_set(Options::DISABLE_DCM_FALLBACK_FW) : option_set(Options::DISABLE_DCM_FALLBACK_VTOL);
if (disable_dcm_fallback) {
// don't fallback
return ret;
}
// Handle loss of global position when we still have a GPS fix
if (hal.util->get_soft_armed() && if (hal.util->get_soft_armed() &&
(!filt_state.flags.using_gps ||
!filt_state.flags.horiz_pos_abs) &&
should_use_gps && should_use_gps &&
AP::gps().status() >= AP_GPS::GPS_OK_FIX_3D) { AP::gps().status() >= AP_GPS::GPS_OK_FIX_3D &&
// if the EKF is not fusing GPS or doesn't have a 2D fix (!filt_state.flags.using_gps || !filt_state.flags.horiz_pos_abs)) {
// and we have a 3D lock, then plane and rover would /*
// prefer to use the GPS position from DCM. This is a If the EKF is not fusing GPS or doesn't have a 2D fix and we have a 3D GPS lock,
// safety net while some issues with the EKF get sorted then plane and rover would prefer to use the GPS position from DCM unless the
// out fallback has been inhibited by the user.
Note: The aircraft could be dead reckoning with acceptable accuracy and rejecting a bad GPS
Note: This is a last resort fallback and makes the navigation highly vulnerable to GPS noise.
Note: When operating in a VTOL flight mode that actively controls height such as QHOVER,
the EKF gives better vertical velocity and position estimates and height control characteristics.
*/
return EKFType::DCM; return EKFType::DCM;
} }
// Handle complete loss of navigation
if (hal.util->get_soft_armed() && filt_state.flags.const_pos_mode) { if (hal.util->get_soft_armed() && filt_state.flags.const_pos_mode) {
/*
Provided the EKF has been configured to use GPS, ie should_use_gps is true, then the
key difference to the case handled above is only the absence of a GPS fix which means
that DCM will not be able to navigate either so we are primarily concerned with
providing an attitude, vertical position and vertical velocity estimate.
*/
return EKFType::DCM; return EKFType::DCM;
} }
if (!filt_state.flags.attitude ||
!filt_state.flags.vert_vel ||
!filt_state.flags.vert_pos) {
return EKFType::DCM;
}
if (!filt_state.flags.horiz_vel || if (!filt_state.flags.horiz_vel ||
(!filt_state.flags.horiz_pos_abs && !filt_state.flags.horiz_pos_rel)) { (!filt_state.flags.horiz_pos_abs && !filt_state.flags.horiz_pos_rel)) {
if ((!AP::compass().use_for_yaw()) && if ((!AP::compass().use_for_yaw()) &&

View File

@ -991,6 +991,15 @@ private:
struct AP_AHRS_Backend::Estimates external_estimates; struct AP_AHRS_Backend::Estimates external_estimates;
#endif #endif
enum class Options : uint16_t {
DISABLE_DCM_FALLBACK_FW=(1U<<0),
DISABLE_DCM_FALLBACK_VTOL=(1U<<1),
};
AP_Int16 _options;
bool option_set(Options option) const {
return (_options & uint16_t(option)) != 0;
}
}; };
namespace AP { namespace AP {