[Sensor]BMI160-加速度计、陀螺仪传感器

最近在搞一个和加速度计相关的项目,所以接触到的传感器比较多,现在写一个总结吧,防止后来者和我一样走这么多的弯路。
  首先看到的是引脚图,如果驱动不了应该首先排除硬件的问题:
SPI接法
在这里插入图片描述

IIC接法
在这里插入图片描述

接着我们就着重看下面的几个寄存器:

传感器名(读/写)寄存器号功能
CHIPID®0x00芯片的ID,一般用来看驱动是否正常,固定值0xD1
PMU_STATUS®0x03显示当前各传感器的电源模式,分normal\low_power\suspend三种模式
ACC_CONF(RW)0x40设置输出数据速率、 带宽和加速度传感器读取的模式
ACC_RANGE0x41允许选择的加速度 g 范围
GYR_CONF(RW)0x42在传感器中设置输出数据速率、 带宽和陀螺仪读取的模式。
GYR_RANGE(RW)0x43定义 BMI160 角速度测量范围
INT_EN(RW)0x50-0x52启用各种中断,包括加速度数据、角速度数据和各种特殊功能的中断,使能后映射到INT1上输出,就可以触发单片机的外部中断了。
INT_OUT_CTRL(RW)0x53输出控制,包括输出使能,触发电平、边沿和输出模式(推挽和开漏)
INT_LATCH(RW)0x54设置中断锁存模式(不是很懂,一开始就是锁存了所以一直没有中断输出…,后来关掉就好了)
CMD®0x7E命令寄存器触发操作,如 softreset、 NVM 编程等。特殊的如:start_foc、acc_set_pmu_mode、gyr_set_pmu_mode、mag_set_pmu_mode、prog_nvm、fifo_flush、int_reset、softreset、step_cnt_clr

接下来就是各种特殊功能寄存器了,就不多说了,用哪个配置那个就可以了
重点说下这个计步的功能吧,现在还比较火:

传感器名(读/写)寄存器号功能
STEP_CONF(RW)0x7A-0x7B步数检测的配置,包括Normal mode,Sensitive mode,Robust mode三种也可以自己配置
STEP_CNT®0x78-0x79直接从这两个寄存器中读出记得步数,要注意的是范围是-32768——32768

下面的代码片是计步的初始化,用的是STM32F405:

void bmi160_init(void)
{
    uint8_t ui8Status = 0;
    uint8_t ui8Attempts = 20;
	uint8_t Device_ID;
	
	BMI160_SPI_Init();
	kprintf("BMI160 Init Ok.\n");
	BMI160_CS=1;			//SPI片选取消

    // Reset the BMI160 sensor
    bmi160_reset();

    // Put accel and gyro in normal mode.
    while (ui8Status != 0x20 && ui8Attempts--)
    {
			BMI160_Write_Reg(AM_DEVICES_BMI160_CMD, 0x12);//设置加速度计为	low_power
			BMI160_Write_Reg(AM_DEVICES_BMI160_CMD, 0x14);//设置陀螺仪			suspend
			delay_ms(1);                 
      ui8Status = BMI160_Read_Reg(AM_DEVICES_BMI160_PMU_STATUS);//读加速度和陀螺仪是否初始化为low_power suspend
    }
   // BMI160 not in correct power mode
    if (!ui8Attempts)
    {
      return;
    }
 			kprintf("PMU_STATUS:0x%X \r\n",ui8Status);
		
		BMI160_Write_Reg( AM_DEVICES_BMI160_STEP_CONF_0, 0x15);//计步功能
		BMI160_Write_Reg( AM_DEVICES_BMI160_STEP_CONF_1, 0x0B);
    BMI160_Write_Reg( AM_DEVICES_BMI160_ACC_RANGE, 0x05);//设置加速度计+-4g
    
		// Read status register to clear it.
    ui8Status = BMI160_Read_Reg(AM_DEVICES_BMI160_ERR_REG);//读错误状态寄存器清ui8Status
                                           
    // Enable INT 1 output as active high
    BMI160_Write_Reg(AM_DEVICES_BMI160_INT_OUT_CTRL, 0x0A);//输出使能INT1引脚,高电平活跃
                                
		//INT1 Set
		// Map INT1 to the Step detection interrupt
		BMI160_Write_Reg(AM_DEVICES_BMI160_INT_MAP_1, 0x80);//映射INT1到 watermark中断

		// Enable INT 1 as FIFO watermark
		BMI160_Write_Reg(AM_DEVICES_BMI160_INT_EN_1, 0x10);//使能data-ready
}
//得到步数
void bmi160_getStep(short *rawStep)
{
  uint8_t buf[2];
	
	buf[0]= BMI160_Read_Reg(AM_DEVICES_BMI160_STEP_CNT_1);
	buf[1]= BMI160_Read_Reg(AM_DEVICES_BMI160_STEP_CNT_0);
	
	*rawStep=((uint16_t)buf[0]<<8)|buf[1];  
}

