计步器
最近在弄ST和瑞萨RA的课程,需要样片的可以加群申请:615061293 。
本文将介绍如何驱动和利用LIS2DUX12传感器,实现计步器感应功能。
LIS2DUX12是一款数字式智能3轴线性加速度计,其MEMS和ASIC旨在将尽可能低的电流消耗与丰富的特性(如常开抗混叠滤波、有限状态机 (FSM)、具有自适应自配置 (ASC) 的机器学习内核 (MLC))相结合。
FSM和MLC(带有ASC)为LIS2DUX12提供了始终可用的出色边缘处理能力。LIS2DUX12 MIPI I3C®从接口和嵌入式128级FIFO缓冲区构成了一系列功能,这让该加速度计在物料清单、处理能力和功耗上成为系统集成方面的参考。
LIS2DUX12具有±2g/±4g/±8g/±16g的用户可选满量程,并且可通过1.6 Hz到800 Hz的输出数据速率测量加速度。
LIS2DUX12包含专用内部引擎,用于处理运动和加速度检测,包括自由落体、唤醒、单/双/三击识别、活动/休止,以及6D/4D方向。
LIS2DUX12采用纤薄的小型塑料平面网格阵列封装(LGA),可确保在更大的温度范围(-40°C至+85°C)内正常工作。
硬件准备
首先需要准备一个开发板,这里我准备的是自己绘制的开发板,需要的可以进行申请。 主控为STM32U073CC,加速度计为LIS2DUX12
视频教学
[https://www.bilibili.com/video/BV1ci421X7bh/]
样品申请
[https://www.wjx.top/vm/OhcKxJk.aspx#]
源码下载
[https://download.csdn.net/download/qq_24312945/89283492]
步数检测说明
LIS2DUX12 集成了一个高级步数计算法,该算法在超低功耗域中运行,确保在电池受限的应用中延长电池寿命。步数计以 25 Hz 运行,不受所选设备功率模式(超低功耗、低功耗、高性能)的影响,保证超低功耗体验并与其他设备功能灵活结合。 步数计算法由以下四个阶段组成:
- 计算加速度量级信号的峰值:检测信号中的峰值以识别步数,独立于设备的方向。
- FIR 滤波器:提取相关频率成分并通过去除高频信号来平滑信号。
- 峰值检测器:找到波形的最大值和最小值并计算峰峰值。
- 步数计数:如果峰峰值大于设定的阈值,则计为一步。
通信模式
对于LIS2DW12,可以使用SPI或者IIC进行通讯。 最小系统图如下所示。
在CS管脚为1的时候,为IIC模式
本文使用的板子原理图如下所示。
管脚定义
IIC通信模式
在使用IIC通讯模式的时候,SA0是用来控制IIC的地址位的。
对于IIC的地址,可以通过SDO/SA0引脚修改。SDO/SA0引脚可以用来修改设备地址的最低有效位。如果SDO/SA0引脚连接到电源电压,LSb(最低有效位)为’1’(地址0011001b);否则,如果SDO/SA0引脚连接到地线,LSb的值为’0’(地址0011000b)。
对应的IIC接口如下所示。 主要使用的管脚为CS、SCL、SDA、SA0。
速率
该模块支持的速度为普通模式(100k)到快速模式+(1M)。
生成STM32CUBEMX
用STM32CUBEMX生成例程,这里使用MCU为STM32U073CC。 配置时钟树,配置时钟为48M。
IIC配置
配置IIC为快速模式,速度为400k。
INT配置
INT1管脚为PB1。
配置如下所示。
开启中断。
配置LED指示灯。
stm32u0xx_it.c添加中断触发后LED翻转电平的代码,添加对应变量。
/* USER CODE BEGIN 0 */
extern uint8_t step_event ;
/* USER CODE END 0 */
添加中断代码。
/**
* @brief This function handles EXTI line 0 and line 1 interrupts.
*/
void EXTI0_1_IRQHandler(void)
{
/* USER CODE BEGIN EXTI0_1_IRQn 0 */
HAL_GPIO_TogglePin(GPIOA, GPIO_PIN_1);
step_event = 1;
/* USER CODE END EXTI0_1_IRQn 0 */
HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_1);
/* USER CODE BEGIN EXTI0_1_IRQn 1 */
/* USER CODE END EXTI0_1_IRQn 1 */
}
串口配置
查看原理图,PA9和PA10设置为开发板的串口。
配置串口。
CS和SA0设置
串口重定向
打开魔术棒,勾选MicroLIB
在main.c中,添加头文件,若不添加会出现 identifier "FILE" is undefined报错。
/* USER CODE BEGIN Includes */
#include "stdio.h"
/* USER CODE END Includes */
函数声明和串口重定向:
/* USER CODE BEGIN PFP */
int fputc(int ch, FILE *f){
HAL_UART_Transmit(&huart1 , (uint8_t *)&ch, 1, 0xFFFF);
return ch;
}
/* USER CODE END PFP */
参考程序
[https://github.com/STMicroelectronics/lis2dux12-pid]
计步器参考
[https://github.com/stm32duino/LIS2DUXS12/tree/main/examples]
初始换管脚
由于需要向LIS2DUX12_I2C_ADD_L写入以及为IIC模式。
所以使能CS为高电平,配置为IIC模式。 配置SA0为低电平。
HAL_GPIO_WritePin(GPIOC, CS_Pin, GPIO_PIN_SET);
HAL_GPIO_WritePin(GPIOC, SA0_Pin, GPIO_PIN_SET);
获取ID
我们可以向WHO_AM_I (0Fh)获取固定值,判断是否为0x47。
lis2dux12_device_id_get为获取函数。
对应的获取ID驱动程序,如下所示。
printf("HELLOn");
HAL_GPIO_WritePin(CS_GPIO_Port, CS_Pin, GPIO_PIN_SET);
HAL_GPIO_WritePin(SA0_GPIO_Port, SA0_Pin, GPIO_PIN_RESET);
lis2dux12_status_t status;
uint8_t id;
lis2dux12_md_t md;
uint16_t steps = 0;
lis2dux12_int_config_t int_mode;
/* Initialize mems driver interface */
dev_ctx.write_reg = platform_write;
dev_ctx.read_reg = platform_read;
dev_ctx.mdelay = platform_delay;
dev_ctx.handle = &SENSOR_BUS;
/* Wait sensor boot time */
platform_delay(BOOT_TIME);
lis2dux12_exit_deep_power_down(&dev_ctx);// 退出深度休眠模式
/* Check device ID */
lis2dux12_device_id_get(&dev_ctx, &id);
printf("LIS2DUX12_ID=0x%x,id=0x%xn",LIS2DUX12_ID,id);
if (id != LIS2DUX12_ID)
while(1);
/* Restore default configuration */
lis2dux12_init_set(&dev_ctx, LIS2DUX12_RESET);// 重置设备
do {
lis2dux12_status_get(&dev_ctx, &status);// 获取设备状态
} while (status.sw_reset);// 获取设备状态
复位操作
可以向CTRL1 (10h)的SW_RESET寄存器写入1进行复位。
lis2dux12_init_set为重置函数。 对应的驱动程序,如下所示。
/* Restore default configuration */
lis2dux12_init_set(&dev_ctx, LIS2DUX12_RESET);
do {
lis2dux12_status_get(&dev_ctx, &status);
} while (status.sw_reset);
BDU设置
在很多传感器中,数据通常被存储在输出寄存器中,这些寄存器分为两部分:MSB和LSB。这两部分共同表示一个完整的数据值。例如,在一个加速度计中,MSB和LSB可能共同表示一个加速度的测量值。
连续更新模式(BDU = ‘0’):在默认模式下,输出寄存器的值会持续不断地被更新。这意味着在你读取MSB和LSB的时候,寄存器中的数据可能会因为新的测量数据而更新。这可能导致一个问题:当你读取MSB时,如果寄存器更新了,接下来读取的LSB可能就是新的测量值的一部分,而不是与MSB相对应的值。这样,你得到的就是一个“拼凑”的数据,它可能无法准确代表任何实际的测量时刻。
块数据更新(BDU)模式(BDU = ‘1’):当激活BDU功能时,输出寄存器中的内容不会在读取MSB和LSB之间更新。这就意味着一旦开始读取数据(无论是先读MSB还是LSB),寄存器中的那一组数据就被“锁定”,直到两部分都被读取完毕。这样可以确保你读取的MSB和LSB是同一测量时刻的数据,避免了读取到代表不同采样时刻的数据。
简而言之,BDU位的作用是确保在读取数据时,输出寄存器的内容保持稳定,从而避免读取到拼凑或错误的数据。这对于需要高精度和稳定性的应用尤为重要。
可以向CTRL4 (13h)的BDU寄存器写入1进行开启。
对应的驱动程序,如下所示。
/* Set bdu and if_inc recommended for driver usage */
lis2dux12_init_set(&dev_ctx, LIS2DUX12_SENSOR_ONLY_ON);
设置传感器的量程
FS[1:0] - 全量程选择:这两个位用于设置传感器的量程。量程决定了传感器可以测量的最大加速度值。例如,量程可以设置为±2g、±4g、±8g或±16g。这允许用户根据应用的特定需求调整传感器的灵敏度。
对应的驱动程序,如下所示。
/* Set Output Data Rate */
md.fs = LIS2DUX12_4g;
md.odr = LIS2DUX12_25Hz_LP;
lis2dux12_mode_set(&dev_ctx, &md);
启用步数计和嵌入式功能
EMB_FUNC_EN 是一个寄存器位,用于启用 LIS2DUX12 传感器的嵌入式功能。这些嵌入式功能可能包括传感器的各种高级特性,如步数计、活动检测、自由落体检测等。这些功能通过内部算法处理传感器数据,而不需要主机处理器进行复杂计算,从而减轻了主机处理器的负担。
if_add_inc 是一个控制寄存器位,用于启用或禁用自动地址递增功能。在多字节访问模式下,当你读取或写入多个连续的寄存器时,启用地址递增功能可以使得寄存器地址自动增加。这在读取或写入一系列数据时非常有用,因为它可以减少通信开销并简化代码。
// Enable Pedometer.
/* 启用传感器的嵌入式功能(例如,步数计),这通常推荐用于驱动程序 */
lis2dux12_init_set(&dev_ctx, LIS2DUX12_SENSOR_EMB_FUNC_ON);// 启用嵌入式功能
platform_delay(10);
PEDO_EN 是一个用于启用步数检测功能的控制位。启用此功能后,传感器可以进行步数计数。这在运动检测和健康监控应用中非常重要,因为它可以记录用户的步数数据。
lis2dux12_stpcnt_mode_t mode;
lis2dux12_stpcnt_mode_get(&dev_ctx, &mode);// 获取步数计模式
/* Enable pedometer algorithm. */
mode.step_counter_enable = PROPERTY_ENABLE;// 启用步数计
mode.false_step_rej = PROPERTY_DISABLE;// 禁用误步拒绝功能
mode.step_counter_in_fifo = PROPERTY_DISABLE; // 禁用步数计 FIFO
/* Turn on embedded features */
lis2dux12_stpcnt_mode_set(&dev_ctx, mode) ;// 设置步数计模式
配置步数检测中断引脚
INT1_STEP_DET 是一个用于步数检测中断的寄存器位。当启用这个中断位时,LIS2DUX12 传感器在检测到步数事件时会触发中断信号,并将该信号输出到指定的中断引脚(如 INT1)。这使得主机微控制器可以通过中断响应步数检测事件,而不需要不断轮询传感器的状态。
INT1_EMB_FUNC 是一个用于配置 LIS2DUX12 传感器的嵌入式功能中断的寄存器位。当该位启用时,传感器会在嵌入式功能(例如步数检测、活动检测等)触发事件时通过 INT1 引脚发出中断信号。这个功能使得主机微控制器可以通过中断快速响应传感器的嵌入式功能事件,而不需要持续轮询传感器状态。
lis2dux12_emb_pin_int_route_t emb_pin_int;
lis2dux12_emb_pin_int1_route_get(&dev_ctx, &emb_pin_int);// 设置步数计模式
emb_pin_int.step_det = PROPERTY_ENABLE;// 设置步数计模式
lis2dux12_emb_pin_int1_route_set(&dev_ctx, &emb_pin_int);// 设置嵌入式中断引脚
中断检测步数
可以通过判断 INT1_STEP_DET 位来确定步数事件是否产生。根据您提供的截图,INT1_STEP_DET 位位于 EMB_FUNC_INT1 寄存器中。当这个位被设置为 1 时,表示步数检测中断被路由到 INT1 引脚。因此,您可以通过读取该寄存器来判断步数事件是否产生。
EMB_FUNC_REG_ACCESS 是用于访问和配置 LIS2DUX12 传感器的嵌入式功能寄存器的寄存器位。启用 EMB_FUNC_REG_ACCESS 后,可以访问和配置与嵌入式功能相关的寄存器。这对于启用步数检测、活动检测、自由落体检测等功能非常重要。
获取步数可以通过STEP_COUNTER_L (28h) 和 STEP_COUNTER_H (29h)进行获取。
获取完毕之后关闭EMB_FUNC_REG_ACCESS 。
/* Infinite loop */
/* USER CODE BEGIN WHILE */
while (1)
{
if (step_event) {
lis2dux12_emb_pin_int_route_t emb_pin_int1;
lis2dux12_emb_pin_int1_route_get(&dev_ctx, &emb_pin_int1);
if (emb_pin_int1.step_det)
{
lis2dux12_stpcnt_steps_get(&dev_ctx, &steps);
/* print number of steps */
printf("Steps: %drn", steps);
}
step_event = 0;
}
HAL_Delay(10);
/* USER CODE END WHILE */
/* USER CODE BEGIN 3 */
}
/* USER CODE END 3 */
演示
审核编辑 黄宇
-
传感器
+关注
关注
2548文章
50642浏览量
751670 -
加速度计
+关注
关注
6文章
698浏览量
45853 -
计步器
+关注
关注
4文章
76浏览量
19897 -
stm32cubemx
+关注
关注
5文章
280浏览量
14746
发布评论请先 登录
相关推荐
评论