STM32/M3/M0关于开关总中断的问题
NVIC 共支持 1 至 240 个外部中断输入(通常外部中断写作 IRQs)。 具体的数值由芯片厂商在设计芯片时决定。此外, NVIC 还支持一个“永垂不朽”的不可屏蔽中断(NMI)输入。NMI 的实际功能亦由芯片制造商决定。在某些情况下, NMI 无法由外部中断源控制。
在 STM32/Cortex-M3 中是通过改变 CPU 的当前优先级来允许或禁止中断。
异常掩蔽寄存器PRIMASK位:只允许 NMI 和 hard fault 异常,其他中断/异常都被屏蔽(当前 CPU 优先级=0,为可编程优先级中的最高优先级) 。
该寄存器可以通过 MRS 和 MSR 以下例方式访问:
1. 关中断MOV R0, #1
MSR PRIMASK, R0
2. 开中断MOV R0, #0
MSR PRIMASK, R0
此外,还可以通过CPS指令快速完成上述功能:
CPSID I;//PRIMASK=1 ; 关中断
CPSIE I;//PRIMASK=0 ; 开中断
CPSID F;//FAULTMASK=1, ; 关异常
CPSIE F;//FAULTMASK=0 ; 开异常
异常掩蔽寄存器FAULTMASK位:只允许 NMI,其他所有中断/异常都被屏蔽(当前 CPU 优先级=-1)。注意的是,FAULTMASK会在异常退出时自动清零。
掩蔽寄存器虽然能一手遮天,却都动不了NMI,因为NMI是用在最危急的情况下的。因此系统为它开出单行道,无需挂号只是不要迟到。
在 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(); //开放总中断
在 3.0 的库中上述库函数已经没有,可以用下列方法实现:
#define CLI() __set_PRIMASK(1)
#define SEI() __set_PRIMASK(0)
或者在编译器里使用:
__disable_irq(); // 关闭总中断
__enable_irq(); // 开启总中断
补充1:异常掩蔽寄存器BASEPRI位
在更精巧的设计中,需要对中断掩蔽进行更细腻的控制——只掩蔽优先级低于某一阈值的中断(它们的优先级在数字上大于等于某个数)。那么这个数存储在哪里?就存储在BASEPRI中。
不过,如果往BASEPRI中写0,则另当别论——BASEPRI将停止掩蔽任何中断。
如果你需要掩蔽所有优先级不高于0x60的中断,则可以如下编程:
MOV R0, #0x60
MSR BASEPRI, R0
如果需要取消 BASEPRI 对中断的掩蔽,则示例代码如下:
MOV R0, #0
MSR BASEPRI, R0
补充2:关闭全局中断时需要注意的问题(未验证是否确有此问题)
STM32在使用时有时需要禁用全局中断。但测试发现一个问题,在关闭总中断后,如果有中断触发,虽然此时不会引发中断,但在调用__enable_irq()开启总中断后,MCU会立即处理之前触发的中断。这说明__disable_irq()只是禁止CPU去响应中断,没有真正的去屏蔽中断的触发,中断发生后,相应的寄存器会将中断标志置位,在__enable_irq()开启中断后,由于相应的中断标志没有清空,因而还会触发中断。
所以要想禁止所有中断,必须对逐个模块的中断进行Disable操作,由于每个模块中断源有很多,对逐个中断Disable的话比较复杂,较为简单的方法是通过XXX_ClearITPendingBit()清除中断标志或者直接通过XXX_DeInit()来清除寄存器的状态。这样在__enable_irq()开启总中断后,MCU就不会响应之前触发的中断了。
原文链接:
http://blog.sina.com.cn/s/blog_a0c22cfd0102x0bj.html
STM32/M3/M0关于开关总中断的问题相关推荐
- STM32中断优先级和开关总中断(很老很经典)
一.中断优先级: STM32(Cortex-M3)中的优先级概念 STM32(Cortex-M3)中有两个优先级的概念--抢占式优先级和响应优先级,有人把响应优先级称作'亚优先级'或'副优先级',每个 ...
- 【stm32f0】stm32 总中断的打开与关闭
问题: 对于基于ARM Cortex M0内核的STM32芯片各类应用开发时,有的时候需要进行总的中断的开.关处理.那就究竟有没有开.关总的中断的函数或者指令呢? 回答: 随着Corte Mn各种内核 ...
- STM32开启和关闭总中断的方法
__set_PRIMASK(1); //关总中断__set_PRIMASK(0); //开总中断
- 【STM32】STM32之限位开关
一.简介 本文介绍如何在STM32上使用限位开关. 二.实验平台 库版本:STM32F10x_StdPeriph_Lib_V3.5.0 编译软件:MDK4.53 硬件平台:STM32开发板(主芯片st ...
- STM32那点事(3)_中断(下)
上节主要讲述了STM32 的中断基础知识以及一般配置过程.外部中断与其他模块中断相对要复杂一点.外部中断一般都是由GPIO触发,是在整个系统中需要经常用到. 介绍 STM32F4 有多达23个中断,每 ...
- STM32那点事(3)_中断(上)
中断是STM32核心模块之一,几乎所有的模块都需要和中断打交道.了解STM32运作机制非常重要.在<STM32那点事(1)_STM32F40_41xx启动文件详解>中讲解启动流程中提到,中 ...
- STM32控制步进电机:基于定时器中断的ULN2003驱动器/步进电机驱动程序
STM32控制步进电机:基于定时器中断的ULN2003驱动器/步进电机驱动程序 一.ULN2003驱动器 1.工作原理 2.步距角以及一圈所需步数的计算 二.硬件连接 三.STM32F103定时器中断 ...
- 【STM32】基于HAL库的中断详细学习
目录 1.中断概述 1.1中断相关概念 1.2 STM32中断系统 2 .HAL库的中断处理 2.1 HAL 库的中断封装 2.2 外部中断处理流程 3.外部中断的HAL库定义 3.1. 外部中断的数 ...
- STM32系统学习——EXTI(外部中断)
一. EXTI 简介 EXTI(External interrupt/event controller)-外部中断/事件控制器,管理了控制器的 20个中断/事件线.每个中断/事件线都对应有一个边沿检测 ...
最新文章
- mongorepository查询条件_MongoDB动态条件之分页查询
- Java 中的字符串(String)与C# 中字符串(string)的异同
- TCP和UDP的优缺点及区别
- pdfbox 第一页加内容_你用代码做过哪些很酷/有趣的事?
- redis 什么是冷数据_阿里Java三面凉凉:微服务,Redis,JVM一个都搞不懂
- USB共享网络:android手机通过USB与Ubuntu进行socket网络通信
- Mybatis-plus常用API全套教程
- 【leetcode 简单】第一题 两数之和
- 浅谈javascript函数劫持
- springboot中,页面访问不到静态资源
- Sqoop导出模式——全量、增量insert、更新update的介绍以及脚本示例
- 倒立摆源码 旋转倒立摆 完整全功能 程序 倒立摆 pid算法 程序使用时可根据硬件需要自行调节
- 优化算法|布谷鸟算法原理及实现
- 计算机用老毛桃u盘备份系统,如何一键备份还原系统?老毛桃一键备份系统详细步骤...
- 云锵基金 2019 年 03 月简报
- L-TAGE预测器 “A 256 Kbits L-TAGE branch predictor”
- 如何使用语音验证码API
- 设计师经常面对的11种职场问题
- 小米12s ultra,索尼xperia1 iv,数码相机 拍照对比
- cp——复制文件\文件夹