DCM: added matrix recovery on reset

when we get a bad DCM error we can recover a matrix corresponding to
the current attitude, making it more likely that the aircraft will be
able to recover
This commit is contained in:
Andrew Tridgell 2012-02-23 22:58:41 +11:00
parent 1a32ececb4
commit b4c8d6491a
2 changed files with 56 additions and 18 deletions

View File

@ -194,20 +194,13 @@ AP_DCM::accel_adjust(void)
extreme errors in the matrix extreme errors in the matrix
*/ */
void void
AP_DCM::matrix_reset(void) AP_DCM::matrix_reset(bool recover_eulers)
{ {
if (_compass != NULL) { if (_compass != NULL) {
_compass->null_offsets_disable(); _compass->null_offsets_disable();
} }
_dcm_matrix.a.x = 1.0f;
_dcm_matrix.a.y = 0.0f; // reset the integration terms
_dcm_matrix.a.z = 0.0f;
_dcm_matrix.b.x = 0.0f;
_dcm_matrix.b.y = 1.0f;
_dcm_matrix.b.z = 0.0f;
_dcm_matrix.c.x = 0.0f;
_dcm_matrix.c.y = 0.0f;
_dcm_matrix.c.z = 1.0f;
_omega_I.x = 0.0f; _omega_I.x = 0.0f;
_omega_I.y = 0.0f; _omega_I.y = 0.0f;
_omega_I.z = 0.0f; _omega_I.z = 0.0f;
@ -218,6 +211,39 @@ AP_DCM::matrix_reset(void)
_error_yaw = _omega_I; _error_yaw = _omega_I;
_errorCourse = 0; _errorCourse = 0;
// if the caller wants us to try to recover to the current
// attitude then calculate the dcm matrix from the current
// roll/pitch/yaw values
if (recover_eulers && !isnan(roll) && !isnan(pitch) && !isnan(yaw)) {
float cp = cos(pitch);
float sp = sin(pitch);
float sr = sin(roll);
float cr = cos(roll);
float sy = sin(yaw);
float cy = cos(yaw);
_dcm_matrix.a.x = cp * cy;
_dcm_matrix.a.y = (sr * sp * cy) - (cr * sy);
_dcm_matrix.a.z = (cr * sp * cy) + (sr * sy);
_dcm_matrix.b.x = cp * sy;
_dcm_matrix.b.y = (sr * sp * sy) + (cr * cy);
_dcm_matrix.b.z = (cr * sp * sy) - (sr * cy);
_dcm_matrix.c.x = -sp;
_dcm_matrix.c.y = sr * cp;
_dcm_matrix.c.z = cr * cp;
} else {
// otherwise make it flat
//Serial.printf("zeroing DCM matrix\n");
_dcm_matrix.a.x = 1.0f;
_dcm_matrix.a.y = 0.0f;
_dcm_matrix.a.z = 0.0f;
_dcm_matrix.b.x = 0.0f;
_dcm_matrix.b.y = 1.0f;
_dcm_matrix.b.z = 0.0f;
_dcm_matrix.c.x = 0.0f;
_dcm_matrix.c.y = 0.0f;
_dcm_matrix.c.z = 1.0f;
}
if (_compass != NULL) { if (_compass != NULL) {
_compass->null_offsets_enable(); // This call is needed to restart the nulling _compass->null_offsets_enable(); // This call is needed to restart the nulling
// Otherwise the reset in the DCM matrix can mess up // Otherwise the reset in the DCM matrix can mess up
@ -231,6 +257,13 @@ AP_DCM::matrix_reset(void)
void void
AP_DCM::check_matrix(void) AP_DCM::check_matrix(void)
{ {
if (_dcm_matrix.is_nan()) {
//Serial.printf("ERROR: DCM matrix NAN\n");
SITL_debug("ERROR: DCM matrix NAN\n");
renorm_blowup_count++;
matrix_reset(true);
return;
}
// some DCM matrix values can lead to an out of range error in // some DCM matrix values can lead to an out of range error in
// the pitch calculation via asin(). These NaN values can // the pitch calculation via asin(). These NaN values can
// feed back into the rest of the DCM matrix via the // feed back into the rest of the DCM matrix via the
@ -240,14 +273,17 @@ AP_DCM::check_matrix(void)
// We have an invalid matrix. Force a normalisation. // We have an invalid matrix. Force a normalisation.
renorm_range_count++; renorm_range_count++;
normalize(); normalize();
if (!(_dcm_matrix.c.x < 1.0 &&
_dcm_matrix.c.x > -1.0)) { if (isnan(_dcm_matrix.c.x) ||
fabs(_dcm_matrix.c.x) > 10) {
// normalisation didn't fix the problem! We're // normalisation didn't fix the problem! We're
// in real trouble. All we can do is reset // in real trouble. All we can do is reset
//Serial.printf("ERROR: DCM matrix error. _dcm_matrix.c.x=%f\n",
// _dcm_matrix.c.x);
SITL_debug("ERROR: DCM matrix error. _dcm_matrix.c.x=%f\n", SITL_debug("ERROR: DCM matrix error. _dcm_matrix.c.x=%f\n",
_dcm_matrix.c.x); _dcm_matrix.c.x);
renorm_blowup_count++; renorm_blowup_count++;
matrix_reset(); matrix_reset(true);
} }
} }
} }
@ -284,7 +320,7 @@ AP_DCM::normalize(void)
_dcm_matrix.c = renorm(temporary[2], problem); _dcm_matrix.c = renorm(temporary[2], problem);
if (problem == 1) { // Our solution is blowing up and we will force back to initial condition. Hope we are not upside down! if (problem == 1) { // Our solution is blowing up and we will force back to initial condition. Hope we are not upside down!
matrix_reset(); matrix_reset(true);
} }
} }
@ -323,6 +359,8 @@ AP_DCM::renorm(Vector3f const &a, int &problem)
// can recover our attitude using drift // can recover our attitude using drift
// correction before we hit the ground! // correction before we hit the ground!
problem = 1; problem = 1;
//Serial.printf("ERROR: DCM renormalisation error. renorm_val=%f\n",
// renorm_val);
SITL_debug("ERROR: DCM renormalisation error. renorm_val=%f\n", SITL_debug("ERROR: DCM renormalisation error. renorm_val=%f\n",
renorm_val); renorm_val);
renorm_blowup_count++; renorm_blowup_count++;
@ -444,7 +482,7 @@ AP_DCM::euler_angles(void)
pitch = asin((_accel_vector.x) / (double)9.81); // asin(acc_x) pitch = asin((_accel_vector.x) / (double)9.81); // asin(acc_x)
yaw = 0; yaw = 0;
#else #else
pitch = -asin(_dcm_matrix.c.x); pitch = -safe_asin(_dcm_matrix.c.x);
roll = atan2(_dcm_matrix.c.y, _dcm_matrix.c.z); roll = atan2(_dcm_matrix.c.y, _dcm_matrix.c.z);
yaw = atan2(_dcm_matrix.b.x, _dcm_matrix.a.x); yaw = atan2(_dcm_matrix.b.x, _dcm_matrix.a.x);
#endif #endif
@ -461,7 +499,7 @@ void
AP_DCM::euler_rp(void) AP_DCM::euler_rp(void)
{ {
check_matrix(); check_matrix();
pitch = -asin(_dcm_matrix.c.x); pitch = -safe_asin(_dcm_matrix.c.x);
roll = atan2(_dcm_matrix.c.y, _dcm_matrix.c.z); roll = atan2(_dcm_matrix.c.y, _dcm_matrix.c.z);
roll_sensor = roll * DEGX100; //degrees(roll) * 100; roll_sensor = roll * DEGX100; //degrees(roll) * 100;
pitch_sensor = pitch * DEGX100; //degrees(pitch) * 100; pitch_sensor = pitch * DEGX100; //degrees(pitch) * 100;

View File

@ -57,7 +57,7 @@ public:
// Methods // Methods
void update_DCM(void); void update_DCM(void);
void update_DCM_fast(void); void update_DCM_fast(void);
void matrix_reset(void); void matrix_reset(bool recover_eulers = false);
long roll_sensor; // Degrees * 100 long roll_sensor; // Degrees * 100
long pitch_sensor; // Degrees * 100 long pitch_sensor; // Degrees * 100