[资源共享] 移植MPU6050-DMP库实现姿态角PRY解算

[复制链接]
2516|36
 楼主| 回复就哭哭 发表于 2022-6-30 10:28 | 显示全部楼层 |阅读模式
本帖最后由 芯圣电子官方QQ 于 2023-7-20 10:37 编辑

【灵动MM32-姿态角解算】移植MPU6050-DMP库实现姿态角PRY解算MM32使用DMP库处理MPU6050数据、姿态解算
实现途径:

①MM32F3277G9P单片机 ②MPU6050模块 ③智能车逐飞MM32开源库 ④正点原子dmp开源库

开源库链接: 逐飞科技MM32F327X_G9P开源库

MM32使用DMP库处理MPU6050数据、姿态解算
一、mpu6050介绍
二、灵动MM32-DMP库移植
1、添加库文件
2、官方 DMP 驱动库移植
3、inv_mpu.c 文件移植
4、 inv_mpu_dmp_motion_driver.c移植
三、MM32程序编写
1、MPU6050初始化
2、主函数编写
四、工程源码


评论

———————————————— 版权声明:本文为CSDN博主「有把原木色的吉他」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。 原文链接:https://blog.csdn.net/weixin_45829708/article/details/125010473  发表于 2022-6-30 10:30
 楼主| 回复就哭哭 发表于 2022-6-30 10:35 | 显示全部楼层
一、mpu6050介绍
MPU6050 是 InvenSense 公司推出的全球首款整合性 6 轴运动处理组件,相较于多组件
方案,免除了组合陀螺仪与加速器时之轴间差的问题,减少了安装空间。

MPU6050 内部整合了 3 轴陀螺仪和 3 轴加速度传感器,并且含有一个第二 IIC 接口(本
模块未引出),可用于连接外部磁力传感器,并利用自带的数字运动处理器( DMP: Digital
Motion Processor)硬件加速引擎,通过主 IIC 接口,向应用端输出完整的 9 轴融合演算数据。
我们可以使用 InvenSense 公司提供的运动处理DMP资料库,非常方便的实现姿态算,降低了运动处理运算对操作系统的负荷,同时大大降低了开发难度。
 楼主| 回复就哭哭 发表于 2022-6-30 10:36 | 显示全部楼层
 楼主| 回复就哭哭 发表于 2022-6-30 10:37 | 显示全部楼层
MPU6050 的特点包括:
① 以数字形式输出 6 轴或 9 轴(需外接磁传感器)的旋转矩阵、四元数(quaternion)、
欧拉角格式(Euler Angle forma)的融合演算数据(需 DMP 支持)
② 具有 131 LSBs/° /sec 敏感度与全格感测范围为±250、±500、±1000 与±2000°
/sec 的 3 轴角速度感测器(陀螺仪)
③ 集成可程序控制,范围为±2g、±4g、±8g 和±16g 的 3 轴加速度传感器
④ 移除加速器与陀螺仪轴间敏感度,降低设定给予的影响与感测器的飘移
⑤ 自带数字运动处理(DMP: Digital Motion Processing)引擎可减少 MCU 复杂的融合演
算数据、感测器同步化、姿势感应等的负荷
⑥ 内建运作时间偏差与磁力感测器校正演算技术, 免除了客户须另外进行校正的需求
⑦ 自带一个数字温度传感器
⑧ 带数字输入同步引脚(Sync pin)支持视频电子影相稳定技术与 GPS
⑨ 可程序控制的中断(interrupt),支持姿势识别、摇摄、画面放大缩小、滚动、快速
下降中断、 high-G 中断、零动作感应、触击感应、摇动感应功能
⑩ VDD 供电电压为 2.5V±5%、 3.0V±5%、 3.3V±5%; VLOGIC 可低至1.8V±5%
⑪ 陀螺仪工作电流: 5mA,陀螺仪待机电流: 5uA;加速器工作电流: 500uA,加速
器省电模式电流: 40uA@10Hz
⑫ 自带 1024 字节 FIFO,有助于降低系统功耗
⑬ 高达 400Khz 的 IIC 通信接口
⑭ 超小封装尺寸: 4x4x0.9mm(QFN)
 楼主| 回复就哭哭 发表于 2022-6-30 10:38 | 显示全部楼层
 楼主| 回复就哭哭 发表于 2022-6-30 10:39 | 显示全部楼层
