LV04-02-点灯-01-IMX6ULL的GPIO
本文主要是I.MX6ULL的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)数值:对于输出引脚,可以设置寄存器让它输出高、低电平
对于输入引脚,可以读取寄存器得到引脚的当前电平。
2. I.MX6ULL的 GPIO 数量
I.MX6ULL有 5 组 GPIO( GPIO1~GPIO5),每组引脚最多有 32 个,但是可能实际上并没有那么多。
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;
3. I.MX6ULL的 IO 命名
像之前学习的STM32 中的 IO 都是 PA0~15、 PB0~15 这样命名的, I.MX6ULL 的 IO 是怎么命名的呢?打开《i.MX 6ULL Applications Processor Reference Manual》参考手册的 Chapter 32 IOMUX Controller (IOMUXC) ,可以看到这一章节对应的书签如下:
I.MX6ULL 的 IO 分为两类: SNVS 域的和通用的,这两类 IO 本质上都是一样的,我们就有下面的常用 IO 为例,说明一下 I.MX6ULL 的 IO 命名方式。
形如“ IOMUXC_SW_MUX_CTL_PAD_GPIO1_IO00 ”的就是 GPIO 命名,命名形式就是“IOMUXC_SW_MUC_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 的发送引脚。
“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 也是可以复用为其它外设引脚的,关于复用,后边再说。
4. I.MX6ULL的 IO 复用
以“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 对应的复用配置寄存器。
5. I.MX6ULL的 IO 配置
我们会发现《 I.MX6UL 参考手册》 第 30 章“ Chapter 30: 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 的默认电平如何设置等等,所以我们接着继续看。
6. I.MX6ULL的 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 参考手册》的第 26章“Chapter 26 General Purpose Input/Ouput(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 个寄存器都是什么含义。
6.1 DR 寄存器
此寄存器是数据寄存器,结构图如图 :
此寄存器是 32 位的,一个 GPIO 组最大只有 32 个 IO,因此 DR 寄存器中的每个位都对应一个 GPIO。当 GPIO 被配置为输出功能以后,向指定的位写入数据那么相应的 IO 就会输出相应的高低电平,比如要设置 GPIO1_IO00 输出高电平,那么就应该设置 GPIO1.DR=1。当 GPIO被配置为输入模式以后, 此寄存器就保存着对应 IO 的电平值,每个位对对应一个 GPIO,例如,当 GPIO1_IO00 这个引脚接地的话,那么 GPIO1.DR 的 bit0 就是 0。
6.2 GDIR 寄存器
GDIR 寄存器,是方向寄存器,用来设置某个 GPIO 的工作方向的,即输入/输出, GDIR 寄存器结构如图 :
GDIR 寄存器也是 32 位的,此寄存器用来设置某个 IO 的工作方向,是输入还是输出。同样的,每个 IO 对应一个位,如果要设置 GPIO 为输入的话就设置相应的位为 0,如果要设置为输出的话就设置为 1。比如要设置 GPIO1_IO00 为输入,那么 GPIO1.GDIR=0;
6.3 PSR 寄存器
PSR 寄存器,这是 GPIO 状态寄存器,如图 :
同样的 PSR 寄存器也是一个 GPIO 对应一个位,读取相应的位即可获取对应的 GPIO 的状态,也就是 GPIO 的高低电平值。功能和输入状态下的 DR 寄存器一样。
6.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 寄存器了。
6.5 IMR 寄存器
IMR 寄存器,这是中断屏蔽寄存器。一个 GPIO 对应一个位, IMR 寄存器用来控制 GPIO 的中断禁止和使能,如果使能某个 GPIO 的中断,那么设置相应的位为 1 即可,反之,如果要禁止中断,那么就设置相应的位为 0 即可。例如,要使能 GPIO1_IO00 的中断,那么就可以设置GPIO1.MIR=1 即可。
6.6 ISR 寄存器
ISR 是中断状态寄存器,寄存器如图 :
ISR 寄存器也是 32 位寄存器,一个 GPIO 对应一个位,只要某个 GPIO 的中断发生,那么ISR 中相应的位就会被置 1。所以,我们可以通过读取 ISR 寄存器来判断 GPIO 中断是否发生,相当于 ISR 中的这些位就是中断标志位。当我们处理完中断以后,必须清除中断标志位,清除方法就是向 ISR 中相应的位写 1,也就是写 1 清零。
6.7 EDGE_SEL 寄存器
EDGE_SEL 寄存器,是边沿选择寄存器,寄存器如图 :
EDGE_SEL 寄存器用来设置边沿中断,这个寄存器会覆盖 ICR1 和 ICR2 的设置,同样是一个 GPIO 对应一个位。如果相应的位被置 1,那么就相当与设置了对应的 GPIO 是上升沿和下降沿(双边沿)触发。例如,我们设置 GPIO1.EDGE_SEL=1,那么就表示 GPIO1_IO01 是双边沿触发中断,无论 GFPIO1_CR1 的设置为多少,都是双边沿触发。
7. 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里面的外设时钟使能寄存器。CMM有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 的所有外设时钟都打开了。 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,把该引脚设置为输出功能
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 |
需要注意的是, 可以设置该引脚的 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,设置输入/输出、是否使用中断、默认输出电平等。