1 /* Renesas / SuperH specific support for Symbian 32-bit ELF files
2 Copyright (C) 2004-2014 Free Software Foundation, Inc.
3 Contributed by Red Hat
4
5 This file is part of BFD, the Binary File Descriptor library.
6
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
11
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
16
17 You should have received a copy of the GNU General Public License
18 along with this program; if not, write to the Free Software
19 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
20 MA 02110-1301, USA. */
21
22
23 /* Stop elf32-sh.c from defining any target vectors. */
24 #define SH_TARGET_ALREADY_DEFINED
25 #define sh_find_elf_flags sh_symbian_find_elf_flags
26 #define sh_elf_get_flags_from_mach sh_symbian_elf_get_flags_from_mach
27 #include "elf32-sh.c"
28
29
30 //#define SYMBIAN_DEBUG 1
31 #define SYMBIAN_DEBUG 0
32
33 #define DIRECTIVE_HEADER "#<SYMEDIT>#\n"
34 #define DIRECTIVE_IMPORT "IMPORT "
35 #define DIRECTIVE_EXPORT "EXPORT "
36 #define DIRECTIVE_AS "AS "
37
38 /* Macro to advance 's' until either it reaches 'e' or the
39 character pointed to by 's' is equal to 'c'. If 'e' is
40 reached and SYMBIAN_DEBUG is enabled then the error message 'm'
41 is displayed. */
42 #define SKIP_UNTIL(s,e,c,m) \
43 do \
44 { \
45 while (s < e && *s != c) \
46 ++ s; \
47 if (s >= e) \
48 { \
49 if (SYMBIAN_DEBUG) \
50 fprintf (stderr, "Corrupt directive: %s\n", m); \
51 result = FALSE; \
52 } \
53 } \
54 while (0); \
55 if (!result) \
56 break;
57
58 /* Like SKIP_UNTIL except there are two terminator characters
59 c1 and c2. */
60 #define SKIP_UNTIL2(s,e,c1,c2,m) \
61 do \
62 { \
63 while (s < e && *s != c1 && *s != c2) \
64 ++ s; \
65 if (s >= e) \
66 { \
67 if (SYMBIAN_DEBUG) \
68 fprintf (stderr, "Corrupt directive: %s\n", m); \
69 result = FALSE; \
70 } \
71 } \
72 while (0); \
73 if (!result) \
74 break;
75
76 /* Macro to advance 's' until either it reaches 'e' or the
77 character pointed to by 's' is not equal to 'c'. If 'e'
78 is reached and SYMBIAN_DEBUG is enabled then the error message
79 'm' is displayed. */
80 #define SKIP_WHILE(s,e,c,m) \
81 do \
82 { \
83 while (s < e && *s == c) \
84 ++ s; \
85 if (s >= e) \
86 { \
87 if (SYMBIAN_DEBUG) \
88 fprintf (stderr, "Corrupt directive: %s\n", m); \
89 result = FALSE; \
90 } \
91 } \
92 while (0); \
93 if (!result) \
94 break;
95
96
97 typedef struct symbol_rename
98 {
99 struct symbol_rename * next;
100 char * current_name;
101 char * new_name;
102 struct elf_link_hash_entry * current_hash;
103 unsigned long new_symndx;
104 }
105 symbol_rename;
106
107 static symbol_rename * rename_list = NULL;
108
109 /* Accumulate a list of symbols to be renamed. */
110
111 static bfd_boolean
sh_symbian_import_as(struct bfd_link_info * info,bfd * abfd,char * current_name,char * new_name)112 sh_symbian_import_as (struct bfd_link_info *info, bfd * abfd,
113 char * current_name, char * new_name)
114 {
115 struct elf_link_hash_entry * new_hash;
116 symbol_rename * node;
117
118 if (SYMBIAN_DEBUG)
119 fprintf (stderr, "IMPORT '%s' AS '%s'\n", current_name, new_name);
120
121 for (node = rename_list; node; node = node->next)
122 if (strcmp (node->current_name, current_name) == 0)
123 {
124 if (strcmp (node->new_name, new_name) == 0)
125 /* Already added to rename list. */
126 return TRUE;
127
128 bfd_set_error (bfd_error_invalid_operation);
129 _bfd_error_handler (_("%B: IMPORT AS directive for %s conceals previous IMPORT AS"),
130 abfd, current_name);
131 return FALSE;
132 }
133
134 if ((node = bfd_malloc (sizeof * node)) == NULL)
135 {
136 if (SYMBIAN_DEBUG)
137 fprintf (stderr, "IMPORT AS: No mem for new rename node\n");
138 return FALSE;
139 }
140
141 if ((node->current_name = bfd_malloc (strlen (current_name) + 1)) == NULL)
142 {
143 if (SYMBIAN_DEBUG)
144 fprintf (stderr, "IMPORT AS: No mem for current name field in rename node\n");
145 free (node);
146 return FALSE;
147 }
148 else
149 strcpy (node->current_name, current_name);
150
151 if ((node->new_name = bfd_malloc (strlen (new_name) + 1)) == NULL)
152 {
153 if (SYMBIAN_DEBUG)
154 fprintf (stderr, "IMPORT AS: No mem for new name field in rename node\n");
155 free (node->current_name);
156 free (node);
157 return FALSE;
158 }
159 else
160 strcpy (node->new_name, new_name);
161
162 node->next = rename_list;
163 node->current_hash = NULL;
164 node->new_symndx = 0;
165 rename_list = node;
166
167 new_hash = elf_link_hash_lookup (elf_hash_table (info), node->new_name, TRUE, FALSE, TRUE);
168 bfd_elf_link_record_dynamic_symbol (info, new_hash);
169 if (new_hash->root.type == bfd_link_hash_new)
170 new_hash->root.type = bfd_link_hash_undefined;
171
172 return TRUE;
173 }
174
175
176 static bfd_boolean
sh_symbian_import(bfd * abfd ATTRIBUTE_UNUSED,char * name)177 sh_symbian_import (bfd * abfd ATTRIBUTE_UNUSED, char * name)
178 {
179 if (SYMBIAN_DEBUG)
180 fprintf (stderr, "IMPORT '%s'\n", name);
181
182 /* XXX: Generate an import somehow ? */
183
184 return TRUE;
185 }
186
187 static bfd_boolean
sh_symbian_export(bfd * abfd ATTRIBUTE_UNUSED,char * name)188 sh_symbian_export (bfd * abfd ATTRIBUTE_UNUSED, char * name)
189 {
190 if (SYMBIAN_DEBUG)
191 fprintf (stderr, "EXPORT '%s'\n", name);
192
193 /* XXX: Generate an export somehow ? */
194
195 return TRUE;
196 }
197
198 /* Process any magic embedded commands in the .directive. section.
199 Returns TRUE upon sucecss, but if it fails it sets bfd_error and
200 returns FALSE. */
201
202 static bfd_boolean
sh_symbian_process_embedded_commands(struct bfd_link_info * info,bfd * abfd,asection * sec,bfd_byte * contents)203 sh_symbian_process_embedded_commands (struct bfd_link_info *info, bfd * abfd,
204 asection * sec, bfd_byte * contents)
205 {
206 char *s;
207 char *e;
208 bfd_boolean result = TRUE;
209 bfd_size_type sz = sec->rawsize ? sec->rawsize : sec->size;
210
211 for (s = (char *) contents, e = s + sz; s < e;)
212 {
213 char * directive = s;
214
215 switch (*s)
216 {
217 /* I want to use "case DIRECTIVE_HEADER [0]:" here but gcc won't let me :-( */
218 case '#':
219 if (strcmp (s, DIRECTIVE_HEADER))
220 result = FALSE;
221 else
222 /* Just ignore the header.
223 XXX: Strictly speaking we ought to check that the header
224 is present and that it is the first thing in the file. */
225 s += strlen (DIRECTIVE_HEADER) + 1;
226 break;
227
228 case 'I':
229 if (! CONST_STRNEQ (s, DIRECTIVE_IMPORT))
230 result = FALSE;
231 else
232 {
233 char * new_name;
234 char * new_name_end;
235 char name_end_char;
236
237 /* Skip the IMPORT directive. */
238 s += strlen (DIRECTIVE_IMPORT);
239
240 new_name = s;
241 /* Find the end of the new name. */
242 while (s < e && *s != ' ' && *s != '\n')
243 ++ s;
244 if (s >= e)
245 {
246 /* We have reached the end of the .directive section
247 without encountering a string terminator. This is
248 allowed for IMPORT directives. */
249 new_name_end = e - 1;
250 name_end_char = * new_name_end;
251 * new_name_end = 0;
252 result = sh_symbian_import (abfd, new_name);
253 * new_name_end = name_end_char;
254 break;
255 }
256
257 /* Remember where the name ends. */
258 new_name_end = s;
259 /* Skip any whitespace before the 'AS'. */
260 SKIP_WHILE (s, e, ' ', "IMPORT: Name just followed by spaces");
261 /* Terminate the new name. (Do this after skiping...) */
262 name_end_char = * new_name_end;
263 * new_name_end = 0;
264
265 /* Check to see if 'AS '... is present. If so we have an
266 IMPORT AS directive, otherwise we have an IMPORT directive. */
267 if (! CONST_STRNEQ (s, DIRECTIVE_AS))
268 {
269 /* Skip the new-line at the end of the name. */
270 if (SYMBIAN_DEBUG && name_end_char != '\n')
271 fprintf (stderr, "IMPORT: No newline at end of directive\n");
272 else
273 s ++;
274
275 result = sh_symbian_import (abfd, new_name);
276
277 /* Skip past the NUL character. */
278 if (* s ++ != 0)
279 {
280 if (SYMBIAN_DEBUG)
281 fprintf (stderr, "IMPORT: No NUL at end of directive\n");
282 }
283 }
284 else
285 {
286 char * current_name;
287 char * current_name_end;
288 char current_name_end_char;
289
290 /* Skip the 'AS '. */
291 s += strlen (DIRECTIVE_AS);
292 /* Skip any white space after the 'AS '. */
293 SKIP_WHILE (s, e, ' ', "IMPORT AS: Nothing after AS");
294 current_name = s;
295 /* Find the end of the current name. */
296 SKIP_UNTIL2 (s, e, ' ', '\n', "IMPORT AS: No newline at the end of the current name");
297 /* Skip (backwards) over spaces at the end of the current name. */
298 current_name_end = s;
299 current_name_end_char = * current_name_end;
300
301 SKIP_WHILE (s, e, ' ', "IMPORT AS: Current name just followed by spaces");
302 /* Skip past the newline character. */
303 if (* s ++ != '\n')
304 if (SYMBIAN_DEBUG)
305 fprintf (stderr, "IMPORT AS: No newline at end of directive\n");
306
307 /* Terminate the current name after having performed the skips. */
308 * current_name_end = 0;
309
310 result = sh_symbian_import_as (info, abfd, current_name, new_name);
311
312 /* The next character should be a NUL. */
313 if (* s != 0)
314 {
315 if (SYMBIAN_DEBUG)
316 fprintf (stderr, "IMPORT AS: Junk at end of directive\n");
317 result = FALSE;
318 }
319 s ++;
320
321 * current_name_end = current_name_end_char;
322 }
323
324 /* Restore the characters we overwrote, since
325 the .directive section will be emitted. */
326 * new_name_end = name_end_char;
327 }
328 break;
329
330 case 'E':
331 if (! CONST_STRNEQ (s, DIRECTIVE_EXPORT))
332 result = FALSE;
333 else
334 {
335 char * name;
336 char * name_end;
337 char name_end_char;
338
339 /* Skip the directive. */
340 s += strlen (DIRECTIVE_EXPORT);
341 name = s;
342 /* Find the end of the name to be exported. */
343 SKIP_UNTIL (s, e, '\n', "EXPORT: no newline at end of directive");
344 /* Skip (backwards) over spaces at end of exported name. */
345 for (name_end = s; name_end[-1] == ' '; name_end --)
346 ;
347 /* name_end now points at the first character after the
348 end of the exported name, so we can termiante it */
349 name_end_char = * name_end;
350 * name_end = 0;
351 /* Skip passed the newline character. */
352 s ++;
353
354 result = sh_symbian_export (abfd, name);
355
356 /* The next character should be a NUL. */
357 if (* s != 0)
358 {
359 if (SYMBIAN_DEBUG)
360 fprintf (stderr, "EXPORT: Junk at end of directive\n");
361 result = FALSE;
362 }
363 s++;
364
365 /* Restore the character we deleted. */
366 * name_end = name_end_char;
367 }
368 break;
369
370 default:
371 result = FALSE;
372 break;
373 }
374
375 if (! result)
376 {
377 if (SYMBIAN_DEBUG)
378 fprintf (stderr, "offset into .directive section: %ld\n",
379 (long) (directive - (char *) contents));
380
381 bfd_set_error (bfd_error_invalid_operation);
382 _bfd_error_handler (_("%B: Unrecognised .directive command: %s"),
383 abfd, directive);
384 break;
385 }
386 }
387
388 return result;
389 }
390
391
392 /* Scan a bfd for a .directive section, and if found process it.
393 Returns TRUE upon success, FALSE otherwise. */
394
395 static bfd_boolean
sh_symbian_process_directives(bfd * abfd,struct bfd_link_info * info)396 sh_symbian_process_directives (bfd *abfd, struct bfd_link_info *info)
397 {
398 bfd_boolean result = FALSE;
399 bfd_byte * contents;
400 asection * sec = bfd_get_section_by_name (abfd, ".directive");
401 bfd_size_type sz;
402
403 if (!sec)
404 return TRUE;
405
406 sz = sec->rawsize ? sec->rawsize : sec->size;
407 contents = bfd_malloc (sz);
408
409 if (!contents)
410 bfd_set_error (bfd_error_no_memory);
411 else
412 {
413 if (bfd_get_section_contents (abfd, sec, contents, 0, sz))
414 result = sh_symbian_process_embedded_commands (info, abfd, sec, contents);
415 free (contents);
416 }
417
418 return result;
419 }
420
421 /* Intercept the normal sh_relocate_section() function
422 and magle the relocs to allow for symbol renaming. */
423
424 static bfd_boolean
sh_symbian_relocate_section(bfd * output_bfd,struct bfd_link_info * info,bfd * input_bfd,asection * input_section,bfd_byte * contents,Elf_Internal_Rela * relocs,Elf_Internal_Sym * local_syms,asection ** local_sections)425 sh_symbian_relocate_section (bfd * output_bfd,
426 struct bfd_link_info * info,
427 bfd * input_bfd,
428 asection * input_section,
429 bfd_byte * contents,
430 Elf_Internal_Rela * relocs,
431 Elf_Internal_Sym * local_syms,
432 asection ** local_sections)
433 {
434 /* When performing a final link we implement the IMPORT AS directives. */
435 if (!info->relocatable)
436 {
437 Elf_Internal_Rela * rel;
438 Elf_Internal_Rela * relend;
439 Elf_Internal_Shdr * symtab_hdr;
440 struct elf_link_hash_entry ** sym_hashes;
441 struct elf_link_hash_entry ** sym_hashes_end;
442 struct elf_link_hash_table * hash_table;
443 symbol_rename * ptr;
444 bfd_size_type num_global_syms;
445 unsigned long num_local_syms;
446
447 BFD_ASSERT (! elf_bad_symtab (input_bfd));
448
449 symtab_hdr = & elf_tdata (input_bfd)->symtab_hdr;
450 hash_table = elf_hash_table (info);
451 num_local_syms = symtab_hdr->sh_info;
452 num_global_syms = symtab_hdr->sh_size / sizeof (Elf32_External_Sym);
453 num_global_syms -= num_local_syms;
454 sym_hashes = elf_sym_hashes (input_bfd);
455 sym_hashes_end = sym_hashes + num_global_syms;
456
457 /* First scan the rename table, caching the hash entry and the new index. */
458 for (ptr = rename_list; ptr; ptr = ptr->next)
459 {
460 struct elf_link_hash_entry * new_hash;
461 struct elf_link_hash_entry ** h;
462
463 ptr->current_hash = elf_link_hash_lookup (hash_table, ptr->current_name, FALSE, FALSE, TRUE);
464
465 if (ptr->current_hash == NULL)
466 {
467 if (SYMBIAN_DEBUG)
468 fprintf (stderr, "IMPORT AS: current symbol '%s' does not exist\n", ptr->current_name);
469 continue;
470 }
471
472 new_hash = elf_link_hash_lookup (hash_table, ptr->new_name,
473 FALSE, FALSE, TRUE);
474 /* If we could not find the symbol then it is a new, undefined symbol.
475 Symbian want this behaviour - ie they want to be able to rename the
476 reference in a reloc from one undefined symbol to another, new and
477 undefined symbol. So we create that symbol here. */
478 if (new_hash == NULL)
479 {
480 struct bfd_link_hash_entry *bh = NULL;
481 bfd_boolean collect = get_elf_backend_data (input_bfd)->collect;
482 if (_bfd_generic_link_add_one_symbol (info, input_bfd,
483 ptr->new_name, BSF_GLOBAL,
484 bfd_und_section_ptr, 0,
485 NULL, FALSE, collect,
486 &bh))
487 {
488 new_hash = (struct elf_link_hash_entry *) bh;
489 new_hash->type = STT_FUNC;
490 new_hash->non_elf = 0;
491
492 if (SYMBIAN_DEBUG)
493 fprintf (stderr, "Created new symbol %s\n", ptr->new_name);
494 }
495 }
496
497 if (new_hash == NULL)
498 {
499 _bfd_error_handler (_("%B: Failed to add renamed symbol %s"),
500 input_bfd, ptr->new_name);
501 continue;
502 }
503
504 /* Convert the new_hash value into a index into the table of symbol hashes. */
505 for (h = sym_hashes; h < sym_hashes_end; h ++)
506 {
507 if (* h == new_hash)
508 {
509 ptr->new_symndx = h - sym_hashes + num_local_syms;
510 if (SYMBIAN_DEBUG)
511 fprintf (stderr, "Converted new hash to index of %ld\n", ptr->new_symndx);
512 break;
513 }
514 }
515 /* If the new symbol is not in the hash table then it must be
516 because it is one of the newly created undefined symbols
517 manufactured above. So we extend the sym has table here to
518 include this extra symbol. */
519 if (h == sym_hashes_end)
520 {
521 struct elf_link_hash_entry ** new_sym_hashes;
522
523 /* This is not very efficient, but it works. */
524 ++ num_global_syms;
525 new_sym_hashes = bfd_alloc (input_bfd, num_global_syms * sizeof * sym_hashes);
526 if (new_sym_hashes == NULL)
527 {
528 if (SYMBIAN_DEBUG)
529 fprintf (stderr, "Out of memory extending hash table\n");
530 continue;
531 }
532 memcpy (new_sym_hashes, sym_hashes, (num_global_syms - 1) * sizeof * sym_hashes);
533 new_sym_hashes[num_global_syms - 1] = new_hash;
534 elf_sym_hashes (input_bfd) = sym_hashes = new_sym_hashes;
535 sym_hashes_end = sym_hashes + num_global_syms;
536 symtab_hdr->sh_size = (num_global_syms + num_local_syms) * sizeof (Elf32_External_Sym);
537
538 ptr->new_symndx = num_global_syms - 1 + num_local_syms;
539
540 if (SYMBIAN_DEBUG)
541 fprintf (stderr, "Extended symbol hash table to insert new symbol as index %ld\n",
542 ptr->new_symndx);
543 }
544 }
545
546 /* Walk the reloc list looking for references to renamed symbols.
547 When we find one, we alter the index in the reloc to point to the new symbol. */
548 for (rel = relocs, relend = relocs + input_section->reloc_count;
549 rel < relend;
550 rel ++)
551 {
552 int r_type;
553 unsigned long r_symndx;
554 struct elf_link_hash_entry * h;
555
556 r_symndx = ELF32_R_SYM (rel->r_info);
557 r_type = ELF32_R_TYPE (rel->r_info);
558
559 /* Ignore unused relocs. */
560 if ((r_type >= (int) R_SH_GNU_VTINHERIT
561 && r_type <= (int) R_SH_LABEL)
562 || r_type == (int) R_SH_NONE
563 || r_type < 0
564 || r_type >= R_SH_max)
565 continue;
566
567 /* Ignore relocs against local symbols. */
568 if (r_symndx < num_local_syms)
569 continue;
570
571 BFD_ASSERT (r_symndx < (num_global_syms + num_local_syms));
572 h = sym_hashes[r_symndx - num_local_syms];
573 BFD_ASSERT (h != NULL);
574
575 while ( h->root.type == bfd_link_hash_indirect
576 || h->root.type == bfd_link_hash_warning)
577 h = (struct elf_link_hash_entry *) h->root.u.i.link;
578
579 /* If the symbol is defined there is no need to rename it.
580 XXX - is this true ? */
581 if ( h->root.type == bfd_link_hash_defined
582 || h->root.type == bfd_link_hash_defweak
583 || h->root.type == bfd_link_hash_undefweak)
584 continue;
585
586 for (ptr = rename_list; ptr; ptr = ptr->next)
587 if (h == ptr->current_hash)
588 {
589 BFD_ASSERT (ptr->new_symndx);
590 if (SYMBIAN_DEBUG)
591 fprintf (stderr, "convert reloc %lx from using index %ld to using index %ld\n",
592 (unsigned long) rel->r_info,
593 (long) ELF32_R_SYM (rel->r_info), ptr->new_symndx);
594 rel->r_info = ELF32_R_INFO (ptr->new_symndx, r_type);
595 break;
596 }
597 }
598 }
599
600 return sh_elf_relocate_section (output_bfd, info, input_bfd, input_section,
601 contents, relocs, local_syms, local_sections);
602 }
603
604 #define TARGET_LITTLE_SYM sh_elf32_symbian_le_vec
605 #define TARGET_LITTLE_NAME "elf32-shl-symbian"
606
607 #undef elf_backend_relocate_section
608 #define elf_backend_relocate_section sh_symbian_relocate_section
609 #undef elf_backend_check_directives
610 #define elf_backend_check_directives sh_symbian_process_directives
611
612 #include "elf32-target.h"
613