三轴加速度计LIS2DUX12开发(3)----计步器

112 篇文章 58 订阅
68 篇文章 3 订阅

计步器

最近在弄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/

三轴加速度计LIS2DUX12开发(3)----计步器

样品申请

https://www.wjx.top/vm/OhcKxJk.aspx#

源码下载

https://download.csdn.net/download/qq_24312945/89283492

步数检测说明

LIS2DUX12 集成了一个高级步数计算法,该算法在超低功耗域中运行,确保在电池受限的应用中延长电池寿命。步数计以 25 Hz 运行,不受所选设备功率模式(超低功耗、低功耗、高性能)的影响,保证超低功耗体验并与其他设备功能灵活结合。
步数计算法由以下四个阶段组成:

  1. 计算加速度量级信号的峰值:检测信号中的峰值以识别步数,独立于设备的方向。
  2. FIR 滤波器:提取相关频率成分并通过去除高频信号来平滑信号。
  3. 峰值检测器:找到波形的最大值和最小值并计算峰峰值。
  4. 步数计数:如果峰峰值大于设定的阈值,则计为一步。
    在这里插入图片描述

通信模式

对于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("HELLO\n");
	
	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%x\n",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: %d\r\n", steps);
				}		
      step_event = 0;
    }		

		HAL_Delay(10);
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */

演示

在这里插入图片描述

  • 29
    点赞
  • 27
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论
以下是一个基于 LIS2DH12 三轴加速度传感器的算法来判断状态的C代码: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <stdbool.h> #include <wiringPiI2C.h> #define ACCEL_ADDR 0x18 #define ACCEL_CTRL_REG1 0x20 #define ACCEL_CTRL_REG4 0x23 #define ACCEL_X_AXIS 0x28 #define ACCEL_Y_AXIS 0x2A #define ACCEL_Z_AXIS 0x2C #define THRESHOLD 1.0 // 加速度变化阈值 #define SAMPLE_RATE 100 // 采样率(Hz) #define WINDOW_SIZE 10 // 窗口大小(个样本) enum State { IDLE, MOVING, FALLING }; int main(void) { int fd = wiringPiI2CSetup(ACCEL_ADDR); if(fd == -1) { printf("Failed to initialize I2C device.\n"); exit(1); } // 配置传感器 wiringPiI2CWriteReg8(fd, ACCEL_CTRL_REG1, 0x77); // 打开传感器并设置采样率 wiringPiI2CWriteReg8(fd, ACCEL_CTRL_REG4, 0x08); // 设置测量范围为 +/- 8g // 初始化前 WINDOW_SIZE 个样本的加速度值 float prev_accel_x[WINDOW_SIZE]; float prev_accel_y[WINDOW_SIZE]; float prev_accel_z[WINDOW_SIZE]; for(int i = 0; i < WINDOW_SIZE; i++) { int x = wiringPiI2CReadReg16(fd, ACCEL_X_AXIS); int y = wiringPiI2CReadReg16(fd, ACCEL_Y_AXIS); int z = wiringPiI2CReadReg16(fd, ACCEL_Z_AXIS); // 将读取到的原始数据转换为实际加速度值 prev_accel_x[i] = (float)x / 1024.0; prev_accel_y[i] = (float)y / 1024.0; prev_accel_z[i] = (float)z / 1024.0; usleep(1000000 / SAMPLE_RATE); // 等待一段时间,以达到采样率 } // 初始化状态 enum State state = IDLE; while(1) { // 读取最新的加速度值 int x = wiringPiI2CReadReg16(fd, ACCEL_X_AXIS); int y = wiringPiI2CReadReg16(fd, ACCEL_Y_AXIS); int z = wiringPiI2CReadReg16(fd, ACCEL_Z_AXIS); // 将读取到的原始数据转换为实际加速度值 float accel_x = (float)x / 1024.0; float accel_y = (float)y / 1024.0; float accel_z = (float)z / 1024.0; // 计算加速度变化值 float delta_accel_x = 0; float delta_accel_y = 0; float delta_accel_z = 0; for(int i = 0; i < WINDOW_SIZE; i++) { delta_accel_x += abs(accel_x - prev_accel_x[i]); delta_accel_y += abs(accel_y - prev_accel_y[i]); delta_accel_z += abs(accel_z - prev_accel_z[i]); } delta_accel_x /= WINDOW_SIZE; delta_accel_y /= WINDOW_SIZE; delta_accel_z /= WINDOW_SIZE; // 根据加速度变化值判断状态 switch(state) { case IDLE: if(delta_accel_x > THRESHOLD || delta_accel_y > THRESHOLD || delta_accel_z > THRESHOLD) { state = MOVING; printf("State: MOVING\n"); } break; case MOVING: if(delta_accel_x > THRESHOLD || delta_accel_y > THRESHOLD || delta_accel_z > THRESHOLD) { state = MOVING; } else { state = FALLING; printf("State: FALLING\n"); } break; case FALLING: if(delta_accel_z > THRESHOLD) { state = IDLE; printf("State: IDLE\n"); } break; } // 更新前 WINDOW_SIZE 个样本的加速度值 for(int i = WINDOW_SIZE - 1; i > 0; i--) { prev_accel_x[i] = prev_accel_x[i - 1]; prev_accel_y[i] = prev_accel_y[i - 1]; prev_accel_z[i] = prev_accel_z[i - 1]; } prev_accel_x[0] = accel_x; prev_accel_y[0] = accel_y; prev_accel_z[0] = accel_z; // 等待一段时间,以达到采样率 usleep(1000000 / SAMPLE_RATE); } return 0; } ``` 该代码使用了和前面类似的方式来读取 LIS2DH12 三轴加速度传感器的数据,并根据加速度变化的情况来判断状态。具体地,根据阈值 THRESHOLD 来判断加速度是否变化,根据窗口大小 WINDOW_SIZE 来计算加速度变化值,然后根据加速度变化值的大小来判断状态。如果加速度变化超过阈值且当前状态为 IDLE,则进入 MOVING 状态;如果加速度变化超过阈值且当前状态为 MOVING,则保持 MOVING 状态;如果加速度变化没有超过阈值且当前状态为 MOVING,则进入 FALLING 状态;如果加速度变化超过阈值且当前状态为 FALLING,则保持 FALLING 状态;如果加速度在 Z 轴方向上超过阈值且当前状态为 FALLING,则回到 IDLE 状态。注意,需要根据具体的传感器型号和配置修改代码中的地址和寄存器值,以及阈值、采样率和窗口大小的大小。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

记帖

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

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

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

打赏作者

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

抵扣说明:

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

余额充值