17. EXTI—外部中断/事件控制器

本章参考资料:《STM32H743用户手册》系统配置控制器以及中断和事件章节。

上一章节我们已经详细介绍了NVIC,对STM32H7xx中断管理系统有个全局的了解,我们这章的内容是NVIC的实例应用,也是STM32H7xx控制器非常重要的一个资源。学习本章时,配合《STM32H743用户手册》系统配置控制器以及中断和事件章节一起阅读,效果会更佳,特别是涉及到寄存器说明的部分。

特别说明,本书内容是以STM32H743xx系列控制器资源讲解。

17.1. EXTI简介

外部中断/事件控制器(EXTI)管理了控制器的25个中断/事件线。每个中断/事件线都对应有一个边沿检测器,可以实现输入信号的上升沿检测和下降沿的检测。EXTI可以实现对每个中断/事件线进行单独配置,可以单独配置为中断或者事件,以及触发事件的属性。

17.2. EXTI功能框图

EXTI的功能框图包含了EXTI最核心内容,掌握了功能框图,对EXTI就有一个整体的把握,在编程时就思路就非常清晰。EXTI功能框图见 图17_1。 EXTI由三个部分组成,分别是通过APB总线访问的寄存器块,事件输入触发块和屏蔽块。其中寄存器块包含了所以得EXTI寄存器, 事件输入触发块主要提供事件输入边沿触发逻辑。事件屏蔽主要是提供不同的唤醒事件,中断/事件输出以及中断/事件屏蔽分配。

图 17‑1 EXTI功能框图

图 17‑1-0 EXTI功能框图

EXTI可分为两大部分功能,一个是产生中断,另一个是产生事件,这两个功能从硬件上就有所不同。

图 17‑1-1 事件配置触发框图

图 17‑1-1 事件配置触发框图

首先我们来看 图17_1_1 红色方框指示的电路流程。它是一个产生中断的线路,最终信号流入到NVIC控制器内。

编号1是输入线,EXTI控制器有88个中断/事件输入线,这些输入线可以通过寄存器设置为任意一个GPIO,也可以是一些外设的事件,这部分内容我们将在后面专门讲解。输入线一般是存在电平变化的信号。除此之外,还有一个软件中断/事件控制寄存器(EXTI_SWIERx),EXTI_SWIER允许我们通过程序控制就可以启动中断/事件线,这在某些地方非常有用。

编号2是一个边沿检测电路,它会根据上升沿触发选择寄存器(EXTI_RTSR)和下降沿触发选择寄存器(EXTI_FTSR)对应位的设置来控制信号触发。EXTI_RTSR和EXTI_FTSR两个寄存器可以控制器需要检测哪些类型的电平跳变过程,可以是只有上升沿触发、只有下降沿触发或者上升沿和下降沿都触发。

编号3电路实际就是一个或门电路,它一个输入来自编号2电路,另外一输入来自挂起请求寄存器(EXTI_CPUPR)。我们知道或门的作用就是有1就为1,所以这两个输入随便一个是有效信号1就可以输出1给编号4电路。

编号4电路是一个或门电路,它有两个输入,分别来自于事件屏蔽寄存器(EXTI_EMR)和来自中断屏蔽寄存器(EXTI_IMR)。这两个输入只要有一个信号为1,则输出1给编号5电路。

编号5是一个与门电路,它的一个输入来自编号3电路,另外一个输入来自编号4电路。与门电路要求输入都为1才输出1,导致的结果是:如果编号4电路输出0,即EXTI_EMR和EXTI_IMR都设置为0时,不管编号3电路的输出信号是1还是0,最终编号5都会输出0;如果EXTI_EMR和EXTI_IMR两者之中有一个为1,则最终编号5电路输出的信号才由编号3电路的输出信号决定的。

编号6是将EXTI_PR寄存器内容输出到NVIC内,从而实现系统中断事件控制。

接下来我们来看看绿色虚线指示的电路流程。它是一个产生事件的线路,最终输出一个脉冲信号。