二、灵动MM32-DMP库移植
InvenSense 提供的 MPU6050 运动驱动库是基于 MSP430 的,我们需要将其移植一下,
才可以用到 MM32上面。
 楼主| 回复就哭哭 发表于 2022-6-30 10:40 | 显示全部楼层
添加库文件
首先需要添加这六个文件到工程中,这六个文件是我从正点原子那里复制过来的,里边大部分不需要改动,只有下面少数几个地方。最后我会将工程文件上传到百度网盘。
6244062bd0d1bca9de.png
 楼主| 回复就哭哭 发表于 2022-6-30 10:43 | 显示全部楼层
官方 DMP 驱动库移植
主要是实现这 4 个函数: i2c_write, i2c_read,delay_ms 和 get_ms
  1. #define i2c_write   MPU_Write_Len
  2. #define i2c_read    MPU_Read_Len
  3. #define delay_ms    systick_delay_ms
  4. #define get_ms      mget_m
 楼主| 回复就哭哭 发表于 2022-6-30 10:44 | 显示全部楼层
其中MPU_Write_Len是我根据逐飞库改变而来,位置在SEEKREE_IIC.c文件之中
  1. //IIC连续写
  2. //addr:器件地址
  3. //reg:寄存器地址
  4. //len:写入长度
  5. //buf:数据区
  6. //返回值:0,正常
  7. //    其他,错误代码
  8. uint8_t MPU_Write_Len(uint8_t addr,uint8_t reg,uint8_t len,uint8_t *buf)
  9. {
  10.         uint8_t i;
  11.     simiic_start();
  12.         send_ch((addr<<1)|0);//发送器件地址+写命令       
  13.     send_ch(reg);        //写寄存器地址
  14.         for(i=0;i<len;i++)
  15.         {
  16.                 send_ch(buf[i]);        //发送数据               
  17.         }   
  18.     simiic_stop();         
  19.         return 0;       
  20. }
 楼主| 回复就哭哭 发表于 2022-6-30 10:47 | 显示全部楼层
MPU_Read_Len是我根据逐飞库改变而来,位置在SEEKREE_IIC.c文件之中
  1. //IIC连续读
  2. //addr:器件地址
  3. //reg:要读取的寄存器地址
  4. //len:要读取的长度
  5. //buf:读取到的数据存储区
  6. //返回值:0,正常
  7. //    其他,错误代码
  8. uint8_t MPU_Read_Len(uint8_t addr,uint8_t reg,uint8_t len,uint8_t *buf)
  9. {
  10.         simiic_start();
  11.         send_ch((addr<<1)|0);//发送器件地址+写命令       
  12.     send_ch(reg);        //写寄存器地址
  13.        
  14.     simiic_start();
  15.         send_ch((addr<<1)|1);//发送器件地址+读命令       
  16.        
  17.         while(len)
  18.         {
  19.                 if(len==1)*buf=read_ch(0);        //读数据,发送nACK
  20.                 else *buf=read_ch(1);                //读数据,发送ACK  
  21.                 len--;
  22.                 buf++;
  23.         }   
  24.     simiic_stop();        //产生一个停止条件
  25.         return 0;       
  26. }
 楼主| 回复就哭哭 发表于 2022-6-30 10:50 | 显示全部楼层
systick_delay_ms函数为逐飞库内毫秒级systick延时函数,位置在zf_systick.h文件中
 楼主| 回复就哭哭 发表于 2022-6-30 10:52 | 显示全部楼层
  1. //-------------------------------------------------------------------------------------------------------------------
  2. // @brief                毫秒级systick延时函数
  3. // @param                time                        延时多少毫秒
  4. // @return                void
  5. // Sample usage:                                systick_delay_ms(1000);   //延时1000毫秒
  6. //-------------------------------------------------------------------------------------------------------------------
  7. void systick_delay_ms (uint32 time)
  8. {
  9.         while(time--) systick_delay(SystemCoreClock / 1000);
  10. }
 楼主| 回复就哭哭 发表于 2022-6-30 12:28 | 显示全部楼层
