AP_NavEKF2: constrain field by table after fusion

this should make for faster convergence
This commit is contained in:
Andrew Tridgell 2019-06-03 11:15:13 +10:00 committed by Randy Mackay
parent 5ac757ad3a
commit 1930103002
3 changed files with 31 additions and 12 deletions

View File

@ -729,6 +729,11 @@ void NavEKF2_core::FuseMagnetometer()
statesArray[j] = statesArray[j] - Kfusion[j] * innovMag[obsIndex]; statesArray[j] = statesArray[j] - Kfusion[j] * innovMag[obsIndex];
} }
// add table constraint here for faster convergence
if (have_table_earth_field && frontend->_mag_ef_limit > 0) {
MagTableConstrain();
}
// the first 3 states represent the angular misalignment vector. This is // the first 3 states represent the angular misalignment vector. This is
// is used to correct the estimated quaternion on the current time step // is used to correct the estimated quaternion on the current time step
stateStruct.quat.rotate(stateStruct.angErr); stateStruct.quat.rotate(stateStruct.angErr);

View File

@ -1407,6 +1407,22 @@ void NavEKF2_core::ConstrainVariances()
for (uint8_t i=22; i<=23; i++) P[i][i] = constrain_float(P[i][i],0.0f,1.0e3f); // wind velocity for (uint8_t i=22; i<=23; i++) P[i][i] = constrain_float(P[i][i],0.0f,1.0e3f); // wind velocity
} }
// constrain states using WMM tables and specified limit
void NavEKF2_core::MagTableConstrain(void)
{
// constrain to error from table earth field
float limit_ga = frontend->_mag_ef_limit * 0.001f;
stateStruct.earth_magfield.x = constrain_float(stateStruct.earth_magfield.x,
table_earth_field_ga.x-limit_ga,
table_earth_field_ga.x+limit_ga);
stateStruct.earth_magfield.y = constrain_float(stateStruct.earth_magfield.y,
table_earth_field_ga.y-limit_ga,
table_earth_field_ga.y+limit_ga);
stateStruct.earth_magfield.z = constrain_float(stateStruct.earth_magfield.z,
table_earth_field_ga.z-limit_ga,
table_earth_field_ga.z+limit_ga);
}
// constrain states to prevent ill-conditioning // constrain states to prevent ill-conditioning
void NavEKF2_core::ConstrainStates() void NavEKF2_core::ConstrainStates()
{ {
@ -1430,17 +1446,8 @@ void NavEKF2_core::ConstrainStates()
// constrain to +/-1Ga // constrain to +/-1Ga
for (uint8_t i=16; i<=18; i++) statesArray[i] = constrain_float(statesArray[i],-1.0f,1.0f); for (uint8_t i=16; i<=18; i++) statesArray[i] = constrain_float(statesArray[i],-1.0f,1.0f);
} else { } else {
// constrain to error from table earth field // use table constrain
float limit_ga = frontend->_mag_ef_limit * 0.001f; MagTableConstrain();
stateStruct.earth_magfield.x = constrain_float(stateStruct.earth_magfield.x,
table_earth_field_ga.x-limit_ga,
table_earth_field_ga.x+limit_ga);
stateStruct.earth_magfield.y = constrain_float(stateStruct.earth_magfield.y,
table_earth_field_ga.y-limit_ga,
table_earth_field_ga.y+limit_ga);
stateStruct.earth_magfield.z = constrain_float(stateStruct.earth_magfield.z,
table_earth_field_ga.z-limit_ga,
table_earth_field_ga.z+limit_ga);
} }
// body magnetic field limit // body magnetic field limit
@ -1511,7 +1518,11 @@ Quaternion NavEKF2_core::calcQuatAndFieldStates(float roll, float pitch)
// don't do this if the earth field has already been learned // don't do this if the earth field has already been learned
if (!magFieldLearned) { if (!magFieldLearned) {
initQuat.rotation_matrix(Tbn); initQuat.rotation_matrix(Tbn);
stateStruct.earth_magfield = Tbn * magDataDelayed.mag; if (have_table_earth_field && frontend->_mag_ef_limit > 0) {
stateStruct.earth_magfield = table_earth_field_ga;
} else {
stateStruct.earth_magfield = Tbn * magDataDelayed.mag;
}
// set the NE earth magnetic field states using the published declination // set the NE earth magnetic field states using the published declination
// and set the corresponding variances and covariances // and set the corresponding variances and covariances

View File

@ -489,6 +489,9 @@ private:
// constrain states // constrain states
void ConstrainStates(); void ConstrainStates();
// constrain earth field using WMM tables
void MagTableConstrain(void);
// fuse selected position, velocity and height measurements // fuse selected position, velocity and height measurements
void FuseVelPosNED(); void FuseVelPosNED();