Zhao_kar 发表于 2023-10-24 20:29

【STM32L476RG】第六节——MPU6050的简单测试

<div class='showpostmsg'> 本帖最后由 Zhao_kar 于 2023-10-24 20:28 编辑

<p style="text-align: center;"><strong><span style="font-size:24px;">【STM32L476RG】第六节&mdash;&mdash;MPU6050的简单测试</span></strong></p>

<p style="text-align: center;"><span style="font-size:18px;"><strong><img height="28" src="https://bbs.eeworld.com.cn/static/editor/plugins/hkemoji/sticker/facebook/expressionless-face_1f611.png" width="28" />之前第五节写成了4,不过无所谓了</strong></span></p>

<p><span style="color:#3498db;">闲聊:这一篇是简单驱动MPU6050的,之前按照预计更新计划,更新了串口和oled的SPI,不过都是简单操作,这一篇同样也是简单测试,只是一个数据的读取和显示,实际配置起来不用几分钟就能配完。但是实际上MPU6050有非常多可以拿来玩的东西,我这一篇实际上只是移植了别人的库,如果按照正常的读手册到配置到使用官方的库,要花费大把时间,要学的东西也很多,这边还是希望各位能自己读读手册或者看看某点原子或者某火的教程深入学习一下,这边只是一个简单的测试,后续会打算补一个完整的mpu6050的测试报告。</span></p>

<p><span style="color:#3498db;">叠甲和声明:本篇只是学习用,只能说是对MPU6050的一个大概理解,希望各位读者不要觉得看了就会了,实际上官方手册里面可是有太多东西了,你照着我这篇去测试下来,除了知道怎么获得这个数据,对mpu6050的所有配置和驱动你基本上都不知道,要深入学习一下建议还是看官方。</span></p>

<p><span style="color:#3498db;">补充:目前已完成作品,是一个简易的计步器,但是作品里面依赖的是官方的算法,使用的是DMP库自带的七步计数器,然后这一篇是我学习过程中找到的快速上手的方法,也就是获取两组数据的办法,只需要移植一个库就行,并不需要一个一个的去更改官方库,至于官方的如何移植在后续总报告进行一个小分析。</span></p>

<p><span style="color:#f1c40f;"><strong><span style="font-size:20px;">一、MPU6050的简单理论</span></strong></span></p>

<p>1、<strong>MPU6050是啥</strong>:MPU6050是一款六轴传感器,包括一个三轴陀螺仪和一个三轴加速度计,所以说可以获得两组数据。</p>

<p>2、<strong>功能</strong>:MPU6050可以测量物体的加速度和角速度。这使其非常适合用于测量物体的运动、方向和倾斜角度。这些数据可以用于各种应用,如无人机、机器人、游戏控制器、虚拟现实设备等。</p>

<p>3、<strong>通信接口</strong>:MPU6050通过I2C接口与微控制器或单片机通信,一般MPU6050有以下引脚</p>

<ul>
        <li>VCC(供电)</li>
        <li>GND</li>
        <li>SCL(IIC通信)</li>
        <li>SDA(IIC通信)</li>
        <li>AD0(地址引脚,一般是接GND)</li>
        <li>INT(中断口,官方库会用到)</li>
        <li>XCL(这两个暂时不管,想了解请自己查查,我没搞过车,不是很了解那些佬用的什么电磁传感器啥的)</li>
        <li>XDA</li>
</ul>

<p><strong><span style="color:#f1c40f;"><span style="font-size:20px;">二、cubemx配置</span></span></strong></p>

<p>1、SYS和RCC不说了,时钟树也是</p>

<p>2、IIC配置打开,且fastmode,快速模式,如下图</p>

<div style="text-align: center;"></div>

<p>3、生成就行了,就是这么简单</p>

<p><strong><span style="font-size:20px;"><span style="color:#f1c40f;">三、添加mpu6050文件</span></span></strong></p>

<p>1、首先把两个.c和.h文件存在一个文件夹里面,然后命名,放到工程目录</p>

<p>2、添加文件路径,不放图了蛤,跟oled那期一样的</p>

<p>3、两个文件我放到末尾</p>

<p><span style="font-size:20px;"><span style="color:#f1c40f;"><strong>四、main函数编写</strong></span></span></p>

<p>1、添加库</p>

<pre>
<code class="language-cpp">/* USER CODE BEGIN Includes */
#include "mpu6050.h"
/* USER CODE END Includes */
</code></pre>

