| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343 | /* -*- ld-script -*- * * Linker script for MAX80 firmware */#define __ASSEMBLY__#define __LDSCRIPT__#include "sys.h"#include "iodevs.h"#include "iodeva.h"OUTPUT_FORMAT("elf32-littleriscv", "elf32-littleriscv",              "elf32-littleriscv")OUTPUT_ARCH(riscv)ENTRY(___reset)EXTERN(_NULL)EXTERN(_reset)EXTERN(_irq)MEMORY{	SRAM	: org = SRAM_ADDR,  len = SRAM_SIZE	DRAM	: org = SDRAM_ADDR, len = SDRAM_SIZE	DRAM2   : org = SDRAM_ADDR+SDRAM_SIZE, len = SDRAM_SIZE}SECTIONS{	/* Debugging sections */  /* Stabs debugging sections.  */  .stab          0 : { *(.stab) }  .stabstr       0 : { *(.stabstr) }  .stab.excl     0 : { *(.stab.excl) }  .stab.exclstr  0 : { *(.stab.exclstr) }  .stab.index    0 : { *(.stab.index) }  .stab.indexstr 0 : { *(.stab.indexstr) }  .comment       0 : { *(.comment) }  .gnu.build.attributes : { *(.gnu.build.attributes .gnu.build.attributes.*) }  /* DWARF debug sections.     Symbols in the DWARF debugging sections are relative to the beginning     of the section so we begin them at 0.  */  /* DWARF 1.  */  .debug          0 : { *(.debug) }  .line           0 : { *(.line) }  /* GNU DWARF 1 extensions.  */  .debug_srcinfo  0 : { *(.debug_srcinfo) }  .debug_sfnames  0 : { *(.debug_sfnames) }  /* DWARF 1.1 and DWARF 2.  */  .debug_aranges  0 : { *(.debug_aranges) }  .debug_pubnames 0 : { *(.debug_pubnames) }  /* DWARF 2.  */  .debug_info     0 : { *(.debug_info .gnu.linkonce.wi.*) }  .debug_abbrev   0 : { *(.debug_abbrev) }  .debug_line     0 : { *(.debug_line .debug_line.* .debug_line_end) }  .debug_frame    0 : { *(.debug_frame) }  .debug_str      0 : { *(.debug_str) }  .debug_loc      0 : { *(.debug_loc) }  .debug_macinfo  0 : { *(.debug_macinfo) }  /* SGI/MIPS DWARF 2 extensions.  */  .debug_weaknames 0 : { *(.debug_weaknames) }  .debug_funcnames 0 : { *(.debug_funcnames) }  .debug_typenames 0 : { *(.debug_typenames) }  .debug_varnames  0 : { *(.debug_varnames) }  /* DWARF 3.  */  .debug_pubtypes 0 : { *(.debug_pubtypes) }  .debug_ranges   0 : { *(.debug_ranges) }  /* DWARF 5.  */  .debug_addr     0 : { *(.debug_addr) }  .debug_line_str 0 : { *(.debug_line_str) }  .debug_loclists 0 : { *(.debug_loclists) }  .debug_macro    0 : { *(.debug_macro) }  .debug_names    0 : { *(.debug_names) }  .debug_rnglists 0 : { *(.debug_rnglists) }  .debug_str_offsets 0 : { *(.debug_str_offsets) }  .debug_sup      0 : { *(.debug_sup) }  .gnu.attributes 0 : { KEEP (*(.gnu.attributes)) }  .riscv.attributes 0 : { KEEP(*(.riscv.attributes)) }  /DISCARD/ : { *(.note.GNU-stack) *(.gnu_debuglink) *(.gnu.lto_*) }	/*	 * Sections we do not need. This program cannot exit, so	 * fini/destructors are never needed. Exception handling	 * is not supported.	 */	 /DISCARD/	: {		*(.note.GNU-stack)		*(.gnu_debuglink)		*(.gnu.lto_*)		*(.discard)		*(.discard.*)		*(.dtors.*)		*(.dtors)		*(.fini_array)		*(.fini_array.*)		*(.eh_frame_hdr) *(.eh_frame_entry .eh_frame_entry.*)		*(.eh_frame) *(.eh_frame.*)		*(.gcc_except_table .gcc_except_table.*)		*(.gnu_extab*)		*(.exception_ranges*)		*(.text.exit .text.exit.*)		*crtbegin.o(*)		*crt0.o(*)	}	. = 0;	PROVIDE (__executable_start = .);	/*	 * Make sure the output binary starts at address 0	 */	.null 0 : {		PROVIDE(___NULL = .);		KEEP (*(SORT_NONE(.null)))	}	.init.reset _PC_RESET : ALIGN(4) {		PROVIDE (___reset = .);		KEEP (*(SORT_NONE(.init.reset)))	}	.init.irq _PC_IRQ : ALIGN(4) {		PROVIDE (___irq = .);		KEEP (*(SORT_NONE(.init.irq)))	}	/* .rwtext is in the zero page */	.rwtext : ALIGN(4) {		PROVIDE (__rwtext_start = .);		*(.rwtext*)		PROVIDE (__rwtext_end = .);	}	/*	 * Put the short data sections in the zero page.	 * This means the initialized sections aren't contiguous, but	 * all memory is intialized during FPGA load anyway.	 */	. = ALIGN(4);	__SDATA_BEGIN__ = .;	.srodata : {		*(.srodata*)	}	.sdata : {		*(.sdata .sdata.* .gnu.linkonce.s.*)	}	.sdata2 : {		*(.sdata2 .sdata2.* .gnu.linkonce.s2.*)	}	. = ALIGN(32);	__BSS_START__ = .;	PROVIDE (__bss_start = .);	.sbss (NOLOAD) : ALIGN(4) {		*(.dynsbss)		*(.sbss .sbss.* .gnu.linkonce.sb.*)		*(.scommon)	}	.sbss2 (NOLOAD) : {	       *(.sbss2 .sbss2.* .gnu.linkonce.sb2.*)	}	HIDDEN($assert_zero_page = ASSERT((. <= 2048), "zero page overflow"));	.bss (NOLOAD) : {		*(.dynbss)		*(.bss.*hot* .gnu.linkonce.b.*)		*(COMMON)	}	. = ALIGN(32);	__BSS_END__ = .;	__BSS_LEN__ = __BSS_END__ - __BSS_START__;	__global_pointer$ = 0;	PROVIDE(___text = .);	.init : ALIGN(4) {		KEEP (*(SORT_NONE(.init)))	}	.text.hot : ALIGN(4) {		*(.text.startup .text.startup.*)		*(.text.hot .text.hot.*)		KEEP(sbrk.o(.text))		*(.gnu.linkonce.t.*)	}	PROVIDE (__etext = .);	PROVIDE (_etext = .);	. = ALIGN(4);	.str.hot : ALIGN(4) {		*(.rodata*.hot.str*)	}	.rodata.hot         : {		*(.rodata*.hot* .gnu.linkonce.r.*)	}	/* Thread Local Storage sections  */	.tdata          : {		PROVIDE_HIDDEN (__tdata_start = .);		*(.tdata .tdata.* .gnu.linkonce.td.*)	}	.tbss       : { *(.tbss .tbss.* .gnu.linkonce.tb.*) *(.tcommon) }	.preinit_array    : {		PROVIDE_HIDDEN (__preinit_array_start = .);		KEEP (*(.preinit_array))		PROVIDE_HIDDEN (__preinit_array_end = .);	}	/* Are these necessary/supportable? */	  .jcr          : { KEEP (*(.jcr)) }	.data.rel.ro	: {			*(.data.rel.ro.local* .gnu.linkonce.d.rel.ro.local.*)			*(.data.rel.ro .data.rel.ro.* .gnu.linkonce.d.rel.ro.*)	}.data           : {		__DATA_BEGIN__ = .;		*(.data .data.* .gnu.linkonce.d.*)		SORT(CONSTRUCTORS)	}	.data1          : { *(.data1) }	_edata = .;	_end = .;	/* Patched in during FPGA compile, must immediately follow _end */	.datestamp (NOLOAD) : { KEEP(*(.datestamp)) }	HIDDEN($sram_size_assert = ASSERT(. <= STACK_BOTTOM, "SRAM overflow"));	. = STACK_BOTTOM;	.stack (NOLOAD) : {		KEEP (*(.stack))	}	/* Sections in SDRAM */	. = SDRAM_ADDR;	__dram_start = .;	/* Always first in DRAM for remote DMA to find fixed addresses */	.dram.esplink (NOLOAD) : ALIGN(4096) {		__esplink_start = .;		KEEP(*(SORT_NONE(.dram.esplink.head)));		*(.dram.esplink .dram.esplink.*);		. = ALIGN(16);		__esplink_end = .;	} >DRAM	. = ALIGN(4096);	/* Align to a flash page */	__dram_init_start = .;	.dram.abcrom : AT(SRAM_SIZE) ALIGN(4096) {		__abcrom_start = .;		KEEP(*(SORT_NONE(.dram.abcrom*)))		__abcrom_end = .;	} >DRAM	.dram.text : ALIGN(4) {		*(SORT(.text.sorted.*))		*(.text .stub .text.*)	        *(.text.*unlikely*)		*(.dram.text*)	} >DRAM	.dram.str : ALIGN(4) {		__dram_str_start = .;		*(.rodata*.str* .dram.rodata*.str*)		__dram_str_end = .;	} >DRAM	.dram.rodata : ALIGN(4) {		__dram_rodata_start = .;		*(.rodata* .dram.rodata*)		__dram_rodata_end = .;	} >DRAM	.dram.init_array : ALIGN(4) {		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 = .);	} >DRAM	.dram.data : ALIGN(4) {		*(.dram.data* .data*)	} >DRAM	. = ALIGN(8);	__dram_init_end = .;	__dram_init_len = __dram_init_end - __dram_init_start;	/* Keeps ld from getting confused */#if 0	. = . + SDRAM_SIZE;	/* Test program image - overlays */	.dram.test __dram_init_end + SDRAM_SIZE :		   AT(SRAM_SIZE + __dram_init_len) ALIGN(4) {		   KEEP(*(.dram.test*))	} >DRAM2#endif	/* bss can overlay the test program image */	. = __dram_init_end;	__dram_bss_start = .;	.dram.bss (NOLOAD) : ALIGN(8) {		*(.dram.bss* .bss*)	} >DRAM	. = ALIGN(8);	__dram_bss_end = .;	__dram_bss_len = __dram_bss_end - __dram_bss_start;	/* Like BSS except no need to clear on boot */	.dram.noinit (NOLOAD) : ALIGN(8) {		*(.dram.noinit*)	} >DRAM	/* No need to zero the heap */	.heap (NOLOAD) : ALIGN(16) {		*(.heap*)	} >DRAM	__dram_end = .;	/* Dummy section containing I/O device symbols */	. = XDEV_ADDR_BASE;	.iodevs (NOLOAD) : {		. = .;		/* Force output */		KEEP(*(.iodevs))	}	/* Catch missing sections */	. = __dram_end;	__junk_start = .;	.junk : {		*(*)	}	__junk_end = .;	HIDDEN($assert_no_junk = ASSERT(__junk_end == __junk_start, "unknown sections present"));}
 |