痞子衡嵌入式:一个奇怪的Keil MDK下变量链接强制对齐报错问题(--legacyalign)


  大家好,我是痞子衡,是正经搞技术的痞子。今天痞子衡给大家分享的是一个奇怪的Keil MDK下变量链接强制对齐报错问题

  痞子衡最近一直在参与恩智浦SBL项目(就是一个适用LPC和i.MXRT的完整OTA方案),这个项目近期会和大家见面,项目需要同时支持GCC, IAR, MDK三大开发环境,项目所属i.MXRT1170工程在GCC和IAR下编译链接一切正常,但是在MDK下出现了链接对齐报错问题,痞子衡花时间研究解决了这个问题,这个问题算是和MDK工具本身紧紧相关,痞子衡觉得挺有意思(其实主要是想吐槽MDK),特分享给大家。

  也许问题和MDK版本有关,在分析问题前,特别交待一下版本信息:

一、L6244E报错问题

  让我们先看一下这是个啥问题,SBL项目源码引入了usb stack,在usb stack源文件usb_device_ehci.c里有如下名为qh_buffer的bss型变量定义,这个变量实际长度为3KB,我们要求MDK链接时将其放在2KB对齐的地址。

#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)自由发挥。

#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起始地址这么焦虑,实在让人费解。

二、尝试解决报错问题

2.1 调整LR_m_text起始地址

  既然链接器对LR_m_text起始地址这么焦虑,干脆不让它焦虑好了,我们直接将起始地址设成0x30000000(FlexSPI映射起始地址),因此链接文件修改如下。注:因为几个i.MXRT启动头的段都是固定地址放置的,所以起始地址的改动对他们没有影响,对其余未指定地址放置的段更没有影响。

#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加载空间没有任何联系。

==============================================================================
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

2.2 为链接器加--legacyalign选项

  上一节的方法虽然解决了问题,但是解决方案没有说服力,仅仅是个替代方案。为此痞子衡翻看了MDK官方文档,找到了如下关于链接对齐方面的一些说明文档:

  默认情况下armlink链接器假设执行区和加载区是4字节对齐的,在链接分配时需要插入一些填充空间来满足区内段的特殊对齐需求,链接器在处理填充时有两个策略:

  • 严苛策略--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主页知乎主页微信公众号 平台上。

微信搜索"痞子衡嵌入式"或者扫描下面二维码,就可以在手机上第一时间看了哦。

