STM32硬件I2C通信外设

这篇具有很好参考价值的文章主要介绍了STM32硬件I2C通信外设。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。


前言

本文主要介绍stm32自带的I2C通信外设,对比与软件模拟I2C,硬件I2C可以自动生成时序,时序的操作更加及时规范,可以实现更加高性能的IIC通信。
本文内容与I2C软件通信有诸多类似之处,I2C软件通信可见:https://blog.csdn.net/qq_53922901/article/details/136662006?spm=1001.2014.3001.5501


I2C硬件介绍

STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机

10 位地址模式

在8位指定地址读的基础下,把第一个字节的前5位改为标志位,值为11110,表示为10位地址模式,剩下的三位加上第二个字节的八位组成十位地址位加上读写位
STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机

硬件I2C的引脚定义

STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机

I2C框图

发送数据:写入数据寄存器,数据寄存器写入移位寄存器,然后数据寄存器为空,继续写入,再通过移位寄存器一位一位的发送完毕,然后置状态寄存器TXE = 1,发送寄存器为空。
接收数据:通过移位寄存器一位一位的接收数据,接收完毕后移入数据寄存器,置RXNE = 1,接收寄存器非空。
STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机
自身地址寄存器&双地址寄存器:在STM32作为从机时,通过这两个寄存器寻址,若与设置的地址对应则使STM32作为从机使用,双地址则表示作为多个从机使用。

I2C基本使用框图:主要需要使用的部分
STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机

主机发送序列图

STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机

主机接收序列图

STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机

通过参考手册,了知道这些标志位的作用与操作
以下为控制寄存器1被需要用到的标志位
STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机

以下为状态寄存器中被需要使用的标志位
STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机
STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机

硬件I2C读写MPU6050

电路连接

STM32硬件I2C通信外设,STM32学习记录,stm32,嵌入式硬件,单片机

按照官方序列图来改写MPU6050.c:

#include "stm32f10x.h"                  // Device header


#define MPU6050_Slave 	0xd0
// 配置滤波、传感器的初始配置
#define SMPLRT_DIV			0X19
#define CONFIG					0X1A
#define GYRO_CONFIG			0X1B
#define ACCEL_CONFIG		0X1C
// 这几个连续的寄存器存储着各个轴的值
#define ACCEL_XOUT_H		0X3B
#define ACCEL_XOUT_L		0X3C
#define ACCEL_YOUT_H		0X3D
#define ACCEL_YOUT_L		0X3E
#define ACCEL_ZOUT_H		0X3F
#define ACCEL_ZOUT_L		0X40
#define TEMP_OUT_H 			0X41
#define TEMP_OUT_L			0X42
#define GYRO_XOUT_H			0X43
#define GYRO_XOUT_L			0X44
#define GYRO_YOUT_H			0X45
#define GYRO_YOUT_L			0X46
#define GYRO_ZOUT_H			0X47
#define GYRO_ZOUT_L			0X48

#define PWR_MGMT_1			0X6B
#define PWR_MGMT_2			0X6C
#define WHO_AM_I				0X75

// 封装I2C_CheckEvent,避免死循环使程序卡死
void Wait_CheckEvent(I2C_TypeDef* I2Cx, uint32_t I2C_EVENT){
	uint16_t count = 10000;
	while(I2C_CheckEvent(I2Cx,I2C_EVENT)==ERROR){
		count--;
		if(count==0)	// 错误处理
			break;
	}
}

// 向寄存器写入数据
void MPU6050_WriteReg(uint8_t RegAddr,uint8_t Data){
	// 生成起始条件
	I2C_GenerateSTART(I2C2,ENABLE);
	// 检测EV5事件,未成功则继续等待
	Wait_CheckEvent(I2C2,I2C_EVENT_MASTER_MODE_SELECT);
	// 发送第一个字节,从机地址+写
	I2C_Send7bitAddress(I2C2,MPU6050_Slave,I2C_Direction_Transmitter);
	// 检测EV6事件,未成功则继续等待
	Wait_CheckEvent(I2C2,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED);
	// 发送数据1(寄存器地址)
	I2C_SendData(I2C2,RegAddr);
	// 检测EV8事件,未成功则继续等待
	Wait_CheckEvent(I2C2,I2C_EVENT_MASTER_BYTE_TRANSMITTING);
	// 发送数据2(数据)
	I2C_SendData(I2C2,Data);
	// 检测EV8_2事件,未成功则继续等待
	Wait_CheckEvent(I2C2,I2C_EVENT_MASTER_BYTE_TRANSMITTED);
	// 产生终止条件
	I2C_GenerateSTOP(I2C2,ENABLE);
}