SPI的初始化(我一开始用的是EEPROM的SPI配置读写等,一直驱动不了,后来才突然发现是SPI的问题):

///以下为BMI160驱动
void BMI160_SPI_Init(void)
{
  GPIO_InitTypeDef  GPIO_InitStructure;
  SPI_InitTypeDef  SPI_InitStructure;
	
  RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);//使能GPIOA时钟
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_SPI1, ENABLE);//使能SPI1时钟
 
  //GPIOF9,F10初始化设置
  GPIO_InitStructure.GPIO_Pin = GPIO_Pin_5|GPIO_Pin_6|GPIO_Pin_7;//PB3~5复用功能输出	
  GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;//复用功能
  GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;//推挽输出
  GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;//100MHz
  GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;//上拉
  GPIO_Init(GPIOA, &GPIO_InitStructure);//初始化
	
	GPIO_PinAFConfig(GPIOA,GPIO_PinSource5,GPIO_AF_SPI1); //PI1复用为 SPI1
	GPIO_PinAFConfig(GPIOA,GPIO_PinSource6,GPIO_AF_SPI1); //PI2复用为 SPI1
	GPIO_PinAFConfig(GPIOA,GPIO_PinSource7,GPIO_AF_SPI1); //PI3复用为 SPI1

 
	//这里只针对SPI口初始化
	RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,ENABLE);//复位SPI1
	RCC_APB2PeriphResetCmd(RCC_APB2Periph_SPI1,DISABLE);//停止复位SPI1

	SPI_I2S_DeInit(SPI1); 
	SPI_InitStructure.SPI_Direction = SPI_Direction_2Lines_FullDuplex;  //设置SPI单向或者双向的数据模式:SPI设置为双线双向全双工
	SPI_InitStructure.SPI_Mode = SPI_Mode_Master;		//设置SPI工作模式:设置为主SPI
	SPI_InitStructure.SPI_DataSize = SPI_DataSize_8b;		//设置SPI的数据大小:SPI发送接收8位帧结构
	SPI_InitStructure.SPI_CPOL = SPI_CPOL_High;		//串行同步时钟的空闲状态为高电平
	SPI_InitStructure.SPI_CPHA = SPI_CPHA_2Edge;	//串行同步时钟的第二个跳变沿(上升或下降)数据被采样
	SPI_InitStructure.SPI_NSS = SPI_NSS_Soft;		//NSS信号由硬件(NSS管脚)还是软件(使用SSI位)管理:内部NSS信号有SSI位控制
	SPI_InitStructure.SPI_BaudRatePrescaler = SPI_BaudRatePrescaler_64;		//定义波特率预分频的值:波特率预分频值为256
	SPI_InitStructure.SPI_FirstBit = SPI_FirstBit_MSB;	//指定数据传输从MSB位还是LSB位开始:数据传输从MSB位开始
	SPI_InitStructure.SPI_CRCPolynomial = 7;	//CRC值计算的多项式
	SPI_Init(SPI1, &SPI_InitStructure);  //根据SPI_InitStruct中指定的参数初始化外设SPIx寄存器
 
	SPI_Cmd(SPI1, ENABLE); //使能SPI外设

	SPI1_ReadWriteByte(0xff);//启动传输		 
}
 

//SPI1速度设置函数
//SPI速度=fAPB2/分频系数
//@ref SPI_BaudRate_Prescaler:SPI_BaudRatePrescaler_2~SPI_BaudRatePrescaler_256  
//fAPB2时钟一般为84Mhz:
void SPI1_SetSpeed(u8 SPI_BaudRatePrescaler)
{
  assert_param(IS_SPI_BAUDRATE_PRESCALER(SPI_BaudRatePrescaler));//判断有效性
	SPI1->CR1&=0XFFC7;//位3-5清零,用来设置波特率
	SPI1->CR1|=SPI_BaudRatePrescaler;	//设置SPI1速度 
	SPI_Cmd(SPI1,ENABLE); //使能SPI1
} 
//SPI1 读写一个字节
//TxData:要写入的字节
//返回值:读取到的字节
u8 SPI1_ReadWriteByte(u8 TxData)
{
	u8 result,Retry=0;//result:返回spi读写的结果; retry:失败重试次数

    while (SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_TXE)==RESET)
    {
        Retry++;
        if(Retry>200) return 0;
    }
    SPI_I2S_SendData(SPI1, TxData);
    while (SPI_I2S_GetFlagStatus(SPI1,SPI_I2S_FLAG_RXNE)==RESET)
    {
        Retry++;
        if(Retry>200) return 0;
    }
    return SPI_I2S_ReceiveData(SPI1);
    SPI_I2S_ClearFlag(SPI1,SPI_I2S_FLAG_RXNE);
}

