]>
Commit | Line | Data |
---|---|---|
2a9fa50c ILT |
1 | # This shell script emits a C file. -*- C -*- |
2 | # It does some substitutions. | |
3 | cat >e${EMULATION_NAME}.c <<EOF | |
4 | /* This file is is generated by a shell script. DO NOT EDIT! */ | |
5 | ||
6 | /* 32 bit ELF emulation code for ${EMULATION_NAME} | |
5f3548d6 | 7 | Copyright (C) 1991, 1993, 1994, 1995 Free Software Foundation, Inc. |
2a9fa50c ILT |
8 | Written by Steve Chamberlain <[email protected]> |
9 | ELF support by Ian Lance Taylor <[email protected]> | |
10 | ||
11 | This file is part of GLD, the Gnu Linker. | |
12 | ||
13 | This program is free software; you can redistribute it and/or modify | |
14 | it under the terms of the GNU General Public License as published by | |
15 | the Free Software Foundation; either version 2 of the License, or | |
16 | (at your option) any later version. | |
17 | ||
18 | This program is distributed in the hope that it will be useful, | |
19 | but WITHOUT ANY WARRANTY; without even the implied warranty of | |
20 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
21 | GNU General Public License for more details. | |
22 | ||
23 | You should have received a copy of the GNU General Public License | |
24 | along with this program; if not, write to the Free Software | |
25 | Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. */ | |
26 | ||
27 | #define TARGET_IS_${EMULATION_NAME} | |
28 | ||
29 | #include "bfd.h" | |
30 | #include "sysdep.h" | |
5efddb2e ILT |
31 | |
32 | #include <ctype.h> | |
33 | ||
2a9fa50c ILT |
34 | #include "bfdlink.h" |
35 | ||
36 | #include "ld.h" | |
37 | #include "config.h" | |
38 | #include "ldmain.h" | |
39 | #include "ldemul.h" | |
40 | #include "ldfile.h" | |
41 | #include "ldmisc.h" | |
42 | #include "ldexp.h" | |
43 | #include "ldlang.h" | |
5efddb2e | 44 | #include "ldgram.h" |
2a9fa50c ILT |
45 | |
46 | static void gld${EMULATION_NAME}_before_parse PARAMS ((void)); | |
5efddb2e ILT |
47 | static boolean gld${EMULATION_NAME}_open_dynamic_archive |
48 | PARAMS ((const char *, lang_input_statement_type *)); | |
2a9fa50c ILT |
49 | static void gld${EMULATION_NAME}_before_allocation PARAMS ((void)); |
50 | static void gld${EMULATION_NAME}_find_statement_assignment | |
51 | PARAMS ((lang_statement_union_type *)); | |
52 | static void gld${EMULATION_NAME}_find_exp_assignment PARAMS ((etree_type *)); | |
5efddb2e ILT |
53 | static boolean gld${EMULATION_NAME}_place_orphan |
54 | PARAMS ((lang_input_statement_type *, asection *)); | |
55 | static void gld${EMULATION_NAME}_place_section | |
56 | PARAMS ((lang_statement_union_type *)); | |
2a9fa50c ILT |
57 | static char *gld${EMULATION_NAME}_get_script PARAMS ((int *isfile)); |
58 | ||
59 | static void | |
60 | gld${EMULATION_NAME}_before_parse() | |
61 | { | |
62 | ldfile_output_architecture = bfd_arch_${ARCH}; | |
5efddb2e ILT |
63 | config.dynamic_link = ${DYNAMIC_LINK-true}; |
64 | } | |
65 | ||
66 | /* Try to open a dynamic archive. This is where we know that ELF | |
67 | dynamic libraries have an extension of .so. */ | |
68 | ||
69 | static boolean | |
70 | gld${EMULATION_NAME}_open_dynamic_archive (arch, entry) | |
71 | const char *arch; | |
72 | lang_input_statement_type *entry; | |
73 | { | |
74 | const char *filename; | |
75 | ||
76 | filename = entry->filename; | |
77 | ||
78 | if (! ldfile_open_file_search (arch, entry, "lib", ".so")) | |
79 | return false; | |
80 | ||
81 | /* We have found a dynamic object to include in the link. The ELF | |
82 | backend linker will create a DT_NEEDED entry in the .dynamic | |
83 | section naming this file. If this file includes a DT_SONAME | |
84 | entry, it will be used. Otherwise, the ELF linker will just use | |
85 | the name of the file. For an archive found by searching, like | |
86 | this one, the DT_NEEDED entry should consist of just the name of | |
87 | the file, without the path information used to find it. Note | |
88 | that we only need to do this if we have a dynamic object; an | |
89 | archive will never be referenced by a DT_NEEDED entry. | |
90 | ||
91 | FIXME: This approach--using bfd_elf_set_dt_needed_name--is not | |
92 | very pretty. I haven't been able to think of anything that is | |
93 | pretty, though. */ | |
94 | if (bfd_check_format (entry->the_bfd, bfd_object) | |
95 | && (entry->the_bfd->flags & DYNAMIC) != 0) | |
96 | { | |
97 | char *needed_name; | |
98 | ||
99 | ASSERT (entry->is_archive && entry->search_dirs_flag); | |
100 | needed_name = (char *) xmalloc (strlen (filename) | |
101 | + strlen (arch) | |
102 | + sizeof "lib.so"); | |
103 | sprintf (needed_name, "lib%s%s.so", filename, arch); | |
104 | bfd_elf_set_dt_needed_name (entry->the_bfd, needed_name); | |
105 | } | |
106 | ||
107 | return true; | |
2a9fa50c ILT |
108 | } |
109 | ||
110 | /* This is called after the sections have been attached to output | |
111 | sections, but before any sizes or addresses have been set. */ | |
112 | ||
113 | static void | |
114 | gld${EMULATION_NAME}_before_allocation () | |
115 | { | |
7fb9ca5f ILT |
116 | asection *sinterp; |
117 | ||
2a9fa50c ILT |
118 | /* If we are going to make any variable assignments, we need to let |
119 | the ELF backend know about them in case the variables are | |
120 | referred to by dynamic objects. */ | |
121 | lang_for_each_statement (gld${EMULATION_NAME}_find_statement_assignment); | |
122 | ||
123 | /* Let the ELF backend work out the sizes of any sections required | |
124 | by dynamic linking. */ | |
5efddb2e ILT |
125 | if (! bfd_elf32_size_dynamic_sections (output_bfd, |
126 | command_line.soname, | |
127 | command_line.rpath, | |
128 | command_line.export_dynamic, | |
129 | &link_info, | |
7fb9ca5f | 130 | &sinterp)) |
2a9fa50c | 131 | einfo ("%P%F: failed to set dynamic section sizes: %E\n"); |
7fb9ca5f ILT |
132 | |
133 | /* Let the user override the dynamic linker we are using. */ | |
134 | if (command_line.interpreter != NULL | |
135 | && sinterp != NULL) | |
136 | { | |
137 | sinterp->contents = (bfd_byte *) command_line.interpreter; | |
138 | sinterp->_raw_size = strlen (command_line.interpreter) + 1; | |
139 | } | |
5f3548d6 ILT |
140 | |
141 | /* Look for any sections named .gnu.warning. As a GNU extensions, | |
142 | we treat such sections as containing warning messages. We print | |
143 | out the warning message, and then zero out the section size so | |
144 | that it does not get copied into the output file. */ | |
145 | ||
146 | { | |
147 | LANG_FOR_EACH_INPUT_STATEMENT (is) | |
148 | { | |
149 | asection *s; | |
150 | bfd_size_type sz; | |
151 | char *msg; | |
152 | boolean ret; | |
153 | ||
154 | if (is->just_syms_flag) | |
155 | continue; | |
156 | ||
157 | s = bfd_get_section_by_name (is->the_bfd, ".gnu.warning"); | |
158 | if (s == NULL) | |
159 | continue; | |
160 | ||
161 | sz = bfd_section_size (is->the_bfd, s); | |
162 | msg = xmalloc ((size_t) sz + 1); | |
163 | if (! bfd_get_section_contents (is->the_bfd, s, msg, (file_ptr) 0, sz)) | |
164 | einfo ("%F%B: Can't read contents of section .gnu.warning: %E\n", | |
165 | is->the_bfd); | |
166 | msg[sz] = '\0'; | |
167 | ret = link_info.callbacks->warning (&link_info, msg); | |
168 | ASSERT (ret); | |
169 | free (msg); | |
170 | ||
171 | /* Clobber the section size, so that we don't waste copying the | |
172 | warning into the output file. */ | |
173 | s->_raw_size = 0; | |
174 | } | |
175 | } | |
2a9fa50c ILT |
176 | } |
177 | ||
178 | /* This is called by the before_allocation routine via | |
179 | lang_for_each_statement. It locates any assignment statements, and | |
180 | tells the ELF backend about them, in case they are assignments to | |
181 | symbols which are referred to by dynamic objects. */ | |
182 | ||
183 | static void | |
184 | gld${EMULATION_NAME}_find_statement_assignment (s) | |
185 | lang_statement_union_type *s; | |
186 | { | |
187 | if (s->header.type == lang_assignment_statement_enum) | |
188 | gld${EMULATION_NAME}_find_exp_assignment (s->assignment_statement.exp); | |
189 | } | |
190 | ||
191 | /* Look through an expression for an assignment statement. */ | |
192 | ||
193 | static void | |
194 | gld${EMULATION_NAME}_find_exp_assignment (exp) | |
195 | etree_type *exp; | |
196 | { | |
197 | switch (exp->type.node_class) | |
198 | { | |
199 | case etree_assign: | |
200 | if (strcmp (exp->assign.dst, ".") != 0) | |
201 | { | |
202 | if (! bfd_elf32_record_link_assignment (output_bfd, &link_info, | |
203 | exp->assign.dst)) | |
204 | einfo ("%P%F: failed to record assignment to %s: %E\n", | |
205 | exp->assign.dst); | |
206 | } | |
207 | gld${EMULATION_NAME}_find_exp_assignment (exp->assign.src); | |
208 | break; | |
209 | ||
210 | case etree_binary: | |
211 | gld${EMULATION_NAME}_find_exp_assignment (exp->binary.lhs); | |
212 | gld${EMULATION_NAME}_find_exp_assignment (exp->binary.rhs); | |
213 | break; | |
214 | ||
215 | case etree_trinary: | |
216 | gld${EMULATION_NAME}_find_exp_assignment (exp->trinary.lhs); | |
217 | gld${EMULATION_NAME}_find_exp_assignment (exp->trinary.lhs); | |
218 | gld${EMULATION_NAME}_find_exp_assignment (exp->trinary.rhs); | |
219 | break; | |
220 | ||
221 | case etree_unary: | |
222 | gld${EMULATION_NAME}_find_exp_assignment (exp->unary.child); | |
223 | break; | |
224 | ||
225 | default: | |
226 | break; | |
227 | } | |
228 | } | |
229 | ||
5efddb2e ILT |
230 | /* Place an orphan section. We use this to put random SHF_ALLOC |
231 | sections in the right segment. */ | |
232 | ||
233 | static asection *hold_section; | |
234 | static lang_output_section_statement_type *hold_use; | |
235 | static lang_output_section_statement_type *hold_text; | |
236 | static lang_output_section_statement_type *hold_data; | |
237 | static lang_output_section_statement_type *hold_bss; | |
238 | ||
239 | /*ARGSUSED*/ | |
240 | static boolean | |
241 | gld${EMULATION_NAME}_place_orphan (file, s) | |
242 | lang_input_statement_type *file; | |
243 | asection *s; | |
244 | { | |
245 | lang_output_section_statement_type *place; | |
246 | asection *snew, **pps; | |
247 | lang_statement_list_type *old; | |
248 | lang_statement_list_type add; | |
249 | etree_type *address; | |
250 | const char *secname, *ps; | |
251 | lang_output_section_statement_type *os; | |
252 | ||
253 | if ((s->flags & SEC_ALLOC) == 0) | |
254 | return false; | |
255 | ||
256 | /* Look through the script to see where to place this section. */ | |
257 | hold_section = s; | |
258 | hold_use = NULL; | |
259 | lang_for_each_statement (gld${EMULATION_NAME}_place_section); | |
260 | ||
261 | if (hold_use != NULL) | |
262 | { | |
263 | /* We have already placed a section with this name. */ | |
264 | wild_doit (&hold_use->children, s, hold_use, file); | |
265 | return true; | |
266 | } | |
267 | ||
268 | /* Decide which segment the section should go in based on the | |
269 | section flags. */ | |
270 | place = NULL; | |
271 | if ((s->flags & SEC_HAS_CONTENTS) == 0 | |
272 | && hold_bss != NULL) | |
273 | place = hold_bss; | |
274 | else if ((s->flags & SEC_READONLY) == 0 | |
275 | && hold_data != NULL) | |
276 | place = hold_data; | |
277 | else if ((s->flags & SEC_READONLY) != 0 | |
278 | && hold_text != NULL) | |
279 | place = hold_text; | |
280 | if (place == NULL) | |
281 | return false; | |
282 | ||
283 | secname = bfd_get_section_name (s->owner, s); | |
284 | ||
5f3548d6 ILT |
285 | /* When generating an object which is to be dynamically linked, we |
286 | do not support orphaned reloc sections. This is because all the | |
287 | reloc sections must be contiguous in order to generate correct | |
288 | DT_REL entries. When this case arises, you can just add the | |
289 | appropriate reloc sections to the linker script. Note that the | |
290 | .rel.plt section must always be the last reloc section. FIXME: | |
291 | This should simply be handled correctly here. */ | |
292 | ASSERT (strncmp (secname, ".rel", 4) != 0 | |
293 | || bfd_get_section_by_name (output_bfd, ".dynamic") == NULL); | |
294 | ||
5efddb2e ILT |
295 | /* Create the section in the output file, and put it in the right |
296 | place. This shuffling to make the output file look neater, and | |
297 | also means that the BFD backend does not have to sort the | |
298 | sections in order by address. */ | |
299 | snew = bfd_make_section (output_bfd, secname); | |
300 | if (snew == NULL) | |
301 | einfo ("%P%F: output format %s cannot represent section called %s\n", | |
302 | output_bfd->xvec->name, secname); | |
303 | for (pps = &output_bfd->sections; *pps != snew; pps = &(*pps)->next) | |
304 | ; | |
305 | *pps = snew->next; | |
306 | snew->next = place->bfd_section->next; | |
307 | place->bfd_section->next = snew; | |
308 | ||
309 | /* Start building a list of statements for this section. */ | |
310 | old = stat_ptr; | |
311 | stat_ptr = &add; | |
312 | lang_list_init (stat_ptr); | |
313 | ||
314 | /* If the name of the section is representable in C, then create | |
315 | symbols to mark the start and the end of the section. */ | |
316 | for (ps = secname; *ps != '\0'; ps++) | |
317 | if (! isalnum (*ps) && *ps != '_') | |
318 | break; | |
319 | if (*ps == '\0' && config.build_constructors) | |
320 | { | |
321 | char *symname; | |
322 | ||
323 | symname = (char *) xmalloc (ps - secname + sizeof "__start_"); | |
324 | sprintf (symname, "__start_%s", secname); | |
325 | lang_add_assignment (exp_assop ('=', symname, | |
326 | exp_nameop (NAME, "."))); | |
327 | } | |
328 | ||
329 | if (! link_info.relocateable) | |
330 | address = NULL; | |
331 | else | |
332 | address = exp_intop ((bfd_vma) 0); | |
333 | ||
334 | lang_enter_output_section_statement (secname, address, 0, | |
335 | (bfd_vma) 0, | |
336 | (etree_type *) NULL, | |
337 | (etree_type *) NULL, | |
338 | (etree_type *) NULL); | |
339 | ||
340 | os = lang_output_section_statement_lookup (secname); | |
341 | wild_doit (&os->children, s, os, file); | |
342 | ||
343 | lang_leave_output_section_statement ((bfd_vma) 0, "*default*"); | |
344 | stat_ptr = &add; | |
345 | ||
346 | if (*ps == '\0' && config.build_constructors) | |
347 | { | |
348 | char *symname; | |
349 | ||
350 | symname = (char *) xmalloc (ps - secname + sizeof "__stop_"); | |
351 | sprintf (symname, "__stop_%s", secname); | |
352 | lang_add_assignment (exp_assop ('=', symname, | |
353 | exp_nameop (NAME, "."))); | |
354 | } | |
355 | ||
356 | /* Now stick the new statement list right after PLACE. */ | |
357 | *add.tail = place->header.next; | |
358 | place->header.next = add.head; | |
359 | ||
360 | stat_ptr = old; | |
361 | ||
362 | return true; | |
363 | } | |
364 | ||
365 | static void | |
366 | gld${EMULATION_NAME}_place_section (s) | |
367 | lang_statement_union_type *s; | |
368 | { | |
369 | lang_output_section_statement_type *os; | |
370 | ||
371 | if (s->header.type != lang_output_section_statement_enum) | |
372 | return; | |
373 | ||
374 | os = &s->output_section_statement; | |
375 | ||
376 | if (strcmp (os->name, hold_section->name) == 0) | |
377 | hold_use = os; | |
378 | ||
379 | if (strcmp (os->name, ".text") == 0) | |
380 | hold_text = os; | |
381 | else if (strcmp (os->name, ".data") == 0) | |
382 | hold_data = os; | |
383 | else if (strcmp (os->name, ".bss") == 0) | |
384 | hold_bss = os; | |
385 | } | |
386 | ||
2a9fa50c ILT |
387 | static char * |
388 | gld${EMULATION_NAME}_get_script(isfile) | |
389 | int *isfile; | |
390 | EOF | |
391 | ||
392 | if test -n "$COMPILE_IN" | |
393 | then | |
394 | # Scripts compiled in. | |
395 | ||
396 | # sed commands to quote an ld script as a C string. | |
397 | sc='s/["\\]/\\&/g | |
398 | s/$/\\n\\/ | |
399 | 1s/^/"/ | |
400 | $s/$/n"/ | |
401 | ' | |
402 | ||
403 | cat >>e${EMULATION_NAME}.c <<EOF | |
404 | { | |
405 | *isfile = 0; | |
406 | ||
407 | if (link_info.relocateable == true && config.build_constructors == true) | |
408 | return `sed "$sc" ldscripts/${EMULATION_NAME}.xu`; | |
409 | else if (link_info.relocateable == true) | |
410 | return `sed "$sc" ldscripts/${EMULATION_NAME}.xr`; | |
411 | else if (!config.text_read_only) | |
412 | return `sed "$sc" ldscripts/${EMULATION_NAME}.xbn`; | |
413 | else if (!config.magic_demand_paged) | |
414 | return `sed "$sc" ldscripts/${EMULATION_NAME}.xn`; | |
5efddb2e ILT |
415 | else if (link_info.shared) |
416 | return `sed "$sc" ldscripts/${EMULATION_NAME}.xs`; | |
2a9fa50c ILT |
417 | else |
418 | return `sed "$sc" ldscripts/${EMULATION_NAME}.x`; | |
419 | } | |
420 | EOF | |
421 | ||
422 | else | |
423 | # Scripts read from the filesystem. | |
424 | ||
425 | cat >>e${EMULATION_NAME}.c <<EOF | |
426 | { | |
427 | *isfile = 1; | |
428 | ||
429 | if (link_info.relocateable == true && config.build_constructors == true) | |
430 | return "ldscripts/${EMULATION_NAME}.xu"; | |
431 | else if (link_info.relocateable == true) | |
432 | return "ldscripts/${EMULATION_NAME}.xr"; | |
433 | else if (!config.text_read_only) | |
434 | return "ldscripts/${EMULATION_NAME}.xbn"; | |
435 | else if (!config.magic_demand_paged) | |
436 | return "ldscripts/${EMULATION_NAME}.xn"; | |
5efddb2e ILT |
437 | else if (link_info.shared) |
438 | return "ldscripts/${EMULATION_NAME}.xs"; | |
2a9fa50c ILT |
439 | else |
440 | return "ldscripts/${EMULATION_NAME}.x"; | |
441 | } | |
442 | EOF | |
443 | ||
444 | fi | |
445 | ||
446 | cat >>e${EMULATION_NAME}.c <<EOF | |
447 | ||
448 | struct ld_emulation_xfer_struct ld_${EMULATION_NAME}_emulation = | |
449 | { | |
450 | gld${EMULATION_NAME}_before_parse, | |
451 | syslib_default, | |
452 | hll_default, | |
453 | after_parse_default, | |
5f3548d6 | 454 | after_open_default, |
2a9fa50c ILT |
455 | after_allocation_default, |
456 | set_output_arch_default, | |
457 | ldemul_default_target, | |
458 | gld${EMULATION_NAME}_before_allocation, | |
459 | gld${EMULATION_NAME}_get_script, | |
460 | "${EMULATION_NAME}", | |
5efddb2e ILT |
461 | "${OUTPUT_FORMAT}", |
462 | NULL, | |
463 | NULL, | |
464 | gld${EMULATION_NAME}_open_dynamic_archive, | |
465 | gld${EMULATION_NAME}_place_orphan | |
2a9fa50c ILT |
466 | }; |
467 | EOF |