江协科技stm32————10-5 硬件I2C读写MPU6050

步骤

一、配置I2C外设,对I2C2外设进行初始化(MyI2C_Init) 

  • 开启I2C外设和对应的GPIO口的时钟
  • 把I2C对应的GPIO口初始化为复用开漏模式
  • 使用结构体配置I2C
  • I2C_Cmd,使能I2C

I2C_GenerateSTART        //生产起始条件

I2C_GenerateSTOP        //生产终止条件

I2C_AcknowledgeConfig        //配置ACK应答使能

I2C_SendData        //把数据写入DR寄存器

I2C_ReceiveData        //读取DR数据

I2C_Send7bitAddress        //发送7位地址

I2C_CheckEvent        //检查标志位

二、控制外设电路,实现指定地址写的时序(WriteReg)

三、控制外设电路,实现指定地址读的实现(ReadReg)

MPU6050.c

#include "stm32f10x.h"                  // Device header
#include "MPU6050_Reg.h"

#define MPU6050_ADDRESS		0xD0

void MPU6050_waitEvent(I2C_TypeDef* I2Cx, uint32_t I2C_EVENT)
{
	uint32_t Timeout;
	Timeout = 10000;
	while(I2C_CheckEvent(I2Cx , I2C_EVENT) != SUCCESS)	
	{
		Timeout --; 	//超时退出机制
		if(Timeout == 0)
		{
			break;
		}
	}
	
}

void MPU6050_WriteReg(uint8_t RegAddress, uint8_t Data)
{

	I2C_GenerateSTART(I2C2 , ENABLE);	//起始
	MPU6050_waitEvent(I2C2 , I2C_EVENT_MASTER_MODE_SELECT);	//检测EV5
	
	
	I2C_Send7bitAddress(I2C2 , MPU6050_ADDRESS  ,I2C_Direction_Transmitter);	//发送7位地址
	MPU6050_waitEvent(I2C2 , I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED);	//检测EV6
	
	I2C_SendData(I2C2 , RegAddress);	//发送数据
	MPU6050_waitEvent(I2C2 , I2C_EVENT_MASTER_BYTE_TRANSMITTING);	//检测EV8
	
	I2C_SendData(I2C2 , Data);
	MPU6050_waitEvent(I2C2 , I2C_EVENT_MASTER_BYTE_TRANSMITTED);	//检测EV8_2

	I2C_GenerateSTOP(I2C2 , ENABLE);	//终止 

}

uint8_t MPU6050_ReadReg(uint8_t RegAddress)
{
	uint8_t Data;
	
	I2C_GenerateSTART(I2C2 , ENABLE);	//起始
	MPU6050_waitEvent(I2C2 , I2C_EVENT_MASTER_MODE_SELECT);	//检测EV5
	
	I2C_Send7bitAddress(I2C2 , MPU6050_ADDRESS  ,I2C_Direction_Transmitter);	//发送7位地址
	MPU6050_waitEvent(I2C2 , I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED);	//检测EV6
	
	I2C_SendData(I2C2 , RegAddress);	//发送数据
	MPU6050_waitEvent(I2C2 , I2C_EVENT_MASTER_BYTE_TRANSMITTED);	//检测EV8
	
	I2C_GenerateSTART(I2C2 , ENABLE);	//重复起始条件
	MPU6050_waitEvent(I2C2 , I2C_EVENT_MASTER_MODE_SELECT);	//检测EV5
	
	I2C_Send7bitAddress(I2C2 , MPU6050_ADDRESS  ,I2C_Direction_Receiver);	//接收7位地址
	MPU6050_waitEvent(I2C2 , I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED);	//检测EV6
	
	I2C_AcknowledgeConfig(I2C2 , DISABLE);	//设置ack=0
	I2C_GenerateSTOP(I2C2 ,ENABLE); //stop = 1 申请产生终止条件

	MPU6050_waitEvent(I2C2 , I2C_EVENT_MASTER_BYTE_RECEIVED);	//检测EV7
	Data = I2C_ReceiveData(I2C2);	//数值赋给Data

	I2C_AcknowledgeConfig(I2C2 , ENABLE);	//设置ack=1

	return Data;
}


