MCU如何从向量表到中断服务

article/2025/8/11 16:36:51

目录

1、中断向量表

2、编写中断服务例程


中断处理的核心是中断向量表(IVT),它是一个存储中断服务例程(ISR)地址的内存结构。当中断发生时,MCU通过IVT找到对应的ISR地址并跳转执行。本文将深入探讨MCU(以ARM Cortex-M架构为例)如何从向量表过渡到中断服务。

1、中断向量表

中断向量表(IVT)是一个存储中断和异常处理程序地址的表格,位于MCU的内存中。在ARM Cortex-M处理器中,IVT通常从地址0x00000000开始,包含初始堆栈指针和各种异常处理程序的地址。每个表项占用4字节,存储一个32位地址,指向对应的ISR或异常处理函数。

IVT的作用是提供一种快速查找机制,使MCU在中断发生时能够迅速定位并执行相应的处理程序。例如,当定时器中断触发时,MCU会根据中断号从IVT中获取ISR地址。

以下是一个简化的IVT结构示例:

IVT通常存储在闪存(Flash)中,但可以通过向量表偏移寄存器(VTOR)将其重定位到其他内存区域(如SRAM),以支持动态更新或引导加载程序(Bootloader)场景。

ARM Cortex-M处理器通过嵌套向量中断控制器(NVIC)管理中断。以下是中断处理的具体步骤:

  • 完成当前指令:处理器首先完成当前正在执行的指令,除非该指令是可中断继续指令(interrupt-continuable instruction)。这确保了指令的原子性。
  • 保存上下文:处理器自动将当前程序的上下文保存到堆栈上,包括寄存器R0-R3、R12、链接寄存器(LR)、程序计数器(PC)和程序状态寄存器(xPSR)。这些寄存器共占用32字节(每个寄存器4字节)。对于使用浮点单元(FPU)的系统,FPU寄存器也可能被保存。
  • 读取中断号:NVIC从中断程序状态寄存器(IPSR)获取中断号(异常号),该中断号用于索引IVT。
  • 获取ISR地址:使用中断号N,处理器计算IVT中的地址(4 × N),从中读取ISR的起始地址。例如,SysTick中断(异常号15)的地址位于0x0000003C。
  • 更新寄存器:处理器更新堆栈指针(SP)、链接寄存器(LR)和程序计数器(PC)。LR被设置为特殊值(如0xFFFFFFF9),指示返回模式(例如,返回线程模式并使用主堆栈指针)。
  • 执行ISR:处理器跳转到ISR地址并执行中断处理程序。ISR通常会清除中断标志并执行必要的操作。
  • 异常返回:ISR完成后,处理器从堆栈恢复保存的上下文,并使用LR中的特殊值返回到被中断的程序,继续执行。

向量表通常在启动代码中定义,由工具链提供或由开发者编写。在C语言中,向量表可以定义为一个函数指针数组。例如:

void (* const vectors[])(void) __attribute__((section(".vectors"))) = {(void (*)(void)) &_stack_top, // 初始堆栈指针Reset_Handler,                // 复位处理程序NMI_Handler,                  // NMI处理程序HardFault_Handler,            // 硬故障处理程序// ... 其他处理程序SysTick_Handler               // SysTick处理程序
};

其中:

  • _stack_top由链接脚本定义,表示堆栈顶部地址。
  • __attribute__((section(".vectors")))确保数组被放置在链接脚本指定的.vectors段,通常映射到0x00000000。
  • 每个处理程序(如Reset_Handler)是一个函数,定义在其他代码文件中。

在汇编语言中,向量表可能如下定义(以Keil MDK的startup.s为例):

    .section .vectors.word _estack.word Reset_Handler.word NMI_Handler.word HardFault_Handler// ....word SysTick_Handler

链接脚本确保.vectors段位于正确地址。开发者通常无需直接修改向量表,只需实现对应的ISR函数。

2、编写中断服务例程

ISR是处理特定中断的函数,必须高效且可靠。以下是编写ISR的最佳实践:

  • 保持简洁:ISR应尽量短,避免复杂计算或阻塞调用。
  • 清除中断标志:确保清除触发中断的标志,以防止重复触发。
  • 使用volatile变量:防止编译器优化导致变量访问错误。
  • 避免嵌套复杂逻辑:如果需要复杂处理,可将任务标记为待处理并在主循环中执行。

以下是一个简单的定时器中断ISR示例:

