mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
fb01e6a3fb
Newer picolibc versions require some additional symbols defined in the linker script to correctly manage alignment constraints on thread local storage. Signed-off-by: Keith Packard <keithp@keithp.com>
258 lines
7.1 KiB
Plaintext
258 lines
7.1 KiB
Plaintext
/*
|
|
* Copyright 2013, Freie Universitaet Berlin (FUB). All rights reserved.
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/* specify the LPC23XX memory areas (see LPC2387 datasheet page 15) */
|
|
MEMORY
|
|
{
|
|
rom (rx) : ORIGIN = 0, LENGTH = 504K /* FLASH ROM 512kByte without ISP bootloader*/
|
|
infomem : ORIGIN = 0x0007D000, LENGTH = 4K /* Last sector in FLASH ROM for config data */
|
|
ram_battery : ORIGIN = 0xE0084000, LENGTH = 2K /* Battery RAM */
|
|
ram (w!rx) : ORIGIN = 0x40000000, LENGTH = 64K /* LOCAL ON-CHIP STATIC RAM */
|
|
ram_usb : ORIGIN = 0x7FD00000, LENGTH = 16K /* USB RAM !!! first 1024 bytes are occupied from GPDMA for MCI */
|
|
ram_ethernet : ORIGIN = 0x7FE00000, LENGTH = 16K /* ethernet RAM */
|
|
}
|
|
|
|
/* now define the output sections */
|
|
SECTIONS
|
|
{
|
|
.text : /* collect all sections that should go into FLASH after startup */
|
|
{
|
|
KEEP(*(.vectors)) /* Exception Vectors and branch table >= 64 bytes */
|
|
. = ALIGN(64);
|
|
KEEP(*(.init))
|
|
KEEP(*(.init0)) /* Start here after reset. */
|
|
*(.text .text.* .gnu.linkonce.t.*)
|
|
*(.glue_7t) *(.glue_7)
|
|
*(.rodata .rodata* .gnu.linkonce.r.*)
|
|
*(.ARM.extab* .gnu.linkonce.armextab.*)
|
|
|
|
/* Support C constructors, and C destructors in both user code
|
|
and the C library. This also provides support for C++ code. */
|
|
. = ALIGN(4);
|
|
KEEP(*(.init))
|
|
. = ALIGN(4);
|
|
__preinit_array_start = .;
|
|
KEEP (*(.preinit_array))
|
|
__preinit_array_end = .;
|
|
|
|
. = ALIGN(4);
|
|
__init_array_start = .;
|
|
KEEP (*(SORT(.init_array.*)))
|
|
KEEP (*(.init_array))
|
|
__init_array_end = .;
|
|
|
|
. = ALIGN(0x4);
|
|
KEEP (*crtbegin.o(.ctors))
|
|
KEEP (*(EXCLUDE_FILE (*crtend.o) .ctors))
|
|
KEEP (*(SORT(.ctors.*)))
|
|
KEEP (*crtend.o(.ctors))
|
|
|
|
. = ALIGN(4);
|
|
KEEP(*(.fini))
|
|
|
|
. = ALIGN(4);
|
|
__fini_array_start = .;
|
|
KEEP (*(.fini_array))
|
|
KEEP (*(SORT(.fini_array.*)))
|
|
__fini_array_end = .;
|
|
|
|
KEEP (*crtbegin.o(.dtors))
|
|
KEEP (*(EXCLUDE_FILE (*crtend.o) .dtors))
|
|
KEEP (*(SORT(.dtors.*)))
|
|
KEEP (*crtend.o(.dtors))
|
|
KEEP (*(SORT(.roxfa.*)))
|
|
|
|
. = ALIGN(4);
|
|
_efixed = .; /* End of text section */
|
|
} > rom
|
|
|
|
/*
|
|
* TLS relocations are offsets relative to the address
|
|
* of the first TLS symbol. That means we just need to
|
|
* allocate them all together so that the TLS region
|
|
* is compact when allocated for each thread.
|
|
*/
|
|
|
|
/*
|
|
* TLS initialization data is loaded into ROM so that
|
|
* each thread can get its values initialized from there
|
|
* at startup
|
|
*/
|
|
.tdata :
|
|
{
|
|
__tdata_start = .;
|
|
*(.tdata .tdata.* .gnu.linkonce.td.*)
|
|
__tdata_end = .;
|
|
} > rom
|
|
__tdata_source = LOADADDR(.tdata);
|
|
__tdata_size = SIZEOF(.tdata);
|
|
|
|
/*
|
|
* TLS zeroed data is relocated as if it immediately followed
|
|
* the tdata values. However, the linker 'magically' erases the
|
|
* memory allocation so that no ROM is consumed by this
|
|
* section
|
|
*/
|
|
.tbss :
|
|
{
|
|
*(.tbss .tbss.* .gnu.linkonce.tb.*)
|
|
*(.tcommon)
|
|
__tbss_end = .;
|
|
} > rom
|
|
__tls_size = __tbss_end - __tdata_start;
|
|
__tbss_size = __tls_size - __tdata_size;
|
|
PROVIDE( __tls_align = MAX(ALIGNOF(.tdata), ALIGNOF(.tbss)) );
|
|
PROVIDE( __arm32_tls_tcb_offset = MAX(8, __tls_align) );
|
|
PROVIDE( __arm64_tls_tcb_offset = MAX(16, __tls_align) );
|
|
|
|
/* .ARM.exidx is sorted, so has to go in its own output section. */
|
|
PROVIDE_HIDDEN (__exidx_start = .);
|
|
.ARM.exidx :
|
|
{
|
|
*(.ARM.exidx* .gnu.linkonce.armexidx.*)
|
|
} > rom
|
|
PROVIDE_HIDDEN (__exidx_end = .);
|
|
|
|
/* exception handling */
|
|
. = ALIGN(4);
|
|
.eh_frame :
|
|
{
|
|
KEEP (*(.eh_frame))
|
|
} > rom
|
|
|
|
. = ALIGN(4);
|
|
_etext = .;
|
|
|
|
.config :
|
|
{
|
|
*(.configmem)
|
|
. = ALIGN(256);
|
|
} > infomem
|
|
|
|
/**************************************************************************
|
|
* RAM
|
|
**************************************************************************/
|
|
|
|
/*
|
|
* Stacks
|
|
*/
|
|
.stack (NOLOAD) :
|
|
{
|
|
PROVIDE(__stack_start = .);
|
|
KEEP (*(.usr_stack))
|
|
__stack_usr_start = .;
|
|
KEEP (*(.und_stack))
|
|
__stack_und_start = .;
|
|
KEEP (*(.fiq_stack))
|
|
__stack_fiq_start = .;
|
|
KEEP (*(.irq_stack))
|
|
__stack_irq_start = .;
|
|
KEEP (*(.abt_stack))
|
|
__stack_abt_start = .;
|
|
KEEP (*(.svc_stack))
|
|
__stack_svc_start = .;
|
|
|
|
PROVIDE(__stack_end = .);
|
|
} > ram
|
|
|
|
.relocate :
|
|
{
|
|
. = ALIGN(4);
|
|
_srelocate = .;
|
|
*(.ramfunc .ramfunc.*);
|
|
*(.data .data.*);
|
|
KEEP (*(.openocd .openocd.*))
|
|
KEEP (*(SORT(.xfa.*)))
|
|
. = ALIGN(4);
|
|
_erelocate = .;
|
|
} > ram AT> rom
|
|
|
|
/* .bss section which is used for uninitialized data */
|
|
.bss (NOLOAD) :
|
|
{
|
|
. = ALIGN(4);
|
|
_sbss = . ;
|
|
_szero = .;
|
|
*(.bss .bss.*)
|
|
*(COMMON)
|
|
. = ALIGN(4);
|
|
_ebss = . ;
|
|
_ezero = .;
|
|
} > ram
|
|
|
|
/*
|
|
* collect all uninitialized sections that go into RAM
|
|
*/
|
|
.noinit (NOLOAD) :
|
|
{
|
|
__noinit_start = .;
|
|
PROVIDE(__fiq_handler = .);
|
|
*(.fiq)
|
|
*(.noinit)
|
|
. = ALIGN(4);
|
|
__noinit_end = .;
|
|
} > ram
|
|
|
|
/* heap section */
|
|
. = ALIGN(4);
|
|
_sheap = . ;
|
|
_eheap = ORIGIN(ram) + LENGTH(ram);
|
|
|
|
/* Populate information about ram size */
|
|
_sram = ORIGIN(ram);
|
|
_eram = ORIGIN(ram) + LENGTH(ram);
|
|
|
|
/* Populate information about rom size */
|
|
_srom = ORIGIN(rom);
|
|
_erom = ORIGIN(rom) + LENGTH(rom);
|
|
|
|
.heap1 ALIGN(4) (NOLOAD) :
|
|
{
|
|
_sheap1 = ORIGIN(ram_ethernet);
|
|
_eheap1 = ORIGIN(ram_ethernet) + LENGTH(ram_ethernet);
|
|
} > ram_ethernet
|
|
|
|
. = ORIGIN(ram_usb);
|
|
.usbdata (NOLOAD) : /* USB RAM section, may be used otherwise if USB is disabled */
|
|
{
|
|
*(.usbdata)
|
|
} > ram_usb
|
|
|
|
.heap2 ALIGN(4) (NOLOAD) :
|
|
{
|
|
_sheap2 = . ;
|
|
_eheap2 = ORIGIN(ram_usb) + LENGTH(ram_usb);
|
|
} > ram_usb
|
|
|
|
_sbackup_data_load = LOADADDR(.backup.data);
|
|
.backup.data : ALIGN(4) {
|
|
_sbackup_data = .;
|
|
*(.backup.data)
|
|
_ebackup_data = .;
|
|
/* Round size so that we can use 4 byte copy in init */
|
|
. = ALIGN(4);
|
|
} > ram_battery AT> rom
|
|
|
|
.backup.bss (NOLOAD) : ALIGN(4) {
|
|
_sbackup_bss = .;
|
|
*(.backup.bss)
|
|
_ebackup_bss = .;
|
|
/* Round size so that we can use 4 byte copy in init */
|
|
. = ALIGN(4);
|
|
} > ram_battery
|
|
|
|
.heap3 (NOLOAD) : ALIGN(4) {
|
|
_sheap3 = . ;
|
|
_eheap3 = ORIGIN(ram_battery) + LENGTH(ram_battery);
|
|
} > ram_battery
|
|
|
|
.end_fw (NOLOAD) : ALIGN(4) {
|
|
_end_fw = . ;
|
|
} > rom
|
|
}
|