驱动LSM6DS3TR-C实现高效运动检测与数据采集(8)----中断获取FIFO数据并应用MotionFX库解析空间坐标

驱动LSM6DS3TR-C实现高效运动检测与数据采集.8--中断获取FIFO数据并应用MotionFX库解析空间坐标

  • 概述
  • 视频教学
  • 样品申请
  • 源码下载
  • 开启LED
  • 开启INT中断
  • 参考驱动程序
  • 中断读取传感器数据
  • 主程序
  • 演示

概述

本文将探讨如何使用中断机制获取FIFO数据并应用MotionFX库解析空间坐标。MotionFX库是一种用于传感器融合的强大工具,可以将加速度计、陀螺仪和磁力计的数据融合在一起,实现精确的姿态和位置估计。本文将介绍如何初始化和配置MotionFX库,使用中断机制读取FIFO中的传感器数据。FIFO可以作为数据缓冲区,存储传感器的临时数据,防止数据丢失,特别是在处理器忙于其他任务时。本文将利用这些数据进行空间坐标的解析。本章案例基于上节的demo进行修改。

需要样片的可以加群申请:615061293 。

在这里插入图片描述

视频教学

https://www.bilibili.com/video/BV1pm421G7XE/

驱动LSM6DS3TR-C实现高效运动检测与数据采集(8)----中断获取FIFO数据并应用MotionFX库解析空间坐标

样品申请

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

源码下载

开启LED

配置PB14为输出模式。
在这里插入图片描述

在这里插入图片描述

开启INT中断

陀螺仪LSM6DS3TR-C的中断管脚接到了PB0,需要将PB0设置为中端口。

在这里插入图片描述
在这里插入图片描述

开启中断。

在这里插入图片描述

参考驱动程序

https://github.com/STMicroelectronics/lsm6ds3tr-c-pid

中断读取传感器数据

为了使用回调函数并获取FIFO中的数据,在main.c定义了以下变量。

float acc_x,acc_y,acc_z;
float gyr_x,gyr_y,gyr_z;
uint32_t deltatime_1,deltatime_2;
uint8_t deltatime_first=0;stmdev_ctx_t dev_ctx;uint8_t waterm = 0;
/// 用于存储FIFO中读取的数据,每条数据包含24个字节,*2保证数据不溢出
uint8_t fifo_data[20*3*2][6];
// FIFO中当前存储的数据数量
uint16_t fifo_num = 0;
// FIFO中断标志,用于标记是否有新的FIFO数据可供读取
uint8_t fifo_flag=0;uint8_t acc_fifo[20*2][6];
uint8_t gyr_fifo[20*2][6];
uint8_t timestamp_fifo[20*2][6];

mian.c中开启中断。

  lsm6ds3tr_c_int1_route_t pin_int;lsm6ds3tr_c_pin_int1_route_get(&dev_ctx, &pin_int);pin_int.int1_fifo_ovr = PROPERTY_ENABLE;lsm6ds3tr_c_pin_int1_route_set(&dev_ctx, pin_int);	

在这里插入图片描述

需要注意优化等级。

在这里插入图片描述

完整初始化如下所示。

  /* USER CODE BEGIN 2 */printf("HELLO!\n");HAL_GPIO_WritePin(CS1_GPIO_Port, CS1_Pin, GPIO_PIN_SET);HAL_GPIO_WritePin(SA0_GPIO_Port, SA0_Pin, GPIO_PIN_RESET);HAL_GPIO_WritePin(CS2_GPIO_Port, CS2_Pin, GPIO_PIN_SET);HAL_Delay(100);/* 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;/* Init test platform */
