0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

一文了解Cortex-M中断向量表对齐原则

strongerHuang 来源:痞子衡嵌入式 作者:痞子衡 2021-10-19 11:06 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

来源 | 痞子衡嵌入式

一、Cortex-M中断向量表对齐原则

中断向量表就是一个集中保存系统全部中断处理函数(xxxIRQHandler)地址的常量数组(函数地址要占 4 个字节,因此数组中每个元素大小为 4 字节),表中元素编号如下:

1. 中断向量表第 0 - 1 个向量比较特殊,是程序初始 SP 和 PC 值
2. 中断向量表第 2 - 15 个向量是系统中断,IRQ 编号为 -14 到 -1
3. 中断向量表第 16 个向量开始是厂商自定义外设中断,IRQ 编号为 0 到 n
   - 对于 Cortex-M0/0+/1,    ARM 建议的 n 值最大为 15(实际一般厂商都会扩展)
   - 对于 Cortex-M3/4/7/23,  ARM 建议的 n 值最大为 239
   - 对于 Cortex-M33/35P/55, ARM 建议的 n 值最大为 479

Cortex-M 内核(除了CM0)模块 SCB 里有个专门的 VTOR 寄存器用来控制中断向量表首地址,程序运行起来后用户可以配置 SCB->VTOR 寄存器来重设中断向量表地址。

SCB->VTOR 寄存器低 7bit 是保留的(永远0),所以中断向量表首地址一定要是 128 字节(0x80)对齐的,这个毫无疑问!但是仅仅 128 字节对齐就行了吗?这个是要看情况的,如下 Cortex-M Generic User Guide 手册里关于 VTOR 寄存器描述里有这样一段话(红框内),这段话的意思是向量表首地址需要按 0x80 向上对齐(还得是 2 的整数幂)以覆盖项目中实际用到的数值最大中断号(xxx_IRQn)。

  • Note: 比如项目中实际用到最大外设中断为 IRQ20,则最小向量表大小为(16 + 21)* 4 字节,那么向量表首地址需要至少以 0x100 对齐。

二、Cortex-M中断向量表不对齐的后果

如果中断向量表首地址没有按规定对齐,会发生什么后果呢?我们找一块板卡来实测下,选择的芯片是恩智浦 i.MXRT1011,这是颗 Cortex-M7 内核的 MCU,除了 16 个系统中断外,还包含 80 个外设中断,中断向量表里一共 96 个有效中断。

因为 i.MXRT1011 里一共 96 个中断,按规定,中断向量表首地址至少要按 0x200 对齐。我们现在故意不按规定来设对齐,先选择一个测试工程 SDK_2.10.0_EVK-MIMXRT1010oardsevkmimxrt1010demo_appshello_worldiar(flexspi_nor_build),修改 hello_world.c 文件,加一个 relocate_vector_table() 函数,将中断向量表重定向到 NEW_VECTOR_ADDRESS:

#defineNEW_VECTOR_ADDRESS(0x00000080)

externuint32_t__VECTOR_TABLE[];
voidrelocate_vector_table(void)
{
__disable_irq();
//将0x60002000处的初始中断向量表拷贝到新地址NEW_VECTOR_ADDRESS
memcpy((void*)NEW_VECTOR_ADDRESS,(void*)__VECTOR_TABLE,0x180);
//将VTOR指向NEW_VECTOR_ADDRESS
SCB->VTOR=NEW_VECTOR_ADDRESS;
__enable_irq();
}

intmain(void)
{
relocate_vector_table();

//其余代码
}

万事俱备,我们现在需要使能一些中断来验证,痞子衡分别选取了 SysTick、LPUART1、GPT2、WDOG2、TEMP_LOW_HIGH、WDOG1 六个中断,它们的使能代码都可以从 SDKoardsevkmimxrt1010driver_examples 里找到,这里不予赘述。

2.1 测试以 0x80 对齐的中断向量表

将 NEW_VECTOR_ADDRESS 设为 ITCM 偏移 0x80 处,则中断向量表被重定向到了按 0x80 对齐的地方,分别测试选定的 6 个中断,最终结果如下:SysTick、TEMP_LOW_HIGH、WDOG1 中断响应是正常的,而 LPUART1、GPT2、WDOG2 实际响应的中断函数却是 MemManage、SysTick、DMA13 位置,这里出现了异常。

