STM32学习9---EXIT外部中断(理论)

server/2025/1/23 13:43:44/

本文参考江科大和其他博主,侵删!

中断系统是管理和执行中断的逻辑结构 ,外部中断是产生中断的外设之一

一、STM32中断

1、中断基本介绍

68个可屏蔽中断通道(中断源),包含EXTI外部、TIM定时器、ADC模数转换器、USART串口、SPI通信、I2C通信、RTC实时时钟等外设。STM32的型号不同,支持的中断不同。

NVIC是统一管理中断,每个中断通道有16个可编程的优先等级,并可对优先级进行分组,进而设置抢占优先级、响应优先级。NVIC就是管理中断分配优先级的。

STM32的中断资源:

比如第一个复位中断,当产生复位事件时,程序就会自动执行复位中断函数,也就是复位后程序开始执行的位置。NMI不可屏蔽中断,硬件失效、存储管理、总线错误、错误应用等等,这些都是内核里面的中断,都比较高深,也难理解,一般用不到,了解一下即可。

STM32的外设中断:

第一个窗口看门狗,用来监测程序运行状态的中断,比如程序卡死没有及时喂狗,窗口看门狗就会申请中断,让你的程序跳到窗口看门狗的中断程序里,这时可以在中断程序里进行一些错误检查,检查问题。

PVD电源电压监测。如果供电电压不足,PVD电路就会申请中断。从而知道现在供电不足,电池是否有电,保存重要数据。类似功能还有:

外设电路检测到异常或者事件,需要提示一下CPU的时候,可以申请中断,让程序跳到对应的中断函数里运行一次,用来处理这个异常或事件。

EXTI0到EXTI4,还有下面的EXTI9_5和EXTI15-10,是本节要学习的内容。

表的右边是中断的地址:

程序中的中断函数的地址是由编译器来分配的,是不固定的,但是中断跳转由于硬件的限制,只能跳到固定的地址执行程序。所以为了能让硬件跳转到一个不固定的中断函数里,这里就需要在内存中定义一个地址的列表。这个列表地址是固定的,中断发生后就跳到这个固定位置。然后在这个固定位置,由编译器,再加上一条跳转到中断函数的代码,这样中断跳转就可以跳到任意位置了

这个中断地址的列表就叫中断向量表,是中断跳转的一个跳板。使用C语言编程不要考虑中断向量表,由编译器完成。

2、NVIC的基本结构

NVIC:嵌套中断向量控制器,在STM32中,统一分配中断优先级和管理中断的。

NVIC是一个内核外设CPU的助手。STM32的中断非常多,如果把这些中断全都接到CPU上,CPU还得引出很多线进行适配,如果很多中断同时申请或者中断很多产生了拥堵,CPU也会很难处理,毕竟CPU主要是用来运算的,所以设置NVIC负责中断分配的任务。

NVIC有很多输入口,中断线路都可以接过来。比如:EXTI、TIM、ADC、USART等

 线上画了个斜杠,上面写个n,这个意思是一个外设可能会同时占用多个中断通道,所以这里有n条线。

3、优先级

NVIC只有一个输出口,NVIC根据中断的优先级分配中断的先后顺序,通过输出口到CPU。怎么分配优先级,CPU不参与。(相当于医院的叫号系统,医生不参与叫号排序)

为了处理不同形式的优先级,STM32的NVIC可以对优先级进行分组。每个中断有16个优先级,这个优先级再可以再划分为抢占优先级和响应优先级,就需要16个优先级进行分组。

NVIC的中断优先级由优先级寄存器的4位决定。(4位二进制可以表示0~15的数对应16个优先级,这个优先级的数是值越小优先级越高0就是最高优先级)。这4位再切分,分位n位的抢占优先级和4-n的响应优先级。

抢占优先级高的可以中断嵌套,响应优先级高的可以优先排队。

当2个中断同时响应,抢占优先级高的先中断。

抢占优先级相同时,响应优先级高的先中断。

抢占优先级相同时,低响应优先级已经进入中断函数,高响应优先级不可打断低响应优先级

抢占优先级和响应优先级均相同,按中断号排队。当抢占优先级和响应优先级均相同时,就按照这里的这个数字来排队,数值小的优先响应。这个中断号就是这里这个表里的这栏数字。

总之,STM32的中断不存在先来后到的排队方式,在任何时候都是优先级高的先响应。

3、优先级分组

因为优先级总共是四位,所以就有五种分组方式。

分组0:0位的抢占优先级,取值只能为0。4位的响应等级取值可以是0~15。

分组1:是1位的抢占优先级,取值是0~1;3位的响应等级取值是0~7。

