AP_AHRS_MPU6000: fixed yaw correction and added _secondary_ahrs parameter

_secondary_ahrs parameter allows mpu6000 ahrs to runin parallel with DCM.
This commit is contained in:
rmackay9 2012-09-30 00:21:18 +09:00
parent db15b3b563
commit c2c2d560bb
2 changed files with 45 additions and 39 deletions

View File

@ -54,8 +54,10 @@ AP_AHRS_MPU6000::update(void)
{ {
float delta_t; float delta_t;
// tell the IMU to grab some data. is this necessary? // tell the IMU to grab some data.
if( !_secondary_ahrs ) {
_imu->update(); _imu->update();
}
// ask the IMU how much time this sensor reading represents // ask the IMU how much time this sensor reading represents
delta_t = _imu->get_delta_time(); delta_t = _imu->get_delta_time();
@ -224,60 +226,59 @@ AP_AHRS_MPU6000::yaw_error_compass(void)
// //
// drift_correction_yaw - yaw drift correction using the compass // drift_correction_yaw - yaw drift correction using the compass
// we have no way to update the dmp with it's actual heading from our compass
// instead we use the yaw_corrected variable to hold what we think is the real heading
// we also record what the dmp said it's last heading was in the yaw_last_uncorrected variable so that on the next iteration we can add the change in yaw to our estimate
// //
void void
AP_AHRS_MPU6000::drift_correction_yaw(void) AP_AHRS_MPU6000::drift_correction_yaw(void)
{ {
static float yaw_last_uncorrected = HEADING_UNKNOWN;
static float yaw_corrected = HEADING_UNKNOWN; static float yaw_corrected = HEADING_UNKNOWN;
float yaw_delta = 0; static float last_dmp_yaw = HEADING_UNKNOWN;
bool new_value = false; float dmp_roll, dmp_pitch, dmp_yaw; // roll pitch and yaw values from dmp
float yaw_error; float yaw_delta; // change in yaw according to dmp
float yaw_error; // difference between heading and corrected yaw
static float heading; static float heading;
// we assume the DCM matrix is completely uncorrect for yaw // get uncorrected yaw values from dmp
// retrieve how much heading has changed according to non-corrected dcm _mpu6000->quaternion.to_euler(&dmp_roll, &dmp_pitch, &dmp_yaw);
if( yaw_last_uncorrected != HEADING_UNKNOWN ) {
yaw_delta = wrap_PI(yaw - yaw_last_uncorrected); // the change in heading according to the gyros only since the last interation // initialise headings on first iteration
yaw_last_uncorrected = yaw; if( yaw_corrected == HEADING_UNKNOWN ) {
yaw_corrected = dmp_yaw;
last_dmp_yaw = dmp_yaw;
} }
// initialise yaw_corrected (if necessary) // change in yaw according to dmp
if( yaw_corrected != HEADING_UNKNOWN ) { yaw_delta = wrap_PI(dmp_yaw - last_dmp_yaw);
yaw_corrected = yaw; yaw_corrected = wrap_PI(yaw_corrected + yaw_delta);
}else{ last_dmp_yaw = dmp_yaw;
yaw_corrected = wrap_PI(yaw_corrected + yaw_delta); // best guess of current yaw is previous iterations corrected yaw + yaw change from gyro
_dcm_matrix.from_euler(roll, pitch, yaw_corrected); // rebuild dcm matrix with best guess at current yaw // rebuild dcm matrix
} _dcm_matrix.from_euler(dmp_roll, dmp_pitch, yaw_corrected);
// if we have new compass data // if we have new compass data
if(_compass && _compass->use_for_yaw() ) { if(_compass && _compass->use_for_yaw() ) {
if(_compass->last_update != _compass_last_update) { if(_compass->last_update != _compass_last_update) {
_compass_last_update = _compass->last_update; _compass_last_update = _compass->last_update;
heading = _compass->calculate_heading(_dcm_matrix); heading = _compass->calculate_heading(_dcm_matrix);
// if this is the first good compass reading then set yaw to this heading
if( !_have_initial_yaw ) { if( !_have_initial_yaw ) {
yaw_corrected = heading;
_have_initial_yaw = true; _have_initial_yaw = true;
_dcm_matrix.from_euler(roll, pitch, yaw_corrected); // rebuild dcm matrix with best guess at current yaw yaw_corrected = wrap_PI(heading);
}
new_value = true;
}
} }
// perform the yaw correction // yaw correction based on compass
if( new_value ) { //yaw_error = yaw_error_compass();
yaw_error = yaw_error_compass(); yaw_error = wrap_PI(heading - yaw_corrected);
// the yaw error is a vector in earth frame
//Vector3f error = Vector3f(0,0, yaw_error);
// convert the error vector to body frame // shift the corrected yaw towards the compass heading a bit
//error = _dcm_matrix.mul_transpose(error); yaw_corrected += wrap_PI(yaw_error * _kp_yaw.get() * 0.1);
// Update the differential component to reduce the error (it´s like a P control)
yaw_corrected += wrap_PI(yaw_error * _kp_yaw.get()); // probably completely wrong
// rebuild the dcm matrix yet again // rebuild the dcm matrix yet again
_dcm_matrix.from_euler(roll, pitch, yaw_corrected); _dcm_matrix.from_euler(dmp_roll, dmp_pitch, yaw_corrected);
}
} }
} }

View File

@ -62,6 +62,9 @@ public:
float get_error_rp(void); float get_error_rp(void);
float get_error_yaw(void); float get_error_yaw(void);
// set_as_secondary - avoid running some steps twice (imu updates) if this is a secondary ahrs
void set_as_secondary(bool set_as_secondary) { _secondary_ahrs = set_as_secondary; }
private: private:
float _ki; float _ki;
float _ki_yaw; float _ki_yaw;
@ -97,6 +100,8 @@ private:
float _error_yaw_sum; float _error_yaw_sum;
uint16_t _error_yaw_count; uint16_t _error_yaw_count;
float _error_yaw_last; float _error_yaw_last;
bool _secondary_ahrs;
}; };
#endif // AP_AHRS_MPU6000_H #endif // AP_AHRS_MPU6000_H