diff --git a/libraries/AP_Compass/AP_Compass.cpp b/libraries/AP_Compass/AP_Compass.cpp
index da8e2a1313a87..355b2bf9f8015 100644
--- a/libraries/AP_Compass/AP_Compass.cpp
+++ b/libraries/AP_Compass/AP_Compass.cpp
@@ -21,6 +21,7 @@
#include "AP_Compass_AK8963.h"
#include "AP_Compass_Backend.h"
#include "AP_Compass_BMM150.h"
+#include "AP_Compass_BMM350.h"
#include "AP_Compass_HMC5843.h"
#include "AP_Compass_IST8308.h"
#include "AP_Compass_IST8310.h"
@@ -516,7 +517,7 @@ const AP_Param::GroupInfo Compass::var_info[] = {
// @Param: DISBLMSK
// @DisplayName: Compass disable driver type mask
// @Description: This is a bitmask of driver types to disable. If a driver type is set in this mask then that driver will not try to find a sensor at startup
- // @Bitmask: 0:HMC5883,1:LSM303D,2:AK8963,3:BMM150,4:LSM9DS1,5:LIS3MDL,6:AK09916,7:IST8310,8:ICM20948,9:MMC3416,11:DroneCAN,12:QMC5883,14:MAG3110,15:IST8308,16:RM3100,17:MSP,18:ExternalAHRS
+ // @Bitmask: 0:HMC5883,1:LSM303D,2:AK8963,3:BMM150,4:LSM9DS1,5:LIS3MDL,6:AK09916,7:IST8310,8:ICM20948,9:MMC3416,11:DroneCAN,12:QMC5883,14:MAG3110,15:IST8308,16:RM3100,17:MSP,18:ExternalAHRS,21:BMM350
// @User: Advanced
AP_GROUPINFO("DISBLMSK", 33, Compass, _driver_type_mask, 0),
@@ -1314,6 +1315,23 @@ void Compass::_probe_external_i2c_compasses(void)
}
}
#endif // AP_COMPASS_BMM150_ENABLED
+
+#if AP_COMPASS_BMM350_ENABLED
+ // BMM350 on I2C
+ FOREACH_I2C_EXTERNAL(i) {
+ for (uint8_t addr=BMM350_I2C_ADDR_MIN; addr <= BMM350_I2C_ADDR_MAX; addr++) {
+ ADD_BACKEND(DRIVER_BMM350,
+ AP_Compass_BMM350::probe(GET_I2C_DEVICE(i, addr), true, ROTATION_NONE));
+ }
+ }
+#if !defined(HAL_SKIP_AUTO_INTERNAL_I2C_PROBE)
+ FOREACH_I2C_INTERNAL(i) {
+ for (uint8_t addr=BMM350_I2C_ADDR_MIN; addr <= BMM350_I2C_ADDR_MAX; addr++) {
+ ADD_BACKEND(DRIVER_BMM350, AP_Compass_BMM350::probe(GET_I2C_DEVICE(i, addr), all_external, ROTATION_NONE));
+ }
+ }
+#endif
+#endif // AP_COMPASS_BMM350_ENABLED
}
/*
diff --git a/libraries/AP_Compass/AP_Compass.h b/libraries/AP_Compass/AP_Compass.h
index fe1a66ad7e310..6498c1d31cd43 100644
--- a/libraries/AP_Compass/AP_Compass.h
+++ b/libraries/AP_Compass/AP_Compass.h
@@ -496,6 +496,9 @@ friend class AP_Compass_Backend;
#if AP_COMPASS_QMC5883P_ENABLED
DRIVER_QMC5883P =20,
#endif
+#if AP_COMPASS_BMM350_ENABLED
+ DRIVER_BMM350 =21,
+#endif
};
bool _driver_enabled(enum DriverType driver_type);
diff --git a/libraries/AP_Compass/AP_Compass_BMM350.cpp b/libraries/AP_Compass/AP_Compass_BMM350.cpp
new file mode 100644
index 0000000000000..d5f83419b8af5
--- /dev/null
+++ b/libraries/AP_Compass/AP_Compass_BMM350.cpp
@@ -0,0 +1,492 @@
+/*
+ * This file is free software: you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License as published by the
+ * Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program. If not, see .
+ */
+#include "AP_Compass_BMM350.h"
+
+#if AP_COMPASS_BMM350_ENABLED
+
+#include
+
+#include
+
+#include
+#include
+
+#define BMM350_REG_CHIP_ID 0x00
+#define BMM350_REG_PMU_CMD_AGGR_SET 0x04
+#define BMM350_REG_PMU_CMD_AXIS_EN 0x05
+#define BMM350_REG_PMU_CMD 0x06
+#define BMM350_REG_PMU_CMD_STATUS_0 0x07
+#define BMM350_REG_INT_CTRL 0x2E
+#define BMM350_REG_MAG_X_XLSB 0x31
+#define BMM350_REG_OTP_CMD 0x50
+#define BMM350_REG_OTP_DATA_MSB 0x52
+#define BMM350_REG_OTP_DATA_LSB 0x53
+#define BMM350_REG_OTP_STATUS 0x55
+#define BMM350_REG_CMD 0x7E
+
+// OTP(one-time programmable memory)
+#define BMM350_OTP_CMD_DIR_READ (0x01<<5U)
+#define BMM350_OTP_CMD_PWR_OFF_OTP (0x04<<5U)
+
+#define BMM350_OTP_STATUS_ERROR_MASK 0xE0
+#define BMM350_OTP_STATUS_CMD_DONE 0x01
+
+#define BMM350_TEMP_OFF_SENS 0x0D
+#define BMM350_MAG_OFFSET_X 0x0E
+#define BMM350_MAG_OFFSET_Y 0x0F
+#define BMM350_MAG_OFFSET_Z 0x10
+#define BMM350_MAG_SENS_X 0x10
+#define BMM350_MAG_SENS_Y 0x11
+#define BMM350_MAG_SENS_Z 0x11
+#define BMM350_MAG_TCO_X 0x12
+#define BMM350_MAG_TCO_Y 0x13
+#define BMM350_MAG_TCO_Z 0x14
+#define BMM350_MAG_TCS_X 0x12
+#define BMM350_MAG_TCS_Y 0x13
+#define BMM350_MAG_TCS_Z 0x14
+#define BMM350_MAG_DUT_T_0 0x18
+#define BMM350_CROSS_X_Y 0x15
+#define BMM350_CROSS_Y_X 0x15
+#define BMM350_CROSS_Z_X 0x16
+#define BMM350_CROSS_Z_Y 0x16
+#define BMM350_SENS_CORR_Y 0.01f
+#define BMM350_TCS_CORR_Z 0.0001f
+
+#define BMM350_CMD_SOFTRESET 0xB6
+#define BMM350_INT_MODE_PULSED (0<<0U)
+#define BMM350_INT_POL_ACTIVE_HIGH (1<<1U)
+#define BMM350_INT_OD_PUSHPULL (1<<2U)
+#define BMM350_INT_OUTPUT_DISABLE (0<<3U)
+#define BMM350_INT_DRDY_EN (1<<7U)
+
+// Averaging performance
+#define BMM350_AVERAGING_4 (0x02 << 4U)
+#define BMM350_AVERAGING_8 (0x03 << 4U)
+
+// Output data rate
+#define BMM350_ODR_100HZ 0x04
+#define BMM350_ODR_50HZ 0x05
+
+// Power modes
+#define BMM350_PMU_CMD_SUSPEND_MODE 0x00
+#define BMM350_PMU_CMD_NORMAL_MODE 0x01
+#define BMM350_PMU_CMD_UPD_OAE 0x02
+#define BMM350_PMU_CMD_FGR 0x05
+#define BMM350_PMU_CMD_BR 0x07
+
+// OTP data length
+#define BMM350_OTP_DATA_LENGTH 32U
+
+// Chip ID of BMM350
+#define BMM350_CHIP_ID 0x33
+
+#define BMM350_XY_SENSITIVE 14.55f
+#define BMM350_Z_SENSITIVE 9.0f
+#define BMM350_TEMP_SENSITIVE 0.00204f
+#define BMM350_XY_INA_GAIN 19.46f
+#define BMM350_Z_INA_GAIN 31.0f
+#define BMM350_ADC_GAIN (1.0f / 1.5f)
+#define BMM350_LUT_GAIN 0.714607238769531f
+#define BMM350_POWER ((float)(1000000.0 / 1048576.0))
+
+#define BMM350_XY_SCALE (BMM350_POWER / (BMM350_XY_SENSITIVE * BMM350_XY_INA_GAIN * BMM350_ADC_GAIN * BMM350_LUT_GAIN))
+#define BMM350_Z_SCALE (BMM350_POWER / (BMM350_Z_SENSITIVE * BMM350_Z_INA_GAIN * BMM350_ADC_GAIN * BMM350_LUT_GAIN))
+#define BMM350_TEMP_SCALE (1.0f / (BMM350_TEMP_SENSITIVE * BMM350_ADC_GAIN * BMM350_LUT_GAIN * 1048576))
+
+extern const AP_HAL::HAL &hal;
+
+AP_Compass_Backend *AP_Compass_BMM350::probe(AP_HAL::OwnPtr dev,
+ bool force_external,
+ enum Rotation rotation)
+{
+ if (!dev) {
+ return nullptr;
+ }
+ AP_Compass_BMM350 *sensor = NEW_NOTHROW AP_Compass_BMM350(std::move(dev), force_external, rotation);
+ if (!sensor || !sensor->init()) {
+ delete sensor;
+ return nullptr;
+ }
+
+ return sensor;
+}
+
+AP_Compass_BMM350::AP_Compass_BMM350(AP_HAL::OwnPtr dev,
+ bool force_external,
+ enum Rotation rotation)
+ : _dev(std::move(dev))
+ , _force_external(force_external)
+ , _rotation(rotation)
+{
+}
+
+/**
+ * @brief Read out OTP(one-time programmable memory) data of sensor which is the compensation coefficients
+ * @see https://github.com/boschsensortec/BMM350-SensorAPI
+ */
+bool AP_Compass_BMM350::read_otp_data()
+{
+ uint16_t otp_data[BMM350_OTP_DATA_LENGTH];
+
+ for (uint8_t index = 0; index < BMM350_OTP_DATA_LENGTH; index++) {
+ // Set OTP address
+ if (!_dev->write_register(BMM350_REG_OTP_CMD, (BMM350_OTP_CMD_DIR_READ | index))) {
+ return false;
+ }
+
+ // Wait for OTP status be ready
+ int8_t tries = 3; // Try polling 3 times
+ while (tries--)
+ {
+ uint8_t status;
+ hal.scheduler->delay_microseconds(300);
+ // Read OTP status
+ if (!read_bytes(BMM350_REG_OTP_STATUS, &status, 1) || ((status & BMM350_OTP_STATUS_ERROR_MASK) != 0)) {
+ return false;
+ }
+ if (status & BMM350_OTP_STATUS_CMD_DONE) {
+ break;
+ }
+ }
+
+ if (tries == -1) {
+ return false;
+ }
+
+ // Read OTP data
+ be16_t reg_data;
+ if (!read_bytes(BMM350_REG_OTP_DATA_MSB, (uint8_t *)®_data, 2)) {
+ return false;
+ }
+
+ otp_data[index] = be16toh(reg_data);
+ }
+
+ // Update magnetometer offset and sensitivity data.
+ // 12-bit unsigned integer to be left-aligned in a 16-bit integer
+ _mag_comp.offset_coef.x = float(int16_t((otp_data[BMM350_MAG_OFFSET_X] & 0x0FFF) << 4) >> 4);
+ _mag_comp.offset_coef.y = float(int16_t((((otp_data[BMM350_MAG_OFFSET_X] & 0xF000) >> 4) +
+ (otp_data[BMM350_MAG_OFFSET_Y] & 0x00FF)) << 4) >> 4);
+ _mag_comp.offset_coef.z = float(int16_t(((otp_data[BMM350_MAG_OFFSET_Y] & 0x0F00) +
+ (otp_data[BMM350_MAG_OFFSET_Z] & 0x00FF)) << 4) >> 4);
+ _mag_comp.offset_coef.temp = float(int8_t(otp_data[BMM350_TEMP_OFF_SENS])) * (1.0f / 5.0f);
+
+ _mag_comp.sensit_coef.x = float(int8_t((otp_data[BMM350_MAG_SENS_X] & 0xFF00) >> 8)) * (1.0f / 256.0f);
+ _mag_comp.sensit_coef.y = float(int8_t(otp_data[BMM350_MAG_SENS_Y])) * (1.0f / 256.0f) + BMM350_SENS_CORR_Y;
+ _mag_comp.sensit_coef.z = float(int8_t((otp_data[BMM350_MAG_SENS_Z] & 0xFF00) >> 8)) * (1.0f / 256.0f);
+ _mag_comp.sensit_coef.temp = float(int8_t((otp_data[BMM350_TEMP_OFF_SENS] & 0xFF00) >> 8)) * (1.0f / 512.0f);
+
+ _mag_comp.tco.x = float(int8_t(otp_data[BMM350_MAG_TCO_X])) * (1.0f / 32.0f);
+ _mag_comp.tco.y = float(int8_t(otp_data[BMM350_MAG_TCO_Y])) * (1.0f / 32.0f);
+ _mag_comp.tco.z = float(int8_t(otp_data[BMM350_MAG_TCO_Z])) * (1.0f / 32.0f);
+
+ _mag_comp.tcs.x = float(int8_t((otp_data[BMM350_MAG_TCS_X] & 0xFF00) >> 8)) * (1.0f / 16384.0f);
+ _mag_comp.tcs.y = float(int8_t((otp_data[BMM350_MAG_TCS_Y] & 0xFF00) >> 8)) * (1.0f / 16384.0f);
+ _mag_comp.tcs.z = float(int8_t((otp_data[BMM350_MAG_TCS_Z] & 0xFF00) >> 8)) * (1.0f / 16384.0f) - BMM350_TCS_CORR_Z;
+
+ _mag_comp.t0_reading = float(int16_t(otp_data[BMM350_MAG_DUT_T_0])) * (1.0f / 512.0f) + 23.0f;
+
+ _mag_comp.cross_axis.cross_x_y = float(int8_t(otp_data[BMM350_CROSS_X_Y])) * (1.0f / 800.0f);
+ _mag_comp.cross_axis.cross_y_x = float(int8_t((otp_data[BMM350_CROSS_Y_X] & 0xFF00) >> 8)) * (1.0f / 800.0f);
+ _mag_comp.cross_axis.cross_z_x = float(int8_t(otp_data[BMM350_CROSS_Z_X])) * (1.0f / 800.0f);
+ _mag_comp.cross_axis.cross_z_y = float(int8_t((otp_data[BMM350_CROSS_Z_Y] & 0xFF00) >> 8)) * (1.0f / 800.0f);
+
+ return true;
+}
+
+/**
+ * @brief Wait PMU_CMD register operation completed. Need to specify which command just sent
+ */
+bool AP_Compass_BMM350::wait_pmu_cmd_ready(const uint8_t cmd, const uint32_t timeout)
+{
+ const uint32_t start_tick = AP_HAL::millis();
+
+ do {
+ hal.scheduler->delay(1);
+ uint8_t status;
+ if (!read_bytes(BMM350_REG_PMU_CMD_STATUS_0, &status, 1)) {
+ return false;
+ }
+ if (((status & 0x01) == 0x00) && (((status & 0xE0) >> 5) == cmd)) {
+ return true;
+ }
+ } while ((AP_HAL::millis() - start_tick) < timeout);
+
+ return false;
+}
+
+/**
+ * @brief Reset bit of magnetic register and wait for change to normal mode
+ */
+bool AP_Compass_BMM350::mag_reset_and_wait()
+{
+ uint8_t data;
+
+ // Get PMU command status 0 data
+ if (!read_bytes(BMM350_REG_PMU_CMD_STATUS_0, &data, 1)) {
+ return false;
+ }
+
+ // Check whether the power mode is Normal
+ if (data & 0x08) {
+ // Set PMU command to suspend mode
+ if (!_dev->write_register(BMM350_REG_PMU_CMD, BMM350_PMU_CMD_SUSPEND_MODE)) {
+ return false;
+ }
+ wait_pmu_cmd_ready(BMM350_PMU_CMD_SUSPEND_MODE, 6);
+ }
+
+ // Set BR(bit reset) to PMU_CMD register
+ // In offical example, it wait for 14ms. But it may not be enough, so we wait an extra 5ms
+ if (!_dev->write_register(BMM350_REG_PMU_CMD, BMM350_PMU_CMD_BR) ||
+ !wait_pmu_cmd_ready(BMM350_PMU_CMD_BR, 19)) {
+ return false;
+ }
+
+ // Set FGR(flux-guide reset) to PMU_CMD register
+ // 18ms got from offical example, we wait an extra 5ms
+ if (!_dev->write_register(BMM350_REG_PMU_CMD, BMM350_PMU_CMD_FGR) ||
+ !wait_pmu_cmd_ready(BMM350_PMU_CMD_FGR, 23)) {
+ return false;
+ }
+
+ // Switch to normal mode
+ if (!set_power_mode(POWER_MODE_NORMAL)) {
+ return false;
+ }
+
+ return true;
+}
+
+/**
+ * @brief Switch sensor power mode
+ */
+bool AP_Compass_BMM350::set_power_mode(const enum power_mode mode)
+{
+ uint8_t pmu_cmd;
+
+ // Get PMU register data as current mode
+ if (!read_bytes(BMM350_REG_PMU_CMD, &pmu_cmd, 1)) {
+ return false;
+ }
+
+ if (pmu_cmd == BMM350_PMU_CMD_NORMAL_MODE || pmu_cmd == BMM350_PMU_CMD_UPD_OAE) {
+ // Set PMU command to suspend mode
+ if (!_dev->write_register(BMM350_REG_PMU_CMD, POWER_MODE_SUSPEND)) {
+ return false;
+ }
+ // Wait for sensor switch to suspend mode
+ // Wait for maximum 6ms, it got from the official example, not explained in datasheet
+ wait_pmu_cmd_ready(POWER_MODE_SUSPEND, 6);
+ }
+
+ // Set PMU command to target mode
+ if (!_dev->write_register(BMM350_REG_PMU_CMD, mode)) {
+ return false;
+ }
+
+ // Wait for mode change. When switch from suspend mode to normal mode, we wait for at most 38ms.
+ // It got from official example too
+ wait_pmu_cmd_ready(mode, 38);
+
+ return true;
+}
+
+/**
+ * @brief Read bytes from sensor
+ */
+bool AP_Compass_BMM350::read_bytes(const uint8_t reg, uint8_t *out, const uint16_t read_len)
+{
+ uint8_t data[read_len + 2];
+
+ if (!_dev->read_registers(reg, data, read_len + 2)) {
+ return false;
+ }
+
+ memcpy(out, &data[2], read_len);
+
+ return true;
+}
+
+bool AP_Compass_BMM350::init()
+{
+ _dev->get_semaphore()->take_blocking();
+
+ // 10 retries for init
+ _dev->set_retries(10);
+
+ // Use checked registers to cope with bus errors
+ _dev->setup_checked_registers(4);
+
+ int8_t boot_retries = 5;
+ while (boot_retries--) {
+ // Soft reset
+ if (!_dev->write_register(BMM350_REG_CMD, BMM350_CMD_SOFTRESET)) {
+ continue;
+ }
+ hal.scheduler->delay(24); // Wait 24ms for soft reset complete, it got from offical example
+
+ // Read and verify chip ID
+ uint8_t chip_id;
+ if (!read_bytes(BMM350_REG_CHIP_ID, &chip_id, 1)) {
+ continue;
+ }
+ if (chip_id == BMM350_CHIP_ID) {
+ break;
+ }
+ }
+
+ if (boot_retries == -1) {
+ goto err;
+ }
+
+ // Read out OTP data
+ if (!read_otp_data()) {
+ goto err;
+ }
+
+ // Power off OTP
+ if (!_dev->write_register(BMM350_REG_OTP_CMD, BMM350_OTP_CMD_PWR_OFF_OTP)) {
+ goto err;
+ }
+
+ // Magnetic reset
+ if (!mag_reset_and_wait()) {
+ goto err;
+ }
+
+ // Configure interrupt settings and enable DRDY
+ // Set INT mode as PULSED | active_high polarity | PUSH_PULL | unmap | DRDY interrupt
+ if (!_dev->write_register(BMM350_REG_INT_CTRL, (BMM350_INT_MODE_PULSED |
+ BMM350_INT_POL_ACTIVE_HIGH |
+ BMM350_INT_OD_PUSHPULL |
+ BMM350_INT_OUTPUT_DISABLE |
+ BMM350_INT_DRDY_EN))) {
+ goto err;
+ }
+
+ // Setup ODR and performance. 100Hz ODR and 4 average for lownoise
+ if (!_dev->write_register(BMM350_REG_PMU_CMD_AGGR_SET, (BMM350_AVERAGING_4 | BMM350_ODR_100HZ))) {
+ goto err;
+ }
+
+ // Update ODR and avg parameter
+ if (!_dev->write_register(BMM350_REG_PMU_CMD, BMM350_PMU_CMD_UPD_OAE)) {
+ goto err;
+ }
+ // Wait at most 20ms for update ODR and avg paramter
+ wait_pmu_cmd_ready(BMM350_PMU_CMD_UPD_OAE, 20);
+
+ // Enable measurement of 3 axis
+ if (!_dev->write_register(BMM350_REG_PMU_CMD_AXIS_EN, 0x07)) {
+ goto err;
+ }
+
+ // Switch power mode to normal mode
+ if (!set_power_mode(POWER_MODE_NORMAL)) {
+ goto err;
+ }
+
+ // Lower retries for run
+ _dev->set_retries(3);
+
+ _dev->get_semaphore()->give();
+
+ /* Register the compass instance in the frontend */
+ _dev->set_device_type(DEVTYPE_BMM350);
+ if (!register_compass(_dev->get_bus_id(), _compass_instance)) {
+ return false;
+ }
+ set_dev_id(_compass_instance, _dev->get_bus_id());
+
+ // printf("BMM350: Found at address 0x%x as compass %u\n", _dev->get_bus_address(), _compass_instance);
+
+ set_rotation(_compass_instance, _rotation);
+
+ if (_force_external) {
+ set_external(_compass_instance, true);
+ }
+
+ // Call timer() at 100Hz
+ _dev->register_periodic_callback(1000000U/100U, FUNCTOR_BIND_MEMBER(&AP_Compass_BMM350::timer, void));
+
+ return true;
+
+err:
+ _dev->get_semaphore()->give();
+ return false;
+}
+
+void AP_Compass_BMM350::timer()
+{
+ struct PACKED {
+ uint8_t magx[3];
+ uint8_t magy[3];
+ uint8_t magz[3];
+ uint8_t temp[3];
+ } data;
+
+ // Read out measurement data
+ if (!read_bytes(BMM350_REG_MAG_X_XLSB, (uint8_t *)&data, sizeof(data))) {
+ return;
+ }
+
+ // Converts 24-bit raw data to signed integer value
+ const int32_t magx_raw = int32_t(((uint32_t)data.magx[0] << 8) + ((uint32_t)data.magx[1] << 16) + ((uint32_t)data.magx[2] << 24)) >> 8;
+ const int32_t magy_raw = int32_t(((uint32_t)data.magy[0] << 8) + ((uint32_t)data.magy[1] << 16) + ((uint32_t)data.magy[2] << 24)) >> 8;
+ const int32_t magz_raw = int32_t(((uint32_t)data.magz[0] << 8) + ((uint32_t)data.magz[1] << 16) + ((uint32_t)data.magz[2] << 24)) >> 8;
+ const int32_t temp_raw = int32_t(((uint32_t)data.temp[0] << 8) + ((uint32_t)data.temp[1] << 16) + ((uint32_t)data.temp[2] << 24)) >> 8;
+
+ // Convert mag lsb to uT and temp lsb to degC
+ float magx = (float)magx_raw * BMM350_XY_SCALE;
+ float magy = (float)magy_raw * BMM350_XY_SCALE;
+ float magz = (float)magz_raw * BMM350_Z_SCALE;
+ float temp = (float)temp_raw * BMM350_TEMP_SCALE;
+
+ if (temp > 0.0f) {
+ temp -= 25.49f;
+ } else if (temp < 0.0f) {
+ temp += 25.49f;
+ }
+
+ // Apply compensation
+ temp = ((1 + _mag_comp.sensit_coef.temp) * temp) + _mag_comp.offset_coef.temp;
+ // Compensate raw magnetic data
+ magx = ((1 + _mag_comp.sensit_coef.x) * magx) + _mag_comp.offset_coef.x + (_mag_comp.tco.x * (temp - _mag_comp.t0_reading));
+ magx /= 1 + _mag_comp.tcs.x * (temp - _mag_comp.t0_reading);
+
+ magy = ((1 + _mag_comp.sensit_coef.y) * magy) + _mag_comp.offset_coef.y + (_mag_comp.tco.y * (temp - _mag_comp.t0_reading));
+ magy /= 1 + _mag_comp.tcs.y * (temp - _mag_comp.t0_reading);
+
+ magz = ((1 + _mag_comp.sensit_coef.z) * magz) + _mag_comp.offset_coef.z + (_mag_comp.tco.z * (temp - _mag_comp.t0_reading));
+ magz /= 1 + _mag_comp.tcs.z * (temp - _mag_comp.t0_reading);
+
+ const float cr_ax_comp_x = (magx - _mag_comp.cross_axis.cross_x_y * magy) / (1 - _mag_comp.cross_axis.cross_y_x * _mag_comp.cross_axis.cross_x_y);
+ const float cr_ax_comp_y = (magy - _mag_comp.cross_axis.cross_y_x * magx) / (1 - _mag_comp.cross_axis.cross_y_x * _mag_comp.cross_axis.cross_x_y);
+ const float cr_ax_comp_z = (magz + (magx * (_mag_comp.cross_axis.cross_y_x * _mag_comp.cross_axis.cross_z_y - _mag_comp.cross_axis.cross_z_x) -
+ magy * (_mag_comp.cross_axis.cross_z_y - _mag_comp.cross_axis.cross_x_y * _mag_comp.cross_axis.cross_z_x)) /
+ (1 - _mag_comp.cross_axis.cross_y_x * _mag_comp.cross_axis.cross_x_y));
+
+ // Store in field vector and convert uT to milligauss
+ Vector3f field { cr_ax_comp_x * 10.0f, cr_ax_comp_y * 10.0f, cr_ax_comp_z * 10.0f };
+ accumulate_sample(field, _compass_instance);
+}
+
+void AP_Compass_BMM350::read()
+{
+ drain_accumulated_samples(_compass_instance);
+}
+
+#endif // AP_COMPASS_BMM350_ENABLED
diff --git a/libraries/AP_Compass/AP_Compass_BMM350.h b/libraries/AP_Compass/AP_Compass_BMM350.h
new file mode 100644
index 0000000000000..4cf5fdf55507d
--- /dev/null
+++ b/libraries/AP_Compass/AP_Compass_BMM350.h
@@ -0,0 +1,112 @@
+/*
+ * This file is free software: you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License as published by the
+ * Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This file is distributed in the hope that it will be useful, but
+ * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
+ * See the GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License along
+ * with this program. If not, see .
+ */
+
+#pragma once
+
+#include "AP_Compass_config.h"
+
+#if AP_COMPASS_BMM350_ENABLED
+
+#include
+#include
+#include
+#include
+
+#include "AP_Compass.h"
+#include "AP_Compass_Backend.h"
+
+#define BMM350_I2C_ADDR_MIN 0x14
+#define BMM350_I2C_ADDR_MAX 0x17
+
+
+class AP_Compass_BMM350 : public AP_Compass_Backend
+{
+public:
+ static AP_Compass_Backend *probe(AP_HAL::OwnPtr dev,
+ bool force_external,
+ enum Rotation rotation);
+
+ void read() override;
+
+ static constexpr const char *name = "BMM350";
+
+private:
+ AP_Compass_BMM350(AP_HAL::OwnPtr dev,
+ bool force_external,
+ enum Rotation rotation);
+
+ AP_HAL::OwnPtr _dev;
+
+ /**
+ * @brief BMM350 offset/sensitivity coefficient structure
+ */
+ struct vector4f
+ {
+ float x; // x axis
+ float y; // y axis
+ float z; // z axis
+ float temp; // Temperature
+ };
+
+ /**
+ * @brief BMM350 magnetometer cross axis compensation structure
+ */
+ struct cross_axis
+ {
+ float cross_x_y;
+ float cross_y_x;
+ float cross_z_x;
+ float cross_z_y;
+ };
+
+ /**
+ * @brief BMM350 magnetometer compensate structure
+ */
+ struct mag_compensate
+ {
+ struct vector4f offset_coef; // Offset coefficient
+ struct vector4f sensit_coef; // Sensitivity coefficient
+ Vector3f tco; // Temperature coefficient of the offset
+ Vector3f tcs; // Temperature coefficient of the sensitivity
+ float t0_reading; // Initialize T0_reading parameter
+ struct cross_axis cross_axis; // Cross axis compensation
+ };
+
+ enum power_mode
+ {
+ POWER_MODE_SUSPEND = 0,
+ POWER_MODE_NORMAL = 1,
+ POWER_MODE_FORCED = 3,
+ POWER_MODE_FORCED_FAST = 4
+ };
+
+ /**
+ * Device periodic callback to read data from the sensor.
+ */
+ bool init();
+ void timer();
+ bool read_otp_data();
+ bool wait_pmu_cmd_ready(const uint8_t cmd, const uint32_t timeout);
+ bool mag_reset_and_wait();
+ bool set_power_mode(const enum power_mode mode);
+ bool read_bytes(const uint8_t reg, uint8_t *out, const uint16_t read_len);
+
+ uint8_t _compass_instance;
+ bool _force_external;
+ enum Rotation _rotation;
+ struct mag_compensate _mag_comp; // Structure for mag compensate
+};
+
+#endif // AP_COMPASS_BMM350_ENABLED
diff --git a/libraries/AP_Compass/AP_Compass_Backend.h b/libraries/AP_Compass/AP_Compass_Backend.h
index 79bfe1f66fd7b..68c0959cccec6 100644
--- a/libraries/AP_Compass/AP_Compass_Backend.h
+++ b/libraries/AP_Compass/AP_Compass_Backend.h
@@ -74,6 +74,7 @@ class AP_Compass_Backend
DEVTYPE_AK09918 = 0x14,
DEVTYPE_AK09915 = 0x15,
DEVTYPE_QMC5883P = 0x16,
+ DEVTYPE_BMM350 = 0x17,
};
#if AP_COMPASS_MSP_ENABLED
diff --git a/libraries/AP_Compass/AP_Compass_config.h b/libraries/AP_Compass/AP_Compass_config.h
index c6c236db17d73..9d135be9dda0c 100644
--- a/libraries/AP_Compass/AP_Compass_config.h
+++ b/libraries/AP_Compass/AP_Compass_config.h
@@ -76,6 +76,10 @@
#define AP_COMPASS_BMM150_DETECT_BACKENDS_ENABLED AP_COMPASS_BMM150_ENABLED
#endif
+#ifndef AP_COMPASS_BMM350_ENABLED
+#define AP_COMPASS_BMM350_ENABLED AP_COMPASS_I2C_BACKEND_DEFAULT_ENABLED
+#endif
+
#ifndef AP_COMPASS_HMC5843_ENABLED
#define AP_COMPASS_HMC5843_ENABLED AP_COMPASS_I2C_BACKEND_DEFAULT_ENABLED
#endif