#defineNEW_VECTOR_ADDRESS(0x00000080)
47d1c196-3007-11ec-82a8-dac502259ad0.png

2.2 测试以 0x100 对齐的中断向量表

将 NEW_VECTOR_ADDRESS 设为 ITCM 偏移 0x100 处,则中断向量表被重定向到了按 0x100 对齐的地方,分别测试选定的 6 个中断,最终结果如下:SysTick、LPUART1、GPT2、WDOG2 中断响应是正常的,而 TEMP_LOW_HIGH、WDOG1 实际响应的中断函数却是 SysTick、DMA10 位置,还是出现了异常。

#defineNEW_VECTOR_ADDRESS(0x00000100)
489426fa-3007-11ec-82a8-dac502259ad0.png

2.3 测试以 0x180 对齐的中断向量表

将 NEW_VECTOR_ADDRESS 设为 ITCM 偏移 0x180 处,则中断向量表被重定向到了按 0x180 对齐的地方,实测效果跟 2.1 节一致。

#defineNEW_VECTOR_ADDRESS(0x00000180)

2.4 测试以 0x200 对齐的中断向量表

将 NEW_VECTOR_ADDRESS 设为 ITCM 偏移 0x200 处,则中断向量表被重定向到了按 0x200 对齐的地方,6 个中断都能正常响应,毕竟是符合 ARM 手册里对齐规定。

#defineNEW_VECTOR_ADDRESS(0x00000200)

2.5 测试结果总结

因为 i.MXRT1011 最多仅 96 个有效中断,有些对齐测试不能完全覆盖,痞子衡后来又在 i.MXRT1176 上(最多 234 个有效中断)以同样方式测了一遍,最终总结到现象如下(该现象可用来精简向量表,下文再聊):

