STM32外设系列—MPU6050角度传感器

🎀 文章作者:二土电子

🌸 关注公众号获取更多资料!

🐸 期待大家一起学习交流!


文章目录

  • 一、MPU6050简介
  • 二、MPU6050寄存器简介
    • 2.1 PWR_MGMT_1寄存器
    • 2.2 GYRO_CONFIG寄存器
    • 2.3 ACCEL_CONFIG寄存器
    • 2.4 PWR_MGMT_2寄存器
  • 三、程序设计
    • 3.1 I2C程序设计
    • 3.2 MPU6050初始化程序
    • 3.3 DMP相关程序
    • 3.4 获取三轴角度信息
  • 四、实现效果

一、MPU6050简介

  MPU6050是由InvenSense公司生产的全球首款整合性六轴运动处理模块,它可以实时获取运动物体的在三维坐标系内的偏转角度,如图所示。

MPU6050偏转角度示意图

  其中roll为绕X轴偏转的角度,pitch为绕Y轴偏转的角度,yaw为绕Z轴偏转的角度。

  MPU6050通过IIC协议与单片机进行通讯,传递偏移角度信息。虽然MPU6050角度传感器体积小,功能强大,但是其内部噪音较大,如果不进行滤波,将会对获取到的偏转角度的准确性带来严重的影响。为此,MPU6050芯片内部集成了一个DMP数据处理模块,该模块已经内置了滤波算法,使MPU6050输出数据的准确性得到了保障。

之所以说MPU6050是六轴运动处理模块,是因为它不仅集成了三轴陀螺仪,还集成了一个三轴加速度计,不仅可以输出三轴角度信息,还可以输出三轴加速度信息。除此之外,MPU6050还集成了一个温度传感器,可以输出温度信息。

  下面我们简单介绍一下MPU6050的引脚。通常我们买到的MPU6050大概是这个样子
MPU6050模块

  • VCC
    电源正极,通常接3.3V
  • GND
    电源地
  • SCL
    IIC的时钟线
  • SDA
    IIC的数据线
  • XDA
    外接IIC设备的数据线
  • XCL
    外接IIC设备的时钟线
  • AD0
    控制IIC从属地址,接地时地址为0X68,如果接VCC的话,从属地址为0X69
  • INT
    中断数字输出

  通过对MPU6050的引脚介绍我们可以知道,MPU6050可以外接一个IIC设备,通常可以外接一个三轴的磁力计来实现完整的九轴输出。

  这里放一下MPU60X0数据手册中的系统结构图,基本信息上面已经介绍了,就不再进行详细标注了

MPU60X0系统结构图

二、MPU6050寄存器简介

  在开始学习怎么使用MPU6050之前,我们先简单看一下它的一些寄存器。

  这里贴一下MPU6050的数据手册和寄存器手册,但是只有英文版的,如果大家只是想了解一些关键寄存器的介绍,又不想自己看英文手册的话,可以继续往下看,下面会有一些关键寄存器的中文介绍。


链接:https://pan.baidu.com/s/1r-xDm7QUp5NfVRPF4IMX9g
提取码:ertu

2.1 PWR_MGMT_1寄存器

  首先看一下寄存器手册中对它的介绍。

PWR_MGMT_1寄存器介绍

  • DEVICE_RESET
    当该位设置为1时,MPU6050会将所有内部寄存器重置为其默认值。在重置完成后,该位将自动清除为0。
  • SLEEP
    当该位设置成1时,MPU6050进入休眠模式。
  • CYCLE
    当这个位被设置为1并且睡眠被禁用时,MPU-60X0将在睡眠模式和醒来之间循环,以由LP_WAKE_CTRL(寄存器108)确定的速率从加速度计中获取单个数据样本。
  • TEMP_DIS
    当设置为1时,此位将禁用温度传感器。
  • CLKSEL
    3位无符号值。指定该设备的时钟源。

2.2 GYRO_CONFIG寄存器


GYRO_CONFIG寄存器
  该寄存器用于触发陀螺仪自检,并配置陀螺仪的量程范围。陀螺仪自检允许用户测试陀螺仪的机械和电气部分。通过控制该寄存器的XG_ST、YG_ST和ZG_ST位,可以激活每个陀螺仪轴的自检。每个轴的自检可以独立进行,也可以同时进行。

  下面我们仔细看一下每一位的作用

  • XG_ST
    设置此位会开启X轴陀螺仪自检。
  • YG_ST
    设置此位会开启Y轴陀螺仪自检。
  • ZG_ST
    设置此位会开启Z轴陀螺仪自检。
  • FS_SEL
    2位无符号值。选择陀螺仪的量程范围。
FS_SEL陀螺仪传感器满量程范围
0±250°/s
1±500°/s
2±1000°/s
3±2000°/s

    这里°/s是角速度的单位。

2.3 ACCEL_CONFIG寄存器