<p>2、定义结构体</p>

<pre>
<code class="language-cpp">/* USER CODE BEGIN PV */
MPU6050_t MPU6050;
/* USER CODE END PV */</code></pre>

<p>3、初始化</p>

<pre>
<code class="language-cpp">/* USER CODE BEGIN 2 */
OLED_Init();
        OLED_Clear();
OLED_Display_On();
        while (MPU6050_Init(&amp;hi2c2) == 1);
//oled别管,上一节测试留下的
/* USER CODE END 2 */</code></pre>

<p>4、读取的函数</p>

<p>这里在h文件里面有如下四个,看着想用哪个用哪个,注,前面cubemx配置的IIC2,所以这里是2</p>

<pre>
<code class="language-cpp">MPU6050_Read_All        //所有参数
MPU6050_Read_Accel        //加速度
MPU6050_Read_Gyro        //陀螺仪
MPU6050_Read_Temp        //温度

//比如我这里用的
MPU6050_Read_All(&amp;hi2c2, &amp;MPU6050);</code></pre>

<p>5、串口调试</p>

<p>说实在,这一篇我10.7就写完了,然后没上实物拍照,拖到现在,我就不用oled展示了,这边直接使用串口打印参数。</p>

<pre>
<code class="language-cpp">        printf("加速度计参数 X1:%.3f \t\r Y1:%.3f \t\r Z1:%.3f\t\r\n",MPU6050.Ax,MPU6050.Ay,MPU6050.Az);
        printf("陀螺仪参数 X2:%.3f \t\r Y2:%.3f \t\r Z2:%.3f\t\r\n",MPU6050.Gx,MPU6050.Gy,MPU6050.Gz);
        printf("温度参数 %.3f\r\n",MPU6050.Temperature);
//补充一下,前面的结构体可以去看看,里面能解释为什么我这边是这样,结构体的内容去看文章后面的两个文件吧</code></pre>

<p>6、可以看到串口打印如下参数,如图</p>

<div style="text-align: center;"></div>

<div>最后给个简单的演示视频吧,因为没用oled,所以看着没那么顺滑。</div>

<div>3ae2aceb990a6113941c1801a215a483<br />
&nbsp;</div>

<p><strong><span style="color:#f1c40f;"><span style="font-size:20px;">五、两个文件+总结</span></span></strong></p>

<p>1、mpu6050.c</p>

<pre>
<code class="language-cpp">/*
* mpu6050.c
*
*Created on: Nov 13, 2019
*      Author: Bulanov Konstantin
*
*Contact information
*-------------------
*
* e-mail   :leech001@gmail.com
*/

/*
* |---------------------------------------------------------------------------------
* | Copyright (C) Bulanov Konstantin,2021
* |
* | This program is free software: you can redistribute it and/or modify
* | it under the terms of the GNU General Public License as published by
* | the Free Software Foundation, either version 3 of the License, or
* | any later version.
* |
* | This program is distributed in the hope that it will be useful,
* | but WITHOUT ANY WARRANTY; without even the implied warranty of
* | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.See the
* | GNU General Public License for more details.
* |
* | You should have received a copy of the GNU General Public License
* | along with this program.If not, see &lt;http://www.gnu.org/licenses/&gt;.
* |
* | Kalman filter algorithm used from https://github.com/TKJElectronics/KalmanFilter
* |---------------------------------------------------------------------------------
*/

#include &lt;math.h&gt;
#include "mpu6050.h"
#include "main.h"


#define RAD_TO_DEG 57.295779513082320876798154814105

#define WHO_AM_I_REG 0x75
#define PWR_MGMT_1_REG 0x6B
#define SMPLRT_DIV_REG 0x19
#define ACCEL_CONFIG_REG 0x1C
#define ACCEL_XOUT_H_REG 0x3B
#define TEMP_OUT_H_REG 0x41
#define GYRO_CONFIG_REG 0x1B
#define GYRO_XOUT_H_REG 0x43

// Setup MPU6050
#define MPU6050_ADDR 0xD0
const uint16_t i2c_timeout = 100;
const double Accel_Z_corrector = 14418.0;
// const double Accel_Z_corrector = 16384.0;

uint32_t timer;

Kalman_t KalmanX = {
    .Q_angle = 0.001f,
    .Q_bias = 0.003f,
    .R_measure = 0.03f};

Kalman_t KalmanY = {
    .Q_angle = 0.001f,
    .Q_bias = 0.003f,
    .R_measure = 0.03f,
};

