记录一下,方便以后翻阅~
主要内容:
1) NVIC中断优先级分组;
2) NVIC中断优先级设置。
官方资料:《STM32中文参考手册V10》第9章——中断和事件
1. NVIC中断优先级分组
1.1 CM3内核支持256个中断,其中包含16个内核中断和240个外部中断,并且具有256级的可编程中断设置;
1.2 STM32并没有使用CM3内核的全部东西,只用了它的一部分;
1.3 STM32有84个中断,包括16个内核中断和68个可屏蔽中断,具有16级可编程的中断优先级;
1.4 STM32F103系列上面,又只有60个可屏蔽中断(在107系列才有68个)。
2. NVIC中断优先级向量表(STM32F10xx产品(非互联型))
3. NVIC中断优先级管理方法
3.1 对STM32中断进行分组,组0~4。同时,对每个中断设置一个抢占优先级和一个响应优先级值;
3.2 分组配置是在寄存器SCB->AIRCR中配置:
3.3 分组配置是在寄存器SCB->AIRCR中配置。
4. 抢占优先级和响应优先级区别
4.1 高优先级(值越小越高)的抢占优先级是可以打断正在进行的低抢占优先级中断的;
4.2 抢占优先级相同的中断,高响应优先级不可以打断低响应优先级的中断;
4.3 抢占优先级相同的中断,当两个中断同时发生的情况下,高响应优先级先执行;
4.4 如果两个中断的抢占优先级和响应优先级都是一样的话,则看哪个中断先发生就先执行。
举例:
假定设置中断优先级组为2,
设置中断3(RTC中断)的抢占优先级为2,响应优先级为1;
设置中断6(外部中断0)的抢占优先级为3,响应优先级为0;
设置中断7(外部中断1)的抢占优先级为2,响应优先级为0。
那么这3个中断的优先级顺序为:中断7>中断3>中断6。
**备注:**一般情况下,系统代码执行过程中,只设置一次中断优先级分组,比如分组2,设置好分组之后一般不会再改变分组。随意改变分组会导致中断管理混乱,程序出现意想不到的执行结果。
5. 中断优先级分组函数:
void NVIC_PriorityGroupConfig(uint32_t NVIC_PriorityGroup)
{
assert_param(IS_NVIC_PRIORITY_GROUP(NVIC_PriorityGroup));
SCB->AIRCR = AIRCR_VECTKEY_MASK | NVIC_PriorityGroup;
}
//例如NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//
6. MDK中NVIC寄存器结构体
typedef struct
{
__IO uint32_t ISER[8]; //中断使能寄存器组//
uint32_t RESERVED0[24];
__IO uint32_t ICER[8]; //中断失能寄存器组//
uint32_t RSERVED1[24];
__IO uint32_t ISPR[8]; //中断挂起寄存器组//
uint32_t RESERVED2[24];
__IO uint32_t ICPR[8]; //中断解挂寄存器组//
uint32_t RESERVED3[24];
__IO uint32_t IABR[8]; //中断激活标志位寄存器组//
uint32_t RESERVED4[56];
__IO uint8_t IP[240]; //中断优先级控制的寄存器组//
uint32_t RESERVED5[644];
__O uint32_t STIR;
}
NVIC_Type;
7. 每个中断的优先级设置
1.1 中断优先级控制的寄存器组:IP[240] (全称:Interrupt Priority Registers)
240个8位寄存器,每个中断使用一个寄存器来确定优先级。STM32F10x系列一共60个可屏蔽中断,使用IP[59]~IP[0]。每个IP寄存器的高4位用来设置抢占和响应优先级(根据分组),低4位没有用到。
相关函数:
void NVIC_Init(NVIC_InitTypeDef* NVIC_InitStruct);
7.2 中断使能寄存器组:ISER[8] (作用:用来使能中断)
32位寄存器,每个位控制一个中断的使能。STM32F10x只有60个可屏蔽中断,所以只使用了其中的ISER[0]和ISER[1]。ISER[0]的bit0bit31分别对应中断031。ISER[1]的bit027对应中断3259。
相关函数:
void NVIC_Init(NVIC_InitTypeDef* NVIC_InitStruct);
7.3 中断失能寄存器组:ICER[8] (作用:用来失能中断)
32位寄存器,每个位控制一个中断的失能。STM32F10x只有60个可屏蔽中断,所以只使用了其中的ICER[0]和ICER[1]。ICER[0]的bit0bit31分别对应中断031。ICER[1]的bit027对应中断3259。
相关函数:
void NVIC_Init(NVIC_InitTypeDef* NVIC_InitStruct);
7.4 中断挂起控制寄存器组:ISPR[8] (作用:用来挂起中断)
7.5 中断解挂控制寄存器组:ICPR[8] (作用:用来解挂中断)
//7.4和7.5涉及的部分相关函数//
static __INLINE void NVIC_SetPendingIRQ(IRQn_Type IRQn);
static __INLINE uint32_t NVIC_GetPendingIRQ(IRQn_Type IRQn);
static __INLINE void NVIC_ClearPendingIRQ(IRQn_Type IRQn);
//以上函数位于core_cm3.h文件中//
7.6 中断激活标志位寄存器组:IABR [8] (作用:只读,通过它可以知道当前在执行的中断是哪一个)
如果对应位为1,说明该中断正在执行。
相关函数:
static __INLINE uint32_t NVIC_GetActive(IRQn_Type IRQn);
8. 中断参数初始化函数解读
void NVIC_Init(NVIC_InitTypeDef* NVIC_InitStruct);
typedef struct
{
uint8_t NVIC_IRQChannel; //设置中断通道//
uint8_t NVIC_IRQChannelPreemptionPriority; //设置抢占优先级//
uint8_t NVIC_IRQChannelSubPriority; //设置相应优先级//
FunctionalState NVIC_IRQChannelCmd; //使能//
}NVIC_InitTypeDef; //位于misc.h文件中//
NVIC_InitTypeDef NVIC_InitStructure;
NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn; //串口1中断,中断类型参考stm32f10x.h中的IRQn_Tpye;//
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=1; //抢占优先级为1//
NVIC_InitStructure.NVIC_IRQChannelSubPriority= 2; //响应优先级位2//
NVIC_InitStructure.NVIC_IRQChannelCmd= ENABLE; //IRQ通道使能//
NVIC_Init(&NVIC_InitStructure); //根据上面指定的参数初始化NVIC寄存器//
9. 中断优先级设置步骤
9.1 系统运行后先设置中断优先级分组;
void NVIC_PriorityGroupConfig(uint32_t NVIC_PriorityGroup); //整个系统执行过程中,只设置一次中断分组;//
9.2 针对每个中断,设置对应的抢占优先级和响应优先级;
void NVIC_Init(NVIC_InitTypeDef* NVIC_InitStruct);
9.3 如果需要挂起/解挂,查看中断当前激活状态,分别调用相关函数即可。
旧知识点:
1)复习基于库函数的初始化函数的一般格式,可参考STM32学习心得三:GPIO实验-基于库函数;
2)复习寄存器地址,可参考STM32学习心得四:GPIO实验-基于寄存器和STM32学习心得五:GPIO实验-基于位操作;
3)复习寄存器地址名称映射,可参考STM32学习心得六:相关C语言学习及寄存器地址名称映射解读。