ACCEL_CONFIG寄存器

  该寄存器用于触发加速度计的自检,并配置加速度计的量程范围。加速度计自检允许用户测试加速度计的机械和电气部分。通过控制该寄存器的XA_ST、YA_ST和ZA_ST位,可以激活每个加速度计轴的自检。每个轴的自检可以独立进行,也可以同时进行。

  下面我们仔细看一下每一位的作用

  • XA_ST
    当设置为1时,X轴加速度计进行自检。
  • YA_ST
    当设置为1时,Y轴加速度计进行自检。
  • ZA_ST
    当设置为1时,Z轴加速度计进行自检。
  • AFS_SEL
    2位无符号值。选择加速度计的量程范围。
AFS_SEL加速度计满量程范围
0±2g
1±4g
3±8g
4±16g

2.4 PWR_MGMT_2寄存器


PWR_MGMT_2寄存器

  此寄存器允许用户在仅加速计低功率模式下配置唤醒频率。这个寄存器还允许用户将加速度计和陀螺仪的各个轴进入待机模式。

  • STBY_XA
    当设置为1时,该位将使X轴加速度计进入待机模式。
  • STBY_YA
    当设置为1时,该位将使Y轴加速度计进入待机模式。
  • STBY_ZA
    当设置为1时,该位将使Z轴加速度计进入待机模式。
  • STBY_XG
    当设置为1时,此位将使X轴陀螺仪进入待机模式。
  • STBY_YG
    当设置为1时,该位将使Y轴陀螺仪进入待机模式。
  • STBY_ZG
    当设置为1时,此位将使Z轴陀螺仪进入待机模式。

三、程序设计

  本次的程序设计使用的是STM32F103C8T6作主控,串口输出三轴角度信息。

3.1 I2C程序设计

  MPU6050使用的是I2C通信,对于I2C这里不再作详细介绍,具体可以看博主STM32外设系列OLED中对I2C的介绍,这里直接给出程序设计

  drv层.c文件

/*
 *==============================================================================
 * 声明:本程序由CSDN博主“二土电子”编写整理,部分程序由外部参考借鉴。
 
 * 如需更多程序资源或STM32教程,可微信公众号搜索“二土电子”关注
 
 * STM32学习交流群:479667779,更多优质资源,等你来发现!
 
 * 2023.11.09 ---------------------------------------------------- code by ertu
 *==============================================================================
*/
#include "drv_mpu6050.h"
#include "delay.h"

