1. 外部中断实验
1.1 NVIC和EXTI简介
1.1.1 NVIC简介
NVIC 即嵌套向量中断控制器,全称 Nested vectored interrupt controller。是ARM Cortex-M处理器中用于管理中断的重要组件。负责处理中断请求,分配优先级,并协调中断的触发和响应。
它是内核的器件,所以它的更多描述可以看内核有关的资料。
M3/M4/M7 内核都是支持 256 个中断,其中包含了 16 个系统中断和 240 个外部中断,并且具有 256 级的可编程中断设置。芯片厂商一般不会把内核的这些资源全部用完,如 STM32F429 的系统中断有 10 个,外部中断有 86 个。
中断的宏定义可以在stm32fxxx.h里面找,中断向量表在startup_stm32fxxxx.s启动文件里面找
1.1.2 NVIC寄存器
NVIC 相关的寄存器定义了可以在 core_cm4.h 文件中找到。
typedef struct
{
__IO uint32_t ISER[8U]; /* 中断使能寄存器 */
uint32_t RESERVED0[24U];
__IO uint32_t ICER[8U]; /* 中断除能寄存器 */
uint32_t RSERVED1[24U];
__IO uint32_t ISPR[8U]; /* 中断使能挂起寄存器 */
uint32_t RESERVED2[24U];
__IO uint32_t ICPR[8U]; /* 中断解挂寄存器 */
uint32_t RESERVED3[24U];
__IO uint32_t IABR[8U]; /* 中断有效位寄存器 */
uint32_t RESERVED4[56U];
__IOM uint8_t IP[240U]; /* 中断优先级寄存器(8Bit 位宽) */
uint32_t RESERVED5[644U];
__O uint32_t STIR; /* 中断触发中断寄存器 */
} NVIC_Type;
STM32F429 的中断在这些寄存器的控制下有序的执行的。只有了解这些中断寄存器,才
能方便的使用 STM32F429 的中断。下面重点介绍这几个寄存器:
ISER[8]:ISER 全称是:Interrupt Set Enable Registers,这是一个中断使能寄存器组,包含8个32位寄存器,每个位控制一个中断。但是STM32F429 的可屏蔽中断最多只有 86 个,所以只用到前4个(ISER[0~3]),总共可以表示 128 个中断。要使能某个中断,必须设置相应的ISER位为1,使该中断被使能(这里仅仅是使能,还要配合中断分组、屏蔽、IO 口映射等设置才算是一个完整的中断设置)。具体每一位对应哪个中断,请参考 STM32F429xx.h文件。
ICER[8]:全称是:Interrupt Clear Enable Registers,是一个中断除能寄存器组。该寄存器组与 ISER 的作用恰好相反,是用来清除某个中断的使能的。其对应位的功能,也和 ICER 一样。这里要专门设置一个 ICER 来清除中断位,而不是向 ISER 写 0 来清除,是因为 NVIC 的这些寄存器都是写 1 有效的,写 0 是无效的。
ISPR[8]:全称是:Interrupt Set Pending Registers,是一个中断挂起控制寄存器组。每个位对应的中断和 ISER 是一样的。通过置 1,可以将正在进行的中断挂起,而执行同级或更高级别的中断。写 0 是无效的。
ICPR[8]:全称是:Interrupt Clear Pending Registers,是一个中断解挂控制寄存器组。其作
用与 ISPR 相反,对应位也和 ISER 是一样的。通过设置 1,可以将挂起的中断解挂。写 0 无效。
IABR[8]:全称是:Interrupt Active Bit Registers,是一个中断激活标志位寄存器组。对应位所代表的中断和 ISER 一样,如果为 1,则表示该位所对应的中断正在被执行。这是一个只读寄存器,通过它可以知道当前在执行的中断是哪一个。在中断执行完了由硬件自动清零。
IPR[240]:全称是:Interrupt Priority Registers,是一个中断优先级控制的寄存器组。IP 寄存器组由 240 个 8bit 的寄存器组成,每个可屏蔽中断占用 8bit,这样总共可以表示 240 个可屏蔽中断。而STM32F429只用到了其中的 86个。IP[85]~IP[0]分别对应中断 85~0。而每个可屏蔽中断占用的8bit 并没有全部使用,而是只用了高 4 位。这 4 位,又分为抢占优先级和子优先级。抢占优先级在前,子优先级在后。而这两个优先级各占几个位又要根据 SCB->AIRCR 中的中断分组设置来决定。
NVIC寄存器总结
NVIC,嵌套向量中断控制器,ARM Cortex-M内核的组件,
用来中断使能、挂起、记录中断激活标志位、配置中断优先级。
分别对应寄存器组有
中断使能和除能 ISER[8] ICER[8] 8个*32位寄存器
中断挂起和解挂 ISPR[8] ICPR[8]
记录中断激活标志位 IABR[8] Interrupt Active Bit Registers
配置中断优先级 IP[240] 240*8位寄存器,仅用高4位,放抢占优先级和子优先级
SCB->AIRCR的中断分组设置决定抢占优先级和子优先级占几位
1.1.3 中断优先级
STM32 中的中断优先级可以分为:抢占式优先级和响应优先级,响应优先级也称子优先
级,每个中断源都需要被指定这两种优先级。
抢占优先级:抢占优先级高的中断 可以 打断正在执行的抢占优先级低的中断。
响应优先级:抢占优先级相同,响应优先级高的中断 不能 打断响应优先级低的中断。但可以按响应优先级顺序先后执行。
当多个中断的抢占式优先级和响应优先级相同时,那么就遵循自然优先级,看中断向量表的中断排序,数值越小,优先级越高。
在NVIC中有 IPR[60] 控制中断优先级。每个 IPR 寄存器每8位又分为一组,可以分4组,所以就有了 240 组宽度为 8bit 的中断优先级控制寄存器,原则上每个外部中断可配置的优先级为 0~255,数值越小,优先级越高。但是实际上 M3 /M4 /M7 芯片为了精简设计,只使用了高四位[7:4],低四位取零,这样以至于最多只有 16 级中断嵌套,即 2^4=16。
正点原子开发指南V1.0该章节所说的 IP[240]和 IPR0~IPR59 其实是一个东西,指的是中断优先级控制的寄存器组,60个32位 IPR 寄存器,8位一组,只使用高4位,由SCB->AIRCR设置抢占优先级和响应优先级分别占几位
1.1.4 NVIC相关函数
ST 公司把 core_cm4.h 文件的 NVIC 相关函数封装到 stm32f4xx_hal_cortex.c 文件中。
1.1.4.1 中断优先级分组
首先要讲解的是中断优先级分组函数 NVIC_SetPriorityGrouping,其声明如下:
void HAL_NVIC_SetPriorityGrouping(uint32_t PriorityGroup);
函数形参:
NVIC_PRIORITYGROUP_0 到 NVIC_PRIORITYGROUP_4(共5组)
这个函数唯一目的就是通过设置 SCB->AIRCR 寄存器来设置中断优先级分组。
1.1.4.2 中断优先级设置
中断优先级设置函数 NVIC_SetPriority,其声明如下:
void HAL_NVIC_SetPriority(IRQn_Type IRQn, uint32_t PreemptPriority,
uint32_t SubPriority);
函数形参:
形参1 中断号, IRQn_Type 定义的枚举类型,定义在 stm32f429xx.h
形参2 抢占优先级
形参3 子优先级
用于设置中断的抢占优先级和子优先级
1.1.4.3 中断使能
中断使能函数 NVIC_EnableIRQ,其声明如下:
void HAL_NVIC_EnableIRQ(IRQn_Type IRQn);
1.1.4.4 中断除能
中断除能函数 NVIC_DisableIRQ,其声明如下:
void HAL_NVIC_disableIRQ(IRQn_Type IRQn);
1.1.4.5 系统复位
系统复位函数 HAL_NVIC_SystemReset 函数,其声明如下:
void HAL_NVIC_SystemReset(void);
//SCB(System Control Block)寄存器中的特定位,例如SCB->AIRCR寄存器的位[2]:SYSRESETREQ。
//通过将这个位设置为1,可以触发系统的软件复位(Software System Reset)。
用于软件复位系统。
1.1.5 EXTI简介
EXTI 即是外部中断和事件控制器,它由三个部分组成:APB 接口访问的寄存器模块、事件输入触发模块和屏蔽模块。其中寄存器块包含了所有 EXTI 寄存器,事件输入触发模块提供事件输入边沿触发逻辑,屏蔽模块为不同的唤醒·、中断和事件输出及其屏蔽功能提供事件分配。EXTI 的功能框图如下图所示:
从 EXTI 功能框图可以看到有两条主线,一条是由输入线到 NVIC 中断控制器,一条是由输入线到脉冲发生器。这就恰恰是 EXTI 的两大部分功能,产生中断与产生事件,两者从硬件上就存在不同。
EXTI 功能框图的产生中断的线路,最终信号是流入 NVIC 中断控制器中。
下面让我们看一下 EXTI 功能框图的产生中断的线路,最终信号是流入 NVIC 控制器中。输入线是线路的信息输入端,它可以通过配置寄存器设置为任何一个 GPIO 口,或者是一些外设事件。
标号①是一个边沿检测电路,包括边沿检测电路,上升沿触发选择寄存器(EXTI_RTSR)和
下降沿触发选择寄存器(EXTI_FTSR)。
标号②是一个或门电路,它的两个信号输入端分别是软件中断事件寄存器(EXTI_SWIER)和边沿检测电路的输入信号。或门电路只要输入端有信号‘1’,就会输出‘1’,所以就会输出‘1’到标号③电路和标号④电路。
标号③是一个与门电路,它的两个信号输入端分别是中断屏蔽寄存器(EXTI_IMR)和标号②电路输出信号。与门电路要求输入都为‘1’才输出‘1’,这样子的情况下,如果中断屏蔽寄存器(EXTI_IMR)设置为 0 时,不管从标号②电路输出的信号特性如何,最终标号③电路输出的信号都是 0;假如中断屏蔽寄存器(EXTI_IMR)设置为 1 时,最终标号③电路输出的信号才由标号②电路输出信号决定,这样子就可以简单控制 EXTI_IMR来实现中断的目的。标号④电路输出‘1’就会把请求挂起寄存器(EXTI_PR)对应位置 1。
最后,请求挂起寄存器(EXTI_PR)的内容就输出到 NVIC 内,实现系统中断事件的控制。
接下来我们看看 EXTI 功能框图的产生事件的线路。
产生事件线路是从标号 2 (软件中断事件寄存器)之后与中断线路有所不用,之前的线路都是共用的。标号④是一个与门,输入端来自标号 2 电路以及来自于事件屏蔽寄存器(EXTI_EMR)。
标号④电路输出有效信号 1 就会使脉冲发生器电路产生一个脉冲,而无效信号就不会使其产生脉冲信号。脉冲信号产生可以给其他外设电路使用,例如定时器,模拟数字转换器等,这样的脉冲信号一般用来触发 TIM 或者 ADC 开始转换。
产生中断线路目的使把输入信号输入到 NVIC,进一步运行中断服务函数,实现功能。而产生事件线路目的是传输一个脉冲信号给其他外设使用,属于硬件级功能。
EXTI功能框图总结
EXTI功能框图有两条线路,
一条是中断发生线路:
GPIO 输入脉冲信号,经过
1. 边沿检测电路
2. 上升沿、下降沿触发选择器
3. 软件中断事件寄存器
4. 中断屏蔽寄存器
5. 中断挂起请求寄存器
最后输入 NVIC,进行中断控制
一条是事件发生线路:
GPIO 输入脉冲信号,经过
1. 边沿检测电路
2. 上升沿、下降沿触发选择器
3. 软件中断事件寄存器
4. 事件屏蔽寄存器
5. 脉冲发生器
最后产生脉冲信号,给其他外设使用
EXTI支持 22个外部中断/事件请求,这些都是信息输入端,也就是上面提及到了输入线,具体如下:
EXTI 线 0~15:对应外部 IO 口的输入中断
EXTI 线 16:连接到 PVD 输出
EXTI 线 17:连接到 RTC 闹钟事件
EXTI 线 18:连接到 USB OTG FS 唤醒事件
EXTI 线 19:连接到以太网唤醒事件
EXTI 线 20:连接到 USB OTG HS(在 FS 中配置)唤醒事件
EXTI 线 21:连接到 RTC 入侵和时间戳事件
EXTI 线 22:连接到 RTC 唤醒事件
STM32F4 供给 IO 口使用的中断线只有 16 个,每个中断线对应了最多 9 个 IO 口,以线 0 为例:它对应了 GPIOA.0、GPIOB.0、GPIOC.0、GPIOD.0、GPIOE.0、GPIOF.0、GPIOG.0、GPIOH.0、GPIOI.0。而中断线每次只能连接到 1 个 IO 口上,这样就需要通过配置决定对应的中断线配置到哪个 GPIO 上了
EXTI 程序设计
例程功能:
通过外部中断的方式,让开发板上的独立按键控制LED灯。
思路:
一个按键按下,信号输入GPIO,触发外部中断点灯。
EXTI 外部中断配置步骤:
1)使能按键的 IO 口时钟
__HAL_RCC_GPIOH_CLK_ENABLE()
2)设置按键的 IO 口模式,触发条件,开启 SYSCFG 时钟,设置 IO 口与中断线的映射关系
3) 配置中断优先级并使能中断
//HAL_INIT函数中设置中断优先级分组,配置Systic时钟
//GPIO_INIT函数配置了SYSCFG时钟
GPIO_InitTypeDef gpio_init_struct;
gpio_init_struct.Pin = GPIO_PIN_3; /* KEY0引脚 */
gpio_init_struct.Mode = GPIO_MODE_INPUT; /* 输入 */
gpio_init_struct.Pull = GPIO_PULLUP; /* 上拉 */
gpio_init_struct.Speed = GPIO_SPEED_FREQ_HIGH; /* 高速 */
gpio_init_struct.Mode = GPIO_MODE_IT_FALLING; /* 下降沿触发 */
HAL_GPIO_Init(KEY0_GPIO_PORT, &gpio_init_struct); /* KEY0引脚模式设置,上拉输入
HAL_NVIC_SetPriority(KEY0_INT_IRQn, 0, 2); /* 抢占0,子优先级2 */
HAL_NVIC_EnableIRQ(EXTI3_IRQn); /* 使能中断线3 */
4)编写中断服务函数
每开启一个中断,就必须编写其对应的中断服务函数,否则将导致死机(CPU 将找不到中断服务函数)。中断服务函数接口厂家已经在 startup_stm32f429xx.s 中做好了。STM32F4 的 IO 口外部中断函数只有 7 个,分别为:
void EXTI0_IRQHandler();
void EXTI1_IRQHandler();
void EXTI2_IRQHandler();
void EXTI3_IRQHandler();
void EXTI4_IRQHandler();
void EXTI9_5_IRQHandler();
void EXTI15_10_IRQHandler();
中断线 0-4,每个中断线对应一个中断函数,中断线 5-9 用中断函数 EXTI9_5_IRQHandler,中断线10-15共用中断函数 EXTI15_10_IRQHandler。一般情况下,我们可以把中断控制逻辑直接编写在中断服务函数中,但是 HAL 库把中断处理过程进行了简单封装,请看下面步骤 5 讲解。
5) 编写中断处理回调函数 GPIO_EXTI_Callback
HAL 库为了用户使用方便,提供了一个中断通用入口函数 HAL_GPIO_EXTI_IRQHandler,
在该函数内部直接调用回调函数 GPIO_EXTI_Callback。
我们先来看一下 HAL_GPIO_EXTI_IRQHandler 函数定义:
void HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin)
{
if(__HAL_GPIO_EXTI_GET_IT(GPIO_Pin) != 0x00U)
{
__HAL_GPIO_EXTI_CLEAR_IT(GPIO_Pin); /* 清中断标志位 */
HAL_GPIO_EXTI_Callback(GPIO_Pin); /* 外部中断回调函数 */
}
}
该函数实现的作用非常简单,通过入口参数 GPIO_Pin 判断中断来自哪个 IO 口,然后清除相应的中断标志位,最后调用回调函数 HAL_GPIO_EXTI_Callback()实现控制逻辑。在所有的外部中断服务函数中直接调用外部中断共用处理函数 HAL_GPIO_EXTI_IRQHandler,然后在回调函数 HAL_GPIO_EXTI_Callback 中通过判断中断是来自哪个 IO 口编写相应的中断服务控制逻辑。
void KEY0_INT_IRQHandler(void)
{
HAL_GPIO_EXTI_IRQHandler(KEY0_INT_GPIO_PIN);
/* 调用中断处理公用函数 清除 KEY0 所在中断线 的中断标志位 */
__HAL_GPIO_EXTI_CLEAR_IT(KEY0_INT_GPIO_PIN);
/* HAL 库默认先清中断再处理回调,退出时再清一次中断,避免按键抖动误触发 */
}
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)
{
delay_ms(20); /* 消抖 */
switch(GPIO_Pin)
{
case KEY0_INT_GPIO_PIN:
if (KEY0 == 0)
{
LED1_TOGGLE(); /* LED1 状态取反 */
LED0_TOGGLE(); /* LED0 状态取反 */
}
break;
default : break;
}
}