mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-05 15:38:29 -04:00
AP_NavEKF3: Fix bug preventing horizontal position reset if badIMUdata
This commit is contained in:
parent
5d00b7d042
commit
36160ba9ce
@ -589,9 +589,8 @@ void NavEKF3_core::SelectVelPosFusion()
|
|||||||
void NavEKF3_core::FuseVelPosNED()
|
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 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 posCheckPassed = false; // boolean true if position measurements have passed innovation consistency check
|
||||||
bool hgtCheckPassed = false; // boolean true if height measurements have passed innovation consistency check
|
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
|
||||||
@ -689,46 +688,60 @@ void NavEKF3_core::FuseVelPosNED()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// calculate innovations and check GPS data validity using an innovation consistency check
|
// Test horizontal position measurements
|
||||||
// test position measurements
|
|
||||||
if (fusePosData) {
|
if (fusePosData) {
|
||||||
// test horizontal position measurements
|
|
||||||
innovVelPos[3] = stateStruct.position.x - velPosObs[3];
|
innovVelPos[3] = stateStruct.position.x - velPosObs[3];
|
||||||
innovVelPos[4] = stateStruct.position.y - velPosObs[4];
|
innovVelPos[4] = stateStruct.position.y - velPosObs[4];
|
||||||
varInnovVelPos[3] = P[7][7] + R_OBS_DATA_CHECKS[3];
|
varInnovVelPos[3] = P[7][7] + R_OBS_DATA_CHECKS[3];
|
||||||
varInnovVelPos[4] = P[8][8] + R_OBS_DATA_CHECKS[4];
|
varInnovVelPos[4] = P[8][8] + R_OBS_DATA_CHECKS[4];
|
||||||
// apply an innovation consistency threshold test, but don't fail if bad IMU data
|
|
||||||
|
// Apply an innovation consistency threshold test
|
||||||
|
// Don't allow test to fail if not navigating and using a constant position
|
||||||
|
// assumption to constrain tilt errors because innovations can become large
|
||||||
|
// due to vehicle motion.
|
||||||
float maxPosInnov2 = sq(MAX(0.01f * (float)frontend->_gpsPosInnovGate, 1.0f))*(varInnovVelPos[3] + varInnovVelPos[4]);
|
float maxPosInnov2 = sq(MAX(0.01f * (float)frontend->_gpsPosInnovGate, 1.0f))*(varInnovVelPos[3] + varInnovVelPos[4]);
|
||||||
posTestRatio = (sq(innovVelPos[3]) + sq(innovVelPos[4])) / maxPosInnov2;
|
posTestRatio = (sq(innovVelPos[3]) + sq(innovVelPos[4])) / maxPosInnov2;
|
||||||
posHealth = ((posTestRatio < 1.0f) || badIMUdata);
|
if (posTestRatio < 1.0f || (PV_AidingMode == AID_NONE)) {
|
||||||
// use position data if healthy or timed out
|
posCheckPassed = true;
|
||||||
if (PV_AidingMode == AID_NONE) {
|
|
||||||
posHealth = true;
|
|
||||||
lastPosPassTime_ms = imuSampleTime_ms;
|
lastPosPassTime_ms = imuSampleTime_ms;
|
||||||
} else if (posHealth || posTimeout) {
|
}
|
||||||
posHealth = true;
|
|
||||||
|
// Use position data if healthy or timed out or bad IMU data
|
||||||
|
// 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.
|
||||||
|
if (posCheckPassed || posTimeout || badIMUdata) {
|
||||||
|
posCheckPassed = true;
|
||||||
lastPosPassTime_ms = imuSampleTime_ms;
|
lastPosPassTime_ms = imuSampleTime_ms;
|
||||||
// if timed out or outside the specified uncertainty radius, reset to the external sensor
|
// if timed out or outside the specified uncertainty radius, reset to the external sensor
|
||||||
if (posTimeout || ((P[8][8] + P[7][7]) > sq(float(frontend->_gpsGlitchRadiusMax)))) {
|
if (posTimeout || ((P[8][8] + P[7][7]) > sq(float(frontend->_gpsGlitchRadiusMax)))) {
|
||||||
// reset the position to the current external sensor position
|
// reset the position to the current external sensor position
|
||||||
ResetPosition(resetDataSource::DEFAULT);
|
ResetPosition(resetDataSource::DEFAULT);
|
||||||
// don't fuse external sensor data on this time step
|
|
||||||
|
// Don't fuse the same data we have used to reset states.
|
||||||
fusePosData = false;
|
fusePosData = false;
|
||||||
|
|
||||||
// Reset the position variances and corresponding covariances to a value that will pass the checks
|
// Reset the position variances and corresponding covariances to a value that will pass the checks
|
||||||
zeroRows(P,7,8);
|
zeroRows(P,7,8);
|
||||||
zeroCols(P,7,8);
|
zeroCols(P,7,8);
|
||||||
P[7][7] = sq(float(0.5f*frontend->_gpsGlitchRadiusMax));
|
P[7][7] = sq(float(0.5f*frontend->_gpsGlitchRadiusMax));
|
||||||
P[8][8] = P[7][7];
|
P[8][8] = P[7][7];
|
||||||
|
|
||||||
// Reset the normalised innovation to avoid failing the bad fusion tests
|
// Reset the normalised innovation to avoid failing the bad fusion tests
|
||||||
posTestRatio = 0.0f;
|
posTestRatio = 0.0f;
|
||||||
// also reset velocity if it has timed out
|
|
||||||
|
// Reset velocity if it has timed out
|
||||||
if (velTimeout) {
|
if (velTimeout) {
|
||||||
// reset the velocity to the external sensor velocity
|
|
||||||
ResetVelocity(resetDataSource::DEFAULT);
|
ResetVelocity(resetDataSource::DEFAULT);
|
||||||
|
|
||||||
|
// Don't fuse the same data we have used to reset states.
|
||||||
fuseVelData = false;
|
fuseVelData = false;
|
||||||
|
|
||||||
|
// Reset the normalised innovation to avoid failing the bad fusion tests
|
||||||
velTestRatio = 0.0f;
|
velTestRatio = 0.0f;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
fusePosData = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -828,7 +841,7 @@ void NavEKF3_core::FuseVelPosNED()
|
|||||||
fuseData[2] = true;
|
fuseData[2] = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (fusePosData && posHealth) {
|
if (fusePosData) {
|
||||||
fuseData[3] = true;
|
fuseData[3] = true;
|
||||||
fuseData[4] = true;
|
fuseData[4] = true;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user