/*
 *==============================================================================
 *函数名称:Drv_Mpu6050_Gpio_Init
 *函数功能:初始化MPU6050引脚
 *输入参数:无
 *返回值:无
 *备  注:这里不再给AD0另外接一个引脚,也就是从机地址固定为0X68
 *==============================================================================
*/
void Drv_Mpu6050_Gpio_Init (void)
{
	GPIO_InitTypeDef GPIO_InitStructure;   // 定义结构体
	// 开启时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);

	// 配置结构体
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_6 | GPIO_Pin_7;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;   // 推挽式输出
	GPIO_Init(GPIOB, &GPIO_InitStructure);
	GPIO_SetBits(GPIOB,GPIO_Pin_6 | GPIO_Pin_7);   // 拉高
}
/*
 *==============================================================================
 *函数名称:MPU_IIC_Delay
 *函数功能:MPU6050 IIC延时
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
*/
void MPU_IIC_Delay (void)
{
	delay_us(2);
}
/*
 *==============================================================================
 *函数名称:MPU_IIC_Start
 *函数功能:MPU6050 IIC发送起始信号
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
*/
void MPU_IIC_Start (void)
{
	MPU_SDA_OUT();   // SDA输出
	MPU_IIC_SDA = 1;	  	  
	MPU_IIC_SCL = 1;
	MPU_IIC_Delay();
	MPU_IIC_SDA = 0;
	MPU_IIC_Delay();
	MPU_IIC_SCL = 0;
}
/*
 *==============================================================================
 *函数名称:MPU_IIC_Stop
 *函数功能:MPU6050 IIC发送终止信号
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
*/
void MPU_IIC_Stop (void)
{
	MPU_SDA_OUT();   // SDA输出
	MPU_IIC_SCL = 0;
	MPU_IIC_SDA = 0;
 	MPU_IIC_Delay();
	MPU_IIC_SCL = 1; 
	MPU_IIC_SDA = 1;
	MPU_IIC_Delay();							   	
}
/*
 *==============================================================================
 *函数名称:MPU_IIC_Wait_Ack
 *函数功能:MPU6050 IIC等待应答信号
 *输入参数:无
 *返回值:0:收到应答信号;1:未收到应答信号
 *备  注:无
 *==============================================================================
*/
u8 MPU_IIC_Wait_Ack (void)
{
	u8 ucErrTime = 0;
	
	MPU_SDA_IN();   // SDA输入
	MPU_IIC_SDA = 1;
	MPU_IIC_Delay();	   
	MPU_IIC_SCL = 1;
	MPU_IIC_Delay();
	
	while(MPU_READ_SDA)
	{
		ucErrTime++;
		if(ucErrTime>250)
		{
			MPU_IIC_Stop();
			return 1;
		}
	}
	MPU_IIC_SCL = 0;   // 时钟输出0 	   
	return 0;  
}
/*
 *==============================================================================
 *函数名称:MPU_IIC_Ack
 *函数功能:MPU6050 IIC发送应答信号
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
*/
void MPU_IIC_Ack (void)
{
	MPU_IIC_SCL = 0;
	MPU_SDA_OUT();   // SDA输出
	MPU_IIC_SDA = 0;
	MPU_IIC_Delay();
	MPU_IIC_SCL = 1;
	MPU_IIC_Delay();
	MPU_IIC_SCL = 0;
}
/*
 *==============================================================================
 *函数名称:MPU_IIC_NAck
 *函数功能:MPU6050 IIC发送非应答信号
 *输入参数:无
 *返回值:无
 *备  注:无
 *==============================================================================
*/
void MPU_IIC_NAck (void)
{
	MPU_IIC_SCL = 0;
	MPU_SDA_OUT();   // SDA输出
	MPU_IIC_SDA = 1;
	MPU_IIC_Delay();
	MPU_IIC_SCL = 1;
	MPU_IIC_Delay();
	MPU_IIC_SCL = 0;
}	
/*
 *==============================================================================
 *函数名称:MPU_IIC_Send_Byte
 *函数功能:MPU6050 IIC发送一个字节
 *输入参数:无
 *返回值:txd:发送的字节
 *备  注:无
 *==============================================================================
*/
void MPU_IIC_Send_Byte (u8 txd)
{                        
	u8 t;
	
	MPU_SDA_OUT();   // SDA输出
	
	MPU_IIC_SCL = 0;   // 拉低时钟开始数据传输
	for(t = 0;t < 8;t ++)
	{              
			MPU_IIC_SDA = (txd & 0x80) >> 7;
			txd <<= 1; 	  
			MPU_IIC_SCL = 1;
			MPU_IIC_Delay(); 
			MPU_IIC_SCL = 0;	
			MPU_IIC_Delay();
	}	 
} 
/*
 *==============================================================================
 *函数名称:MPU_IIC_Read_Byte
 *函数功能:MPU6050 IIC读取一个字节
 *输入参数:ack = 1时,发送ACK;ack = 0时,发送NACK
 *返回值:接收到的一个自己数据
 *备  注:无
 *==============================================================================
*/
u8 MPU_IIC_Read_Byte (u8 ack)
{
	u8 i,receive = 0;
	
	MPU_SDA_IN();   // SDA输入
	
	for(i = 0;i < 8;i ++ )
	{
		MPU_IIC_SCL = 0; 
		MPU_IIC_Delay();
		MPU_IIC_SCL = 1;
		
		receive <<= 1;
		if(MPU_READ_SDA)
		{
			receive++;
		}			
		MPU_IIC_Delay(); 
	}
	
	if (!ack)
	{
		MPU_IIC_NAck();   // 发送NACK
	}
	else
	{
		MPU_IIC_Ack();   // 发送ACK
	}
	return receive;
}
/*
 *==============================================================================
 *函数名称:MPU_Write_Len
 *函数功能:IIC在固定地址开始连续写入数据
 *输入参数:addr:从设备地址;reg:寄存器地址;len:写入长度;
						*buf:要写入的数据地址
 *返回值:0:写入成功;1:写入失败
 *备  注:无
 *==============================================================================
*/
u8 MPU_Write_Len (u8 addr,u8 reg,u8 len,u8 *buf)
{
	u8 i; 
	
	MPU_IIC_Start(); 
	MPU_IIC_Send_Byte((addr << 1) | 0);   // 发送器件地址+写命令	
	
	if(MPU_IIC_Wait_Ack())   // 等待应答
	{
		MPU_IIC_Stop();		 
		return 1;		
	}
	
	MPU_IIC_Send_Byte(reg);   // 写寄存器地址
	MPU_IIC_Wait_Ack();   // 等待应答
	for(i = 0;i < len;i ++)
	{
		MPU_IIC_Send_Byte(buf[i]);	//发送数据
		if(MPU_IIC_Wait_Ack())		//等待ACK
		{
			MPU_IIC_Stop();	 
			return 1;		 
		}		
	} 
	
	MPU_IIC_Stop();	 
	return 0;	
} 
/*
 *==============================================================================
 *函数名称:MPU_Read_Len
 *函数功能:IIC在固定地址开始连续读取数据
 *输入参数:addr:从设备地址;reg:寄存器地址;len:读取长度;
						*buf:要读取的数据存储地址
 *返回值:0:读取成功;1:读取失败
 *备  注:无
 *==============================================================================
*/
u8 MPU_Read_Len (u8 addr,u8 reg,u8 len,u8 *buf)
{ 
 	MPU_IIC_Start(); 
	MPU_IIC_Send_Byte((addr << 1) | 0);   // 发送器件地址+写命令	
	if(MPU_IIC_Wait_Ack())   // 等待应答
	{
		MPU_IIC_Stop();		 
		return 1;		
	}
	MPU_IIC_Send_Byte(reg);   // 写寄存器地址
	MPU_IIC_Wait_Ack();   // 等待应答
	MPU_IIC_Start();
	MPU_IIC_Send_Byte((addr << 1) | 1);   // 发送器件地址+读命令	
	MPU_IIC_Wait_Ack();   // 等待应答 
	
	while(len)
	{
		if(len == 1)*buf = MPU_IIC_Read_Byte(0);   // 读数据,发送nACK 
		else *buf = MPU_IIC_Read_Byte(1);   // 读数据,发送ACK  
		len --;
		buf ++; 
	}    
	MPU_IIC_Stop();   // 产生一个停止条件 
	return 0;	
}
/*
 *==============================================================================
 *函数名称:MPU_Read_Len
 *函数功能:IIC在固定地址写入一字节数据
 *输入参数:reg:寄存器地址;data:数据;
 *返回值:0:写入成功;1:写入失败
 *备  注:无
 *==============================================================================
*/
u8 MPU_Write_Byte(u8 reg,u8 data)				 
{ 
	MPU_IIC_Start(); 
	MPU_IIC_Send_Byte((0X68 << 1) | 0);   // 发送器件地址+写命令	
	if(MPU_IIC_Wait_Ack())   // 等待应答
	{
		MPU_IIC_Stop();		 
		return 1;		
	}
	MPU_IIC_Send_Byte(reg);   // 写寄存器地址
	MPU_IIC_Wait_Ack();   // 等待应答 
	MPU_IIC_Send_Byte(data);   // 发送数据
	if(MPU_IIC_Wait_Ack())   // 等待ACK
	{
		MPU_IIC_Stop();	 
		return 1;		 
	}		 
	MPU_IIC_Stop();	 
	return 0;
}
/*
 *==============================================================================
 *函数名称:MPU_Read_Byte
 *函数功能:IIC在固定地址读取一字节数据
 *输入参数:reg:寄存器地址;
 *返回值:读取的数据
 *备  注:无
 *==============================================================================
*/
u8 MPU_Read_Byte (u8 reg)
{
	u8 res;
	MPU_IIC_Start(); 
	MPU_IIC_Send_Byte((0X68 << 1) | 0);   // 发送器件地址+写命令	
	MPU_IIC_Wait_Ack();   // 等待应答 
	MPU_IIC_Send_Byte(reg);   // 写寄存器地址
	MPU_IIC_Wait_Ack();   // 等待应答
	MPU_IIC_Start();
	MPU_IIC_Send_Byte((0X68 << 1) | 1);   // 发送器件地址+读命令	
	MPU_IIC_Wait_Ack();   // 等待应答 
	res=MPU_IIC_Read_Byte(0);   // 读取数据,发送nACK 
	MPU_IIC_Stop();   // 产生一个停止条件 
	return res;		
}
//设置MPU6050的数字低通滤波器
//lpf:数字低通滤波频率(Hz)
//返回值:0,设置成功
//    其他,设置失败 
/*
 *==============================================================================
 *函数名称:Drv_Mpu6050_Set_Lpf
 *函数功能:设置MPU6050的数字低通滤波器
 *输入参数:lpf:数字低通滤波频率(Hz)
 *返回值:0:成功;1:失败
 *备  注:无
 *==============================================================================
*/
u8 Drv_Mpu6050_Set_Lpf (u16 lpf)
{
	u8 data = 0;
	if(lpf >= 188)
	{
		data=1;
	}
	else if(lpf>=98)
	{
		data=2;
	}
	else if(lpf>=42)
	{
		data=3;
	}
	else if(lpf>=20)
	{
		data=4;
	}
	else if(lpf>=10)
	{
		data=5;
	}
	else
	{
		data=6;
	}
	return MPU_Write_Byte(MPU_CFG_REG,data);   // 设置数字低通滤波器  
}
/*
 *==============================================================================
 *函数名称:Drv_Mpu6050_Set_Rate
 *函数功能:设置MPU6050的采样率(假定Fs=1KHz)
 *输入参数:rate:4~1000(Hz)
 *返回值:0:成功;1:失败
 *备  注:无
 *==============================================================================
*/
u8 Drv_Mpu6050_Set_Rate (u16 rate)
{
	u8 data;
	
	if(rate > 1000)
	{
		rate=1000;
	}
	if(rate<4)
	{
		rate=4;
	}
	data = 1000 / rate - 1;
	data = MPU_Write_Byte(MPU_SAMPLE_RATE_REG,data);   // 设置数字低通滤波器
 	return Drv_Mpu6050_Set_Lpf(rate / 2);   // 自动设置LPF为采样率的一半
}

  drv层.h文件