//读取SPI寄存器值
//reg:要读的寄存器
u8 BMI160_Read_Reg(u8 reg)
{	
	u8 reg_val;	    
  BMI160_CS = 0;          //使能SPI传输	
	delay_ms(1);	
  SPI1_ReadWriteByte(reg|0x80);   //1.发送寄存器号 //Ored with "read request" bit
  reg_val=SPI1_ReadWriteByte(0XFF);//读取寄存器内容  // send a value of 0 to read the first byte returned:
  delay_ms(1);	
	BMI160_CS = 1;          //禁止SPI传输  		
  return(reg_val);           //返回状态值
}

//SPI写寄存器
//reg:指定寄存器地址
//value:写入的值
u8 BMI160_Write_Reg(u8 reg,u8 value)
{
	u8 status;	
  BMI160_CS=0;                 //使能SPI传输
  status =SPI1_ReadWriteByte(reg&0x7f);//2.发送寄存器号
  SPI1_ReadWriteByte(value);      //写入寄存器的值
  BMI160_CS=1;                 //禁止SPI传输	   
  return(status);       			//返回状态值
}
  • 25
    点赞
  • 82
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 48
    评论
全志R58的官方开发板加载bmi160驱动的步骤3B.txt 开发板:全志R58的官方开发板R58_PER3_LPDDR3_32X1_V1_1.pdf(板载加速度传感器bma250) 目标:外挂bmi160模块可以检测到加速度和角速度(acc+gyr/加速度传感器+陀螺仪) BSP:r58_20160823.tar.gz(2016/8/22从全志的git服务器拿下来的系统) 显示:HDMI输出1080p分辨率的LCD显示器。 计划步骤: 1、打通开发板上的bma250(证明开发板硬件是好的。全志官方的BSP也是好的。) 2、将驱动程序bma250.c中的bma250全部替换为bmi160,验证是可以加入新的gsensor的(陀螺仪类似)。 3、借用bma250.c这个驱动程序,初始化的部分修改为初始化bmi160,调通BMI160的gsensor部分。 4、完善全志/博世提供的bmi160的驱动程序,调通BMI160的gsensor部分。 (陀螺仪部分鱼刺类似,陀螺仪部分借用l3gd20.c来验证bmi160的gyr部分) 下面进行第三步:借用bma250.c这个驱动程序,初始化的部分修改为初始化bmi160,调通BMI160的gsensor部分。 为了方便观察,直接注释掉除了bma250之外的全部的gsensor: Z:\home\wwt\only_bma250_r58\android\device\softwinner\common\hardware-common\libhardware\libsensors\aw_sensors\sensorDetect.cpp struct sensor_extend_t gsensorList[] = { { { "bma250", LSG_BMA250, }, { "Bosch 3-axis Accelerometer", "Bosch Sensortec", 1, 0, SENSOR_TYPE_ACCELEROMETER, 4.0f*9.81f, (4.0f*9.81f)/1024.0f, 0.2f, 0,0,0, { }, }, }, }; Z:\home\wwt\only_bma250_r58\android\device\softwinner\octopus-perf\configs\gsensor.cfg ;Direction parameter adjustment, including the x, y, z axis, and xy interchange four variables, ;the name of the module used for identification, and drive registered name consistent ;-------------------------- ;name:bma250 ;-------------------------- gsensor_name = bma250 gsensor_direct_x = false gsensor_direct_y = true gsensor_direct_z = true gsensor_xy_revert = true Z:\home\wwt\only_bma250_r58\android\device\softwinner\octopus-perf\BoardConfig.mk #gsensor & Gyr sensor SW_BOARD_USES_SENSORS_TYPE = aw_sensors 注意:lunch的f1选项在HAL层中使用的ST的9轴(ACC+GYR+MAG)传感器。 #gsensor & Gyr sensor SW_BOARD_USES_SENSORS_TYPE = lsm9ds0 Z:\home\wwt\only_bma250_r58\android\device\softwinner\octopus-perf\init.sun8i.rc on boot # use automatic det
以下是使用ESP32-IDF读写SD卡并获取BMI160数据的示例代码: ```C #include <stdio.h> #include <string.h> #include <sys/unistd.h> #include <sys/stat.h> #include <fcntl.h> #include <esp_err.h> #include <esp_log.h> #include <esp_vfs_fat.h> #include <driver/sdmmc_host.h> #include <driver/sdspi_host.h> #include <driver/gpio.h> #include "bmi160.h" #define SD_CS_PIN GPIO_NUM_5 #define TAG "SD_IO" static const char *SD_MOSI_PIN = "GPIO12"; static const char *SD_MISO_PIN = "GPIO13"; static const char *SD_CLK_PIN = "GPIO14"; static const char *SD_CS_PIN = "GPIO15"; static spi_device_handle_t spi; static void sdcard_init(void) { ESP_LOGI(TAG, "Initializing SD card"); esp_err_t ret; sdmmc_host_t host = SDSPI_HOST_DEFAULT(); sdspi_slot_config_t slot_config = SDSPI_SLOT_CONFIG_DEFAULT(); gpio_set_pull_mode(SD_CS_PIN, GPIO_PULLUP_ONLY); slot_config.gpio_cs = SD_CS_PIN; slot_config.gpio_miso = gpio_num_t; slot_config.gpio_mosi = gpio_num_t; slot_config.gpio_sck = gpio_num_t; ret = gpio_set_direction(SD_CS_PIN, GPIO_MODE_OUTPUT); if (ret != ESP_OK) { ESP_LOGE(TAG, "Failed to set direction for CS GPIO"); return; } ret = gpio_set_level(SD_CS_PIN, 1); if (ret != ESP_OK) { ESP_LOGE(TAG, "Failed to set level for CS GPIO"); return; } spi_bus_config_t bus_config = { .mosi_io_num = SD_MOSI_PIN, .miso_io_num = SD_MISO_PIN, .sclk_io_num = SD_CLK_PIN, .quadwp_io_num = -1, .quadhd_io_num = -1 }; ret = spi_bus_initialize(host.slot, &bus_config, 1); if (ret != ESP_OK) { ESP_LOGE(TAG, "Failed to initialize SPI bus"); return; } ret = spi_bus_add_device(host.slot, &slot_config, &spi); if (ret != ESP_OK) { ESP_LOGE(TAG, "Failed to add SPI device"); return; } esp_vfs_fat_sdmmc_mount_config_t mount_config = { .format_if_mount_failed = true, .max_files = 5, .allocation_unit_size = 16 * 1024 }; sdmmc_card_t *card; ret = esp_vfs_fat_sdmmc_mount("/sdcard", &host, &slot_config, &mount_config, &card); if (ret != ESP_OK) { ESP_LOGE(TAG, "Failed to mount filesystem"); return; } sdmmc_card_print_info(stdout, card); } void app_main(void) { sdcard_init(); struct bmi160_dev bmi160; int8_t rslt; rslt = bmi160_init(&bmi160); if (rslt != BMI160_OK) { ESP_LOGE(TAG, "Failed to initialize BMI160"); return; } rslt = bmi160_set_power_mode(BMI160_NORMAL_MODE, &bmi160); if (rslt != BMI160_OK) { ESP_LOGE(TAG, "Failed to set BMI160 power mode"); return; } struct bmi160_sensor_data accel = {0}; struct bmi160_sensor_data gyro = {0}; FILE *f = fopen("/sdcard/data.csv", "w"); if (f == NULL) { ESP_LOGE(TAG, "Failed to open file for writing"); return; } while (1) { rslt = bmi160_get_sensor_data(BMI160_ACCEL_SEL | BMI160_GYRO_SEL, &accel, &gyro, &bmi160); if (rslt != BMI160_OK) { ESP_LOGE(TAG, "Failed to get sensor data"); continue; } fprintf(f, "%d,%d,%d,%d,%d,%d,%d\n", accel.x, accel.y, accel.z, gyro.x, gyro.y, gyro.z, (int) (esp_timer_get_time() / 1000)); fflush(f); vTaskDelay(pdMS_TO_TICKS(100)); } fclose(f); } ``` 这个示例代码使用SPI协议与SD卡通信,并在SD卡上创建一个名为“data.csv”的文件,将获取的BMI160加速度和陀螺仪数据写入文件中。 请注意,此代码仅用于演示目的。 在实际应用中,您需要根据具体需求进行修改和优化。
评论 48
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

山猫Show

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值