STM32硬件I2C通信外设

文章目录

  • 前言
  • I2C硬件介绍
    • 10 位地址模式
    • 硬件I2C的引脚定义
    • I2C框图
    • 主机发送序列图
    • 主机接收序列图
  • 硬件I2C读写MPU6050
  • 总结


前言

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


I2C硬件介绍

在这里插入图片描述

10 位地址模式

在8位指定地址读的基础下,把第一个字节的前5位改为标志位,值为11110,表示为10位地址模式,剩下的三位加上第二个字节的八位组成十位地址位加上读写位
在这里插入图片描述

硬件I2C的引脚定义

在这里插入图片描述

I2C框图

发送数据:写入数据寄存器,数据寄存器写入移位寄存器,然后数据寄存器为空,继续写入,再通过移位寄存器一位一位的发送完毕,然后置状态寄存器TXE = 1,发送寄存器为空。
接收数据:通过移位寄存器一位一位的接收数据,接收完毕后移入数据寄存器,置RXNE = 1,接收寄存器非空。
在这里插入图片描述
自身地址寄存器&双地址寄存器:在STM32作为从机时,通过这两个寄存器寻址,若与设置的地址对应则使STM32作为从机使用,双地址则表示作为多个从机使用。

I2C基本使用框图:主要需要使用的部分
在这里插入图片描述

主机发送序列图

在这里插入图片描述

主机接收序列图

在这里插入图片描述

通过参考手册,了知道这些标志位的作用与操作
以下为控制寄存器1被需要用到的标志位
在这里插入图片描述

以下为状态寄存器中被需要使用的标志位
在这里插入图片描述
在这里插入图片描述

硬件I2C读写MPU6050

电路连接

在这里插入图片描述

按照官方序列图来改写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使用了大量的库函数来配置,所以需要了解这些库函数的使用,配合寄存器的详细介绍来理解。

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

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

相关文章

什么是网页抓取 Web Scraping?如何进行网页抓取?

现在&#xff0c;不论是个人开发者还是庞大的企业都需要从互联网抓取大量数据&#xff0c;而网页抓取&#xff08;Web Scraping&#xff09;技术正是获取互联网上无尽信息宝库的一把钥匙。通过网页抓取工具&#xff0c;我们可以快速收集产品价格、市场趋势、用户评论等关键数据…

uniapp h5 touch事件踩坑记录

场景&#xff1a;悬浮球功能 当我给悬浮球设置了 position: fixed; 然后监听悬浮球的touch事件&#xff0c;从事件对象中拿到clientY和clientX赋值给悬浮球的left和top属性。当直接赋值后效果应该是这样子&#xff1a; 注意鼠标相对悬浮球的位置&#xff0c;应该就是左上角&a…

力扣_876_ 链表的中间结点(c语言)

题目描述&#xff1a; 解题方法&#xff1a; struct ListNode* middleNode(struct ListNode* head) {struct ListNode* l1,*l2;l1l2head;while(l2&&l2->next){l1l1->next;l2l2->next->next;}return l1; }

element UI季度选择器的实现

效果展示 用elementUI的select实现季度选择器 代码实现 generateQuarterOption放在methods中&#xff0c;需要近几年的只需要修改第一个循环的次数即可&#xff0c;mounted生命周期函数中调用generateQuarterOption() generateQuarterOption() {//近3年所有季度let now ne…

6行代码,1行命令!轻松实现多模态(视觉)模型离线推理 在线服务

早在去年年底&#xff0c;LMDeploy 已经悄悄地支持了多模态&#xff08;视觉&#xff09;模型&#xff08;下文简称 VLM&#xff09;推理&#xff0c;只不过它静静地躺在仓库的 examples/vl 角落里&#xff0c;未曾与大家正式照面。 LMDeploy 开源链接&#xff1a; https://gi…

Android | 开发过程遇到的报错以及解决方法

注&#xff1a; 此博客为记录个人开发过程中遇到的报错问题以及解决方案。 由于不同版本环境等因素影响&#xff0c;解决方案对其他人可能无效。 本博客仅提供一种解决思路&#xff0c;具体问题请具体分析。 报错&#xff1a;Connection timed out: connect解决&#xff1a;在G…

763. 划分字母区间(力扣LeetCode)

763. 划分字母区间 题目描述 给你一个字符串 s 。我们要把这个字符串划分为尽可能多的片段&#xff0c;同一字母最多出现在一个片段中。 注意&#xff0c;划分结果需要满足&#xff1a;将所有划分结果按顺序连接&#xff0c;得到的字符串仍然是 s 。 返回一个表示每个字符串…

pip wheel直接为离线环境打包需要的python包