uint8_t MPU6050_Init(I2C_HandleTypeDef *I2Cx)
{
    uint8_t check;
    uint8_t Data;

    // check device ID WHO_AM_I

    HAL_I2C_Mem_Read(I2Cx, MPU6050_ADDR, WHO_AM_I_REG, 1, &amp;check, 1, i2c_timeout);

    if (check == 104) // 0x68 will be returned by the sensor if everything goes well
    {
      // power management register 0X6B we should write all 0's to wake the sensor up
      Data = 0;
      HAL_I2C_Mem_Write(I2Cx, MPU6050_ADDR, PWR_MGMT_1_REG, 1, &amp;Data, 1, i2c_timeout);

      // Set DATA RATE of 1KHz by writing SMPLRT_DIV register
      Data = 0x07;
      HAL_I2C_Mem_Write(I2Cx, MPU6050_ADDR, SMPLRT_DIV_REG, 1, &amp;Data, 1, i2c_timeout);

      // Set accelerometer configuration in ACCEL_CONFIG Register
      // XA_ST=0,YA_ST=0,ZA_ST=0, FS_SEL=0 -&gt; � 2g
      Data = 0x00;
      HAL_I2C_Mem_Write(I2Cx, MPU6050_ADDR, ACCEL_CONFIG_REG, 1, &amp;Data, 1, i2c_timeout);

      // Set Gyroscopic configuration in GYRO_CONFIG Register
      // XG_ST=0,YG_ST=0,ZG_ST=0, FS_SEL=0 -&gt; � 250 �/s
      Data = 0x00;
      HAL_I2C_Mem_Write(I2Cx, MPU6050_ADDR, GYRO_CONFIG_REG, 1, &amp;Data, 1, i2c_timeout);
      return 0;
    }
    return 1;
}

void MPU6050_Read_Accel(I2C_HandleTypeDef *I2Cx, MPU6050_t *DataStruct)
{
    uint8_t Rec_Data;

    // Read 6 BYTES of data starting from ACCEL_XOUT_H register

    HAL_I2C_Mem_Read(I2Cx, MPU6050_ADDR, ACCEL_XOUT_H_REG, 1, Rec_Data, 6, i2c_timeout);

    DataStruct-&gt;Accel_X_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    DataStruct-&gt;Accel_Y_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    DataStruct-&gt;Accel_Z_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);

    /*** convert the RAW values into acceleration in 'g'
         we have to divide according to the Full scale value set in FS_SEL
         I have configured FS_SEL = 0. So I am dividing by 16384.0
         for more details check ACCEL_CONFIG Register            ****/

    DataStruct-&gt;Ax = DataStruct-&gt;Accel_X_RAW / 16384.0;
    DataStruct-&gt;Ay = DataStruct-&gt;Accel_Y_RAW / 16384.0;
    DataStruct-&gt;Az = DataStruct-&gt;Accel_Z_RAW / Accel_Z_corrector;
}

void MPU6050_Read_Gyro(I2C_HandleTypeDef *I2Cx, MPU6050_t *DataStruct)
{
    uint8_t Rec_Data;

    // Read 6 BYTES of data starting from GYRO_XOUT_H register

    HAL_I2C_Mem_Read(I2Cx, MPU6050_ADDR, GYRO_XOUT_H_REG, 1, Rec_Data, 6, i2c_timeout);

    DataStruct-&gt;Gyro_X_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    DataStruct-&gt;Gyro_Y_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    DataStruct-&gt;Gyro_Z_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);

    /*** convert the RAW values into dps (�/s)
         we have to divide according to the Full scale value set in FS_SEL
         I have configured FS_SEL = 0. So I am dividing by 131.0
         for more details check GYRO_CONFIG Register            ****/

    DataStruct-&gt;Gx = DataStruct-&gt;Gyro_X_RAW / 131.0;
    DataStruct-&gt;Gy = DataStruct-&gt;Gyro_Y_RAW / 131.0;
    DataStruct-&gt;Gz = DataStruct-&gt;Gyro_Z_RAW / 131.0;
}

void MPU6050_Read_Temp(I2C_HandleTypeDef *I2Cx, MPU6050_t *DataStruct)
{
    uint8_t Rec_Data;
    int16_t temp;

    // Read 2 BYTES of data starting from TEMP_OUT_H_REG register

    HAL_I2C_Mem_Read(I2Cx, MPU6050_ADDR, TEMP_OUT_H_REG, 1, Rec_Data, 2, i2c_timeout);

    temp = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    DataStruct-&gt;Temperature = (float)((int16_t)temp / (float)340.0 + (float)36.53);
}