void TIM2_IRQHandler(void) {TIM2->SR &= ~TIM_SR_UIF; // 清除更新中断标志// 执行操作,例如切换LED状态
}

ARM Cortex-M通过NVIC支持嵌套中断,允许高优先级中断抢占低优先级中断。优先级通过NVIC的优先级寄存器(IPR)配置,值越小优先级越高。例如,Cortex-M3/M4支持最多256个优先级级别,但具体实现可能只使用部分位(如4位,支持16个级别)。

开发者可以通过以下方式管理中断优先级:

  • 使用NVIC函数(如NVIC_SetPriority(IRQn, priority))设置优先级。
  • 通过__enable_irq()和__disable_irq()全局启用或禁用中断。
  • 使用BASEPRI寄存器屏蔽低于某优先级的中断(在Cortex-M3/M4中支持)。

嵌套中断的典型场景是高优先级中断(如紧急传感器触发)打断低优先级中断(如定时器任务),确保关键任务优先执行。

从向量表到中断服务的过程是嵌入式系统中中断管理的核心。通过理解IVT的结构、中断处理流程以及如何配置和编写ISR,开发者可以设计高效、实时的嵌入式系统。ARM Cortex-M的NVIC和硬件优化(如尾链式处理)进一步降低了中断延迟,提高了系统性能。

虽然本文以ARM Cortex-M为例,但其他MCU(如8051)也有类似的中断处理机制,尽管向量表地址和异常号可能不同。开发者应参考具体MCU的数据手册和工具链文档以获取详细信息。


http://www.hkcw.cn/article/rjxLNJKyTN.shtml

相关文章

Docker Compose(容器编排)

目录 什么是 Docker Compose Docker Compose 的功能 Docker Compose 使用场景 Docker Compose 文件(docker-compose.yml) Docker Compose 命令清单 常见命令说明 操作案例 总结 什么是 Docker Compose docker-compose 是 Docker 官方的开源项…

安卓jetpack compose学习笔记-UI基础学习

哲学知识应该用哲学的方式学习,技术知识也应该用技术的方式学习。没必要用哲学的态度来学习技术。 学完安卓技术能做事就ok了,安卓技术肯定是有哲学的,但是在初学阶段没必要讨论什么安卓哲学。 学习一们复杂技术的路径有很多,这里…

[蓝桥杯]螺旋折线

