/* * Copyright (C) 2017, 2019 Ken Rabold * * 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. */ /** * @addtogroup cpu_riscv_common * @{ * * @file * @brief Common linker directives for the RISC-V CPU * * @author Ken Rabold * * @} */ OUTPUT_ARCH( "riscv" ) ENTRY( _start ) PHDRS { flash PT_LOAD; ram_init PT_LOAD; ram PT_NULL; tls PT_TLS; } SECTIONS { __stack_size = DEFINED(__stack_size) ? __stack_size : 256; /* Populate information about rom size */ _srom = ORIGIN(flash); _erom = ORIGIN(flash) + LENGTH(flash); .init : { KEEP (*(SORT_NONE(.init))) } >flash AT>flash :flash .text : { *(.text.unlikely .text.unlikely.*) *(.text.startup .text.startup.*) *(.text .text.*) *(.gnu.linkonce.t.*) } >flash AT>flash :flash .fini : { KEEP (*(SORT_NONE(.fini))) } >flash AT>flash :flash PROVIDE (__etext = .); PROVIDE (_etext = .); PROVIDE (etext = .); .rodata : { *(.rdata) *(.rodata .rodata.*) *(.gnu.linkonce.r.*) KEEP (*(SORT(.roxfa.*))) } >flash AT>flash :flash . = ALIGN(4); .preinit_array : { PROVIDE_HIDDEN (__preinit_array_start = .); KEEP (*(.preinit_array)) PROVIDE_HIDDEN (__preinit_array_end = .); } >flash AT>flash :flash .init_array : { PROVIDE_HIDDEN (__init_array_start = .); KEEP (*(SORT_BY_INIT_PRIORITY(.init_array.*) SORT_BY_INIT_PRIORITY(.ctors.*))) KEEP (*(.init_array EXCLUDE_FILE (*crtbegin.o *crtbegin?.o *crtend.o *crtend?.o ) .ctors)) PROVIDE_HIDDEN (__init_array_end = .); } >flash AT>flash :flash .fini_array : { PROVIDE_HIDDEN (__fini_array_start = .); KEEP (*(SORT_BY_INIT_PRIORITY(.fini_array.*) SORT_BY_INIT_PRIORITY(.dtors.*))) KEEP (*(.fini_array EXCLUDE_FILE (*crtbegin.o *crtbegin?.o *crtend.o *crtend?.o ) .dtors)) PROVIDE_HIDDEN (__fini_array_end = .); } >flash AT>flash :flash .ctors : { /* gcc uses crtbegin.o to find the start of the constructors, so we make sure it is first. Because this is a wildcard, it doesn't matter if the user does not actually link against crtbegin.o; the linker won't look for a file to match a wildcard. The wildcard also means that it doesn't matter which directory crtbegin.o is in. */ KEEP (*crtbegin.o(.ctors)) KEEP (*crtbegin?.o(.ctors)) /* We don't want to include the .ctor section from the crtend.o file until after the sorted ctors. The .ctor section from the crtend file contains the end of ctors marker and it must be last */ KEEP (*(EXCLUDE_FILE (*crtend.o *crtend?.o ) .ctors)) KEEP (*(SORT(.ctors.*))) KEEP (*(.ctors)) } >flash AT>flash :flash .dtors : { KEEP (*crtbegin.o(.dtors)) KEEP (*crtbegin?.o(.dtors)) KEEP (*(EXCLUDE_FILE (*crtend.o *crtend?.o ) .dtors)) KEEP (*(SORT(.dtors.*))) KEEP (*(.dtors)) } >flash AT>flash :flash /* * 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 = .; } >flash AT>flash :tls __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 = .; } >flash : tls __tls_size = __tbss_end - __tdata_start; __tbss_size = __tls_size - __tdata_size; .lalign : { . = ALIGN(4); PROVIDE( _data_lma = . ); } >flash AT>flash :flash .dalign : { . = ALIGN(4); PROVIDE( _data = . ); } >ram AT>flash :ram_init .data : { *(.ramfunc .ramfunc.*) *(.data .data.*) *(.gnu.linkonce.d.*) . = ALIGN(8); PROVIDE( __global_pointer$ = . + 0x800 ); *(.sdata .sdata.*) *(.gnu.linkonce.s.*) . = ALIGN(8); *(.srodata.cst16) *(.srodata.cst8) *(.srodata.cst4) *(.srodata.cst2) *(.srodata .srodata.*) KEEP (*(SORT(.xfa.*))) } >ram AT>flash :ram_init . = ALIGN(4); PROVIDE( _edata = . ); PROVIDE( edata = . ); PROVIDE( _fbss = . ); PROVIDE( __bss_start = . ); .bss : { *(.sbss*) *(.gnu.linkonce.sb.*) *(.bss .bss.*) *(.gnu.linkonce.b.*) *(COMMON) . = ALIGN(4); } >ram AT>ram :ram PROVIDE( __bss_end = . ); .noinit (NOLOAD) : { __noinit_begin__ = .; *(.noinit .noinit.*) . = ALIGN(4) ; __noinit_end__ = .; } >ram AT>ram :ram . = ALIGN(8); PROVIDE( _end = . ); PROVIDE( end = . ); PROVIDE( _sheap = . ); .stack ORIGIN(ram) + LENGTH(ram) - __stack_size : { PROVIDE( _eheap = . ); . = __stack_size; PROVIDE( _sp = . ); } >ram AT>ram :ram .flash_writable (NOLOAD) : { KEEP(*(SORT(.flash_writable.*))) } > flash .end_fw (NOLOAD) : ALIGN(4) { _end_fw = . ; } > flash }