0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

陀螺仪LSM6DSOW开发(4)----中断获取FIFO数据

嵌入式单片机MCU开发 来源:嵌入式单片机MCU开发 作者:嵌入式单片机MCU开 2024-08-08 16:03 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

概述

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

最近在弄ST和瑞萨RA的课程,需要样片的可以加群申请:615061293 。

视频教学

[https://www.bilibili.com/video/BV1Tm42137BS/]

样品申请

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

源码下载

[https://download.csdn.net/download/qq_24312945/89602062]

硬件准备

首先需要准备一个开发板,这里我准备的是自己绘制的开发板,需要的可以进行申请。 主控为STM32H503CB,陀螺仪为LSM6DSOW,磁力计为LIS2MDL。

开启LED

配置PB14为输出模式。

开启INT中断

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

开启中断。

中断读取传感器数据

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

/// 用于存储FIFO中读取的数据,每条数据包含7个字节(1个标签字节和6个数据字节)
uint8_t fifo_data[10*2][7];
stmdev_ctx_t dev_ctx;
uint8_t wmflag = 0;
uint16_t num = 0;
lsm6dso_fifo_tag_t reg_tag;
axis3bit16_t dummy;

// FIFO中当前存储的数据数量
uint16_t fifo_num = 0;
// FIFO中断标志,用于标记是否有新的FIFO数据可供读取
uint8_t fifo_flag=0;

static int16_t *datax;
static int16_t *datay;
static int16_t *dataz;


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;
int out_num=0;

需要注意优化等级。

完整初始化如下所示。

/* 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);



  /* Uncomment to configure INT 1 */
  lsm6dso_pin_int1_route_t int1_route;    
  /* 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 */
  lsm6dso_device_id_get(&dev_ctx, &whoamI);
    printf("LSM6DSO_ID=0x%x,whoamI=0x%x",LSM6DSO_ID,whoamI);
  if (whoamI != LSM6DSO_ID)
    while (1);

  /* Restore default configuration */
  lsm6dso_reset_set(&dev_ctx, PROPERTY_ENABLE);

  do {
    lsm6dso_reset_get(&dev_ctx, &rst);
  } while (rst);

  /* Disable I3C interface */
  lsm6dso_i3c_disable_set(&dev_ctx, LSM6DSO_I3C_DISABLE);
  /* Enable Block Data Update */
  lsm6dso_block_data_update_set(&dev_ctx, PROPERTY_ENABLE);
  /* Set full scale */
  lsm6dso_xl_full_scale_set(&dev_ctx, LSM6DSO_2g);
  lsm6dso_gy_full_scale_set(&dev_ctx, LSM6DSO_2000dps);
  /* Set FIFO watermark (number of unread sensor data TAG + 6 bytes
   * stored in FIFO) to 10 samples
   */
  lsm6dso_fifo_watermark_set(&dev_ctx, 10);
  /* Set FIFO batch XL/Gyro ODR to 12.5Hz */
  lsm6dso_fifo_xl_batch_set(&dev_ctx, LSM6DSO_XL_BATCHED_AT_417Hz);
  lsm6dso_fifo_gy_batch_set(&dev_ctx, LSM6DSO_GY_BATCHED_AT_417Hz);
  /* Set FIFO mode to Stream mode (aka Continuous Mode) */
  lsm6dso_fifo_mode_set(&dev_ctx, LSM6DSO_STREAM_MODE);
  /* Enable drdy 75 μs pulse: uncomment if interrupt must be pulsed */
  lsm6dso_data_ready_mode_set(&dev_ctx, LSM6DSO_DRDY_PULSED);
  /* Uncomment if interrupt generation on Free Fall INT1 pin */
  lsm6dso_pin_int1_route_get(&dev_ctx, &int1_route);
  int1_route.fifo_th = PROPERTY_ENABLE;
  lsm6dso_pin_int1_route_set(&dev_ctx, int1_route);
  /* Uncomment if interrupt generation on Free Fall INT2 pin */
  //lsm6dso_pin_int2_route_get(&dev_ctx, &int2_route);
  //int2_route.reg.int2_ctrl.int2_fifo_th = PROPERTY_ENABLE;
  //lsm6dso_pin_int2_route_set(&dev_ctx, &int2_route);
  /* Set Output Data Rate */
  lsm6dso_xl_data_rate_set(&dev_ctx, LSM6DSO_XL_ODR_417Hz);
  lsm6dso_gy_data_rate_set(&dev_ctx, LSM6DSO_GY_ODR_417Hz);



    lsm6dso_fifo_timestamp_decimation_set(&dev_ctx, LSM6DSO_DEC_1);
  /* Enable timestamp */
  lsm6dso_timestamp_set(&dev_ctx, PROPERTY_ENABLE);

  /* USER CODE END 2 */

开启时间戳

FIFO_CTRL4 (0Ah) 寄存器用于配置LSM6DSOW传感器的FIFO行为,包括时间戳批处理减量、温度数据批处理速率和FIFO模式选择。该寄存器包含多个字段,每个字段的功能如下所述。

DEC_TS_BATCH 位于 FIFO_CTRL4 (0Ah) 寄存器中,用于配置时间戳在FIFO中的批处理减量。它决定了时间戳数据在FIFO中写入的频率。

CTRL10_C 寄存器用于启用和配置LSM6DSOW传感器的时间戳计数器。启用时间戳计数器可以在数据记录过程中提供精确的时间标记,以便更好地分析和同步传感器数据。
TIMESTAMP_EN: 用于启用或禁用时间戳计数器。

时间戳单位为25us,当启用时间戳计数器时,计数器的值可以从以下寄存器读取:TIMESTAMPO (40h),TIMESTAMP1 (41h),TIMESTAMP2 (42h),TIMESTAMP3 (43h)

lsm6dso_fifo_timestamp_decimation_set(&dev_ctx, LSM6DSO_DEC_1);
  /* Enable timestamp */
  lsm6dso_timestamp_set(&dev_ctx, PROPERTY_ENABLE);

中断配置

在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 的状态,可以直观地了解中断的发生。

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin){

    if(GPIO_Pin == GPIO_PIN_0)
    {
        HAL_GPIO_TogglePin(LED1_GPIO_Port, LED1_Pin);
    /* Read watermark flag */
    lsm6dso_fifo_wtm_flag_get(&dev_ctx, &wmflag);

    if (wmflag > 0) {
            fifo_flag=1;
      /* Read number of samples in FIFO */
      lsm6dso_fifo_data_level_get(&dev_ctx, &num);
            fifo_num=num;
//                    printf("num=%dn",num);
      for(int i=0;i< num;i++)  {
        /* Read FIFO tag */
        lsm6dso_fifo_sensor_tag_get(&dev_ctx, ®_tag);
                fifo_data[i][0]=reg_tag;
                memset(data_raw_acceleration.u8bit, 0x00, 3 * sizeof(int16_t));
                lsm6dso_fifo_out_raw_get(&dev_ctx, data_raw_acceleration.u8bit);            
                fifo_data[i][1]=data_raw_acceleration.u8bit[0];
                fifo_data[i][2]=data_raw_acceleration.u8bit[1];
                fifo_data[i][3]=data_raw_acceleration.u8bit[2];
                fifo_data[i][4]=data_raw_acceleration.u8bit[3];
                fifo_data[i][5]=data_raw_acceleration.u8bit[4];
                fifo_data[i][6]=data_raw_acceleration.u8bit[5];        
//    printf("f1=%d f2=%d",fifo_data[i][1],fifo_data[i][2]);            

//        switch (reg_tag) {
//          case LSM6DSO_XL_NC_TAG:
//            memset(data_raw_acceleration.u8bit, 0x00, 3 * sizeof(int16_t));
//            lsm6dso_fifo_out_raw_get(&dev_ctx, data_raw_acceleration.u8bit);
//            acceleration_mg[0] =
//              lsm6dso_from_fs2_to_mg(data_raw_acceleration.i16bit[0]);
//            acceleration_mg[1] =
//              lsm6dso_from_fs2_to_mg(data_raw_acceleration.i16bit[1]);
//            acceleration_mg[2] =
//              lsm6dso_from_fs2_to_mg(data_raw_acceleration.i16bit[2]);
//            printf("Acceleration [mg]:%4.2ft%4.2ft%4.2frn",
//                    acceleration_mg[0], acceleration_mg[1], acceleration_mg[2]);
//            break;

//          case LSM6DSO_GYRO_NC_TAG:
//            memset(data_raw_angular_rate.u8bit, 0x00, 3 * sizeof(int16_t));
//            lsm6dso_fifo_out_raw_get(&dev_ctx, data_raw_angular_rate.u8bit);
//            angular_rate_mdps[0] =
//              lsm6dso_from_fs2000_to_mdps(data_raw_angular_rate.i16bit[0]);
//            angular_rate_mdps[1] =
//              lsm6dso_from_fs2000_to_mdps(data_raw_angular_rate.i16bit[1]);
//            angular_rate_mdps[2] =
//              lsm6dso_from_fs2000_to_mdps(data_raw_angular_rate.i16bit[2]);
//            printf("Angular rate [mdps]:%4.2ft%4.2ft%4.2frn",
//                    angular_rate_mdps[0], angular_rate_mdps[1], angular_rate_mdps[2]);
//            break;

//                    
//          case LSM6DSO_TIMESTAMP_TAG:    
//                        /* Read temperature data */
//                        memset(&data_raw_timestamp, 0x00, sizeof(int16_t));
//            lsm6dso_fifo_out_raw_get(&dev_ctx, data_raw_timestamp.u8bit);
//                        uint32_t timestamp=0;
//                        timestamp+= data_raw_timestamp.u8bit[0];
//                        timestamp+= data_raw_timestamp.u8bit[1]< < 8;                    
//                        timestamp+= data_raw_timestamp.u8bit[2]< < 16;                        
//                        timestamp+= data_raw_timestamp.u8bit[3]< < 24;                        
//                        printf("timestamp=%drn",
//              timestamp);
//            break;                
//                    
//                    
//                    
//                    
//          default:
//            /* Flush unused samples */
//            memset(dummy.u8bit, 0x00, 3 * sizeof(int16_t));
//            lsm6dso_fifo_out_raw_get(&dev_ctx, dummy.u8bit);
//            break;
//        }
      }
    }        

    }
}

主程序

在主循环中检查FIFO中断标志,如果有新的FIFO数据,则读取并处理这些数据。

/* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    if(fifo_flag)// 如果 FIFO 中断标志被设置
    {        
        uint8_t acc_flag=0,gyr_flag=0;//加速度角速度标志位
        uint8_t deltatime_flag=0;//时间标志位        
        printf("fifo_num=%dn",fifo_num);
        for(int i=0;i< fifo_num;i++)// 遍历 FIFO 数据数组
        {
            // 获取数据指针
            datax = (int16_t *)&fifo_data[i][1];
            datay = (int16_t *)&fifo_data[i][3];
            dataz = (int16_t *)&fifo_data[i][5];

            // 根据数据标签处理不同类型的数据
            switch (fifo_data[i][0]) {
        case LSM6DSO_XL_NC_TAG:// 加速度数据
                    acc_flag=1;                
                    acc_x=lsm6dso_from_fs2_to_mg(*datax);
                    acc_y=lsm6dso_from_fs2_to_mg(*datay);
                    acc_z=lsm6dso_from_fs2_to_mg(*dataz);        
                    printf("Acceleration [mg]:%4.2ft%4.2ft%4.2frn",
                    acc_x, acc_y, acc_z);                
          break;                
        case LSM6DSO_GYRO_NC_TAG:// 角速度数据
                    gyr_flag=1;
                    gyr_x=lsm6dso_from_fs2000_to_mdps(*datax);
                    gyr_y=lsm6dso_from_fs2000_to_mdps(*datay);
                    gyr_z=lsm6dso_from_fs2000_to_mdps(*dataz);    
                    printf("Angular rate [mdps]:%4.2ft%4.2ft%4.2frn",
                                    gyr_x,gyr_y,gyr_z);            
          break;                
        case LSM6DSO_TIMESTAMP_TAG:// 时间戳数据
                    deltatime_flag=1;
             /* 读取时间戳数据 */
                uint32_t timestamp=0;
                        timestamp+= fifo_data[i][1];
                        timestamp+= fifo_data[i][2]< < 8;                    
                        timestamp+= fifo_data[i][3]< < 16;                        
                        timestamp+= fifo_data[i][4]< < 24;    
                    if(deltatime_first==0)//第一次
                    {
                        deltatime_1=timestamp;
                        deltatime_2=deltatime_1;
                        deltatime_first=1;
                    }
                    else
                    {
                        deltatime_2=timestamp;
                    }
                    printf("timestamp=%drn",timestamp);                            
          break;    
        default:
          break;                
                }
            // 如果加速度、角速度和时间戳数据都已获取
            if(acc_flag&&gyr_flag&&deltatime_flag)
            {
//                lsm6dso_motion_fx_determin();// 调用 MotionFX 处理函数
                acc_flag=0;
                gyr_flag=0;
                deltatime_flag=0;
                deltatime_1=deltatime_2;    // 更新时间戳        
            }                        
        }

        // 清除 FIFO 标志和数据量
        fifo_flag=0;
        fifo_num=0;

        }


    /* USER CODE END WHILE */

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

