mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-12 10:58:30 -04:00
Quaternion: added more numerical safety in the quaternion code
prevent infinities from creeping in and prevent large drift changes
This commit is contained in:
parent
f2e6714598
commit
89a10c584d
@ -31,6 +31,13 @@ AP_Quaternion::set_compass(Compass *compass)
|
|||||||
_compass = compass;
|
_compass = compass;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// to keep the code as close to the original as possible, we use these
|
||||||
|
// macros for quaternion access
|
||||||
|
#define SEq_1 q.q1
|
||||||
|
#define SEq_2 q.q2
|
||||||
|
#define SEq_3 q.q3
|
||||||
|
#define SEq_4 q.q4
|
||||||
|
|
||||||
// Function to compute one quaternion iteration without magnetometer
|
// Function to compute one quaternion iteration without magnetometer
|
||||||
void AP_Quaternion::update_IMU(float deltat, Vector3f &gyro, Vector3f &accel)
|
void AP_Quaternion::update_IMU(float deltat, Vector3f &gyro, Vector3f &accel)
|
||||||
{
|
{
|
||||||
@ -55,6 +62,11 @@ void AP_Quaternion::update_IMU(float deltat, Vector3f &gyro, Vector3f &accel)
|
|||||||
|
|
||||||
// normalise accelerometer vector
|
// normalise accelerometer vector
|
||||||
accel.normalize();
|
accel.normalize();
|
||||||
|
if (accel.is_inf()) {
|
||||||
|
// discard this data point
|
||||||
|
renorm_range_count++;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Compute the objective function and Jacobian
|
// Compute the objective function and Jacobian
|
||||||
f_1 = twoSEq_2 * SEq_4 - twoSEq_1 * SEq_3 - accel.x;
|
f_1 = twoSEq_2 * SEq_4 - twoSEq_1 * SEq_3 - accel.x;
|
||||||
@ -75,12 +87,16 @@ void AP_Quaternion::update_IMU(float deltat, Vector3f &gyro, Vector3f &accel)
|
|||||||
|
|
||||||
// Normalise the gradient
|
// Normalise the gradient
|
||||||
norm = 1.0/sqrt(SEqHatDot_1 * SEqHatDot_1 + SEqHatDot_2 * SEqHatDot_2 + SEqHatDot_3 * SEqHatDot_3 + SEqHatDot_4 * SEqHatDot_4);
|
norm = 1.0/sqrt(SEqHatDot_1 * SEqHatDot_1 + SEqHatDot_2 * SEqHatDot_2 + SEqHatDot_3 * SEqHatDot_3 + SEqHatDot_4 * SEqHatDot_4);
|
||||||
if (!isinf(norm)) {
|
if (isinf(norm)) {
|
||||||
SEqHatDot_1 *= norm;
|
// we can't do an update - discard this data point and
|
||||||
SEqHatDot_2 *= norm;
|
// hope the next one is better
|
||||||
SEqHatDot_3 *= norm;
|
renorm_range_count++;
|
||||||
SEqHatDot_4 *= norm;
|
return;
|
||||||
}
|
}
|
||||||
|
SEqHatDot_1 *= norm;
|
||||||
|
SEqHatDot_2 *= norm;
|
||||||
|
SEqHatDot_3 *= norm;
|
||||||
|
SEqHatDot_4 *= norm;
|
||||||
|
|
||||||
// Compute the quaternion derrivative measured by gyroscopes
|
// Compute the quaternion derrivative measured by gyroscopes
|
||||||
SEqDot_omega_1 = -halfSEq_2 * gyro.x - halfSEq_3 * gyro.y - halfSEq_4 * gyro.z;
|
SEqDot_omega_1 = -halfSEq_2 * gyro.x - halfSEq_3 * gyro.y - halfSEq_4 * gyro.z;
|
||||||
@ -96,12 +112,17 @@ void AP_Quaternion::update_IMU(float deltat, Vector3f &gyro, Vector3f &accel)
|
|||||||
|
|
||||||
// Normalise quaternion
|
// Normalise quaternion
|
||||||
norm = 1.0/sqrt(SEq_1 * SEq_1 + SEq_2 * SEq_2 + SEq_3 * SEq_3 + SEq_4 * SEq_4);
|
norm = 1.0/sqrt(SEq_1 * SEq_1 + SEq_2 * SEq_2 + SEq_3 * SEq_3 + SEq_4 * SEq_4);
|
||||||
if (!isinf(norm)) {
|
if (isinf(norm)) {
|
||||||
SEq_1 *= norm;
|
// our quaternion is bad! Reset based on roll/pitch/yaw
|
||||||
SEq_2 *= norm;
|
// and hope for the best ...
|
||||||
SEq_3 *= norm;
|
renorm_blowup_count++;
|
||||||
SEq_4 *= norm;
|
quaternion_from_euler(q, roll, pitch, yaw);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
SEq_1 *= norm;
|
||||||
|
SEq_2 *= norm;
|
||||||
|
SEq_3 *= norm;
|
||||||
|
SEq_4 *= norm;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -124,9 +145,19 @@ void AP_Quaternion::update_MARG(float deltat, Vector3f &gyro, Vector3f &accel, V
|
|||||||
|
|
||||||
// normalise accelerometer vector
|
// normalise accelerometer vector
|
||||||
accel.normalize();
|
accel.normalize();
|
||||||
|
if (accel.is_inf()) {
|
||||||
|
// discard this data point
|
||||||
|
renorm_range_count++;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// normalise the magnetometer measurement
|
// normalise the magnetometer measurement
|
||||||
mag.normalize();
|
mag.normalize();
|
||||||
|
if (mag.is_inf()) {
|
||||||
|
// discard this data point
|
||||||
|
renorm_range_count++;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// auxiliary variables to avoid repeated calculations
|
// auxiliary variables to avoid repeated calculations
|
||||||
float halfSEq_1 = 0.5f * SEq_1;
|
float halfSEq_1 = 0.5f * SEq_1;
|
||||||
@ -189,6 +220,11 @@ void AP_Quaternion::update_MARG(float deltat, Vector3f &gyro, Vector3f &accel, V
|
|||||||
|
|
||||||
// normalise the gradient to estimate direction of the gyroscope error
|
// normalise the gradient to estimate direction of the gyroscope error
|
||||||
norm = 1.0 / sqrt(SEqHatDot_1 * SEqHatDot_1 + SEqHatDot_2 * SEqHatDot_2 + SEqHatDot_3 * SEqHatDot_3 + SEqHatDot_4 * SEqHatDot_4);
|
norm = 1.0 / sqrt(SEqHatDot_1 * SEqHatDot_1 + SEqHatDot_2 * SEqHatDot_2 + SEqHatDot_3 * SEqHatDot_3 + SEqHatDot_4 * SEqHatDot_4);
|
||||||
|
if (isinf(norm)) {
|
||||||
|
// discard this data point
|
||||||
|
renorm_range_count++;
|
||||||
|
return;
|
||||||
|
}
|
||||||
SEqHatDot_1 *= norm;
|
SEqHatDot_1 *= norm;
|
||||||
SEqHatDot_2 *= norm;
|
SEqHatDot_2 *= norm;
|
||||||
SEqHatDot_3 *= norm;
|
SEqHatDot_3 *= norm;
|
||||||
@ -205,8 +241,18 @@ void AP_Quaternion::update_MARG(float deltat, Vector3f &gyro, Vector3f &accel, V
|
|||||||
_error_rp_count++;
|
_error_rp_count++;
|
||||||
_error_yaw_count++;
|
_error_yaw_count++;
|
||||||
|
|
||||||
// compute and remove the gyroscope baises
|
// compute the gyroscope bias delta
|
||||||
gyro_bias += w_err * (deltat * zeta);
|
Vector3f drift_delta = w_err * (deltat * zeta);
|
||||||
|
|
||||||
|
// don't allow the drift rate to be exceeded. This prevents a
|
||||||
|
// sudden drift change coming from a outage in the compass
|
||||||
|
float max_change = gyroMeasDrift * deltat;
|
||||||
|
drift_delta.x = constrain(drift_delta.x, -max_change, max_change);
|
||||||
|
drift_delta.y = constrain(drift_delta.y, -max_change, max_change);
|
||||||
|
drift_delta.z = constrain(drift_delta.z, -max_change, max_change);
|
||||||
|
gyro_bias += drift_delta;
|
||||||
|
|
||||||
|
// correct the gyro reading for drift
|
||||||
gyro -= gyro_bias;
|
gyro -= gyro_bias;
|
||||||
|
|
||||||
// compute the quaternion rate measured by gyroscopes
|
// compute the quaternion rate measured by gyroscopes
|
||||||
@ -223,12 +269,17 @@ void AP_Quaternion::update_MARG(float deltat, Vector3f &gyro, Vector3f &accel, V
|
|||||||
|
|
||||||
// normalise quaternion
|
// normalise quaternion
|
||||||
norm = 1.0/sqrt(SEq_1 * SEq_1 + SEq_2 * SEq_2 + SEq_3 * SEq_3 + SEq_4 * SEq_4);
|
norm = 1.0/sqrt(SEq_1 * SEq_1 + SEq_2 * SEq_2 + SEq_3 * SEq_3 + SEq_4 * SEq_4);
|
||||||
if (!isinf(norm)) {
|
if (isinf(norm)) {
|
||||||
SEq_1 *= norm;
|
// our quaternion is bad! Reset based on roll/pitch/yaw
|
||||||
SEq_2 *= norm;
|
// and hope for the best ...
|
||||||
SEq_3 *= norm;
|
renorm_blowup_count++;
|
||||||
SEq_4 *= norm;
|
quaternion_from_euler(q, roll, pitch, yaw);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
SEq_1 *= norm;
|
||||||
|
SEq_2 *= norm;
|
||||||
|
SEq_3 *= norm;
|
||||||
|
SEq_4 *= norm;
|
||||||
|
|
||||||
// compute flux in the earth frame
|
// compute flux in the earth frame
|
||||||
// recompute axulirary variables
|
// recompute axulirary variables
|
||||||
@ -265,6 +316,13 @@ void AP_Quaternion::update(void)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!_have_initial_yaw && _compass && _compass->use_for_yaw()) {
|
||||||
|
// setup the quaternion with initial compass yaw
|
||||||
|
quaternion_from_euler(q, 0, 0, _compass->heading);
|
||||||
|
_have_initial_yaw = true;
|
||||||
|
gyro_bias.zero();
|
||||||
|
}
|
||||||
|
|
||||||
// get current IMU state
|
// get current IMU state
|
||||||
gyro = _imu->get_gyro();
|
gyro = _imu->get_gyro();
|
||||||
accel = _imu->get_accel();
|
accel = _imu->get_accel();
|
||||||
@ -293,7 +351,7 @@ void AP_Quaternion::update(void)
|
|||||||
accel.z += (gyro.y - gyro_bias.y) * veloc;
|
accel.z += (gyro.y - gyro_bias.y) * veloc;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_compass == NULL) {
|
if (_compass == NULL || !_compass->use_for_yaw()) {
|
||||||
update_IMU(deltat, gyro, accel);
|
update_IMU(deltat, gyro, accel);
|
||||||
} else {
|
} else {
|
||||||
// mag.z is reversed in quaternion code
|
// mag.z is reversed in quaternion code
|
||||||
|
@ -22,12 +22,6 @@ public:
|
|||||||
_gps(gps),
|
_gps(gps),
|
||||||
_compass(compass)
|
_compass(compass)
|
||||||
{
|
{
|
||||||
// initial quaternion
|
|
||||||
SEq_1 = 1;
|
|
||||||
SEq_2 = 0;
|
|
||||||
SEq_3 = 0;
|
|
||||||
SEq_4 = 0;
|
|
||||||
|
|
||||||
// reference direction of flux in earth frame
|
// reference direction of flux in earth frame
|
||||||
b_x = 0;
|
b_x = 0;
|
||||||
b_z = -1;
|
b_z = -1;
|
||||||
@ -97,8 +91,8 @@ private:
|
|||||||
// true if we are doing centripetal acceleration correction
|
// true if we are doing centripetal acceleration correction
|
||||||
bool _centripetal;
|
bool _centripetal;
|
||||||
|
|
||||||
// maximum gyroscope measurement error in rad/s (set to 10 degrees/second)
|
// maximum gyroscope measurement error in rad/s (set to 20 degrees/second)
|
||||||
static const float gyroMeasError = 10.0 * (M_PI/180.0);
|
static const float gyroMeasError = 20.0 * (M_PI/180.0);
|
||||||
|
|
||||||
// maximum gyroscope drift rate in radians/s/s (set to 0.02
|
// maximum gyroscope drift rate in radians/s/s (set to 0.02
|
||||||
// degrees/s/s, which is 1.2 degrees/s/minute)
|
// degrees/s/s, which is 1.2 degrees/s/minute)
|
||||||
@ -108,8 +102,10 @@ private:
|
|||||||
float zeta;
|
float zeta;
|
||||||
|
|
||||||
// quaternion elements
|
// quaternion elements
|
||||||
float SEq_1, SEq_2, SEq_3, SEq_4;
|
Quaternion q;
|
||||||
|
|
||||||
|
// magnetic flux estimates. These are used for the automatic
|
||||||
|
// magnetometer calibration
|
||||||
float b_x;
|
float b_x;
|
||||||
float b_z;
|
float b_z;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user