@ -14,510 +14,366 @@
@@ -14,510 +14,366 @@
/* Includes */
# include <stdint.h>
# include <stdbool.h>
# include <math.h>
# include <stddef.h>
# 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)
@@ -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)
@@ -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 ] ;
}
}