打印
[其他ST产品]

模块介绍之六轴陀螺仪MPU6050篇(STM32基本使用

[复制链接]
709|28
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
慢醇|  楼主 | 2022-6-28 10:34 | 只看该作者 |只看大图 回帖奖励 |倒序浏览 |阅读模式
已经有很多大佬介绍过了MPU6050的协议、数据处理方式、滤波算法,所以这部分就不再复述了,本篇主要是针对看不懂长篇大论的小白的简易上手的方法。
首先呢还是得简单介绍下什么是陀螺仪:


MPU-60X0是世界上第一款集成 6 轴MotionTracking设备。

使用特权

评论回复
沙发
慢醇|  楼主 | 2022-6-28 10:39 | 只看该作者
它集成了3轴MEMS陀螺仪,3轴MEMS加速度计,以及一个可扩展的数字运动处理器 DMP( DigitalMotion Processor),可用I2C接口连接一个第三方的数字传感器,比如磁力计。

使用特权

评论回复
板凳
慢醇|  楼主 | 2022-6-28 10:40 | 只看该作者
么本篇文章呢,主要介绍通过复制粘贴的代码可以实现对当前角度的读取(软件IIC),检测三轴方向上的角度,假如该陀螺仪平放在水平面上,从侧面看为一个线段,现取一个顶点(圆点),让另一边绕该点旋转,形成一个角度的示意图,如下图,这样便呈现了一个角度,我们的单片机通过协议和处理方法,将这个角度读取出来(-90-90°),当换一个平面时,相同的绕圆点旋转便可以出现Y轴,

使用特权

评论回复
地板
慢醇|  楼主 | 2022-6-28 10:41 | 只看该作者

使用特权

评论回复
5
慢醇|  楼主 | 2022-6-28 10:43 | 只看该作者
当如果我们想读取Z轴时(不常用),选择的方式是以该平面的中心为圆点进行旋转,可以得到一个角度值,该值通过本篇方法处理后有一定的偏差,常用的为X和Y轴

使用特权

评论回复
6
慢醇|  楼主 | 2022-6-28 10:43 | 只看该作者

使用特权

评论回复
7
慢醇|  楼主 | 2022-6-28 10:45 | 只看该作者
上面简单介绍了X、Y、Z轴读取后的数据分别代码什么内容,下面来讲解针对STM32平台的代码移植过程,本篇使用的方式为软件IIC(通过IO的读和写来实现数据读取,方便移植)

使用特权

评论回复
8
慢醇|  楼主 | 2022-6-28 11:02 | 只看该作者
首先我们打开该工程的OLED_I2C.H文件(本文最下面附链,以STM32F1为例)

使用特权

评论回复
9
慢醇|  楼主 | 2022-6-28 11:02 | 只看该作者
#define OLED_I2C                          I2C1
#define OLED_I2C_CLK                      RCC_APB1Periph_I2C1
#define OLED_I2C_CLK_INIT                                                                  RCC_APB1PeriphClockCmd

#define OLED_I2C_SCL_PIN                  GPIO_Pin_6                 
#define OLED_I2C_SCL_GPIO_PORT            GPIOB                       
#define OLED_I2C_SCL_GPIO_CLK             RCC_AHB1Periph_GPIOB
#define OLED_I2C_SCL_SOURCE               GPIO_PinSource6
#define OLED_I2C_SCL_AF                   GPIO_AF_I2C1

#define OLED_I2C_SDA_PIN                  GPIO_Pin_7                  
#define OLED_I2C_SDA_GPIO_PORT            GPIOB                       
#define OLED_I2C_SDA_GPIO_CLK             RCC_AHB1Periph_GPIOB
#define OLED_I2C_SDA_SOURCE               GPIO_PinSource7
#define OLED_I2C_SDA_AF                   GPIO_AF_I2C1

void I2C_Configuration(void);

void I2C_WriteByte(uint8_t addr,uint8_t data);
void WriteCmd(unsigned char I2C_Command);
void WriteDat(unsigned char I2C_Data);

void OLED_Init(void);
void OLED_SetPos(unsigned char x, unsigned char y);
void OLED_Fill(unsigned char fill_Data);
void OLED_CLS(void);
void OLED_ON(void);
void OLED_OFF(void);
void OLED_ShowStr(unsigned char x, unsigned char y, unsigned char ch[], unsigned char TextSize);
void OLED_ShowCN(unsigned char x, unsigned char y, unsigned char N);
void OLED_DrawBMP(unsigned char x0,unsigned char y0,unsigned char x1,unsigned char y1,unsigned char BMP[]);

使用特权

评论回复
10
慢醇|  楼主 | 2022-6-28 11:04 | 只看该作者
该头文件部分对使用到的引脚进行了声明,例如该工程中使用到的SCL引脚为PB6,SDA引脚为PB7,同时我们看一下OLED_I2C.C中是如何使用这些内容的

使用特权

评论回复
11
慢醇|  楼主 | 2022-6-28 11:05 | 只看该作者
void I2C_Configuration(void)
{
        I2C_InitTypeDef  I2C_InitStructure;
        GPIO_InitTypeDef  GPIO_InitStructure;

        OLED_I2C_CLK_INIT(OLED_I2C_CLK,ENABLE);
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);

    GPIO_InitStructure.GPIO_Pin = OLED_I2C_SCL_PIN;
    GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
    GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD;               // 开漏输出
    GPIO_Init(OLED_I2C_SCL_GPIO_PORT, &GPIO_InitStructure);
       
    GPIO_InitStructure.GPIO_Pin = OLED_I2C_SDA_PIN;
    GPIO_Init(OLED_I2C_SDA_GPIO_PORT, &GPIO_InitStructure);       
       
        I2C_InitStructure.I2C_Mode = I2C_Mode_I2C;
        I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2;/* 高电平数据稳定,低电平数据变化 SCL 时钟线的占空比 */
        I2C_InitStructure.I2C_OwnAddress1 = OLED_ADDRESS;//OLED的I2C地址
        I2C_InitStructure.I2C_Ack = I2C_Ack_Enable;
        I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;/* I2C的寻址模式 */
        I2C_InitStructure.I2C_ClockSpeed = I2C_Speed;

        I2C_Init(I2C1, &I2C_InitStructure); /* I2C1 初始化 */
        I2C_Cmd(I2C1, ENABLE);               /* 使能 I2C1 */

}

