模拟IIC——关于模拟IIC的IO口的配置选取推挽输出还是开漏输出,以及是否需要更改IO口输入输出模式和是否需要对IO配置上拉

问题如下

当时我以为引脚配错了,原理图明明是B引脚,为何程序是C呢
在这里插入图片描述
在这里插入图片描述

查了一下资料,顿悟了

https://blog.youkuaiyun.com/m0_62243928/article/details/125779308

在使用模拟IIC的时候,观看别人的程序的时候发现了程序之间的一些不一样的地方

——————————————————————————————————代码1————————————————————————————————————
 
//IO方向设置
#define SDA_IN()  {GPIOB->MODER&=~(3<<(9*2));GPIOB->MODER|=0<<9*2;}	//PB9输入模式
#define SDA_OUT() {GPIOB->MODER&=~(3<<(9*2));GPIOB->MODER|=1<<9*2;} //PB9输出模式
//IO操作函数	 
#define IIC_SCL    PBout(8) //SCL
#define IIC_SDA    PBout(9) //SDA	 
#define READ_SDA   PBin(9)  //输入SDA 
———————————————————————————————————————————————————————————————————————————
——————————————————————————————————代码2————————————————————————————————————
 
#define BH1750_I2C_SCL_1()  GPIO_SetBits(GPIOB, GPIO_Pin_6)		/* SCL = 1 */
#define BH1750_I2C_SCL_0()  GPIO_ResetBits(GPIOB, GPIO_Pin_6)	/* SCL = 0 */
	
#define BH1750_I2C_SDA_1()  GPIO_SetBits(GPIOB, GPIO_Pin_7)		/* SDA = 1 */
#define BH1750_I2C_SDA_0()  GPIO_ResetBits(GPIOB, GPIO_Pin_7)	/* SDA = 0 */
	
#define BH1750_I2C_SDA_READ()  GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_7)	
/* 读SDA口线状态 */
———————————————————————————————————————————————————————————————————————————

代码1中间有一个对SDA数据线的模式的配置, 也就是输入输出的设置;

在向从设备写数据的时候会先调用SDA_OUT(),将引脚配置为输出模式;

在向从设备写数据的时候会先调用SDA_IN(),将引脚配置为输入模式;

但是问题来了,代码2中间并没有对引脚的输入输出模式进行改变,代码2也能和从机进行正常的通信,这是为什么呢?模拟IIC通信时对引脚的输入输出模式的配置是否有必要呢?

我仔细对比了两份代码,发现两份代码在引脚的初始化部分不一样。

——————————————————————————————————代码1————————————————————————————————————
void SHT3x_Init(void)
{    
    RCC->APB2ENR|=1<<6;              //使能PORTE时钟					 
	GPIOE->CRL&=0XFF00FFFF;          //PE4,PE5
	GPIOE->CRL|=0X00330000;	         //推挽输出 
	GPIOE->ODR|=3<<4;                //将PE4、PE5设为1
}
——————————————————————————————————代码2————————————————————————————————————
 
static void I2C_BH1750_GPIOConfig(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;
 
	RCC_APB2PeriphClockCmd(BH1750_RCC_I2C_PORT, ENABLE);	/* 打开GPIO时钟 */
 
	GPIO_InitStructure.GPIO_Pin = BH1750_I2C_SCL_PIN | BH1750_I2C_SDA_PIN;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;  	    /* 开漏输出 */
	GPIO_Init(BH1750_GPIO_PORT_I2C, &GPIO_InitStructure);
 
	/* 给一个停止信号, 复位I2C总线上的所有设备到待机模式 */
	i2c_Stop();
}
RCC_APB2PeriphClockCmd(BH1750_RCC_I2C_PORT, ENABLE);	/* 打开GPIO时钟 */

GPIO_InitStructure.GPIO_Pin = BH1750_I2C_SCL_PIN | BH1750_I2C_SDA_PIN;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD;  	    /* 开漏输出 */
GPIO_Init(BH1750_GPIO_PORT_I2C, &GPIO_InitStructure);

/* 给一个停止信号, 复位I2C总线上的所有设备到待机模式 */
i2c_Stop();

}

可以发现:推挽输出对应->需要切换输入输出模式;开漏输出对应->不需要切换输入输出模式;

我学习IIC的时候明明记得IIC需要使用开漏输出和接上拉电阻的;如果使用推挽输出,当多个设备

连接到一个总线上面时,如果一个设备输出低电平一个设备输出高电平就会出现短路的情况。而且

不能推挽输出实现线与。

为什么这里可以使用推挽呢?我猜测这是模拟IIC和硬件IIC的不同了。

我使用模拟IIC一般来说不会出现需要多个设备连接到一个总线上面情况,也就不会出现上面的短

路的可能以及线与的需求了。所以模拟IIC这里是可以使用推挽输出的。

