AP_NavEKF3: Fix bug preventing height reset if badIMUdata

This commit is contained in:
Paul Riseborough 2021-06-17 10:06:42 +10:00 committed by Randy Mackay
parent e4eb8339d0
commit 5d00b7d042

View File

@ -590,9 +590,9 @@ void NavEKF3_core::FuseVelPosNED()
{ {
// health is set bad until test passed // health is set bad until test passed
bool posHealth = false; // boolean true if position measurements have passed innovation consistency check bool posHealth = false; // boolean true if position measurements have passed innovation consistency check
bool hgtHealth = false; // boolean true if height measurements have passed innovation consistency check
bool velCheckPassed = false; // boolean true if velocity measurements have passed innovation consistency checks bool velCheckPassed = false; // boolean true if velocity measurements have passed innovation consistency checks
bool hgtCheckPassed = false; // boolean true if height measurements have passed innovation consistency check
// declare variables used to control access to arrays // declare variables used to control access to arrays
bool fuseData[6] = {false,false,false,false,false,false}; bool fuseData[6] = {false,false,false,false,false,false};
@ -776,43 +776,47 @@ void NavEKF3_core::FuseVelPosNED()
} }
} }
// test height measurements // Test height measurements
if (fuseHgtData) { if (fuseHgtData) {
// calculate height innovations // Calculate height innovations
innovVelPos[5] = stateStruct.position.z - velPosObs[5]; innovVelPos[5] = stateStruct.position.z - velPosObs[5];
varInnovVelPos[5] = P[9][9] + R_OBS_DATA_CHECKS[5]; varInnovVelPos[5] = P[9][9] + R_OBS_DATA_CHECKS[5];
// calculate the innovation consistency test ratio
// Calculate the innovation consistency test ratio
hgtTestRatio = sq(innovVelPos[5]) / (sq(MAX(0.01f * (float)frontend->_hgtInnovGate, 1.0f)) * varInnovVelPos[5]); hgtTestRatio = sq(innovVelPos[5]) / (sq(MAX(0.01f * (float)frontend->_hgtInnovGate, 1.0f)) * varInnovVelPos[5]);
// when on ground we accept a larger test ratio to allow // When on ground we accept a larger test ratio to allow the filter to handle large switch on IMU
// the filter to handle large switch on IMU bias errors // bias errors without rejecting the height sensor.
// without rejecting the height sensor const float maxTestRatio = (PV_AidingMode == AID_NONE && onGround)? 3.0f : 1.0f;
const float maxTestRatio = (PV_AidingMode == AID_NONE && onGround)? 3.0 : 1.0; if (hgtTestRatio < maxTestRatio) {
hgtCheckPassed = true;
lastHgtPassTime_ms = imuSampleTime_ms;
}
// fail if the ratio is > 1, but don't fail if bad IMU data // Use height data if innovation check passed or timed out or if bad IMU data
hgtHealth = ((hgtTestRatio < maxTestRatio) || badIMUdata); // Always fuse data if bad IMU to prevent aliasing and clipping pulling the state estimate away
// from the measurement un-opposed if test threshold is exceeded.
// Fuse height data if healthy or timed out or in constant position mode if (hgtCheckPassed || hgtTimeout || badIMUdata) {
if (hgtHealth || hgtTimeout) {
// Calculate a filtered value to be used by pre-flight health checks // Calculate a filtered value to be used by pre-flight health checks
// We need to filter because wind gusts can generate significant baro noise and we want to be able to detect bias errors in the inertial solution // We need to filter because wind gusts can generate significant baro noise and we want to be able to detect bias errors in the inertial solution
if (onGround) { if (onGround) {
float dtBaro = (imuSampleTime_ms - lastHgtPassTime_ms)*1.0e-3f; float dtBaro = (imuSampleTime_ms - lastHgtPassTime_ms) * 1.0e-3f;
const float hgtInnovFiltTC = 2.0f; const float hgtInnovFiltTC = 2.0f;
float alpha = constrain_float(dtBaro/(dtBaro+hgtInnovFiltTC),0.0f,1.0f); float alpha = constrain_float(dtBaro/(dtBaro+hgtInnovFiltTC), 0.0f, 1.0f);
hgtInnovFiltState += (innovVelPos[5]-hgtInnovFiltState)*alpha; hgtInnovFiltState += (innovVelPos[5] - hgtInnovFiltState)*alpha;
} else { } else {
hgtInnovFiltState = 0.0f; hgtInnovFiltState = 0.0f;
} }
// if timed out, reset the height
if (hgtTimeout) { if (hgtTimeout) {
ResetHeight(); ResetHeight();
// Don't fuse the same data we have used to reset states.
fuseHgtData = false;
} }
// If we have got this far then declare the height data as healthy and reset the timeout counter } else {
hgtHealth = true; fuseHgtData = false;
lastHgtPassTime_ms = imuSampleTime_ms;
} }
} }
@ -828,7 +832,7 @@ void NavEKF3_core::FuseVelPosNED()
fuseData[3] = true; fuseData[3] = true;
fuseData[4] = true; fuseData[4] = true;
} }
if (fuseHgtData && hgtHealth) { if (fuseHgtData) {
fuseData[5] = true; fuseData[5] = true;
} }