mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-24 09:38:29 -04:00
AP_InertialSensor: use filtered data in BMI270 and implement fifo reset
increase gyro ODR to 3.2Khz to allow higher AAF use OSR4 on accel filter to achieve nominal 188Hz
This commit is contained in:
parent
06a1749a0f
commit
8855a54720
@ -210,7 +210,6 @@ void AP_InertialSensor_BMI270::start()
|
|||||||
|
|
||||||
_dev->get_semaphore()->give();
|
_dev->get_semaphore()->give();
|
||||||
|
|
||||||
// using headerless mode so gyro and accel ODRs must match
|
|
||||||
if (!_imu.register_accel(_accel_instance, BMI270_BACKEND_SAMPLE_RATE, _dev->get_bus_id_devtype(DEVTYPE_BMI270)) ||
|
if (!_imu.register_accel(_accel_instance, BMI270_BACKEND_SAMPLE_RATE, _dev->get_bus_id_devtype(DEVTYPE_BMI270)) ||
|
||||||
!_imu.register_gyro(_gyro_instance, BMI270_BACKEND_SAMPLE_RATE, _dev->get_bus_id_devtype(DEVTYPE_BMI270))) {
|
!_imu.register_gyro(_gyro_instance, BMI270_BACKEND_SAMPLE_RATE, _dev->get_bus_id_devtype(DEVTYPE_BMI270))) {
|
||||||
return;
|
return;
|
||||||
@ -318,8 +317,10 @@ void AP_InertialSensor_BMI270::check_err_reg()
|
|||||||
|
|
||||||
void AP_InertialSensor_BMI270::configure_accel()
|
void AP_InertialSensor_BMI270::configure_accel()
|
||||||
{
|
{
|
||||||
// set acc in high performance mode with normal filtering at 1600Hz
|
// set acc in high performance mode with OSR4 filtering (751Hz/4) at 1600Hz
|
||||||
write_register(BMI270_REG_ACC_CONF, 1U<<7 | 0x2U<<4 | 0x0C);
|
// see https://community.bosch-sensortec.com/t5/MEMS-sensors-forum/BMI270-OSR-mode-behaviour/td-p/52020
|
||||||
|
// OSR4 is a 188Hz filter cutoff, acc_bwp == 0, equivalent to other driver filters
|
||||||
|
write_register(BMI270_REG_ACC_CONF, 1U<<7 | 0x0C);
|
||||||
// set acc to 16G full scale
|
// set acc to 16G full scale
|
||||||
write_register(BMI270_REG_ACC_RANGE, 0x03);
|
write_register(BMI270_REG_ACC_RANGE, 0x03);
|
||||||
|
|
||||||
@ -328,8 +329,9 @@ void AP_InertialSensor_BMI270::configure_accel()
|
|||||||
|
|
||||||
void AP_InertialSensor_BMI270::configure_gyro()
|
void AP_InertialSensor_BMI270::configure_gyro()
|
||||||
{
|
{
|
||||||
// set gyro in high performance filter mode, high performance noise mode. normal filtering at 1600Hz
|
// set gyro in high performance filter mode, high performance noise mode, normal filtering at 3.2KHz
|
||||||
write_register(BMI270_REG_GYRO_CONF, 1U<<7 | 1<<6 | 2<<4 | 0x0C);
|
// filter cutoff 751hz
|
||||||
|
write_register(BMI270_REG_GYRO_CONF, 1U<<7 | 1<<6 | 2<<4 | 0x0D);
|
||||||
// set gyro to 2000dps full scale
|
// set gyro to 2000dps full scale
|
||||||
// for some reason you have to enable the ois_range bit (bit 3) for 2000dps as well
|
// for some reason you have to enable the ois_range bit (bit 3) for 2000dps as well
|
||||||
// or else the gyro scale will be 250dps when in prefiltered FIFO mode (not documented in datasheet!)
|
// or else the gyro scale will be 250dps when in prefiltered FIFO mode (not documented in datasheet!)
|
||||||
@ -340,27 +342,44 @@ void AP_InertialSensor_BMI270::configure_gyro()
|
|||||||
|
|
||||||
void AP_InertialSensor_BMI270::configure_fifo()
|
void AP_InertialSensor_BMI270::configure_fifo()
|
||||||
{
|
{
|
||||||
// don't stop when full, disable sensortime frame
|
// stop when full, disable sensortime frame
|
||||||
write_register(BMI270_REG_FIFO_CONFIG_0, 0x00);
|
write_register(BMI270_REG_FIFO_CONFIG_0, 0x01);
|
||||||
// accel + gyro data in FIFO together with headers
|
// accel + gyro data in FIFO together with headers
|
||||||
write_register(BMI270_REG_FIFO_CONFIG_1, 1U<<7 | 1U<<6 | 1U<<4);
|
write_register(BMI270_REG_FIFO_CONFIG_1, 1U<<7 | 1U<<6 | 1U<<4);
|
||||||
// unfiltered with gyro downsampled by 2^2 to give 1600Hz
|
// filtered data downsampled by 2**1 to 1600Hz
|
||||||
write_register(BMI270_REG_FIFO_DOWNS, 0x02);
|
write_register(BMI270_REG_FIFO_DOWNS, 1U<<7 | 1U<<3 | 0x01);
|
||||||
// disable advanced power save, enable FIFO self-wake
|
// disable advanced power save, enable FIFO self-wake
|
||||||
write_register(BMI270_REG_PWR_CONF, 0x02);
|
write_register(BMI270_REG_PWR_CONF, 0x02);
|
||||||
// Enable the gyro, accelerometer and temperature sensor - disable aux interface
|
// Enable the gyro, accelerometer and temperature sensor - disable aux interface
|
||||||
write_register(BMI270_REG_PWR_CTRL, 0x0E);
|
write_register(BMI270_REG_PWR_CTRL, 0x0E);
|
||||||
// flush FIFO
|
|
||||||
write_register(BMI270_REG_CMD, BMI270_CMD_FIFOFLUSH);
|
fifo_reset();
|
||||||
|
|
||||||
check_err_reg();
|
check_err_reg();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void AP_InertialSensor_BMI270::fifo_reset()
|
||||||
|
{
|
||||||
|
// flush and reset FIFO
|
||||||
|
write_register(BMI270_REG_CMD, BMI270_CMD_FIFOFLUSH);
|
||||||
|
|
||||||
|
notify_accel_fifo_reset(_accel_instance);
|
||||||
|
notify_gyro_fifo_reset(_gyro_instance);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
read fifo
|
read fifo
|
||||||
*/
|
*/
|
||||||
void AP_InertialSensor_BMI270::read_fifo(void)
|
void AP_InertialSensor_BMI270::read_fifo(void)
|
||||||
{
|
{
|
||||||
|
// check for FIFO errors/overflow
|
||||||
|
uint8_t err = 0;
|
||||||
|
read_registers(BMI270_REG_ERR_REG, &err, 1);
|
||||||
|
if ((err>>6 & 1) == 1) {
|
||||||
|
fifo_reset();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
uint8_t len[2];
|
uint8_t len[2];
|
||||||
if (!read_registers(BMI270_REG_FIFO_LENGTH_LSB, len, 2)) {
|
if (!read_registers(BMI270_REG_FIFO_LENGTH_LSB, len, 2)) {
|
||||||
_inc_accel_error_count(_accel_instance);
|
_inc_accel_error_count(_accel_instance);
|
||||||
@ -418,17 +437,22 @@ void AP_InertialSensor_BMI270::read_fifo(void)
|
|||||||
break;
|
break;
|
||||||
case 0x48:
|
case 0x48:
|
||||||
// fifo config frame
|
// fifo config frame
|
||||||
frame_len = 2;
|
frame_len = 5;
|
||||||
break;
|
break;
|
||||||
case 0x50:
|
case 0x50:
|
||||||
// sample drop frame
|
// sample drop frame
|
||||||
frame_len = 2;
|
frame_len = 2;
|
||||||
break;
|
break;
|
||||||
|
case 0x80:
|
||||||
|
// invalid frame
|
||||||
|
fifo_reset();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
p += frame_len;
|
p += frame_len;
|
||||||
fifo_length -= frame_len;
|
fifo_length -= frame_len;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// temperature sensor updated every 10ms
|
||||||
if (temperature_counter++ == 100) {
|
if (temperature_counter++ == 100) {
|
||||||
temperature_counter = 0;
|
temperature_counter = 0;
|
||||||
uint8_t tbuf[2];
|
uint8_t tbuf[2];
|
||||||
@ -436,13 +460,15 @@ void AP_InertialSensor_BMI270::read_fifo(void)
|
|||||||
_inc_accel_error_count(_accel_instance);
|
_inc_accel_error_count(_accel_instance);
|
||||||
_inc_gyro_error_count(_gyro_instance);
|
_inc_gyro_error_count(_gyro_instance);
|
||||||
} else {
|
} else {
|
||||||
uint16_t temp_uint11 = (tbuf[0]<<3) | (tbuf[1]>>5);
|
uint16_t tval = tbuf[0] | (tbuf[1] << 8);
|
||||||
int16_t temp_int11 = temp_uint11>1023?temp_uint11-2048:temp_uint11;
|
if (tval != 0x8000) { // 0x8000 is invalid
|
||||||
float temp_degc = temp_int11 * 0.125f + 23;
|
int16_t klsb = static_cast<int16_t>(tval);
|
||||||
|
float temp_degc = klsb * 0.002f + 23.0f;
|
||||||
_publish_temperature(_accel_instance, temp_degc);
|
_publish_temperature(_accel_instance, temp_degc);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void AP_InertialSensor_BMI270::parse_accel_frame(const uint8_t* d)
|
void AP_InertialSensor_BMI270::parse_accel_frame(const uint8_t* d)
|
||||||
{
|
{
|
||||||
|
@ -93,10 +93,13 @@ private:
|
|||||||
*/
|
*/
|
||||||
void configure_gyro();
|
void configure_gyro();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reset FIFO.
|
||||||
|
*/
|
||||||
|
void fifo_reset();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Configure FIFO.
|
* Configure FIFO.
|
||||||
*
|
|
||||||
* @return true on success, false otherwise.
|
|
||||||
*/
|
*/
|
||||||
void configure_fifo();
|
void configure_fifo();
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user