你们好,我是痞子衡,是正经搞技术的痞子。今天痞子衡给你们分享的是一个奇怪的Keil MDK下变量连接强制对齐报错问题。微信
痞子衡最近一直在参与恩智浦SBL项目(就是一个适用LPC和i.MXRT的完整OTA方案),这个项目近期会和你们见面,项目须要同时支持GCC, IAR, MDK三大开发环境,项目所属i.MXRT1170工程在GCC和IAR下编译连接一切正常,可是在MDK下出现了连接对齐报错问题,痞子衡花时间研究解决了这个问题,这个问题算是和MDK工具自己牢牢相关,痞子衡以为挺有意思(其实主要是想吐槽MDK),特分享给你们。工具
也许问题和MDK版本有关,在分析问题前,特别交待一下版本信息:flex
让咱们先看一下这是个啥问题,SBL项目源码引入了usb stack,在usb stack源文件usb_device_ehci.c里有以下名为qh_buffer的bss型变量定义,这个变量实际长度为3KB,咱们要求MDK连接时将其放在2KB对齐的地址。ui
#define USB_DEVICE_CONFIG_EHCI (2) #define USB_DEVICE_CONFIG_ENDPOINTS (8U) __attribute__((aligned(2048))) static uint8_t qh_buffer[(USB_DEVICE_CONFIG_EHCI - 1) * 2048 + USB_DEVICE_CONFIG_ENDPOINTS * 2 * sizeof(usb_device_ehci_qh_struct_t)];
以下是SBL项目的配套MDK连接文件(MIMXRT1176xxxxx_cm7_flexspi_nor.scf),工程代码是XIP执行的。从连接文件内容来看,这是一个很是普通的连接文件,除了为i.MXRT启动头(FDCB、IVT、BootData)作了一些特殊放置外,其他都是常规连接语句,没有再为其余代码或变量作特殊放置,基本就是让连接器(armlink)自由发挥。.net
#define m_flash_config_start 0x30000400 #define m_flash_config_size 0x00000C00 #define m_ivt_start 0x30001000 #define m_ivt_size 0x00001000 #define m_interrupts_start 0x30002000 #define m_interrupts_size 0x00000400 #define m_text_start 0x30002400 #define m_text_size 0x00FBDC00 #define m_data_start 0x20000000 #define m_data_size 0x00040000 #define Stack_Size 0x0400 #define Heap_Size 0x0400 LR_m_text m_flash_config_start m_text_start+m_text_size-m_flash_config_start { ; 放置FDCB(i.MXRT特点) RW_m_config_text m_flash_config_start FIXED m_flash_config_size { * (.boot_hdr.conf, +FIRST) } ; 放置IVT, BootData(i.MXRT特点) RW_m_ivt_text m_ivt_start FIXED m_ivt_size { * (.boot_hdr.ivt, +FIRST) * (.boot_hdr.boot_data) } ; 放置中断向量表 VECTOR_ROM m_interrupts_start FIXED m_interrupts_size { * (.isr_vector,+FIRST) } ; 放置程序代码 ER_m_text m_text_start FIXED m_text_size { * (InRoot$$Sections) .ANY (+RO) } ; 放置程序变量 RW_m_data m_data_start m_data_size-Stack_Size-Heap_Size { .ANY (+RW +ZI) * (RamFunction) * (NonCacheable.init) * (*NonCacheable) } ; 放置程序堆、栈 ARM_LIB_HEAP +0 EMPTY Heap_Size { } ARM_LIB_STACK m_data_start+m_data_size EMPTY -Stack_Size { } }
编译工程获得一个以下图所示奇怪连接错误,连接器说LR_m_text起始地址没有按2KB对齐。连接文件里指定的LR_m_text加载区地址范围[m_flash_config_start, m_text_start+m_text_size-m_flash_config_start]只是一个最大的RO存储范围,虽然m_flash_config_start等于0x30000400,可是这个起始地址是指定用于放置FDCB的,何况本文主角qh_buffer是个bss型变量(初始化值为0,不需在flash里放初值),彻底不占用RO区,仅需分配RW区便可,连接器由于qh_buffer的对齐需求而对LR_m_text起始地址这么焦虑,实在让人费解。3d
既然连接器对LR_m_text起始地址这么焦虑,干脆不让它焦虑好了,咱们直接将起始地址设成0x30000000(FlexSPI映射起始地址),所以连接文件修改以下。注:由于几个i.MXRT启动头的段都是固定地址放置的,因此起始地址的改动对他们没有影响,对其他未指定地址放置的段更没有影响。code
#define m_flash_start 0x30000000 #define m_flash_size 0x00FC0000 #define m_flash_config_start 0x30000400 #define m_flash_config_size 0x00000C00 LR_m_text m_flash_start m_flash_size { ;改动在这里!!! ; 放置FDCB RW_m_config_text m_flash_config_start FIXED m_flash_config_size { * (.boot_hdr.conf, +FIRST) } ; 放置IVT, BootData ; 放置中断向量表 ; 放置程序代码 ; 放置程序变量 ; 放置程序堆、栈 }
改完连接文件后从新编译MDK工程,此次没有连接错误了,咱们打开工程映射文件(sbl.map),找出其中跟qh_buffer相关的内容,能够看到qh_buffer被放置在了0x20004800处,这个地址确实是2KB对齐的,但这是RW区,其实跟咱们设定/改动的LR_m_text加载空间没有任何联系。htm
============================================================================== Image Symbol Table Local Symbols Symbol Name Value Ov Type Size Object(Section) qh_buffer 0x20004800 Data 3072 usb_device_ehci.o(.bss.qh_buffer) [Anonymous Symbol] 0x20004800 Section 0 usb_device_ehci.o(.bss.qh_buffer) ============================================================================== Memory Map of the image Image Entry point : 0x30002401 Load Region LR_m_text (Base: 0x30000000, Size: 0x00011800, Max: 0x00fc0000, ABSOLUTE, COMPRESSED[0x00011518]) Execution Region RW_m_data (Exec base: 0x20000000, Load base: 0x30010000, Size: 0x00005ed8, Max: 0x0003f800, ABSOLUTE, COMPRESSED[0x00000800]) Exec Addr Load Addr Size Type Attr Idx E Section Name Object 0x20004800 - 0x00000c00 Zero RW 2164 .bss.qh_buffer usb_device_ehci.o
上一节的方法虽然解决了问题,可是解决方案没有说服力,仅仅是个替代方案。为此痞子衡翻看了MDK官方文档,找到了以下关于连接对齐方面的一些说明文档:blog
- 关于--legacyalign, --no_legacyalign 用法解释:https://www.keil.com/support/man/docs/armlink/armlink_pge1362075504330.htm
- Section alignment with the linker 用法规定:https://www.keil.com/support/man/docs/armclang_ref/armclang_ref_pge1362065911965.htm
默认状况下armlink连接器假设执行区和加载区是4字节对齐的,在连接分配时须要插入一些填充空间来知足区内段的特殊对齐需求,连接器在处理填充时有两个策略:ci
- 严苛策略--no_legacyalign(默认):指示连接器插入填充以强制执行区首地址天然对齐,这里的天然对齐是该区域内已知的最大对齐。这个选项能够确保严格符合ELF规范。
- 宽松策略--legacyalign:指示连接器按最小化对齐方式来插入填充。
读到这里,咱们好像找到了一开始报错的缘由,就是默认的--no_legacyalign捣的鬼,连接器应该根据LR_m_text区首地址按qh_buffer对齐要求来填充,但实际上连接器却直接撂挑子不干了,报了个错。那咱们就不让连接器为难了,给它个宽松策略:
这样改动以后,不须要调整连接文件,MDK工程也能正常编译链接了。再来看映射文件(sbl/map),qh_buffer连接地址相比前一个方案发生了变化,从0x20004800移到了0x20004000,但依然知足2KB对齐的。
============================================================================== Image Symbol Table Local Symbols Symbol Name Value Ov Type Size Object(Section) qh_buffer 0x20004000 Data 3072 usb_device_ehci.o(.bss.qh_buffer) [Anonymous Symbol] 0x20004000 Section 0 usb_device_ehci.o(.bss.qh_buffer) ============================================================================== Memory Map of the image Image Entry point : 0x30002401 Load Region LR_m_text (Base: 0x30000400, Size: 0x00010944, Max: 0x00fbfc00, ABSOLUTE, COMPRESSED[0x00010408]) Execution Region RW_m_data (Exec base: 0x20000000, Load base: 0x3000f944, Size: 0x000056d8, Max: 0x0003f800, ABSOLUTE, COMPRESSED[0x000001ac]) Exec Addr Load Addr Size Type Attr Idx E Section Name Object 0x20004000 - 0x00000c00 Zero RW 2164 .bss.qh_buffer usb_device_ehci.o
最后再提一个MDK自相矛盾的地方,咱们加了--legacyalign选项后编译给了个以下警告,说--legacyalign选项不推荐使用。
然而咱们在MDK官方文档里看到了备注,说的是armlink v6.6版本以上不推荐加--no_legacyalign选项,那痞子衡正在使用的armlink v6.14版本应该是建议使用--legacyalign选项的,可是为什么给警告呢?MDK啊,想说爱你不容易!
至此,一个奇怪的Keil MDK下变量连接强制对齐报错问题痞子衡便介绍完毕了,掌声在哪里~~~
文章会同时发布到个人 博客园主页、CSDN主页、知乎主页、微信公众号 平台上。
微信搜索"痞子衡嵌入式"或者扫描下面二维码,就能够在手机上第一时间看了哦。