产生事件线路是在编号3电路之后与中断线路有所不同,之前电路都是共用的。编号6电路是一个与门,它一个输入编号3电路,另外一个输入来自事件屏蔽寄存器(EXTI_EMR)。如果EXTI_EMR设置为0时,那不管编号3电路的输出信号是1还是0,最终编号6电路输出的信号都为0;如果EXTI_EMR设置为1时,最终编号6电路输出的信号才由编号3电路的输出信号决定,这样我们可以简单的控制EXTI_EMR来实现是否产生事件的目的。

编号7是一个脉冲发生器电路,当它的输入端,即编号6电路的输出端,是一个有效信号1时就会产生一个脉冲;如果输入端是无效信号就不会输出脉冲。

编号8是一个脉冲信号,就是产生事件的线路最终的产物,这个脉冲信号可以给其他外设电路使用,比如定时器TIM、模拟数字转换器ADC等等。 产生中断线路目的是把输入信号输入到NVIC,进一步会运行中断服务函数,实现功能,这样是软件级的。而产生事件线路目的就是传输一个脉冲信号给其他外设使用,并且是电路级别的信号传输,属于硬件级的。

另外,EXTI是在APB2总线上的,在编程时候需要注意到这点。

17.3. 中断/事件线

EXTI有25个中断/事件线,每个GPIO都可以被设置为输入线,占用EXTI0至EXTI15,还有另外七根用于特定的外设事件,见 表17_1

七根特定外设中断/事件线由外设触发,具体用法参考《STM32F7xx参考手册》中对外设的具体说明。

表 17‑1 EXTI中断/事件线

中断/事件线

输入源

EXTI0

PX0(X可为A,B,C,D,E,F,G,H,I)

EXTI1

PX1(X可为A,B,C,D,E,F,G,H,I)

EXTI2

PX2(X可为A,B,C,D,E,F,G,H,I)

EXTI3

PX3(X可为A,B,C,D,E,F,G,H,I)

EXTI4

PX4(X可为A,B,C,D,E,F,G,H,I)

EXTI5

PX5(X可为A,B,C,D,E,F,G,H,I)

EXTI6

PX6(X可为A,B,C,D,E,F,G,H,I)

EXTI7

PX7(X可为A,B,C,D,E,F,G,H,I)

EXTI8

PX8(X可为A,B,C,D,E,F,G,H,I)

EXTI9

PX9(X可为A,B,C,D,E,F,G,H,I)

EXTI10

PX10(X可为A,B,C,D,E,F,G,H,I)

EXTI11

PX11(X可为A,B,C,D,E,F,G,H,I)

EXTI12

PX12(X可为A,B,C,D,E,F,G,H,I)

EXTI13

PX13(X可为A,B,C,D,E,F,G,H,I)

EXTI14

PX14(X可为A,B,C,D,E,F,G,H,I)

EXTI15

PX15(X可为A,B,C,D,E,F,G,H)

EXTI0至EXTI15用于GPIO,通过编程控制可以实现任意一个GPIO作为EXTI的输入源。由 表17_1 可知, EXTI0可以通过SYSCFG外部中断配置寄存器1(SYSCFG_EXTICR1)的EXTI0[3:0]位选择配置为PA0、PB0、PC0、PD0、PE0、PF0、PG0、PH0或者PI0, 见 图17_2。其他EXTI线(EXTI中断/事件线)使用配置都是类似的。

图 17‑2 EXTI0输入源选择

图 17‑2 EXTI0输入源选择

17.4. EXTI初始化详解

HAL库函数的EXIT初始化非常简单,只需配置好IO口的模式,然后配置中断源、中断优先级、使能中断。

  1. HAL_NVIC_SetPriority:该函数负责EXTI中断/事件线选择,可选EXTI0至EXTI15,可参考表 17-1选择,配置优先级。

  2. HAL_NVIC_EnableIRQ:该函数负责控制使能中断。

17.5. 外部中断控制实验

