历史上的今天
返回首页

历史上的今天

今天是:2025年12月26日(星期五)

正在发生

2022年12月26日 | 关于STM32 (Cortex-M3) 中NVIC的分析

2022-12-26 来源:zhihu

  一、STM32 (Cortex-M3) 中的优先级概念


  STM32(Cortex-M3)中有两个优先级的概念:抢占式优先级和响应优先级,也把响应优先级称作“亚优先级”或“副优先级”,每个中断源都需要被指定这两种优先级。


  1. 何为占先式优先级(pre-emption priority)


  高占先式优先级的中断事件会打断当前的主程序/中断程序运行—抢断式优先响应,俗称中断嵌套。


  2. 何为副优先级(subpriority)


  在占先式优先级相同的情况下,高副优先级的中断优先被响应;


  在占先式优先级相同的情况下,如果有低副优先级中断正在执行, 高副优先级的中断要等待已被响应的低副优先级中断执行结束后才能得到响应—非抢断式响应(不能嵌套)。


  3. 判断中断是否会被响应的依据


  首先是占先式优先级,其次是副优先级;


  占先式优先级决定是否会有中断嵌套;


  Reset、NMI、Hard Fault 优先级为负(高于普通中断优先级)且不可调整。


  4. 优先级冲突的处理


  具有高抢占式优先级的中断可以在具有低抢占式优先级的中断处理过程中被响应,即中断的嵌套,或者说高抢占式优先级的中断可以嵌套低抢占式优先级的中断。


  当两个中断源的抢占式优先级相同时,这两个中断将没有嵌套关系,当一个中断到来后,如果正在处理另一个中断,这个后到来的中断就要等到前一个中断处理完之后才能被处理。如果这两个中断同时到达,则中断控制器根据他们的响应优先级高低来决定先处理哪一个;如果他们的抢占式优先级和响应优先级都相等,则根据他们在中断表中的排位顺序决定先处理哪一个。


  5. Cortex-M3中对中断优先级的定义


  既然每个中断源都需要被指定这两种优先级,就需要有相应的寄存器位记录每个中断的优先级;在Cortex-M3中定义了8个比特位用于设置中断源的优先级,这8个比特位可以有8种分配方式,如下:


  所有8位用于指定响应优先级


  最高1位用于指定抢占式优先级,最低7位用于指定响应优先级


  最高2位用于指定抢占式优先级,最低6位用于指定响应优先级


  最高3位用于指定抢占式优先级,最低5位用于指定响应优先级


  最高4位用于指定抢占式优先级,最低4位用于指定响应优先级


  最高5位用于指定抢占式优先级,最低3位用于指定响应优先级


  最高6位用于指定抢占式优先级,最低2位用于指定响应优先级


  最高7位用于指定抢占式优先级,最低1位用于指定响应优先级


  这就是优先级分组的概念。


  6. STM32中对中断优先级的定义


  Cortex-M3允许具有较少中断源时使用较少的寄存器位指定中断源的优先级,因此STM32把指定中断优先级的寄存器位减少到4位,这4个寄存器位的分组方式如下:


  第0组:所有4位用于指定响应优先级


  第1组:最高1位用于指定抢占式优先级,最低3位用于指定响应优先级


  第2组:最高2位用于指定抢占式优先级,最低2位用于指定响应优先级


  第3组:最高3位用于指定抢占式优先级,最低1位用于指定响应优先级


  第4组:所有4位用于指定抢占式优先级

  AIRC(Application Interrupt and Reset Register)寄存器中有用于指定优先级的 4 bits。这4个bits用于分配preemption优先级和sub优先级,在STM32的固件库中定义如下:


  /* Preemption Priority Group */

  #define NVIC_PriorityGroup_0 ((u32)0x700) /* 0 bits for pre-emption  priority

  4 bits for subpriority */

  #define NVIC_PriorityGroup_1 ((u32)0x600) /* 1 bits for pre-emption  priority

  3 bits for subpriority */

  #define NVIC_PriorityGroup_2 ((u32)0x500) /* 2 bits for pre-emption  priority

  2 bits for subpriority */

  #define NVIC_PriorityGroup_3 ((u32)0x400) /* 3 bits for pre-emption  priority

  1 bits for subpriority */

  #define NVIC_PriorityGroup_4 ((u32)0x300) /* 4 bits for pre-emption  priority

  0 bits for subpriority */

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


  NVIC_PriorityGroup_0 => 选择第0组

  NVIC_PriorityGroup_1 => 选择第1组

  NVIC_PriorityGroup_2 => 选择第2组

  NVIC_PriorityGroup_3 => 选择第3组

  NVIC_PriorityGroup_4 => 选择第4组

  接下来就是指定中断源的优先级,下面以一个简单的例子说明如何指定中断源的抢占式优先级和响应优先级:


  // 选择使用优先级分组第1组

  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);

  // 使能EXTI0中断

  NVIC_InitStructure.NVIC_IRQChannel = EXTI0_IRQChannel;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1; // 指定抢占式优先级别1

  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0; // 指定响应优先级别0

  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  NVIC_Init(&NVIC_InitStructure);

  // 使能EXTI9_5中断

  NVIC_InitStructure.NVIC_IRQChannel = EXTI9_5_IRQChannel;

  NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; // 指定抢占式优先级别0

  NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1; // 指定响应优先级别1

  NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;

  NVIC_Init(&NVIC_InitStructure);

  要注意的几点是:


  1. 如果指定的抢占式优先级别或响应优先级别超出了选定的优先级分组所限定的范围,将可能得到意想不到的结果;


  2. 抢占式优先级别相同的中断源之间没有嵌套关系;


  3. 如果某个中断源被指定为某个抢占式优先级别,又没有其它中断源处于同一个抢占式优先级别,则可以为这个中断源指定任意有效的响应优先级别。


  二、开关总中断


  在STM32/Cortex-M3中是通过改变CPU的当前优先级来允许或禁止中断。


  PRIMASK位:只允许NMI和hard fault异常,其他中断/异常都被屏蔽(当前CPU优先级=0)。


  FAULTMASK位:只允许NMI,其他所有中断/异常都被屏蔽(当前CPU优先级=-1)。


  在STM32固件库中(stm32f10x_nvic.c和stm32f10x_nvic.h) 定义了四个函数操作PRIMASK位和FAULTMASK位,改变CPU的当前优先级,从而达到控制所有中断的目的。


  下面两个函数等效于关闭总中断:


  void NVIC_SETPRIMASK(void);

  void NVIC_SETFAULTMASK(void);

  下面两个函数等效于开放总中断:


  void NVIC_RESETPRIMASK(void);

  void NVIC_RESETFAULTMASK(void);

  上面两组函数要成对使用,但不能交叉使用。


  例如:


  第一种方法:


  NVIC_SETPRIMASK(); //关闭总中断

  NVIC_RESETPRIMASK();//开放总中断

  第二种方法:


  NVIC_SETFAULTMASK(); //关闭总中断

  NVIC_RESETFAULTMASK();//开放总中断

  常常使用:


  NVIC_SETPRIMASK(); // Disable Interrupts

  NVIC_RESETPRIMASK(); // Enable Interrupts

  补充:


  可以用:


  #define CLI() __set_PRIMASK(1)

  #define SEI() __set_PRIMASK(0)

  来实现开关总中断的功能。