void MPU6050_Read_All(I2C_HandleTypeDef *I2Cx, MPU6050_t *DataStruct)
{
    uint8_t Rec_Data;
    int16_t temp;

    // Read 14 BYTES of data starting from ACCEL_XOUT_H register

    HAL_I2C_Mem_Read(I2Cx, MPU6050_ADDR, ACCEL_XOUT_H_REG, 1, Rec_Data, 14, i2c_timeout);

    DataStruct-&gt;Accel_X_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    DataStruct-&gt;Accel_Y_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    DataStruct-&gt;Accel_Z_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    temp = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    DataStruct-&gt;Gyro_X_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    DataStruct-&gt;Gyro_Y_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);
    DataStruct-&gt;Gyro_Z_RAW = (int16_t)(Rec_Data &lt;&lt; 8 | Rec_Data);

    DataStruct-&gt;Ax = DataStruct-&gt;Accel_X_RAW / 16384.0;
    DataStruct-&gt;Ay = DataStruct-&gt;Accel_Y_RAW / 16384.0;
    DataStruct-&gt;Az = DataStruct-&gt;Accel_Z_RAW / Accel_Z_corrector;
    DataStruct-&gt;Temperature = (float)((int16_t)temp / (float)340.0 + (float)36.53);
    DataStruct-&gt;Gx = DataStruct-&gt;Gyro_X_RAW / 131.0;
    DataStruct-&gt;Gy = DataStruct-&gt;Gyro_Y_RAW / 131.0;
    DataStruct-&gt;Gz = DataStruct-&gt;Gyro_Z_RAW / 131.0;

    // Kalman angle solve
    double dt = (double)(HAL_GetTick() - timer) / 1000;
    timer = HAL_GetTick();
    double roll;
    double roll_sqrt = sqrt(
      DataStruct-&gt;Accel_X_RAW * DataStruct-&gt;Accel_X_RAW + DataStruct-&gt;Accel_Z_RAW * DataStruct-&gt;Accel_Z_RAW);
    if (roll_sqrt != 0.0)
    {
      roll = atan(DataStruct-&gt;Accel_Y_RAW / roll_sqrt) * RAD_TO_DEG;
    }
    else
    {
      roll = 0.0;
    }
    double pitch = atan2(-DataStruct-&gt;Accel_X_RAW, DataStruct-&gt;Accel_Z_RAW) * RAD_TO_DEG;
    if ((pitch &lt; -90 &amp;&amp; DataStruct-&gt;KalmanAngleY &gt; 90) || (pitch &gt; 90 &amp;&amp; DataStruct-&gt;KalmanAngleY &lt; -90))
    {
      KalmanY.angle = pitch;
      DataStruct-&gt;KalmanAngleY = pitch;
    }
    else
    {
      DataStruct-&gt;KalmanAngleY = Kalman_getAngle(&amp;KalmanY, pitch, DataStruct-&gt;Gy, dt);
    }
    if (fabs(DataStruct-&gt;KalmanAngleY) &gt; 90)
      DataStruct-&gt;Gx = -DataStruct-&gt;Gx;
    DataStruct-&gt;KalmanAngleX = Kalman_getAngle(&amp;KalmanX, roll, DataStruct-&gt;Gx, dt);
}

double Kalman_getAngle(Kalman_t *Kalman, double newAngle, double newRate, double dt)
{
    double rate = newRate - Kalman-&gt;bias;
    Kalman-&gt;angle += dt * rate;

    Kalman-&gt;P += dt * (dt * Kalman-&gt;P - Kalman-&gt;P - Kalman-&gt;P + Kalman-&gt;Q_angle);
    Kalman-&gt;P -= dt * Kalman-&gt;P;
    Kalman-&gt;P -= dt * Kalman-&gt;P;
    Kalman-&gt;P += Kalman-&gt;Q_bias * dt;

    double S = Kalman-&gt;P + Kalman-&gt;R_measure;
    double K;
    K = Kalman-&gt;P / S;
    K = Kalman-&gt;P / S;

    double y = newAngle - Kalman-&gt;angle;
    Kalman-&gt;angle += K * y;
    Kalman-&gt;bias += K * y;

    double P00_temp = Kalman-&gt;P;
    double P01_temp = Kalman-&gt;P;

    Kalman-&gt;P -= K * P00_temp;
    Kalman-&gt;P -= K * P01_temp;
    Kalman-&gt;P -= K * P00_temp;
    Kalman-&gt;P -= K * P01_temp;

    return Kalman-&gt;angle;
};
</code></pre>