中断在嵌入式应用中占有非常重要的地位,几乎每个控制器都有中断功能。中断对保证紧急事件得到第一时间处理是非常重要的

我们设计使用外接的按键来作为触发源,使得控制器产生中断,并在中断服务函数中实现控制RGB彩灯的任务。

17.5.1. 硬件设计

轻触按键在按下时会使得引脚接通,通过电路设计可以使得按下时产生电平变化,见 图17_3

图 17‑3 按键电路设计

图 17‑3 按键电路设计

17.5.2. 软件设计

这里只讲解核心的部分代码,有些变量的设置,头文件的包含等并没有涉及到,完整的代码请参考本章配套的工程。我们创建了两个文件:bsp_exti.c和bsp_exti.h文件用来存放EXTI驱动程序及相关宏定义,中断服务函数放在stm32h7xx_it.c文件中。

17.5.2.1. 编程要点

  1. 初始化系统时钟;

  2. 初始化RGB彩灯的GPIO;

  3. 开启按键GPIO时钟;

  4. 配置NVIC;

  5. 配置按键GPIO为输入模式;

  6. 将按键GPIO连接到EXTI源输入;

  7. 配置按键EXTI中断/事件线;

  8. 编写EXTI中断服务函数。

17.5.2.2. 软件分析

17.5.2.2.1. 按键和EXTI宏定义

代码清单 17‑1 按键和EXTI 宏定义

 #define KEY1_INT_GPIO_PORT                GPIOA
 #define KEY1_INT_GPIO_CLK_ENABLE()        __GPIOA_CLK_ENABLE();
 #define KEY1_INT_GPIO_PIN                 GPIO_PIN_0
 #define KEY1_INT_EXTI_IRQ                 EXTI0_IRQn
 #define KEY1_IRQHandler                   EXTI0_IRQHandler

 #define KEY2_INT_GPIO_PORT                GPIOC
 #define KEY2_INT_GPIO_CLK_ENABLE()        __GPIOA_CLK_ENABLE();
 #define KEY2_INT_GPIO_PIN                 GPIO_PIN_13
 #define KEY2_INT_EXTI_IRQ                 EXTI15_10_IRQn
 #define KEY2_IRQHandler                   EXTI15_10_IRQHandler

使用宏定义方法指定与电路设计相关配置,这对于程序移植或升级非常有用的。

17.5.2.2.2. EXTI中断配置

代码清单 17‑2 EXTI中断配置(bsp_exit.c文件)

 void EXTI_Key_Config(void)
 {
     GPIO_InitTypeDef GPIO_InitStructure;

     /*开启按键GPIO口的时钟*/
     KEY1_INT_GPIO_CLK_ENABLE();
     KEY2_INT_GPIO_CLK_ENABLE();

     /* 选择按键1的引脚 */
     GPIO_InitStructure.Pin = KEY1_INT_GPIO_PIN;
     /* 设置引脚为输入模式 */
     GPIO_InitStructure.Mode = GPIO_MODE_IT_RISING;
     /* 设置引脚不上拉也不下拉 */
     GPIO_InitStructure.Pull = GPIO_NOPULL;
     /* 使用上面的结构体初始化按键 */
     HAL_GPIO_Init(KEY1_INT_GPIO_PORT, &GPIO_InitStructure);
     /* 配置 EXTI 中断源 到key1 引脚、配置中断优先级*/
     HAL_NVIC_SetPriority(KEY1_INT_EXTI_IRQ, 0, 0);
     /* 使能中断 */
     HAL_NVIC_EnableIRQ(KEY1_INT_EXTI_IRQ);

     /* 选择按键2的引脚 */
     GPIO_InitStructure.Pin = KEY2_INT_GPIO_PIN;
     /* 其他配置与上面相同 */
     HAL_GPIO_Init(KEY2_INT_GPIO_PORT, &GPIO_InitStructure);
     /* 配置 EXTI 中断源 到key1 引脚、配置中断优先级*/
     HAL_NVIC_SetPriority(KEY2_INT_EXTI_IRQ, 0, 0);
     /* 使能中断 */
     HAL_NVIC_EnableIRQ(KEY2_INT_EXTI_IRQ);
 }

