LV04-02-GPIO-01-GPIO简介
本文主要是I.MX6ULL的GPIO——GPIO简介的相关笔记,若笔记中有错误或者不合适的地方,欢迎批评指正😃。
点击查看使用工具及版本
Windows版本 | windows11 |
Ubuntu版本 | Ubuntu16.04的64位版本 |
VMware® Workstation 16 Pro | 16.2.3 build-19376536 |
终端软件 | MobaXterm(Professional Edition v23.0 Build 5042 (license)) |
Linux开发板 | 正点原子 i.MX6ULL Linux 阿尔法开发板 |
uboot | NXP官方提供的uboot,NXP提供的版本为uboot-imx-rel_imx_4.1.15_2.1.0_ga(使用的uboot版本为U-Boot 2016.03) |
linux内核 | linux-4.15(NXP官方提供) |
Win32DiskImager | Win32DiskImager v1.0 |
点击查看本文参考资料
分类 | 网址 | 说明 |
官方网站 | https://www.arm.com/ | ARM官方网站,在这里我们可以找到Cotex-Mx以及ARMVx的一些文档 |
https://www.nxp.com.cn/ | NXP官方网站 | |
https://www.nxpic.org.cn/ | NXP 官方社区 | |
https://u-boot.readthedocs.io/en/latest/ | u-boot官网 | |
https://www.kernel.org/ | linux内核官网 |
点击查看相关文件下载
分类 | 网址 | 说明 |
NXP | https://github.com/nxp-imx | NXP imx开发资源GitHub组织,里边会有u-boot和linux内核的仓库 |
https://elixir.bootlin.com/linux/latest/source | 在线阅读linux kernel源码 | |
nxp-imx/linux-imx/releases/tag/rel_imx_4.1.15_2.1.0_ga | NXP linux内核仓库tags中的rel_imx_4.1.15_2.1.0_ga | |
nxp-imx/uboot-imx/releases/tag/rel_imx_4.1.15_2.1.0_ga | NXP u-boot仓库tags中的rel_imx_4.1.15_2.1.0_ga | |
I.MX6ULL | i.MX 6ULL Applications Processors for Industrial Products | I.MX6ULL 芯片手册(datasheet,可以在线查看) |
i.MX 6ULL Applications ProcessorReference Manual | I.MX6ULL 参考手册(下载后才能查看,需要登录NXP官网) |
一、GPIO概述
1. 硬件框图
GPIO就是General-purpose input/output,通用的输入输出口。GPIO 模块一般结构如下:
(1)有多组 GPIO,每组有多个 GPIO
(2)使能:电源/时钟
(3)模式(Mode):引脚可用于 GPIO 或其他功能
(4)方向:引脚 Mode 设置为 GPIO 时,可以继续设置它是输出引脚,还是输入引脚
(5)数值:对于输出引脚,可以设置寄存器让它输出高、低电平对于输入引脚,可以读取寄存器得到引脚的当前电平。
我们可以看一下《i.MX 6UltraLite Applications Processor Reference Manual》 Chapter 28 General Purpose Input/Output (GPIO) ,里面有一个GPIO的硬件框图:
2. 基本结构分析
接下来我们按照上图的编号分析一下这个硬件框图。
2.1 ①——PAD
PAD代表了一个i.MX6ULL的GPIO引脚,就是我们拿到芯片能看到的那些金属管脚。在它的左侧是一系列信号通道及控制线, 如input_on控制输入开关,Dir控制引脚的输入输出方向,Data_out控制引脚输出高低电平, Data_in作为信号输入,这些信号都经过一个IOMUX的器件连接到左侧的寄存器。
对于每个引脚都有很多关于属性的配置如下所示:
- A——PAD引脚
框图中的最右侧的PAD同样是代表一个i.MX6ULL的引脚。
- B——输出缓冲区
当输出缓冲区使能时,引脚被配置为输出模式。在输出缓冲区中,又包含了如下的属性配置:
(1)DSE驱动能力:DSE可以调整芯片内部与引脚串联电阻R0的大小,从而改变引脚的驱动能力。 例如,R0的初始值为260欧姆,在3.3V电压下其电流驱动能力为12.69mA,通过DSE可以把R0的值配置为原值的1/2、1/3…1/7等。
(2)SRE压摆率配置:压摆率是指电压转换速率,可理解为电压由波谷升到波峰的时间。 增大压摆率可减少输出电压的上升时间。i.MX6ULL的引脚通过SRE支持低速和高速压摆率这两种配置。 压摆率是大信号特性,下面的带宽是小信号特性。
(3)SPEED带宽配置:通过SPEED可以设置IO的带宽,分别可设置为50MHz、100MHz以及200MHz。 带宽的意思是能通过这个IO口最高的信号频率,通俗点讲就是方波不失真,如果超过这个频率方波就变正弦波。 但是这个带宽要区别于IO的翻转速率,IO的翻转速率的信号来自于GPIO这个外设, 而IO的带宽只是限制了IO口引脚的物理特性,IO口的信号可以来自于内部定时器输出的PWM信号, 也可以来自于GPIO翻转输出的信号,两者相比之下,PWM信号的频率是远远高于GPIO翻转输出的信号频率。
(4)ODE开漏输出配置:通过ODE可以设置引脚是否工作在开漏输出模式。在该模式时引脚可以输出高阻态和低电平, 输出高阻态时可由外部上拉电阻拉至高电平。开漏输出模式常用在一些通讯总线中,如I2C。
- C——输入缓冲区
当输入缓冲区使能时,引脚被配置为输入模式。在输入缓冲区中,又包含了如下的属性配置:
HYS滞后使能:i.MX 6U的输入检测可以使用普通的CMOS检测或施密特触发器模式(滞后模式)。 施密特触发器具有滞后效应,对正向和负向变化的输入信有不同的阈值电压, 常被用于电子开关、波形变换等场合,其转换特性和对比如下,如检测按键时, 使用施密特模式即可起到消抖的功能。
- D——Pull/Keeper上下拉、保持器
引脚的控制逻辑中还包含了上下拉、保持器的功能。 芯片内部的上拉和下拉电阻可以将不确定的信号钳位在高、低电平,或小幅提高的电流输出能力, 上拉提供输出电流,下拉提供输入电流。注意这些上下拉配置只是弱拉,对于类似I2C之类的总线, 还是必须使用外部上拉电阻。i.MX6ULL芯片的电源模块中包含转换器,当转换器停止工作时,保持器会保持输入输出电压。上下拉、保持器可以通过如下属性配置:
(1)PUS上下拉配置:PUS可配置项可选为100K欧下拉以及22K欧、47K欧及100K欧上拉。
(2)PUE上下拉、保持器选择:上下拉功能和保持器功能是二选一的,可以通过PUE来选择。
(3)PKE上下拉、保持器配置:上下拉功能和保持器还通过PKE来控制是否使能。
注意,当引脚被配置为输出模式时,不管上下拉、保持器是什么配置,它们都会被关闭。
2.2 ②——IOMUX复用选择器
IOMUX译为IO复用选择器。i.MX6ULL的芯片每个GPIO都通过IOMUX支持多种功能, 例如一个IO可用于网络外设ENET的数据接收引脚,也可以被配置成PWM外设的输出引脚, 这样的设计大大增加了芯片的适用性,这样可选的功能就是由IOMUX实现的。 IOMUX相当于增加了多根内部信号线与IO引脚相连,最多有8根,也就是说一个IO最多可支持8种可选的功能。
IOMUX由其左侧的IOMUXC控制(C表示Controler),IOMUXC提供寄存器给用户进行配置, 它又分成MUX Mode(IO模式控制)以及Pad Settings(Pad配置)两个部分:
- MUX Mode配置
MUX Mode就是用来配置引脚的复用功能,按上面的例子,即是具体是用于网络外设ENET的数据接收, 还是用于PWM外设的输出引脚,当然,也可以配置成普通的IO口,仅用于控制输出高低电平。
- Pad Settings配置
Pad Settings用于配置引脚的属性,例如驱动能力,是否使用上下拉电阻, 是否使用保持器,是否使用开漏模式以及使用施密特模式还是CMOS模式等。 关于属性的介绍会在后面学习,在学习各种外设时,也将会接触到这些属性在不同场合下的应用。
在IOMUXC外设中关于MUX Mode和Pad Settings寄存器命名格式如下
1 | IOMUXC控制类型 寄存器名称 |
2.3 ③——Block外设功能控制块
Block是外设功能控制块,例如具有ENET的数据接收功能的引脚,它就需要网络外设ENET的支持, 具有PWM输出功能的引脚,它需要PWM外设的支持,这些外设在芯片内部会有独立的功能逻辑控制块, 这些控制块通过IOMUX的复用信号与IO引脚相连。使用时通过IOMUX选择具体哪个外设连接到IO。
2.4 ④——GPIO外设
GPIO模块是每个IO都具有的外设,它具有IO控制最基本的功能,如输出高低电平、检测电平输入等。 它也占用IOMUX分配的复用信号,也就是说使用GPIO模块功能时同样需要使用IOMUX选中GPIO外设。 图中的GPIO.DR、GPIO.GDIR、GPIO.PSR等是指GPIO外设相关的控制寄存器, 它们分别是数据寄存器、方向寄存器以及引脚状态寄存器,功能介绍如下:
(1)GPIO.GDIR方向寄存器:控制一个GPIO引脚时,要先用GDIR方向寄存器配置该引脚用于输出电平信号还是用作输入检测。 典型的例子是使用输出模式可以控制LED灯的亮灭,输入模式时可以用来检测按键是否按下。GDIR寄存器的每一个数据位代表一个引脚的方向,对应的位被设置为0时该引脚为输入模式,被设置为1时该引脚为输出模式。这个后面会具体再了解。
(2)GPIO.DR数据寄存器:DR数据寄存器直接代表了引脚的电平状态,它也使用1个数据位表示1个引脚的电平,每位用1表示高电平,用0表示低电平。当GDIR方向寄存器设置引脚为输出模式时,写入DR数据寄存器对应的位即可控制该引脚输出的电平状态。当GDIR方向寄存器设置引脚为输入模式时,读取DR数据寄存器对应的位即可获取该引脚当前的输入电平状态。
(3)GPIO.PSR引脚状态寄存器:PSR引脚状态寄存器相当于DR寄存器的简化版,它仅在GDIR方向寄存器设置为输入模式时有效, 它的每个位表示一个引脚当前的输入电平状态。PSR寄存器的权限是只读的,对它进行写操作是无效的。特别地,当引脚被配置成输出模式时,若IOMUXC中的MUX寄存器使能了SION功能(输出通道回环至输入), 可以通过PSR寄存器读取回引脚的状态值。
2.5 ⑤——与其它引脚的连接
GPIO功能框中的第5项表示另一个引脚PAD2,它与PAD1有一根信号线连接,表示部分引脚的输出可以作为另一个引脚的输入。
2.6 总结一下
这一部分主要包括PAD、IOMUX复用选选择器以及GPIO外设三部分内容。剩下两个相对没那么重要,可以先不管。
- PAD就是我们拿到芯片能看到的那些金属管脚,在这部分学习了芯片引脚驱动能力、上下拉、输出最高频率等等和STM32非常相似的内容, 但是这部分并没有提供配置GPIO驱动能力、上下拉等属性的寄存器,这些配置寄存器在引脚复用(IOMUX)模块配置。
- IOMUX复用选择器,这部分内容很多,简单来说每个可用的引脚拥有两个寄存器, 一个用于配置引脚的复用功能,另外一个用于配置引脚的驱动能力、上下拉、带宽等等引脚属性。
- GPIO外设,需要强调的GPIO不是引脚,GPIO是外设,同串口、SPI一样是芯片的一种片上外设。 某个引脚可以通过IOMUX复用选择器选择为GPIO功能,同样也可能可以选择为其他外设的功能引脚。 芯片上电后每个引脚有自己的默认复用功能,当然我们可修改它。
3. 一共有多少个GPIO?
I.MX6ULL有 5 组 GPIO( GPIO1~GPIO5),每组引脚最多有 32 个,但是可能实际上并没有那么多。 一共有124个GPIO:
GPIO1 有 32 个引脚: GPIO1_IO0~GPIO1_IO31;
GPIO2 有 22 个引脚: GPIO2_IO0~GPIO2_IO21;
GPIO3 有 29 个引脚: GPIO3_IO0~GPIO3_IO28;
GPIO4 有 29 个引脚: GPIO4_IO0~GPIO4_IO28;
GPIO5 有 12 个引脚: GPIO5_IO0~GPIO5_IO11;
二、怎么用参考手册?
看一下参考手册《i.MX 6ULL Applications Processor Reference Manual》就会发现,里面的GPIO命名并不是GPIOx这样简单,还有一堆名称很长的,接下来我们来了解一下。
1. I.MX6ULL的 IO 命名方式
像之前学习的STM32 中的 IO 都是 PA0~15、 PB0~15 这样命名的, I.MX6ULL 的 IO 是怎么命名的呢?打开《i.MX 6ULL Applications Processor Reference Manual》参考手册的 Chapter 32 IOMUX Controller (IOMUXC) ,我们先看一下书签:
我们先看一下参考手册的32.1.1 Features 这一节:
I.MX6ULL 的 IO 分为两类: SNVS 域的和通用的,这两类 IO 本质上都是一样的,我们就有下面的常用 IO 为例,说明一下 I.MX6ULL 的 IO 命名方式。
形如“ IOMUXC_SW_MUX_CTL_PAD_GPIO1_IO00 ”的就是 GPIO 命名,命名形式就是“IOMUXC_SW_MUX_CTL_PAD_XX_XX”,后面的“XX_XX”就是 GPIO 命名,比如: GPIO1_IO01、 UART1_TX_DATA、 JTAG_MOD 等等。I.MX6ULL 的 GPIO 并不像 STM32一样以 PA0~15 这样命名,它是根据某个 IO 所拥有的功能来命名的。比如我们一看到GPIO1_IO01 就知道这个肯定能做 GPIO,看到 UART1_TX_DATA 肯定就知道这个 IO 肯定能做为 UART1 的发送引脚。
那上面的还有些绿色和蓝色区域的名字为什么有的是MUX,有的是PAD?他们最后的命名都一样,但是这里不同?比如:
1 | IOMUXC_SW_MUX_CTL_PAD_GPIO1_IO04 |
其实前面有提到的,对于某个/某组引脚, IOMUXC 中有 2 个寄存器用来设置它 :
- 选择功能
1 | IOMUXC_SW_MUX_CTL_PAD_<PADNAME> # Mux pad xxx,选择某个 pad 的功能 |
- 设置上下拉电阻等参数
1 | IOMUXC_SW_PAD_CTL_PAD_<PAD_NAME> # pad pad xxx,设置某个 pad 的参数 |
还有一些不符合这个规律的,但是看一下寄存器功能大概都就知道怎么回事了,这里就不过多的去了解了。
另外“Chapter 32: IOMUX Controller(IOMUXC)”这一章列出了 I.MX6ULL 的所有 IO,如果找遍 32 章的书签,我们会发现貌似 GPIO 只有GPIO1_IO00 ~ GPIO1_IO09,难道I.MX6ULL 的 GPIO 只有这 10 个?显然不是的, 我们学习STM32的时候知道 STM32 的很多 IO 是可以复用为其它功能的,类比一下就可以知道其实 I.MX6ULL 的其它IO也是可以复用为 GPIO 功能。同样的,GPIO1_IO00~GPIO_IO09 也是可以复用为其它外设引脚的,关于复用,后边再说。
2. IO的MUX Mode配置说明
以“IOMUXC_SW_MUX_CTL_PAD_GPIO1_IO00”这个 IO 为例,打开参考手册的 1568 页(32.6.7 SW_MUX_CTL_PAD_GPIO1_IO00 SW MUX Control Register (IOMUXC_SW_MUX_CTL_PAD_GPIO1_IO00) ):
可以看到有个名为: IOMUXC_SW_MUX_CTL_PAD_GPIO1_IO00 的寄存器,寄存器地址为 0X020E005C,这个寄存器是 32 位的,但是只用到了最低 5 位,其中bit0~bit3(MUX_MODE)就是设置 GPIO1_IO00 的复用功能的。 GPIO1_IO00 一共可以复用为 9种功能 IO,分别对应 ALT0~ALT8,其中 ALT5 就是作为 GPIO1_IO00。 GPIO1_IO00 还可以作为 I2C2_SCL、 GPT1_CAPTURE1、 ANATOP_OTG1_ID 等。这个就是 I.MX6U 的 IO 复用。
由此可见, I.MX6U 的 GPIO 不止 GPIO1_IO00~GPIO1_IO09 这 10 个,其它的 IO 都可以复用为 GPIO 来使用。 I.MX6U 的 GPIO 一共有 5 组:GPIO1、 GPIO2、 GPIO3、 GPIO4 和 GPIO5,其中 GPIO1 有 32 个 IO, GPIO2 有 22 个 IO, GPIO3 有 29 个 IO、 GPIO4 有 29 个 IO, GPIO5最少,只有 12 个 IO,这样一共有 124 个 GPIO。如果只想看每个 IO 能复用什么外设的话可以直接查阅《IMX6ULL 参考手册》的第 4 章“Chapter 4 External Signals and Pin Multiplexing”。如果我们要编写代码,设置某个 IO 的复用功能的话就需要查阅第 32 章“Chapter 32: IOMUX Controller(IOMUXC)” ,第 32 章详细的列出了所有 IO 对应的复用配置寄存器。
3. IO的Pad Settings配置说明
我们会发现《 I.MX6UL 参考手册》 第 32章“ Chapter 32 : IOMUX Controller(IOMUXC)”的书签中,每一个 IO 会出现两次,
它们的名字差别很小,不仔细看就看不出来,比如 GPIO1_IO00 有如下两个书签:
1 | IOMUXC_SW_MUX_CTL_PAD_GPIO1_IO00 |
上面两个都是跟 GPIO_IO00 有关的寄存器,名字上的区别就是红色部分,一个是“MUX”,一个是“PAD”。
IOMUX_SW_MUX_CTL_PAD_GPIO1_IO00 我们前面已经说了,是用来配置GPIO1_IO00 复用功能的。那么IOMUXC_SW_PAD_CTL_PAD_GPIO1_IO00 是做什么的呢?找到这个书签对应的 1787 页(32.6.153 SW_PAD_CTL_PAD_GPIO1_IO00 SW PAD ControlRegister (IOMUXC_SW_PAD_CTL_PAD_GPIO1_IO00) ):
可以看出, IOMUXC_SW_PAD_CTL_PAD_GPIO1_IO00 也是个寄存器,寄存器地址为 0X020E02E8。这也是个 32 位寄存器,但是只用到了其中的低 17 位,在看这写位的具体含义之前,先来看一下下图所示的 GPIO 功能图(参考手册的28.4.2 GPIO pad structure ):
我们对照着上图来详细看一下寄存器 IOMUXC_SW_PAD_CTL_PAD_GPIO1_IO00 的各个位的含义:
**HYS(bit16)**:对应图中 HYS,用来使能迟滞比较器,当 IO 作为输入功能的时候有效,用于设置输入接收器的施密特触发器是否使能。如果需要对输入波形进行整形的话可以使能此位。此位为 0 的时候禁止迟滞比较器,为 1 的时候使能迟滞比较器。
**PUS(bit15:14)**: 对应图中的 PUS,用来设置上下拉电阻的,一共有四种选项可以选择:
位设置 | 含义 |
---|---|
00 | 100K 下拉 |
01 | 47K 上拉 |
10 | 100K 上拉 |
11 | 22K 上拉 |
**PUE(bit13)**: 图没有给出来,当 IO 作为输入的时候,这个位用来设置 IO 使用上下拉还是状态保持器。当为 0 的时候使用状态保持器,当为 1 的时候使用上下拉。状态保持器在IO 作为输入的时候才有用,顾名思义,就是当外部电路断电以后此 IO 口可以保持住以前的状态。
**PKE(bit12)**: 对应图中的 PKE,此位用来使能或者禁止上下拉/状态保持器功能,为0 时禁止上下拉/状态保持器,为 1 时使能上下拉和状态保持器。
**ODE(bit11)**:对应图中的 ODE,当 IO 作为输出的时候,此位用来禁止或者使能开路输出,此位为 0 的时候禁止开路输出,当此位为 1 的时候就使能开路输出功能。
**SPEED(bit7:6)**: 对应图中的 SPEED,当 IO 用作输出的时候,此位用来设置 IO 速度,设置如表所示:
位设置 | 速度 |
---|---|
00 | 低速 50M |
01 | 中速 100M |
10 | 中速 100M |
11 | 最大速度 200M |
- **DSE(bit5:3)**:对应图中的 DSE,当 IO 用作输出的时候用来设置 IO 的驱动能力,总共有 8 个可选选项,如表所示:
位设置 | 速度 |
---|---|
000 | 输出驱动关闭 |
001 | R0(3.3V 下 R0 是 260Ω, 1.8V 下 R0 是 150Ω,接 DDR 的时候是 240Ω ) |
010 | R0/2 |
011 | R0/3 |
100 | R0/4 |
101 | R0/5 |
110 | R0/6 |
111 | R0/7 |
- **SRE(bit0)**: 对应图中的 SRE,设置压摆率,当此位为 0 的时候是低压摆率,当为 1的时候是高压摆率。这里的压摆率就是 IO 电平跳变所需要的时间,比如从 0 到 1 需要多少时间,时间越小波形就越陡,说明压摆率越高;反之,时间越多波形就越缓,压摆率就越低。如果产品要过 EMC 的话那就可以使用小的压摆率,因为波形缓和,如果当前所使用的 IO做高速通信的话就可以使用高压摆率。
通过上面的介绍,可以看出寄存器 IOMUXC_SW_PAD_CTL_PAD_GPIO1_IO00 是用来配置 GPIO1_IO00 的,包括速度设置、驱动能力设置、压摆率设置等等。到这里我们就知 I.MX6U 的 IO 是可以设置速度的、而且比 STM32 的设置要更多。但是我们没有看到如何设置 IO 为输入还是输出? IO 的默认电平如何设置等等,所以我们接着继续看。
4.GPIO 外设说明
IOMUXC_SW_MUX_CTL_PAD_XX_XX 和 IOMUXC_SW_PAD_CTL_PAD_XX_XX 这两种寄存器都是配置 IO 的,注意是 IO!不是 GPIO, GPIO 是一个 IO 众多复用功能中的一种。比如 GPIO1_IO00 这个 IO 可以复用为: I2C2_SCL、 GPT1_CAPTURE1、 ANATOP_OTG1_ID、ENET1_REF_CLK 、 MQS_RIGHT 、 GPIO1_IO00 、 ENET1_1588_EVENT0_IN 、SRC_SYSTEM_RESET 和 WDOG3_WDOG_B 这 9 个功能, GPIO1_IO00 是其中的一种,我们想要把 GPIO1_IO00 用作哪个外设就复用为哪个外设功能即可。
如果我们要用 GPIO1_IO00 来点个灯、作为按键输入啥的就是使用其 GPIO(通用输入输出)的功能。将其复用为 GPIO 以后还需要对其 GPIO 的功能进行配置,关于 I.MX6U 的 GPIO 请参考《IMX6UL 参考手册》的第 28 章“Chapter 28 General Purpose Input/Output (GPIO) ”, GPIO 结构如图:
左下角的 IOMUXC 框图里面就有 SW_MUX_CTL_PAD_* 和SW_PAD_CTL_PAD_*两种寄存器。这两种寄存器前面说了用来设置 IO 的复用功能和 IO 属性配置。左上角部分的 GPIO 框图就是,当 IO 用作 GPIO 的时候需要设置的寄存器,一共有八个:DR、 GDIR、 PSR、 ICR1、 ICR2、EDGE_SEL、 IMR 和 ISR。前面我们说了 I.MX6U 一共有GPIO1~GPIO5 共五组 GPIO,每组 GPIO 都有这 8 个寄存器。我们来看一下这 8 个寄存器都是什么含义。
4.1 DR 寄存器
此寄存器是数据寄存器,结构图如图 :
此寄存器是 32 位的,一个 GPIO 组最大只有 32 个 IO,因此 DR 寄存器中的每个位都对应一个 GPIO。当 GPIO 被配置为输出功能以后,向指定的位写入数据那么相应的 IO 就会输出相应的高低电平,比如要设置 GPIO1_IO00 输出高电平,那么就应该设置 GPIO1.DR=1。当 GPIO被配置为输入模式以后, 此寄存器就保存着对应 IO 的电平值,每个位对应一个 GPIO,例如,当 GPIO1_IO00 这个引脚接地的话,那么 GPIO1.DR 的 bit0 就是 0。
4.2 GDIR 寄存器
GDIR 寄存器,是方向寄存器,用来设置某个 GPIO 的工作方向的,即输入/输出, GDIR 寄存器结构如图 :
GDIR 寄存器也是 32 位的,此寄存器用来设置某个 IO 的工作方向,是输入还是输出。同样的,每个 IO 对应一个位,如果要设置 GPIO 为输入的话就设置相应的位为 0,如果要设置为输出的话就设置为 1。比如要设置 GPIO1_IO00 为输入,那么 GPIO1.GDIR=0;
4.3 PSR 寄存器
PSR 寄存器,这是 GPIO 状态寄存器,如图 :
同样的 PSR 寄存器也是一个 GPIO 对应一个位,读取相应的位即可获取对应的 GPIO 的状态,也就是 GPIO 的高低电平值。功能和输入状态下的 DR 寄存器一样。 PSR引脚状态寄存器相当于DR寄存器的简化版,它仅在GDIR方向寄存器设置为输入模式时有效, 它的每个位表示一个引脚当前的输入电平状态。PSR寄存器的权限是只读的,对它进行写操作是无效的。
4.4 ICR1 和 ICR2 寄存器
ICR1和ICR2这两个寄存器,都是中断控制寄存器, ICR1用于配置低16个GPIO,ICR2 用于配置高 16 个 GPIO, ICR1 寄存器如图:
ICR1 用于 IO0~15 的配置, ICR2 用于 IO16~31 的配置。 ICR1 寄存器中一个 GPIO 用两个位,这两个位用来配置中断的触发方式,和 STM32 的中断很类似,可配置的选线如表所示:
位设置 | 速度 |
---|---|
00 | 低电平触发 |
01 | 高电平触发 |
10 | 上升沿触发 |
11 | 下降沿触发 |
以GPIO1_IO15为例,如果要设置GPIO1_IO15为上升沿触发中断,那么GPIO1.ICR1=2<<30,如果要设置 GPIO1 的 IO16~31 的话就需要设置 ICR2 寄存器了。
4.5 IMR 寄存器
IMR 寄存器,这是中断屏蔽寄存器。一个 GPIO 对应一个位, IMR 寄存器用来控制 GPIO 的中断禁止和使能,如果使能某个 GPIO 的中断,那么设置相应的位为 1 即可,反之,如果要禁止中断,那么就设置相应的位为 0 即可。例如,要使能 GPIO1_IO00 的中断,那么就可以设置GPIO1.MIR=1 即可。
4.6 ISR 寄存器
ISR 是中断状态寄存器,寄存器如图 :
ISR 寄存器也是 32 位寄存器,一个 GPIO 对应一个位,只要某个 GPIO 的中断发生,那么ISR 中相应的位就会被置 1。所以,我们可以通过读取 ISR 寄存器来判断 GPIO 中断是否发生,相当于 ISR 中的这些位就是中断标志位。当我们处理完中断以后,必须清除中断标志位,清除方法就是向 ISR 中相应的位写 1,也就是写 1 清零。
4.7 EDGE_SEL 寄存器
EDGE_SEL 寄存器,是边沿选择寄存器,寄存器如图 :
EDGE_SEL 寄存器用来设置边沿中断,这个寄存器会覆盖 ICR1 和 ICR2 的设置,同样是一个 GPIO 对应一个位。如果相应的位被置 1,那么就相当与设置了对应的 GPIO 是上升沿和下降沿(双边沿)触发。例如,我们设置 GPIO1.EDGE_SEL=1,那么就表示 GPIO1_IO01 是双边沿触发中断,无论 GFPIO1_CR1 的设置为多少,都是双边沿触发。
5. I.MX6ULL的 GPIO 时钟使能
I.MX6U 的 GPIO 是否需要使能时钟? STM32 的每个外设都有一个外设时钟, GPIO 也不例外,要使用某个外设,必须要先使能对应的时钟。 I.MX6U 其实也一样的,每个外设的时钟都可以独立的使能或禁止,这样可以关闭掉不使用的外设时钟,起到省电的目的。
I.MX6U 的系统时钟参考《I.MX6UL 参考手册》的第 18 章“Chapter 18: ClockController Module(CCM)”,这一章主要讲解 I.MX6U 的时钟系统,很复杂。我们先不研究 I.MX6U的 时 钟 系 统 , 我们只看一下CCM里面的外设时钟使能寄存器。CCM有CCM_CCGR0 ~ CCM_CCGR6 这 7 个寄存器,这 7 个寄存器控制着 I.MX6U 的所有外设时钟开关,我们以 CCM_CCGR0 为例来看一下如何禁止或使能一个外设的时钟, CCM_CCGR0 结构体如图 :
CCM_CCGR0 是个 32 位寄存器,其中每 2 位控制一个外设的时钟,比如 bit31:30 控制着GPIO2 的外设时钟,两个位就有 4 种操作方式,如表:
位设置 | 时钟控制 |
---|---|
00 | 所有模式下都关闭外设时钟。 |
01 | 只有在运行模式下打开外设时钟,等待模式和停止模式下均关闭外设时钟。 |
10 | 未使用(保留)。 |
11 | 除了停止模式以外,其他所有模式下时钟都打开。 |
根据表中的位设置,如果我们要打开 GPIO2 的外设时钟,那么只需要设置CCM_CCGR0 的 bit31 和 bit30 都为 1 即可,也就是 CCM_CCGR0=3 << 30。反之,如果要关闭GPIO2 的外设时钟 , 那就设置 CCM_CCGR0 的 bit31 和 bit30 都为 0 即可 。
CCM_CCGR0~CCM_CCGR6 这 7 个寄存器操作都是类似的,只是不同的寄存器对应不同的外设时钟而已。 I.MX6U 的每个外设的时钟都可以独立的禁止和使能,这个和 STM32 是一样的。
三、GPIO的控制模块
在《i.MX 6ULL Applications ProcessorReference Manual》参考手册的 Chapter 28 General Purpose Input/Output (GPIO) 一节中有I.MX6ULL的GPIO模块的框图:
GPIO 的控制涉及 4 大模块: CCM、 IOMUXC、 GPIO 模块本身 ,上图中的CCM是我画上去的,该模块用于使能GPIO的时钟。接下来我们来了解一下这几个模块。
1. CCM模块
CCM 用于设置是否向 GPIO 模块提供时钟。这一部分可以参考《i.MX 6ULL Applications Processor Reference Manual》的 Chapter 18 Clock Controller Module (CCM) 。GPIOx 要用 CCM_CCGRy 寄存器中的 2 位来决定该组 GPIO 是否使能。哪组GPIO 用哪个 CCM_CCGR 寄存器来设置可以看这里:
CCM_CCGR 寄存器中某 2 位的取值含义如下
取值 | 说明 |
---|---|
00 | 该 GPIO 模块全程被关闭 |
01 | 该 GPIO 模块在 CPU run mode 情况下是使能的;在 WAIT 或 STOP 模式下,关闭 |
10 | 保留 |
11 | 该 GPIO 模块全程使能 |
2. IOMUXC 模块
IOMUXC模块用于控制引脚的模式(Mode、功能) 等,我们可以参考《i.MX 6ULL Applications Processor Reference Manual》的Chapter 32 OMUX Controller (IOMUXC) 。对于某个/某组引脚, IOMUXC 中有 2 个寄存器用来设置它 :
1 | IOMUXC_SW_MUX_CTL_PAD_<PADNAME> |
2.1 选择引脚功能
- IOMUXC_SW_MUX_CTL_PAD_<PADNAME> : Mux pad xxx,选择某个 pad 的功能
- IOMUXC_SW_MUX_CTL_GRP_<GROUP NAME>: Mux grp xxx,选择某组引脚的功能
我们来看一下命名的情况:
我们来看一下寄存器的详情,这里以 IOMUXC_SW_MUX_CTL_PAD_GPIO1_IO03 为例:
图中的MUX_MODE就是用于选择该引脚功能。SION的话是一种loopback回环模式,一般是用于测试。
2.2 设置引脚参数
- IOMUXC_SW_PAD_CTL_PAD_<PAD_NAME>: pad pad xxx,设置某个 pad 的参数
- IOMUXC_SW_PAD_CTL_GRP_<GROUP NAME>: pad grp xxx,设置某组引脚的参数
我们来看一下寄存器的详情,这里就不写了,因为其实我们之前学习过STM32的话,这些都是大差不差,无非就是参数多一些少一些,功能大体是一样的,按我自己的理解,无非就是I.MX6ULL性能比较强,可以跑操作系统,感觉就像是性能比较强的单片机,它们有很多都是想通的。
3. GPIO模块内部
内部框图如下:
我们暂时只需要关心 3 个寄存器 :
(1)GPIOx_GDIR:设置引脚方向,每位对应一个引脚, 1-output, 0-input
(2)GPIOx_DR:设置输出引脚的电平,每位对应一个引脚, 1-高电平, 0-低电平
(3)GPIOx_PSR:读取引脚的电平,每位对应一个引脚, 1-高电平, 0-低电平
4. 如何编程?
可以参考《i.MX 6ULL Applications Processor Reference Manual》的28.4.3 GPIO Programming 。
4.1 读GPIO
(1)设置 CCM_CCGRx 寄存器中某位使能对应的 GPIO 模块 ,默认是使能的,上图省略了
(2)设置 IOMUX 来选择引脚用于 GPIO
(3)设置 GPIOx_GDIR 中某位为 0,把该引脚设置为输入功能
(4)读 GPIOx_DR 或 GPIOx_PSR 得到某位的值(读 GPIOx_DR 返回的是GPIOx_PSR 的值)
4.2 写GPIO
(1)设置 CCM_CCGRx 寄存器中某位使能对应的 GPIO 模块 ,默认是使能的,上图省略了
(2)设置 IOMUX 来选择引脚用于 GPIO
(3)设置 GPIOx_GDIR 中某位为 1,把该引脚设置为输出功能
需要注意的是, 可以设置该引脚的 loopback 功能,这样就可以从GPIOx_PSR 中读到引脚的有实电平;从 GPIOx_DR 中读回的只是上次设置的值,它并不能反应引脚的真实电平,比如可能因为硬件故障导致该引脚跟地短路了,通过设置 GPIOx_DR 让它输出高电平并不会起效果。
四、GPIO使用总结
要将 I.MX6U 的 IO 作为 GPIO 使用,我们需要一下几步:
①、使能 GPIO 对应的时钟。
②、设置寄存器 IOMUXC_SW_MUX_CTL_PAD_XX_XX,设置 IO 的复用功能,使其复用为 GPIO 功能。
③、设置寄存器 IOMUXC_SW_PAD_CTL_PAD_XX_XX,设置 IO 的上下拉、速度等等。
④、第②步已经将 IO 复用为了 GPIO 功能,所以需要配置 GPIO,设置输入/输出、是否使用中断、默认输出电平等。