From d0f01e53a43b0a53697dcb02b204ba00eca9c482 Mon Sep 17 00:00:00 2001 From: Daniel Peter Chokola Date: Fri, 22 Jan 2021 15:02:23 -0500 Subject: [PATCH] implement ST-like API for EM7180 pared down EM7180 to just enough to set some config parameters and get back real data LIS2MDL, LPS22HB, and LSM6DSM are non-functional --- Drivers/EM7180/Inc/em7180.h | 119 +---- Drivers/EM7180/Inc/em7180_common.h | 38 +- Drivers/EM7180/Src/em7180.c | 794 +++++++++++------------------ Drivers/EM7180/Src/em7180_common.c | 36 ++ Drivers/EM7180/Src/lis2mdl.c | 32 +- Drivers/EM7180/Src/lps22hb.c | 15 +- Drivers/EM7180/Src/lsm6dsm.c | 40 +- 7 files changed, 421 insertions(+), 653 deletions(-) create mode 100644 Drivers/EM7180/Src/em7180_common.c diff --git a/Drivers/EM7180/Inc/em7180.h b/Drivers/EM7180/Inc/em7180.h index a34b066..09872bd 100644 --- a/Drivers/EM7180/Inc/em7180.h +++ b/Drivers/EM7180/Inc/em7180.h @@ -17,92 +17,17 @@ /* Includes */ #include -#include "i2c.h" -#include "lsm6dsm.h" -#include "lps22hb.h" -#include "lis2mdl.h" +#include "em7180_common.h" /* Definitions */ -/* - * EM7180 SENtral register map - * see http://www.emdeveloper.com/downloads/7180/EMSentral_EM7180_Register_Map_v1_3.pdf - */ -#define EM7180_QX 0x00 // this is a 32-bit normalized floating point number read from registers 0x00-03 -#define EM7180_QY 0x04 // this is a 32-bit normalized floating point number read from registers 0x04-07 -#define EM7180_QZ 0x08 // this is a 32-bit normalized floating point number read from registers 0x08-0B -#define EM7180_QW 0x0C // this is a 32-bit normalized floating point number read from registers 0x0C-0F -#define EM7180_QTIME 0x10 // this is a 16-bit unsigned integer read from registers 0x10-11 -#define EM7180_MX 0x12 // int16_t from registers 0x12-13 -#define EM7180_MY 0x14 // int16_t from registers 0x14-15 -#define EM7180_MZ 0x16 // int16_t from registers 0x16-17 -#define EM7180_MTIME 0x18 // uint16_t from registers 0x18-19 -#define EM7180_AX 0x1A // int16_t from registers 0x1A-1B -#define EM7180_AY 0x1C // int16_t from registers 0x1C-1D -#define EM7180_AZ 0x1E // int16_t from registers 0x1E-1F -#define EM7180_ATIME 0x20 // uint16_t from registers 0x20-21 -#define EM7180_GX 0x22 // int16_t from registers 0x22-23 -#define EM7180_GY 0x24 // int16_t from registers 0x24-25 -#define EM7180_GZ 0x26 // int16_t from registers 0x26-27 -#define EM7180_GTIME 0x28 // uint16_t from registers 0x28-29 -#define EM7180_Baro 0x2A // start of two-byte MS5637 pressure data, 16-bit signed interger -#define EM7180_BaroTIME 0x2C // start of two-byte MS5637 pressure timestamp, 16-bit unsigned -#define EM7180_Temp 0x2E // start of two-byte MS5637 temperature data, 16-bit signed interger -#define EM7180_TempTIME 0x30 // start of two-byte MS5637 temperature timestamp, 16-bit unsigned -#define EM7180_QRateDivisor 0x32 // uint8_t -#define EM7180_EnableEvents 0x33 -#define EM7180_HostControl 0x34 -#define EM7180_EventStatus 0x35 -#define EM7180_SensorStatus 0x36 -#define EM7180_SentralStatus 0x37 -#define EM7180_AlgorithmStatus 0x38 -#define EM7180_FeatureFlags 0x39 -#define EM7180_ParamAcknowledge 0x3A -#define EM7180_SavedParamByte0 0x3B -#define EM7180_SavedParamByte1 0x3C -#define EM7180_SavedParamByte2 0x3D -#define EM7180_SavedParamByte3 0x3E -#define EM7180_ActualMagRate 0x45 -#define EM7180_ActualAccelRate 0x46 -#define EM7180_ActualGyroRate 0x47 -#define EM7180_ActualBaroRate 0x48 -#define EM7180_ActualTempRate 0x49 -#define EM7180_ErrorRegister 0x50 -#define EM7180_AlgorithmControl 0x54 -#define EM7180_MagRate 0x55 -#define EM7180_AccelRate 0x56 -#define EM7180_GyroRate 0x57 -#define EM7180_BaroRate 0x58 -#define EM7180_TempRate 0x59 -#define EM7180_LoadParamByte0 0x60 -#define EM7180_LoadParamByte1 0x61 -#define EM7180_LoadParamByte2 0x62 -#define EM7180_LoadParamByte3 0x63 -#define EM7180_ParamRequest 0x64 -#define EM7180_ROMVersion1 0x70 -#define EM7180_ROMVersion2 0x71 -#define EM7180_RAMVersion1 0x72 -#define EM7180_RAMVersion2 0x73 -#define EM7180_ProductID 0x90 -#define EM7180_RevisionID 0x91 -#define EM7180_RunStatus 0x92 -#define EM7180_UploadAddress 0x94 // uint16_t registers 0x94 (MSB)-5(LSB) -#define EM7180_UploadData 0x96 -#define EM7180_CRCHost 0x97 // uint32_t from registers 0x97-9A -#define EM7180_ResetRequest 0x9B -#define EM7180_PassThruStatus 0x9E -#define EM7180_PassThruControl 0xA0 - -#define EM7180_ADDRESS 0x28 // Address of the EM7180 SENtral sensor hub -#define M24512DFM_DATA_ADDRESS 0x50 // Address of the 500 page M24512DFM EEPROM data buffer, 1024 bits (128 8-bit bytes) per page -#define M24512DFM_IDPAGE_ADDRESS 0x58 // Address of the single M24512DFM lockable EEPROM ID page +#define EM7180_OK (0) +#define EM7180_BAD_ARG (1 << 0) +#define EM7180_BAD_COMM (1 << 1) /* Data Structures */ -typedef struct em7180_s +typedef int32_t em7180_status_t; +typedef struct em7180_init_s { - I2C_HandleTypeDef *hi2c; - lsm6dsm_t *lsm6dsm; - lis2mdl_t *lis2mdl; - lps22hb_t *lps22hb; uint16_t acc_fs; uint16_t gyro_fs; uint16_t mag_fs; @@ -111,18 +36,30 @@ typedef struct em7180_s uint8_t acc_rate; uint8_t gyro_rate; uint8_t baro_rate; +} em7180_init_t; +typedef struct em7180_s +{ + em7180_init_t *init; + uint8_t i2c_addr; + delay_func_t delay_func; + i2c_read_func_t i2c_read_func; + i2c_write_func_t i2c_write_func; } em7180_t; /* Function Prototypes */ -void em7180_init(em7180_t *em7180, I2C_HandleTypeDef *hi2c, lsm6dsm_t *lsm6dsm, - lis2mdl_t *lis2mdl, lps22hb_t *lps22hb, uint16_t acc_fs, - uint16_t gyro_fs, uint16_t mag_fs, uint8_t q_rate_div, - uint8_t mag_rate, uint8_t acc_rate, uint8_t gyro_rate, - uint8_t baro_rate); -void em7180_config(em7180_t *em7180); -void em7180_gyro_set_fs(em7180_t *em7180, uint16_t gyro_fs); -void em7180_mag_acc_set_fs(em7180_t *em7180, uint16_t mag_fs, uint16_t acc_fs); -void em7180_set_integer_param(em7180_t *em7180, uint8_t param, - uint32_t param_val); +em7180_status_t em7180_init(em7180_t *em7180, em7180_init_t *init); +void em7180_set_delay_cb(em7180_t *em7180, delay_func_t delay_func); +void em7180_set_i2c_cbs(em7180_t *em7180, i2c_read_func_t i2c_read_func, + i2c_write_func_t i2c_write_func, uint8_t dev_addr); +em7180_status_t em7180_config(em7180_t *em7180); +void em7180_reset(em7180_t *em7180); +void em7180_passthrough_enable(em7180_t *em7180); +void em7180_passthrough_disable(em7180_t *em7180); +em7180_status_t em7180_quatdata_get(em7180_t *em7180, float *destination); +em7180_status_t em7180_acceldata_get(em7180_t *em7180, int16_t *destination); +em7180_status_t em7180_gyrodata_get(em7180_t *em7180, int16_t *destination); +em7180_status_t em7180_magdata_get(em7180_t *em7180, int16_t *destination); +em7180_status_t em7180_barodata_get(em7180_t *em7180, int16_t *destination); +em7180_status_t em7180_tempdata_get(em7180_t *em7180, int16_t *destination); #endif /* EM7180_H_ */ diff --git a/Drivers/EM7180/Inc/em7180_common.h b/Drivers/EM7180/Inc/em7180_common.h index 1e395bc..0a1687f 100644 --- a/Drivers/EM7180/Inc/em7180_common.h +++ b/Drivers/EM7180/Inc/em7180_common.h @@ -12,7 +12,6 @@ /* Includes */ #include -#include "i2c.h" /* Definitions */ #define return_if_fail(cond) \ @@ -26,28 +25,21 @@ return (val); \ } -/* Function Prototypes */ -inline __attribute__((always_inline)) HAL_StatusTypeDef i2c_write_byte( - I2C_HandleTypeDef *hi2c, uint16_t addr, uint16_t sub_addr, uint8_t data) -{ - return HAL_I2C_Mem_Write(hi2c, addr << 1, sub_addr, 1, &data, 1, - HAL_MAX_DELAY); -} -inline __attribute__((always_inline)) uint8_t i2c_read_byte( - I2C_HandleTypeDef *hi2c, uint16_t addr, uint16_t sub_addr) -{ - uint8_t temp; - - HAL_I2C_Mem_Read(hi2c, addr << 1, sub_addr, 1, &temp, 1, HAL_MAX_DELAY); +/* Data Structures */ +typedef void (*delay_func_t)(uint32_t); +typedef int32_t (*i2c_write_func_t)(uint16_t, uint16_t, uint16_t, uint8_t*, + uint16_t); +typedef int32_t (*i2c_read_func_t)(uint16_t, uint16_t, uint16_t, uint8_t*, + uint16_t); - return temp; -} -inline __attribute__((always_inline)) HAL_StatusTypeDef i2c_read( - I2C_HandleTypeDef *hi2c, uint16_t addr, uint16_t sub_addr, uint8_t *data, - uint16_t len) -{ - return HAL_I2C_Mem_Read(hi2c, addr << 1, sub_addr, 1, data, len, - HAL_MAX_DELAY); -} +/* Function Prototypes */ +int32_t i2c_read_byte(i2c_read_func_t read, uint16_t dev_addr, uint16_t addr, + uint8_t *byte); +int32_t i2c_write_byte(i2c_write_func_t write, uint16_t dev_addr, uint16_t addr, + uint8_t byte); +int32_t i2c_read(i2c_read_func_t read, uint16_t dev_addr, uint16_t addr, + uint8_t *data, uint16_t len); +int32_t i2c_write(i2c_write_func_t write, uint16_t dev_addr, uint16_t addr, + uint8_t *data, uint16_t len); #endif /* EM7180_COMMON_h */ diff --git a/Drivers/EM7180/Src/em7180.c b/Drivers/EM7180/Src/em7180.c index 4cc33a0..17f2247 100644 --- a/Drivers/EM7180/Src/em7180.c +++ b/Drivers/EM7180/Src/em7180.c @@ -14,510 +14,366 @@ /* Includes */ #include -#include -#include +#include #include "em7180_common.h" #include "em7180.h" -#include "i2c.h" +/* Definitions */ +/* + * EM7180 SENtral register map + * see http://www.emdeveloper.com/downloads/7180/EMSentral_EM7180_Register_Map_v1_3.pdf + */ +#define EM7180_QX 0x00 /* this is a 32-bit normalized floating point number read from registers 0x00-03 */ +#define EM7180_QY 0x04 /* this is a 32-bit normalized floating point number read from registers 0x04-07 */ +#define EM7180_QZ 0x08 /* this is a 32-bit normalized floating point number read from registers 0x08-0B */ +#define EM7180_QW 0x0C /* this is a 32-bit normalized floating point number read from registers 0x0C-0F */ +#define EM7180_QTIME 0x10 /* uint16_t from registers 0x10-11 */ +#define EM7180_MX 0x12 /* int16_t from registers 0x12-13 */ +#define EM7180_MY 0x14 /* int16_t from registers 0x14-15 */ +#define EM7180_MZ 0x16 /* int16_t from registers 0x16-17 */ +#define EM7180_MTIME 0x18 /* uint16_t from registers 0x18-19 */ +#define EM7180_AX 0x1A /* int16_t from registers 0x1A-1B */ +#define EM7180_AY 0x1C /* int16_t from registers 0x1C-1D */ +#define EM7180_AZ 0x1E /* int16_t from registers 0x1E-1F */ +#define EM7180_ATIME 0x20 /* uint16_t from registers 0x20-21 */ +#define EM7180_GX 0x22 /* int16_t from registers 0x22-23 */ +#define EM7180_GY 0x24 /* int16_t from registers 0x24-25 */ +#define EM7180_GZ 0x26 /* int16_t from registers 0x26-27 */ +#define EM7180_GTIME 0x28 /* uint16_t from registers 0x28-29 */ +#define EM7180_Baro 0x2A /* start of two-byte MS5637 pressure data, 16-bit signed interger */ +#define EM7180_BaroTIME 0x2C /* start of two-byte MS5637 pressure timestamp, 16-bit unsigned */ +#define EM7180_Temp 0x2E /* start of two-byte MS5637 temperature data, 16-bit signed interger */ +#define EM7180_TempTIME 0x30 /* start of two-byte MS5637 temperature timestamp, 16-bit unsigned */ +#define EM7180_QRateDivisor 0x32 /* uint8_t */ +#define EM7180_EnableEvents 0x33 +#define EM7180_HostControl 0x34 +#define EM7180_EventStatus 0x35 +#define EM7180_SensorStatus 0x36 +#define EM7180_SentralStatus 0x37 +#define EM7180_AlgorithmStatus 0x38 +#define EM7180_FeatureFlags 0x39 +#define EM7180_ParamAcknowledge 0x3A +#define EM7180_SavedParamByte0 0x3B +#define EM7180_SavedParamByte1 0x3C +#define EM7180_SavedParamByte2 0x3D +#define EM7180_SavedParamByte3 0x3E +#define EM7180_ActualMagRate 0x45 +#define EM7180_ActualAccelRate 0x46 +#define EM7180_ActualGyroRate 0x47 +#define EM7180_ActualBaroRate 0x48 +#define EM7180_ActualTempRate 0x49 +#define EM7180_ErrorRegister 0x50 +#define EM7180_AlgorithmControl 0x54 +#define EM7180_MagRate 0x55 +#define EM7180_AccelRate 0x56 +#define EM7180_GyroRate 0x57 +#define EM7180_BaroRate 0x58 +#define EM7180_TempRate 0x59 +#define EM7180_LoadParamByte0 0x60 +#define EM7180_LoadParamByte1 0x61 +#define EM7180_LoadParamByte2 0x62 +#define EM7180_LoadParamByte3 0x63 +#define EM7180_ParamRequest 0x64 +#define EM7180_ROMVersion1 0x70 +#define EM7180_ROMVersion2 0x71 +#define EM7180_RAMVersion1 0x72 +#define EM7180_RAMVersion2 0x73 +#define EM7180_ProductID 0x90 +#define EM7180_RevisionID 0x91 +#define EM7180_RunStatus 0x92 +#define EM7180_UploadAddress 0x94 /* uint16_t registers 0x94 (MSB)-5(LSB) */ +#define EM7180_UploadData 0x96 +#define EM7180_CRCHost 0x97 /* uint32_t from registers 0x97-9A */ +#define EM7180_ResetRequest 0x9B +#define EM7180_PassThroughStatus 0x9E +#define EM7180_PassThroughControl 0xA0 + +/* Macros */ +#define em7180_read_byte(addr, byte) i2c_read_byte((em7180->i2c_read_func), (em7180->i2c_addr), (addr), (byte)) +#define em7180_write_byte(addr, byte) i2c_write_byte((em7180->i2c_write_func), (em7180->i2c_addr), (addr), (byte)) +#define em7180_read(addr, data, len) i2c_read((em7180->i2c_read_func), (em7180->i2c_addr), (addr), (data), (len)) +#define em7180_write(addr, data, len) i2c_write((em7180->i2c_write_func), (em7180->i2c_addr), (addr), (data), (len)) /* Data Structures */ /* Private Global Variables */ /* Function Prototypes */ -static void em7180_passthrough(em7180_t *em7180); static float uint32_reg_to_float(uint8_t *buf); -static void float_to_bytes(float param_val, uint8_t *buf); /* Function Definitions */ -void em7180_init(em7180_t *em7180, I2C_HandleTypeDef *hi2c, lsm6dsm_t *lsm6dsm, - lis2mdl_t *lis2mdl, lps22hb_t *lps22hb, uint16_t acc_fs, - uint16_t gyro_fs, uint16_t mag_fs, uint8_t q_rate_div, - uint8_t mag_rate, uint8_t acc_rate, uint8_t gyro_rate, - uint8_t baro_rate) +em7180_status_t em7180_init(em7180_t *em7180, em7180_init_t *init) { - return_if_fail(em7180); + int8_t *ptr = (int8_t*) em7180; + size_t i; - em7180->hi2c = hi2c; - em7180->lsm6dsm = lsm6dsm; - em7180->lis2mdl = lis2mdl; - em7180->lps22hb = lps22hb; - em7180->acc_fs = acc_fs; - em7180->gyro_fs = gyro_fs; - em7180->mag_fs = mag_fs; - em7180->q_rate_div = q_rate_div; - em7180->mag_rate = mag_rate; - em7180->acc_rate = acc_rate; - em7180->gyro_rate = gyro_rate; - em7180->baro_rate = baro_rate; - - /* configure the EM7180 */ - em7180_config(em7180); - /* enter passthrough mode */ - em7180_passthrough(em7180); - /* and configure the devices on the slave bus */ - if(em7180->lsm6dsm) - { - lsm6dsm_config(em7180->lsm6dsm, em7180->hi2c); - } - if(em7180->lis2mdl) - { - lis2mdl_config(em7180->lis2mdl, em7180->hi2c); - } - if(em7180->lps22hb) - { - lps22hb_config(em7180->lps22hb, em7180->hi2c); - } -} + return_val_if_fail(em7180, EM7180_BAD_ARG); + return_val_if_fail(init, EM7180_BAD_ARG); -void em7180_config(em7180_t *em7180) -{ - uint8_t runStatus; - uint8_t algoStatus; - uint8_t passthruStatus; - uint8_t eventStatus; - uint8_t sensorStatus; - - // Enter EM7180 initialized state - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_HostControl, 0x00); // set SENtral in initialized state to configure registers - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_PassThruControl, 0x00); // make sure pass through mode is off - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_HostControl, 0x01); // Force initialize - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_HostControl, 0x00); // set SENtral in initialized state to configure registers - - /* Legacy MPU6250 stuff, it seems - // Setup LPF bandwidth (BEFORE setting ODR's) - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ACC_LPF_BW, accBW); // accBW = 3 = 41Hz - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_GYRO_LPF_BW, gyroBW); // gyroBW = 3 = 41Hz */ - // Set accel/gyro/mag desired ODR rates - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_QRateDivisor, - em7180->q_rate_div); // quat rate = gyroRt/(1 QRTDiv) - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_MagRate, - em7180->mag_rate); // 0x64 = 100 Hz - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AccelRate, - em7180->acc_rate); // 200/10 Hz, 0x14 = 200 Hz - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_GyroRate, - em7180->gyro_rate); // 200/10 Hz, 0x14 = 200 Hz - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_BaroRate, - 0x80 | em7180->baro_rate); // set enable bit and set Baro rate to 25 Hz, rate = baroRt/2, 0x32 = 25 Hz - - // Configure operating mode - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AlgorithmControl, 0x00); // read scale sensor data - // Enable interrupt to host upon certain events - // choose host interrupts when any sensor updated (0x40), new gyro data (0x20), new accel data (0x10), - // new mag data (0x08), quaternions updated (0x04), an error occurs (0x02), or the SENtral needs to be reset(0x01) - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_EnableEvents, 0x07); - // Enable EM7180 run mode - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_HostControl, 0x01); // set SENtral in normal run mode - HAL_Delay(100); - - // EM7180 parameter adjustments - /* Serial.println("Beginning Parameter Adjustments"); */ - - // Disable stillness mode for balancing robot application - em7180_set_integer_param(em7180, 0x49, 0x00); - - // Write desired sensor full scale ranges to the EM7180 - em7180_mag_acc_set_fs(em7180, em7180->mag_fs, em7180->acc_fs); // 1000 uT == 0x3E8, 8 g == 0x08 - em7180_gyro_set_fs(em7180, em7180->gyro_fs); // 2000 dps == 0x7D0 - - // Read EM7180 status - runStatus = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_RunStatus); - if(runStatus & 0x01) - { - /* Serial.println(" EM7180 run status = normal mode"); */ - } - algoStatus = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_AlgorithmStatus); - if(algoStatus & 0x01) - { - /* Serial.println(" EM7180 standby status"); */ - } - if(algoStatus & 0x02) - { - /* Serial.println(" EM7180 algorithm slow"); */ - } - if(algoStatus & 0x04) - { - /* Serial.println(" EM7180 in stillness mode"); */ - } - if(algoStatus & 0x08) - { - /* Serial.println(" EM7180 mag calibration completed"); */ - } - if(algoStatus & 0x10) + /* zero em7180_t struct */ + for(i = 0; i < sizeof(em7180_t); i++) { - /* Serial.println(" EM7180 magnetic anomaly detected"); */ - } - if(algoStatus & 0x20) - { - /* Serial.println(" EM7180 unreliable sensor data"); */ - } - passthruStatus = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_PassThruStatus); - if(passthruStatus & 0x01) - { - /* Serial.print(" EM7180 in passthru mode!"); */ - } - eventStatus = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_EventStatus); - if(eventStatus & 0x01) - { - /* Serial.println(" EM7180 CPU reset"); */ - } - if(eventStatus & 0x02) - { - /* Serial.println(" EM7180 Error"); */ - } - if(eventStatus & 0x04) - { - /* Serial.println(" EM7180 new quaternion result"); */ - } - if(eventStatus & 0x08) - { - /* Serial.println(" EM7180 new mag result"); */ - } - if(eventStatus & 0x10) - { - /* Serial.println(" EM7180 new accel result"); */ - } - if(eventStatus & 0x20) - { - /* Serial.println(" EM7180 new gyro result"); */ + *ptr++ = 0; } - HAL_Delay(1000); // give some time to read the screen + em7180->init = init; - // Check sensor status - sensorStatus = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_SensorStatus); - /* Serial.print(" EM7180 sensor status = "); */ - /* Serial.println(sensorStatus); */ - if(sensorStatus & 0x01) - { - /* Serial.print("Magnetometer not acknowledging!"); */ - } - if(sensorStatus & 0x02) - { - /* Serial.print("Accelerometer not acknowledging!"); */ - } - if(sensorStatus & 0x04) - { - /* Serial.print("Gyro not acknowledging!"); */ - } - if(sensorStatus & 0x10) - { - /* Serial.print("Magnetometer ID not recognized!"); */ - } - if(sensorStatus & 0x20) - { - /* Serial.print("Accelerometer ID not recognized!"); */ - } - if(sensorStatus & 0x40) - { - /* Serial.print("Gyro ID not recognized!"); */ - } - - /* Serial.print("Actual MagRate = "); */ - /* Serial.print(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ActualMagRate)); */ - /* Serial.println(" Hz"); */ - /* Serial.print("Actual AccelRate = "); */ - /* Serial.print(10 * i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ActualAccelRate)); */ - /* Serial.println(" Hz"); */ - /* Serial.print("Actual GyroRate = "); */ - /* Serial.print(10 * i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ActualGyroRate)); */ - /* Serial.println(" Hz"); */ - /* Serial.print("Actual BaroRate = "); */ - /* Serial.print(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ActualBaroRate)); */ - /* Serial.println(" Hz"); */ + return EM7180_OK; } -#if(0) -void em7180_chip_id_get(em7180_t *em7180) +void em7180_set_delay_cb(em7180_t *em7180, delay_func_t delay_func) { - // Read SENtral device information - uint16_t ROM1 = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ROMVersion1); - uint16_t ROM2 = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ROMVersion2); - /* Serial.print("EM7180 ROM Version: 0x"); */ - /* Serial.print(ROM1, HEX); */ - /* Serial.println(ROM2, HEX); */ - /* Serial.println("Should be: 0xE609"); */ - uint16_t RAM1 = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_RAMVersion1); - uint16_t RAM2 = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_RAMVersion2); - /* Serial.print("EM7180 RAM Version: 0x"); */ - /* Serial.print(RAM1); */ - /* Serial.println(RAM2); */ - uint8_t PID = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ProductID); - /* Serial.print("EM7180 ProductID: 0x"); */ - /* Serial.print(PID, HEX); */ - /* Serial.println(" Should be: 0x80"); */ - uint8_t RID = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_RevisionID); - /* Serial.print("EM7180 RevisionID: 0x"); */ - /* Serial.print(RID, HEX); */ - /* Serial.println(" Should be: 0x02"); */ + return_if_fail(em7180); + + em7180->delay_func = delay_func; } -#endif -void em7180_load_fw_from_eeprom(em7180_t *em7180) +void em7180_set_i2c_cbs(em7180_t *em7180, i2c_read_func_t i2c_read_func, + i2c_write_func_t i2c_write_func, uint8_t dev_addr) { - // Check which sensors can be detected by the EM7180 - uint8_t featureflag = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_FeatureFlags); - if(featureflag & 0x01) - { - /* Serial.println("A barometer is installed"); */ - } - if(featureflag & 0x02) - { - /* Serial.println("A humidity sensor is installed"); */ - } - if(featureflag & 0x04) - { - /* Serial.println("A temperature sensor is installed"); */ - } - if(featureflag & 0x08) - { - /* Serial.println("A custom sensor is installed"); */ - } - if(featureflag & 0x10) - { - /* Serial.println("A second custom sensor is installed"); */ - } - if(featureflag & 0x20) - { - /* Serial.println("A third custom sensor is installed"); */ - } + return_if_fail(em7180); - HAL_Delay(1000); // give some time to read the screen + em7180->i2c_read_func = i2c_read_func; + em7180->i2c_write_func = i2c_write_func; + em7180->i2c_addr = dev_addr; +} - // Check SENtral status, make sure EEPROM upload of firmware was accomplished - uint8_t STAT = (i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_SentralStatus) - & 0x01); - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) & 0x01) - { - /* Serial.println("EEPROM detected on the sensor bus!"); */ - } - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) & 0x02) - { - /* Serial.println("EEPROM uploaded config file!"); */ - } - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) & 0x04) - { - /* Serial.println("EEPROM CRC incorrect!"); */ - } - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) & 0x08) - { - /* Serial.println("EM7180 in initialized state!"); */ - } - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) & 0x10) - { - /* Serial.println("No EEPROM detected!"); */ - } - int count = 0; - while(!STAT) - { - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ResetRequest, 0x01); - HAL_Delay(500); - count++; - STAT = (i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_SentralStatus) - & 0x01); - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) & 0x01) - { - /* Serial.println("EEPROM detected on the sensor bus!"); */ - } - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) & 0x02) - { - /* Serial.println("EEPROM uploaded config file!"); */ - } - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) & 0x04) - { - /* Serial.println("EEPROM CRC incorrect!"); */ - } - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) & 0x08) - { - /* Serial.println("EM7180 in initialized state!"); */ - } - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) & 0x10) - { - /* Serial.println("No EEPROM detected!"); */ - } - if(count > 10) - { - break; - } - } +em7180_status_t em7180_config(em7180_t *em7180) +{ + uint8_t resp; + int32_t ret = 0; - if(!(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_SentralStatus) - & 0x04)) + /* wait for EM7180 to enter unprogrammed or initialized state */ + do { - /* Serial.println("EEPROM upload successful!"); */ + em7180_read_byte(EM7180_SentralStatus, &resp); } -} + while(!(resp & 0x08)); -uint8_t em7180_status(em7180_t *em7180) -{ - // Check event status register, way to check data ready by polling rather than interrupt - uint8_t c = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_EventStatus); // reading clears the register and interrupt + /* configure EM7180 to accept initial register set-up */ + ret |= em7180_write_byte(EM7180_HostControl, 0x00); /* put EM7180 in initialized state to configure registers */ + ret |= em7180_write_byte(EM7180_PassThroughControl, 0x00); /* make sure pass through mode is off */ + ret |= em7180_write_byte(EM7180_HostControl, 0x01); /* force initialize */ + ret |= em7180_write_byte(EM7180_HostControl, 0x00); /* put EM7180 in initialized state to configure registers */ - return c; -} + /* set output data rates */ + ret |= em7180_write_byte(EM7180_MagRate, em7180->init->mag_rate); + ret |= em7180_write_byte(EM7180_AccelRate, em7180->init->acc_rate); + ret |= em7180_write_byte(EM7180_GyroRate, em7180->init->gyro_rate); + ret |= em7180_write_byte(EM7180_BaroRate, 0x80 | em7180->init->baro_rate); + ret |= em7180_write_byte(EM7180_QRateDivisor, em7180->init->q_rate_div); -uint8_t em7180_errors(em7180_t *em7180) -{ - uint8_t c = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_ErrorRegister); // check error register - - return c; -} + /* configure operating mode */ + ret |= em7180_write_byte(EM7180_AlgorithmControl, 0x00); + /* enable interrupts for CPUReset, Error, QuaternionResult, MagResult, AccelResult, GyroResult */ + ret |= em7180_write_byte(EM7180_EnableEvents, 0x07); + /* run EM7180 */ + ret |= em7180_write_byte(EM7180_HostControl, 0x01); -void em7180_gyro_set_fs(em7180_t *em7180, uint16_t gyro_fs) -{ - uint8_t bytes[4], STAT; - bytes[0] = gyro_fs & (0xFF); - bytes[1] = (gyro_fs >> 8) & (0xFF); - bytes[2] = 0x00; - bytes[3] = 0x00; - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte0, - bytes[0]); // Gyro LSB - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte1, - bytes[1]); // Gyro MSB - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte2, - bytes[2]); // Unused - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte3, - bytes[3]); // Unused - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamRequest, 0xCB); // Parameter 75; 0xCB is 75 decimal with the MSB set high to indicate a parameter write process - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AlgorithmControl, 0x80); // Request parameter transfer procedure - STAT = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamAcknowledge); // Check the parameter acknowledge register and loop until the result matches parameter request byte - while(!(STAT == 0xCB)) - { - STAT = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_ParamAcknowledge); - } - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamRequest, 0x00); // Parameter request = 0 to end parameter transfer process - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AlgorithmControl, 0x00); // Re-start algorithm + return ret ? EM7180_BAD_COMM : EM7180_OK; } -void em7180_mag_acc_set_fs(em7180_t *em7180, uint16_t mag_fs, uint16_t acc_fs) +void em7180_reset(em7180_t *em7180) { - uint8_t bytes[4], STAT; - bytes[0] = mag_fs & (0xFF); - bytes[1] = (mag_fs >> 8) & (0xFF); - bytes[2] = acc_fs & (0xFF); - bytes[3] = (acc_fs >> 8) & (0xFF); - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte0, - bytes[0]); // Mag LSB - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte1, - bytes[1]); // Mag MSB - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte2, - bytes[2]); // Acc LSB - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte3, - bytes[3]); // Acc MSB - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamRequest, 0xCA); // Parameter 74; 0xCA is 74 decimal with the MSB set high to indicate a paramter write processs - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AlgorithmControl, 0x80); // Request parameter transfer procedure - STAT = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamAcknowledge); // Check the parameter acknowledge register and loop until the result matches parameter request byte - while(!(STAT == 0xCA)) - { - STAT = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_ParamAcknowledge); - } - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamRequest, 0x00); // Parameter request = 0 to end parameter transfer process - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AlgorithmControl, 0x00); // Re-start algorithm + /* reset EM7180 */ + em7180_write_byte(EM7180_ResetRequest, 0x01); } -void em7180_set_integer_param(em7180_t *em7180, uint8_t param, - uint32_t param_val) +void em7180_passthrough_enable(em7180_t *em7180) { - uint8_t bytes[4], STAT; - bytes[0] = param_val & (0xFF); - bytes[1] = (param_val >> 8) & (0xFF); - bytes[2] = (param_val >> 16) & (0xFF); - bytes[3] = (param_val >> 24) & (0xFF); - param = param | 0x80; // Parameter is the decimal value with the MSB set high to indicate a paramter write processs - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte0, - bytes[0]); // Param LSB - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte1, - bytes[1]); - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte2, - bytes[2]); - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte3, - bytes[3]); // Param MSB - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamRequest, param); - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AlgorithmControl, 0x80); // Request parameter transfer procedure - STAT = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamAcknowledge); // Check the parameter acknowledge register and loop until the result matches parameter request byte - while(!(STAT == param)) - { - STAT = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_ParamAcknowledge); - } - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamRequest, 0x00); // Parameter request = 0 to end parameter transfer process - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AlgorithmControl, 0x00); // Re-start algorithm + /* put EM7180 in standby mode */ + em7180_write_byte(EM7180_AlgorithmControl, 0x01); + /* put EM7180 in passthrough mode */ + em7180_write_byte(EM7180_PassThroughControl, 0x01); } -void em7180_param_set_float(em7180_t *em7180, uint8_t param, float param_val) +void em7180_passthrough_disable(em7180_t *em7180) { - uint8_t bytes[4], STAT; - float_to_bytes(param_val, &bytes[0]); - param = param | 0x80; // Parameter is the decimal value with the MSB set high to indicate a paramter write processs - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte0, - bytes[0]); // Param LSB - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte1, - bytes[1]); - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte2, - bytes[2]); - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_LoadParamByte3, - bytes[3]); // Param MSB - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamRequest, param); - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AlgorithmControl, 0x80); // Request parameter transfer procedure - STAT = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamAcknowledge); // Check the parameter acknowledge register and loop until the result matches parameter request byte - while(!(STAT == param)) - { - STAT = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_ParamAcknowledge); - } - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_ParamRequest, 0x00); // Parameter request = 0 to end parameter transfer process - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AlgorithmControl, 0x00); // Re-start algorithm + /* put EM7180 in standby mode */ + em7180_write_byte(EM7180_PassThroughControl, 0x00); + /* put EM7180 in normal operation */ + em7180_write_byte(EM7180_AlgorithmControl, 0x00); } -void em7180_quatdata_get(em7180_t *em7180, float *destination) +em7180_status_t em7180_quatdata_get(em7180_t *em7180, float *destination) { - uint8_t data[16]; // x/y/z quaternion register data stored here + int32_t ret = 0; + uint8_t data[16]; - i2c_read(em7180->hi2c, EM7180_ADDRESS, EM7180_QX, data, 16); // Read the sixteen raw data registers into data array + ret |= em7180_read(EM7180_QX, data, 16); + /* EM7180 quaternion is xi + yj + zk + s */ + destination[0] = uint32_reg_to_float(&data[12]); destination[1] = uint32_reg_to_float(&data[0]); destination[2] = uint32_reg_to_float(&data[4]); destination[3] = uint32_reg_to_float(&data[8]); - destination[0] = uint32_reg_to_float(&data[12]); // SENtral stores quats as qx, qy, qz, q0! + + return ret ? EM7180_BAD_COMM : EM7180_OK; } -void em7180_acceldata_get(em7180_t *em7180, int16_t *destination) +em7180_status_t em7180_acceldata_get(em7180_t *em7180, int16_t *destination) { - uint8_t data[6]; // x/y/z accel register data stored here + int32_t ret = 0; + uint8_t data[6]; - i2c_read(em7180->hi2c, EM7180_ADDRESS, EM7180_AX, data, 6); // Read the six raw data registers into data array - destination[0] = (int16_t) (((int16_t) data[1] << 8) | data[0]); // Turn the MSB and LSB into a signed 16-bit value + ret |= em7180_read(EM7180_AX, data, 6); + destination[0] = (int16_t) (((int16_t) data[1] << 8) | data[0]); destination[1] = (int16_t) (((int16_t) data[3] << 8) | data[2]); destination[2] = (int16_t) (((int16_t) data[5] << 8) | data[4]); + + return ret ? EM7180_BAD_COMM : EM7180_OK; } -void em7180_gyrodata_get(em7180_t *em7180, int16_t *destination) +em7180_status_t em7180_gyrodata_get(em7180_t *em7180, int16_t *destination) { - uint8_t data[6]; // x/y/z gyro register data stored here + int32_t ret = 0; + uint8_t data[6]; - i2c_read(em7180->hi2c, EM7180_ADDRESS, EM7180_GX, data, 6); // Read the six raw data registers sequentially into data array - destination[0] = (int16_t) (((int16_t) data[1] << 8) | data[0]); // Turn the MSB and LSB into a signed 16-bit value + ret |= em7180_read(EM7180_GX, data, 6); + destination[0] = (int16_t) (((int16_t) data[1] << 8) | data[0]); destination[1] = (int16_t) (((int16_t) data[3] << 8) | data[2]); destination[2] = (int16_t) (((int16_t) data[5] << 8) | data[4]); + + return ret ? EM7180_BAD_COMM : EM7180_OK; } -void em7180_magdata_get(em7180_t *em7180, int16_t *destination) +em7180_status_t em7180_magdata_get(em7180_t *em7180, int16_t *destination) { - uint8_t data[6]; // x/y/z mag register data stored here + int32_t ret = 0; + uint8_t data[6]; - i2c_read(em7180->hi2c, EM7180_ADDRESS, EM7180_MX, data, 6); // Read the six raw data registers sequentially into data array - destination[0] = (int16_t) (((int16_t) data[1] << 8) | data[0]); // Turn the MSB and LSB into a signed 16-bit value + ret |= em7180_read(EM7180_MX, data, 6); + destination[0] = (int16_t) (((int16_t) data[1] << 8) | data[0]); destination[1] = (int16_t) (((int16_t) data[3] << 8) | data[2]); destination[2] = (int16_t) (((int16_t) data[5] << 8) | data[4]); + + return ret ? EM7180_BAD_COMM : EM7180_OK; +} + +em7180_status_t em7180_barodata_get(em7180_t *em7180, int16_t *destination) +{ + int32_t ret = 0; + uint8_t data[2]; + + ret |= em7180_read(EM7180_Baro, data, 2); + + *destination = (((int16_t) data[1] << 8) | data[0]); + + return ret ? EM7180_BAD_COMM : EM7180_OK; +} + +em7180_status_t em7180_tempdata_get(em7180_t *em7180, int16_t *destination) +{ + int32_t ret = 0; + uint8_t data[2]; + + ret |= em7180_read(EM7180_Temp, data, 2); + + *destination = (((int16_t) data[1] << 8) | data[0]); + + return ret ? EM7180_BAD_COMM : EM7180_OK; +} + +/* FIXME: haven't explored the usagge/usefulness of these yet: */ +#if(0) +uint8_t em7180_status(em7180_t *em7180) +{ + // Check event status register, way to check data ready by polling rather than interrupt + uint8_t c; + + em7180_read_byte(EM7180_EventStatus, &c); // reading clears the register and interrupt + + return c; +} + +uint8_t em7180_errors(em7180_t *em7180) +{ + uint8_t c; + + em7180_read_byte(EM7180_ErrorRegister, &c); // check error register + + return c; +} + +void em7180_gyro_set_fs(em7180_t *em7180, uint16_t gyro_fs) +{ + uint8_t resp; + + em7180_write_byte(EM7180_LoadParamByte0, gyro_fs & (0xFF)); // Gyro LSB + em7180_write_byte(EM7180_LoadParamByte1, (gyro_fs >> 8) & (0xFF)); // Gyro MSB + em7180_write_byte(EM7180_LoadParamByte2, 0x00); // Unused + em7180_write_byte(EM7180_LoadParamByte3, 0x00); // Unused + em7180_write_byte(EM7180_ParamRequest, 0xcb); // Parameter 75; 0xCB is 75 decimal with the MSB set high to indicate a parameter write process + em7180_write_byte(EM7180_AlgorithmControl, 0x80); // Request parameter transfer procedure + /* Check the parameter acknowledge register and loop until the result matches parameter request byte */ + do + { + em7180_read_byte(EM7180_ParamAcknowledge, &resp); + } + while(resp != 0xCB); + em7180_write_byte(EM7180_ParamRequest, 0x00); // Parameter request = 0 to end parameter transfer process + em7180_write_byte(EM7180_AlgorithmControl, 0x00); // Re-start algorithm +} + +void em7180_mag_acc_set_fs(em7180_t *em7180, uint16_t mag_fs, uint16_t acc_fs) +{ + uint8_t resp; + + em7180_write_byte(EM7180_LoadParamByte0, mag_fs & (0xFF)); // Mag LSB + em7180_write_byte(EM7180_LoadParamByte1, (mag_fs >> 8) & (0xFF)); // Mag MSB + em7180_write_byte(EM7180_LoadParamByte2, acc_fs & (0xFF)); // Acc LSB + em7180_write_byte(EM7180_LoadParamByte3, (acc_fs >> 8) & (0xFF)); // Acc MSB + em7180_write_byte(EM7180_ParamRequest, 0xca); // Parameter 74; 0xCA is 74 decimal with the MSB set high to indicate a parameter write process + em7180_write_byte(EM7180_AlgorithmControl, 0x80); // Request parameter transfer procedure + /* Check the parameter acknowledge register and loop until the result matches parameter request byte */ + do + { + em7180_read_byte(EM7180_ParamAcknowledge, &resp); + } + while(!(resp == 0xCA)); + em7180_write_byte(EM7180_ParamRequest, 0x00); // Parameter request = 0 to end parameter transfer process + em7180_write_byte(EM7180_AlgorithmControl, 0x00); // Re-start algorithm +} + +void em7180_set_integer_param(em7180_t *em7180, uint8_t param, + uint32_t param_val) +{ + uint8_t resp; + + em7180_write_byte(EM7180_LoadParamByte0, param_val & (0xFF)); // Param LSB + em7180_write_byte(EM7180_LoadParamByte1, (param_val >> 8) & (0xFF)); + em7180_write_byte(EM7180_LoadParamByte2, (param_val >> 16) & (0xFF)); + em7180_write_byte(EM7180_LoadParamByte3, (param_val >> 24) & (0xFF)); // Param MSB + em7180_write_byte(EM7180_ParamRequest, param | 0x80); // Parameter is the decimal value with the MSB set high to indicate a parameter write process + em7180_write_byte(EM7180_AlgorithmControl, 0x80); // Request parameter transfer procedure + /* Check the parameter acknowledge register and loop until the result matches parameter request byte */ + do + { + em7180_read_byte(EM7180_ParamAcknowledge, &resp); + } + while(resp != param); + em7180_write_byte(EM7180_ParamRequest, 0x00); // Parameter request = 0 to end parameter transfer process + em7180_write_byte(EM7180_AlgorithmControl, 0x00); // Re-start algorithm +} + +void em7180_param_set_float(em7180_t *em7180, uint8_t param, float param_val) +{ + uint8_t resp; + uint8_t bytes[4]; + + float_to_bytes(param_val, bytes); + em7180_write_byte(EM7180_LoadParamByte0, bytes[0]); // Param LSB + em7180_write_byte(EM7180_LoadParamByte1, bytes[1]); + em7180_write_byte(EM7180_LoadParamByte2, bytes[2]); + em7180_write_byte(EM7180_LoadParamByte3, bytes[3]); // Param MSB + em7180_write_byte(EM7180_ParamRequest, param | 0x80); // Parameter is the decimal value with the MSB set high to indicate a parameter write process + em7180_write_byte(EM7180_AlgorithmControl, 0x80); // Request parameter transfer procedure + /* Check the parameter acknowledge register and loop until the result matches parameter request byte */ + do + { + em7180_read_byte(EM7180_ParamAcknowledge, &resp); + } + while(resp != param); + em7180_write_byte(EM7180_ParamRequest, 0x00); // Parameter request = 0 to end parameter transfer process + em7180_write_byte(EM7180_AlgorithmControl, 0x00); // Re-start algorithm } float em7180_mres_get(uint8_t Mscale) @@ -605,48 +461,7 @@ float em7180_ares_get(uint8_t ascale) return a_res; } - -int16_t em7180_baro_get(em7180_t *em7180) -{ - uint8_t data[2]; // baro register data stored here - - i2c_read(em7180->hi2c, EM7180_ADDRESS, EM7180_Baro, data, 2); // Read the two raw data registers sequentially into data array - - return (((int16_t) data[1] << 8) | data[0]); // Turn the MSB and LSB into a signed 16-bit value -} - -int16_t em7180_temp_get(em7180_t *em7180) -{ - uint8_t data[2]; // temp register data stored here - - i2c_read(em7180->hi2c, EM7180_ADDRESS, EM7180_Temp, data, 2); // Read the two raw data registers sequentially into data array - - return (((int16_t) data[1] << 8) | data[0]); // Turn the MSB and LSB into a signed 16-bit value -} - -static void em7180_passthrough(em7180_t *em7180) -{ - // First put SENtral in standby mode - uint8_t c = i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, - EM7180_AlgorithmControl); - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_AlgorithmControl, - c | 0x01); - // c = readByte(EM7180_ADDRESS, EM7180_AlgorithmStatus); - /* // Serial.print("c = "); Serial.println(c); */ - // Verify standby status - // if(readByte(EM7180_ADDRESS, EM7180_AlgorithmStatus) & 0x01) { - /* Serial.println("SENtral in standby mode"); */ - // Place SENtral in pass-through mode - i2c_write_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_PassThruControl, 0x01); - if(i2c_read_byte(em7180->hi2c, EM7180_ADDRESS, EM7180_PassThruStatus) & 0x01) - { - /* Serial.println("SENtral in pass-through mode"); */ - } - else - { - /* Serial.println("ERROR! SENtral not in pass-through mode!"); */ - } -} +#endif static float uint32_reg_to_float(uint8_t *buf) { @@ -661,24 +476,3 @@ static float uint32_reg_to_float(uint8_t *buf) return u.f; } - -static void float_to_bytes(float param_val, uint8_t *buf) -{ - union - { - float f; - uint8_t u8[sizeof(float)]; - } u; - - u.f = param_val; - for(uint8_t i = 0; i < sizeof(float); i++) - { - buf[i] = u.u8[i]; - } - // Convert to LITTLE ENDIAN - /* FIXME: What the hell? */ - for(uint8_t i = 0; i < sizeof(float); i++) - { - buf[i] = buf[(sizeof(float) - 1) - i]; - } -} diff --git a/Drivers/EM7180/Src/em7180_common.c b/Drivers/EM7180/Src/em7180_common.c new file mode 100644 index 0000000..be1c56a --- /dev/null +++ b/Drivers/EM7180/Src/em7180_common.c @@ -0,0 +1,36 @@ +/* + * em7180_common.c + * + * Created on: Jan 22, 2021 + * Author: dan + */ + +/* Includes */ +#include +#include "em7180_common.h" + +/* Function Definitions */ + +int32_t i2c_read_byte(i2c_read_func_t read, uint16_t dev_addr, uint16_t addr, + uint8_t *byte) +{ + return read(dev_addr, addr, 1, byte, 1); +} + +int32_t i2c_write_byte(i2c_write_func_t write, uint16_t dev_addr, uint16_t addr, + uint8_t byte) +{ + return write(dev_addr, addr, 1, &byte, 1); +} + +int32_t i2c_read(i2c_read_func_t read, uint16_t dev_addr, uint16_t addr, + uint8_t *data, uint16_t len) +{ + return read(dev_addr, addr, 1, data, len); +} + +int32_t i2c_write(i2c_write_func_t write, uint16_t dev_addr, uint16_t addr, + uint8_t *data, uint16_t len) +{ + return write(dev_addr, addr, 1, data, len); +} diff --git a/Drivers/EM7180/Src/lis2mdl.c b/Drivers/EM7180/Src/lis2mdl.c index 54eec70..789acf1 100644 --- a/Drivers/EM7180/Src/lis2mdl.c +++ b/Drivers/EM7180/Src/lis2mdl.c @@ -37,20 +37,21 @@ void lis2mdl_init(lis2mdl_t *lis2mdl, uint8_t m_odr) void lis2mdl_config(lis2mdl_t *lis2mdl, I2C_HandleTypeDef *hi2c) { // enable temperature compensation (bit 7 == 1), continuous mode (bits 0:1 == 00) - i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_A, - 0x80 | lis2mdl->m_odr << 2); + broken_i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_A, + 0x80 | lis2mdl->m_odr << 2); // enable low pass filter (bit 0 == 1), set to ODR/4 - i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_B, 0x01); + broken_i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_B, 0x01); // enable data ready on interrupt pin (bit 0 == 1), enable block data read (bit 4 == 1) - i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_C, 0x01 | 0x10); + broken_i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_C, + 0x01 | 0x10); } uint8_t lis2mdl_chip_id_get(I2C_HandleTypeDef *hi2c) { - uint8_t c = i2c_read_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_WHO_AM_I); + uint8_t c = broken_i2c_read_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_WHO_AM_I); return c; } @@ -58,25 +59,30 @@ uint8_t lis2mdl_chip_id_get(I2C_HandleTypeDef *hi2c) void lis2mdl_reset(I2C_HandleTypeDef *hi2c) { // reset device - uint8_t temp = i2c_read_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_A); + uint8_t temp = broken_i2c_read_byte(hi2c, LIS2MDL_ADDRESS, + LIS2MDL_CFG_REG_A); - i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_A, temp | 0x20); // Set bit 5 to 1 to reset LIS2MDL + broken_i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_A, + temp | 0x20); // Set bit 5 to 1 to reset LIS2MDL HAL_Delay(1); - i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_A, temp | 0x40); // Set bit 6 to 1 to boot LIS2MDL + broken_i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_A, + temp | 0x40); // Set bit 6 to 1 to boot LIS2MDL HAL_Delay(100); // Wait for all registers to reset } uint8_t lis2mdl_status(I2C_HandleTypeDef *hi2c) { // Read the status register of the altimeter - uint8_t temp = i2c_read_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_STATUS_REG); + uint8_t temp = broken_i2c_read_byte(hi2c, LIS2MDL_ADDRESS, + LIS2MDL_STATUS_REG); return temp; } void lis2mdl_data_get(I2C_HandleTypeDef *hi2c, int16_t *destination) { uint8_t data[6]; // x/y/z mag register data stored here - i2c_read(hi2c, LIS2MDL_ADDRESS, (0x80 | LIS2MDL_OUTX_L_REG), data, 8); // Read the 6 raw data registers into data array + broken_i2c_read(hi2c, LIS2MDL_ADDRESS, (0x80 | LIS2MDL_OUTX_L_REG), data, + 8); // Read the 6 raw data registers into data array destination[0] = ((int16_t) data[1] << 8) | data[0]; // Turn the MSB and LSB into a signed 16-bit value destination[1] = ((int16_t) data[3] << 8) | data[2]; @@ -167,8 +173,8 @@ void lis2mdl_self_test(I2C_HandleTypeDef *hi2c) magNom[1] = (float) sum[1] / 50.0f; magNom[2] = (float) sum[2] / 50.0f; - uint8_t c = i2c_read_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_C); - i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_C, c | 0x02); // enable self test + uint8_t c = broken_i2c_read_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_C); + broken_i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_C, c | 0x02); // enable self test HAL_Delay(100); // let mag respond sum[0] = 0; @@ -187,7 +193,7 @@ void lis2mdl_self_test(I2C_HandleTypeDef *hi2c) magTest[1] = (float) sum[1] / 50.0f; magTest[2] = (float) sum[2] / 50.0f; - i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_C, c); // return to previous settings/normal mode + broken_i2c_write_byte(hi2c, LIS2MDL_ADDRESS, LIS2MDL_CFG_REG_C, c); // return to previous settings/normal mode HAL_Delay(100); // let mag respond /* Serial.println("Mag Self Test:"); */ diff --git a/Drivers/EM7180/Src/lps22hb.c b/Drivers/EM7180/Src/lps22hb.c index 5387a9d..90db3a8 100644 --- a/Drivers/EM7180/Src/lps22hb.c +++ b/Drivers/EM7180/Src/lps22hb.c @@ -37,15 +37,15 @@ void lps22hb_config(lps22hb_t *lps22hb, I2C_HandleTypeDef *hi2c) // enable low-pass filter by setting bit 3 to one // bit 2 == 0 means bandwidth is odr/9, bit 2 == 1 means bandwidth is odr/20 // make sure data not updated during read by setting block data udate (bit 1) to 1 - i2c_write_byte(hi2c, LPS22H_ADDRESS, LPS22H_CTRL_REG1, - lps22hb->p_odr << 4 | 0x08 | 0x02); - i2c_write_byte(hi2c, LPS22H_ADDRESS, LPS22H_CTRL_REG3, 0x04); // enable data ready as interrupt source + broken_i2c_write_byte(hi2c, LPS22H_ADDRESS, LPS22H_CTRL_REG1, + lps22hb->p_odr << 4 | 0x08 | 0x02); + broken_i2c_write_byte(hi2c, LPS22H_ADDRESS, LPS22H_CTRL_REG3, 0x04); // enable data ready as interrupt source } uint8_t lps22hb_chip_id_get(I2C_HandleTypeDef *hi2c) { // Read the WHO_AM_I register of the altimeter this is a good test of communication - uint8_t temp = i2c_read_byte(hi2c, LPS22H_ADDRESS, LPS22H_WHOAMI); // Read WHO_AM_I register for LPS22H + uint8_t temp = broken_i2c_read_byte(hi2c, LPS22H_ADDRESS, LPS22H_WHOAMI); // Read WHO_AM_I register for LPS22H return temp; } @@ -53,7 +53,7 @@ uint8_t lps22hb_chip_id_get(I2C_HandleTypeDef *hi2c) uint8_t lps22hb_status(I2C_HandleTypeDef *hi2c) { // Read the status register of the altimeter - uint8_t temp = i2c_read_byte(hi2c, LPS22H_ADDRESS, LPS22H_STATUS); + uint8_t temp = broken_i2c_read_byte(hi2c, LPS22H_ADDRESS, LPS22H_STATUS); return temp; } @@ -62,7 +62,8 @@ int32_t lps22hb_pressure_get(I2C_HandleTypeDef *hi2c) { uint8_t data[3]; // 24-bit pressure register data stored here - i2c_read(hi2c, LPS22H_ADDRESS, (LPS22H_PRESS_OUT_XL | 0x80), data, 3); // bit 7 must be one to read multiple bytes + broken_i2c_read(hi2c, LPS22H_ADDRESS, (LPS22H_PRESS_OUT_XL | 0x80), data, + 3); // bit 7 must be one to read multiple bytes return ((int32_t) data[2] << 16 | (int32_t) data[1] << 8 | data[0]); } @@ -71,7 +72,7 @@ int16_t lps22hb_temp_get(I2C_HandleTypeDef *hi2c) { uint8_t data[2]; // 16-bit pressure register data stored here - i2c_read(hi2c, LPS22H_ADDRESS, (LPS22H_TEMP_OUT_L | 0x80), data, 2); // bit 7 must be one to read multiple bytes + broken_i2c_read(hi2c, LPS22H_ADDRESS, (LPS22H_TEMP_OUT_L | 0x80), data, 2); // bit 7 must be one to read multiple bytes return ((int16_t) data[1] << 8 | data[0]); } diff --git a/Drivers/EM7180/Src/lsm6dsm.c b/Drivers/EM7180/Src/lsm6dsm.c index 6819c7f..5ff36cf 100644 --- a/Drivers/EM7180/Src/lsm6dsm.c +++ b/Drivers/EM7180/Src/lsm6dsm.c @@ -40,30 +40,32 @@ void lsm6dsm_init(lsm6dsm_t *lsm6dsm, uint8_t ascale, uint8_t gscale, void lsm6dsm_config(lsm6dsm_t *lsm6dsm, I2C_HandleTypeDef *hi2c) { - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL1_XL, - lsm6dsm->a_odr << 4 | lsm6dsm->ascale << 2); + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL1_XL, + lsm6dsm->a_odr << 4 | lsm6dsm->ascale << 2); - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL2_G, - lsm6dsm->g_odr << 4 | lsm6dsm->gscale << 2); + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL2_G, + lsm6dsm->g_odr << 4 | lsm6dsm->gscale << 2); - uint8_t temp = i2c_read_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL3_C); + uint8_t temp = broken_i2c_read_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL3_C); // enable block update (bit 6 = 1), auto-increment registers (bit 2 = 1) - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL3_C, temp | 0x40 | 0x04); + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL3_C, + temp | 0x40 | 0x04); // by default, interrupts active HIGH, push pull, little endian data // (can be changed by writing to bits 5, 4, and 1, resp to above register) // enable accel LP2 (bit 7 = 1), set LP2 tp ODR/9 (bit 6 = 1), enable input_composite (bit 3) for low noise - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL8_XL, 0x80 | 0x40 | 0x08); + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL8_XL, + 0x80 | 0x40 | 0x08); // interrupt handling - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_DRDY_PULSE_CFG, 0x80); // latch interrupt until data read - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_INT1_CTRL, 0x40); // enable significant motion interrupts on INT1 - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_INT2_CTRL, 0x03); // enable accel/gyro data ready interrupts on INT2 + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_DRDY_PULSE_CFG, 0x80); // latch interrupt until data read + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_INT1_CTRL, 0x40); // enable significant motion interrupts on INT1 + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_INT2_CTRL, 0x03); // enable accel/gyro data ready interrupts on INT2 } uint8_t lsm6dsm_chip_id_get(lsm6dsm_t *lsm6dsm, I2C_HandleTypeDef *hi2c) { - uint8_t c = i2c_read_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_WHO_AM_I); + uint8_t c = broken_i2c_read_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_WHO_AM_I); return c; } @@ -128,9 +130,9 @@ float lsm6dsm_gres_get(lsm6dsm_t *lsm6dsm) void lsm6dsm_reset(lsm6dsm_t *lsm6dsm, I2C_HandleTypeDef *hi2c) { // reset device - uint8_t temp = i2c_read_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL3_C); + uint8_t temp = broken_i2c_read_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL3_C); - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL3_C, temp | 0x01); // Set bit 0 to 1 to reset LSM6DSM + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL3_C, temp | 0x01); // Set bit 0 to 1 to reset LSM6DSM HAL_Delay(100); // Wait for all registers to reset } @@ -152,35 +154,35 @@ void lsm6dsm_self_test(lsm6dsm_t *lsm6dsm, I2C_HandleTypeDef *hi2c) gyroNom[1] = temp[2]; gyroNom[2] = temp[3]; - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL5_C, 0x01); // positive accel self test + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL5_C, 0x01); // positive accel self test HAL_Delay(100); // let accel respond lsm6dsm_read_data(lsm6dsm, hi2c, temp); accelPTest[0] = temp[4]; accelPTest[1] = temp[5]; accelPTest[2] = temp[6]; - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL5_C, 0x03); // negative accel self test + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL5_C, 0x03); // negative accel self test HAL_Delay(100); // let accel respond lsm6dsm_read_data(lsm6dsm, hi2c, temp); accelNTest[0] = temp[4]; accelNTest[1] = temp[5]; accelNTest[2] = temp[6]; - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL5_C, 0x04); // positive gyro self test + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL5_C, 0x04); // positive gyro self test HAL_Delay(100); // let gyro respond lsm6dsm_read_data(lsm6dsm, hi2c, temp); gyroPTest[0] = temp[1]; gyroPTest[1] = temp[2]; gyroPTest[2] = temp[3]; - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL5_C, 0x0C); // negative gyro self test + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL5_C, 0x0C); // negative gyro self test HAL_Delay(100); // let gyro respond lsm6dsm_read_data(lsm6dsm, hi2c, temp); gyroNTest[0] = temp[1]; gyroNTest[1] = temp[2]; gyroNTest[2] = temp[3]; - i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL5_C, 0x00); // normal mode + broken_i2c_write_byte(hi2c, LSM6DSM_ADDRESS, LSM6DSM_CTRL5_C, 0x00); // normal mode HAL_Delay(100); // let accel and gyro respond /* Serial.println("Accel Self Test:"); */ @@ -280,7 +282,7 @@ static void lsm6dsm_read_data(lsm6dsm_t *lsm6dsm, I2C_HandleTypeDef *hi2c, { uint8_t data[14]; // x/y/z accel register data stored here - i2c_read(hi2c, LSM6DSM_ADDRESS, LSM6DSM_OUT_TEMP_L, data, 14); // Read the 14 raw data registers into data array + broken_i2c_read(hi2c, LSM6DSM_ADDRESS, LSM6DSM_OUT_TEMP_L, data, 14); // Read the 14 raw data registers into data array destination[0] = ((int16_t) data[1] << 8) | data[0]; // Turn the MSB and LSB into a signed 16-bit value destination[1] = ((int16_t) data[3] << 8) | data[2]; destination[2] = ((int16_t) data[5] << 8) | data[4];