// 从寄存器读取数据
uint8_t MPU6050_ReadReg(uint8_t RegAddr){
	uint8_t Data;
	// 生成起始条件
	I2C_GenerateSTART(I2C2,ENABLE);
	// 检测EV5事件,未成功则继续等待
	Wait_CheckEvent(I2C2,I2C_EVENT_MASTER_MODE_SELECT);
	// 发送第一个字节,从机地址+写
	I2C_Send7bitAddress(I2C2,MPU6050_Slave,I2C_Direction_Transmitter);
	// 检测EV6事件,未成功则继续等待
	Wait_CheckEvent(I2C2,I2C_EVENT_MASTER_TRANSMITTER_MODE_SELECTED);
	// 发送数据1(寄存器地址)
	I2C_SendData(I2C2,RegAddr);
	// 检测EV8_2事件,未成功则继续等待
	Wait_CheckEvent(I2C2,I2C_EVENT_MASTER_BYTE_TRANSMITTED);
	// 生成重复起始条件
	I2C_GenerateSTART(I2C2,ENABLE);
	// 检测EV5事件,未成功则继续等待
	Wait_CheckEvent(I2C2,I2C_EVENT_MASTER_MODE_SELECT);
	// 指定地址读
	I2C_Send7bitAddress(I2C2,MPU6050_Slave,I2C_Direction_Receiver);
	// 检测EV6事件,未成功则继续等待
	Wait_CheckEvent(I2C2,I2C_EVENT_MASTER_RECEIVER_MODE_SELECTED);
	// 只发送一个字节,所以要提前将ACK置0并提前停止
	I2C_AcknowledgeConfig(I2C2,DISABLE);
	I2C_GenerateSTOP(I2C2,ENABLE);
	// 检测EV7事件,未成功则继续等待
	Wait_CheckEvent(I2C2,I2C_EVENT_MASTER_BYTE_RECEIVED);
	Data = I2C_ReceiveData(I2C2);
	// ACK置回1
	I2C_AcknowledgeConfig(I2C2,ENABLE);
	return Data;
}

// 初始化
void MPU6050_Init(void){
	// 开启硬件IIC的时钟
	RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C2,ENABLE);
	// 开启GPIO时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB,ENABLE);
	
	// 配置GPIO,复用开漏模式
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD;
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10 | GPIO_Pin_11;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOB,&GPIO_InitStructure);
	
	// 配置IIC外设
	I2C_InitTypeDef I2C_InitStructure;
	I2C_InitStructure.I2C_Ack = I2C_Ack_Enable;		// 发送应答
	I2C_InitStructure.I2C_AcknowledgedAddress = I2C_AcknowledgedAddress_7bit;		// 作为从机应答几位地址
	I2C_InitStructure.I2C_ClockSpeed = 50000;		// 50KHz
	// 占空比,由于弱上拉的电路原因,当高速通讯时,数据的变化比较缓慢,需要更多的低电平来等待SDA改变电平,所以占空比在高速模式下会低电平占比更多,标志速度下此配置不起作用。
	I2C_InitStructure.I2C_DutyCycle = I2C_DutyCycle_2;
	I2C_InitStructure.I2C_Mode = I2C_Mode_I2C;	// I2C模式
	I2C_InitStructure.I2C_OwnAddress1 = 0x00;		// 作为从机时的地址,需与作为从机时应答几位地址配置对应
	I2C_Init(I2C2,&I2C_InitStructure);
	
	// I2C外设使能
	I2C_Cmd(I2C2,ENABLE);
	
	
	// 配置寄存器
	MPU6050_WriteReg(PWR_MGMT_1,0x01);		// 解除睡眠,选择推荐的陀螺仪x轴时钟
	MPU6050_WriteReg(PWR_MGMT_2,0x00);		// 不用待机
	MPU6050_WriteReg(SMPLRT_DIV,0x09);		// 10分频
	MPU6050_WriteReg(CONFIG,0x06);	
	MPU6050_WriteReg(GYRO_CONFIG,0x18);		// 自测不使能,使用最大量程
	MPU6050_WriteReg(ACCEL_CONFIG,0x18);
}