1. 当中断向量表以 0x80 对齐时:
  - 表中 (2n*0x80)/4 处开始的连续 32 个中断均能够正常响应,n 可取值 0 - 7
  - 表中 ((2n+1)*0x80)/4 处开始的连续 32 个中断发生时,实际响应的却是表中((2n*0x80)/4 处对应的连续 32 个中断函数
2. 当中断向量表以 0x100 对齐时:
  - 表中 (2n*0x100)/4 处开始的连续 64 个中断均能够正常响应,n 可取值 0 - 4
  - 表中 ((2n+1)*0x100)/4 处开始的连续 64 个中断发生时,实际响应的却是表中((2n*0x100)/4 处对应的连续 64 个中断函数
3. 当中断向量表以 0x200 对齐时:
  - 表中 (2n*0x200)/4 处开始的连续 128 个中断均能够正常响应,n 可取值 0 - 1
  - 表中 ((2n+1)*0x200)/4 处开始的连续 128 个中断发生时,实际响应的却是表中((2n*0x200)/4 处对应的连续 128 个中断函数
4. 当中断向量表以 0x400 对齐时:
  - 表中前 256 个中断均能够正常响应
  - 推测表中第 256 - 511 个中断发生时,实际响应的是表中 0 - 255 个中断函数
5. 当中断向量表以 0x800 对齐时:
  - 表中前 512 个中断均能够正常响应

6. 当中断向量表以 0x180 对齐时:未详尽测试,效果似乎与以 0x80 对齐一致
7. 当中断向量表以 0x280 对齐时:未详尽测试,第 100 个中断误触发第 68 个中断函数,第 136 个中断触发 HardFault
8. 当中断向量表以 0x300 对齐时:未详尽测试,第 100/136 个中断均触发 HardFault
...

至此,Cortex-M中断向量表对齐原则介绍完了~~~

编辑:jq
声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 寄存器
    +关注

    关注

    31

    文章

    5590

    浏览量

    129158
  • 函数
    +关注

    关注

    3

    文章

    4408

    浏览量

    66905
  • Cortex-M
    +关注

    关注

    2

    文章

    234

    浏览量

    30973

原文标题:一文了解Cortex-M中断向量表对齐原则

文章出处:【微信号:strongerHuang,微信公众号:strongerHuang】欢迎添加关注!文章转载请注明出处。

收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    如何在CW32F030上实现IAP功能实现远程升级?

    FLASH 中存放的地址,本例设置为 0x00002000。这样用户程序的中断向量表就被重定位到 0x00002000 处,与 ARM®Cortex®-M0+ 内核默认的中断向量表
    发表于 12-11 06:15

    Cortex-M产品的特色

    的处理能力。 中断控制器:Cortex-M处理器内置了灵活的中断控制器,能够支持多种中断优先级和高效的中断处理机制。 访问控制单
    发表于 11-26 07:22

    Cortex-M内核中的精确延时的方法

    使用 CYCCNT寄存器来测量执行某个任务所花的周期数,这也可以用作时间基准相关的目的(操作系统中统计 CPU使用率可以用到它)。” Cortex-M中的DWT它有个32位的寄存器叫CYCCNT
    发表于 11-21 07:51

    Cortex-M级别的转换

    、 简述 Cortex-M 里面有特权级别的概念,不同级别可以设定不同的权限,如何转换特权级别基本是本章的内容。 二、操作模式 ARM M 核操作模式有两个: 线程(Thread)模式:在复位时或
    发表于 11-19 07:32

    MCU中断不触发的排查

    1、NVIC中断是否使能? 2、外设本身的中断是否使能(例如 UART 的接收中断使能位)? 3、中断服务函数名是否与向量表中的名称完全
    发表于 11-18 08:19

    从riscv底层原理分析gd32vf103的中断行为

    从riscv底层原理分析gd32vf103的中断行为 1.概述 2.中断向量表初始化 3.详细分析下irq_entry 4.关于gd32vf103中断编程模型的理解 1.概
    发表于 10-31 08:04

    RVMCU课堂「12」: 手把手教你玩转RVSTAR—外部中断

    配置不同的中断,需要修改此参数为对应的中断号。 最后个参数配置的是中断处理函数。直接来看的话,虽然这里写的是“NULL”,但是并不代表没有中断
    发表于 10-31 07:39

    RVMCU课堂「10」: 手把手教你玩转RVSTAR—处理器内部中断

    /quickstart-doc-u-nuclei_interrupt_quickstart.html)进行学习。 6 mtimer_irq_handler 第六个参数配置的是中断处理函数。在讲解此参数前,需要中断向量表
    发表于 10-31 06:12

    中断向量表中,数字较小的中断向量能否以 larg 中断中断向量

    中断向量表中,数字较小的中断向量能否以 larg 中断中断向量
    发表于 08-21 08:17

    请问NuMicro® Cortex-M® 系列芯片是否支持 I2C 监视器功能?

    NuMicro® Cortex-M® 系列芯片是否支持 I2C 监视器功能?
    发表于 08-21 06:04

    AT32 IAP using the USART

    的复位中断向量起始地址为 0x08000004+N+M),跳转至新写入程序的复位向量表,取出新程序的复位中断向量的地址,并跳转执行新程序的复位中断
    发表于 07-11 10:51

    第八章 启动文件详解

    本章讲解了W55MH32的启动文件,其由汇编编写,系统上电后首执行,完成初始化堆栈、中断向量表、配置系统时钟等工作,还介绍了常用ARM汇编指令及代码结构。
    的头像 发表于 05-22 16:52 1939次阅读
    第八章 启动文件详解

    一天一个嵌入式面试知识(1:中断处理机制详解)

    事件(如定时器溢出)发送中断请求(IRQ)。 中断使能:在NVIC(嵌套向量中断控制器)中开启对应中断通道。
    发表于 03-21 10:18

    瑞萨RA8快速上手指南:Cortex-M85内核瑞萨RA8开发环境搭建 并点亮个LED

    因为Cortex-M内核,瑞萨RA8系列单片机支持多种市面上常见的开发环境,像Keil MDK、IAR EWARM等,而本文讲述的是瑞萨自家官方的IDE(e2 studio)。
    的头像 发表于 03-17 14:35 1606次阅读
    瑞萨RA8快速上手指南:<b class='flag-5'>Cortex-M</b>85内核瑞萨RA8开发环境搭建 并点亮<b class='flag-5'>一</b>个LED

    程序中断的100种写法

    。 信号处理:在Unix/Linux系统中,通过信号(signal)处理机制实现中断。 异常处理:由程序中的异常(如除零错误、段错误)触发。 3. 中断控制器配置 中断向量表:在嵌入式系统中,通过配置
    发表于 01-28 08:25