首先,使用GPIO_InitTypeDef结构体定义用于GPIO初始化配置的变量,关于这个结构体前面都已经做了详细的讲解。

使用GPIO之前必须开启GPIO端口的时钟;

调用HAL_NVIC_SetPriority和HAL_NVIC_EnableIRQ函数完成对按键1、按键2优先级配置并使能中断通道。

作为中断/时间输入线把GPIO配置为中断上升沿触发模式,这里不使用上拉或下拉,有外部电路完全决定引脚的状态。

我们的目的是产生中断,执行中断服务函数,EXTI选择中断模式,按键1使用下降沿触发方式,并使能EXTI线。

按键2基本上采用与按键1相关参数配置。

17.5.2.2.3. EXTI中断服务函数

代码清单 17‑3 EXTI中断服务函数

 void KEY1_IRQHandler(void)
 {
     //确保是否产生了EXTI Line中断
     if (__HAL_GPIO_EXTI_GET_IT(KEY1_INT_GPIO_PIN) != RESET) {
         // LED1 取反
         LED1_TOGGLE;
         //清除中断标志位
         __HAL_GPIO_EXTI_CLEAR_IT(KEY1_INT_GPIO_PIN);
     }
 }

 void KEY2_IRQHandler(void)
 {
     //确保是否产生了EXTI Line中断
     if (__HAL_GPIO_EXTI_GET_IT(KEY2_INT_GPIO_PIN) != RESET) {
         // LED2 取反
         LED2_TOGGLE;
         //清除中断标志位
         __HAL_GPIO_EXTI_CLEAR_IT(KEY2_INT_GPIO_PIN);
     }
 }

当中断发生时,对应的中断服务函数就会被执行,我们可以在中断服务函数实现一些控制。

一般为确保中断确实发生,我们会在中断服务函数调用中断标志位状态读取函数读取外设中断标志位并判断标志位状态。

__HAL_GPIO_EXTI_GET_IT函数用来获取EXTI的中断标志位状态,如果EXTI线有中断发生函数返回“SET”否则返回“RESET”。实际上,__HAL_GPIO_EXTI_GET_IT函数是通过读取EXTI_PR寄存器值来判断EXTI线状态的。

按键1的中断服务函数我们让LED1翻转其状态,按键2的中断服务函数我们让LED2翻转其状态。执行任务后需要调用__HAL_GPIO_EXTI_CLEAR_IT函数清除EXTI线的中断标志位。

17.5.2.2.4. 主函数

代码清单 17‑4 主函数

 int main(void)
 {
     /* 系统时钟初始化成400 MHz */
     SystemClock_Config();
     /* LED 端口初始化 */
     LED_GPIO_Config();

     /* 初始化EXTI中断,按下按键会触发中断,
     *  触发中断会进入stm32f7xx_it.c文件中的函数
     *
     KEY1_IRQHandler和KEY2_IRQHandler,处理中断,反转LED灯。
     */
     EXTI_Key_Config();

     /* 等待中断,由于使用中断方式,CPU不用轮询按键
     */
     while (1) {
     }
 }

主函数非常简单,只有三个任务函数。SystemClock_Config 初始化系统时钟, LED_GPIO_Config函数定义在bsp_led.c文件内,完成RGB彩灯的GPIO初始化配置。EXTI_Key_Config函数完成两个按键的GPIO和EXTI配置。

17.5.3. 下载验证

保证开发板相关硬件连接正确,把编译好的程序下载到开发板。此时RGB彩色灯是暗的,如果我们按下开发板上的按键1, RGB彩灯变亮,再按下按键1,RGB彩灯又变暗;如果我们按下开发板上的按键2并弹开,RGB彩灯变亮, 再按下开发板上的KEY2并弹开,RGB彩灯又变暗。