I2C通信
- I2C(Inter IC Bus)是由Philips公司开发的一种通用数据总线
- 两根通信线:SCL(Serial Clock)、SDA(Serial Data)
- 同步,半双工
- 带数据应答
- 支持总线挂载多设备(一主多从、多主多从)
- 使用同步时序可以极大降低单片机对硬件电路的依赖
I2C硬件电路
- 所有I2C设备的SCL连在一起,SDA连在一起
- 设备的SCL和SDA均要配置成开漏输出模式
- SCL和SDA各添加一个上拉电阻,阻值一般为4.7KΩ左右
任何时候都是主机完全掌握SCL,在空闲状态下,主机可以主动发起对SDA的控制。只有从机在发送数据和从机应答的时候,主机才会把SDA的控制权交给从机。
把SDA和SCL想象成一根杆子,并且让所有人都不能往上拉,只能往下拉或松手,之后我们外置一根弹簧到这根杆子上,输出低电平就往下拽,输出高电平就放手,这是一个弱上拉的高电平,完全不影响数据传输。
I2C时序基本单元
起始条件:SCL高电平期间,SDA从高电平切换到低电平
终止条件:SCL高电平期间,SDA从低电平切换到高电平
发送一个字节:SCL低电平期间,主机将数据位依次放到SDA线上(高位先行),然后释放SCL,从机将在SCL高电平期间读取数据位,所以SCL高电平期间SDA不允许有数据变化,依次循环上述过程8次,即可发送一个字节。低电平主机放数据,高电平从机读数据。
起始条件后,第一个字节必须为主机发送。如果主机想发送0,则让拉低SDA到低电平。如果发送1,则放手让SDA回弹到高低电平。在SCL低电平期间,允许改变SDA电平,在SCL高电平期间,不允许改变SDA的电平,并且是从机读取SDA的时候,从机必须尽快读取SDA,一般在上升沿时刻就读取完数据。主机放手SCL一段时间后,继续拉低SCL传输下一位,主机也需要在SCL下降沿之后尽快把数据放在SDA上,但主机掌握着时钟的主导权,所以在低电平的任意时刻把数据放在SDA上即可。
接收一个字节:SCL低电平期间,从机将数据位依次放到SDA线上(高位先行),然后释放SCL,主机将在SCL高电平期间读取数据位,所以SCL高电平期间SDA不允许有数据变化,依次循环上述过程8次,即可接收一个字节(主机在接收之前,需要释放SDA)。低电平从机放数据,高电平主机读数据。
实线是主机控制部分,虚线是从机控制部分。
发送应答信号:主机在接收完一个字节之后,在下一个时钟发送一位数据,数据0表示应答,数据1表示非应答
接收应答信号:主机在发送完一个字节之后,在下一个时钟接收一位数据,判断从机是否应答,数据0表示应答,数据1表示非应答(主机在接收之前,需要释放SDA)
主机释放SDA的时候,从机就应该把SDA拉下来,在SCL高电平期间,主机读取应答位,应答位为0,说明从机确实收到了。
发送应答位目的是告诉从机,是不是要继续发,如果从机发送一个数据后,得到主机的应答,说明要继续发送,如果主机没有应答则从机停止发送,交出SDA的控制权。
I2C时序
指定地址写;对于指定设备(Slave Address),在指定地址(Reg Address)下,写入指定数据(Data)。
这个数据帧的目的式在指定从机地址1101000的设备,在其内部0x19地址的寄存器下,写入0xAA数据。
在起始条件后,紧跟着的时序必须是发送一个字节的时序,字节的内容必须是从机地址+读写位。从机地址为7位,读写位为1位,加起来正好8位。发送从机地址就是确定通信的对象,发送读写位就是确定接下来是写入还是读出。紧跟着的单元式接受从机的应答位。
指定地址读;对于指定设备(Slave Address),在指定地址(Reg Address)下,读取从机数据(Data)。
在Sr前面就是指定地址写,后面就是当前地址读。
先起始写入地址,停止,因为写入的地址会存在地址指针里,所以这个地址不会因为时序停止而消失,我们可以再起始,读当前位置,停止。
I2C外设
- STM32内部集成了硬件I2C收发电路,可以由硬件自动执行时钟生成、起始终止条件生成、应答位收发、数据收发等功能,减轻CPU的负担
- 支持多主机模型
- 支持7位/10位地址模式
- 支持不同的通讯速度,标准速度(高达100 kHz),快速(高达400 kHz)
- 支持DMA
- 兼容SMBus协议
- STM32F103C8T6 硬件I2C资源:I2C1、I2C2
I2C框图
SDA的核心部分就是数据寄存器和数据移位寄存器。
发送流程:当需要发送数据时,可以把一个字节数据写到数据寄存器DR中,当移位寄存器没有数据移位时,数据寄存器的值就会进一步转到移位寄存器里。在移位的过程中,可以把下一个数据放在数据寄存器等着,一旦前一个数据移位完成,下个数据就可以无缝衔接,继续发送。当数据寄存器转到移位寄存器时,就会置状态寄存器TXE位为1(表示发送寄存器为空)。
接收流程:输入的数据移一位一位传输到移位寄存器,当一个字节数据接收完毕,数据整体从移位寄存器转到数据寄存器,同时置标志位RXNE(表示接收寄存器为非空)。
I2C外设基本结构
首先移位寄存器和数据寄存器DR的配合是通信的核心部分,因为I2C是高位先行,所以移位寄存器是向左移位,在发送的时候,最高位先移出去,然后是次高位等等。一个SCL时钟移位一次,移位8次就可以把一个字节,由高位到低位一次放到SDA线上。
发送的时候:数据先写入数据寄存器,如果移位寄存器没有数据,再转到移位寄存器进行发送。
接收的时候:数据通过GPIO口从右边依次移进来,最终移8次,一个字节就接收完成了。
使用硬件I2C的时候,需要把GPIO口配置成复用开漏输出模式(复用:GPIO状态是交由片上外设来控制)。
主机发送
7位主发送:起始条件后的一个字节是寻址。
10位主发送:起始条件后的两个字节是寻址。
EVX:组合多个标志位的大标志位。
7位主发送流程:当检测(EV5)起始条件已发送后,就可以发送一个字节的从机地址,从机地址需要写到数据寄存器DR中,写入DR之后,硬件电路会自动把这个字节转到移位寄存器里,再把这个字节发送到I2C总线上,之后硬件会自动接收应答并判断,如果没有应答就会置失败的标志位(这个标志位可以利用中断来提醒)。当寻址完成后会发生EV6事件,接下来发生EV8_1事件,一旦写入DR后,DR会立刻转移到移位寄存器进行发送,这时就是EV8事件,这时就是移位寄存器正在发送数据的状态。EV8结束时,数据2写入到数据寄存器等着,接收应答位之后,数据2就转入移位寄存器进行发送。一旦检测到EV8事件,就可以写入下一个数据。最后,当我们想发的数据写完之后,就没有新数据写入数据寄存器中。当移位寄存器当前数据移位完成时,此时是移位寄存器空,数据寄存器也空的状态,这就是EV8_2事件。
主机接收
首先写入控制寄存器的START位,产生起始条件,等待EV5事件,之后是寻址,接收应答(A),结束后产生EV6事件,数据1代表正在通过移位寄存器进行输入。当这个时序单元结束后,说明移位寄存器已经成功移入一个字节的数据1,移入的一个字节整体转移到数据寄存器。同时置RxNE标志位,表示数据寄存器非空,也就是收到一个字节的数据,这就是EV7事件。当不需要接收时,需要在最后一个时序单元发生时,提前把应答位控制的寄存器Ack置0,并且设置终止条件请求,这就是EV7_1事件。在时序完成之后,由于之前设置了Ack=0,所以这里会给出非应答。由于设置了STOP位,最后产生终止条件。
MPU6050
- MPU6050是一个6轴姿态传感器,可以测量芯片自身X、Y、Z轴的加速度、角速度参数,通过数据融合,可进一步得到姿态角,常应用于平衡车、飞行器等需要检测自身姿态的场景
- 3轴加速度计(Accelerometer):测量X、Y、Z轴的加速度
- 3轴陀螺仪传感器(Gyroscope):测量X、Y、Z轴的角速度
- 加速度计具有静态稳定性,不具有动态稳定性
MUP6050参数
- 16位ADC采集传感器的模拟信号,量化范围:-32768~32767
- 加速度计满量程选择:±2、±4、±8、±16(g)(如果测量的物体运动非常剧烈,可以把满量程选择大一些。反之同理)
- 陀螺仪满量程选择: ±250、±500、±1000、±2000(°/sec)(满量程选得越小,测量分辨率越高)
- 可配置的数字低通滤波器
- 可配置的时钟源
- 可配置的采样分频
- I2C从机地址:1101000(AD0=0) 1101001(AD0=1)
如果认为0x68是从地址,在发送第一个字节需要把0x68左移一位,再按位或上读写位。
把0x68左移一位后的数据当作从机地址,左移一位是0xD0。在实际发送第一个字节的时候,如果需要写就直接把0xD0当作第一个字节,如果需要读则把0xD0或0x01,即0xD1当作第一个字节。这种操作方式是读写位融入到从机地址中。0xD0是写地址,0xD1是读地址。
MPU6050硬件电路
XDA和XCL通常用于外接磁力计或气压计,当接上之后,MPU6050主机接口可以直接访问扩展芯片的数据。
MPU6050框图
Self test(自测模块):先使能Self test测得X Accel数据,再失能Self test测X Acce数据,测出来的数据两个进行相减,再根据手册里面数据进行对比,看是否在这个区间内,在能正常使用。
Interrupt Status Register(中断状态寄存器):控制内部事件到中断引脚的输出。
FIFO(先入先出寄存器):对数据流进行缓存。
Config Registers(配置寄存器):对内部的各个电路进行配置。
Sensor Registers(传感器寄存器或数据寄存器):存储各个传感器的数据。
Serial Interface Bypass Mux(接口旁路选择器):如果拨到上面,辅助的I2C引脚和正常的I2C引脚接在一起,两路总线结合在一起,STM32能控制所有设备。如果拨到下面辅助的I2C引脚就由MPU6050控制,这时STM32是MPU6050大哥,MPU6050是扩展芯片的大哥。
代码部分
I2C软件配置代码
#include "Bsp_I2C.h"
#include "Delay.h"
/* SCL写功能 */
void Bsp_I2C_W_SCL(uint8_t BitValue)
{
GPIO_WriteBit(GPIOB, GPIO_Pin_10, (BitAction)BitValue);
Delay_us(10);
}
/* SDA写功能 */
void Bsp_I2C_W_SDA(uint8_t BitValue)
{
GPIO_WriteBit(GPIOB, GPIO_Pin_11, (BitAction)BitValue);
Delay_us(10);
}
/* SDA读功能 */
uint8_t Bsp_I2C_R_SDA(void)
{
uint8_t BitValue;
BitValue = GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_11);
Delay_us(10);
return BitValue;
}
/* I2C初始化*/
void Bsp_I2C_Init(void)
{
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10 | GPIO_Pin_11;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOB, &GPIO_InitStructure);
}
/* I2C起始条件和重复起始条件(先让SDA置低电平,之后让SCL置低电平。为什么要先让他们置高电平,是因为防止之后起始的时候SCL先置低电平) */
void Bsp_I2C_Strat(void)
{
Bsp_I2C_W_SDA(1);
Bsp_I2C_W_SCL(1);
Bsp_I2C_W_SDA(0);
Bsp_I2C_W_SCL(0);
}
/* I2C终止条件 (为什么要先让SDA置低电平,因为SDA不一定是低电平))*/
void Bsp_I2C_Stop(void)
{
Bsp_I2C_W_SDA(0);
Bsp_I2C_W_SCL(1);
Bsp_I2C_W_SDA(1);
}
/* I2C发送数据 */
void Bsp_I2C_SendByte(uint8_t Byte)
{
for (uint8_t i = 0; i < 8; i++)
{
Bsp_I2C_W_SDA(Byte & (0x80 >> i)); // 只与最高位,最终结果为0x80、0x40...,为什么最后写进去的是0或1呢,因为这个强整型(BitAction)
Bsp_I2C_W_SCL(1);
Bsp_I2C_W_SCL(0);
}
}
/* I2C接收数据 */
uint8_t Bsp_I2C_ReceiveByte(void)
{
uint8_t Byte = 0x00;
Bsp_I2C_W_SDA(1);
for (uint8_t i = 0; i < 8; i++)
{
Bsp_I2C_W_SCL(1);
if (Bsp_I2C_R_SDA() == 1)
{
Byte |= (0x80 >> i);
}
Bsp_I2C_W_SCL(0);
}
return Byte;
}
/* I2C发送应答 */
void Bsp_I2C_WaitSendAck(uint8_t AckBit)
{
Bsp_I2C_W_SDA(AckBit);
Bsp_I2C_W_SCL(1);
Bsp_I2C_W_SCL(0);
}
/* I2C接收应答 */
uint8_t Bsp_I2C_WaitReceiveAck(void)
{
uint8_t AckBit;
Bsp_I2C_W_SDA(1);
Bsp_I2C_W_SCL(1);
AckBit = Bsp_I2C_R_SDA();
Bsp_I2C_W_SCL(0);
return AckBit;
}
MPU6050寄存器配置代码
#ifndef __BSP_MPU6050_REG_H
#define __BSP_MPU6050_REG_H
/* 这些起始把MPU6050寄存器一些关键寄存器地址封装 让我们明了有哪些寄存器 */
#define MPU6050_SMPLRT_DIV 0x19
#define MPU6050_CONFIG 0x1A
#define MPU6050_GYRO_CONFIG 0x1B
#define MPU6050_ACCEL_CONFIG 0x1C
#define MPU6050_ACCEL_XOUT_H 0x3B
#define MPU6050_ACCEL_XOUT_L 0x3C
#define MPU6050_ACCEL_YOUT_H 0x3D
#define MPU6050_ACCEL_YOUT_L 0x3E
#define MPU6050_ACCEL_ZOUT_H 0x3F
#define MPU6050_ACCEL_ZOUT_L 0x40
#define MPU6050_TEMP_OUT_H 0x41
#define MPU6050_TEMP_OUT_L 0x42
#define MPU6050_GYRO_XOUT_H 0x43
#define MPU6050_GYRO_XOUT_L 0x44
#define MPU6050_GYRO_YOUT_H 0x45
#define MPU6050_GYRO_YOUT_L 0x46
#define MPU6050_GYRO_ZOUT_H 0x47
#define MPU6050_GYRO_ZOUT_L 0x48
#define MPU6050_PWR_MGMT_1 0x6B
#define MPU6050_PWR_MGMT_2 0x6C
#define MPU6050_WHO_AM_I 0x75
#endif
MPU6050配置代码
Bsp_MPU6050.h代码
#ifndef __BSP_MPU6050_H
#define __BSP_MPU6050_H
#include "stm32f10x.h"
#include "Bsp_MPU6050_Reg.h"
#include "Bsp_I2C.h"
void MPU6050_WriteReg(uint8_t RegAddress, uint8_t Data);
uint8_t MPU6050_ReadReg(uint8_t RegAddress);
void MPU6050_GetData(void);
uint8_t MPU6050_IDGet(void);
void MPU6050_Init(void);
typedef struct
{
int16_t AccX;
int16_t AccY;
int16_t AccZ;
int16_t GyroX;
int16_t GyroY;
int16_t GyroZ;
}MPU6050_Data;
#endif
Bsp_MPU6050.c代码
#include "Bsp_MPU6050.h"
#define MPU6050_Address 0xD0 // 定义MPU6050地址
MPU6050_Data Data;
/* MPU6050的I2C写数据 */
void MPU6050_WriteReg(uint8_t RegAddress, uint8_t Data)
{
Bsp_I2C_Strat();
Bsp_I2C_SendByte(MPU6050_Address); // 发送从机地址
Bsp_I2C_WaitReceiveAck(); // 等待接收应答
Bsp_I2C_SendByte(RegAddress); // 发送寄存器地址
Bsp_I2C_WaitReceiveAck();
Bsp_I2C_SendByte(Data); // 发送数据(这里还可以写for循环写入多个数据)
Bsp_I2C_WaitReceiveAck();
Bsp_I2C_Stop();
}
/* MPU6050的I2C读数据 */
uint8_t MPU6050_ReadReg(uint8_t RegAddress)
{
uint8_t Data;
Bsp_I2C_Strat();
Bsp_I2C_SendByte(MPU6050_Address);
Bsp_I2C_WaitReceiveAck();
Bsp_I2C_SendByte(RegAddress);
Bsp_I2C_WaitReceiveAck();
Bsp_I2C_Strat();
Bsp_I2C_SendByte(MPU6050_Address | 0x01); // 改为读
Bsp_I2C_WaitReceiveAck();
Data = Bsp_I2C_ReceiveByte(); // 这里可以写for循环读多个数据,但完成前的所有从机给 应答 都给0,最后一个从机给 非应答 给1
Bsp_I2C_WaitSendAck(1); // 0:给从机应答 1:不给从机应答
Bsp_I2C_Stop();
return Data;
}
/* MPU6050初始化, 配置完成后,陀螺仪内部就在不断进行数据转换,输出的数据就在数据寄存器里,如果想获取数据,读取相应的寄存器就即可 */
void MPU6050_Init(void)
{
Bsp_I2C_Init();
MPU6050_WriteReg(MPU6050_PWR_MGMT_1, 0x01); // 解除睡眠,选择陀螺仪时钟
MPU6050_WriteReg(MPU6050_PWR_MGMT_2, 0x00); // 6个轴均不待机
MPU6050_WriteReg(MPU6050_SMPLRT_DIV, 0x09); // 10分频
MPU6050_WriteReg(MPU6050_CONFIG, 0x06); // 滤波参数选择(这里选择最大)
MPU6050_WriteReg(MPU6050_GYRO_CONFIG, 0x18); // 陀螺仪量程选择(这里选择最大量程)
MPU6050_WriteReg(MPU6050_ACCEL_CONFIG, 0x18); // 加速度量程选择(这里选择最大量程)
}
/* 获取MPU6050数据 */
void MPU6050_GetData(void)
{
uint8_t DataH, DataL; // 定义数据高位和低位
DataH = MPU6050_ReadReg(MPU6050_ACCEL_XOUT_H); // 读MPU6050_ACCEL_XOUT_H寄存器的高八位
DataL = MPU6050_ReadReg(MPU6050_ACCEL_XOUT_L); // 读MPU6050_ACCEL_XOUT_L寄存器的低八位
Data.AccX = (DataH << 8) | DataL; // 高八位左移八位再与上第八位,就可以得到16位数据
DataH = MPU6050_ReadReg(MPU6050_ACCEL_YOUT_H);
DataL = MPU6050_ReadReg(MPU6050_ACCEL_YOUT_L);
Data.AccY = (DataH << 8) | DataL;
DataH = MPU6050_ReadReg(MPU6050_ACCEL_ZOUT_H);
DataL = MPU6050_ReadReg(MPU6050_ACCEL_ZOUT_L);
Data.AccZ = (DataH << 8) | DataL;
DataH = MPU6050_ReadReg(MPU6050_GYRO_XOUT_H);
DataL = MPU6050_ReadReg(MPU6050_GYRO_XOUT_L);
Data.GyroX = (DataH << 8) | DataL;
DataH = MPU6050_ReadReg(MPU6050_GYRO_YOUT_H);
DataL = MPU6050_ReadReg(MPU6050_GYRO_YOUT_L);
Data.GyroY = (DataH << 8) | DataL;
DataH = MPU6050_ReadReg(MPU6050_GYRO_ZOUT_H);
DataL = MPU6050_ReadReg(MPU6050_GYRO_ZOUT_L);
Data.GyroZ = (DataH << 8) | DataL;
}
/* 获取ID */
uint8_t MPU6050_IDGet(void)
{
return MPU6050_ReadReg(MPU6050_WHO_AM_I);
}
I2C硬件配置代码
#include "Bsp_MPU6050.h"
#define MPU6050_Address 0xD0 // 定义MPU6050地址
MPU6050_Data Data;
/* 对I2C_CheakEvent进行封装 加入了超时退出机制 */
void MPU6050_WaitCheckEvent(I2C_TypeDef* I2Cx, uint32_t I2C_EVENT)
{
uint32_t TimeOut;
while (I2C_CheckEvent(I2Cx, I2C_EVENT) != SUCCESS)
{
TimeOut --;
if (TimeOut == 0)
{
break;
}
}
}
/* MPU6050的I2C写数据 */
void MPU6050_WriteReg(uint8_t RegAddress, uint8_t Data)
{
I2C_GenerateSTART(I2C2, ENABLE); // 起始
MPU6050_WaitCheckEvent(I2C2, I2C_EVENT_MASTER_MODE_SELECT); // 等待EV5事件
I2C_Send7bitAddress(I2C2, MPU6050_Address, I2C_Direction_Transmitter); // 发送地址
MPU6050_WaitCheckEvent(I2C2, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED); // 等待EV6事件
I2C_SendData(I2C2, RegAddress);
MPU6050_WaitCheckEvent(I2C2, I2C_EVENT_MASTER_BYTE_TRANSMITTING); // 等待EV8事件
I2C_SendData(I2C2, Data);
MPU6050_WaitCheckEvent(I2C2, I2C_EVENT_MASTER_BYTE_TRANSMITTED); // 等待EV8_2事件,因为到这里就结束了
I2C_GenerateSTOP(I2C2, ENABLE); // 终止
}
/* 在程序中,大量的死循环等待非常危险,一旦一个环节没有产生,则会产生死循环。所以对这种情况可以加入超时退出机制 */
/* MPU6050的I2C读数据 */
uint8_t MPU6050_ReadReg(uint8_t RegAddress)
{
uint8_t Data;
I2C_GenerateSTART(I2C2, ENABLE);
MPU6050_WaitCheckEvent(I2C2, I2C_EVENT_MASTER_MODE_SELECT); // 等待EV5事件
I2C_Send7bitAddress(I2C2, MPU6050_Address, I2C_Direction_Transmitter); // 从机地址:发送模式
MPU6050_WaitCheckEvent(I2C2, I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED); // 等待EV6事件
I2C_SendData(I2C2, RegAddress);
MPU6050_WaitCheckEvent(I2C2, I2C_EVENT_MASTER_BYTE_TRANSMITTED); // 等待EV8事件
// 重新启动
I2C_GenerateSTART(I2C2, ENABLE);
MPU6050_WaitCheckEvent(I2C2, I2C_EVENT_MASTER_MODE_SELECT); // 等待EV5事件
I2C_Send7bitAddress(I2C2, MPU6050_Address, I2C_Direction_Receiver); // 从机地址:接收模式
MPU6050_WaitCheckEvent(I2C2, I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED); // 等待EV6事件
I2C_AcknowledgeConfig(I2C2, DISABLE); // 在接收最后一个字节之前,需要临时把ACK置0。
I2C_GenerateSTOP(I2C2, ENABLE);
MPU6050_WaitCheckEvent(I2C2, I2C_EVENT_MASTER_BYTE_RECEIVED); // 等待EV7_1事件 因为这里只读取了一个字节,所以就要立刻把Ack置0,STOP置1
Data = I2C_ReceiveData(I2C2); // 如果需要指定多个地址,那么需要在47-51加for循环,并在最后一个字节的时候,利用if加入47和48行代码。
I2C_AcknowledgeConfig(I2C2, ENABLE);
return Data;
}
/* MPU6050初始化, 配置完成后,陀螺仪内部就在不断进行数据转换,输出的数据就在数据寄存器里,如果想获取数据,读取相应的寄存器就即可 */
void MPU6050_Init(void)
{
// Bsp_I2C_Init();
RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C2, ENABLE);
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
GPIO_InitTypeDef GPIO_InitStrcuture;
GPIO_InitStrcuture.GPIO_Mode = GPIO_Mode_AF_OD; // 引脚模式为复用开漏输出
GPIO_InitStrcuture.GPIO_Pin = GPIO_Pin_11 | GPIO_Pin_10;
GPIO_InitStrcuture.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOB, &GPIO_InitStrcuture);
I2C_InitTypeDef I2C_InitStructure;
I2C_InitStructure.I2C_Ack = I2C_Ack_Enable; // 开启应答位
I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit; // 选择7位地址还是确认10位地址,这里选择7位地址
I2C_InitStructure.I2C_ClockSpeed = 100000; // 通讯速度,这里选择标准哦通信速度100KHZ
I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2; // I2C快速模式占空比
I2C_InitStructure.I2C_Mode = I2C_Mode_I2C; // 选择I2C模式
I2C_InitStructure.I2C_OwnAddress1 = 0x00; // 设备自身地址(如果选择7位地址就要写自身的7位地址,10位同理)
I2C_Init(I2C2, &I2C_InitStructure);
I2C_Cmd(I2C2, ENABLE);
MPU6050_WriteReg(MPU6050_PWR_MGMT_1, 0x01); // 解除睡眠,选择陀螺仪时钟
MPU6050_WriteReg(MPU6050_PWR_MGMT_2, 0x00); // 6个轴均不待机
MPU6050_WriteReg(MPU6050_SMPLRT_DIV, 0x09); // 10分频
MPU6050_WriteReg(MPU6050_CONFIG, 0x06); // 滤波参数选择(这里选择最大)
MPU6050_WriteReg(MPU6050_GYRO_CONFIG, 0x18); // 陀螺仪量程选择(这里选择最大量程)
MPU6050_WriteReg(MPU6050_ACCEL_CONFIG, 0x18); // 加速度量程选择(这里选择最大量程)
}
/* 获取MPU6050数据 */
void MPU6050_GetData(void)
{
uint8_t DataH, DataL; // 定义数据高位和低位
DataH = MPU6050_ReadReg(MPU6050_ACCEL_XOUT_H); // 读MPU6050_ACCEL_XOUT_H寄存器的高八位
DataL = MPU6050_ReadReg(MPU6050_ACCEL_XOUT_L); // 读MPU6050_ACCEL_XOUT_L寄存器的低八位
Data.AccX = (DataH << 8) | DataL; // 高八位左移八位再与上第八位,就可以得到16位数据
DataH = MPU6050_ReadReg(MPU6050_ACCEL_YOUT_H);
DataL = MPU6050_ReadReg(MPU6050_ACCEL_YOUT_L);
Data.AccY = (DataH << 8) | DataL;
DataH = MPU6050_ReadReg(MPU6050_ACCEL_ZOUT_H);
DataL = MPU6050_ReadReg(MPU6050_ACCEL_ZOUT_L);
Data.AccZ = (DataH << 8) | DataL;
DataH = MPU6050_ReadReg(MPU6050_GYRO_XOUT_H);
DataL = MPU6050_ReadReg(MPU6050_GYRO_XOUT_L);
Data.GyroX = (DataH << 8) | DataL;
DataH = MPU6050_ReadReg(MPU6050_GYRO_YOUT_H);
DataL = MPU6050_ReadReg(MPU6050_GYRO_YOUT_L);
Data.GyroY = (DataH << 8) | DataL;
DataH = MPU6050_ReadReg(MPU6050_GYRO_ZOUT_H);
DataL = MPU6050_ReadReg(MPU6050_GYRO_ZOUT_L);
Data.GyroZ = (DataH << 8) | DataL;
}
/* 获取ID */
uint8_t MPU6050_IDGet(void)
{
return MPU6050_ReadReg(MPU6050_WHO_AM_I);
}