<p>2、mpu6050.h</p>

<pre>
<code class="language-cpp">/*
* mpu6050.h
*
*Created on: Nov 13, 2019
*      Author: Bulanov Konstantin
*/

#ifndef INC_GY521_H_
#define INC_GY521_H_

#endif /* INC_GY521_H_ */

#include &lt;stdint.h&gt;
#include "i2c.h"

// MPU6050 structure
typedef struct
{

    int16_t Accel_X_RAW;
    int16_t Accel_Y_RAW;
    int16_t Accel_Z_RAW;
    double Ax;
    double Ay;
    double Az;

    int16_t Gyro_X_RAW;
    int16_t Gyro_Y_RAW;
    int16_t Gyro_Z_RAW;
    double Gx;
    double Gy;
    double Gz;

    float Temperature;

    double KalmanAngleX;
    double KalmanAngleY;
} MPU6050_t;

// Kalman structure
typedef struct
{
    double Q_angle;
    double Q_bias;
    double R_measure;
    double angle;
    double bias;
    double P;
} Kalman_t;

uint8_t MPU6050_Init(I2C_HandleTypeDef *I2Cx);

void MPU6050_Read_Accel(I2C_HandleTypeDef *I2Cx, MPU6050_t *DataStruct);

void MPU6050_Read_Gyro(I2C_HandleTypeDef *I2Cx, MPU6050_t *DataStruct);

void MPU6050_Read_Temp(I2C_HandleTypeDef *I2Cx, MPU6050_t *DataStruct);

void MPU6050_Read_All(I2C_HandleTypeDef *I2Cx, MPU6050_t *DataStruct);

double Kalman_getAngle(Kalman_t *Kalman, double newAngle, double newRate, double dt);
</code></pre>

<p>3、总结</p>

<p>本篇只是一个移植了别人的<span style="color:#e74c3c;">开源库</span>进行使用的,<span style="color:#e74c3c;">非常非常非常非常简单</span>的一个办法,当然,你除了能获得这个参数外,基本不可能学到mpu6050的其他知识,包括四元数等等,滤波算法的计算,都是不知道的,然后作品已经做完了,下一篇报告就是完整的项目展示,同时会重点介绍官方的DMP库是如何实现的。</p>

<p>补充两点</p>

<p>1、移植官方库的过程非常繁琐,我也无法短时间内,一个十分钟内的视频去介绍,所以我在最终的视频报告只会讲主要部分。</p>

<p>2、官方的DMP库确实有计步器的功能,这个算法依赖于七步计数,但是市面上的计步器产品,大多都是自己写的算法,作者也是首次学习mpu6050,所以使用的官方库。</p>

<p>3、声明:这个库是我看别人文章发现的,如有侵权,请联系我</p>
</div><script>                                        var loginstr = '<div class="locked">查看本帖全部内容,请<a href="javascript:;"   style="color:#e60000" class="loginf">登录</a>或者<a href="https://bbs.eeworld.com.cn/member.php?mod=register_eeworld.php&action=wechat" style="color:#e60000" target="_blank">注册</a></div>';
                                       
                                        if(parseInt(discuz_uid)==0){
                                                                                                (function($){
                                                        var postHeight = getTextHeight(400);
                                                        $(".showpostmsg").html($(".showpostmsg").html());
                                                        $(".showpostmsg").after(loginstr);
                                                        $(".showpostmsg").css({height:postHeight,overflow:"hidden"});
                                                })(jQuery);
                                        }                </script><script type="text/javascript">(function(d,c){var a=d.createElement("script"),m=d.getElementsByTagName("script"),eewurl="//counter.eeworld.com.cn/pv/count/";a.src=eewurl+c;m.parentNode.insertBefore(a,m)})(document,523)</script>

Jacktang 发表于 2023-10-25 07:26

<p>移植了别人的开源库进行使用,也是实际开发人常用的技法</p>

<p>只要关键地方是自己写的就行,,,</p>

<p>哈哈,支持楼主</p>

Verifone 发表于 2023-10-25 08:47

<p>有开源库可以用还是可以缩短很多开发时间的,自己写库太麻烦了</p>
页: [1]
查看完整版本: 【STM32L476RG】第六节——MPU6050的简单测试