//  platform_init();/* Wait sensor boot time */platform_delay(BOOT_TIME);/* Check device ID */whoamI = 0;lsm6ds3tr_c_device_id_get(&dev_ctx, &whoamI);printf("LSM6DS3TR-C_ID=0x%x,whoamI=0x%x",LSM6DS3TR_C_ID,whoamI);if ( whoamI != LSM6DS3TR_C_ID )while (1); /*manage here device not found *//* Restore default configuration */lsm6ds3tr_c_reset_set(&dev_ctx, PROPERTY_ENABLE);do {lsm6ds3tr_c_reset_get(&dev_ctx, &rst);} while (rst);/* 设置加速度计和陀螺仪的满量程范围 */lsm6ds3tr_c_xl_full_scale_set(&dev_ctx, LSM6DS3TR_C_4g);lsm6ds3tr_c_gy_full_scale_set(&dev_ctx, LSM6DS3TR_C_2000dps);/* 启用块数据更新(BDU),当FIFO支持时 */lsm6ds3tr_c_block_data_update_set(&dev_ctx, PROPERTY_ENABLE);lsm6ds3tr_c_xl_power_mode_set(&dev_ctx, LSM6DS3TR_C_XL_HIGH_PERFORMANCE);	/* 设置加速度计和陀螺仪的输出数据速率:* 在本例中,我们将加速度计和陀螺仪的速率设置为26 Hz*/lsm6ds3tr_c_xl_data_rate_set(&dev_ctx, LSM6DS3TR_C_XL_ODR_416Hz);lsm6ds3tr_c_gy_data_rate_set(&dev_ctx, LSM6DS3TR_C_GY_ODR_416Hz);lsm6ds3tr_c_fifo_mode_set(&dev_ctx, LSM6DS3TR_C_BYPASS_MODE);	HAL_Delay(10);/* 设置FIFO水印为模式的倍数* 在本例中,我们将水印设置为10个模式* 这意味着10个序列:* (陀螺仪 + 加速度计) = 12字节* (外部传感器 + 时间戳) = 12字节*/lsm6ds3tr_c_int1_route_t int_1_reg;uint16_t pattern_len = 24;  // 每个数据集由6个字节组成,4*6=24lsm6ds3tr_c_fifo_watermark_set(&dev_ctx, 10 * pattern_len);/* 将FIFO模式设置为流模式 *///FIFO_CTRL5(0x0A)->STREAM_MODElsm6ds3tr_c_fifo_mode_set(&dev_ctx, LSM6DS3TR_C_STREAM_MODE);/* 启用时间戳并将其添加到FIFO *///CTRL10_C (19h)->TIMER_ENlsm6ds3tr_c_timestamp_set(&dev_ctx, PROPERTY_ENABLE);//CTRL10_C (19h)->PEDO_EN	lsm6ds3tr_c_pedo_sens_set(&dev_ctx, PROPERTY_ENABLE); // 根据需求配置步数计数/* 将时间戳分辨率设置为25 μs (WAKE_UP_DUR寄存器中的TIMER_HR位) *///WAKE_UP_DUR (5Ch)->TIMER_HRlsm6ds3tr_c_timestamp_res_set(&dev_ctx, LSM6DS3TR_C_LSB_25us);//设置第3数据集(Dataset 3)的降采样因子lsm6ds3tr_c_fifo_dataset_3_batch_set(&dev_ctx, LSM6DS3TR_C_FIFO_DS3_NO_DEC);	//设置第4数据集(Dataset 4)的降采样因子//FIFO_CTRL4 (09h)->DEC_DS4_FIFO[2:0]lsm6ds3tr_c_fifo_dataset_4_batch_set(&dev_ctx, LSM6DS3TR_C_FIFO_DS4_NO_DEC);// 启用时间戳写入FIFO第四数据集//FIFO_CTRL2 (07h)->TIMER_PEDO_FIFO_ENlsm6ds3tr_c_fifo_pedo_and_timestamp_batch_set(&dev_ctx, PROPERTY_ENABLE);			/* 设置FIFO传感器的降采样因子 */lsm6ds3tr_c_fifo_xl_batch_set(&dev_ctx, LSM6DS3TR_C_FIFO_XL_NO_DEC);lsm6ds3tr_c_fifo_gy_batch_set(&dev_ctx, LSM6DS3TR_C_FIFO_GY_NO_DEC);/* 设置FIFO的输出数据速率 *///FIFO_CTRL5 (0Ah)lsm6ds3tr_c_fifo_data_rate_set(&dev_ctx, LSM6DS3TR_C_FIFO_416Hz);lsm6ds3tr_c_int1_route_t pin_int;lsm6ds3tr_c_pin_int1_route_get(&dev_ctx, &pin_int);pin_int.int1_fth = PROPERTY_ENABLE;lsm6ds3tr_c_pin_int1_route_set(&dev_ctx, pin_int);	lsm6ds3tr_c_init();/* USER CODE END 2 */

在stm32h5xx_it.c中添加回调函数引用。

/* USER CODE BEGIN 0 */
extern void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin);/* USER CODE END 0 */

处理PB0外部中断线0(EXTI Line0)的中断。

