STM32的NVIC和中断的总结

本文详细介绍了STM32中断的概念及实现方法,包括STM32对中断优先级的定义、中断向量表、配置相关寄存器等内容。适用于有一定51单片机开发经验的学习者。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言:
1.要想学习STM32中断,要先掌握STM32对优先级的定义;
2.有51单片机开发经验会比较容易理解中断优先级;
3.本篇博文基于STM32F103ZET6芯片和3.5.0标准库编写;
4.本篇博文从寄存器入手,最终实现编程的步骤;如有不足之处,还请前辈多多指教;

一 基础知识
1. cortex-m3支持256个中断,其中包含了16个内核中断,240个外部中断。(本博文只介绍60个外部可屏蔽中断)
2. stm32只有84个中断,包括16个内核中断和68个可屏蔽中断
3. stm32f103上只有60个可屏蔽中断,f107上才有68个中断
4. 先占优先级也就是抢占优先级,概念等同于51单片机中的中断。假设有两中断先后触发,已经在执行的中断先占优先级如果没有后触发的中断 先占优先级更高,就会先处理先占优先级高的中断。也就是说又有较高的先占优先级的中断可以打断先占优先级较低的中断。这是实现中断嵌套的基础。
次占优先级,也就是响应优先级,只在同一先占优先级的中断同时触发时起作用,先占优先级相同,则优先执行次占优先级较高的中断。次占优先级不会造成中断嵌套。 如果中断的两个优先级都一致,则优先执行位于中断向量表中位置较高的中断。

  1. NVIC是什么?
    嵌套向量中断控制器;用于为中断分组,从而分配抢占优先级和响应优先级;
    分组的方式有两种:
    (1)Cortex-m3内核提供了一种3位宽度的PRIGROUP数据区,用于指示一个8位数据序列中的小数点的位置,从而表示中断优先级的分组。见下表:
    这里写图片描述

(2)而实际上STM32并没有用到这么多中断,所以在分组上只分了5个组,并且表示方法有所不同;见下表:
这里写图片描述

我们在应用当中只会用到STM32的分组(5组)方式,所以下面着重于5组分组方式;

二 中断向量表 (STM32F10x系列)
这里写图片描述
这里写图片描述
这里写图片描述
这里写图片描述
(图片来自STM32使用手册,只需看,不需要熟记,知道大概这么多中断就好)

三 配置中断相关寄存器

/*
cortex-m3内核分组方式(8组)结构体表达方式:
*/
typedef struct
{
  __IO uint32_t ISER[8];             中断使能设置寄存器          /*!< 偏移量: 0x000  Interrupt Set Enable Register           */
       uint32_t RESERVED0[24];                                   
  __IO uint32_t ICER[8];              中断清除使能寄存器         /*!<偏移量: 0x080  Interrupt Clear Enable Register        */
       uint32_t RSERVED1[24];                                    
  __IO uint32_t ISPR[8];              中断挂起设置寄存器        /*!< 偏移量: 0x100  Interrupt Set Pending Register          */
       uint32_t RESERVED2[24];                                   
  __IO uint32_t ICPR[8];              中断清除挂起寄存器        /*!<偏移量: 0x180  Interrupt Clear Pending Register        */
       uint32_t RESERVED3[24];                                   
  __IO uint32_t IABR[8];               中断激活状态位寄存器       /*!< 偏移量: 0x200  Interrupt Active bit Register           */
       uint32_t RESERVED4[56];                                   
  __IO uint8_t  IP[240];               中断优先级寄存器       /*!< 偏移量: 0x300  Interrupt Priority Register (8Bit wide) */
       uint32_t RESERVED5[644];        软件触发方式寄存器                          
  __O  uint32_t STIR;                         /*!< 偏移量: 0xE00  Software Trigger Interrupt Register     */
}  NVIC_Type;  
/*
STM32分组(5组)方式结构体表达方式
typedef struct
{
  vu32 ISER[2];
  u32  RESERVED0[30];
  vu32 ICER[2];
  u32  RSERVED1[30];
  vu32 ISPR[2];
  u32  RESERVED2[30];
  vu32 ICPR[2];
  u32  RESERVED3[30];
  vu32 IABR[2];
  u32  RESERVED4[62];
  vu32 IPR[15];
} NVIC_TypeDef;
*/

以上寄存器介绍:
这里写图片描述
对一些概念的解释:
1. 挂起:当置位中断挂起寄存器的时候,相应的中断将会被挂起,这是这个中断将不会立即执行,而是等待可执行的时候再执行;比如高低级别的中断同时产生,就先挂起低级别的中断,等高级别的中断执行完毕,解除并执行低级中断;
2. 对中断优先级控制寄存器的解释
这里写图片描述

