STM32 外部中断的原理与应用分析

2022-05-27     大方老师单片机

原标题:STM32 外部中断的原理与应用分析

STM32外部中断的原理与应用分析

1、抢占优先级和响应优先级

在了解中断之前,先了解一下它们之间的优先级别STM32的中断源具有两种优先级:一种为抢占优先级;另一种为响应优先级(亚优先级,其属性编号越小,表明它的优先级别越高。抢占是指打断其他中断的属性,即低抢占优先级的中A可以被高抢占优先级的中B打断,执行完中断服务函B后,再返回继续执行中断服务函A,由此会出现中断嵌套。响应属性则应用在抢占属性相同的情况下,即当两个中断源的抢占优先级相同时,分以下几种情况处理:

(1)如果两个中断同时到达,则中断控制器会先处理响应优先级高的中断。

(2)当一个中断到来后,如果正在处理另一个中断,则这个后到的中断就要等到前一个中断处理完之后才能被处理(高响应优先级的中断不可以打断低响应优先级的中断)。

(3)如果它们的抢占式优先级和响应优先级都相等,则根据它们在中断表中的排位顺序决定先处理哪一个。

举个例子,现在3个中断向量,如下图,若内核正在执C的中断服务函数,则它能被抢占优先级更高的中A打断,由BC的抢占优先级相同,所C不能B打断。但如BC中断是同时到达的,内核就会首先执行响应优先级别更高B中断。如BC中有一个先到,不管谁的响应优先级高,都不能打断,只能等待。

2NVIC的优先级组

STM32使用4个中断优先级的寄存器位,只可以配16种优先级,即抢占优先级和响应优先级的数量由一4位的数字来决定,把这4位数字的位数分配成抢占优先级部分和响应优先级部分。有以5种分配方式:

0种:所4位用于指定响应优先级,NVIC配置24次方16种中断向量都是只有响应属性,没有抢占属性。

1种:最1位用来配置抢占优先级,3位用来配置响应优先级,表示2种级别的抢占优先级01,23次方8种响应优先级,即16种中断向量之中,8种中断的抢占优先级都0级,而它们的响应优先级分别0~ 7,8种中断的抢占优先级则都1级,响应优先级别分别0~7

22位用来配置抢占优先级2位用来配置响应优先,22次方4种抢占优先级22次方4种响应优先级。

3种:3位用来配置抢占优先级,最1位用来配置响应优先级,即8种抢占优先级2种响应优先级。

4种:所4位用来指定抢占优先级,16种中断具有不相同的抢占优先级。

可以通过调STM32固件库中的函NVIC_PriorityGroupConfig()选择使用哪种优先级分组,这个函数5个参数:

#define NVIC_PriorityGroup_0 ((uint32_t)0x700) //选择0 #define NVIC_PriorityGroup_1 ((uint32_t)0x600) //选择1 #define NVIC_PriorityGroup_2 ((uint32_t)0x500) //选择2#define NVIC_PriorityGroup_3 ((uint32_t)0x400) //选择3 #define NVIC_PriorityGroup_4 ((uint32_t)0x300) //选择4

这里我说一下为什么要设置分组和优先级?首先中断需要有优先级,事件有重要和次要之分,CPU接到更重要的中断请求时,应当先去处理重要的事情。设计优先级分组是一个更灵活的选择STM32使用4个中断优先级的寄存器位,这就意味着可以设16种优先级,比如说我做的项目会突发很多种情况,这时就可以设置为3种,8种抢占(先后)级别2种响应级别;如果中断的情况不多,可以设置为2种,那就会4种抢占(先后)4种响应,这样的设计就可以灵活对付很多复杂的情况。注意:

一般情况下,系统代码执行过程中,只设置一次中断优先级分组,比如分3,设置好分组之后一般不会再改变分组。随意改变分组会导致中断管理混乱,程序出现意想不到的执行结果。

3、中断线的概念

STM32的每IO都可以作为外部中断输入。

STM32的中断控制器支19个外部中/事件请求:

线0~15:对应外IO口的输入中断。

线16:连接PVD输出。

线17:连接RTC闹钟事件。

线18:连接USB唤醒事件。

每个外部中断线可以独立的配置触发方式(上升沿,下降沿或者双边沿触发),触/屏蔽,专用的状态位STM32IO使用的中断线只16个,但STM32F10x系列IO口多达上百个,那么中断线怎么io口对应呢?其实啊,一IO口对应一根中断线,每IO口都可以使用这跟中断线,但是在同一时刻,只能响应一个端口的事件触发,不能同时响应所GPIO端口的事件,也就是分时复用。

4、中断服务函数

STM32分配16根中断线,但是中断服务函数只分配7个,毋庸置疑,肯定有几个也是要大家共享的了。

·中断线0 ~ 4各对应一个中断函数(下图少了一个外部中0

·中断线5 ~ 9共用中断函EXTI9_5_IRQHandler

·中断线10 ~ 15共用中断函EXTI15_10_IRQHandler

中断效劳函数列表:

EXTI0_IRQHandler EXTI1_IRQHandlerEXTI2_IRQHandler EXTI3_IRQHandler EXTI4_IRQHandler EXTI9_5_IRQHandler EXTI15_10_IRQHandler

在编写中断效劳函数的时候会经常使用到两个函数实现下面的功能:

(1)判断某个中断线上的中断是否发生(标志位是否置位,一般使用在中断效劳函数的开头,判断中断是否发生:

ITStatus EXTI_GetITStatus(uint32_t EXTI_Line);

(2)革除某个中断线上的中断标志位,这个函数一般应用在中断效劳函数完毕之前,革除中断标志位:

void EXTI_ClearITPendingBit(uint32_t EXTI_Line)

中断效劳函数的格式能够说是一个套路的,常用的中断效劳函数格式为:

void EXTI9_5_IRQHandler(void){

if(EXTI_GetITStatus(EXTI_Line8)!=RESET)//判断线8上的中断是否发

{

?中断逻

EXTI_ClearITPendingBit(EXTI_Line8);//LINE上的中断标志

}

}

另外,固件库还提供了两个函数用来判断外部中断状态和革除外部状态标志位,EXTI_GetFlagStatus()EXTI_ClearFlag(),其作用和前面两个函数的作用类似,只是EXTI_GetITStatus()中会先判断这种中断是否使能,使能了才去判断中断标志位,EXTI_GetFlagStatus()直接用来判断状态标志位。

5、外部中断的一般配置步骤:

初始IO口为输写。

GPIO_Init();

IO口复用时钟。

RCC_APB2PeriphClockCmd(RCC_APB2Periph_AFIO,ENABLE);

【文章福利】:小编整理了一些个人觉得比较好的学习书籍、视频资料共享在群文件里面,有需要的可以自行添加哦!~点击绿色通讯软件搜airuimcu加入。

IO口与中断线的映射关系。

void GPIO_EXTILineConfig();例如GPIO_EXTILineConfig(GPIO_PortSourceGPIOA,GPIO_PinSource0);

初始化线上中断,设置触发条件等。

EXTI_Init();例如 EXTI_InitStructure.EXTI_Line=EXTI_Line2;//指定中断线EXTI_InitStructure.EXTI_Mode = EXTI_Mode_Interrupt; //模式:事件或者中断EXTI_InitStructure.EXTI_Trigger = EXTI_Trigger_Falling;//触发方式下降沿EXTI_InitStructure.EXTI_LineCmd = ENABLE; //使能 EXTI_Init(&EXTI_InitStructure);

配置中断分组NVIC),并使能中断。

NVIC_Init();例如 NVIC_InitTypeDef NVIC_InitStructure; NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;//1中断NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1 ;//抢占优先级1NVIC_InitStructure.NVIC_IRQChannelSubPriority = 2;//子优先级2NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;//IRQ通道使能NVIC_Init(&NVIC_InitStructure); //根据上面指定的参数初始NVIC寄存器

编写中断服务函数。

EXTIx_IRQHandler();

清除中断标志位

EXTI_ClearITPendingBit();

文章来源: https://twgreatdaily.com/zh-hans/8939d776b380327d9fe8fdb684795c12.html