You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					234 lines
				
				5.7 KiB
			
		
		
			
		
	
	
					234 lines
				
				5.7 KiB
			| 
								 
											5 years ago
										 
									 | 
							
								#include "lsm6ds3_gyro.h"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								#include <cassert>
							 | 
						||
| 
								 | 
							
								#include <cmath>
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								#include <cstring>
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								
							 | 
						||
| 
								 
											4 years ago
										 
									 | 
							
								#include "common/swaglog.h"
							 | 
						||
| 
								 | 
							
								#include "common/timing.h"
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								#include "common/util.h"
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								
							 | 
						||
| 
								 | 
							
								#define DEG2RAD(x) ((x) * M_PI / 180.0)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								LSM6DS3_Gyro::LSM6DS3_Gyro(I2CBus *bus, int gpio_nr, bool shared_gpio) :
							 | 
						||
| 
								 | 
							
								  I2CSensor(bus, gpio_nr, shared_gpio) {}
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								void LSM6DS3_Gyro::wait_for_data_ready() {
							 | 
						||
| 
								 | 
							
								  uint8_t drdy = 0;
							 | 
						||
| 
								 | 
							
								  uint8_t buffer[6];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  do {
							 | 
						||
| 
								 | 
							
								    read_register(LSM6DS3_GYRO_I2C_REG_STAT_REG, &drdy, sizeof(drdy));
							 | 
						||
| 
								 | 
							
								    drdy &= LSM6DS3_GYRO_DRDY_GDA;
							 | 
						||
| 
								 | 
							
								  } while (drdy == 0);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  read_register(LSM6DS3_GYRO_I2C_REG_OUTX_L_G, buffer, sizeof(buffer));
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								void LSM6DS3_Gyro::read_and_avg_data(float* out_buf) {
							 | 
						||
| 
								 | 
							
								  uint8_t drdy = 0;
							 | 
						||
| 
								 | 
							
								  uint8_t buffer[6];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  for (int i = 0; i < 5; i++) {
							 | 
						||
| 
								 | 
							
								    do {
							 | 
						||
| 
								 | 
							
								      read_register(LSM6DS3_GYRO_I2C_REG_STAT_REG, &drdy, sizeof(drdy));
							 | 
						||
| 
								 | 
							
								      drdy &= LSM6DS3_GYRO_DRDY_GDA;
							 | 
						||
| 
								 | 
							
								    } while (drdy == 0);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    int len = read_register(LSM6DS3_GYRO_I2C_REG_OUTX_L_G, buffer, sizeof(buffer));
							 | 
						||
| 
								 | 
							
								    assert(len == sizeof(buffer));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    for (int j = 0; j < 3; j++) {
							 | 
						||
| 
								 | 
							
								      out_buf[j] += (float)read_16_bit(buffer[j*2], buffer[j*2+1]) * 70.0f;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // calculate the mg average values
							 | 
						||
| 
								 | 
							
								  for (int i = 0; i < 3; i++) {
							 | 
						||
| 
								 | 
							
								    out_buf[i] /= 5.0f;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								int LSM6DS3_Gyro::self_test(int test_type) {
							 | 
						||
| 
								 | 
							
								  float val_st_off[3] = {0};
							 | 
						||
| 
								 | 
							
								  float val_st_on[3] = {0};
							 | 
						||
| 
								 | 
							
								  float test_val[3] = {0};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // prepare sensor for self-test
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // full scale: 2000dps, ODR: 208Hz
							 | 
						||
| 
								 | 
							
								  int ret = set_register(LSM6DS3_GYRO_I2C_REG_CTRL2_G, LSM6DS3_GYRO_ODR_208HZ | LSM6DS3_GYRO_FS_2000dps);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    return ret;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // wait for stable output, and discard first values
							 | 
						||
| 
								 | 
							
								  util::sleep_for(150);
							 | 
						||
| 
								 | 
							
								  wait_for_data_ready();
							 | 
						||
| 
								 | 
							
								  read_and_avg_data(val_st_off);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // enable Self Test positive (or negative)
							 | 
						||
| 
								 | 
							
								  ret = set_register(LSM6DS3_GYRO_I2C_REG_CTRL5_C, test_type);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    return ret;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // wait for stable output, and discard first values
							 | 
						||
| 
								 | 
							
								  util::sleep_for(50);
							 | 
						||
| 
								 | 
							
								  wait_for_data_ready();
							 | 
						||
| 
								 | 
							
								  read_and_avg_data(val_st_on);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // disable sensor
							 | 
						||
| 
								 | 
							
								  ret = set_register(LSM6DS3_GYRO_I2C_REG_CTRL2_G, 0);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    return ret;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // disable self test
							 | 
						||
| 
								 | 
							
								  ret = set_register(LSM6DS3_GYRO_I2C_REG_CTRL5_C, 0);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    return ret;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // calculate the mg values for self test
							 | 
						||
| 
								 | 
							
								  for (int i = 0; i < 3; i++) {
							 | 
						||
| 
								 | 
							
								    test_val[i] = fabs(val_st_on[i] - val_st_off[i]);
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // verify test result
							 | 
						||
| 
								 | 
							
								  for (int i = 0; i < 3; i++) {
							 | 
						||
| 
								 | 
							
								    if ((LSM6DS3_GYRO_MIN_ST_LIMIT_mdps > test_val[i]) ||
							 | 
						||
| 
								 | 
							
								        (test_val[i] > LSM6DS3_GYRO_MAX_ST_LIMIT_mdps)) {
							 | 
						||
| 
								 | 
							
								      return -1;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  return ret;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											4 years ago
										 
									 | 
							
								int LSM6DS3_Gyro::init() {
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  uint8_t value = 0;
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  bool do_self_test = false;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  const char* env_lsm_selftest =env_lsm_selftest = std::getenv("LSM_SELF_TEST");
							 | 
						||
| 
								 | 
							
								  if (env_lsm_selftest != nullptr && strncmp(env_lsm_selftest, "1", 1) == 0) {
							 | 
						||
| 
								 | 
							
								    do_self_test = true;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  int ret = verify_chip_id(LSM6DS3_GYRO_I2C_REG_ID, {LSM6DS3_GYRO_CHIP_ID, LSM6DS3TRC_GYRO_CHIP_ID});
							 | 
						||
| 
								 | 
							
								  if (ret == -1) return -1;
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  if (ret == LSM6DS3TRC_GYRO_CHIP_ID) {
							 | 
						||
| 
								 
											4 years ago
										 
									 | 
							
								    source = cereal::SensorEventData::SensorSource::LSM6DS3TRC;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  ret = init_gpio();
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    goto fail;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  ret = self_test(LSM6DS3_GYRO_POSITIVE_TEST);
							 | 
						||
| 
								 | 
							
								  if (ret < 0 ) {
							 | 
						||
| 
								 | 
							
								    LOGE("LSM6DS3 gyro positive self-test failed!");
							 | 
						||
| 
								 | 
							
								    if (do_self_test) goto fail;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  ret = self_test(LSM6DS3_GYRO_NEGATIVE_TEST);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    LOGE("LSM6DS3 gyro negative self-test failed!");
							 | 
						||
| 
								 | 
							
								    if (do_self_test) goto fail;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								  // TODO: set scale. Default is +- 250 deg/s
							 | 
						||
| 
								 | 
							
								  ret = set_register(LSM6DS3_GYRO_I2C_REG_CTRL2_G, LSM6DS3_GYRO_ODR_104HZ);
							 | 
						||
| 
								 
											4 years ago
										 
									 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								    goto fail;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  ret = set_register(LSM6DS3_GYRO_I2C_REG_DRDY_CFG, LSM6DS3_GYRO_DRDY_PULSE_MODE);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    goto fail;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // enable data ready interrupt for gyro on INT1
							 | 
						||
| 
								 | 
							
								  // (without resetting existing interrupts)
							 | 
						||
| 
								 | 
							
								  ret = read_register(LSM6DS3_GYRO_I2C_REG_INT1_CTRL, &value, 1);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    goto fail;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  value |= LSM6DS3_GYRO_INT1_DRDY_G;
							 | 
						||
| 
								 | 
							
								  ret = set_register(LSM6DS3_GYRO_I2C_REG_INT1_CTRL, value);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								fail:
							 | 
						||
| 
								 | 
							
								  return ret;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								int LSM6DS3_Gyro::shutdown() {
							 | 
						||
| 
								 | 
							
								  int ret = 0;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  // disable data ready interrupt for gyro on INT1
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  uint8_t value = 0;
							 | 
						||
| 
								 | 
							
								  ret = read_register(LSM6DS3_GYRO_I2C_REG_INT1_CTRL, &value, 1);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    goto fail;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  value &= ~(LSM6DS3_GYRO_INT1_DRDY_G);
							 | 
						||
| 
								 | 
							
								  ret = set_register(LSM6DS3_GYRO_I2C_REG_INT1_CTRL, value);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								    LOGE("Could not disable lsm6ds3 gyroscope interrupt!")
							 | 
						||
| 
								 | 
							
								    goto fail;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  // enable power-down mode
							 | 
						||
| 
								 | 
							
								  value = 0;
							 | 
						||
| 
								 | 
							
								  ret = read_register(LSM6DS3_GYRO_I2C_REG_CTRL2_G, &value, 1);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    goto fail;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								  value &= 0x0F;
							 | 
						||
| 
								 | 
							
								  ret = set_register(LSM6DS3_GYRO_I2C_REG_CTRL2_G, value);
							 | 
						||
| 
								 | 
							
								  if (ret < 0) {
							 | 
						||
| 
								 | 
							
								    LOGE("Could not power-down lsm6ds3 gyroscope!")
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								    goto fail;
							 | 
						||
| 
								 | 
							
								  }
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								
							 | 
						||
| 
								 | 
							
								fail:
							 | 
						||
| 
								 | 
							
								  return ret;
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								bool LSM6DS3_Gyro::get_event(MessageBuilder &msg, uint64_t ts) {
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  // INT1 shared with accel, check STATUS_REG who triggered
							 | 
						||
| 
								 | 
							
								  uint8_t status_reg = 0;
							 | 
						||
| 
								 | 
							
								  read_register(LSM6DS3_GYRO_I2C_REG_STAT_REG, &status_reg, sizeof(status_reg));
							 | 
						||
| 
								 | 
							
								  if ((status_reg & LSM6DS3_GYRO_DRDY_GDA) == 0) {
							 | 
						||
| 
								 | 
							
								    return false;
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  }
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								
							 | 
						||
| 
								 | 
							
								  uint8_t buffer[6];
							 | 
						||
| 
								 | 
							
								  int len = read_register(LSM6DS3_GYRO_I2C_REG_OUTX_L_G, buffer, sizeof(buffer));
							 | 
						||
| 
								 | 
							
								  assert(len == sizeof(buffer));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								  float scale = 8.75 / 1000.0;
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								  float x = DEG2RAD(read_16_bit(buffer[0], buffer[1]) * scale);
							 | 
						||
| 
								 | 
							
								  float y = DEG2RAD(read_16_bit(buffer[2], buffer[3]) * scale);
							 | 
						||
| 
								 | 
							
								  float z = DEG2RAD(read_16_bit(buffer[4], buffer[5]) * scale);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  auto event = msg.initEvent().initGyroscope();
							 | 
						||
| 
								 
											4 years ago
										 
									 | 
							
								  event.setSource(source);
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								  event.setVersion(2);
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								  event.setSensor(SENSOR_GYRO_UNCALIBRATED);
							 | 
						||
| 
								 | 
							
								  event.setType(SENSOR_TYPE_GYROSCOPE_UNCALIBRATED);
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  event.setTimestamp(ts);
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								
							 | 
						||
| 
								 | 
							
								  float xyz[] = {y, -x, z};
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								  auto svec = event.initGyroUncalibrated();
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								  svec.setV(xyz);
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								  svec.setStatus(true);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 
											3 years ago
										 
									 | 
							
								  return true;
							 | 
						||
| 
								 
											5 years ago
										 
									 | 
							
								}
							 |