然后分组2、3、4都是按照这个规律来分配的。

这个分组方式在程序中可以选择的,选好分组方式之后,在配置优先级的时候,就要注意抢占优先级和响应优先级的取值范围了,不要超出这个表里规定的取值范围。

二、EXTI外部中断

1、外部中断基本介绍 

EXTI可以监测指定GPIO口的电平信号,指定GPIO口的电平发生变化时,EXTI立即向NVIC申请中断,经过NVIC裁决后,可中断CPU主程序,使CPU执行EXTI对应的中断程序。

支持的触发方式:

触发方式:上升沿、下降沿、双边沿、软件触发(使用代码触发,与引脚无关)。

支持的GPIO口:

所有GPIO口,但相同的Pin不能同时触发中断,比如PA0和PB0不能同时触发,PA1、PB1、PC1等都不能同时触发中断,只能选择一个。有多个中断引脚,要选择不同Pin脚,如:PA7、PA9。

外部中断占用的通道:
通道数:16个GPIO_Pin(GPIO_Pin_0到GPIO_Pin_15),再加PVD输出、RTC时钟、USB唤醒、以太网唤醒,共计20条中断线路。16个GPIO_Pin是外部中断的主要功能。

外部中断的触发响应方式

分为中断响应/事件响应。

中断响应:申请中断,让CPU执行中断函数。

事件响应:外部中断的额外功能。当外部中断检测到引脚电平变化时,一般时申请中断,但还可以选择触发事件,外部中断的信号不会通向CPU,而是通向其他外设,触发外设的操作,比如ADC转换、触发DMA。总之,中断响应是正常的操作,引脚电平触发中断。事件响应不会触发中断,而是触发别的外设,属于外设之间的联合工作。

2、基本结构

最左边是GPIO口的外设,每个GPIO外设有16个引脚,所以每个外设进来16根线

AFIO:中断引脚选择

EXTI模块只有16个GPIO的通道。但这里每个GPIO外设都有16个引脚,如果每个引脚占用一个通道,EXTI的16个通道显然就不够用了。所以在这里会有一个AFIO中断引脚选择的电路模块

AFIO:是一个数据选择器。在GPIO外设各自的16个引脚里,选择其中一个连接到后面的EXTI的通道里,所以这前面说相同的Pin不能同时触发中断。因为对于PA0、PB0、PC0,通过AFIO选择之后,只有其中一个能接到EXTI的通道0上。同理,PA1、PB1、PC1,也只能有一个接到通道1上。这就是所有GPIO口都能触发中断,但相同的pin不能同时触发中断的原因。

EXTI边沿检测及控制:

AFIO选择之后的16个通道,接到EXTI边沿检测及控制电路上。还有下面PVD等4个蹭网的外设,共计组成EXTI的20个中断通道。

NVIC中断选择分配优先级

经过EXTI电路后,分为2种输出,接到NVIC的,是触发中断的。

注意:本来20个输入,应该20个输出,但是为了节省NVIC的通道资源,外部中断的9~5,15~10给分配了同一个通道。即:外部中断的9~5会触发同一个中断函数15~10也会触发同一个中断函数。编程时在这两个中断函数里,根据标志位来区分到底是哪个中断。

右下角有20条输出线路到了其他外设,这就是用来触发其他外设的,也就是上文提到的事件响应。

AFIO和NVIC的内部电路:

先看一下AFIO复用lO口:

上图是AFIO选择中断引脚的结构图,这里就是一系列的数据选择器。

 第一个输入是PA0、PB0、PC0等,一直到PG0,尾号都是0,然后通过数据选择器,最终选择一个接到EXTI0上。配置AFIO_EXTICR1寄存器的EXIT0可以决定选择哪一个输入。

 AFIO注意用于引脚复用功能的选择和重定义,也就是数据选择器的作用。

在STM32中,AFIO主要2个作用:复用功能引脚重映射、中断引脚选择。

复用功能引脚重映射:把这些默认复用功能的引脚换到重定义的这个位置来,使用AFIO来完成的,这也是AFIO的一大主要功能。

EXTI的内部框图

 

从框图看到,EXTI的右边就是20根输入线,输入线首先进入边沿检测电路,在上面的上升沿寄存器和下降沿寄存器,可以选择是上升沿触发还是下降沿触发,或者两个都触发。接着触发信号就进入到这个或门的输入端了。

硬件触发和软件中断寄存器的值,接在这个或门上,也就是任意一个为1,或门就可以输出1,所以才支持多种触发方式,如上升沿、下降沿、双边沿和软件触发。

触发信号通过这个或门之后就分两路,上一路是触发中断,下一路是触发事件。