#ifndef _DRV_MPU6050_H
#define _DRV_MPU6050_H

#include "sys.h"   // STM32库函数头文件

// IIC引脚宏定义
#define MPU_IIC_SCL    PBout(6) 		// SCL
#define MPU_IIC_SDA    PBout(7) 		// SDA	 
#define MPU_READ_SDA   PBin(7) 	  	// 输入SDA

// SDA方向
#define MPU_SDA_IN()  {GPIOB->CRL&=0X0FFFFFFF;GPIOB->CRL|=(u32)8<<28;}
#define MPU_SDA_OUT() {GPIOB->CRL&=0X0FFFFFFF;GPIOB->CRL|=(u32)3<<28;}

// MPU6050寄存器
#define MPU_SELF_TESTX_REG		0X0D	//自检寄存器X
#define MPU_SELF_TESTY_REG		0X0E	//自检寄存器Y
#define MPU_SELF_TESTZ_REG		0X0F	//自检寄存器Z
#define MPU_SELF_TESTA_REG		0X10	//自检寄存器A
#define MPU_SAMPLE_RATE_REG		0X19	//采样频率分频器
#define MPU_CFG_REG				    0X1A	//配置寄存器
#define MPU_GYRO_CFG_REG		  0X1B	//陀螺仪配置寄存器
#define MPU_ACCEL_CFG_REG		  0X1C	//加速度计配置寄存器
#define MPU_MOTION_DET_REG		0X1F	//运动检测阀值设置寄存器
#define MPU_FIFO_EN_REG			  0X23	//FIFO使能寄存器
#define MPU_I2CMST_CTRL_REG		0X24	//IIC主机控制寄存器
#define MPU_I2CSLV0_ADDR_REG	0X25	//IIC从机0器件地址寄存器
#define MPU_I2CSLV0_REG		  	0X26	//IIC从机0数据地址寄存器
#define MPU_I2CSLV0_CTRL_REG	0X27	//IIC从机0控制寄存器
#define MPU_I2CSLV1_ADDR_REG	0X28	//IIC从机1器件地址寄存器
#define MPU_I2CSLV1_REG		  	0X29	//IIC从机1数据地址寄存器
#define MPU_I2CSLV1_CTRL_REG	0X2A	//IIC从机1控制寄存器
#define MPU_I2CSLV2_ADDR_REG	0X2B	//IIC从机2器件地址寄存器
#define MPU_I2CSLV2_REG			  0X2C	//IIC从机2数据地址寄存器
#define MPU_I2CSLV2_CTRL_REG	0X2D	//IIC从机2控制寄存器
#define MPU_I2CSLV3_ADDR_REG	0X2E	//IIC从机3器件地址寄存器
#define MPU_I2CSLV3_REG		  	0X2F	//IIC从机3数据地址寄存器
#define MPU_I2CSLV3_CTRL_REG	0X30	//IIC从机3控制寄存器
#define MPU_I2CSLV4_ADDR_REG	0X31	//IIC从机4器件地址寄存器
#define MPU_I2CSLV4_REG		  	0X32	//IIC从机4数据地址寄存器
#define MPU_I2CSLV4_DO_REG		0X33	//IIC从机4写数据寄存器
#define MPU_I2CSLV4_CTRL_REG	0X34	//IIC从机4控制寄存器
#define MPU_I2CSLV4_DI_REG		0X35	//IIC从机4读数据寄存器

