STM32L031 BMI160 陀螺仪检测到移动向单片机发送中断

3 篇文章 0 订阅
3 篇文章 0 订阅

开发环境:
电路板硬件:stm32L031+BMI160
软件环境:keil5.27+stm32cubemx6.2.0
开发库: HAL库

问题点:
BMI160是一个微小型的陀螺仪芯片,经过两天的调试,终于搞定了移动或晃动电路板,使BMI160芯片的INT1向单片机发出中断信号,BMI160的配置,还是挺麻烦的,特此记录一下,防止记性太差。stm32L301与BMI160 通信使用的I2C。

直接上代码:

// 0x68 << 1 = 0xD0
#define MBI160_ADDRESS 0xD0

#define AM_DEVICES_BMI160_CMD 0x7e
#define AM_DEVICES_BMI160_INT_MOTION_3 0x62
#define AM_DEVICES_BMI160_INT_MOTION_2 0x61
#define AM_DEVICES_BMI160_INT_MOTION_1 0x60
#define AM_DEVICES_BMI160_INT_MOTION_0 0x5F
#define AM_DEVICES_BMI160_INT_MAP_1 0x56
#define AM_DEVICES_BMI160_INT_MAP_0 0x55
#define AM_DEVICES_BMI160_INT_LATCH 0x54
#define AM_DEVICES_BMI160_INT_OUT_CTRL 0x53
#define AM_DEVICES_BMI160_INT_EN_2 0x52
#define AM_DEVICES_BMI160_INT_EN_1 0x51
#define AM_DEVICES_BMI160_INT_EN_0 0x50
#define AM_DEVICES_BMI160_PMU_STATUS 0x03
#define AM_DEVICES_BMI160_ERR_REG 0x02

// 写指令 buffer
uint8_t I2C_Buffer_Write[16] = {0};

// 定义变量
int gyr_x = 0, gyr_y = 0, gyr_z = 0,  \
                                  acc_x = 0, acc_y = 0, acc_z = 0;

void BMI160_Write_Reg(uint8_t reg,  uint8_t val)
{
    I2C_Buffer_Write[0] = reg ; ;
    I2C_Buffer_Write[1] = val;
    HAL_I2C_Master_Transmit(&hi2c1, MBI160_ADDRESS, I2C_Buffer_Write, 2, 1000);
	HAL_Delay(50);
	
}


void BMI160_Read_Reg(uint8_t reg, uint16_t data_len)
{    
	I2C_Buffer_Write[0] = reg;
    I2C_Buffer_Write[1] = 0x0c;

    HAL_I2C_Master_Transmit(&hi2c1, MBI160_ADDRESS, I2C_Buffer_Write, 1, 1000);

    HAL_Delay(5);
    memset(I2C_Buffer_Write, 0, sizeof(I2C_Buffer_Write));
    HAL_I2C_Master_Receive(&hi2c1, MBI160_ADDRESS, I2C_Buffer_Write, data_len, 1000);
}


