mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-13 11:28:30 -04:00
AP_InertialSensor: catch FIFO overruns on BMI088
we have seen errors where the BMI088 gets out of sync, so that the 3 axes are rotated. The data is shifted by 4 bytes, so that X=Z, Y=X and Z=Y this changes the BMI088 to "stop on full" mode, which is what Bosch use in their example drivers, and also catches FIFO overrun events and triggers a full FIFO reset. This should fix the problem with the FIFO sync
This commit is contained in:
parent
fa70d62f08
commit
3e87782cc0
@ -254,8 +254,8 @@ bool AP_InertialSensor_BMI088::gyro_init()
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// setup FIFO for streaming X,Y,Z
|
// setup FIFO for streaming X,Y,Z, with stop-at-full
|
||||||
if (!dev_gyro->write_register(REGG_FIFO_CONFIG_1, 0x80, true)) {
|
if (!dev_gyro->write_register(REGG_FIFO_CONFIG_1, 0x40, true)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -374,38 +374,41 @@ void AP_InertialSensor_BMI088::read_fifo_gyro(void)
|
|||||||
_inc_gyro_error_count(gyro_instance);
|
_inc_gyro_error_count(gyro_instance);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
const float scale = radians(2000.0f) / 32767.0f;
|
||||||
|
const uint8_t max_frames = 8;
|
||||||
|
Vector3i data[max_frames];
|
||||||
|
|
||||||
|
if (num_frames & 0x80) {
|
||||||
|
// fifo overrun, reset, likely caused by scheduling error
|
||||||
|
dev_gyro->write_register(REGG_FIFO_CONFIG_1, 0x40, true);
|
||||||
|
goto check_next;
|
||||||
|
}
|
||||||
|
|
||||||
num_frames &= 0x7F;
|
num_frames &= 0x7F;
|
||||||
|
|
||||||
// don't read more than 8 frames at a time
|
// don't read more than 8 frames at a time
|
||||||
if (num_frames > 8) {
|
num_frames = MIN(num_frames, max_frames);
|
||||||
num_frames = 8;
|
|
||||||
}
|
|
||||||
if (num_frames == 0) {
|
if (num_frames == 0) {
|
||||||
return;
|
goto check_next;
|
||||||
}
|
}
|
||||||
uint8_t data[6*num_frames];
|
|
||||||
if (!dev_gyro->read_registers(REGG_FIFO_DATA, data, num_frames*6)) {
|
if (!dev_gyro->read_registers(REGG_FIFO_DATA, (uint8_t *)data, num_frames*6)) {
|
||||||
_inc_gyro_error_count(gyro_instance);
|
_inc_gyro_error_count(gyro_instance);
|
||||||
return;
|
goto check_next;
|
||||||
}
|
}
|
||||||
|
|
||||||
// data is 16 bits with 2000dps range
|
// data is 16 bits with 2000dps range
|
||||||
const float scale = radians(2000.0f) / 32767.0f;
|
|
||||||
for (uint8_t i = 0; i < num_frames; i++) {
|
for (uint8_t i = 0; i < num_frames; i++) {
|
||||||
const uint8_t *d = &data[i*6];
|
Vector3f gyro(data[i].x, data[i].y, data[i].z);
|
||||||
int16_t xyz[3] {
|
|
||||||
int16_t(uint16_t(d[0] | d[1]<<8)),
|
|
||||||
int16_t(uint16_t(d[2] | d[3]<<8)),
|
|
||||||
int16_t(uint16_t(d[4] | d[5]<<8)) };
|
|
||||||
Vector3f gyro(xyz[0], xyz[1], xyz[2]);
|
|
||||||
gyro *= scale;
|
gyro *= scale;
|
||||||
|
|
||||||
_rotate_and_correct_gyro(gyro_instance, gyro);
|
_rotate_and_correct_gyro(gyro_instance, gyro);
|
||||||
_notify_new_gyro_raw_sample(gyro_instance, gyro);
|
_notify_new_gyro_raw_sample(gyro_instance, gyro);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
check_next:
|
||||||
AP_HAL::Device::checkreg reg;
|
AP_HAL::Device::checkreg reg;
|
||||||
if (!dev_gyro->check_next_register()) {
|
if (!dev_gyro->check_next_register(reg)) {
|
||||||
log_register_change(dev_gyro->get_bus_id(), reg);
|
log_register_change(dev_gyro->get_bus_id(), reg);
|
||||||
_inc_gyro_error_count(gyro_instance);
|
_inc_gyro_error_count(gyro_instance);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user