触发中断:首先会置一个请求挂起寄存器,即中断标志位,可以读取该标志位,判断哪个通道触发的中断。该寄存器置1,沿着线路继续往左走,和中断屏蔽寄存器共同进入一个与门,最后至NVIC中断控制器。

这里的与门起了开关的作用,对于与门来说,1&x=x,0&x=0。中断屏蔽寄存器给1,那另一个输入就是直接输出,也就是允许中断。中断屏蔽寄存器给0,另一个输入无论是什么,输出都是0,相当于屏蔽了这个中断。这就是这个与门的作用,相当于一个开关控制。

触发事件:

首先事件屏蔽寄存器进行开关控制,通过一个脉冲发生器到其他外设。这个脉冲发生器就是给一个电平脉冲,用来触发其他外设的动作。

另外,这个框图上画一个斜杠写着20,表示20根线,代表20个通道。最上面是外设接口和APB总线,可以通过总线访问这些寄存器。

以上就是EXTI的内部结构。

三、外部中断使用的硬件模块

1、编码器介绍 

什么样的设备需要用到外部中断?对于STM32来说,想要获取的信号是外部驱动的很快的突发信号。比如旋转编码器的输出信号。这个信号是突发的,STM32不知道什么时候会来,同时它是外部驱动的,STM32只能被动读取,而且这个信号非常快,STM32稍微晚读取,就会错过很多波形。对于这种情况,就考虑使用STM32的外部中断。一有脉冲过来,STM32立即进入中断函数处理。没有脉冲的时候,STM32处理其他。

再比如红外遥控接收头的输出,接收到遥控数据之后,它会输出一段波形。这个波形转瞬即逝,所以就需要用外部中断来读取。

但按键的动作可以看成外部驱动的突发事件,但并不推荐使用外部中断来读取按键,因为外部中断不好处理按键抖动和松手检测的问题,对于按键来说,它的输出波形也不是转瞬即逝的。所以要求不高的话,可以在主程序中循环读取。如果不想用主循环读取的话,采用定时器中断读取的方式。这样既能后台读取按键值不阻塞主程序,也能处理按键抖动和松手检测的问题。

旋转编码器

它是测量速度、位置、和旋转方向的装置。当旋转轴旋转时,输出端可以输出与旋转方向和速度对应的方波信号,读取方波信号的频率和相位就可以测出旋转轴的旋转方向和速度。

旋转编码器的类型:机械触点式、霍尔传感器式、光栅式等

这是最简单的编码器,使用对射式红外传感器测速,需要配合光栅编码盘使用。

当编码盘转动时,红外传感器的红外光就会出现遮挡、透过、遮挡、透过这样的现象。对应模块输出的电平就是高低电平交替的方波。方波的个数代表旋转的角度,频率代表速度。所以可以使用外部中断捕获方波的边沿,以此判断位置和速度。该种只有一路输出,正反转波形无法区分,无法判断旋转方向。

为了进一步测量方向,我们就可以用这几种编码器:

第一个是套件里的旋转编码器了,左边是它的外观,右边是它的内部拆解结构。

它内部是用金属触点来进行通断的,所以它是一种机械触点式编码器,这里左右是两部分开关触点 。这种编码器可以进行调节的,比如音响调节音量,因为它是触点接触的形式,所以不适合电机这种高速旋转的应用,另外几种都是非接触的形式,可以用于电机测速,电机测速的功能比较常见。

 其中内侧的两根细的触点都是和中间的这个引脚连接的

外侧的触点,左边的接在这个引脚,右边的接在这个引脚,这就是这些触点的连接方式。

中间这个圆的金属片是一个按键,旋转编码器的轴是可以按下去的

按键的两根线就在上面引出来, 按键的轴按下,则上面两根线短路。松手则上面两根线断开。

再看一下编码盘,是一系列像光栅一样的东西,这是金属触点,在旋转时依次接通和断开两边的触点。

两侧触点的通断产生一个九十度的相位差,配合外部电路,这个编码器的两个输出就会输出这样的波形:

当正转时,A侧引脚输出一个方波信号,同时B相引脚输出一个和它相位相差90度的波形,也就是正向旋转时,B相输出是滞后90度,

反向旋转时A相还是方波信号,B相就会提前90度。这样正反转就可以区分。

这种相位相差90°的波形叫正交波形。正交波形输出的编码器是可以测量方向的。这也是与单相输出波形的区别。

接着看,直接附在电机后面的编码器,称为霍尔传感器形式的编码器。中间是圆形磁铁,边上有两个位置错开的霍尔传感器。当磁铁旋转时,通过霍尔传感器就可以输出正交的方波信号。

接着是独立的编码器元件:也可以测速和测方向。

