【问题标题】:GNU Linker Script - move flash memory to new regionGNU 链接器脚本 - 将闪存移动到新区域
【发布时间】:2020-05-15 10:43:52
【问题描述】:

我正在尝试使用 STM32F446ZE 微控制器的链接器脚本来移动内存部分。我最初的设置是这样的:

MEMORY
{
RAM (xrw)      : ORIGIN = 0x20000000, LENGTH = 128K
FLASH (rx)      : ORIGIN = 0x8000000, LENGTH = 512K - 128k
DATA (rwx)      : ORIGIN = 0x08060000, LENGTH = 5120
}

SECTIONS
{
  .user_data :
  {
    . = ALIGN(4);
    KEEP(*(.user_data))
    . = ALIGN(4);
  } >DATA
  .isr_vector :
  {
    . = ALIGN(4);
    KEEP(*(.isr_vector)) /* Startup code */
    . = ALIGN(4);
  } >FLASH

  .text :
  {
    . = ALIGN(4);
    *(.text)           /* .text sections (code) */
    *(.text*)          /* .text* sections (code) */
    *(.glue_7)         /* glue arm to thumb code */
    *(.glue_7t)        /* glue thumb to arm code */
    *(.eh_frame)
    KEEP (*(.init))
    KEEP (*(.fini))
    . = ALIGN(4);
    _etext = .;        /* define a global symbols at end of code */
  } >FLASH

  .rodata :
  {
    . = ALIGN(4);
    *(.rodata)         /* .rodata sections (constants, strings, etc.) */
    *(.rodata*)        /* .rodata* sections (constants, strings, etc.) */
    . = ALIGN(4);
  } >FLASH

  .data : 
  {
    . = ALIGN(4);
    _sdata = .;        /* create a global symbol at data start */
    *(.data)           /* .data sections */
    *(.data*)          /* .data* sections */
    . = ALIGN(4);
    _edata = .;        /* define a global symbol at data end */
  } >RAM AT> FLASH

  . = ALIGN(4);
  .bss :
  {
    /* This is used by the startup in order to initialize the .bss secion */
    _sbss = .;         /* define a global symbol at bss start */
    __bss_start__ = _sbss;
    *(.bss)
    *(.bss*)
    *(COMMON)
    . = ALIGN(4);
    _ebss = .;         /* define a global symbol at bss end */
    __bss_end__ = _ebss;
  } >RAM

  /* User_heap_stack section, used to check that there is enough RAM left */
  ._user_heap_stack :
  {
    . = ALIGN(4);
    PROVIDE ( end = . );
    PROVIDE ( _end = . );
    . = . + _Min_Heap_Size;
    . = . + _Min_Stack_Size;
    . = ALIGN(4);
  } >RAM

我想要做的是将 DATA 移动到从 0x08000000(闪存当前正在启动的位置)开始,并在 0x08040000(在 DATA 之后)启动 FLASH。我可以很容易地在内存部分更改它,但是我的程序无法启动。我相信 SECTIONS 块中的某些代码可能必须更改,但我不确定如何更改。问题是:如何将闪存(程序代码所在的位置)移动到以后的内存地址。

【问题讨论】:

标签: memory linker stm32 gnu linker-scripts


【解决方案1】:

How can I change the start address on flash? 如果你有 stmf7 这将适用,遗憾的是你没有从你喜欢的任何地址启动的功能。您有固定数量的选项。

查看数据表,7.1.2 复位:RESET 服务程序向量固定在内存映射中的地址 0x0000_0004。这意味着闪存中的第二个 4 个字节是复位处理程序的地址。

但是,您可以使用 BOOT 引脚更改启动的位置,再次参考 2.4 版的数据表 STM32F4xx 微控制器实现了一种特殊机制,能够从其他存储器(如内部SRAM)。

因此,您唯一的选择是更改用于引导的内存类型。但在你的情况下,它根本没有帮助。

【讨论】:

    【解决方案2】:

    正如 P__J__ 提到的,您不能将整个数据区域移动到地址 0x0800 0000,因为 MCU 期望中断向量从那里开始。当 MCU 从闪存启动时,地址0x0800 0000 映射到地址0x0000 0000。 您可以做的是为向量表的长度创建另一个区域,并根据需要移动部分的其他部分。

    MEMORY
    {
    RAM (xrw)       : ORIGIN = 0x20000000, LENGTH = 128K
    VECTORS (rx)    : ORIGIN = 0x08000000, LENGTH = 0xB8
    FLASH (rx)      : ORIGIN = 0x080000B8, LENGTH = 512K - 128k - 0xB8
    DATA (rwx)      : ORIGIN = 0x08060000, LENGTH = 5120
    }
    
      .isr_vector :
      {
        KEEP(*(.isr_vector))
      } > VECTORS
    
    

    【讨论】:

      【解决方案3】:

      这是不可能的,因为您的 STM32 uC 从闪存启动时从地址 0x8000000 开始。

      问题是:如何将闪存(程序代码所在的位置)移动到以后的内存地址。

      答案:不可能。从 FLASH 启动时,向量表必须从 0x8000000 开始

      【讨论】:

      • 好吧,你可以用 BOOT 选项字节告诉你微控制器哪个地址用于引导。所以你可以在 DTCM RAM 的实际 0x0 启动它,如果另一个设备在那里放置了向量表,它将启动获取堆栈指针并重置处理程序,就像它在 0x08000000 所做的那样
      • 您显然没有阅读数据表 :) 但这里有一个 stackoverflow 问题stackoverflow.com/questions/56896375/…
      • 对不起,stm32f4 没有这个功能。我的错,刚刚检查了数据表
      • @AntonStafeyev 是时候删除那些错误的、不相关的 cmets 了
      猜你喜欢
      • 1970-01-01
      • 2013-10-16
      • 2013-09-09
      • 2018-04-25
      • 2021-08-08
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2022-06-26
      相关资源
      最近更新 更多