Memory GNU链接器脚本-将闪存移动到新区域
我正在尝试使用STM32F446ZE微控制器的链接器脚本移动内存部分。我最初的设置包括:Memory GNU链接器脚本-将闪存移动到新区域,memory,linker,stm32,gnu,linker-scripts,Memory,Linker,Stm32,Gnu,Linker Scripts,我正在尝试使用STM32F446ZE微控制器的链接器脚本移动内存部分。我最初的设置包括: MEMORY { RAM (xrw) : ORIGIN = 0x20000000, LENGTH = 128K FLASH (rx) : ORIGIN = 0x8000000, LENGTH = 512K - 128k DATA (rwx) : ORIGIN = 0x08060000, LENGTH = 5120 } SECTIONS { .user_data : {
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
我要做的是将数据移动到0x08000000(闪存当前正在启动的位置)并在0x08040000(数据之后)启动闪存。我可以很容易地在内存区更改它,但我的程序无法启动。我相信SECTIONS块中的一些代码可能需要更改,但我不确定如何更改。问题是:如何将闪存(程序代码所在的位置)移动到以后的内存地址。从闪存引导时,STM32 uC从地址0x8000000开始,这是不可能的 问题是:如何将闪存(程序代码所在的位置)移动到以后的内存地址
答案是:这是不可能的。从闪存引导时,矢量表必须从0x8000000开始。这是不可能的,因为从闪存引导时,STM32 uC从地址0x8000000开始 问题是:如何将闪存(程序代码所在的位置)移动到以后的内存地址
答案是:这是不可能的。从闪存引导时,向量表必须从0x8000000开始,正如p_uuj_u所述,您不能将整个数据区域移动到地址
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
正如p_uuj_uu所提到的,您不能将整个数据区域移动到地址
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
如果您有stmf7,这将适用,遗憾的是您没有从任何您喜欢的地址启动的功能。你有固定数量的选择
查看数据表7.1.2重置:重置服务例程向量固定在内存映射中的地址0x0000_0004。这意味着闪存中的第二个4字节是重置处理程序的地址
但是,您可以使用boot pin更改引导位置,再次查阅2.4中的数据表STM32F4xx微控制器实现一种特殊机制,以便能够从其他内存(如内部SRAM)引导
所以,您唯一的选择是更改用于引导的内存类型。但对你来说,这根本没有帮助。如果你有stmf7,这将适用,遗憾的是,你没有从任何你喜欢的地址启动的功能。你有固定数量的选择
查看数据表7.1.2重置:重置服务例程向量固定在内存映射中的地址0x0000_0004。这意味着闪存中的第二个4字节是重置处理程序的地址
但是,您可以使用boot pin更改引导位置,再次查阅2.4中的数据表STM32F4xx微控制器实现一种特殊机制,以便能够从其他内存(如内部SRAM)引导
所以,您唯一的选择是更改用于引导的内存类型。但对你来说,这根本没用。你可以用启动选项字节告诉微控制器哪个地址用于启动。因此,您可以在实际0x0(即DTCM RAM)处启动它,如果另一个设备在那里放置了向量表,它将启动获取堆栈指针并重置处理程序,就像在0x08000000时一样。您显然没有阅读数据表:)但是这里有一个堆栈溢出问题哦,对不起,stm32f4没有此功能。我的错,只是检查了一下datasheet@AntonStafeyev现在是时候删除那些错误的、不相关的注释了。你可以用引导选项字节告诉微控制器哪个地址用于引导。因此,您可以在实际0x0(即DTCM RAM)处启动它,如果另一个设备在那里放置了向量表,它将启动获取堆栈指针并重置处理程序,就像在0x08000000时一样。您显然没有阅读数据表:)但是这里有一个堆栈溢出问题哦,对不起,stm32f4没有此功能。我的错,只是检查了一下datasheet@AntonStafeyev是时候删除那些错误的、不相关的评论了吗?这是否回答了你的问题?这回答了你的问题吗?