// 初始化一下BMI160 若有晃动,移动,INT1 发出中断信号
void init_bmi160()
{
	uint8_t ui8Attempts = 20;
	uint8_t ui8Status = 0;
	
	// 重启 BMI160
	BMI160_Write_Reg(AM_DEVICES_BMI160_CMD, 0xB6);

	while (ui8Status != 0x24 && ui8Attempts--)
	{
		// BMI160 陀螺仪 通用模式
		BMI160_Write_Reg(AM_DEVICES_BMI160_CMD, 0x15);
	
		// BMI160 加速器 low power
		BMI160_Write_Reg(AM_DEVICES_BMI160_CMD, 0x12);
		BMI160_Read_Reg(AM_DEVICES_BMI160_PMU_STATUS, 1);
		ui8Status =	I2C_Buffer_Write[0];
		
#if GYROSCOPE_DEBUG		
		printf ("PMU_STATUS=%02Xr\n", I2C_Buffer_Write[0]);
#endif
	}

	BMI160_Read_Reg(AM_DEVICES_BMI160_ERR_REG, 1);
#if GYROSCOPE_DEBUG	
	printf ("_ERR_REG=%02X\r\n", I2C_Buffer_Write[0]);
#endif
	
	
	BMI160_Write_Reg(AM_DEVICES_BMI160_INT_MOTION_0, 0x00);
	BMI160_Write_Reg(AM_DEVICES_BMI160_INT_MOTION_1, 0x14);
	//BMI160_Write_Reg(AM_DEVICES_BMI160_INT_MOTION_2, 0xFF);
	BMI160_Write_Reg(AM_DEVICES_BMI160_INT_MOTION_3, 0x02);
	
	
	BMI160_Write_Reg(AM_DEVICES_BMI160_INT_OUT_CTRL, 0x0A);
	
	//BMI160_Write_Reg(AM_DEVICES_BMI160_INT_MAP_2, 0x07);
	//BMI160_Write_Reg(AM_DEVICES_BMI160_INT_MAP_1, 0x07);
	BMI160_Write_Reg(AM_DEVICES_BMI160_INT_MAP_0, 0x07);
	
	// INT_EN_0
	BMI160_Write_Reg(AM_DEVICES_BMI160_INT_EN_0, 0x07);
	BMI160_Write_Reg(AM_DEVICES_BMI160_INT_EN_1, 0x07);
	BMI160_Write_Reg(AM_DEVICES_BMI160_INT_EN_2, 0x07);
	
}


// 读取I2c数据。 
void i2c_read_data()
{

    HAL_Delay(100);

    I2C_Buffer_Write[0] = 0x0c;
    I2C_Buffer_Write[1] = 0x0c;

    HAL_I2C_Master_Transmit(&hi2c1, MBI160_ADDRESS, I2C_Buffer_Write, 1, 1000);

    HAL_Delay(5);
    memset(I2C_Buffer_Write, 0, sizeof(I2C_Buffer_Write));
    HAL_I2C_Master_Receive(&hi2c1, MBI160_ADDRESS, I2C_Buffer_Write, 6, 1000);

    gyr_x  = I2C_Buffer_Write[0] | (I2C_Buffer_Write[1] << 8);
    if(gyr_x > 0x7fff)
    {
        gyr_x = -(0xffff - gyr_x);
    }
    gyr_x = (gyr_x * 2000) / 0x8000;

    gyr_y  = I2C_Buffer_Write[2] | (I2C_Buffer_Write[3] << 8);
    if(gyr_y > 0x7fff)
    {
        gyr_y = -(0xffff - gyr_y);
    }

    gyr_y = (gyr_y * 2000) / 0x8000;

    gyr_z  = I2C_Buffer_Write[4] | (I2C_Buffer_Write[5] << 8);
    if(gyr_z > 0x7fff)
    {
        gyr_z = -(0xffff - gyr_z);
    }
    gyr_z = (gyr_z * 2000) / 0x8000;

#if GYROSCOPE_DEBUG
    printf("read i2c=\r\n %02x, %02x, %02x, %02x, %02x, %02x\r\n",
           I2C_Buffer_Write[0], I2C_Buffer_Write[1], I2C_Buffer_Write[2],
           I2C_Buffer_Write[3], I2C_Buffer_Write[4], I2C_Buffer_Write[5]);
    printf("gyr_x=%d, gyr_y=%d, gyr_z=%d\r\n", gyr_x, gyr_y, gyr_z);
#endif

    if (abs(gyr_z) + abs(gyr_x) + abs(gyr_y) >= 5)
    {    
#if GYROSCOPE_DEBUG
        printf("warnning REMOVE \r\n");
#endif
    }
}

参考:
https://blog.csdn.net/wangyijieonline/article/details/52925908

<完毕>

  • 1
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
全志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
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值