【STM32 HAL库】MPU6050 DMP库移植 与 自检失败的处理

【STM32 HAL库】MPU6050 DMP库移植 与 自检失败的处理

  • 本文参考
  • 移植步骤
  • 文件配置
  • 代码修改
  • inv_mpu.c
  • inv_mpu.h
  • inv_mpu_dmp_motion_driver.c
  • 使用
  • 自检失败怎么处理
  • ret = -1
  • 改正
  • DEBUG过程
  • ret = -9
  • 改正
  • DEBUG过程
  • 本文参考

    B站
    CSDN

    移植步骤

    文件配置

    新建一个 dmp 文件夹
    并将 8 个官方的驱动文件复制到这个文件夹中

    将该dmp文件夹复制到工程文件夹中

    新建Groups组,并添加这8个驱动文件

    添加编译路径

    代码修改

    inv_mpu.c

    第 39 行到第 161 行删掉
    #if defined EMPL_TARGET_STM32F4#endif
    然后在删除代码的原位置添加以下代码

    #define MPU6050
    #include "main.h"
    extern I2C_HandleTypeDef hi2c1;
    #define i2c_write(dev_addr,reg_addr,data_size,p_data)  \
    HAL_I2C_Mem_Write(&hi2c1,dev_addr,reg_addr,I2C_MEMADD_SIZE_8BIT,p_data,data_size,0x100)
    #define i2c_read(dev_addr,reg_addr,data_size,p_data)   \
    HAL_I2C_Mem_Read(&hi2c1,dev_addr,reg_addr,I2C_MEMADD_SIZE_8BIT,p_data,data_size,0x100)
    #define delay_ms    HAL_Delay
    #define get_ms(p)      do{*p = HAL_GetTick();}while(0)
    #define log_i(...)     do {} while (0)
    #define log_e(...)     do {} while (0)
    /* labs is already defined by TI's toolchain. */
    /* fabs is for doubles. fabsf is for floats. */
    #define fabs        fabsf
    #define min(a,b) ((a<b)?a:b)
    

    找到这段代码,删掉

    修改.addr值为0xd0
    注意这个addr大概在400行左右,如果你修改成480行的那个addr值,则后续自检会不通过!!!

    inv_mpu.h

    删掉 31 行到 44 行的这段代码

    并在删去的位置添加下面这段代码

    struct int_param_s {
    	  void *arg;
    };
    

    inv_mpu_dmp_motion_driver.c

    删掉第 27 到 77 行的代码

    添加下面代码

    #include "main.h"
    extern I2C_HandleTypeDef hi2c1;
    #define delay_ms    HAL_Delay
    #define get_ms(p)      do{*p = HAL_GetTick();}while(0)
    #define log_i(...)     do {} while (0)
    #define log_e(...)     do {} while (0)
    

    修改

    使用

    引用头文件
    #include "MPU6050.h"

    在main.c的初始化部分 添加MPU6050 DMP库初始化代码

      int ret=0;
    	do{
    	 ret=MPU6050_DMP_init();
    	 //printf输出ret值,判断初始化是否成功
    	 printf("ret=%d\n",ret);
    	}while(ret);
    

    定义全局变量

    float pitch,yaw,roll;
    

    函数使用

    void loop(void){
         MPU6050_DMP_Get_Date(&pitch,&roll,&yaw);
    	 printf("pitch=%f yaw=%f roll=%f\n",pitch,roll,yaw);
    }
    

    自检失败怎么处理

    ret = -1

    改正

    inv_mpu.c 400行左右的.addr的值修改为0xd0
    不是480行的那个!!!

    DEBUG过程

    我们可以发现ret为 MPU6050_DMP_init(); 的返回值

    进入该函数实现
    可以看到在该函数中返回值为宏定义

    进入宏定义
    确定 ret = -1 对应 ERROR_MPU_INIT

    也即ret = mpu_init(&int_param);中的mpu初始化函数报错,返回值不为0。


    再看下该函数实现
    发现有多个return -1的出口,无法判断是哪里出去

    改写该函数,将所有的return -1的出口处添加error_code的标志位,判断到底从哪个出口出去

    打印的error_code为1
    说明设备复位错误

    我们排查i2c_write(st.hw->addr, st.reg->pwr_mgmt_1, 1, data)函数
    发现是st.hw->addr该参数错误

    错误原因:在移植DMP库的时候,修改inv_mpu.c时,修改成了MPU6500的.addr值,而非MPU6050的
    480行左右的.addr是定义MPU6500时的设备地址,也即MPU6500的设备地址,并非400行左右的那个MPU6050的.addr
    注意,该段代码为灰色,说明被编译器忽视了(因为MPU6500没有被define)
    修改:将400行左右的那个MPU6050的设备地址.addr的值修改为0xd0

    ret = -9

    改正

    inv_mpu.cint mpu_run_self_test(long *gyro, long *accel) 的2730行左右的这段代码注释掉

    DEBUG过程

    ret=-9对应MPU6050自检函数错误

    进入run_self_test();看该函数为何报错

    在result后将其值赋给error_code并打印,发现result值为0x7

    我们看下为什么result不为0x3
    进入该陀螺仪、加速度计自检函数

    该函数内部

    函数作用
    用result的第0位、第1位、第2位分别存储陀螺仪、加速度计、磁力计的自检结果。
    若自检通过,则对应位置1(位运算,按位或)
    若陀螺仪、加速度计、磁力计均自检通过,则result = 0111,也即0x7
    但我们没有接磁力计,所以最高位应该为0,也即result = 0011,也即0x3(这也是我们想要的
    我们看下,为什么最高位会被置1,也即为什么磁力计会自检通过

    我们看下这段代码,如果定义了磁力计,则巴拉巴拉。#else result |= 0x04;
    说明,即使没有定义磁力计(没有外接磁力计),自检也通过,最高位置1(result=0x7),这显然不对,我们注释掉这行代码

    改正

    作者:Abaaba+

    物联沃分享整理
    物联沃-IOTWORD物联网 » 【STM32 HAL库】MPU6050 DMP库移植 与 自检失败的处理

    发表回复