mget_ms函数为空函数未用到 位置在inv_mpu.h文件中
 楼主| 回复就哭哭 发表于 2022-6-30 12:28 | 显示全部楼层
  1. void mget_ms(unsigned long *time)

  2. }
 楼主| 回复就哭哭 发表于 2022-6-30 12:30 | 显示全部楼层
inv_mpu.c 文件移植
包括头文件调用及 4 个函数: i2c_write, i2c_read,delay_ms 和 get_ms*函数重定义
 楼主| 回复就哭哭 发表于 2022-6-30 12:30 | 显示全部楼层
  1. #include <stdio.h>
  2. #include <stdint.h>
  3. #include <stdlib.h>
  4. #include <string.h>
  5. #include <math.h>
  6. #include "inv_mpu.h"
  7. #include "inv_mpu_dmp_motion_driver.h"
  8. #include "mpu6050.h"

  9. #include "SEEKFREE_IIC.h"
  10. #include "zf_systick.h"

  11. #define MPU6050                                                        //定义我们使用的传感器为MPU6050
  12. #define MOTION_DRIVER_TARGET_MSP430                //定义驱动部分,采用MSP430的驱动(移植到STM32F4)

  13. #if defined MOTION_DRIVER_TARGET_MSP430
  14. //#include "msp430.h"
  15. //#include "msp430_i2c.h"
  16. //#include "msp430_clock.h"
  17. //#include "msp430_interrupt.h"

  18. #define i2c_write   MPU_Write_Len
  19. #define i2c_read    MPU_Read_Len
  20. #define delay_ms    systick_delay_ms
  21. #define get_ms      mget_ms
  22. //static inline int reg_int_cb(struct int_param_s *int_param)
  23. //{
  24. //    return msp430_reg_int_cb(int_param->cb, int_param->pin, int_param->lp_exit,
  25. //        int_param->active_low);
  26. //}
  27. #define log_i         printf        //打印信息
  28. #define log_e          printf        //打印信息
  29. /* labs is already defined by TI's toolchain. */
  30. /* fabs is for doubles. fabsf is for floats. */
  31. #define fabs        fabsf
  32. #define min(a,b) ((a<b)?a:b)
 楼主| 回复就哭哭 发表于 2022-6-30 12:31 | 显示全部楼层
修改两个函数mpu_dmp_init 、mpu_dmp_get_data

mpu_dmp_init是 MPU6050 DMP 初始化函数
 楼主| 回复就哭哭 发表于 2022-6-30 12:32 | 显示全部楼层
该函数代码如下:
  1. //mpu6050,dmp初始化
  2. //返回值:0,正常
  3. //    其他,失败
  4. u8 mpu_dmp_init(void)
  5. {
  6.         u8 res=0;
  7.         //simiic_init();         //iic总线初始化,这里不需要加,因为在主函数中dmp初始化已经调用过了
  8.         if(mpu_init()==0)        //初始化MPU6050
  9.         {         
  10.                 res=mpu_set_sensors(INV_XYZ_GYRO|INV_XYZ_ACCEL);//设置所需要的传感器
  11.                 if(res)return 1;
  12.                 res=mpu_configure_fifo(INV_XYZ_GYRO | INV_XYZ_ACCEL);//设置FIFO
  13.                 if(res)return 2;
  14.                 res=mpu_set_sample_rate(DEFAULT_MPU_HZ);        //设置采样率
  15.                 if(res)return 3;
  16.                 res=dmp_load_motion_driver_firmware();                //加载dmp固件
  17.                 if(res)return 4;
  18.                 res=dmp_set_orientation(inv_orientation_matrix_to_scalar(gyro_orientation));//设置陀螺仪方向
  19.                 if(res)return 5;
  20.                 res=dmp_enable_feature(DMP_FEATURE_6X_LP_QUAT|DMP_FEATURE_TAP|        //设置dmp功能
  21.                     DMP_FEATURE_ANDROID_ORIENT|DMP_FEATURE_SEND_RAW_ACCEL|DMP_FEATURE_SEND_CAL_GYRO|
  22.                     DMP_FEATURE_GYRO_CAL);
  23.                 if(res)return 6;
  24.                 res=dmp_set_fifo_rate(DEFAULT_MPU_HZ);        //设置DMP输出速率(最大不超过200Hz)
  25.                 if(res)return 7;   
  26.                 res=run_self_test();                //自检
  27.                 if(res)return 8;   
  28.                 res=mpu_set_dmp_state(1);        //使能DMP
  29.                 if(res)return 9;     
  30.         }
  31.         return 0;
  32. }
 楼主| 回复就哭哭 发表于 2022-6-30 12:34 | 显示全部楼层
