【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.c
中int 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),这显然不对,我们注释掉这行代码
改正: