
1 GPIO简介 GPIO,即通用I/O(输入/输出)端口,是STM32可控制的引脚。STM32芯片的GPIO引脚与外部设备连接起来,可实现与外部通讯、控制外部硬件或者采集外部硬件数据的功能。6 A/ P0 c5 P0 L4 O7 Z/ F4 m STM32F407有7组IO。分别为GPIOA~GPIOG,每组IO有16个IO口,共有112个IO口 通常称为 PAx、PBx、PCx、PDx、PEx、PFx、PGx,其中x为0-15。 并且F4系列是基于Cortex-M4内核 4 I/ b% V9 W3 `5 W3 J ! Q/ I5 s( [7 f/ {+ ?9 F# N C$ I6 E GPIO的复用:5 e* v3 Z+ W# K* U9 r9 q% j . C' b1 v" Z9 G- l: G# `/ u STM32F4 有很多的内置外设,这些外设的外部引脚都是与 GPIO 共用的。也就是说,一个引脚可以有很多作用,但是默认为IO口,如果想使用一个 GPIO内置外设的功能引脚,就需要GPIO的复用,那么当这个 GPIO 作为内置外设使用的时候,就叫做复用。 比如说串口 就是GPIO复用为串口2 z8 e, W2 z& x0 d3 {1 C 2 GPIO的工作模式$ o u( b5 \0 c* k: s \: {, R 1、4种输入模式 (1)GPIO_Mode_IN_FLOATING 浮空输入6 \8 i- c% G" s) {- d1 u: p (2)GPIO_Mode_IPU 上拉输入 (3)GPIO_Mode_IPD 下拉输入9 g5 L d3 m/ Q2 R. | (4)GPIO_Mode_AIN 模拟输入& l5 @, G6 D/ o; Q% X' L8 K 2、4种输出模式 4 B6 `* ]+ L* \8 N (5)GPIO_Mode_Out_OD 开漏输出(带上拉或者下拉)4 i* W8 x, P& b g( w! |- z (6)GPIO_Mode_AF_OD 复用开漏输出(带上拉或者下拉) (7)GPIO_Mode_Out_PP 推挽输出(带上拉或者下拉) (8)GPIO_Mode_AF_PP 复用推挽输出(带上拉或者下拉) 3、4种最大输出速度# s, ^7 h6 P# X' G O7 k (1)2MHZ (低速)' N9 G& z( S4 C& M" z (2)25MHZ (中速) (3)50MHZ (快速) ]7 P" l4 O& @3 H- k (4)100MHZ (高速)1 Q* `, ]; t4 O ) q) T5 A F. m' a' f- t6 c. d 关于他们的定义,都在 stm32f4xx_gpio.h 中,都为结构体形式的定义 % I: S* R8 C3 k 3 GPIO框图剖析. O5 r" ^0 }9 Y4 x7 D$ t ![]() # V1 k9 a$ g! f+ N* U4 o7 i 我们所用到的每一个GPIO其内部结构都是这样,分别对应着GPIO的八种模式 这里我们简单的介绍下: 保护二极管: IO引脚上下两边两个二极管用于防止引脚外部过高、过低的电压输入,当引脚电压高于VDD_FT时,上方的二极管导通,当引脚电压低于VSS时,下方的二极管导通,防止不正常电压引入芯片导致芯片烧毁 0 t; _: Y, d$ g# ` t 上拉、下拉电阻:控制引脚默认状态的电压,开启上拉的时候引脚默认电压为高电平,开启下拉的时候引脚默认电压为低电平, }! h# I/ a1 {, V/ M; }$ Q6 O TTL施密特触发器:基本原理是当输入电压高于正向阈值电压,输出为高;当输入电压低于负向阈值电压,输出为低;IO口信号经过触发器后,模拟信号转化为0和1的数字信号 也就是高低电平 并且是TTL电平协议 这也是为什么STM32是TTL电平协议的原因 P-MOS管和N-MOS管:信号由P-MOS管和N-MOS管,依据两个MOS管的工作方式,使得GPIO具有“推挽输出”和“开漏输出”的模式 P-MOS管高电平导通,低电平关闭,下方的N-MOS低电平导通,高电平关闭7 X9 h( t8 J% Y5 i8 \% C% a! t 注: VDD_FT 代表IO口,兼容3.3V和5V,如果没有标注“FT”,就代表着不兼容5V ) c j! W" N/ y; h& A 在芯片数据手册的引脚定义中,会看到有“I/O电平”一列 有FT即为支持5V ![]() 4 GPIO的八种工作模式剖析:) D) L& D2 j0 D! A0 r# l3 w 浮空输入模式 , t( N& R: X( ~! H/ D0 r ![]() " l! M1 ~$ h5 Q3 N 浮空输入模式下,I/O端口的电平信号直接进入输入数据寄存器。MCU直接读取I/O口电平,I/O的电平状态是不确定的,完全由外部输入决定;如果在该引脚悬空(在无信号输入)的情况下,读取该端口的电平是不确定的。 (接用电压表测量其引脚电压为1点几伏,这是个不确定值) 以用来做KEY识别 4 S# Z& T( z2 o7 j8 l3 L% Z5 I, I 上拉输入模式+ u L5 o" O1 v y) v9 V 9 P% i- P6 }$ Y$ p: \. Z, D ![]() IO内部接上拉电阻,此时如果IO口外部没有信号输入或者引脚悬空,IO口默认为高电平 如果I/O口输入低电平,那么引脚就为低电平,MCU读取到的就是低电平 9 ~# _4 F( O' Q1 i# p STM32的内部上拉是"弱上拉",即通过此上拉输出的电流是很弱的,如要求大电流还是需要外部上拉。 4 \3 L" K( u- ^0 q 下拉输入模式 $ m- i6 r/ \( b ![]() % J6 V' I( A3 w/ V! m! w/ Q2 Q IO内部接下拉电阻,此时如果IO口外部没有信号输入或者引脚悬空,IO口默认为低电平 如果I/O口输入高电平,那么引脚就为高电平,MCU读取到的就是高电平 5 U5 V: M! Z$ M( y 模拟输入模式" `, x5 h# F2 p / A2 A* r0 w, O3 \ ![]() 当GPIO引脚用于ADC采集电压的输入通道时,用作"模拟输入"功能,此时信号不经过施密特触发器,直接直接进入ADC模块,并且输入数据寄存器为空 ,CPU不能在输入数据寄存器上读到引脚状态 # h: {1 o. F- I7 U% H 当GPIO用于模拟功能时,引脚的上、下拉电阻是不起作用的,这个时候即使配置了上拉或下拉模式,也不会影响到模拟信号的输入输出' k5 k- `5 m8 V! I- w: p' @3 o 6 K- h; x4 v' Q M& ^6 y; h1 E L 除了 ADC 和 DAC 要将 IO 配置为模拟通道之外其他外设功能一律 要配置为复用功能模式,6 b( c4 h+ k( a* { [9 G) B: i# c. j# `2 o) {/ [ 开漏输出模式(带上拉或者下拉) ![]() 7 w) e5 m2 s- {, E0 p0 `: u8 F , _$ r% s+ X% M, K6 ]6 G 在开漏输出模式时,只有N-MOS管工作,如果我们控制输出为0,低电平,则P-MOS管关闭,N-MOS管导通,使输出低电平,I/O端口的电平就是低电平,若控制输出为1时,高电平,则P-MOS管和N-MOS管都关闭,输出指令就不会起到作用,此时I/O端口的电平就不会由输出的高电平决定,而是由I/O端口外部的上拉或者下拉决定 如果没有上拉或者下拉 IO口就处于悬空状态 / F3 Z; r# D/ e2 F3 [9 t 并且此时施密特触发器是打开的,即输入可用,通过输入数据寄存器GPIOx_IDR可读取I/O的实际状态。,I/O口的电平不一定是输出的电平 1 R2 G) H) b% M3 H+ l$ p, ? ![]() 在推挽输出模式时,N-MOS管和P-MOS管都工作,如果我们控制输出为0,低电平,则P-MOS管关闭,N-MOS管导通,使输出低电平,I/O端口的电平就是低电平,若控制输出为1 高电平,则P-MOS管导通N-MOS管关闭,使输出高电平,I/O端口的电平就是高电平, 外部上拉和下拉的作用是控制在没有输出时IO口电平- V/ K7 U9 L- l) r9 R- n; Z% y . d6 W: ?; w4 U7 o 此时施密特触发器是打开的,即输入可用,通过输入数据寄存器GPIOx_IDR可读取I/O的实际状态。I/O口的电平一定是输出的电平 " ?0 m0 O( D3 q5 H3 z/ N* `" T, ` 复用开漏输出(带上拉或者下拉) ![]() ( \7 x4 v+ m8 C GPIO复用为其他外设,输出数据寄存器GPIOx_ODR无效; 输出的高低电平的来源于其它外设,施密特触发器打开,输入可用,通过输入数据寄存器可获取I/O实际状态 除了输出信号的来源改变 其他与开漏输出功能相同4 }$ [; j* C& r0 E7 V' F: a 复用推挽输出(带上拉或者下拉) ![]() GPIO复用为其他外设(如 I2C),输出数据寄存器GPIOx_ODR无效; 输出的高低电平的来源于其它外设,施密特触发器打开,输入可用,通过输入数据寄存器可获取I/O实际状态 除了输出信号的来源改变 其他与开漏输出功能相同 开漏输出和推挽输出的区别:* u E0 M( Y+ o& [) \" z * E- g/ t3 } y9 i/ A% A 推挽输出:, P: d1 Z: @& n- H6 W 6 @2 P! M/ c. |: {; u; { 可以输出强高低电平,连接数字器件 . V# `( p/ _6 v% S 推挽结构一般是指两个三极管分别受两互补信号的控制,总是在一个三极管导通的时候另一个截止.! d/ o- ]2 p0 d. Y- ^; a 5 z- K& Y6 ?% O4 T2 } 开漏输出:7 T( Q2 d/ u, l! \9 o/ H # h' Q& c0 z: Y# p; ] 可以输出强低电平,高电平得靠外部电阻拉高。输出端相当于三极管的集电极. 需要外接上拉电阻,才能实现输出高电平 合于做电流型的驱动,其吸收电流的能力相对强(一般20ma以内); . K1 B1 e2 \" ?, | 在使用任何一种开漏模式时,都需要接上拉电阻,否则只能输出低电平9 U- N% b+ I/ X7 M2 [ + _4 c1 H& @/ s j1 N) U ![]() 推挽输出电路: 其中IN端输出高电平时下面的PNP三极管截止,而上面NPN三极管导通,输出电平VS+;当IN端输出低电平时则恰恰相反,PNP三极管导通,输出和地相连,为低电平 ! X0 Y" E9 o+ N+ c8 D4 K: b3 e# P4 [ 6 i9 H0 H4 r( x9 H) ? 开漏输出电路:IN端输出低电平时,三极管导通,使输出接地,IN端输出高电平时,三极管截止,所以引脚既不输出高电平,也不输出低电平,为高阻态。为正常使用时必须接上拉电阻, 3 ^! h& f- j* i/ P1 c; ]6 o* `# | 在STM32中选用IO模式: p& ~! i9 r7 f/ L0 D+ P 上拉输入、下拉输入可以用来检测外部信号;例如,按键等; 模拟输入 ——应用ADC模拟输入,或者低功耗下省电 开漏输出一般应用在I2C、SMBUS通讯等需要"线与"功能的总线电路中。% S2 ?9 E1 [" h m o" l 推挽输出模式一般应用在输出电平为0和3.3伏而且需要高速切换开关状态的场合。在STM32的应用中,除了必须用开漏模式的场合,我们都习惯使用推挽输出模式。 复用功能的推挽输出_AF_PP ——片内外设功能(I2C的SCL,SDA), T& l+ O8 K/ M9 @* G# d, p1 e& k. j 复用功能的开漏输出_AF_OD——片内外设功能(TX1,MOSI,MISO.SCK.SS)( ?. p7 ^" C4 q$ ^ F4系列与F1系列区别: 本质上的区别是F4系列采用了Cortex-M4内核 而F1系列采用Cortex-M3内核) e/ n4 R. | R" [1 x- B F1系列(M3)IO口基本结构: ![]() $ s% @+ p, p: S2 m( x8 L- H F4系列(M4)IO口基本结构: ![]() F4系列设计的更加高级与人性化,他将外部上下拉电阻转移到了输出/输入驱动器外部,使得输出模式下也可以实现内部上拉与下拉,方便了用户的使用,增加了灵活性& Q( m0 c9 q# m/ s ? GPIO的初始化(F4) 这里我们以初始化LED为例$ k: O7 ?3 m2 n5 |8 V" w D 1.定义一个 GPIO_InitTypeDef 类型的结构体$ a' M: V* A, }- Z3 R! z
: o& E7 g0 ?8 e: k. a7 w ![]() 5 g- _, x! @( o: y7 M( ~: ^& i 1 S: A% a0 S J p 2开启 LED 相关的 GPIO 外设时钟
Q:为什么要设置时钟? 任何外设都需要时钟,51单片机,stm32,430等等,因为寄存器是由D触发器组成的,往触发器里面写东西,前提条件是有时钟输入。stm32是低功耗,他将所有的门都默认设置为disable(不使能),在你需要用哪个门的时候,开哪个门就可以,也就是说用到什么外设,只要打开对应外设的时钟就可以, 其他的没用到的可以还是disable(不使能),这样耗能就会减少。 5 s/ R" h$ \* q, o Q:为什么 STM32 要有多个时钟源呢?1 s! m! J+ s9 J3 t9 n & d/ w# p* I: A/ K) |: ` 因为首 先 STM32 本身非常复杂,外设非常的多,但是并不是所有外设都需要系统时钟这么高的频率, 比如看门狗以及 RTC 只需要几十 k 的时钟即可。同一个电路,时钟越快功耗越大,同时抗电磁 干扰能力也会越弱,所以对于较为复杂的 MCU 一般都是采取多时钟源的方法来解决这些问题。' T9 }; h7 R4 p( x. O. y0 U 而相对应的外设功能所使用的时钟 在stm32f4xx.h 中即可查看到 RCC_AHB1# q- o9 T9 z; v1 o7 C- I ; `6 A) |$ e6 y' E ![]() RCC_APB1 ( g7 ]8 g( Z+ t; v6 i }" T ![]() 1 g% F! y1 D8 z `" c8 {4 l: t 3选择要控制的 GPIO 引脚 " q' x- s) l0 l/ |1 ?
可选引脚为0-15 一组IO口有16个引脚" C0 `4 l" j! [" b ![]() 4设置所选引脚的模式
引脚的模式共有四种,分别为输入,输出,复用,和模拟模式2 Q; J' k6 c; c' Z' s" b ![]() ) J8 r0 `1 E' R0 R" j" L 5 设定所选引脚的输出类型" d# a8 H: }, W7 }5 Y+ n9 ~7 ? * @: B! f* B/ Y' h0 E1 g
输出模式有两种:推挽输出和开漏输出 $ O; g# `% d; P6 X, I6 G* } # }) r$ X; E! r7 l, B6 Q3 k% u 只有输出模式才需要配置,输入模式下不需要配置+ }: E4 T) R+ Y' M+ G ^( M- i2 `, Y 6 设定所选管脚的速度2 Y2 d+ n- K3 P2 d
![]() + B- c% ^2 X' Q ! L' q- A2 m4 _8 `; _/ v9 X$ e+ C 7 设定所选管脚的上拉与下拉, j+ I. X' p9 `# O4 V. I3 f. i) s . n( a: q: d: D5 O
可设置为:上拉,下拉,与浮空 2 {* P1 ]0 }' N ![]() ( t# F8 l: \1 C0 E6 i 8初始化GPIO" ]- d7 J$ a3 q( F% ]% B
GPIO_Init() 是官方配置的初始化函数 第一个参数是GPIOX 第二个参数是结构体所对应GPIO各种参数的配置 ^' @) W- }; H2 q GPIO的初始化(F1) F4的初始化相较于F1系列有了很大的改变,在配置F1系列时,结构体只有三个参数 , [5 ]& ]* J/ D; m; `6 X GPIO_InitTypeDef 类型的结构体$ d, g: l5 q. p) P: l' R- U! Y. V( A5 ]: P ![]() 9 T. x5 x: d+ o; K/ K+ j 所选管脚的速度 ![]() , _3 c" }5 q# O2 z 所选管脚的8种模式7 A1 s. [( A/ X, D( N5 @ H ![]() 区别: 9 ?2 \ t* `) Y. F: j F1(M-3)系列管脚速度只有三种模式 并且管脚的模式配置八种模式全部都在一起定义,直接设置即可 ) z# n$ Y% u& ^ 总归还是大同小异,不做过多介绍,看下下方配置即可, `+ K( J' _! f N" a0 ^( B( a7 M/ y
: `" S1 j% J! A/ A0 ~8 Y |
Nucleo-F429ZI开发板体验LWIP功能
TouchGFX应用移植自定义屏幕
STM32固件库分享,超全系列整理
【经验分享】新建基于STM32F40x 固件库的MDK5 工程
年终中奖---F429板卡网络应用
【NUCLEO STM32F429ZI开发板】之点灯大师
OpenBLT移植到STM32F405开发板
串口通信波特率异常的解决办法
《RT-Thread设备驱动开发指南》书籍学习记录
STM32F407/STM32F417通过以太网实现在应用中编程 (IAP)