推荐去观看一位前辈的博客:
http://blog.youkuaiyun.com/DLUTXIE/article/details/7059184?locationNum=4&fps=1

四 编程步骤

(1)选择优先级分组
/*
1. 此函数在库文件misc.h文件下;
2. 参数可参照下面图片:
这里写图片描述
3. 功能:选择分组方式;
*/
void NVIC_PriorityGroupConfig(uint32_t NVIC_PriorityGroup);

栗子:NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
//抢占优先级可选0~3,响应优先级可选0~3;

(2)选择,配置,并使能中断
NVIC_InitStructure.NVIC_IRQChannel = EXTI2_IRQn; //选择EXTI2中断
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0x02; //抢占优先级为2
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0x02; //响应优先级为2
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //使能EXTI2中断;
NVIC_Init(&NVIC_InitStructure); //初始化以上参数;

(3)写出相应中断函数
栗子:
void EXTI2_IRQHandler(void)
{

//逻辑代码;

EXTI_ClearITPendingBit(EXTI_Line2);
}

<think>我们首先需要理解用户的问题:如何在NVIC中配置EXTI15_10_IRQn的优先级分组,特别是使用NVIC_PriorityGroupConfig函数。根据引用[1],NVIC_PriorityGroupConfig函数用于设置优先级分组,可选的分组有0到4组。引用[2][3]解释了不同分组的意义:分组决定了抢占优先级子优先级的位数分配。用户特别提到了EXTI15_10_IRQn,这是外部中断10到15的中断通道。配置步骤(使用标准外设库):1.首先,设置整个系统的优先级分组(使用NVIC_PriorityGroupConfig)。注意,整个系统只能设置一次,通常在初始化时设置。2.然后,针对具体的中断通道(如EXTI15_10_IRQn)配置其抢占优先级子优先级。根据用户之前的上下文,用户希望使用NVIC_PriorityGroup_2(即第2组)。在分组2中,4位优先级被分为2位抢占优先级(0-3)2位子优先级(0-3)。因此,配置EXTI15_10_IRQn的步骤为:a.设置优先级分组为Group2。b.初始化EXTI15_10_IRQn的NVIC结构体,指定抢占优先级子优先级的值(范围均为0-3)。下面给出具体代码示例(基于标准外设库):</think>### 配置 EXTI15_10_IRQn 优先级分组详解 #### 配置步骤(标准外设库) ```c // 1. 设置全局优先级分组为Group2(仅需执行一次) NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2); // 2. 配置EXTI15_10中断通道 NVIC_InitTypeDef NVIC_InitStructure; NVIC_InitStructure.NVIC_IRQChannel = EXTI15_10_IRQn; // 选择外部中断10-15通道 NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 2; // 抢占优先级(0-3) NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; // 子优先级(0-3) NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStructure); ``` #### 关键配置说明 1. **分组选择**: - `NVIC_PriorityGroup_2` 表示: - **2位抢占优先级**(0-3,4个级别) - **2位子优先级**(0-3,4个级别) - 此分组提供中等粒度的中断嵌套控制[^1] 2. **优先级数值计算**: - 实际优先级值 = 抢占优先级 × 4 + 子优先级 - 示例配置值:$2 \times 4 + 1 = 9$ - 数值越小优先级越高 3. **中断行为规则**: - 不同抢占优先级:高抢占优先级可打断低抢占优先级中断[^2] - 相同抢占优先级:子优先级高的先执行(数值小的优先级高) - 相同抢占子优先级:按硬件中断号顺序执行 #### CubeMX配置注意事项 1. **分组设置位置**: - `System Core` → `NVIC` → `Priority Group` 选择 `Group 2` ```mermaid graph LR A[CubeMX] --> B[System Core] B --> C[NVIC] C --> D[Priority Group] D --> E[Group 2] ``` 2. **EXTI15_10配置**: - 在 `NVIC Configuration` 选项卡找到 `EXTI line[15:10] interrupts` - 设置: - `Preemption Priority`:2 - `Sub Priority`:1 - 勾选 `Enabled` 复选框 3. **设计建议**: - 将实时性要求高的中断(如电机控制)设为抢占优先级0-1 - EXTI15_10这类GPIO中断通常设为抢占优先级2-3 - 避免在同一抢占优先级内设置过多中断[^3] > **重要原则**:优先级分组设置必须在所有中断初始化前完成,且整个系统只需设置一次。不同分组的选择会影响抢占优先级子优先级的可用级别数量[^1]。 --- ###
评论 21
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值