// 用于存储获取的加速度与陀螺仪各轴的值
struct MPU6050_DataDef{
	int16_t AccX;
	int16_t AccY;
	int16_t AccZ;
	int16_t GyroX;
	int16_t GyroY;
	int16_t GyroZ;
}MPU6050_Data;
// 把寄存器的值高低位封装好了保存到结构体中
void MPU6050_GetData(void){
	uint16_t DataH,DataL;
	
	DataH = MPU6050_ReadReg(ACCEL_XOUT_H);
	DataL = MPU6050_ReadReg(ACCEL_XOUT_L);
	MPU6050_Data.AccX = (DataH << 8) | DataL;
	
	DataH = MPU6050_ReadReg(ACCEL_YOUT_H);
	DataL = MPU6050_ReadReg(ACCEL_YOUT_L);
	MPU6050_Data.AccY = (DataH << 8) | DataL;
	
	DataH = MPU6050_ReadReg(ACCEL_ZOUT_H);
	DataL = MPU6050_ReadReg(ACCEL_ZOUT_L);
	MPU6050_Data.AccZ = (DataH << 8) | DataL;
	
	DataH = MPU6050_ReadReg(GYRO_XOUT_H);
	DataL = MPU6050_ReadReg(GYRO_XOUT_L);
	MPU6050_Data.GyroX = (DataH << 8) | DataL;
	
	DataH = MPU6050_ReadReg(GYRO_YOUT_H);
	DataL = MPU6050_ReadReg(GYRO_YOUT_L);
	MPU6050_Data.GyroY = (DataH << 8) | DataL;
	
	DataH = MPU6050_ReadReg(GYRO_ZOUT_H);
	DataL = MPU6050_ReadReg(GYRO_ZOUT_L);
	MPU6050_Data.GyroZ = (DataH << 8) | DataL;
}

uint8_t MPU8050_GetId(void){
	return MPU6050_ReadReg(WHO_AM_I);
}

主函数main.c:

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include "OLED.h"
#include "MPU6050.h"
#include "Timer.h"


int main(void)
{
	uint8_t ID;
	OLED_Init();
	OLED_ShowString(1,1,"ID:");
	MPU6050_Init();
	Timer_Init();
	ID = MPU8050_GetId();
	OLED_ShowHexNum(1,7,ID,2);
	MPU6050_GetData();
	while (1)
	{
		OLED_ShowSignedNum(2,1,MPU6050_Data.AccX,5);
		OLED_ShowSignedNum(2,9,MPU6050_Data.AccY,5);
		OLED_ShowSignedNum(3,1,MPU6050_Data.AccZ,5);
		OLED_ShowSignedNum(3,9,MPU6050_Data.GyroX,5);
		OLED_ShowSignedNum(4,1,MPU6050_Data.GyroY,5);
		OLED_ShowSignedNum(4,9,MPU6050_Data.GyroZ,5);

	}
}


//中断函数
void TIM2_IRQHandler(void){
	// 获取中断标志位
	if(TIM_GetITStatus(TIM2,TIM_IT_Update) == SET){
		MPU6050_GetData();
		// 清除标志位
		TIM_ClearITPendingBit(TIM2,TIM_IT_Update);
	}
}


总结

硬件I2C使用了大量的库函数来配置,所以需要了解这些库函数的使用,配合寄存器的详细介绍来理解。文章来源地址https://www.toymoban.com/news/detail-844479.html