很多情况下&#xff0c;需要离线安装python库&#xff0c;直接下载所需的库包时&#xff0c;可能又要求更新或安装相关的依赖包&#xff08;这就非常麻烦了&#xff09;&#xff0c;所以推荐一条命令一步到位&#xff0c;命令如下&#xff1a; pip wheel -r requirements.txt …

设计模式-设配器模式

目录 &#x1f38a;1.适配器模式介绍 &#x1f383;2.适配器类型 &#x1f38f;3.接口适配器 &#x1f390;4.类的适配器 &#x1f38e;5.优缺点 1.适配器模式介绍 适配器模式&#xff08;Adapter Pattern&#xff09;是作为两个不兼容的接口之间的桥梁。这种类型的设…

什么?想让视频号小店领先同行,竟然这么简单!

大家好&#xff0c;我是电商小布。 视频号小店从推出到现在&#xff0c;逐渐也是被越来越多的人所熟知了。 虽然说当前市场内部的商家数量并不多&#xff0c;竞争力不大。 但是在入驻之后想要领先同行商家&#xff0c;产生更好的店铺数据&#xff0c;该怎么来做呢&#xff1…

学习JavaEE的日子 Day29 yield,join,线程的中断,守护线程,线程局部变量共享,线程生命周期

Day29 多线程 12. 线程的礼让 Thread.yield(); 理解&#xff1a;此方法为静态方法&#xff0c;此方法写在哪个线程中&#xff0c;哪个线程就礼让 注意&#xff1a;所谓的礼让是指当前线程退出CPU资源&#xff0c;并转到就绪状态&#xff0c;接着再抢 需求&#xff1a;创建两个…

多叉树题目:N 叉树的后序遍历

文章目录 题目标题和出处难度题目描述要求示例数据范围进阶 解法一思路和算法代码复杂度分析 解法二思路和算法代码复杂度分析 解法三思路和算法代码复杂度分析 题目 标题和出处 标题&#xff1a;N 叉树的后序遍历 出处&#xff1a;590. N 叉树的后序遍历 难度 3 级 题目…

Android笔记(三十):PorterDuffXfermode实现旋转进度View

背景 核心原理是使用PorterDuffXfermode Path来绘制进度&#xff0c;并实现圆角 效果图 Android笔记(三十)效果演示 进度条绘制步骤 将ImageView矩形七个点的坐标存储起来&#xff08;configNodes&#xff09; 他们对应着7个不同的刻度&#xff0c;每个刻度的值 i * &#…

Unity | 射线检测及EventSystem总结

目录 一、知识概述 1.Input.mousePosition 2.Camera.ScreenToWorldPoint 3.Camera.ScreenPointToRay 4.Physics2D.Raycast 二、射线相关 1.3D&#xff08;包括UI&#xff09;、射线与ScreenPointToRay 2.3D&#xff08;包括UI&#xff09;、射线与ScreenToWorldPoint …

计算机基础,挑战全网最全解析

1.什么是计算机&#xff1f; 2.冯诺依曼结构 3.进制 4.摩尔斯码和布莱叶盲文 摩尔斯码 布莱叶盲文

如何使用群晖WebDAV实现固定公网地址同步Zotero文献管理器

文章目录 前言1. Docker 部署 Trfɪk2. 本地访问traefik测试3. Linux 安装cpolar4. 配置Traefik公网访问地址5. 公网远程访问Traefik6. 固定Traefik公网地址 前言 Trfɪk 是一个云原生的新型的 HTTP 反向代理、负载均衡软件&#xff0c;能轻易的部署微服务。它支持多种后端 (D…

蓝桥杯嵌入式学习笔记(6):IIC程序设计

目录 前言 1. IIC基本原理 2. 电路原理 3. 代码编程 3.1 预备工作 3.2 AT24C02写读功能编写 3.2.1 AT24C02写操作实现 3.2.2 AT24C02读操作实现 3.3 MCP4017写读功能编写 3.3.1 MCP4017写操作实现 3.3.2 MCP4017读操作实现 3.4 main.c编写 3.4.1 头文件引用 3.4.…

基于javaweb(springboot+mybatis)网上酒类商城项目设计和实现以及文档报告

基于javaweb(springbootmybatis)网上酒类商城项目设计和实现以及文档报告 博主介绍&#xff1a;多年java开发经验&#xff0c;专注Java开发、定制、远程、文档编写指导等,csdn特邀作者、专注于Java技术领域 作者主页 央顺技术团队 Java毕设项目精品实战案例《1000套》 欢迎点赞…

Redis数据类型介绍和使用

数据类型 String&#xff08;字符串&#xff09;&#xff1a;最基本的数据类型&#xff0c;可以存储任何类型的数据&#xff0c;如文本、数字等。Hash&#xff08;哈希&#xff09;&#xff1a;用于存储字段-值对的散列集合&#xff0c;适用于存储对象。List&#xff08;列表&…