那为什么推挽输出需要切换输入输出模式,开漏输出不需要切换输入输出模式呢?

我们首先需要了解到GPIO口的输入和输出模式有什么不同以及推挽输出和开漏输出的不同:

下面对于GPIO口的内容引用

GPIO口的输入,输出模式及其说明
https://blog.youkuaiyun.com/qq_42384937/article/details/82428812
在输入模式下只有红色圈出的部分处于工作状态,也就是说下半部分的输出电路,实际上是与端口处于隔离状态,不能工作,这个时候我们不能去读取端口的电平。
在这里插入图片描述
在输出模式下,图的上半部,施密特触发器处于开启状态,这意味着CPU可以在“输入数据寄存器”的另一端,随时监控I/O端口的状态;
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
我们可以发现:

在输出模式下,施密特触发器处于开启状态,这意味着CPU可以在“输入数据寄存器”的另一端,随时监控I/O端口的状态,也就是可以读取IO口的值;

但是对于推挽输出而言:推挽输出是强输出电流模式,在此模式下的输出通道上的推挽结构MOS管,属于强上拉和强下拉的,这会影响读取IDR时的值,强上拉意味着会将来自外部的低电平输入强制置高,强下拉意味着会将来自外部的高电平输入强制置低

在开漏模式下,实现了虚拟的I/O端口双向通信:只要CPU输出逻辑“1”,由于N-MOS管处于关闭状态,I/O端口的电平将完全由外部电路决定,因此,CPU可以在“输入数据寄存器”读到外部电路的信号,而不是它自己输出的逻辑“1”。

了解stm32的双向io口

请问STM32F4的GPIO管脚可以同时配置成输入、输出模式吗
详细一点的内容可以看这两个链接

现在我们可以分析两个代码的差异:推挽输出对应->需要切换输入输出模式;开漏输出对应->不需要切换输入输出模式;

因为设置为推挽输出时 输出通道上的推挽结构MOS管,属于强上拉和强下拉的,这会影响读取IDR时的值,强上拉意味着会将来自外部的低电平输入强制置高,强下拉意味着会将来自外部的高电平输入强制置低。所以我们去读取IO口的值会是输出的值,并不能得到外部电路的值。所以在IIC通信时需要读取外部数据的时候需要将IO的模式配置成输入模式;

但是对于开漏输出来说,只要CPU输出逻辑“1”,由于N-MOS管处于关闭状态,I/O端口的电平将完全由外部电路决定,因此,CPU可以在“输入数据寄存器”读到外部电路的信号,而不是它自己输出的逻辑“1”。所以不需要去将IO口的模式配置成输入模式

这也就造成了上面两个代码的差异。

上拉电阻:查看F4的中文参考手册可以知道
在这里插入图片描述
IO口配置成上拉的时候可以解决推挽输出输出高电平时高阻态没办法拉高电平的情况。

注意:
使用开漏输时,如果需要读取电平值,最好输出高电平,由于N-MOS管处于关闭状态,I/O端口的电平将完全由外部电路决定,这时读到的才是外部电路的电平值。

每个线上加了4.7K的上拉电阻【上拉电阻具体数值需要计算和测试,基本选择10K往下】

对于MCU和EE单对单的:1.SCL并不是加了上拉电阻就得用推挽输出;2.SCL可以用推挽也可以用开漏,SDA必须用开漏输出;3.对于主从电平不一致的情况,SCL需要做上拉用开漏,将电平拉到从电平。

不会存在所谓的线与,IO口短路。SDA采用推挽,有可能会出现高电平被下拉,大概率是驱动负载过大的原因。