原文地址:https://www.cnblogs.com/henjay724/p/14069961.html

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐


  译序:JWMediaPlayer是开源的网页使用的Flash播放器。本文采摘于JWPlayer的官方文档,讲解了JWPlayer对于RTMP的使用方法,我们可以从JWPlayer客户端的角度来了解RTMP协议。以下是官方原文:      简介    RTMP(RealTimeMessagingProtocol
    Flash编程原理都是只能将1写为0,而不能将0写成1.所以在Flash编程之前,必须将对应的块擦除,而擦除的过程就是将所有位都写为1的过程,块内的所有字节变为0xFF.因此可以说,编程是将相应位写0的过程,而擦除是将相应位写1的过程,两者的执行过程完全相反.一、Nor和NandFlash
 上传setenvgatewayip192.168.1.1;setenvserverip192.168.1.7;setenvipaddr192.168.1.156;mw.b0x820000000xff0x1000000sfprobe0sfread0x8200000000x1000000tftp0x82000000test.bin0x1000000 下载mw.b82000000ff1000000tftp82000000test.bi
Error:FlashDownloadFailed-"Cortex-M3"出现一般有两种情况:1.SWD模式下,Debug菜单中,Reset菜单选项(Autodetect/HWreset/sysresetReq/Vectreset)默认是AutoDetect,改成SysResetReq即可。2.Jtag模式下,主要是芯片大小选错。Flash->ConfigureFalshTools配置窗口,切换到“Utilities"
jPlayer是一个用于控制和播放mp3文件的jQuery插件。它在后台使用Flash来播放mp3文件,前台播放器外观完全可以使用XHML/CSS自定义。支持:有一点比较好的是,在支持html5的浏览器上会使用html5的标签audio或者video,而不支持的浏览器上使用swf来播放选择需要播放的Mp3文件。播放、暂停
#ifndef__FONTUPD_H__#define__FONTUPD_H__#include"sys.h" //字库信息结构体定义33字节__packedtypedefstruct{u8fontok;//字库存在标志,0XAA,字库正常;其他,字库不存在u32ugbkaddr;//unigbk的地址u32ugbksize;//unigbk的大小u32f12addr;//gbk12地址u32g
ROM(ReadOnlyMemory)和RAM(RandomAccessMemory)指的都是半导体存储器。ROM在系统停止供电的时候仍然可以保持数据,而RAM通常都是在掉电之后就丢失数据,但是访问速度快。典型的RAM就是计算机的内存。RAM有两大类,一种称为静态RAM(StaticRAM/SRAM),SRAM速度非常快,是目前读写最快的存储
JSpc端和移动端实现复制到剪贴板功能实现在网页上复制文本到剪切板,一般是使用JS+Flash结合的方法,网上有很多相关文章介绍。随着HTML5技术的发展,Flash已经在很多场合不适用了,甚至被屏蔽。本文介绍的一款JS插件,实现了纯JS方法复制文本到剪切板。插件名是Clipboard.js,该插件不依
例子:R0=1R1=1R2=10R3=e000ed10R12=0LR=fffffff9(中断返回值)PC=0PSR=60000013或60000016或60000036(Z、C、EXCEPT_NUM:RTC_WKUP_IRQn、EXTI0_IRQn、USART2_IRQn)BFAR=e000ed38(不关心)CFSR=20000(INVSTATE:Invalidstateusagefault thePCvaluestackedf
 内存接口概念首先来分析下操作GPIO控制器和操作UART控制器两者的区别如图是S3C2440是个片上系统,有GPIO控制器(接有GPIO管脚),有串口控制器(接有TXDRXD引脚)配置GPIO控制器相应的寄存器,即可让引脚输出高低电平;配置UART控制器相应的寄存器,即可让引脚输出波形。前者相对简单,类
小编导语:    近几年来,网页游戏成为了游戏界关注的焦点,由于其制作简单,成本低并且收益率较高,因此成为了众多游戏厂商追逐的对象,但是除了商家夸张的炒作宣传外,很少有页游佳作出现。然而,随着Unity3D游戏引擎的出现,网页游戏的3D化成了页游冲出重围的杀手锏,那么在flash网页游戏称
1.指定数组到特定的Flash单元#pragmalocation=0x000FFF00 __rootconstcharFlash_config[]={0x0,0x1,0x2,0x3,0x4,0x5,0x6,0x7,0x8,0x9,0xA,0xB,0xC,0xD,0xE,0xF,0x10,0x11,0x12,0x13,0x14,0x15,0x16,0x17,0x18,0x19,0x1A,0x1B,0x1C,0x1D,0x1E,0x1F,0x20,0x21,0x22,0x23,0
继续研究发现,计算机的固件真的很有趣。参考了一些重要的资料,比如http://donovan6000.blogspot.com/2013/06/insyde-bios-modding-advanced-and-power-tabs.html等,对于IDA的使用也了解了一些。最后,总结一下目前看来可行性的方案:0.基础知识储备,包括UEFIBIOS的概念,InsydeBIOS的
<!DOCTYPEhtml><htmllang="en"><head><metacharset="UTF-8"><title>navigator对象<itle></head><body><buttononclick="checkFlash()">检测</button>
修改网上流传的flash-marker.js(function(global,factory){typeofexports==='object'&&typeofmodule!=='undefined'?module.exports=factory():typeofdefine==='function'&&define.amd?define(factory
shareObject本地缓存存储位置:win7系统用户到C:\Users\[你的用户名]\AppData\Roaming\Macromedia\FlashPlayer\#SharedObjects\XP或2003用户到:C:\DocumentsandSettings\用户名\ApplicationData\Macromedia\FlashPlayer\#SharedObjects\ ---------------------作者:iteye_
安装谷歌浏览器之后经常遇到Flash崩溃或者浏览器在浏览Flash内容时卡死的情况。在网上查找资料大多都认为应该是浏览器自带的Flash插件工作模式引起的问题,解决方法如下:首先在地址栏输入chrome://plugins/显示浏览器使用的插件。点击右上角的详细信息,可以看到Flash插件为进程外
之前一直使用的W25Q16spiflash都没问题,换了一款W25Q80后发现工作不正常,经过测试,初步定位到问题在于初始化SPI后是否将CS拉高。于是又去查看了一下原厂代码:发现原厂的代码初始化SPI接口时是专门拉高CS的。结论:网上很多代码初始化SPI接口时没有专门拉高CS,对某些型号可能确实
======================================================NANDFlash最小存储单元:写数据操作:通过对控制闸(ControlGate)施加高电压,然后允许源极(SOURCE)和汲极(RRAIN)间的N信道(N-Channel)流入电子,等到电流够强,电子获得足够能量时,便会越过浮置闸(FloatingGate)底下的二氧化硅层(S
安装CnarioPlayer3.8.1.156或其他版本时,有时会出现如下提示:Warning4154.AdobeFlashPlayer13...notcorrectlyinstalled:请前往AdobeFlash网站,并选择下图示的版本下载安装: