21.1 NVIC基础知识
NVIC的全称是Nested vectored interrupt controller,即嵌套向量中断控制器。
对于M3/M4/M7内核的MCU,每个中断的优先级都是用寄存器中的8位来设置的。8位的话就可以设置2^8 = 256级中断,实际中用不了这么多,所以芯片厂商根据自己生产的芯片做出了调整。比如ST的STM32F1xx,F4xx和H7只使用了这个8位中的高四位[7:4],低四位取零,这样2^4=16,只能表示16级中断嵌套。
对于这个NVIC,有个重要的知识点就是优先级分组、抢占优先级和子优先级。STM32F1xx,F4xx和H7都是只使用了这个8位寄存器的高四位[7:4]。
从上面的表格可以看出,STM32支持5种优先级分组。系统上电复位后,默认使用的是优先级分组0,也就是没有抢占式优先级,只有子优先级。关于这个抢占优先级和子优先级有几点一定要说清楚。
具有高抢占式优先级的中断可以在具有低抢占式优先级的中断服务程序执行过程中被响应,即中断嵌套,或者说高抢占式优先级的中断可以抢占低抢占式优先级的中断的执行。
在抢占式优先级相同的情况下,有几个子优先级不同的中断同时到来,那么高子优先级的中断优先被响应。
在抢占式优先级相同的情况下,如果有低子优先级中断正在执行,高子优先级的中断要等待已被响应的低子优先级中断执行结束后才能得到响应,即子优先级不支持中断嵌套。
Reset、NMI、Hard Fault 优先级为负数,高于普通中断优先级,且优先级不可配置。
初学者还有一个比较纠结的问题,就是系统中断(比如:PendSV,SVC,SysTick)是不是一定比外部中断(比如SPI,USART)要高。答案:不是的,它们是在同一个NVIC下面设置的。
掌握了这些基础知识基本就够用了。另外特别注意一点,配置抢占优先级和子优先级,它们合并成的4bit数字的数值越小,优先级越高,这一点千万不要搞错了。
21.2 常用的NVIC API
下面将外设配置中经常用到的两个函数做个说明。
HAL_NVIC_SetPriority
HAL_NVIC_EnableIRQ
21.2.1 函数HAL_NVIC_SetPriority
函数原型:
void HAL_NVIC_SetPriority(IRQn_Type IRQn, uint32_t PreemptPriority, uint32_t SubPriority)
函数描述:
此函数主要用于设置中断的抢占优先级和子优先级。
函数参数:
第1个参数IRQn是中断号,由IRQn_Type定义的枚举类型,原始定义在stm32f429x.h。
typedef enum
{
/* 部分定义,其它省略未写 */
PendSV_IRQn = -2,
SysTick_IRQn = -1,
DMA1_Stream3_IRQn = 14,
DMA1_Stream4_IRQn = 15,
DMA1_Stream5_IRQn = 16,
DMA1_Stream6_IRQn = 17,
ADC_IRQn = 18,
} IRQn_Type;
第2个参数PreemptPriority用于设置抢占优先级,范围0 - 15。
第3个参数SubPriority用于设置子优先级,范围0 -15。
使用举例:
此函数的调用比较简单,比如设置ADC中断的抢占优先级是1,子优先级是0,那么此函数的设置就是:HAL_NVIC_SetPriority(ADC_IRQn, 1, 0)。
21.2.2 函数HAL_NVIC_EnableIRQ
函数原型:
void HAL_NVIC_EnableIRQ(IRQn_Type IRQn)
函数描述:
此函数主要用于使能中断。
函数参数:
第1个参数IRQn是中断号,由IRQn_Type定义的枚举类型,原始定义在stm32f429x.h。
typedef enum
{
/* 部分定义,其它省略未写 */
PendSV_IRQn = -2,
SysTick_IRQn = -1,
DMA1_Stream3_IRQn = 14,
DMA1_Stream4_IRQn = 15,
DMA1_Stream5_IRQn = 16,
DMA1_Stream6_IRQn = 17,
ADC_IRQn = 18,
} IRQn_Type;
注意事项:
调用此函数前要先调用优先级分组设置函数NVIC_PriorityGroupConfig,而这个函数会在HAL_Init里面被调用。函数基本都会在main函数里面优先被调用,所以保证HAL_Init优先被调用即可。
使用举例:
此函数的调用比较省事,比如使能ADC中断,那么此函数的设置就是:
HAL_NVIC_EnableIRQ (ADC_IRQn, )。
21.3 MDK中查看实际优先级配置
如果想查看配置的外设优先级是否正确设置了,MDK进入调试状态后,先点击全速运行,然后查看如下调试组件:
弹出如下窗口,在这个窗口里面就可以查看实际优先级了,比如滴答定时器Systick的优先级就是240,即240>>4 = 15。
21.4 开关中断
V6开发板的BSP驱动包在bsp.h文件里面对开关中断做了一个宏定义:
/* 开关全局中断的宏 */
#define ENABLE_INT() __set_PRIMASK(0) /* 使能全局中断 */
#define DISABLE_INT() __set_PRIMASK(1) /* 禁止全局中断 */
__set_PRIMASK就是对寄存器primask做的开关设置,如下表所示:
对于寄存器basepri我们举一个例子,帮助大家理解。比我们配置寄存器basepri(对于STM32F429,范围0-255)的数值为16,所有优先级数值大于等于16的中断都会被关闭,优先级数值小于16的中断不会被关闭。但0比较特殊,对寄存器basepri寄存器赋值0,那么被关闭的中断会被打开。
21.5 总结
最后,我们再强调一下,在NVIC分组为4的情况下,抢占优先级可配置范围是0-15,那么数值越小,抢占优先级的级别越高,即0代表最高优先级,15代表最低优先级。
上一篇:第22章 STM32F429的SysTick实现多组软件定时器
下一篇:第20章 STM32F429的GPIO应用之无源蜂鸣器