推荐阅读

史海拾趣

DAICO公司的发展小趣事

在电子行业中,技术创新是企业持续发展的关键。DAICO公司始终将研发创新作为核心战略,不断投入大量资源用于新技术和新产品的研发。通过持续的创新和积累,DAICO在中频/射频和微波控制领域取得了多项技术领先成果,成为行业内的佼佼者。

Advanced Illumination Inc公司的发展小趣事

Advanced Illumination Inc公司成立于XXXX年,初创时期,公司便专注于LED照明光源的研发与生产。面对当时市场上众多的竞争对手,Ai公司凭借其对技术的深刻理解和对市场需求的敏锐洞察,选择了一条专注于机器视觉工业应用的道路。这一决策为公司后续的发展奠定了坚实的基础。

上海晶岳(AFSEMI)公司的发展小趣事

随着市场的不断变化和技术的不断进步,晶岳电子意识到单一的产品线已无法满足市场需求。于是,在2012年,公司开始拓展产品线,研发并推出了LDO、DCDC、复位IC等电源管理芯片。这一举措不仅丰富了公司的产品线,也进一步提升了公司在电源管理领域的竞争力。同时,晶岳电子还加大了对研发团队的投入,引进了一批具有丰富经验的研发人员,为公司的技术创新提供了有力保障。