#define MPU_I2CMST_STA_REG		0X36	//IIC主机状态寄存器
#define MPU_INTBP_CFG_REG	  	0X37	//中断/旁路设置寄存器
#define MPU_INT_EN_REG			  0X38	//中断使能寄存器
#define MPU_INT_STA_REG	  		0X3A	//中断状态寄存器

#define MPU_ACCEL_XOUTH_REG		0X3B	//加速度值,X轴高8位寄存器
#define MPU_ACCEL_XOUTL_REG		0X3C	//加速度值,X轴低8位寄存器
#define MPU_ACCEL_YOUTH_REG		0X3D	//加速度值,Y轴高8位寄存器
#define MPU_ACCEL_YOUTL_REG		0X3E	//加速度值,Y轴低8位寄存器
#define MPU_ACCEL_ZOUTH_REG		0X3F	//加速度值,Z轴高8位寄存器
#define MPU_ACCEL_ZOUTL_REG		0X40	//加速度值,Z轴低8位寄存器

#define MPU_TEMP_OUTH_REG	  	0X41	//温度值高八位寄存器
#define MPU_TEMP_OUTL_REG	  	0X42	//温度值低8位寄存器

#define MPU_GYRO_XOUTH_REG		0X43	//陀螺仪值,X轴高8位寄存器
#define MPU_GYRO_XOUTL_REG		0X44	//陀螺仪值,X轴低8位寄存器
#define MPU_GYRO_YOUTH_REG		0X45	//陀螺仪值,Y轴高8位寄存器
#define MPU_GYRO_YOUTL_REG		0X46	//陀螺仪值,Y轴低8位寄存器
#define MPU_GYRO_ZOUTH_REG		0X47	//陀螺仪值,Z轴高8位寄存器
#define MPU_GYRO_ZOUTL_REG		0X48	//陀螺仪值,Z轴低8位寄存器

#define MPU_I2CSLV0_DO_REG		0X63	//IIC从机0数据寄存器
#define MPU_I2CSLV1_DO_REG		0X64	//IIC从机1数据寄存器
#define MPU_I2CSLV2_DO_REG		0X65	//IIC从机2数据寄存器
#define MPU_I2CSLV3_DO_REG		0X66	//IIC从机3数据寄存器