①:调用mpu_init 函数(MSP430库函数,不用管),初始化 MPU6050。
②:设置 DMP 所用传感器、 FIFO、采样率和加载固件等一系列操作。
③:最后通过 mpu_set_dmp_state(1)使能 DMP 功能,在使能成功以后,我们便可以通过 mpu_dmp_get_data 来读取姿态解算后的数据了。
 楼主| 回复就哭哭 发表于 2022-6-30 12:56 | 显示全部楼层
mpu_dmp_get_data 函数代码如下:
  1. //得到dmp处理后的数据(注意,本函数需要比较多堆栈,局部变量有点多)
  2. //pitch:俯仰角 精度:0.1°   范围:-90.0° <---> +90.0°
  3. //roll:横滚角  精度:0.1°   范围:-180.0°<---> +180.0°
  4. //yaw:航向角   精度:0.1°   范围:-180.0°<---> +180.0°
  5. //返回值:0,正常
  6. //    其他,失败
  7. u8 mpu_dmp_get_data(float *pitch,float *roll,float *yaw)
  8. {
  9.         float q0=1.0f,q1=0.0f,q2=0.0f,q3=0.0f;
  10.         unsigned long sensor_timestamp;
  11.         short gyro[3], accel[3], sensors;
  12.         unsigned char more;
  13.         long quat[4];
  14.         if(dmp_read_fifo(gyro, accel, quat, &sensor_timestamp, &sensors,&more))return 1;         
  15.         /* Gyro and accel data are written to the FIFO by the DMP in chip frame and hardware units.
  16.          * This behavior is convenient because it keeps the gyro and accel outputs of dmp_read_fifo and mpu_read_fifo consistent.
  17.         **/
  18.         /*if (sensors & INV_XYZ_GYRO )
  19.         send_packet(PACKET_TYPE_GYRO, gyro);
  20.         if (sensors & INV_XYZ_ACCEL)
  21.         send_packet(PACKET_TYPE_ACCEL, accel); */
  22.         /* Unlike gyro and accel, quaternions are written to the FIFO in the body frame, q30.
  23.          * The orientation is set by the scalar passed to dmp_set_orientation during initialization.
  24.         **/
  25.         if(sensors&INV_WXYZ_QUAT)
  26.         {
  27.                 q0 = quat[0] / q30;        //q30格式转换为浮点数
  28.                 q1 = quat[1] / q30;
  29.                 q2 = quat[2] / q30;
  30.                 q3 = quat[3] / q30;
  31.                 //计算得到俯仰角/横滚角/航向角
  32.                 *pitch = asin(-2 * q1 * q3 + 2 * q0* q2)* 57.3;        // pitch
  33.                 *roll  = atan2(2 * q2 * q3 + 2 * q0 * q1, -2 * q1 * q1 - 2 * q2* q2 + 1)* 57.3;        // roll
  34.                 *yaw   = atan2(2*(q1*q2 + q0*q3),q0*q0+q1*q1-q2*q2-q3*q3) * 57.3;        //yaw
  35.         }else return 2;
  36.         return 0;
  37. }
您需要登录后才可以回帖 登录 | 注册

本版积分规则

29

主题

436

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部