
1 GPIO简介 GPIO,即通用I/O(输入/输出)端口,是STM32可控制的引脚。STM32芯片的GPIO引脚与外部设备连接起来,可实现与外部通讯、控制外部硬件或者采集外部硬件数据的功能。 8 b* O" T" K& o1 C STM32F407有7组IO。分别为GPIOA~GPIOG,每组IO有16个IO口,共有112个IO口 通常称为 PAx、PBx、PCx、PDx、PEx、PFx、PGx,其中x为0-15。 并且F4系列是基于Cortex-M4内核 # e0 s& ?( a4 S; {+ R; }# y+ M GPIO的复用: STM32F4 有很多的内置外设,这些外设的外部引脚都是与 GPIO 共用的。也就是说,一个引脚可以有很多作用,但是默认为IO口,如果想使用一个 GPIO内置外设的功能引脚,就需要GPIO的复用,那么当这个 GPIO 作为内置外设使用的时候,就叫做复用。 比如说串口 就是GPIO复用为串口 1 H7 H! ~, f7 b2 D! H5 \" P 2 GPIO的工作模式4 ^2 D! _) h Q* s+ \) J 1、4种输入模式3 X% o! a! ?, a) e, o* `0 @ (1)GPIO_Mode_IN_FLOATING 浮空输入. ~) I8 O$ t& ?; F( ] (2)GPIO_Mode_IPU 上拉输入7 w F8 B. m5 j (3)GPIO_Mode_IPD 下拉输入 (4)GPIO_Mode_AIN 模拟输入 : c: ?# Q. A: } x7 Y% f 2、4种输出模式 8 ~/ T s. x/ V7 _4 J (5)GPIO_Mode_Out_OD 开漏输出(带上拉或者下拉) (6)GPIO_Mode_AF_OD 复用开漏输出(带上拉或者下拉) (7)GPIO_Mode_Out_PP 推挽输出(带上拉或者下拉): v/ O+ J) U t, I (8)GPIO_Mode_AF_PP 复用推挽输出(带上拉或者下拉) 7 c, {9 O" `8 z$ N6 h) _* v) s u. i5 t 3、4种最大输出速度 (1)2MHZ (低速)$ _2 j7 j# V( l9 R/ ^ (2)25MHZ (中速)8 {5 @0 \1 |7 q& q4 D/ c8 J2 C+ O, h (3)50MHZ (快速)7 F+ o1 j* O4 ~& i (4)100MHZ (高速) 关于他们的定义,都在 stm32f4xx_gpio.h 中,都为结构体形式的定义 . x- p. X3 `1 v! _# M! M/ B 3 GPIO框图剖析 / r. f6 F) ^& G3 O. |! y) c ![]() " j! @$ r% \# f 我们所用到的每一个GPIO其内部结构都是这样,分别对应着GPIO的八种模式 这里我们简单的介绍下:* L2 Z% G3 I v9 ?9 |% L% c3 A0 p % M) W5 i6 X- y 保护二极管: IO引脚上下两边两个二极管用于防止引脚外部过高、过低的电压输入,当引脚电压高于VDD_FT时,上方的二极管导通,当引脚电压低于VSS时,下方的二极管导通,防止不正常电压引入芯片导致芯片烧毁 上拉、下拉电阻:控制引脚默认状态的电压,开启上拉的时候引脚默认电压为高电平,开启下拉的时候引脚默认电压为低电平) V( n" w7 w6 @0 Y; S TTL施密特触发器:基本原理是当输入电压高于正向阈值电压,输出为高;当输入电压低于负向阈值电压,输出为低;IO口信号经过触发器后,模拟信号转化为0和1的数字信号 也就是高低电平 并且是TTL电平协议 这也是为什么STM32是TTL电平协议的原因 P-MOS管和N-MOS管:信号由P-MOS管和N-MOS管,依据两个MOS管的工作方式,使得GPIO具有“推挽输出”和“开漏输出”的模式 P-MOS管高电平导通,低电平关闭,下方的N-MOS低电平导通,高电平关闭& e( p2 K/ T$ l' i 注: VDD_FT 代表IO口,兼容3.3V和5V,如果没有标注“FT”,就代表着不兼容5V 4 _. t% e) W: T( f9 U1 I# w 在芯片数据手册的引脚定义中,会看到有“I/O电平”一列 有FT即为支持5V9 z6 w/ U' Y2 D* K" z# u 3 ?3 J, v L+ d) ~+ b ![]() 4 GPIO的八种工作模式剖析:6 v; R4 w! y1 E, g( y/ ^! g 浮空输入模式, T- R+ r# T1 t" ] y! s8 X3 w 9 P# ~" \9 Q. b, y7 | F8 A ![]() ! j+ [, S% N" d/ u# e: J1 s 浮空输入模式下,I/O端口的电平信号直接进入输入数据寄存器。MCU直接读取I/O口电平,I/O的电平状态是不确定的,完全由外部输入决定;如果在该引脚悬空(在无信号输入)的情况下,读取该端口的电平是不确定的。 (接用电压表测量其引脚电压为1点几伏,这是个不确定值) 以用来做KEY识别 上拉输入模式 8 |2 f6 c4 \) d; |1 w ![]() IO内部接上拉电阻,此时如果IO口外部没有信号输入或者引脚悬空,IO口默认为高电平 如果I/O口输入低电平,那么引脚就为低电平,MCU读取到的就是低电平 ?$ M9 L) ^: @( S' Y5 e2 h; r 4 ]& D" {5 |! A) h STM32的内部上拉是"弱上拉",即通过此上拉输出的电流是很弱的,如要求大电流还是需要外部上拉。8 M# f0 Q0 {6 h- F z 下拉输入模式( c" A0 o9 U- ]* R( i 1 c" o& S1 s- H ![]() 0 T0 v4 h- [( } IO内部接下拉电阻,此时如果IO口外部没有信号输入或者引脚悬空,IO口默认为低电平 如果I/O口输入高电平,那么引脚就为高电平,MCU读取到的就是高电平 模拟输入模式& p# ~( N( m; e5 K( W0 y1 S; f/ P. w' b ![]() : ?( L5 Z6 @5 {9 \- U- p 当GPIO引脚用于ADC采集电压的输入通道时,用作"模拟输入"功能,此时信号不经过施密特触发器,直接直接进入ADC模块,并且输入数据寄存器为空 ,CPU不能在输入数据寄存器上读到引脚状态 当GPIO用于模拟功能时,引脚的上、下拉电阻是不起作用的,这个时候即使配置了上拉或下拉模式,也不会影响到模拟信号的输入输出 除了 ADC 和 DAC 要将 IO 配置为模拟通道之外其他外设功能一律 要配置为复用功能模式,; i# E5 x6 m( v5 l% d+ _ , U1 r7 k. Y" w$ ]" |! E. @ 开漏输出模式(带上拉或者下拉)" w& M8 k) [2 K3 ~ w. ?3 b3 _0 B# |/ Y ![]() - o1 Q4 a% |: _# ` 在开漏输出模式时,只有N-MOS管工作,如果我们控制输出为0,低电平,则P-MOS管关闭,N-MOS管导通,使输出低电平,I/O端口的电平就是低电平,若控制输出为1时,高电平,则P-MOS管和N-MOS管都关闭,输出指令就不会起到作用,此时I/O端口的电平就不会由输出的高电平决定,而是由I/O端口外部的上拉或者下拉决定 如果没有上拉或者下拉 IO口就处于悬空状态6 w! t% N% D; H2 b- x, T ' H- W+ w$ C; o" g' K! R 并且此时施密特触发器是打开的,即输入可用,通过输入数据寄存器GPIOx_IDR可读取I/O的实际状态。,I/O口的电平不一定是输出的电平 1 M; t3 y- P8 l3 ]) o, Q ![]() 在推挽输出模式时,N-MOS管和P-MOS管都工作,如果我们控制输出为0,低电平,则P-MOS管关闭,N-MOS管导通,使输出低电平,I/O端口的电平就是低电平,若控制输出为1 高电平,则P-MOS管导通N-MOS管关闭,使输出高电平,I/O端口的电平就是高电平, 外部上拉和下拉的作用是控制在没有输出时IO口电平; a7 x' Y/ G' K: Q 此时施密特触发器是打开的,即输入可用,通过输入数据寄存器GPIOx_IDR可读取I/O的实际状态。I/O口的电平一定是输出的电平 8 M8 n# p( X8 y1 b- o( b9 h 复用开漏输出(带上拉或者下拉)# b6 ~ ~- ^, Q8 O* V ![]() / ^* Z- Q1 Q/ d& s# W GPIO复用为其他外设,输出数据寄存器GPIOx_ODR无效; 输出的高低电平的来源于其它外设,施密特触发器打开,输入可用,通过输入数据寄存器可获取I/O实际状态 除了输出信号的来源改变 其他与开漏输出功能相同 , g2 y8 Q( U" d2 Y* c! u& k( R 复用推挽输出(带上拉或者下拉) s$ U# d9 ?) a9 o0 r ![]() GPIO复用为其他外设(如 I2C),输出数据寄存器GPIOx_ODR无效; 输出的高低电平的来源于其它外设,施密特触发器打开,输入可用,通过输入数据寄存器可获取I/O实际状态 除了输出信号的来源改变 其他与开漏输出功能相同+ z8 Z( `! _9 |& E s1 }: V4 E2 ` . N1 B. N4 B$ E0 @+ f; Z ) E% w# Z$ Z- l( i. P# {% s 开漏输出和推挽输出的区别: ! t8 d9 e1 x: m) l0 i, `+ r* L8 l% G, D 推挽输出:. W# ^+ e; k! D" l- d$ C 7 a3 G# E7 p- h6 K 可以输出强高低电平,连接数字器件 推挽结构一般是指两个三极管分别受两互补信号的控制,总是在一个三极管导通的时候另一个截止. 开漏输出: 可以输出强低电平,高电平得靠外部电阻拉高。输出端相当于三极管的集电极. 需要外接上拉电阻,才能实现输出高电平 合于做电流型的驱动,其吸收电流的能力相对强(一般20ma以内); 4 q. }8 K& \% C 在使用任何一种开漏模式时,都需要接上拉电阻,否则只能输出低电平! ^0 z! c% {+ l 7 }. ?, I! C% W8 F1 {8 J/ @ ] ![]() 推挽输出电路: 其中IN端输出高电平时下面的PNP三极管截止,而上面NPN三极管导通,输出电平VS+;当IN端输出低电平时则恰恰相反,PNP三极管导通,输出和地相连,为低电平 . f" z# f# I4 }% K- i 开漏输出电路:IN端输出低电平时,三极管导通,使输出接地,IN端输出高电平时,三极管截止,所以引脚既不输出高电平,也不输出低电平,为高阻态。为正常使用时必须接上拉电阻, " I# v7 E6 `4 l% x 在STM32中选用IO模式: 上拉输入、下拉输入可以用来检测外部信号;例如,按键等; 模拟输入 ——应用ADC模拟输入,或者低功耗下省电 开漏输出一般应用在I2C、SMBUS通讯等需要"线与"功能的总线电路中。 推挽输出模式一般应用在输出电平为0和3.3伏而且需要高速切换开关状态的场合。在STM32的应用中,除了必须用开漏模式的场合,我们都习惯使用推挽输出模式。+ b6 Z3 }' X% I) b4 `! h1 k$ d5 A 复用功能的推挽输出_AF_PP ——片内外设功能(I2C的SCL,SDA) 复用功能的开漏输出_AF_OD——片内外设功能(TX1,MOSI,MISO.SCK.SS) |! u8 |8 _* ?8 n* ^ F4系列与F1系列区别:8 z |5 I: f1 M 本质上的区别是F4系列采用了Cortex-M4内核 而F1系列采用Cortex-M3内核- h) W0 J. W `: I1 s( T! k3 T3 h$ m- Y 7 H; O9 e9 F, a/ {$ _! c2 t F1系列(M3)IO口基本结构: ![]() F4系列(M4)IO口基本结构: - A) g' `& u: s2 v ![]() F4系列设计的更加高级与人性化,他将外部上下拉电阻转移到了输出/输入驱动器外部,使得输出模式下也可以实现内部上拉与下拉,方便了用户的使用,增加了灵活性7 u- q- f( q( f( L! q/ } GPIO的初始化(F4) 这里我们以初始化LED为例- q7 c" `2 O2 F- { 1.定义一个 GPIO_InitTypeDef 类型的结构体8 J9 _5 z2 s# b6 A
![]() - j0 I/ t' z/ O0 T) \- i% Z5 C* h 2开启 LED 相关的 GPIO 外设时钟7 X9 C' h9 Q4 T( X Y
Q:为什么要设置时钟?, O3 O3 M$ R3 ~! I) X# r 任何外设都需要时钟,51单片机,stm32,430等等,因为寄存器是由D触发器组成的,往触发器里面写东西,前提条件是有时钟输入。stm32是低功耗,他将所有的门都默认设置为disable(不使能),在你需要用哪个门的时候,开哪个门就可以,也就是说用到什么外设,只要打开对应外设的时钟就可以, 其他的没用到的可以还是disable(不使能),这样耗能就会减少。 ' O$ E# h5 o0 v4 k Q:为什么 STM32 要有多个时钟源呢?1 K! H! H J3 c+ ]" H% j! z 0 \/ r2 o) c- d. z$ w* W( T- x ~ 因为首 先 STM32 本身非常复杂,外设非常的多,但是并不是所有外设都需要系统时钟这么高的频率, 比如看门狗以及 RTC 只需要几十 k 的时钟即可。同一个电路,时钟越快功耗越大,同时抗电磁 干扰能力也会越弱,所以对于较为复杂的 MCU 一般都是采取多时钟源的方法来解决这些问题。) S6 i0 I6 m" |( j: C + {, C9 {. w7 I8 p. X* j4 f' w+ _ 3 V- V" ~; M" }0 r4 Y 而相对应的外设功能所使用的时钟 在stm32f4xx.h 中即可查看到 s" g3 V: ?7 G RCC_AHB1/ z2 M8 [# Q0 n! h" t/ h4 { $ D0 n w5 t; T8 A" @# d! H3 U ![]() RCC_APB1 , `( J8 T# f1 O# J$ } ![]() * w: r, g; I5 ?9 a6 F6 ^1 V* ] 3选择要控制的 GPIO 引脚6 u; l9 X) @0 ?; W' g
可选引脚为0-15 一组IO口有16个引脚0 Z! S6 ~4 ~6 L- p6 Z* D ![]() 4设置所选引脚的模式
引脚的模式共有四种,分别为输入,输出,复用,和模拟模式7 G$ I5 q- O6 J$ g + x" U+ m5 g D ![]() - `2 h# O- H6 q% A5 G$ G0 u 5 设定所选引脚的输出类型+ G9 p/ ?6 ?+ X* @ 2 }- \! c" t4 {1 ?- N
输出模式有两种:推挽输出和开漏输出 0 o7 z3 X/ x5 S, w1 R4 o ! o* s$ I& ~- I. { 只有输出模式才需要配置,输入模式下不需要配置 ! V! n' }; Y1 Y) m+ J6 I. d 6 设定所选管脚的速度# F- n7 d: e P5 Y& L- w5 f # Y" q! R; E2 \& a
![]() 9 O8 @* t( \( S. l' D 7 设定所选管脚的上拉与下拉
可设置为:上拉,下拉,与浮空, l: u! X8 Y* n. n) ~ ![]() ' @" C! c+ _4 v) ^3 ]; a/ t" h+ R 8初始化GPIO3 D& Y6 V+ U' m
GPIO_Init() 是官方配置的初始化函数 第一个参数是GPIOX 第二个参数是结构体所对应GPIO各种参数的配置 GPIO的初始化(F1) F4的初始化相较于F1系列有了很大的改变,在配置F1系列时,结构体只有三个参数 8 y+ e! Y1 @" U3 v3 n1 I8 z6 M GPIO_InitTypeDef 类型的结构体3 `2 j4 U' G9 Z1 V+ I) W& T4 k! Y 2 x" Y) H5 I1 ` ?% ~ ![]() 所选管脚的速度! j- F0 q0 n. m; ~7 t5 P' X" V ( I7 b2 ^2 p3 b9 \ ![]() 所选管脚的8种模式 ![]() 区别: 8 R& v8 \: S& ^9 F @, M& C F1(M-3)系列管脚速度只有三种模式 并且管脚的模式配置八种模式全部都在一起定义,直接设置即可2 S) {# d9 L% N/ G9 A( d( E# Q 总归还是大同小异,不做过多介绍,看下下方配置即可
4 d* v8 D0 \( ]3 y- N |