#define MPU_I2CMST_DELAY_REG	0X67	//IIC主机延时管理寄存器
#define MPU_SIGPATH_RST_REG		0X68	//信号通道复位寄存器
#define MPU_MDETECT_CTRL_REG	0X69	//运动检测控制寄存器
#define MPU_USER_CTRL_REG	  	0X6A	//用户控制寄存器
#define MPU_PWR_MGMT1_REG	  	0X6B	//电源管理寄存器1
#define MPU_PWR_MGMT2_REG		  0X6C	//电源管理寄存器2 
#define MPU_FIFO_CNTH_REG	  	0X72	//FIFO计数寄存器高八位
#define MPU_FIFO_CNTL_REG		  0X73	//FIFO计数寄存器低八位
#define MPU_FIFO_RW_REG		  	0X74	//FIFO读写寄存器
#define MPU_DEVICE_ID_REG	  	0X75	//器件ID寄存器

void Drv_Mpu6050_Gpio_Init (void);   // 初始化MPU6050引脚
void MPU_IIC_Delay (void);   // MPU6050 IIC延时
void MPU_IIC_Start (void);   // MPU6050 IIC发送起始信号
void MPU_IIC_Stop (void);   // MPU6050 IIC发送终止信号
u8 MPU_IIC_Wait_Ack (void);   // MPU6050 IIC等待应答信号
void MPU_IIC_Ack (void);   // MPU6050 IIC发送应答信号
void MPU_IIC_NAck (void);   // MPU6050 IIC发送非应答信号
void MPU_IIC_Send_Byte (u8 txd);   // MPU6050 IIC发送一个字节
u8 MPU_IIC_Read_Byte (u8 ack);   // MPU6050 IIC读取一个字节
u8 MPU_Write_Len (u8 addr,u8 reg,u8 len,u8 *buf);   // IIC在固定地址开始连续写入数据
u8 MPU_Read_Len (u8 addr,u8 reg,u8 len,u8 *buf);   // IIC在固定地址开始连续读取数据
u8 MPU_Write_Byte(u8 reg,u8 data);   // IIC在固定地址写入一字节数据
u8 MPU_Read_Byte (u8 reg);   // IIC在固定地址读取一字节数据
u8 Drv_Mpu6050_Set_Lpf (u16 lpf);   // 设置MPU6050的数字低通滤波器
u8 Drv_Mpu6050_Set_Rate (u16 rate);   // 设置MPU6050的采样率(假定Fs=1KHz)

#endif

3.2 MPU6050初始化程序

  MPU6050初始化程序如下

/*
 *==============================================================================
 *函数名称:Drv_Mpu6050_Gpio_Init
 *函数功能:初始化MPU6050
 *输入参数:无
 *返回值:0:初始化成功;1:初始化失败;
 *备  注:这里不再给AD0另外接一个引脚,也就是从机地址固定为0X68
 *==============================================================================
*/
u8 Med_Mpu6050_Init (void)
{
	u8 mpu6050Id = 0;   // 存储读取的ID(从设备地址)
	
	Drv_Mpu6050_Gpio_Init();   // 初始化GPIO
	
	MPU_Write_Byte(MPU_PWR_MGMT1_REG,0X80);   // 复位MPU6050
	delay_ms(100);
	MPU_Write_Byte(MPU_PWR_MGMT1_REG,0X00);   // 唤醒MPU6050
	MPU_Write_Byte(MPU_GYRO_CFG_REG,3 << 3);   // 设置陀螺仪满量程范围0:±250°/s;1:±500°/s;2:±1000°/s;3:±2000°/s
	MPU_Write_Byte(MPU_ACCEL_CFG_REG,0 << 3);   // 设置加速度传感器满量程范围0:±2g;1:±4g;2:±8g;3:±16g
	Drv_Mpu6050_Set_Rate(50);   // 设置采样率50Hz
	MPU_Write_Byte(MPU_INT_EN_REG,0X00);   // 关闭所有中断
	MPU_Write_Byte(MPU_USER_CTRL_REG,0X00);   // I2C主模式关闭
	MPU_Write_Byte(MPU_FIFO_EN_REG,0X00);   // 关闭FIFO
	MPU_Write_Byte(MPU_INTBP_CFG_REG,0X80);   // INT引脚低电平有效
	
	mpu6050Id = MPU_Read_Byte(MPU_DEVICE_ID_REG);
	
	if(mpu6050Id == 0X68)   // 器件ID正确
	{
		MPU_Write_Byte(MPU_PWR_MGMT1_REG,0X01);   // 设置CLKSEL,PLL X轴为参考
		MPU_Write_Byte(MPU_PWR_MGMT2_REG,0X00);   // 加速度与陀螺仪都工作
		Drv_Mpu6050_Set_Rate(50);   // 设置采样率为50Hz
 	}
	else
	{
		return 1;
	}
	return 0;
}

  上面的程序是MPU6050初始化函数设计,我们在进行函数设计时加入了一个返回值,如果我们获取到的MPU6050的从机地址是正确的,就返回0,说明MPU6050连接正常,否则说明MPU6050异常。当MPU6050异常时,我们最好设置一个超时检测,防止我们在初始化的时候一直卡在这里。具体的初始化程序设计如下

	u8 watCunt = 0;   // 超时间检测计数变量

	//初始化MPU6050
	printf ("MPU6050 Init");
	while (Med_Mpu6050_Init())
	{
		printf (".");
		delay_ms(200);
		watCunt = watCunt + 1;
		
		// 超时跳出
		if (watCunt >= 150)
		{
			printf ("\r\n");
			printf ("DMP Error!\r\n");
			break;
		}
	}
	printf ("\r\n");
	printf ("MPU6050 Init OK!\r\n");

