关注+星标公众,不错过精彩内容

来源 | 痞子衡嵌入式

一、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 个有效中断,见如下 startup_MIMXRT1011.s 文件中具体中断响应函数定义:

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

#define NEW_VECTOR_ADDRESS (0x00000080)extern uint32_t __VECTOR_TABLE[];
void relocate_vector_table(void)
{__disable_irq();// 将 0x60002000 处的初始中断向量表拷贝到新地址 NEW_VECTOR_ADDRESSmemcpy((void *)NEW_VECTOR_ADDRESS, (void *)__VECTOR_TABLE, 0x180);// 将 VTOR 指向 NEW_VECTOR_ADDRESSSCB->VTOR = NEW_VECTOR_ADDRESS;__enable_irq();
}int main(void)
{relocate_vector_table();// 其余代码
}

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

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

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

#define NEW_VECTOR_ADDRESS (0x00000080)

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

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

#define NEW_VECTOR_ADDRESS (0x00000100)

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

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

#define NEW_VECTOR_ADDRESS (0x00000180)

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

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

#define NEW_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中断向量表对齐原则介绍完了~~~

------------ END ------------

后台回复『Cortex-M』『单片机』相关文章。

欢迎关注我的公众号回复“加群”按规则加入技术交流群,回复“1024”查看更多内容。

欢迎关注我的视频号:

点击“阅读原文”查看更多分享,欢迎点分享、收藏、点赞、在看。

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

  1. 用对齐原则求结构体长度

    用对齐原则求结构体长度 [日期:2009-09-18] 来源:中嵌信息  作者:chinaeda-news [字体:大 中 小] 1. 熟悉Win32下VC6.0各种基本数据长度: size of i ...

  2. 32位操作系统和64位操作系统每种类型占用的字节数、内存对齐原则

    32位操作系统: 1:整形 int 4字节 long int 4字节 short 2字节 unsigned int 4字节 unsigned long int 4字节 unsigned short 2 ...

  3. mysql左对齐原则_MySQL 设计与开发规范

    MySQL 设计与开发规范 1 目的 本规范的主要目的是希望规范数据库设计与开发,尽量避免由于数据库设计与开发不当而产生的麻烦:同时好的规范,在执行的时候可以培养出好的习惯,好的习惯是软件质量的很好保 ...

  4. 一文了解结构体字节对齐

    结构体字节对齐详解 表述如有不正确的地方,欢迎批评指正. C++/C 常见的基本数据类型: bool short (short int) int long (long int) long long ( ...

  5. java字节对齐原则_C struct 中字节对齐问题

    为了提高CPU的存储速度,VC对一些变量的 起始地址做了"对齐"处理.在默认情况下,VC规定各成员变量存放的起始地址相对于结 构的起始地址的偏移量必须为该变量的类型所占用的字节数的 ...

  6. word右顶格,word文档最右边怎么对齐

    我们通过大纲级别自动生成目录之后这样目录页码也同时自动生成,目录页码右对齐到目录中.这里你先学会如何自动添加目录的方法即可.word目录页码对不齐,word目录页码右对齐出现错位是怎么回事? 故障分析 ...

  7. 万字好文!探究Java的设计原则,看了都说好!

    点击上方 "程序员小乐"关注, 星标或置顶一起成长 后台回复"大礼包"有惊喜礼包! 关注订阅号「程序员小乐」,收看更多精彩内容 每日英文 Man has to ...

  8. Cortex M4 中断向量表

    前言 重新学习Cortex M4的中断向量表内容.中断向量表是Cortex M4自带的功能,厂家可以根据需要自行定义,所以需要学习该内容,直接看Cortex M4内核相关即可.我这查看的是<Co ...

  9. 【转】彻底搞清计算结构体大小和数据对齐原则

    数据对齐: 许多计算机系统对基本数据类型合法地址做出了一些限制,要求某种类型对象的地址必须是 某个值K(通常是2,4或8)的倍数.这种对齐限制简化了形成处理器和存储器系统之间的接口的硬件 设计.例如, ...

最新文章

  1. sulime text 3
  2. java mac postgresql_Mac上安装配置和简单使用PostgreSQL(仍然很不懂)
  3. object overview page打开后白屏问题分析
  4. Java项目几个月能完成_c#项目转JAVA,第5个月,基本完成
  5. 19年8月 字母哥 第五章 静态资源与模板引擎的整合 用热点公司网不行
  6. Anti-debugging Skills in APK
  7. 业务建模重的几个概念
  8. 深入了解 Loader
  9. jvisualvm/Jconsole监控WAS(WebSphere)中间件
  10. rust大油田分解机_辽河油田曙光采油厂:智慧党建建强战斗堡垒
  11. html显示日期时间代码,JS全中文显示日期时间代码
  12. Android Studio 设置/更改 SDK 路径
  13. 4. jQuery 事件
  14. Office server webs app 集成(JAVA)
  15. 我的漫漫程序人生路(真诚的长文,慎点)
  16. 国内各大企业邮箱,选择看重哪几个方面?
  17. git push 时出现错误error: failed to push some refs to ‘https://gitee.com/**.git‘
  18. 《计算之魂》Task3:关于排序的讨论
  19. 如何将视频中的音频提取出来
  20. 如何看待中科院合肥研究院 90 多名科研人员集体辞职?

热门文章

  1. Announcing the program for the 2018 LLVM Developers’ Meeting Bay Area
  2. empirecms php 很慢,帝国cms提高网站网页打开速度的方法总结
  3. 生活随笔:遇到电梯吸烟烟鬼
  4. Android Stock Browser Web App开发当中遇到的问题(持续更新)
  5. JavaWEB三:Javascript
  6. CATIA CAA 32位和64位编译
  7. Linux中du -h与df -h 区别
  8. html5走势图动态,CSS3+jQuery的互动走势图
  9. 【开心一刻】为什么程序员怕改需求?看完这些神解释我笑了
  10. 如何组装一台安全机器人?天线是必不可少的组件