/*** @brief This function handles EXTI Line0 interrupt.*/
void EXTI0_IRQHandler(void)
{/* USER CODE BEGIN EXTI0_IRQn 0 */HAL_GPIO_EXTI_Callback(GPIO_PIN_0);/* USER CODE END EXTI0_IRQn 0 */HAL_GPIO_EXTI_IRQHandler(INT1_Pin);/* USER CODE BEGIN EXTI0_IRQn 1 *//* USER CODE END EXTI0_IRQn 1 */
}

在main.c中添加回调函数的定义,检查中断是否由 GPIO_PIN_0 引脚触发,每次发生中断时从传感器获取当前的FIFO状态,并存储在 fifo_status 变量中。读取FIFO数据,并将这些数据存储在一个全局数组 fifo_data 中,以便在主循环或其他地方进行处理。通过切换 LED 的状态,可以直观地了解中断的发生。

/* USER CODE BEGIN 4 */void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin){if(GPIO_Pin == GPIO_PIN_0){HAL_GPIO_TogglePin(LED1_GPIO_Port, LED1_Pin);/* 读取LSM6DS3TR-C的水印标志 */lsm6ds3tr_c_fifo_wtm_flag_get(&dev_ctx, &waterm);		uint16_t num = 0,num1=0;uint16_t num_pattern = 0;if (waterm) {fifo_flag=1;/* 读取FIFO中的字数 */lsm6ds3tr_c_fifo_data_level_get(&dev_ctx, &num);		num_pattern = num / 24*2;		
//			printf( "-- FIFO num %d num_pattern=%d\r\n", num,num_pattern);fifo_num=num_pattern;for(int i=0;i<num_pattern;i++)  {/* 根据传感器的ODR配置,FIFO模式由以下样本序列组成:GYRO, XL 外部传感器 时间戳*/lsm6ds3tr_c_fifo_raw_data_get(&dev_ctx,&gyr_fifo[i][0],3 * sizeof(int16_t));				lsm6ds3tr_c_fifo_raw_data_get(&dev_ctx,&acc_fifo[i][0],6);						//外部传感器数据				lsm6ds3tr_c_fifo_raw_data_get(&dev_ctx,data_raw_none.u8bit,3 * sizeof(int16_t));						lsm6ds3tr_c_fifo_raw_data_get(&dev_ctx,&timestamp_fifo[i][0],3 * sizeof(int16_t));			}	}}
}
/* USER CODE END 4 */

主程序

在主循环中检查FIFO中断标志,如果有新的FIFO数据,则读取并处理这些数据。处理完成后,调用MotionFX库函数进行数据融合计算,以获得传感器的姿态和位置。

  /* Infinite loop *//* USER CODE BEGIN WHILE */while (1){if(fifo_flag){for(int i=0;i<fifo_num;i++)// 遍历 FIFO 数据数组{	int16_t gyr;				gyr=(gyr_fifo[i][1]<<8) + gyr_fifo[i][0];gyr_x =lsm6ds3tr_c_from_fs2000dps_to_mdps(gyr);gyr=(gyr_fifo[i][3]<<8) + gyr_fifo[i][2];				gyr_y =lsm6ds3tr_c_from_fs2000dps_to_mdps(gyr);gyr=(gyr_fifo[i][5]<<8) + gyr_fifo[i][4];				gyr_z =lsm6ds3tr_c_from_fs2000dps_to_mdps(gyr);
//			printf(
//							"gyr_x:%4.2f\t%4.2f\t%4.2f\r\n",
//							gyr_x, gyr_y, gyr_z);			int16_t acc;acc=(acc_fifo[i][1]<<8) + acc_fifo[i][0];acc_x =lsm6ds3tr_c_from_fs4g_to_mg(acc);acc=(acc_fifo[i][3]<<8) + acc_fifo[i][2];				acc_y =lsm6ds3tr_c_from_fs4g_to_mg(acc);acc=(acc_fifo[i][5]<<8) + acc_fifo[i][4];				acc_z =lsm6ds3tr_c_from_fs4g_to_mg(acc);//			printf(
//							"acc_x:%4.2f\t%4.2f\t%4.2f\r\n",
//							acc_x, acc_y, acc_z);	/* 读取时间戳数据 */uint32_t timestamp=0;timestamp=(timestamp_fifo[i][1]<<16)|(timestamp_fifo[i][0]<<8)|(timestamp_fifo[i][3]);
//        printf("Timestamp: %u\r\n", timestamp);		if(deltatime_first==0)//第一次{deltatime_1=timestamp;deltatime_2=deltatime_1;deltatime_first=1;}else{deltatime_2=timestamp;}lsm6ds3tr_c_motion_fx_determin();	deltatime_1=deltatime_2;}fifo_flag=0;				}/* USER CODE END WHILE *//* USER CODE BEGIN 3 */}/* USER CODE END 3 */

演示

初始位置和数据输出如下所示。

在这里插入图片描述
在这里插入图片描述

逆时针旋转90°

在这里插入图片描述

在这里插入图片描述

逆时针旋转180°
在这里插入图片描述

在这里插入图片描述
逆时针旋转270°

在这里插入图片描述

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/45943.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

在 PostgreSQL 里如何处理数据的索引碎片整理的自动化?

&#x1f345;关注博主&#x1f397;️ 带你畅游技术世界&#xff0c;不错过每一次成长机会&#xff01;&#x1f4da;领书&#xff1a;PostgreSQL 入门到精通.pdf 文章目录 PostgreSQL 中数据索引碎片整理的自动化处理 PostgreSQL 中数据索引碎片整理的自动化处理 在数据库管…

接口基础知识2:http通信的组成

课程大纲 一、http协议 HTTP&#xff08;Hypertext Transfer Protocol&#xff0c;超文本传输协议&#xff09;是互联网中被使用最广的一种网络协议&#xff0c;用于客户端与服务器之间的通信。 HTTP协议定义了一系列的请求方法&#xff0c;例如 GET、POST、PUT、DELETE 等&…

特惠电影票api安全性如何评测

评测特惠电影票API的安全性是确保用户数据安全和系统稳定运行的关键步骤。以下是评测特惠电影票API安全性的一些方法和步骤&#xff1a; ### 1. **认证和授权** - **JWT认证**&#xff1a;使用JSON Web Token (JWT) 进行用户身份验证和授权&#xff0c;确保只有合法用户可以访…

Ubuntu搭建Android架构so库交叉编译环境

目录 前言一、下载NDK并安装二、安装NDK三、配置交叉编译工具链四、编写交叉编译脚本 前言 需要将一些源码编译成Android可用的架构的so库 一、下载NDK并安装 https://developer.android.google.cn/ndk/downloads/ 二、安装NDK 将下载下来的android-ndk-r23b-linux.zip解压…

uniapp内置组件scroll-view案例解析

参考资料 文档地址&#xff1a;https://uniapp.dcloud.net.cn/component/scroll-view.html 官方给的完整代码 <script>export default {data() {return {scrollTop: 0,old: {scrollTop: 0}}},methods: {upper: function(e) {console.log(e)},lower: function(e) {cons…

【SVPWM】空间矢量调制

SVPWM仿真模型 扇区315462 切换时间马鞍波 三角载波和马鞍波比较 三相电压 FFT分析 参考文献&#xff1a; 现代永磁同步电机控制原理及MATLAB仿真.袁雷.

WPF学习(3) -- 控件模板

一、操作过程 二、代码 <Window x:Class"学习.MainWindow"xmlns"http://schemas.microsoft.com/winfx/2006/xaml/presentation"xmlns:x"http://schemas.microsoft.com/winfx/2006/xaml"xmlns:d"http://schemas.microsoft.com/expressio…

树莓派pico入坑笔记,esp01/01s使用

目录 关于树莓派pico和circuitpython的更多玩法&#xff0c;请看树莓派pico专栏 说明 关于at指令 WiFi的at指令 UDP的at指令 样例程序 调试助手端输入指令 sta端程序 效果 进阶使用 库函数说明 样例代码 关于树莓派pico和circuitpython的更多玩法&#xff0c;请看树…

秋招突击——7/12——复习{每日温度、完全平方数、无重复最长子串}——新作{字节面试——控制多线程按照顺序输出}

文章目录 引言复习每日温度复习实现参考学习 完全平方数复习实现参考学习 无重复字符的最长子串复习实现参考学习 新作控制多线程输出Java实现线程——不使用锁实现使用synchronized关键实现——使用锁实现使用synchronized、wait和notify关键字实现 总结 引言 今天又要面试字…

安全防御,防火墙配置NAT转换智能选举综合实验

一、实验拓扑图 二、实验需求 1、办公区设备可以通过电信链路和移动链路上网(多对多的NAT&#xff0c;并且需要保留一个公网IP不能用来转换) 2、分公司设备可以通过总公司的移动链路和电信链路访问到Dmz区的http服务器 3、多出口环境基于带宽比例进行选路&#xff0c;但是&…

DP(3) | 0-1背包 | Java | LeetCode 1049, 494, 474 做题总结(474未完)

1049. 最后一块石头的重量 II 和 LC 416.分割等和子集 类似 思路&#xff08;我没有思路&#xff09;&#xff1a; 两块石头相撞&#xff0c;这里没有想到的一个点是&#xff0c;相撞的两个石头要几乎相似 以示例1为例&#xff0c;stones [2,7,4,1,8,1]&#xff0c;如果从左到…

Collections:专为集合框架而生的工具类

Collections 是 JDK 提供的一个工具类&#xff0c;位于 java.util 包下&#xff0c;提供了一系列的静态方法。 排序操作 reverse(List list)&#xff1a;反转顺序shuffle(List list)&#xff1a;洗牌&#xff0c;将顺序打乱sort(List list)&#xff1a;自然升序sort(List lis…

解答|服务器只能开22端口可以申请IP地址SSL证书吗?

IP地址SSL证书&#xff0c;是一种专门颁发给公网IP地址的SSL证书&#xff0c;而不是常见的基于域名的SSL证书。SSL证书主要用于保障数据在客户端&#xff08;如用户的浏览器&#xff09;和服务器之间传输时的加密性和安全性&#xff0c;以防止数据被截取或篡改。 服务器只能开…

github actions方式拉取docker镜像

参考&#xff1a; https://wkdaily.cpolar.cn/archives/gc 注意github actions提供的免费虚拟机空间有限&#xff0c;空间不足会报错&#xff0c;查看大概语句有10来G 我在workflow file里加了df -h 运行查看磁盘情况&#xff1a; 通过pwd命令&#xff0c;可以知道运行目录/ho…

ETL数据集成丨主流ETL工具(ETLCloud、DataX、Kettle)数据传输性能大PK

目前市面上的ETL工具众多&#xff0c;为了方便广大企业用户在选择ETL工具时有一个更直观性能方面的参考值&#xff0c;我们选取了目前市面上最流行的三款ETL工具&#xff08;ETLCloud、DataX、Kettle&#xff09;来作为本次性能传输的代表&#xff0c;虽然性能测试数据有很多相…

【JavaScript】解决 JavaScript 语言报错:Uncaught TypeError: XYZ is not a function

文章目录 一、背景介绍常见场景 二、报错信息解析三、常见原因分析1. 变量或对象属性类型错误2. 函数名拼写错误或覆盖3. 作用域问题导致的函数未定义4. 调用未初始化的函数 四、解决方案与预防措施1. 确保变量类型正确2. 检查拼写错误3. 注意作用域4. 初始化变量 五、示例代码…

C#中的反射

dll和exe文件的区别 用途&#xff1a; .exe&#xff08;可执行文件&#xff09;&#xff1a;是可以直接运行的程序文件。当你双击一个 .exe 文件或在命令行中输入它的名字&#xff0c;操作系统会加载并执行这个程序。 .dll&#xff08;动态链接库&#xff09;&#xff1a;包含…

graphviz subgraph添加边界框

subgraph name 属性必须要以cluster开头。 A Quick Introduction to GraphvizAn awesome tool for software documentation and visualizing graphshttps://www.worthe-it.co.za/blog/2017-09-19-quick-introduction-to-graphviz.html digraph {rankdir"LR"// the n…

【探索Linux】P.39(传输层 —— TCP的三次 “握手” 和四次 “挥手” )

阅读导航 引言一、TCP的三次握手1. 简介2. 图解三次握手3. 名词解释&#xff08;1&#xff09;SYN&#xff08;同步序列编号&#xff09;包&#xff08;2&#xff09;SYN-ACK&#xff08;同步确认&#xff09;包&#xff08;3&#xff09;ACK&#xff08;确认&#xff09;包 4.…

基于matlab的SVR回归模型

1 原理 SVR&#xff08;Support Vector Regression&#xff09;回归预测原理&#xff0c;基于支持向量机&#xff08;SVM&#xff09;的回归分支&#xff0c;其核心思想是通过寻找一个最优的超平面来进行回归预测&#xff0c;并处理非线性回归问题。以下是SVR回归预测原理的系统…