前段时间由于其他原因,专栏暂停更新了较长一段时间,现在恢复更新,争取继续为大家创造有价值的内容,期待大家的订阅关注,欢迎互相学习交流。
在STM32速成笔记系列专栏中其实已经对GPIO的一些必要知识进行了介绍,但是如果我们想要深入了解GPIO的一些寄存器知识和各种输入输出模式,还需要继续深入学习,本文就让我们一起来深入学习GPIO的相关内容。
PS:笔者能力有限,文章如有错漏之处,还请各位不吝赐教,感谢大家的支持!
文章目录
- 一、GPIO简介
- 二、GPIO寄存器
- 2.1 端口配置寄存器低/高
- 2.2 端口输出寄存器
- 2.3 端口设置/清除寄存器
- 2.4 端口清除寄存器
- 三、硬件分析
- 3.1 推挽输出
- 3.2 开漏输出
- 四、程序分析
一、GPIO简介
每个GPI/O端口有两个32位配置寄存器(GPIOX CRL,GPIOx CRH),两个32位数据寄存器(GPIOX IDR和GPIOx ODR),一个32位置位1复位寄存器(GPIOx BSRR),一个16位复位寄存器(GPIOx BRR)和一个32位锁定寄存器(GPIOx LCKR)。根据数据手册中列出的每个!/O端口的特定硬件特征,GPIO端口的每个位可以由软件分别配置成多种模式。
每个I/O端口位可以自由编程,然而I/O端口寄存器必须按32位字被访问(不允许半字或字节访问)。GPIOX BSRR和GPIOX BRR寄存器允许对任何GPIO寄存器的读/更改的独立访问:这样,在读和更改访问之间产生IRQ时不会发生危险。
上面一段文字是STM32中文参考手册中对于GPIO功能的描述,这有一个关键就是I/O寄存器必须按照32位来操作,我们在看一些程序工程时可能会看到如下的宏定义
// SDA方向
#define MPU_SDA_IN() {GPIOB->CRL&=0X0FFFFFFF;GPIOB->CRL|=(u32)8<<28;}
#define MPU_SDA_OUT() {GPIOB->CRL&=0X0FFFFFFF;GPIOB->CRL|=(u32)3<<28;}
这里也是严格按照32位来操作了GPIO,关于上面两句具体是什么作用,是怎么实现的,我们下面随着学习的深入会一步一步来分析。
我们知道I/O有多种输入输出模式,在STM32中文参考手册的GPIO功能描述部分给出了各种模式的配置方法
复位期间和刚复位后,复用功能未开启,I/O端口被配置成浮空输入模式,复位后JTAG引脚会被配置成对应的输入输出模式。
在使用时需要注意们尽量不要使用这些引脚,具体是哪些引脚以及如果必须使用这几个引脚改怎么办,在STM32速成笔记专栏中有介绍,这里就不再赘述了。
当作为输出配置时,写到输出数据寄存器上的值(GPIOx_ODR)输出到相应的I/O引脚,输入数据寄存器(GPIOx_IDR)在每个APB2时钟周期捕捉I/O引脚上的数据。
二、GPIO寄存器
我们这里介绍几个关键的GPIO寄存器
2.1 端口配置寄存器低/高
每一个GPIOx都有16个引脚0~15,每一个GPIO引脚的输入输出配置需要4bit,一个寄存器是32bit,所以想要能配置16个引脚的输入输出模式需要两个GPIO配置寄存器,所以在STM32参考手册中有高低之分,低控制0到7,高控制8到15,我们只附一个GPIO配置寄存器低的图,高的是类似的。
我们看到这里介绍的时候只给出了“上拉/下拉式输入模式”并没有说明怎么配置是上拉输入,怎么配置是下拉输入,这时我们可以结合上面的表格17来参考。
另外笔者在看的时候注意到“输入模式”后面写着“复位后的状态”,导致笔者以为如果想从输出模式配置成输入模式需要复位操作,后来明白这里的意思应该是引脚复位后的默认状态应该是浮空输入模式。
2.2 端口输出寄存器
结合GPIO配置寄存器和上面的表格,加入我们想要配置一个GPIO的某各引脚为上拉输入模式,我们需要配置它的CNFy[1:0]为10,配置它的MODEy[1:0]为00,还需要再配置ODR寄存器来确定是上拉输入模式而不是下拉,我们来看一下端口数据输出寄存器(GPIOx_ODR寄存器)的描述。
2.3 端口设置/清除寄存器
我们想要配置成上拉输入模式,需要将GPIOx_ODR对应位设置成1,从上面对于ODR寄存器描述来看,我们可以借助GPIOx_BSRR寄存器来单独操作ODR寄存器的值,STM32参考手册中对于段端口位设置/清除寄存器(GPIOx_BSRR)的描述如下。
根据寄存器描述,我们只需要将对应端口的BS位设置成1即可完成上拉输入模式的配置。
2.4 端口清除寄存器
除了可以通过上面的GPIOx_BSRR来清除对应的ODR位之外,还提供了一个端口清楚寄存器(GPIOx_BRR)来单独清除,和上面的GPIOx_BSRR的清除方法是一样的,这里就不再做赘述了。
此外还有一个端口配置锁定寄存器(GPIOx_LCKR)这里也不做介绍了。
三、硬件分析
下面我们来简单分析一下端口配置成不同模式时的硬件电路,从而更深的理解他们的区别,这里之分析推挽式输出与开漏输出。在STM32中文参考手册中给出了输出的配置图并且针对推挽式输出和开漏输出给出了一些描述
- 开漏模式:输出寄存器上的’0’激活N-MOS,而输出寄存器上的’1’将端口置于高阻状态(PMOS从不被激活)。
- 推挽模式:输出寄存器上的’0’激活N-MOS,而输出寄存器上的’1’将激活P-MOS。
如果只是从表面上来理解的话,推挽式输出是具有实际输出能力的,也就是说推挽式输出的高电平可以真正的驱动负载,而开漏输出无法输出真正的高电平,也就是说开漏输出的高电平是无法驱动负载的。
针对其中的细节电路已经有很多博主作了介绍,由于笔者对于硬件不擅长,所以就不露怯了,指从表面上来分析一下推挽输出与开漏输出的区别。
我们单独把上图的输出驱动器部分摘出来看一下
3.1 推挽输出
我们先分析一下推挽式输出
- 当P-MOS打开,N-MOS关闭时,输出高电平,电流会流出去,这就叫“推”
- 当P-MOS关闭,N-MMOS打开时,输出低电平,电流会流进来,这就叫“挽”
3.2 开漏输出
下面我们再来看一下开漏输出,端口配置为开漏输出时,P-MOS一直处于关闭状态,当N-MOS打开时,输出高电平,当N-MOS关闭时,处于一个高阻态/浮空。我们既然已经有了推挽输出,而且开漏输出还需要一个上拉电阻才能输出高电平,那为什么还需要开漏输出呢?下面我们就介绍一下开漏输出的用处。
首先第一个,根据STM32中文参考手册中的描述,在开漏模式时,对输入数据寄存器的读访问可得到I/O状态,也就是说,如果我们把引脚初始化为开漏输出模式,我们需要读取引脚输入时就不需要再把引脚配置成输入模式了。但是在推挽式模式时,对输出数据寄存器的读访问得到最后一次写的值,所以如果引脚配置的是推挽输出,那么需要读取引脚数据时需要重新初始化成输入模式。开漏输出的这个特点对于在进行IIC通信或者其他共用数据线的总线通信中十分友好,不需要重新初始化SDA引脚为输入模式也可以读取引脚数据。
第二个,开漏输出可以用来匹配电平,比如我们有一个单片机的引脚输出是5V,但是我们的目标器件电压是3.3V,此时我们可以初始化引脚为开漏输出,外部接一个3.3V上拉电阻,这样的话当我们的GPIO处于高阻态时输出端就会被上拉电阻拉到3.3V,同理反过来也是一样的,也就是说STM32的引脚配置成开漏输出外部接一个5V上拉电阻就能实现5V输出,STM官方社区也给出了这个说明,但是笔者没有尝试过,所以大家请谨慎尝试。
第三呢就是可以支持几个设备同时挂接到同一条线上,比如我们使用IIC通信,有多个主设备时如果我们设置成推挽输出,可能会导致主设备之间产生短路。
最后就是开漏输出能够实现线与操作,什么是线与?线与简单理解就是当一根总线上的设备输出全部为高电平时,总线处于高电平状态,一旦有一个设备的输出是低电平,总线处于低电平状态。比如在IIC总线的仲裁机制就是通过线与逻辑实现的,我们知道当主机要开启IIC通信时需要先发送一个起始信号,也就是在SCL为高时,将SDA拉高,再拉低,但是当IIC总线上有多个主设备时,就需要有一个仲裁机制来判断谁先占用总线,谁后占用总线。当主设备将SDA拉高后,需要检查SDA的电平如果此时SDA电平为高,说明主设备可以占用总线,如果此时SDA电平为低,说明总线已经被其他设备占用。
四、程序分析
最后我们说回来最开始介绍的两行代码
// SDA方向
#define MPU_SDA_IN() {GPIOB->CRL&=0X0FFFFFFF;GPIOB->CRL|=(u32)8<<28;}
#define MPU_SDA_OUT() {GPIOB->CRL&=0X0FFFFFFF;GPIOB->CRL|=(u32)3<<28;}
对照上面的表格与寄存器我们可以知道,第一行是将SDA配置成了下拉输入模式,第二行是将SDA配置成了通用推挽输出模式,对应的引脚是PB7。
我们从这个配置来说,可以推测他在初始化IIC引脚是配置的是推挽输出模式,所以需要不断地切换SDA引脚的输入输出状态,不是很友好,而且当总线上需要有多个主设备时会造成短路,所以建议后续IIC引脚还是初始化成开漏输出模式。