ECM [ECM Electronics Limited.]公司的发展小趣事

面对日益严重的环境问题,ECM Electronics Limited.积极响应国家节能减排的号召,将绿色发展和可持续发展作为企业的重要战略。公司采用环保材料和节能技术,降低产品能耗和排放。同时,公司还积极参与环保公益活动,推动行业绿色发展。这些举措不仅提升了公司的社会形象,也为公司的长期发展奠定了坚实的基础。

APTA Group Inc公司的发展小趣事

APTA Group Inc在电子行业中崭露头角,得益于其一次重要的技术突破。公司在研发新型半导体材料方面取得了重大进展,这种材料不仅性能稳定,而且成本大大降低。这一创新使得APTA的产品在市场上具有极强的竞争力,吸引了众多合作伙伴。随着技术的逐步推广和应用,APTA的业绩迅速攀升,逐渐在行业中占据了一席之地。

智烽维(CDA)公司的发展小趣事

在超级电容器领域不断取得突破的同时,智烽维也在积极探索新的技术领域。2018年,公司成功实现了锂离子电容器的研发突破。这一成果不仅填补了公司在这一领域的空白,也为公司未来的发展开辟了新的道路。锂离子电容器以其高能量密度、高功率密度和长寿命等特点,在新能源汽车、智能电网等领域具有广阔的应用前景。

问答坊 | AI 解惑

新手一定要看的学习笔记 -ARM开发

以下问题常被问到,我就想到哪说到哪吧。   一 首先说说ARM的发展     可以用一片大好来形容,翻开各个公司的网站,招聘里面嵌入式占据了大半工程师职位。 广义的嵌入式无非几种:传统的什么51、AVR、PIC称做嵌入式微控制器 ...…

查看全部问答>

一次维修TCD的经历(转)

自从转职做仪器维修应用以来,有很多的第一次:第一次参观GC生产车间;第一次安装调试;第一次培训客户操作;第一次更换GC电路板;第一次更换电机;第一次......今天把自己一次更换TCD热导元件的经历发布出来,和版友交流! 1.首先是检测TCD的损 ...…

查看全部问答>

想用S3C2440+GPS模块实现简单的GPS功能

各位老大,我的ARM开发板是FL2440,系统是WINCE 5.0,本人想DIY个车载GPS,学习学习,想问问各位老大,常用的GPS模块有哪些?电子地图是自己绘制,还是利用其他公司提供的地图。有没有免费的可用电子地图?用mapinfo可以吗? 只要能实现简单的功能就 ...…

查看全部问答>

驱动入门级的问题,麻烦高人指点下,谢谢。

NTSTATUS HelloDispatch(IN PDEVICE_OBJECT DeviceObject, IN PIRP pIrp) {         NTSTATUS ntStatus = STATUS_SUCCESS;         PIO_STACK_LOCATION IrpStack = NULL;       &nbs ...…

查看全部问答>

关于访问外部RAM的数据问题

TAB:DB 00H,01H,02H,03H 我怎么更改TAB中第二块的值? 用MOVX吗?…

查看全部问答>

IAR发布了POWERPAC实时嵌入式操作系统文件管理系统

IAR 发布了具有实时嵌入式操作系统(RTOS)并包含一个高性能的文件管理系统的POWERPAC        IAR PowerPac for ARM,V1.10 IAR PowerPac for ARM 包含 ...…

查看全部问答>

100~200价位的GSM+GPRS模块哪个性能比较稳定?广和通的怎样?

RT,现在要把以前用单片机430做的远程医疗产品移植到ARM上,想找个比较稳定的gprs模块,但不知道选择哪块,比如有 SIMcom的900b, 西门子的MC52i, 华为的MG323等等,广和通的听说性能不错,就是比较新,怕不稳定,求达人建议~~…

查看全部问答>

MSP430G2553

今天奋斗了一个晚上终于实现了第一个程序,跟大伙分享一下喜悦! …

查看全部问答>

求整合好的代码

请问一下,mspg2553(测温度的那个代码)的代码怎么整合到g2452(就是那个风火轮的那个代码)代码中?…

查看全部问答>