-
Notifications
You must be signed in to change notification settings - Fork 111
/
n64.ld
161 lines (136 loc) · 3.99 KB
/
n64.ld
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
/* ========================================================================
*
* n64.ld -- Libdragon linker script
*
* Originally based on n64ld.x, developed by Frank Somers <[email protected]>,
* with modifications by hcs ([email protected]),
* released under the Unlicense.
* ========================================================================
*/
OUTPUT_FORMAT ("elf32-bigmips", "elf32-bigmips", "elf32-littlemips")
OUTPUT_ARCH (mips)
EXTERN (_start)
ENTRY (_start)
PHDRS
{
irq PT_LOAD AT ( 0x80000000 );
main PT_LOAD AT ( 0x80000400 );
}
SECTIONS {
.intvectors 0x80000000 : {
. = ALIGN(32);
KEEP(*(.intvectors))
__intvectors_end = .;
} :irq
/* The text section carries the app code and its relocation addr is
* the first byte of the cart domain in cached, unmapped memory
*/
.text 0x80000400 : {
__text_start = .;
*(.boot)
. = ALIGN(16);
*(.text)
*(.text.*)
KEEP(*(keep.text.*))
*(.init)
*(.fini)
*(.gnu.linkonce.t.*)
. = ALIGN(16);
__text_end = .;
} :main
.eh_frame_hdr : { *(.eh_frame_hdr) }
.eh_frame : {
__EH_FRAME_BEGIN__ = .; /* Define symbol for accessing eh_frame section */
KEEP (*(.eh_frame))
}
.gcc_except_table : { *(.gcc_except_table*) }
.jcr : { KEEP (*(.jcr)) }
.rodata : {
*(.rdata)
*(.rodata)
*(.rodata.*)
KEEP(*(keep.rodata.*))
*(.gnu.linkonce.r.*)
. = ALIGN(8);
}
/* This is important to keep __CTOR_END__ consistent */
. = ALIGN(4);
.ctors : {
__CTOR_LIST__ = .;
/* Actually we should place the count here like;
LONG((__CTOR_END__ - __CTOR_LIST__) / 4 - 2)
but g++ seem to already do this (though incorrectly placing an 0xFFFFFFFF)
and that breaks things if we also add it. Ideally this should be parsed
by the code but we already know the end as well.
The new build system links everything via g++ by default and should use
__wrap___do_global_ctors and enables it via the --wrap linker option.
When linking with ld, we should use __do_global_ctors, which is the default
if you don't provide the --wrap option. */
KEEP(*(.ctors))
/* Similarly we should have a;
LONG(0)
here, and g++ seem to insert it at __CTOR_END__ */
__CTOR_END__ = .;
}
. = ALIGN(8);
/* Data section has relocation address at start of RAM in cached,
* unmapped memory, but is loaded just at the end of the text segment,
* and must be copied to the correct location at startup
* Gather all initialised data together. The memory layout
* will place the global initialised data at the lowest addrs.
* The lit8, lit4, sdata and sbss sections have to be placed
* together in that order from low to high addrs with the _gp symbol
* positioned (aligned) at the start of the sdata section.
* We then finish off with the standard bss section
*/
.data : {
__data_start = .;
*(.data)
*(.data.*)
KEEP(*(keep.data.*))
*(.gnu.linkonce.d.*)
. = ALIGN(8);
}
/* Small data START */
.sdata : {
_gp = . + 0x8000;
*(.sdata)
*(.sdata.*)
*(.gnu.linkonce.s.*)
. = ALIGN(8);
}
.lit8 : {
*(.lit8)
. = ALIGN(8);
}
.lit4 : {
*(.lit4)
. = ALIGN(8);
}
. = ALIGN(8);
__data_end = .;
/* Here the ROM is finished. The rest is just in RAM */
. = ALIGN(8);
__rom_end = .;
.sbss : {
__bss_start = .;
*(.sbss)
*(.sbss.*)
*(.gnu.linkonce.sb.*)
*(.scommon)
*(.scommon.*)
}
/* Small data END */
. = ALIGN(8);
.bss : {
*(.bss)
*(.bss*)
*(.gnu.linkonce.b.*)
*(COMMON)
. = ALIGN(8);
__bss_end = .;
}
. = ALIGN(8);
/* Deprecated */
end = .;
}