到了这里,关于STM32硬件I2C通信外设的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • RT-Thread在STM32硬件I2C的踩坑记录

    参考文章: 1.将硬件I2C巧妙地将“嫁接”到RTT原生的模拟I2C驱动框架 2.基于STM32F4平台的硬件I2C驱动实现笔记 3.《rt-thread驱动框架分析》- i2c驱动   最近打算用RT-Thread做一个小demo玩玩,其中需要用I2C通信驱动一个oled屏幕,但是找了一圈也没找到RTT中对硬件I2C的支持方式以及

    2024年02月11日
    浏览(74)
  • 【STM32】STM32学习笔记-I2C通信协议(31)

    I2C(Inter-Integrated Circuit)总线 是一种由NXP(原PHILIPS)公司开发的两线式串行总线,用于连接微控制器及其外围设备。多用于主控制器和从器件间的主从通信,在小数据量场合使用,传输距离短,任意时刻只能有一个主机等特性。 串行的 8 位双向数据传输位速率在标准模式下可

    2024年01月23日
    浏览(75)
  • 【STM32】| 02——常用外设 | I2C

    系列文章目录 【STM32】| 01——常用外设 | USART 【STM32】| 02——常用外设 | I2C 失败了也挺可爱,成功了就超帅。 本文详细介绍 I2C协议及 MCU I2C配置使用 I2C是一种常用的串行通信总线,由串行数据线SDA 和串线时钟线SCL组成。I2C是一种多主机控制总线,由飞利浦公司为了让主板

    2024年01月19日
    浏览(42)
  • STM32--MPU6050与I2C外设

    在51单片机专栏中,用过I2C通信来进行实现AT24C02的数据存储; 里面介绍的是 利用程序的编程来实现I2C的时序 ,进而实现AT24C02与单片机之间的关系连接; 本章将介绍使用I2C的硬件外设来实现I2C通信,和介绍MPU6050,利用I2C通信实现STM32对MPU6050的控制. I2C通信软件实现程序链接

    2024年02月11日
    浏览(57)
  • 【STM32学习】——I2C通信协议&MPU6050姿态传感器&软件I2C读写MPU6050

    ​   目录 前言 一、I2C通信协议 1.简介 2.硬件电路设计 3.I2C时序(软件)

    2024年02月16日
    浏览(56)
  • 【STM32】入门(七):I2C硬件控制方式

    之所以叫“I2C硬件控制方式”是与“软件控制方式”相对。I2C软件控制,就是写程序直接操作两个GPIO引脚,分别作为时钟线SCL和数据线SDA,按照I2C协议的时序要求,操作GPIO输入、输出、高电平、低电平。 听着就很复杂,好在STM32中有I2C的硬件实现,即通过简单的操作寄存器

    2023年04月08日
    浏览(49)
  • STM32——I2C通信

            I2C(Inter IC Bus)是由Philips公司开发的一种通用数据总线,它是两线式串行总线,它具有两根通信线: SCL(Serial Clock)、SDA(Serial Data) ,多用于主控制器和从器件间的主从通信,在小数据量场合使用,传输距离短,任意时刻只能有一个主机等特性。I2C是同步半双

    2024年01月25日
    浏览(51)
  • STM32 HAL FreeRTOS 硬件I2C 使用

    因为某个项目想要颜色识别,去识别球的颜色,但是又不想多来个摄像头,所以想尝试一下颜色传感器的方案。但是经过尝试,HAL库生成的 FreeRTOS 硬件 I2C 读写一直在报错。 刚好手头上有九轴陀螺仪的例程代码。最后用FreeRTOS 硬件 I2C 读取数据。 这里提到了阻塞式 HAL 函数(

    2024年02月20日
    浏览(52)
  • 10:STM32------I2C通信

    目录 一:I2C通信协议 1:I2C简历 2:硬件电路 3:I2C时序基本单元 A : 开/ 终条件 2:发送一个字节 3:接收一个字节 4:应答机制  4:I2C时序  1:指定地址写 2:当前地址读 3: 指定地址读 二:MPU6050 1:简历 2:参数 3:硬件电路 4:框图 5:寄存器地址 三:案例 A:软件I2C读写 MPU6050 1:连接图 2:代码 B:硬

    2024年02月20日
    浏览(59)
  • 【【STM32----I2C通信协议】】

    我们会发现I2C有两根通信线: SCL和SDA 同步 半双工 带数据应答 支持总线挂载多设备(一主多从,多主多从) 硬件电路 所有I2C设备的SCL连在一起,SDA连在一起 设备的SCL和SDA均要配置成开漏输出模式 SCL和SDA各添加一个上拉电阻,阻值一般为4.7KΩ左右 左边的CPU就是主机,他的权

    2024年02月12日
    浏览(51)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包