3.3 DMP相关程序

  DMP相关程序我们直接使用大家常见的例程中的一些文件,不再进行介绍。

DMP相关程序

  在初始化DMP时我们也设置一个超时检测,具体程序设计如下

	u8 watCunt = 0;   // 超时间检测计数变量
	
	// 初始化DMP
	printf ("DMP Init");
	while(mpu_dmp_init())
 	{
		printf (".");
		delay_ms(200);
		watCunt = watCunt + 1;
		
		// 超时跳出
		if (watCunt >= 150)
		{
			printf ("\r\n");
			printf ("MPU6050 Connect Error!\r\n");
			break;
		}
	}
	printf ("\r\n");
	printf ("DMP Init OK!\r\n");

3.4 获取三轴角度信息

  最后我们在main函数中获取三轴角度,串口打印输出

int main(void)
{
	float pitch,roll,yaw;   // 欧拉角
	
	Med_Mcu_Iint();   // 系统初始化
	
	while(1)
  {
		if(mpu_dmp_get_data(&pitch,&roll,&yaw) == 0)
		{ 
			printf("Pitch:%.1f Roll:%.1f Yaw:%.1f\r\n",pitch,roll,yaw);
		}
		delay_ms(100);
	}
}

四、实现效果

  下面我们来看一下具体的效果

实现效果

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/125187.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

到蒙古包了,这边天气-9度 很冷

【点我-这里送书】 本人详解 作者&#xff1a;王文峰&#xff0c;参加过 CSDN 2020年度博客之星&#xff0c;《Java王大师王天师》 公众号&#xff1a;JAVA开发王大师&#xff0c;专注于天道酬勤的 Java 开发问题中国国学、传统文化和代码爱好者的程序人生&#xff0c;期待你的…

lua脚本实现redis分布式锁(脚本解析)

文章目录 lua介绍lua基本语法redis执行lua脚本 - EVAL指令使用lua保证删除原子性 lua介绍 Lua 是一种轻量小巧的脚本语言&#xff0c;用标准C语言编写并以源代码形式开放&#xff0c; 其设计目的是为了嵌入应用程序中&#xff0c;从而为应用程序提供灵活的扩展和定制功能。 设…

(动手学习深度学习)第13章 计算机视觉---图像增广与微调

13.1 图像增广 总结 数据增广通过变形数据来获取多样性从而使得模型泛化性能更好常见图片增广包裹翻转、切割、变色。 图像增广代码实现

【开源三方库】Easyui:基于OpenAtom OpenHarmony ArkUI深度定制的组件框架

万冬阳 公司&#xff1a;中国科学院软件所 小组&#xff1a;知识体系工作组 简介 Easyui是一套基于ArkTS语言开发的轻量、可靠的移动端组件库&#xff0c;它是对OpenAtom OpenHarmony&#xff08;以下简称“OpenHarmony”&#xff09; ArkUI进行深度定制的组件框架。Easyui可扩…

C/C++特殊求和 2021年6月电子学会青少年软件编程(C/C++)等级考试一级真题答案解析

目录 C/C幻数求和 一、题目要求 1、编程实现 2、输入输出 二、算法分析 三、程序编写 四、程序说明 五、运行结果 六、考点分析 C/C幻数求和 2021年6月 C/C编程等级考试一级编程题 一、题目要求 1、编程实现 如果一个数能够被7整除或者十进制表示中含有数字7&…

NVM安装使用

文章目录 简要说明下载nvm安装nvm使用说明使用nvm下载各个版本的node.js查看已经下载到本地的node有哪些切换到对应的node版本后 简要说明 当我们在项目开发时&#xff0c;我们接手别人的项目、是当时开发的项目使用的node版、找开源项目学习的时候开源项目要求的node版本。和…

【数据结构】线性表的链式存储结构

&#x1f984;个人主页:修修修也 &#x1f38f;所属专栏:数据结构 ⚙️操作环境:Visual Studio 2022 顺序存储结构的不足的解决办法 从上一节我们对顺序表的讨论中可见,线性表的顺序存储结构的特点是: 逻辑关系上相邻的两个元素在物理位置(内存)上也相邻,因此可以随机存取表中…

借PVE8.0的Debian 12系统配置一下NFS服务器

正文共&#xff1a;1234 字 16 图&#xff0c;预估阅读时间&#xff1a;2 分钟 前面我们介绍了基于Windows Server 2012 R2创建的共享NFS&#xff08;Network File System&#xff0c;网络文件系统&#xff09;存储&#xff08;Windows Server2012 R2搭建NFS服务器&#xff09;…