使用特权

评论回复
12
慢醇|  楼主 | 2022-6-28 11:05 | 只看该作者
发现在该C文件中使用到初始化的部分都是通过#define的值进行调用的,所以!! 在本工程中只需要改动OLED_I2C.H种的内容就可以完全实现移植,兼容任何STM32的标准库

使用特权

评论回复
13
慢醇|  楼主 | 2022-6-28 11:06 | 只看该作者
说完了移植代码,下面看一下如何获取数据:

首先在main()函数里面需要有两个初始化,
//I2C初始化
i2c_GPIO_Config();
//MPU6050初始化
MPU6050_Init();
while(MPU6050ReadID() == 0);
               

使用特权

评论回复
14
慢醇|  楼主 | 2022-6-28 11:16 | 只看该作者
通过这两个初始化和再次校验,MPU6050的初始化就成功了,现在我们进入数据读取环节
/**
  * [url=home.php?mod=space&uid=247401]@brief[/url]   读取MPU6050的加速度数据
  * @param   
  * @retval  
  */
void MPU6050ReadAcc(short *accData)
{
    u8 buf[6];
    MPU6050_ReadData(MPU6050_ACC_OUT, buf, 6);
    accData[0] = (buf[0] << 8) | buf[1];
    accData[1] = (buf[2] << 8) | buf[3];
    accData[2] = (buf[4] << 8) | buf[5];
}

/**
  * @brief   读取MPU6050的角加速度数据
  * @param   
  * @retval  
  */
void MPU6050ReadGyro(short *gyroData)
{
    u8 buf[6];
    MPU6050_ReadData(MPU6050_GYRO_OUT,buf,6);
    gyroData[0] = (buf[0] << 8) | buf[1];
    gyroData[1] = (buf[2] << 8) | buf[3];
    gyroData[2] = (buf[4] << 8) | buf[5];
}


/**
  * @brief   读取MPU6050的原始温度数据
  * @param   
  * @retval  
  */
void MPU6050ReadTemp(short *tempData)
{
        u8 buf[2];
    MPU6050_ReadData(MPU6050_RA_TEMP_OUT_H,buf,2);     //读取温度值
    *tempData = (buf[0] << 8) | buf[1];
}

使用特权

评论回复
15
慢醇|  楼主 | 2022-6-28 14:22 | 只看该作者
所有的读取数据都是通过这三个函数实现的,经过上面的初始化后可以直接在主函数中进行使用,我们这里用到的主要是MPU6050ReadGyro()函数,函数内封装有数据转换,可以直接得到角度值,此时我们先新建一个变量

short Gyro[3];

使用特权

评论回复
16
慢醇|  楼主 | 2022-6-28 14:36 | 只看该作者
新建一个成员为3的数组,方便一次性获取X、Y、Z轴的数据,使用方式为:

使用特权

评论回复
17
慢醇|  楼主 | 2022-6-28 14:37 | 只看该作者
MPU6050ReadGyro(Gyro);
JD1[0]=abs(Gyro[0])/100+'0';
JD1[1]=abs(Gyro[0])/10%10+'0';
JD1[2]=abs(Gyro[0])%10+'0';

使用特权

评论回复
18
慢醇|  楼主 | 2022-6-28 14:38 | 只看该作者
该语句是将X轴的角度数据转换为字符串数据,方便新手下面通过串口输出数值进行转换

因为本人身边现在没有MPU6050,所以串口助手生成数据这一部分就不再演示了

使用特权

评论回复
19
慢醇|  楼主 | 2022-6-28 14:40 | 只看该作者
工程文件链接: https://pan.baidu.com/s/1luZSUbHr9ZiVQGdiI5_umw 提取码: 05lh

使用特权

评论回复
20
Uriah| | 2022-10-5 07:07 | 只看该作者

通过访问寄存器来控制I2C1工作时钟的开启。

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

125

主题

1241

帖子

5

粉丝