2、旋转编码器硬件电路

电路图:

图里中间这个就是旋转编码器

上面按键的两根线(方框),这个模块没有使用,是悬空的。红圈的是编码器内部的两个触点。

旋转轴旋转时,这两个触点以相位相差九十度的方式交替导通。该开关信号要配合外围电路才能输出高低电平。

电路图左边,接了一个10k的上拉电阻,默认编码器没旋转时,这个点被拉为高电平,通过R3输出到A端口的也是高电平。

当旋转时,触点导通,这个点就直接被拉低到GND了(低电平)。

R3是限流电阻,防止引脚电流过大。C1是滤波电容,防止信号抖动。右边电路与左边一样。

中间的C直接接GND。

该模块的外部图:

A相输出和B相输出接到STM32的两个引脚比如PB0和PB1。注意引脚的GPIO_Pin编号不要一样。

3、手册建议

NVIC属于内核外设,参考Cortex_M3编程手册。找到NVIC的一些寄存器,包括中断使能寄存器、中断清除使能寄存器、中断设置挂起寄存器、中断清除挂起寄存器、中断活动位寄存器、中断优先级寄存器,这个中断优先级寄存器是设置每个中段的优先级的、接着还有一个软件触发中断寄存器。

另外,中断分组的配置,中断分组的配置寄存器是被分配到了这个SCB里面的。在SCB_AIRCR里,这三位就是用来配置中断分组的。

打开STM32的手册,有中断系统和外部中断的介绍以及NVIC的概述,然后是中断相量表,接着就是外部中断EXTI的介绍。还有一些框图,和外部中断唤醒事件的说明、外部中断的功能说明。

还有中断的线路映象,也就是中断引脚选择,这一部分实际上是在AFIO里的。

最后就是外部中断的寄存器描述,这里有中断屏蔽寄存器、事件屏蔽寄存器、上升沿触发选择寄存器、下降沿触发选择寄存器、软件触发寄存器、挂起寄存器。
本文结束!

 

 


http://www.ppmy.cn/server/160752.html

相关文章

重学设计模式-单例模式

一、什么是单例模式 单例模式,从字面意思理解,就是保证一个类只有一个实例,并提供一个全局访问点来访问这个实例。想象一下,在一个大型游戏中,游戏的配置信息类,整个游戏运行期间只需要一份配置数据就够了…

标签编码和独热编码对线性模型和树模型的影响

本人主页:机器学习司猫白 机器学习专栏:机器学习实战 PyTorch入门专栏:PyTorch入门 深度学习实战:深度学习 ok,话不多说,我们进入正题吧 概述 相信大家在建模中经常会用到标签编码和独热编码,这…

VSCode最新离线插件拓展下载方式

之前在vscode商店有以下类似的download按钮,但是2025年更新之后这个按钮就不提供了,所以需要使用新的方式下载 ps:给自己的网站推广下~~(国内直连GPT/Claude) 新的下载方式1 首先打开vscode商店官网:vscode插件下载…

“深入浅出”系列之数通篇:(5)TCP的三次握手和四次挥手

TCP(传输控制协议)的三次握手和四次挥手是TCP连接建立和释放的过程。 一、TCP三次握手 TCP三次握手是为了建立可靠的连接,确保客户端和服务器之间的通信能力。具体过程如下: 第一次握手:客户端向服务器发送一个带有…

AF3 ConditionedTransitionBlock类源码解读

ConditionedTransitionBlock类的核心功能是对输入特征进行非线性变换,并通过条件输入(s)自适应地调整特征的表示。 主要模块包括: 自适应层归一化(ada_ln):用于动态调整特征分布。两组线性变换(hidden_gating_linear 和 hidden_linear):用于构造特征空间中的非线性变…

Spark SQL中的from_json函数详解

Spark SQL中的from_json函数详解 在Spark SQL中,from_json是一个用于解析JSON数据的函数,主要用于将JSON格式的字符串解析为结构化的数据(即StructType或其他Spark SQL数据类型)。这个函数在处理半结构化数据(如JSON日…

前端Vue2项目使用md编辑器

项目中有一个需求,要在前端给用户展示内容,内容有 AI 生成的,返回来的是 md 格式,所以需要给用户展示 md 格式,并且管理端也可以编辑这个 md 格式的文档。 使用组件库 v-md-editor。 https://code-farmer-i.github.i…

开源轻量级文件分享服务Go File本地Docker部署与远程访问

???欢迎来到我的博客,很高兴能够在这里和您见面!希望您在这里可以感受到一份轻松愉快的氛围,不仅可以获得有趣的内容和知识,也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学习,不断总结,共同进步,活到老学到老…