演示

由于分别开启了加速度,角速度,时间戳到FIFO,FIFO溢出阈值设置为10,那么到12的时候数据溢出进行打印。 时间戳数据16697842-16697746=96*25us=2400us。 速率为416HZ,即2.4ms

审核编辑 黄宇

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 陀螺仪
    +关注

    关注

    44

    文章

    873

    浏览量

    101094
  • fifo
    +关注

    关注

    3

    文章

    406

    浏览量

    45500
收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    LSM6DSV16BX:集成机器学习核心与Qvar感应的先进惯性测量单元

    STMicroelectronics LSM6DSV16BX iNEMO三轴加速度计和陀螺仪是一款三轴数字加速度计和一款三轴数字陀螺仪LSM6DSV16BX具有三核,用于在三个独立
    的头像 发表于 10-28 13:45 277次阅读
    <b class='flag-5'>LSM6</b>DSV16BX:集成机器学习核心与Qvar感应的先进惯性测量单元

    LSM6DSV16X:集成机器学习与静电感应的高性能惯性测量单元

    STMicroelectronics LSM6DSV16X iNEMO惯性模块是一款三轴数字加速度计和三轴数字陀螺仪LSM6DSV16X具有三个内核,用于处理三个独立通道上的加速度和角速率
    的头像 发表于 10-28 11:52 353次阅读
    <b class='flag-5'>LSM6</b>DSV16X:集成机器学习与静电感应的高性能惯性测量单元

    LSM6DSV16B:面向TWS与可穿戴设备的6轴IMU,融合传感与低功耗智能

    STMicroelectronics LSM6DSV16B iNEMO惯性模块具有一个3轴数字加速度计和一个3轴数字陀螺仪。 STMicroelectronics LSM6DSV16B的设计可用
    的头像 发表于 10-25 14:28 833次阅读
    <b class='flag-5'>LSM6</b>DSV16B:面向TWS与可穿戴设备的<b class='flag-5'>6</b>轴IMU,融合传感与低功耗智能

    LSM6DSV80X 6轴惯性测量单元技术解析

    陀螺仪。利用LSM6DSV80X,设备可提供训练分析和性能基准等功能,为运动员提供帮助。低G加速计记录和识别跑步、步行和手势交互等活动。 该IMU适用于体育设备的各种应用。
    的头像 发表于 10-15 17:42 562次阅读
    <b class='flag-5'>LSM6</b>DSV80X <b class='flag-5'>6</b>轴惯性测量单元技术解析

    光纤陀螺仪与惯性导航系统在无人车驾驶中的应用

    一、光纤陀螺仪与惯性导航系统简介 光纤陀螺仪是一种基于光学传感技术的精密仪器,利用Sagnac效应来检测旋转角速度。与传统的机械陀螺仪相比,光纤陀螺仪具有全固态、无旋转部件和摩擦部件、
    的头像 发表于 10-09 10:29 191次阅读
    光纤<b class='flag-5'>陀螺仪</b>与惯性导航系统在无人车驾驶中的应用

    什么是光纤陀螺仪陀螺仪有哪些作用?

    陀螺仪是干什么用的?陀螺仪是用来感知和测量物体旋转的一个传感器。简单来说,它可以帮助设备知道自己当前的角度和运动方向。通常,我们会在智能手机、无人机、VR设备等科技产品中看到陀螺仪的身影。我第一次
    的头像 发表于 08-26 17:36 1657次阅读
    什么是光纤<b class='flag-5'>陀螺仪</b>?<b class='flag-5'>陀螺仪</b>有哪些作用?

    为什么三轴MEMS陀螺仪是无人机飞控系统的核心传感器?

    在现代无人机技术中,三轴MEMS陀螺仪扮演着至关重要的角色。作为飞行控制系统的核心传感器,它通过实时测量飞行器的角速度变化,为稳定飞行提供关键数据支持。以ER-3MG-063 MEMS陀螺仪为例,让我们深入了解三轴MEMS
    的头像 发表于 08-07 17:37 875次阅读
    为什么三轴MEMS<b class='flag-5'>陀螺仪</b>是无人机飞控系统的核心传感器?

    MEMS陀螺仪如何实现高精度寻北?

    MEMS陀螺仪的寻北功能,其核心在于精确测量地球自转角速度的分量。通过解算这些分量,即可确定地理北向。得益于MEMS技术的持续发展,此类陀螺仪在精度与稳定性方面已实现显著跃升。
    的头像 发表于 07-29 18:00 511次阅读

    MEMS陀螺仪有哪些分类?

    你是否好奇,手机里的指南针、无人机稳定的飞行姿态,甚至火箭精准的导航,背后都离不开一个微型“旋转感知器”?这就是MEMS陀螺仪!它凭借小巧身材、低廉成本和优异性能,正在取代笨重的传统陀螺仪,深入我们
    的头像 发表于 07-21 15:58 589次阅读

    MEMS陀螺仪正在取代光纤陀螺仪

    一、微型科技巨匠:MEMS陀螺仪揭秘 何谓MEMS? MEMS(微机电系统)是融合了微电子与微机械的神奇技术。它能在指甲盖大小的硅芯片上集成复杂的传感器、执行器和处理电路,实现微观世界的数据感知
    的头像 发表于 07-08 16:45 647次阅读

    振动陀螺仪传感器的工作原理

    陀螺仪传感器,也称为角速率传感器或角速度传感器,是一种感测角速度的设备。陀螺仪传感器种类繁多,比较常见的有机械陀螺仪,光学陀螺仪,流体陀螺仪
    的头像 发表于 06-16 16:29 1017次阅读
    振动<b class='flag-5'>陀螺仪</b>传感器的工作原理

    陀螺仪在钻井或测井中有何用途?

    钻井与测井作业对定向和井眼轨迹控制要求较高,传统光纤陀螺仪和磁性测斜体积大、抗干扰能力弱、成本高。MEMS陀螺仪凭借其微型化、高可靠性和环境适应性,成为钻井、测井核心器件。
    的头像 发表于 06-10 17:46 711次阅读
    <b class='flag-5'>陀螺仪</b>在钻井或测井中有何用途?

    LSM6DS3TRC FIFO模式,长时间运行\"卡死\"有人遇到过吗

    LSM6DS3TRC_Set_CTRL6_C(0x10); // 设置陀螺仪为低功耗模式(bit7 = 1) LSM6DS3TRC_Set_CTRL7_G(0x80); 每次唤醒后执行的 bool
    发表于 06-08 15:08

    MEMS陀螺仪在钻井中发挥什么作用?

    在油气勘探、矿产开发及地质工程领域,定向钻孔与测井作业面临着复杂地层环境下的姿态控制、轨迹监测与数据精度挑战。陀螺钻井,是用于精确井眼定位和定向钻井的一项技术,通常使用陀螺仪和加速度计
    的头像 发表于 06-06 15:58 550次阅读
    MEMS<b class='flag-5'>陀螺仪</b>在钻井中发挥什么作用?

    MEMS陀螺仪的寻北原理是什么?精度如何?

    MEMS陀螺仪的寻北技术核心原理基于地球自转特性,通过测量角速度分量解算出地理北向。随着MEMS技术的不断进步,MEMS陀螺仪性能也在不断提升,已经具备了较高的测量精度和稳定性。
    的头像 发表于 06-04 17:50 867次阅读
    MEMS<b class='flag-5'>陀螺仪</b>的寻北原理是什么?精度如何?