1 GPIO简介 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内核 8 e/ q1 @7 @: T8 T1 T' V- Q$ w# E GPIO的复用:2 g2 \& n5 k+ O1 M9 Z' j STM32F4 有很多的内置外设,这些外设的外部引脚都是与 GPIO 共用的。也就是说,一个引脚可以有很多作用,但是默认为IO口,如果想使用一个 GPIO内置外设的功能引脚,就需要GPIO的复用,那么当这个 GPIO 作为内置外设使用的时候,就叫做复用。 比如说串口 就是GPIO复用为串口' X( Q; F4 j8 h) q( H# O 2 GPIO的工作模式 1、4种输入模式4 j' l2 J5 o% W- d, O& t2 W. W7 n (1)GPIO_Mode_IN_FLOATING 浮空输入 (2)GPIO_Mode_IPU 上拉输入6 h6 J1 m" e. u3 j9 v# P (3)GPIO_Mode_IPD 下拉输入 (4)GPIO_Mode_AIN 模拟输入! t, U ~& Y: u4 z1 y : q& t8 v8 z3 i+ k" l 2、4种输出模式 (5)GPIO_Mode_Out_OD 开漏输出(带上拉或者下拉)* G; U( V8 A# ]0 o. L# r (6)GPIO_Mode_AF_OD 复用开漏输出(带上拉或者下拉) (7)GPIO_Mode_Out_PP 推挽输出(带上拉或者下拉)& C! {$ ?( ~# e (8)GPIO_Mode_AF_PP 复用推挽输出(带上拉或者下拉) ; r+ g( v7 q6 A* \1 t 3、4种最大输出速度 (1)2MHZ (低速)+ |0 P' V4 A2 `: Z1 r (2)25MHZ (中速) (3)50MHZ (快速)% c; L8 e3 i8 q1 p% g5 c. K (4)100MHZ (高速)0 |: |: Y6 u4 G; M1 A! y$ ]6 p 关于他们的定义,都在 stm32f4xx_gpio.h 中,都为结构体形式的定义 1 Y4 Y3 ]4 v& Z 3 GPIO框图剖析8 U8 ~! P$ p I# O6 H+ ` 6 p3 ^ R& g' X* b/ K" I6 W 我们所用到的每一个GPIO其内部结构都是这样,分别对应着GPIO的八种模式 这里我们简单的介绍下: + I9 \ ]: @* r5 N" ? 保护二极管: IO引脚上下两边两个二极管用于防止引脚外部过高、过低的电压输入,当引脚电压高于VDD_FT时,上方的二极管导通,当引脚电压低于VSS时,下方的二极管导通,防止不正常电压引入芯片导致芯片烧毁 + a4 |6 c" I9 n8 N 上拉、下拉电阻:控制引脚默认状态的电压,开启上拉的时候引脚默认电压为高电平,开启下拉的时候引脚默认电压为低电平, T j4 _: I5 r. O5 U/ d# B TTL施密特触发器:基本原理是当输入电压高于正向阈值电压,输出为高;当输入电压低于负向阈值电压,输出为低;IO口信号经过触发器后,模拟信号转化为0和1的数字信号 也就是高低电平 并且是TTL电平协议 这也是为什么STM32是TTL电平协议的原因 P-MOS管和N-MOS管:信号由P-MOS管和N-MOS管,依据两个MOS管的工作方式,使得GPIO具有“推挽输出”和“开漏输出”的模式 P-MOS管高电平导通,低电平关闭,下方的N-MOS低电平导通,高电平关闭 注: VDD_FT 代表IO口,兼容3.3V和5V,如果没有标注“FT”,就代表着不兼容5V 1 d% |1 h9 w* e0 F 在芯片数据手册的引脚定义中,会看到有“I/O电平”一列 有FT即为支持5V 4 GPIO的八种工作模式剖析: 浮空输入模式 % W7 K1 ^/ \* ~' I" T 浮空输入模式下,I/O端口的电平信号直接进入输入数据寄存器。MCU直接读取I/O口电平,I/O的电平状态是不确定的,完全由外部输入决定;如果在该引脚悬空(在无信号输入)的情况下,读取该端口的电平是不确定的。 (接用电压表测量其引脚电压为1点几伏,这是个不确定值) 以用来做KEY识别 上拉输入模式 IO内部接上拉电阻,此时如果IO口外部没有信号输入或者引脚悬空,IO口默认为高电平 如果I/O口输入低电平,那么引脚就为低电平,MCU读取到的就是低电平 ) t# a* m- h4 i6 ~/ B$ O$ f STM32的内部上拉是"弱上拉",即通过此上拉输出的电流是很弱的,如要求大电流还是需要外部上拉。) A. _ T. [' B6 u! K& q$ o 下拉输入模式- W8 L- F' t+ h5 k9 M* w' T IO内部接下拉电阻,此时如果IO口外部没有信号输入或者引脚悬空,IO口默认为低电平 如果I/O口输入高电平,那么引脚就为高电平,MCU读取到的就是高电平 2 z' i4 ]; X5 z C- S 模拟输入模式 $ ?- N) _ P9 P% x+ y" Q 当GPIO引脚用于ADC采集电压的输入通道时,用作"模拟输入"功能,此时信号不经过施密特触发器,直接直接进入ADC模块,并且输入数据寄存器为空 ,CPU不能在输入数据寄存器上读到引脚状态 当GPIO用于模拟功能时,引脚的上、下拉电阻是不起作用的,这个时候即使配置了上拉或下拉模式,也不会影响到模拟信号的输入输出 8 N, G9 N- I l2 U( O 除了 ADC 和 DAC 要将 IO 配置为模拟通道之外其他外设功能一律 要配置为复用功能模式, ) t( u1 Q% O" o" [ 开漏输出模式(带上拉或者下拉) e' \0 t4 y0 H# ] - E" {# B- D: [2 y, Q 8 ]1 n% U$ x1 t: X 在开漏输出模式时,只有N-MOS管工作,如果我们控制输出为0,低电平,则P-MOS管关闭,N-MOS管导通,使输出低电平,I/O端口的电平就是低电平,若控制输出为1时,高电平,则P-MOS管和N-MOS管都关闭,输出指令就不会起到作用,此时I/O端口的电平就不会由输出的高电平决定,而是由I/O端口外部的上拉或者下拉决定 如果没有上拉或者下拉 IO口就处于悬空状态$ ]2 i* Y, Z' a1 X/ x( R * e, [% w y) R3 L; T Q 并且此时施密特触发器是打开的,即输入可用,通过输入数据寄存器GPIOx_IDR可读取I/O的实际状态。,I/O口的电平不一定是输出的电平 : X6 K! D( t& R/ d9 b ' ~, T7 @0 x, L1 F9 j 9 u4 o# p: X6 ~ 在推挽输出模式时,N-MOS管和P-MOS管都工作,如果我们控制输出为0,低电平,则P-MOS管关闭,N-MOS管导通,使输出低电平,I/O端口的电平就是低电平,若控制输出为1 高电平,则P-MOS管导通N-MOS管关闭,使输出高电平,I/O端口的电平就是高电平, 外部上拉和下拉的作用是控制在没有输出时IO口电平- R5 V8 U/ O+ y" n x* c- K 此时施密特触发器是打开的,即输入可用,通过输入数据寄存器GPIOx_IDR可读取I/O的实际状态。I/O口的电平一定是输出的电平 ; y! }, y. C: g3 }) R / s" g* Y6 r- R 复用开漏输出(带上拉或者下拉) ) x A; ?6 t$ Q. \( ?1 G% I7 S! A GPIO复用为其他外设,输出数据寄存器GPIOx_ODR无效; 输出的高低电平的来源于其它外设,施密特触发器打开,输入可用,通过输入数据寄存器可获取I/O实际状态 除了输出信号的来源改变 其他与开漏输出功能相同2 T; N; Y* R Z( g/ ~7 a, J/ A 2 x. l3 p/ v% H 复用推挽输出(带上拉或者下拉)6 r, a6 z" W3 s* A 1 m" R5 ~9 S p) [( Z+ B; j9 j , {& D' P/ O4 H4 x GPIO复用为其他外设(如 I2C),输出数据寄存器GPIOx_ODR无效; 输出的高低电平的来源于其它外设,施密特触发器打开,输入可用,通过输入数据寄存器可获取I/O实际状态 除了输出信号的来源改变 其他与开漏输出功能相同 # y6 |: y& {: F* T# F8 d 开漏输出和推挽输出的区别:. g7 O4 {) Y4 r: H% W+ F* z3 u 推挽输出:" E& P0 X( `, r ! I8 [) d7 p& t4 E6 g 可以输出强高低电平,连接数字器件 : }+ E+ g! \; W F" e- X1 T5 B$ U 推挽结构一般是指两个三极管分别受两互补信号的控制,总是在一个三极管导通的时候另一个截止. & Y( a+ x+ H% o: s& q 开漏输出:- y% X' J: G0 H9 B/ j* T! Q 7 W) N+ H6 _; Q$ ]. J4 w 可以输出强低电平,高电平得靠外部电阻拉高。输出端相当于三极管的集电极. 需要外接上拉电阻,才能实现输出高电平 合于做电流型的驱动,其吸收电流的能力相对强(一般20ma以内);6 e% I0 Y+ C& v/ P5 V9 J 在使用任何一种开漏模式时,都需要接上拉电阻,否则只能输出低电平( E9 a8 J( Z3 A- M' K/ H " N, j. Z) X' X O& B% W! O! @ 推挽输出电路: 其中IN端输出高电平时下面的PNP三极管截止,而上面NPN三极管导通,输出电平VS+;当IN端输出低电平时则恰恰相反,PNP三极管导通,输出和地相连,为低电平 " m" [8 Z, _8 @( q 开漏输出电路:IN端输出低电平时,三极管导通,使输出接地,IN端输出高电平时,三极管截止,所以引脚既不输出高电平,也不输出低电平,为高阻态。为正常使用时必须接上拉电阻, 8 x8 ~, @4 K/ [+ t0 i5 t- w1 I 在STM32中选用IO模式:* d4 m' U" ]* [9 l$ N, a5 D/ K 上拉输入、下拉输入可以用来检测外部信号;例如,按键等; 模拟输入 ——应用ADC模拟输入,或者低功耗下省电5 l: b0 F( w0 @% [ 开漏输出一般应用在I2C、SMBUS通讯等需要"线与"功能的总线电路中。7 h7 G! U |# L$ }/ A- b l 推挽输出模式一般应用在输出电平为0和3.3伏而且需要高速切换开关状态的场合。在STM32的应用中,除了必须用开漏模式的场合,我们都习惯使用推挽输出模式。" I% B* V6 a7 I7 t 复用功能的推挽输出_AF_PP ——片内外设功能(I2C的SCL,SDA)) X7 x0 q7 B% `8 q6 ^ 复用功能的开漏输出_AF_OD——片内外设功能(TX1,MOSI,MISO.SCK.SS)7 V) \. V, g8 D* ]8 z F4系列与F1系列区别: 本质上的区别是F4系列采用了Cortex-M4内核 而F1系列采用Cortex-M3内核 # o% P0 [2 t# m! O% V F1系列(M3)IO口基本结构:& f ?& q1 m/ i* o 3 ?% H9 P! v- l( U" Z F4系列(M4)IO口基本结构:; o$ H( G }. M+ W' i $ ]" U) ]) s' p6 f/ b% y7 k F4系列设计的更加高级与人性化,他将外部上下拉电阻转移到了输出/输入驱动器外部,使得输出模式下也可以实现内部上拉与下拉,方便了用户的使用,增加了灵活性& D2 u2 d: Z$ y; v. j4 e, E: a6 l 3 v0 N& W1 _; g GPIO的初始化(F4) 这里我们以初始化LED为例5 D6 v; V6 v1 H 1.定义一个 GPIO_InitTypeDef 类型的结构体
$ r$ \* s1 P7 d: c 3 N. G. X& [! N3 } 2开启 LED 相关的 GPIO 外设时钟 # B, l' c: J2 m
Q:为什么要设置时钟?& N Q( u; ?" m# f 任何外设都需要时钟,51单片机,stm32,430等等,因为寄存器是由D触发器组成的,往触发器里面写东西,前提条件是有时钟输入。stm32是低功耗,他将所有的门都默认设置为disable(不使能),在你需要用哪个门的时候,开哪个门就可以,也就是说用到什么外设,只要打开对应外设的时钟就可以, 其他的没用到的可以还是disable(不使能),这样耗能就会减少。 Q:为什么 STM32 要有多个时钟源呢? 因为首 先 STM32 本身非常复杂,外设非常的多,但是并不是所有外设都需要系统时钟这么高的频率, 比如看门狗以及 RTC 只需要几十 k 的时钟即可。同一个电路,时钟越快功耗越大,同时抗电磁 干扰能力也会越弱,所以对于较为复杂的 MCU 一般都是采取多时钟源的方法来解决这些问题。. ~' {3 @: D$ L; ^) |3 t1 v! `) y6 j 而相对应的外设功能所使用的时钟 在stm32f4xx.h 中即可查看到6 j% m7 o; d4 I: I" ^ + Z% \- b0 \& [- f3 d RCC_AHB1 " @" |' \, p$ x7 n7 ^: B2 j % ~6 e2 k% ^, I8 T' S7 x3 |# p RCC_APB17 _6 i8 e: b$ d. H$ {* l* e1 f $ t9 V( q% G5 u& e - J. }6 F' j% n5 {4 } 3选择要控制的 GPIO 引脚
可选引脚为0-15 一组IO口有16个引脚1 ?9 ^9 e0 n+ P 4设置所选引脚的模式; b& }1 r7 X, r: A
引脚的模式共有四种,分别为输入,输出,复用,和模拟模式+ V. N3 u& k! S0 h, z; C! y % v7 P/ y+ n8 r1 o/ t1 @ 5 设定所选引脚的输出类型. H+ \) P/ u/ x8 X. c1 l
输出模式有两种:推挽输出和开漏输出 . P2 U. e; K- {4 _* @ 只有输出模式才需要配置,输入模式下不需要配置* d/ g( Z: @: g3 U, U0 Q9 d6 ]: a; O, w ; v _; f# f/ \9 I 6 设定所选管脚的速度+ b8 b5 H- p$ C$ |
/ ]" r8 z0 a# x: l1 U0 X 7 设定所选管脚的上拉与下拉 8 M; G* z' h' {0 O7 W
可设置为:上拉,下拉,与浮空" `! U3 ~- c; v3 S4 n1 R/ O( N 8初始化GPIO 2 L- C- z! I$ P" p6 G: y
GPIO_Init() 是官方配置的初始化函数 第一个参数是GPIOX 第二个参数是结构体所对应GPIO各种参数的配置( T1 z5 f, o& y6 a $ O: R1 {! X, \- a8 H* A8 E GPIO的初始化(F1) f) H% K6 g) l7 p! r F4的初始化相较于F1系列有了很大的改变,在配置F1系列时,结构体只有三个参数 1 I* E5 N/ q# ]2 J& D) ~# `$ X' u GPIO_InitTypeDef 类型的结构体 ; X H* g' K' Y* e) d: _+ ? 所选管脚的速度! { I0 L! |' T; |9 q) @0 b , |% I l# Q. _. H - [8 }0 v& B0 U9 x: F: w, U 所选管脚的8种模式 6 O% h) t" G7 b9 c$ F2 E 区别: 8 p2 g5 p( ^' f1 L' e F1(M-3)系列管脚速度只有三种模式 并且管脚的模式配置八种模式全部都在一起定义,直接设置即可0 q9 ?" s4 c% r, ^2 ] 总归还是大同小异,不做过多介绍,看下下方配置即可6 p+ V1 J6 u! Y2 O+ J
) ^% ~4 j' I2 ]/ p7 U% y 1 P3 \% r( U9 X |' Z/ ^. Q/ b1 f |
串口通信波特率异常的解决办法
《RT-Thread设备驱动开发指南》书籍学习记录
STM32F407/STM32F417通过以太网实现在应用中编程 (IAP)
STM32固件库分享,超全系列整理
分享STM32F407的学习资料 附学习笔记
STM32F4中文资料--基础篇
基于STM32F407和Cubemx的ADC采集+DMA传输实现简易示波器经验分享
基于STM32F407的DMA采样+FFT时域经验分享
基于STM32F407中的RNG获取随机数经验分享
【中文文档】AN3965_STM32F40x和STM32F41x基于串口的IAP