void MPU6050_Init(void)
{
//	MyI2C_Init();
	
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C2 , ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB , ENABLE);
	
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD;	//复用AF:GPIO的控制权要交给硬件外设 开漏OD:I2C协议的要求
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10 | GPIO_Pin_11;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB, &GPIO_InitStructure);
	
	I2C_InitTypeDef I2C_Initstructure;
	I2C_Initstructure.I2C_Mode = I2C_Mode_I2C;
	I2C_Initstructure.I2C_ClockSpeed = 50000;
	I2C_Initstructure.I2C_DutyCycle = I2C_DutyCycle_2;
	I2C_Initstructure.I2C_Ack = I2C_Ack_Enable;
	I2C_Initstructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;
	I2C_Initstructure.I2C_OwnAddress1 = 0x00;
	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);//加速度配置寄存器
}

uint8_t MPU6050_GetID(void)
{
	return MPU6050_ReadReg(MPU6050_WHO_AM_I);
}


void MPU6050_GetData(int16_t *AccX, int16_t *AccY, int16_t *AccZ, 
					int16_t *GyroX, int16_t *GyroY, int16_t *GyroZ)	
//分别读取6个轴数据寄存器的高位和低位,拼接为16位数据再通过指针变量返回																
{
	uint8_t DataH, DataL;
	
	DataH = MPU6050_ReadReg(MPU6050_ACCEL_XOUT_H);
	DataL = MPU6050_ReadReg(MPU6050_ACCEL_XOUT_L);
	*AccX = (DataH << 8) | DataL;
	
	DataH = MPU6050_ReadReg(MPU6050_ACCEL_YOUT_H);
	DataL = MPU6050_ReadReg(MPU6050_ACCEL_YOUT_L);
	*AccY = (DataH << 8) | DataL;
	
	DataH = MPU6050_ReadReg(MPU6050_ACCEL_ZOUT_H);
	DataL = MPU6050_ReadReg(MPU6050_ACCEL_ZOUT_L);
	*AccZ = (DataH << 8) | DataL;
	
	DataH = MPU6050_ReadReg(MPU6050_GYRO_XOUT_H);
	DataL = MPU6050_ReadReg(MPU6050_GYRO_XOUT_L);
	*GyroX = (DataH << 8) | DataL;
	
	DataH = MPU6050_ReadReg(MPU6050_GYRO_YOUT_H);
	DataL = MPU6050_ReadReg(MPU6050_GYRO_YOUT_L);
	*GyroY = (DataH << 8) | DataL;
	
	DataH = MPU6050_ReadReg(MPU6050_GYRO_ZOUT_H);
	DataL = MPU6050_ReadReg(MPU6050_GYRO_ZOUT_L);
	*GyroZ = (DataH << 8) | DataL;
}

 

  • 8
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
江科大自化协软件模拟I2C读写MPU6050的过程可以分为两个主要步骤。首先,需要学习I2C协议规则,并使用软件模拟的方式实现协议。这包括了了解I2C协议的引脚配置、时序要求等相关内容。其次,需要学习STM32I2C外设,并使用硬件实现协议。这一步骤可以进一步学习STM32I2C外设和协议的硬件实现方式。\[1\]\[2\] 在软件模拟I2C读写MPU6050的过程中,可以采用多层的模块架构。最底层是I2C协议层,主要关注引脚配置、时序要求等与协议相关的内容。其次是MPU6050的驱动层,主要关注如何读写寄存器、配置寄存器以及读取数据等与驱动相关的内容。最后是主函数应用层,通过调用MPU6050的驱动函数获取数据,并根据需要进行进一步的功能设计。\[3\] 因此,江科大自化协软件模拟I2C读写MPU6050的过程包括学习I2C协议规则、软件模拟实现协议,以及学习STM32I2C外设并使用硬件实现协议。在软件模拟过程中,需要关注引脚配置、时序要求等与协议相关的内容。在硬件实现过程中,需要学习STM32I2C外设和协议的硬件实现方式。最后,通过多层的模块架构,实现对MPU6050读写操作,并根据需要进行进一步的功能设计。 #### 引用[.reference_title] - *1* *2* *3* [【STM32学习】——I2C通信协议&MPU6050姿态传感器&软件I2C读写MPU6050](https://blog.csdn.net/weixin_51658186/article/details/129821841)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^control_2,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值