学习本章时,配合《STM32F4xx 中文参考手册》 “通用 I/O(GPIO)” 章节一起阅读,效果会更佳,特别是涉及到寄存器说明的部分。 关于建立工程时使用 KEIL5 的基本操作,请参考前面的章节。
本讲建议看火哥视频,很重要。
链接:https://pan.baidu.com/s/1Dusgd-K1pPJpILpcp7xawg
提取码:w6nx
5.1 GPIO 简介:
GPIO 是通用输入输出端口的简称,简单来说就是 STM32 可控制的引脚, STM32 芯片的 GPIO 引脚与外部设备连接起来,从而实现与外部通讯、控制以及数据采集的功能。STM32 芯片的 GPIO 被分成很多组,每组有 16 个引脚,如型号为 STM32F4IGT6 型号的芯片有 GPIOA、 GPIOB、 GPIOC 至 GPIOI 共 9 组 GPIO,芯片一共 176 个引脚,其中 GPIO就占了一大部分,所有的 GPIO 引脚都有基本的输入输出功能。
最基本的输出功能是由 STM32 控制引脚输出高、低电平,实现开关控制,如把 GPIO引脚接入到 LED 灯,那就可以控制 LED 灯的亮灭,引脚接入到继电器或三极管,那就可以通过继电器或三极管控制外部大功率电路的通断。
最基本的输入功能是检测外部输入电平,如把 GPIO 引脚连接到按键,通过电平高低区分按键是否被按下。
5.2 GPIO 框图剖析:
通过 GPIO 硬件结构框图,就可以从整体上深入了解 GPIO 外设及它的各种应用模式。该图从最右端看起,最右端就是代表 STM32 芯片引出的 GPIO 引脚,其余部件都位于芯片内部。
5.2.1 基本结构分析:
下面我们按图中的编号对 GPIO 端口的结构部件进行说明。
1. 保护二极管及上、下拉电阻:
引脚的两个保护二级管可以防止引脚外部过高或过低的电压输入,当引脚电压高于VDD_FT 时, 上方的二极管导通,当引脚电压低于 VSS 时,下方的二极管导通,防止不正常电压引入芯片导致芯片烧毁。尽管有这样的保护,并不意味着 STM32 的引脚能直接外接大功率驱动器件,如直接驱动电机,强制驱动要么电机不转,要么导致芯片烧坏,必须要加大功率及隔离电路驱动。具体电压、电流范围可查阅《STM32F4xx 规格书》。
上拉、下拉电阻, 从它的结构我们可以看出, 通过上、下拉对应的开关配置,我们可以控制引脚默认状态的电压,开启上拉的时候引脚电压为高电平,开启下拉的时候引脚电压为低电平,这样可以消除引脚不定状态的影响。如引脚外部没有外接器件,或者外部的器件不干扰该引脚电压时, STM32 的引脚都会有这个默认状态。
也可以设置“既不上拉也不下拉模式”,我们也把这种状态称为浮空模式,配置成这个模式时,直接用电压表测量其引脚电压为 1 点几伏,这是个不确定值。所以一般来说我们都会选择给引脚设置“上拉模式”或“下拉模式”使它有默认状态。
STM32 的内部上拉是“弱上拉”,即通过此上拉输出的电流是很弱的,如要求大电流还是需要外部上拉。
通过“上拉/下拉寄存器 GPIOx_PUPDR”控制引脚的上、下拉以及浮空模式。
2. P-MOS 管和 N-MOS 管:
GPIO 引脚线路经过上、下拉电阻结构后,向上流向“输入模式”结构,向下流向“输出模式”结构。先看输出模式部分,线路经过一个由 P-MOS 和 N-MOS 管组成的单元电路。这个结构使 GPIO 具有了“推挽输出”和“开漏输出”两种模式。
所谓的推挽输出模式,是根据这两个 MOS 管的工作方式来命名的。在该结构中输入高电平时,上方的 P-MOS 导通,下方的 N-MOS 关闭,对外输出高电平;而在该结构中输入低电平时, N-MOS 管导通, P-MOS 关闭,对外输出低电平。当引脚高低电平切换时,两个管子轮流导通,一个负责灌电流,一个负责拉电流,使其负载能力和开关速度都比普通的方式有很大的提高。推挽输出的低电平为 0 伏,高电平为 3.3 伏,参考图 5-2 左侧,它是推挽输出模式时的等效电路。
而在开漏输出模式时,上方的 P-MOS 管完全不工作。如果我们控制输出为 0,低电平,则 P-MOS 管关闭, N-MOS 管导通,使输出接地,若控制输出为 1 (它无法直接输出高电平)时,则 P-MOS 管和 N-MOS 管都关闭,所以引脚既不输出高电平,也不输出低电平,为高阻态。为正常使用时必须接上拉电阻(可用 STM32 的内部上拉,但建议在 STM32 外部再接一个上拉电阻),参考图 5-2 中的右侧等效电路。它具“线与”特性,也就是说,若有很多个开漏模式引脚连接到一起时,只有当所有引脚都输出高阻态,才由上拉电阻提供高电平,此高电平的电压为外部上拉电阻所接的电源的电压。若其中一个引脚为低电平,那线路就相当于短路接地,使得整条线路都为低电平, 0 伏。
推挽输出模式一般应用在输出电平为 0 和 3.3 伏而且需要高速切换开关状态的场合。在 STM32 的应用中,除了必须用开漏模式的场合,我们都习惯使用推挽输出模式。
开漏输出一般应用在 I2C、 SMBUS 通讯等需要“线与”功能的总线电路中。除此之外,还用在电平不匹配的场合,如需要输出 5 伏的高电平,就可以在外部接一个上拉电阻, 上拉电源为 5 伏, 并且把 GPIO 设置为开漏模式,当输出高阻态时,由上拉电阻和电源向外输出 5 伏的电平。
通过 “输出类型寄存器 GPIOx_OTYPER”可以控制 GPIO 端口是推挽模式还是开漏模式。
3. 输出数据寄存器:
前面提到的双 MOS 管结构电路的输入信号,是由 GPIO“输出数据寄存器GPIOx_ODR”提供的,因此我们通过修改输出数据寄存器的值就可以修改 GPIO 引脚的输出电平。而“置位/复位寄存器 GPIOx_BSRR”可以通过修改输出数据寄存器的值从而影响电路的输出。
4. 复用功能输出:
“复用功能输出”中的“复用”是指 STM32 的其它片上外设对 GPIO 引脚进行控制,此时 GPIO 引脚用作该外设功能的一部分,算是第二用途。从其它外设引出来的“复用功能输出信号”与 GPIO 本身的数据据寄存器都连接到双 MOS 管结构的输入中,通过图中的梯形结构作为开关切换选择。
例如我们使用 USART 串口通讯时,需要用到某个 GPIO 引脚作为通讯发送引脚,这个时候就可以把该 GPIO 引脚配置成 USART 串口复用功能,由串口外设控制该引脚,发送数据。
5. 输入数据寄存器:
看 GPIO 结构框图的上半部分,它是 GPIO 引脚经过上、下拉电阻后引入的,它连接到施密特触发器,信号经过触发器后,模拟信号转化为 0、 1 的数字信号,然后存储在“输入数据寄存器 GPIOx_IDR”中,通过读取该寄存器就可以了解 GPIO 引脚的电平状态。
6. 复用功能输入:
与“复用功能输出”模式类似,在“复用功能输出模式”时, GPIO 引脚的信号传输到STM32 其它片上外设,由该外设读取引脚状态。
同样,如我们使用 USART 串口通讯时,需要用到某个 GPIO 引脚作为通讯接收引脚,这个时候就可以把该 GPIO 引脚配置成 USART 串口复用功能,使 USART 可以通过该通讯引脚的接收远端数据。
7. 模拟输入输出:
当 GPIO 引脚用于 ADC 采集电压的输入通道时,用作“模拟输入”功能,此时信号是不经过施密特触发器的,因为经过施密特触发器后信号只有 0、 1 两种状态,所以 ADC 外设要采集到原始的模拟信号,信号源输入必须在施密特触发器之前。类似地,当 GPIO 引脚用于 DAC 作为模拟电压输出通道时,此时作为“模拟输出”功能, DAC 的模拟信号输出就不经过双 MOS 管结构了,在 GPIO 结构框图的右下角处,模拟信号直接输出到引脚。同时,当 GPIO 用于模拟功能时(包括输入输出),引脚的上、下拉电阻是不起作用的,这个时候即使在寄存器配置了上拉或下拉模式,也不会影响到模拟信号的输入输出。
5.2.2 GPIO 工作模式:
总结一下,由 GPIO 的结构决定了 GPIO 可以配置成以下模式:
1. 输入模式(上拉/下拉/浮空):
在输入模式时, 施密特触发器打开, 输出被禁止。 数据寄存器每隔 1 个 AHB1 时钟周期更新一次,可通过输入数据寄存器GPIOx_IDR 读取 I/O 状态。 其中 AHB1 的时钟如按默认配置一般为 180MHz。用于输入模式时,可设置为上拉、下拉或浮空模式。
2. 输出模式(推挽/开漏,上拉/下拉):
在输出模式中, 输出使能,推挽模式时双 MOS 管以方式工作,输出数据寄存器GPIOx_ODR 可控制 I/O 输出高低电平。开漏模式时,只有 N-MOS 管工作,输出数据寄存器可控制 I/O 输出高阻态或低电平。 输出速度可配置,有 2MHz25MHz50MHz100MHz 的选项。
此处的输出速度即 I/O 支持的高低电平状态最高切换频率,支持的频率越高,功耗越大,如果功耗要求不严格,把速度设置成最大即可。此时施密特触发器是打开的,即输入可用,通过输入数据寄存器 GPIOx_IDR 可读取I/O 的实际状态。
用于输出模式时,可使用上拉、 下拉模式或浮空模式。但此时由于输出模式时引脚电平会受到 ODR 寄存器影响,而 ODR 寄存器对应引脚的位为 0,即引脚初始化后默认输出低电平,所以在这种情况下,上拉只起到小幅提高输出电流能力,但不会影响引脚的默认状态。
3. 复用功能(推挽/开漏,上拉/下拉):
复用功能模式中,输出使能,输出速度可配置,可工作在开漏及推挽模式,但是输出信号源于其它外设,输出数据寄存器 GPIOx_ODR 无效;输入可用,通过输入数据寄存器可获取 I/O 实际状态,但一般直接用外设的寄存器来获取该数据信号。
用于复用功能时,可使用上拉、 下拉模式或浮空模式。同输出模式,在这种情况下,初始化后引脚默认输出低电平,上拉只起到小幅提高输出电流能力,但不会影响引脚的默认状态。
4. 模拟输入输出:
模拟输入输出模式中,双 MOS 管结构被关闭,施密特触发器停用,上/下拉也被禁止。其它外设通过模拟通道进行输入输出。
通过对 GPIO 寄存器写入不同的参数,就可以改变 GPIO 的应用模式,再强调一下,要了解具体寄存器时一定要查阅《STM32F4xx 参考手册》中对应外设的寄存器说明。在GPIO 外设中,通过设置“模式寄存器 GPIOx_MODER”可配置 GPIO 的输入/输出/复用/模拟模式,“输出类型寄存器 GPIOx_OTYPER”配置推挽/开漏模式,配置“输出速度寄存器 GPIOx_OSPEEDR”可选 2/25/50/100MHz 输出速度,“上/下拉寄存器 GPIOx_PUPDR”可配置上拉/下拉/浮空模式,各寄存器的具体参数值见表 5-1。
5.3 实验:使用寄存器点亮 LED 灯:
本小节中,我们以实例讲解如何通过控制寄存器来点亮 LED 灯。此处侧重于讲解原理,请您直接用 KEIL5 软件打开我们提供的实验例程配合阅读,先了解原理,学习完本小节后,再尝试自己建立一个同样的工程。本节配套例程名称为“GPIO 输出—寄存器点亮 LED灯”,在工程目录下找到后缀为“.uvprojx”的文件,用 KEIL5 打开即可。
自己尝试新建工程时,请对照查阅《用 KEIL5 新建工程模版 寄存器版本》章节。
若没有安装 KEIL5 软件,请参考《如何安装 KEIL5》章节。
打开该工程,可看到一共有三个文件,分别 startup_stm32f429_439xx.s 、stm32f4xx.h 以及 main.c,下面我们对这三个工程进行讲解。
5.3.1 硬件连接:
在本教程中 STM32 芯片与 LED 灯的连接见下图。
图中从 3 个 LED 灯的阳极引出连接到 3.3V 电源,阴极各经过 1 个电阻引入至 STM32的 3 个 GPIO 引脚 PH10、 PH11、 PH12 中,所以我们只要控制这三个引脚输出高低电平,即可控制其所连接 LED 灯的亮灭。如果您的实验板 STM32 连接到 LED 灯的引脚或极性不一样,只需要修改程序到对应的 GPIO 引脚即可,工作原理都是一样的。
我们的目标是把 GPIO 的引脚设置成推挽输出模式并且默认下拉,输出低电平,这样就能让 LED 灯亮起来了。
5.3.2 启动文件:
名为“startup_stm32f429_439xx.s”的文件,它里边使用汇编语言写好了基本程序,当STM32 芯片上电启动的时候,首先会执行这里的汇编程序,从而建立起 C 语言的运行环境,所以我们把这个文件称为启动文件。该文件使用的汇编指令是 Cortex-M4 内核支持的指令,可从《Cortex-M4 Technical Reference Manual》查到,也可参考《Cortex-M3 权威指南中文》, M3 跟 M4 大部分汇编指令相同。
startup_stm32f429_439xx.s 文件是由官方提供的,一般有需要也是在官方的基础上修改,不会自己完全重写。该文件可以从 KEIL5 安装目录找到,也可以从 ST 库里面找到,找到该文件后把启动文件添加到工程里面即可。不同型号的芯片以及不同编译环境下使用的汇编文件是不一样的,但功能相同。
对于启动文件这部分我们主要总结它的功能,不详解讲解里面的代码,其功能如下:
初始化堆栈指针 SP;
初始化程序计数器指针 PC;
设置堆、栈的大小;
设置中断向量表的入口地址;
配置外部 SRAM 作为数据存储器(这个由用户配置,一般的开发板可没有外部SRAM) ;
调用 SystemIni() 函数配置 STM32 的系统时钟。
设置 C 库的分支入口“__main”(最终用来调用 main 函数) ;
先去除繁枝细节,挑重点的讲,主要理解最后两点,在启动文件中有一段复位后立即执行的程序,代码见代码清单5-1。在实际工程中阅读时,可使用编辑器的搜索(Ctrl+F)功能查找这段代码在文件中的位置。
代码清单 5-1 复位后执行的程序
;Reset handler
Reset_Handler PROC
EXPORT Reset_Handler [WEAK]
IMPORT SystemInit
IMPORT __main
LDR R0, =SystemInit
BLX R0
LDR R0, =__main
BX R0
ENDP
开头的是程序注释,在汇编里面注释用的是“;”,相当于 C 语言的“//”注释符
第二行是定义了一个子程序: Reset_Handler。 PROC 是子程序定义伪指令。这里就相当于 C 语言里定义了一个函数,函数名为 Reset_Handler。
第三行 EXPORT 表示 Reset_Handler 这个子程序可供其他模块调用。 相当于 C 语言的函数声明。关键字[WEAK] 表示弱定义,如果编译器发现在别处定义了同名的函数,则在链接时用别处的地址进行链接,如果其它地方没有定义,编译器也不报错,以此处地址进行链接,如果不理解 WEAK,那就忽略它好了。
第四行和第五行 IMPORT 说明 SystemInit 和__main 这两个标号在其他文件,在链接的时候需要到其他文件去寻找。相当于 C 语言中,从其它文件引入函数声明。以便下面对外部函数进行调用。
SystemInit 需要由我们自己实现,即我们要编写一个具有该名称的函数,用来初始化STM32 芯片的时钟,一般包括初始化 AHB、 APB 等各总线的时钟,需要经过一系列的配置 STM32 才能达到稳定运行的状态。
__main 其实不是我们定义的(不要与 C 语言中的 main 函数混淆),当编译器编译时,只要遇到这个标号就会定义这个函数,该函数的主要功能是:负责初始化栈、堆,配置系统环境,准备好 C 语言并在最后跳转到用户自定义的 main 函数,从此来到 C 的世界。
第六行把 SystemInit 的地址加载到寄存器 R0。
第七行程序跳转到 R0 中的地址执行程序,即执行 SystemInit 函数的内容。
第八行把__main 的地址加载到寄存器 R0。
第九行程序跳转到 R0 中的地址执行程序,即执行__main 函数,执行完毕之后就去到我们熟知的 C 世界,进入 main 函数。
第十行表示子程序的结束。
总之,看完这段代码后,了解到如下内容即可:我们需要在外部定义一个 SystemInit函数设置 STM32 的时钟; STM32 上电后,会执行 SystemInit 函数,最后执行我们 C 语言中的 main 函数。
5.3.3 stm32f4xx.h 文件:
看完启动文件,那我们立即写 SystemInit 和 main 函数吧?别着急,定义好了SystemInit 函数和 main 我们又能写什么内容?连接 LED 灯的 GPIO 引脚,是要通过读写寄存器来控制的,就这样空着手,如何控制寄存器呢。在上一章,我们知道寄存器就是特殊的内存空间,可以通过指针操作访问寄存器。所以此处我们根据 STM32 的存储分配先定义好各个寄存器的地址,把这些地址定义都统一写在 stm32f4xx.h 文件中,见代码清单 5-2。
代码清单 5-2 外设地址定义
/*片上外设基地址 */
#define PERIPH_BASE ((unsigned int)0x40000000)
/*总线基地址 */
#define AHB1PERIPH_BASE (PERIPH_BASE + 0x00020000)
/*GPIO 外设基地址*/
#define GPIOH_BASE (AHB1PERIPH_BASE + 0x1C00)
/* GPIOH 寄存器地址,强制转换成指针 */
#define GPIOH_MODER *(unsigned int*)(GPIOH_BASE+0x00)
#define GPIOH_OTYPER *(unsigned int*)(GPIOH_BASE+0x04)
#define GPIOH_OSPEEDR *(unsigned int*)(GPIOH_BASE+0x08)
#define GPIOH_PUPDR *(unsigned int*)(GPIOH_BASE+0x0C)
#define GPIOH_IDR *(unsigned int*)(GPIOH_BASE+0x10)
#define GPIOH_ODR *(unsigned int*)(GPIOH_BASE+0x14)
#define GPIOH_BSRR *(unsigned int*)(GPIOH_BASE+0x18)
#define GPIOH_LCKR *(unsigned int*)(GPIOH_BASE+0x1C)
#define GPIOH_AFRL *(unsigned int*)(GPIOH_BASE+0x20)
#define GPIOH_AFRH *(unsigned int*)(GPIOH_BASE+0x24)
/*RCC 外设基地址*/
#define RCC_BASE (AHB1PERIPH_BASE + 0x3800)
/*RCC 的 AHB1 时钟使能寄存器地址,强制转换成指针*/
#define RCC_AHB1ENR *(unsigned int*)(RCC_BASE+0x30)
GPIO 外设的地址跟上一章讲解的相同,不过此处把寄存器的地址值都直接强制转换成了指针,方便使用。代码的最后两段是 RCC 外设寄存器的地址定义, RCC 外设是用来设置时钟的,以后我们会详细分析,本实验中只要了解到使用 GPIO 外设必须开启它的时钟即可。
5.3.4 main 文件:
现在就可以开始编写程序了,在 main 文件中先编写一个 main 函数,里面什么都没有,暂时为空.
int main (void)
{
}
此时直接编译的话,会出现如下错误:
“Error: L6218E: Undefined symbol SystemInit (referred from startup_stm32f429_439xx.o)”
错误提示 SystemInit 没有定义。从分析启动文件时我们知道, Reset_Handler 调用了该函数用来初始化 SMT32 系统时钟,为了简单起见,我们在 main 文件里面定义一个SystemInit 空函数,什么也不做,为的是骗过编译器,把这个错误去掉。关于配置系统时钟我们在后面再写。当我们不配置系统时钟时, STM32 芯片会自动按系统内部的默认时钟运行,程序还是能跑的。我们在 main 中添加如下函数:
上一篇:再造STM32---第六部分:自己写库—构建库函数雏形
下一篇:再造STM32---第四部分:新建工程---寄存器版
推荐阅读最新更新时间:2024-11-07 11:14
设计资源 培训 开发板 精华推荐
- NCP562 80 mA CMOS 低 Iq 低压降稳压器的典型应用
- 具有手动复位按钮的 LTC1727EMS8-2.5 三路电源监视器的典型应用
- 使用 Richtek Technology Corporation 的 RT2515A 的参考设计
- AZ7023ZTR低压指示检测器典型应用电路
- #第三届立创大赛#STM32/STM8离线下载器
- LT1587CT 3A 低压差稳压器的典型应用
- LT1074IT 负升压转换器的典型应用
- 1800 - 2500 MHz 接收器应用中的 BGA622 硅锗通用低噪声放大器 MMIC 配置 C
- ADL5545-EVALZ,基于 ADL5545 30 MHz 至 6 GHz RF/IF 增益模块的评估板
- LTC6401-20 的典型应用 - 用于 140MHz IF 的 1.3GHz 低噪声、低失真差分 ADC 驱动器
- 科学家研发基于AI的身份验证工具 可保护车辆免受网络攻击威胁
- Microchip推出广泛的IGBT 7 功率器件组合,专为可持续发展、电动出行和数据中心应用而设计
- 面向未来驾驶体验 博世推出新型微电子技术
- 英飞凌与马瑞利合作 利用AURIX™ TC4x MCU系列推动区域控制单元创新
- 5C超充,该怎么卷?
- 《2025年度中国汽车十大技术趋势》正式揭晓!你最看好哪个?
- Microchip推出新型VelocityDRIVE™软件平台和车规级多千兆位以太网交换芯片,支持软件定义汽车
- 英特尔中国正式发布2023-2024企业社会责任报告
- can转485数据是如何对应的
- MCU今年的重点:NPU和64位