快速搭建开源分布式任务调度系统DolphinScheduler并远程访问

使用Docker部署开源分布式任务调度系统DolphinScheduler 文章目录 使用Docker部署开源分布式任务调度系统DolphinScheduler前言1. 安装部署DolphinScheduler1.1 启动服务 2. 登录DolphinScheduler界面3. 安装内网穿透工具4. 配置Dolphin Scheduler公网地址5. 固定DolphinSchedu…

移动医疗科技:开发互联网医院系统源码

在这个数字化时代&#xff0c;互联网医院系统成为了提供便捷、高效医疗服务的重要手段。本文将介绍利用移动医疗科技开发互联网医院系统的源码&#xff0c;为医疗行业的数字化转型提供有力支持。 智慧医疗、互联网医院这一类平台可以通过线上的形式进行部分医疗服务&#xff…

Python的版本如何查询?

要查询Python的版本&#xff0c;可以使用以下方法之一&#xff1a; 1.在命令行中使用python --version命令。这会显示安装在计算机上的Python解释器的版本号。 # Author : 小红牛 # 微信公众号&#xff1a;wdPython2.在Python脚本中使用import sys语句&#xff0c;然后打印sy…

P6入门:项目初始化1-项目详情介绍

前言 使用项目详细信息查看和编辑有关所选项目的详细信息&#xff0c;在项目创建完成后&#xff0c;初始化项目是一项非常重要的工作&#xff0c;涉及需要设置的内容包括项目名&#xff0c;ID,责任人&#xff0c;日历&#xff0c;预算&#xff0c;资金&#xff0c;分类码等等&…

视频剪辑教程:视频嵌套技巧深度解析,提升剪辑水平的捷径

在视频剪辑的世界里&#xff0c;视频嵌套是一项强大的技术&#xff0c;也是许多专业剪辑师提升剪辑水平的重要手段。通过巧妙地运用视频嵌套技巧&#xff0c;可以在视频中创造出丰富的视觉效果&#xff0c;让观众眼前一亮。简单来说&#xff0c;就是在同一个视频轨道上&#xf…

JAVA客户端使用账号密码调用influxdb2报错:{“code“:“unauthorized“,“message“:“Unauthorized“}

问题&#xff1a;JAVA客户端访问influxdb2报错 说明&#xff1a;当前influxdb版本&#xff1a;2.6.1 使用依赖&#xff1a; <dependency><groupId>org.influxdb</groupId><artifactId>influxdb-java</artifactId><version>2.10</vers…

基于公共业务提取的架构演进——外部依赖防腐篇

背景 有了前两篇的帐号权限提取和功能设置提取的架构演进后&#xff0c;有一个问题就紧接着诞生了&#xff0c;对于诸多业务方来说&#xff0c;关键数据源的迁移如何在各个产品落地&#xff1f; 要知道这些数据都很关键&#xff1a; - 对于帐号&#xff0c;获取不到帐号信息是…

第四章《全景图:机器学习路线图》笔记

4.1 通俗讲解机器学习是什么 4.1.1 究竟什么是机器学习 卡内基梅隆大学机器学习领域的著名学者汤姆米切尔曾经在 1997 年对机器学习做出过更为严谨和经典的定义: A program can be said to learn from experience E with respect to some class of tasks T and performance …

kantts底膜训练篇-----个性化模型底膜训练

我是kantts群里的老友了&#xff0c;群里有很多热心肠的人安念、马静等很多老哥&#xff0c;还有群主格真、渡航等开源作者的支持。在里面摸爬滚打了3天&#xff0c;现在才能出这个教程。 因为kantts多年没维护了&#xff0c;只有简单的运行教程&#xff0c;很多深入的&#x…

【Redis】Java连接redis进行数据访问及项目的实例应用场景

目录 一、连接 二、数据访问 1. 字符串(String) 2. 哈希(Hash) 3. 列表(List) 4. 集合(Set) 三、项目应用 1. 作用 2. 实例 一、连接 打开开发工具( IDEA ) &#xff0c;在需要连接Redis的项目中&#xff0c;找到 pom.xml 配置文件导入依赖 在pom.xml 配置文件中导入以…

【LeetCode力扣】42.接雨水(困难)

目录 1、题目介绍 2、解题 2.1、解题思路 2.2、图解说明 2.3、解题代码 1、题目介绍 原题链接&#xff1a;42. 接雨水 - 力扣&#xff08;LeetCode&#xff09; 输入&#xff1a;height [0,1,0,2,1,0,1,3,2,1,2,1] 输出&#xff1a;6 解释&#xff1a;上面是由数组 [0,1,…

【Go 编程实践】从零到一:创建、测试并发布自己的 Go 库

为什么需要开发自己的 Go 库 在编程语言中&#xff0c;包&#xff08;Package&#xff09;和库&#xff08;Library&#xff09;是代码组织和复用的重要工具。在 Go 中&#xff0c;包是代码的基本组织单位&#xff0c;每个 Go 程序都由包构成。包的作用是帮助组织代码&#xf…