
1 GPIO简介6 _, T! e7 D( H GPIO,即通用I/O(输入/输出)端口,是STM32可控制的引脚。STM32芯片的GPIO引脚与外部设备连接起来,可实现与外部通讯、控制外部硬件或者采集外部硬件数据的功能。 STM32F407有7组IO。分别为GPIOA~GPIOG,每组IO有16个IO口,共有112个IO口 通常称为 PAx、PBx、PCx、PDx、PEx、PFx、PGx,其中x为0-15。 并且F4系列是基于Cortex-M4内核 # u" R: Y" p6 ~+ x( Q GPIO的复用:7 j4 m, g! J. _ STM32F4 有很多的内置外设,这些外设的外部引脚都是与 GPIO 共用的。也就是说,一个引脚可以有很多作用,但是默认为IO口,如果想使用一个 GPIO内置外设的功能引脚,就需要GPIO的复用,那么当这个 GPIO 作为内置外设使用的时候,就叫做复用。 比如说串口 就是GPIO复用为串口6 i( t7 }2 @# ?" b. `# L 2 GPIO的工作模式9 i) p: }; }9 X, ?; A 1、4种输入模式 (1)GPIO_Mode_IN_FLOATING 浮空输入& [2 _1 @% a* E6 _ (2)GPIO_Mode_IPU 上拉输入 (3)GPIO_Mode_IPD 下拉输入5 |1 ]) k: f. ?# }& s (4)GPIO_Mode_AIN 模拟输入 2、4种输出模式 (5)GPIO_Mode_Out_OD 开漏输出(带上拉或者下拉)* X/ F: F" z4 Z( L (6)GPIO_Mode_AF_OD 复用开漏输出(带上拉或者下拉) (7)GPIO_Mode_Out_PP 推挽输出(带上拉或者下拉)$ K2 I5 o3 y2 ~% M (8)GPIO_Mode_AF_PP 复用推挽输出(带上拉或者下拉)5 Z; }6 }& X1 k/ u0 i : _1 [+ S; b8 _) S1 E 3、4种最大输出速度 (1)2MHZ (低速)! @; h7 ~" ]0 }7 p+ ^; O$ B8 } (2)25MHZ (中速) (3)50MHZ (快速) (4)100MHZ (高速) C+ g" ~+ U" D, }7 Q9 ? 关于他们的定义,都在 stm32f4xx_gpio.h 中,都为结构体形式的定义 - H* G6 x# ^* h: D! g5 a8 H 3 GPIO框图剖析 ![]() + Y; r& F) C# T7 R, Z* { 我们所用到的每一个GPIO其内部结构都是这样,分别对应着GPIO的八种模式 这里我们简单的介绍下: 5 D+ {, f* y* ^% z 保护二极管: IO引脚上下两边两个二极管用于防止引脚外部过高、过低的电压输入,当引脚电压高于VDD_FT时,上方的二极管导通,当引脚电压低于VSS时,下方的二极管导通,防止不正常电压引入芯片导致芯片烧毁 / [4 c& c9 Y2 X 上拉、下拉电阻:控制引脚默认状态的电压,开启上拉的时候引脚默认电压为高电平,开启下拉的时候引脚默认电压为低电平 TTL施密特触发器:基本原理是当输入电压高于正向阈值电压,输出为高;当输入电压低于负向阈值电压,输出为低;IO口信号经过触发器后,模拟信号转化为0和1的数字信号 也就是高低电平 并且是TTL电平协议 这也是为什么STM32是TTL电平协议的原因- T( z+ H- U4 Z+ W/ q P-MOS管和N-MOS管:信号由P-MOS管和N-MOS管,依据两个MOS管的工作方式,使得GPIO具有“推挽输出”和“开漏输出”的模式 P-MOS管高电平导通,低电平关闭,下方的N-MOS低电平导通,高电平关闭& u# F1 ?2 m m4 x 注: VDD_FT 代表IO口,兼容3.3V和5V,如果没有标注“FT”,就代表着不兼容5V 8 O$ m2 P$ X0 v' a% k: c 在芯片数据手册的引脚定义中,会看到有“I/O电平”一列 有FT即为支持5V3 K1 g2 k7 C. ` , S1 x% a" w0 p( B1 y: \# W9 \ ![]() " B! v0 N+ g7 q 4 GPIO的八种工作模式剖析:( P) d$ n7 R9 v H. | 浮空输入模式 ![]() 4 y* t* f# B% P* A" _+ L* ?; h d( K u 浮空输入模式下,I/O端口的电平信号直接进入输入数据寄存器。MCU直接读取I/O口电平,I/O的电平状态是不确定的,完全由外部输入决定;如果在该引脚悬空(在无信号输入)的情况下,读取该端口的电平是不确定的。 (接用电压表测量其引脚电压为1点几伏,这是个不确定值) 以用来做KEY识别 上拉输入模式$ u. Q5 q+ k _0 U# d 2 A/ j. _7 G3 S$ U4 j5 m$ O ![]() + i1 p0 W1 Q M IO内部接上拉电阻,此时如果IO口外部没有信号输入或者引脚悬空,IO口默认为高电平 如果I/O口输入低电平,那么引脚就为低电平,MCU读取到的就是低电平' t: x" o3 m( ~& d, Z. Z 2 S: v1 W3 u; _) e* `8 b$ M STM32的内部上拉是"弱上拉",即通过此上拉输出的电流是很弱的,如要求大电流还是需要外部上拉。) ^7 A; L5 q0 E- b2 F# |; a$ l & o: u# E0 A) z3 ]% A: p 下拉输入模式 U; _6 [# H. J6 S ; ^$ r3 f( B( s. r/ V7 |" F, ` ![]() 9 z# ?4 P: Y* w' K3 L; @, ]3 a6 y IO内部接下拉电阻,此时如果IO口外部没有信号输入或者引脚悬空,IO口默认为低电平 如果I/O口输入高电平,那么引脚就为高电平,MCU读取到的就是高电平2 f1 y* P6 [: U6 {% {& } 模拟输入模式 ![]() 当GPIO引脚用于ADC采集电压的输入通道时,用作"模拟输入"功能,此时信号不经过施密特触发器,直接直接进入ADC模块,并且输入数据寄存器为空 ,CPU不能在输入数据寄存器上读到引脚状态 ! A' v h4 {% t& b2 j 当GPIO用于模拟功能时,引脚的上、下拉电阻是不起作用的,这个时候即使配置了上拉或下拉模式,也不会影响到模拟信号的输入输出* ^* }+ {$ l: r3 ~ Q, t 除了 ADC 和 DAC 要将 IO 配置为模拟通道之外其他外设功能一律 要配置为复用功能模式,; `( {, _) ^' U+ b : E1 y2 t: o" W8 K* a 开漏输出模式(带上拉或者下拉)% ^. T: m G6 I1 i: _! h* c/ h ![]() 1 O2 ~) ]2 S9 J0 H% v 在开漏输出模式时,只有N-MOS管工作,如果我们控制输出为0,低电平,则P-MOS管关闭,N-MOS管导通,使输出低电平,I/O端口的电平就是低电平,若控制输出为1时,高电平,则P-MOS管和N-MOS管都关闭,输出指令就不会起到作用,此时I/O端口的电平就不会由输出的高电平决定,而是由I/O端口外部的上拉或者下拉决定 如果没有上拉或者下拉 IO口就处于悬空状态 并且此时施密特触发器是打开的,即输入可用,通过输入数据寄存器GPIOx_IDR可读取I/O的实际状态。,I/O口的电平不一定是输出的电平 ![]() # j( Z5 ~5 u O- O' z 在推挽输出模式时,N-MOS管和P-MOS管都工作,如果我们控制输出为0,低电平,则P-MOS管关闭,N-MOS管导通,使输出低电平,I/O端口的电平就是低电平,若控制输出为1 高电平,则P-MOS管导通N-MOS管关闭,使输出高电平,I/O端口的电平就是高电平, 外部上拉和下拉的作用是控制在没有输出时IO口电平. U& W+ F% C; P" W4 q 此时施密特触发器是打开的,即输入可用,通过输入数据寄存器GPIOx_IDR可读取I/O的实际状态。I/O口的电平一定是输出的电平 , o4 r/ h0 g, [. p( x2 ^0 P# G 0 |: [+ G6 q0 M' r( C 复用开漏输出(带上拉或者下拉) ![]() % o. R; I* _2 O6 i8 J GPIO复用为其他外设,输出数据寄存器GPIOx_ODR无效; 输出的高低电平的来源于其它外设,施密特触发器打开,输入可用,通过输入数据寄存器可获取I/O实际状态 除了输出信号的来源改变 其他与开漏输出功能相同 # @& ~/ |, j5 r4 o' O- d8 `" `; o 复用推挽输出(带上拉或者下拉) 5 n" u' z1 @, |. t: L ![]() GPIO复用为其他外设(如 I2C),输出数据寄存器GPIOx_ODR无效; 输出的高低电平的来源于其它外设,施密特触发器打开,输入可用,通过输入数据寄存器可获取I/O实际状态 除了输出信号的来源改变 其他与开漏输出功能相同 开漏输出和推挽输出的区别: + \& ^) t6 L; |. y' B, s3 q7 | 推挽输出:: x2 x2 K2 i1 ~3 E V 7 g" T9 m+ k$ q3 V, Z; x% R: _# W 可以输出强高低电平,连接数字器件 推挽结构一般是指两个三极管分别受两互补信号的控制,总是在一个三极管导通的时候另一个截止.; {! t! T% i' ^. r 3 f0 P) B4 B( Y8 k 开漏输出: 7 R- q- l3 Z9 v( v9 ^4 a 可以输出强低电平,高电平得靠外部电阻拉高。输出端相当于三极管的集电极. 需要外接上拉电阻,才能实现输出高电平 合于做电流型的驱动,其吸收电流的能力相对强(一般20ma以内);2 e ?: W2 m% }3 P/ B9 }6 d ! k/ T' A; j0 d- h+ S+ R# J! w! S 在使用任何一种开漏模式时,都需要接上拉电阻,否则只能输出低电平* i; N0 F: e9 j$ p ![]() * F4 q) J7 W; G- u0 W9 d 推挽输出电路: 其中IN端输出高电平时下面的PNP三极管截止,而上面NPN三极管导通,输出电平VS+;当IN端输出低电平时则恰恰相反,PNP三极管导通,输出和地相连,为低电平 - a" y t# G7 Q" r2 l 开漏输出电路:IN端输出低电平时,三极管导通,使输出接地,IN端输出高电平时,三极管截止,所以引脚既不输出高电平,也不输出低电平,为高阻态。为正常使用时必须接上拉电阻,7 G* k1 }% n& W* l6 y4 G ' N" D1 g: F. o' K. D4 f 在STM32中选用IO模式: 上拉输入、下拉输入可以用来检测外部信号;例如,按键等; 模拟输入 ——应用ADC模拟输入,或者低功耗下省电 开漏输出一般应用在I2C、SMBUS通讯等需要"线与"功能的总线电路中。' c$ Z7 q; L) ? ^* J 推挽输出模式一般应用在输出电平为0和3.3伏而且需要高速切换开关状态的场合。在STM32的应用中,除了必须用开漏模式的场合,我们都习惯使用推挽输出模式。0 x! E2 ^& B3 J/ Y: z 复用功能的推挽输出_AF_PP ——片内外设功能(I2C的SCL,SDA)' d& Q: a' d- I% y 复用功能的开漏输出_AF_OD——片内外设功能(TX1,MOSI,MISO.SCK.SS)0 }3 H. S7 R W% o) t1 X7 O5 c F4系列与F1系列区别:1 Y! ^0 x5 t) l% ?; l 本质上的区别是F4系列采用了Cortex-M4内核 而F1系列采用Cortex-M3内核 F1系列(M3)IO口基本结构: ![]() F4系列(M4)IO口基本结构:5 S5 C/ O: i4 {( _8 _ ![]() 4 M' m: Y! P$ o z F4系列设计的更加高级与人性化,他将外部上下拉电阻转移到了输出/输入驱动器外部,使得输出模式下也可以实现内部上拉与下拉,方便了用户的使用,增加了灵活性 ]- l* g5 p2 S' n GPIO的初始化(F4)" Y; o/ b/ E7 V/ f 这里我们以初始化LED为例& h) Q/ B2 I1 D& h4 R/ d4 D 1.定义一个 GPIO_InitTypeDef 类型的结构体+ d6 M/ r0 Q* Y8 B$ c0 P+ l' }! o9 z 5 x/ S. @9 \: Q+ D$ j) H1 T& q6 ?3 Z
$ K5 R5 E, W; X) y ![]() - ]7 V- Q) ~2 T8 s, J $ l) t6 l; j4 e8 E- V5 O 2开启 LED 相关的 GPIO 外设时钟0 b; X8 K5 x; j9 r7 M Z% }* f $ F2 O; T: v5 {$ M) ?: F
Q:为什么要设置时钟?3 ?8 L# t. D% Q; ? _& j3 P K# Y 任何外设都需要时钟,51单片机,stm32,430等等,因为寄存器是由D触发器组成的,往触发器里面写东西,前提条件是有时钟输入。stm32是低功耗,他将所有的门都默认设置为disable(不使能),在你需要用哪个门的时候,开哪个门就可以,也就是说用到什么外设,只要打开对应外设的时钟就可以, 其他的没用到的可以还是disable(不使能),这样耗能就会减少。 * @- J* |/ D+ _7 F% U+ x Q:为什么 STM32 要有多个时钟源呢? 因为首 先 STM32 本身非常复杂,外设非常的多,但是并不是所有外设都需要系统时钟这么高的频率, 比如看门狗以及 RTC 只需要几十 k 的时钟即可。同一个电路,时钟越快功耗越大,同时抗电磁 干扰能力也会越弱,所以对于较为复杂的 MCU 一般都是采取多时钟源的方法来解决这些问题。& g. @2 c3 b+ F0 l2 S, ] 而相对应的外设功能所使用的时钟 在stm32f4xx.h 中即可查看到* `) E: ~0 k$ Q ]; r9 ?% T& S( q RCC_AHB1 ![]() 9 w( u; v, X# Q& H, [3 R3 i v RCC_APB1 ![]() 3选择要控制的 GPIO 引脚3 _2 Z6 c2 S4 ? i8 b- J
可选引脚为0-15 一组IO口有16个引脚 . e) ^9 R1 e3 v2 X6 L r6 ~ O, i ![]() * e c5 U, x8 S6 T4 |* G 4设置所选引脚的模式+ v, `7 f, m8 D0 R4 v5 T
引脚的模式共有四种,分别为输入,输出,复用,和模拟模式% Z$ [) B# K0 o : w$ D: k$ c! b; A6 s3 j ![]() . h; x7 u8 T. K* `6 P 5 设定所选引脚的输出类型
输出模式有两种:推挽输出和开漏输出 ' R; K) r1 |, y & j. @9 D' I# N# x* x) p) T- v. z! m$ O8 d" N 只有输出模式才需要配置,输入模式下不需要配置* V# _( V; O% ?% @ ; h1 x7 o" [8 D6 [ 6 设定所选管脚的速度
![]() ' N0 n# j9 ^; A% \2 ^ 7 设定所选管脚的上拉与下拉 : Q3 c$ _8 c1 k
可设置为:上拉,下拉,与浮空 ![]() 8初始化GPIO
GPIO_Init() 是官方配置的初始化函数 第一个参数是GPIOX 第二个参数是结构体所对应GPIO各种参数的配置 GPIO的初始化(F1). _- w/ g u! Z* d) R) t F4的初始化相较于F1系列有了很大的改变,在配置F1系列时,结构体只有三个参数 # Z1 m. h" S. Y; ?: k GPIO_InitTypeDef 类型的结构体 1 @6 V! B& e: n6 I- E# w& T; K ![]() 所选管脚的速度 ![]() 所选管脚的8种模式 ![]() 区别: , b8 X4 A' f( T4 ~) T ^" D. Y3 s F1(M-3)系列管脚速度只有三种模式 并且管脚的模式配置八种模式全部都在一起定义,直接设置即可 2 Q. t; w8 y9 c3 o8 i 总归还是大同小异,不做过多介绍,看下下方配置即可 " [/ i) q) c9 h- @( @0 `
+ d+ ?- i4 o3 t/ u( k( i$ O; E ( H: l0 F+ `% k. K2 ] |