### STM32 HAL库 IO模拟 IIC协议 输出 示例代码 为了实现通过STM32的GPIO模拟IIC通信,通常需要设置GPIO输出模式(Open-Drain Output)。这种模式允许SCLSDA信号线被多个设备共享,并支持上拉电阻来确保总线电平稳定。以下是基于HAL库的一个完整示例代码。 #### 配置说明 在使用HAL库时,需注意以下几点: 1. 使用`GPIO_InitTypeDef`结构体初始化GPIO引脚。 2. 设置GPIO模式为`GPIO_MODE_OUTPUT_OD`(输出),并启用内部上拉电阻`GPIO_PULLUP`[^3]。 3. 编写函数手动控制SCLSDA的状态变化以完成起始条件、停止条件以及数据传输过程。 #### 初始化代码 下面是一个简单的初始化代码片段: ```c #include "stm32f1xx_hal.h" #define SDA_PIN GPIO_PIN_7 #define SCL_PIN GPIO_PIN_6 #define GPIO_PORT GPIOB void GPIO_I2C_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; __HAL_RCC_GPIOB_CLK_ENABLE(); // 启用GPIOB时钟 // 配置SDA引脚 (PB7) GPIO_InitStruct.Pin = SDA_PIN; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_OD; // 输出 GPIO_InitStruct.Pull = GPIO_PULLUP; // 上拉电阻 GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIO_PORT, &GPIO_InitStruct); // 配置SCL引脚 (PB6) GPIO_InitStruct.Pin = SCL_PIN; HAL_GPIO_Init(GPIO_PORT, &GPIO_InitStruct); } ``` 上述代码完成了对SCLSDA两个引脚的基础配置。 #### 模拟IIC核心功能函数 接下来定义几个用于操作IIC的关键函数,包括启动、发送字节、接收字节等功能。 ```c // 延迟函数 void delay_us(uint16_t us) { __HAL_TIM_SET_COUNTER(&htim2, 0); // 清零定时器计数值 while (__HAL_TIM_GET_COUNTER(&htim2) < us); // 循环等待直到达到指定微秒数 } // 发送START信号 void I2C_Start() { HAL_GPIO_WritePin(GPIO_PORT, SDA_PIN, GPIO_PIN_SET); // SDA先释放至高电平 delay_us(5); HAL_GPIO_WritePin(GPIO_PORT, SCL_PIN, GPIO_PIN_RESET); // 下拉SCL delay_us(5); HAL_GPIO_WritePin(GPIO_PORT, SDA_PIN, GPIO_PIN_RESET); // 下拉SDA delay_us(5); HAL_GPIO_WritePin(GPIO_PORT, SCL_PIN, GPIO_PIN_SET); // 释放SCL delay_us(5); } // 发送STOP信号 void I2C_Stop() { HAL_GPIO_WritePin(GPIO_PORT, SDA_PIN, GPIO_PIN_RESET); // 下拉SDA delay_us(5); HAL_GPIO_WritePin(GPIO_PORT, SCL_PIN, GPIO_PIN_SET); // 释放SCL delay_us(5); HAL_GPIO_WritePin(GPIO_PORT, SDA_PIN, GPIO_PIN_SET); // 释放SDA delay_us(5); } // 主机向从机发送一个字节的数据 uint8_t I2C_SendByte(uint8_t byte) { uint8_t i, ack; for(i=0;i<8;i++) { if(byte & 0x80) { HAL_GPIO_WritePin(GPIO_PORT, SDA_PIN, GPIO_PIN_SET); // 数据位为1 } else { HAL_GPIO_WritePin(GPIO_PORT, SDA_PIN, GPIO_PIN_RESET); // 数据位为0 } byte <<= 1; HAL_GPIO_WritePin(GPIO_PORT, SCL_PIN, GPIO_PIN_SET); // 拉高SCL delay_us(5); HAL_GPIO_WritePin(GPIO_PORT, SCL_PIN, GPIO_PIN_RESET); // 拉低SCL delay_us(5); } HAL_GPIO_WritePin(GPIO_PORT, SDA_PIN, GPIO_PIN_SET); // 准备读取ACK/NACK HAL_GPIO_WritePin(GPIO_PORT, SCL_PIN, GPIO_PIN_SET); // 拉高SCL delay_us(5); ack = HAL_GPIO_ReadPin(GPIO_PORT, SDA_PIN); // 读取从机响应 HAL_GPIO_WritePin(GPIO_PORT, SCL_PIN, GPIO_PIN_RESET); // 拉低SCL return ack; } // 接收来自从机的一个字节的数据 uint8_t I2C_ReceiveByte(void) { uint8_t i, data = 0; HAL_GPIO_WritePin(GPIO_PORT, SDA_PIN, GPIO_PIN_SET); // 将SDA设置为输入状态 for(i=0;i<8;i++) { HAL_GPIO_WritePin(GPIO_PORT, SCL_PIN, GPIO_PIN_SET); // 拉高SCL delay_us(5); data = (data << 1) | HAL_GPIO_ReadPin(GPIO_PORT, SDA_PIN); // 读取数据位 HAL_GPIO_WritePin(GPIO_PORT, SCL_PIN, GPIO_PIN_RESET); // 拉低SCL delay_us(5); } return data; } ``` 以上代码实现了基本的IIC通信逻辑,包括起始信号、停止信号、单字节发送与接收的功能[^1]。 #### 主函数调用实例 最后,在主循环中可以这样调用来测试EEPROM或其他外设连接情况: ```c int main(void) { HAL_Init(); SystemClock_Config(); GPIO_I2C_Init(); // 初始化GPIO作为IIC I2C_Start(); // 发送START信号 I2C_SendByte(0xA0); // 设定目标地址(假设为0xA0) I2C_SendByte(0x00); // 写入存储位置索引 I2C_SendByte('H'); // 写入字符'H' I2C_Stop(); // 结束通讯 while (1) {} } ``` 此部分展示了如何利用之前编写的子函数构建完整的IIC事务处理流程[^2]。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

不熬夜,早点睡

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值