螺旋折线 题目描述 如下图所示的螺旋折线经过平面上所有整点恰好一次。 对于整点 (X,Y)(X,Y),我们定义它到原点的距离 dis(X,Y)dis(X,Y) 是从原点到 (X,Y)(X,Y) 的螺旋折线段的长度。 例如 dis(0,1)3,dis(−2,−1)9dis(0,1)3,dis(−2,−1)9。 给出整点坐标 (X,Y…

【动态规划】子序列问题(一)

📝前言说明: 本专栏主要记录本人的动态规划算法学习以及LeetCode刷题记录,按专题划分每题主要记录:(1)本人解法 本人屎山代码;(2)优质解法 优质代码;&…

一文读懂Ingress-Nginx以及实践攻略

一文读懂Ingress-Nginx以及实践攻略 目录 1 概念 1.1 什么是Ingress? 1.1.1 主要功能:1.2 Ingress的组件1.3 什么是ingress-nginx1.4 ingress-nginx优点和限制1.5 版本兼容性矩阵2 实践: Ingress nginx部署 2.1 使用helm部署ingress-nginx 2.1.1 安装和配置Helm2.1.2 配置和…

一、【专栏启动篇】:为什么是 Django + Vue3?测试平台的技术选型与架构蓝图

【专栏启动篇】:为什么是 Django Vue3?测试平台的技术选型与架构蓝图 前言一、为什么是 Django Vue3?二、测试平台的架构设计蓝图三、测试平台模块功能概述 结语 前言 一个高效、稳定、易用的测试平台,不仅能够帮助团队提升测试…

基于OAuth2+SpringSecurity+Jwt实现身份认证和权限管理后端服务

1、简介 本文讲述了如何实现简易的后端鉴权服务。所谓“鉴权”,就是“身份鉴定”“权限判断”。涉及的技术有:OAuth2、SpringSecurity、Jwt、过滤器、拦截器。OAuth2用于授权,使用Jwt签发Access Token和Refresh Token,并管理token…

基于SpringBoot和PostGIS的云南与缅甸的千里边境线实战

目录 前言 一、PostGIS空间求解 1、相邻的求解 二、后台程序实现 1、数据查询的实现 2、API接口实现 三、WebGIS可视化实现 1、空间面展示 2、增加面标注 3、图例展示 4、与缅甸距离较近的区县信息 四、总结 前言 云南,这个位于中国西南边陲的省份&…

【带小白做项目】如何在SpringBoot项目中接入AI大模型?

随着chatGPT的兴起,越来越多的应用接入了AI大模型,那么我们要怎么在自己的项目中接入AI大模型呢?本节我们一起做一个简单的demo来尝试一下。 一 为什么要在项目中接入大模型 1. 增强业务功能和用户体验 AI 大模型(如 GPT、BERT…

【计算机主板架构】ATX架构

一、引言 在计算机的世界里,主板就如同一个城市的基础设施,承载着各种重要的组件并协调它们的工作。而ATX(Advanced Technology Extended)架构的主板,自问世以来,一直在计算机硬件领域占据着举足轻重的地位…

精选了几道MySQL的大厂面试题,被提问的几率很高!

🎥 作者简介: CSDN\阿里云\腾讯云\华为云开发社区优质创作者,专注分享大数据、Python、数据库、人工智能等领域的优质内容 🌸个人主页: 长风清留杨的博客 🍃形式准则: 无论成就大小,…

搞定mysql的 行转列(7种方法) 和 列转行

一、行转列 1、使用case…when…then 2、使用SUM(IF()) 生成列 3、使用SUM(IF()) 生成列 WITH ROLLUP 生成汇总行 4、使用SUM(IF()) 生成列,直接生成汇总结果,不再利用子查询 5、使用SUM(IF()) 生成列 UNION 生成汇总行,并利用 IFNULL将汇总行标题…

高并发场景下的热点key问题探析与应对策略

目录 一、问题描述 二、发现机制 三、解决策略分析 (一)解决策略一:多级缓存策略 客户端本地缓存 代理节点本地缓存 (二)解决策略二:多副本策略 (三)解决策略三:热点…

SQL Server——SSMS中数据库、表的创建

目录 一、引言 二、数据库、表的创建与删除 (一)方法一:在SSMS控制平台上进行创建 (二)方法二:使用 SQL 代码实现对数据库和表的创建 三、SQL 和 T-SQL 一、引言 在学习数据库的过程中,初…

spring AOP详解

文章目录 AOP1 环境准备1.1 工程及接口创建1.2 工程存在的问题1.2.1 问题1.2.2 解决思路 2 AOP面向切面编程2.1 AOP概述2.2 AOP原理分析 3 基于注解的AOP3.1 入门示例3.2 使用流程3.3 切入点表达式3.4 练习3.5 通知类型 AOP ​ AOP(Aspect Orient Programming&…

重看Spring聚焦ApplicationContext分析

目录 一、理解下ApplicationContext的设计 (一)功能性的理解 (二)ApplicationContext 结构类图 二、ApplicationContext根接口 (一)源码展示 (二)分析说明 三、子接口Configu…

【MySQL安装】—报错“Can‘t connect to local MySQL server through socket ‘varlibmysqlmysql.sock‘”

项目场景: 执行 “mysql -uroot -p” 命令,进入MySQL数据库。 问题描述: 报错: Cant connect to local MySQL server through socket /var/lib/mysql/mysql.sock 原因分析: /var/lib/mysql路径下缺少mysql.sock文件。 …

本地部署Vanna实战,快速解决NLP2SQL

一、背景 ​ 随着DeepSeek的火爆,基于AI的应用也如雨后春笋般迸发出来,如何根据用户的一句话来找到用户所需要的信息,采用传统的方式无法通过模糊匹配等实现复杂的业务场景,故探索一种新的思路来实现信息获取。Text2SQL将自然语言…

【MySQL】基础操作

MySQL(二)基础操作 一、数据库操作 1.创建库 2.查看库 3.选中库 4.删除库 二、表操作 1.创建表 1.1[comment 注释]: 1.2,...: 2.查看表 2.1查看所有表 2.2查看表结构 3.删除表 三、记录操作 1.插入记录 1.1全列插入 1.2指定列插入 1.3…

嵌入式硬件篇---蜂鸣器

蜂鸣器是一种常用的电子发声元件,主要分为有源蜂鸣器和无源蜂鸣器两类。它们在结构、工作原理、驱动方式、应用场景等方面存在显著差异。以下是详细介绍: 一、核心定义与结构差异 1. 有源蜂鸣器 定义: “有源” 指内部自带振荡电路&#x…