Mach-O: add objdump -P function_starts to display function starts.
[binutils-gdb.git] / bfd / elf32-moxie.c
blob5a4fd85fad768436bc116d1f0936699a514ad810
1 /* moxie-specific support for 32-bit ELF.
2 Copyright (C) 2009-2014 Free Software Foundation, Inc.
4 Copied from elf32-fr30.c which is..
5 Copyright (C) 1998-2014 Free Software Foundation, Inc.
7 This file is part of BFD, the Binary File Descriptor library.
9 This program is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 3 of the License, or
12 (at your option) any later version.
14 This program is distributed in the hope that it will be useful,
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 GNU General Public License for more details.
19 You should have received a copy of the GNU General Public License
20 along with this program; if not, write to the Free Software
21 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
22 MA 02110-1301, USA. */
24 #include "sysdep.h"
25 #include "bfd.h"
26 #include "libbfd.h"
27 #include "elf-bfd.h"
28 #include "elf/moxie.h"
30 /* Forward declarations. */
32 static reloc_howto_type moxie_elf_howto_table [] =
34 /* This reloc does nothing. */
35 HOWTO (R_MOXIE_NONE, /* type */
36 0, /* rightshift */
37 2, /* size (0 = byte, 1 = short, 2 = long) */
38 32, /* bitsize */
39 FALSE, /* pc_relative */
40 0, /* bitpos */
41 complain_overflow_bitfield, /* complain_on_overflow */
42 bfd_elf_generic_reloc, /* special_function */
43 "R_MOXIE_NONE", /* name */
44 FALSE, /* partial_inplace */
45 0, /* src_mask */
46 0, /* dst_mask */
47 FALSE), /* pcrel_offset */
49 /* A 32 bit absolute relocation. */
50 HOWTO (R_MOXIE_32, /* type */
51 0, /* rightshift */
52 2, /* size (0 = byte, 1 = short, 2 = long) */
53 32, /* bitsize */
54 FALSE, /* pc_relative */
55 0, /* bitpos */
56 complain_overflow_bitfield, /* complain_on_overflow */
57 bfd_elf_generic_reloc, /* special_function */
58 "R_MOXIE_32", /* name */
59 FALSE, /* partial_inplace */
60 0x00000000, /* src_mask */
61 0xffffffff, /* dst_mask */
62 FALSE), /* pcrel_offset */
64 /* A 10 bit PC-relative relocation. */
65 HOWTO (R_MOXIE_PCREL10, /* type. */
66 1, /* rightshift. */
67 1, /* size (0 = byte, 1 = short, 2 = long). */
68 10, /* bitsize. */
69 TRUE, /* pc_relative. */
70 0, /* bitpos. */
71 complain_overflow_signed, /* complain_on_overflow. */
72 bfd_elf_generic_reloc, /* special_function. */
73 "R_MOXIE_PCREL10", /* name. */
74 FALSE, /* partial_inplace. */
75 0, /* src_mask. */
76 0x000003FF, /* dst_mask. */
77 TRUE), /* pcrel_offset. */
80 /* Map BFD reloc types to MOXIE ELF reloc types. */
82 struct moxie_reloc_map
84 bfd_reloc_code_real_type bfd_reloc_val;
85 unsigned int moxie_reloc_val;
88 static const struct moxie_reloc_map moxie_reloc_map [] =
90 { BFD_RELOC_NONE, R_MOXIE_NONE },
91 { BFD_RELOC_32, R_MOXIE_32 },
92 { BFD_RELOC_MOXIE_10_PCREL, R_MOXIE_PCREL10 },
95 static reloc_howto_type *
96 moxie_reloc_type_lookup (bfd *abfd ATTRIBUTE_UNUSED,
97 bfd_reloc_code_real_type code)
99 unsigned int i;
101 for (i = sizeof (moxie_reloc_map) / sizeof (moxie_reloc_map[0]);
102 --i;)
103 if (moxie_reloc_map [i].bfd_reloc_val == code)
104 return & moxie_elf_howto_table [moxie_reloc_map[i].moxie_reloc_val];
106 return NULL;
109 static reloc_howto_type *
110 moxie_reloc_name_lookup (bfd *abfd ATTRIBUTE_UNUSED, const char *r_name)
112 unsigned int i;
114 for (i = 0;
115 i < sizeof (moxie_elf_howto_table) / sizeof (moxie_elf_howto_table[0]);
116 i++)
117 if (moxie_elf_howto_table[i].name != NULL
118 && strcasecmp (moxie_elf_howto_table[i].name, r_name) == 0)
119 return &moxie_elf_howto_table[i];
121 return NULL;
124 /* Set the howto pointer for an MOXIE ELF reloc. */
126 static void
127 moxie_info_to_howto_rela (bfd *abfd ATTRIBUTE_UNUSED,
128 arelent *cache_ptr,
129 Elf_Internal_Rela *dst)
131 unsigned int r_type;
133 r_type = ELF32_R_TYPE (dst->r_info);
134 BFD_ASSERT (r_type < (unsigned int) R_MOXIE_max);
135 cache_ptr->howto = & moxie_elf_howto_table [r_type];
138 /* Perform a single relocation. By default we use the standard BFD
139 routines, but a few relocs, we have to do them ourselves. */
141 static bfd_reloc_status_type
142 moxie_final_link_relocate (reloc_howto_type *howto,
143 bfd *input_bfd,
144 asection *input_section,
145 bfd_byte *contents,
146 Elf_Internal_Rela *rel,
147 bfd_vma relocation)
149 bfd_reloc_status_type r = bfd_reloc_ok;
151 switch (howto->type)
153 default:
154 r = _bfd_final_link_relocate (howto, input_bfd, input_section,
155 contents, rel->r_offset,
156 relocation, rel->r_addend);
159 return r;
162 /* Relocate an MOXIE ELF section.
164 The RELOCATE_SECTION function is called by the new ELF backend linker
165 to handle the relocations for a section.
167 The relocs are always passed as Rela structures; if the section
168 actually uses Rel structures, the r_addend field will always be
169 zero.
171 This function is responsible for adjusting the section contents as
172 necessary, and (if using Rela relocs and generating a relocatable
173 output file) adjusting the reloc addend as necessary.
175 This function does not have to worry about setting the reloc
176 address or the reloc symbol index.
178 LOCAL_SYMS is a pointer to the swapped in local symbols.
180 LOCAL_SECTIONS is an array giving the section in the input file
181 corresponding to the st_shndx field of each local symbol.
183 The global hash table entry for the global symbols can be found
184 via elf_sym_hashes (input_bfd).
186 When generating relocatable output, this function must handle
187 STB_LOCAL/STT_SECTION symbols specially. The output symbol is
188 going to be the section symbol corresponding to the output
189 section, which means that the addend must be adjusted
190 accordingly. */
192 static bfd_boolean
193 moxie_elf_relocate_section (bfd *output_bfd,
194 struct bfd_link_info *info,
195 bfd *input_bfd,
196 asection *input_section,
197 bfd_byte *contents,
198 Elf_Internal_Rela *relocs,
199 Elf_Internal_Sym *local_syms,
200 asection **local_sections)
202 Elf_Internal_Shdr *symtab_hdr;
203 struct elf_link_hash_entry **sym_hashes;
204 Elf_Internal_Rela *rel;
205 Elf_Internal_Rela *relend;
207 symtab_hdr = & elf_tdata (input_bfd)->symtab_hdr;
208 sym_hashes = elf_sym_hashes (input_bfd);
209 relend = relocs + input_section->reloc_count;
211 for (rel = relocs; rel < relend; rel ++)
213 reloc_howto_type *howto;
214 unsigned long r_symndx;
215 Elf_Internal_Sym *sym;
216 asection *sec;
217 struct elf_link_hash_entry *h;
218 bfd_vma relocation;
219 bfd_reloc_status_type r;
220 const char *name;
221 int r_type;
223 r_type = ELF32_R_TYPE (rel->r_info);
224 r_symndx = ELF32_R_SYM (rel->r_info);
225 howto = moxie_elf_howto_table + r_type;
226 h = NULL;
227 sym = NULL;
228 sec = NULL;
230 if (r_symndx < symtab_hdr->sh_info)
232 sym = local_syms + r_symndx;
233 sec = local_sections [r_symndx];
234 relocation = _bfd_elf_rela_local_sym (output_bfd, sym, &sec, rel);
236 name = bfd_elf_string_from_elf_section
237 (input_bfd, symtab_hdr->sh_link, sym->st_name);
238 name = (name == NULL) ? bfd_section_name (input_bfd, sec) : name;
240 else
242 bfd_boolean unresolved_reloc, warned, ignored;
244 RELOC_FOR_GLOBAL_SYMBOL (info, input_bfd, input_section, rel,
245 r_symndx, symtab_hdr, sym_hashes,
246 h, sec, relocation,
247 unresolved_reloc, warned, ignored);
249 name = h->root.root.string;
252 if (sec != NULL && discarded_section (sec))
253 RELOC_AGAINST_DISCARDED_SECTION (info, input_bfd, input_section,
254 rel, 1, relend, howto, 0, contents);
256 if (info->relocatable)
257 continue;
259 r = moxie_final_link_relocate (howto, input_bfd, input_section,
260 contents, rel, relocation);
262 if (r != bfd_reloc_ok)
264 const char * msg = NULL;
266 switch (r)
268 case bfd_reloc_overflow:
269 r = info->callbacks->reloc_overflow
270 (info, (h ? &h->root : NULL), name, howto->name,
271 (bfd_vma) 0, input_bfd, input_section, rel->r_offset);
272 break;
274 case bfd_reloc_undefined:
275 r = info->callbacks->undefined_symbol
276 (info, name, input_bfd, input_section, rel->r_offset,
277 TRUE);
278 break;
280 case bfd_reloc_outofrange:
281 msg = _("internal error: out of range error");
282 break;
284 case bfd_reloc_notsupported:
285 msg = _("internal error: unsupported relocation error");
286 break;
288 case bfd_reloc_dangerous:
289 msg = _("internal error: dangerous relocation");
290 break;
292 default:
293 msg = _("internal error: unknown error");
294 break;
297 if (msg)
298 r = info->callbacks->warning
299 (info, msg, name, input_bfd, input_section, rel->r_offset);
301 if (! r)
302 return FALSE;
306 return TRUE;
309 /* Return the section that should be marked against GC for a given
310 relocation. */
312 static asection *
313 moxie_elf_gc_mark_hook (asection *sec,
314 struct bfd_link_info *info,
315 Elf_Internal_Rela *rel,
316 struct elf_link_hash_entry *h,
317 Elf_Internal_Sym *sym)
319 return _bfd_elf_gc_mark_hook (sec, info, rel, h, sym);
322 /* Look through the relocs for a section during the first phase.
323 Since we don't do .gots or .plts, we just need to consider the
324 virtual table relocs for gc. */
326 static bfd_boolean
327 moxie_elf_check_relocs (bfd *abfd,
328 struct bfd_link_info *info,
329 asection *sec,
330 const Elf_Internal_Rela *relocs)
332 Elf_Internal_Shdr *symtab_hdr;
333 struct elf_link_hash_entry **sym_hashes;
334 const Elf_Internal_Rela *rel;
335 const Elf_Internal_Rela *rel_end;
337 if (info->relocatable)
338 return TRUE;
340 symtab_hdr = &elf_tdata (abfd)->symtab_hdr;
341 sym_hashes = elf_sym_hashes (abfd);
343 rel_end = relocs + sec->reloc_count;
344 for (rel = relocs; rel < rel_end; rel++)
346 struct elf_link_hash_entry *h;
347 unsigned long r_symndx;
349 r_symndx = ELF32_R_SYM (rel->r_info);
350 if (r_symndx < symtab_hdr->sh_info)
351 h = NULL;
352 else
354 h = sym_hashes[r_symndx - symtab_hdr->sh_info];
355 while (h->root.type == bfd_link_hash_indirect
356 || h->root.type == bfd_link_hash_warning)
357 h = (struct elf_link_hash_entry *) h->root.u.i.link;
359 /* PR15323, ref flags aren't set for references in the same
360 object. */
361 h->root.non_ir_ref = 1;
365 return TRUE;
368 #define ELF_ARCH bfd_arch_moxie
369 #define ELF_MACHINE_CODE EM_MOXIE
370 #define ELF_MAXPAGESIZE 0x1
372 #define TARGET_BIG_SYM bfd_elf32_bigmoxie_vec
373 #define TARGET_BIG_NAME "elf32-bigmoxie"
374 #define TARGET_LITTLE_SYM bfd_elf32_littlemoxie_vec
375 #define TARGET_LITTLE_NAME "elf32-littlemoxie"
377 #define elf_info_to_howto_rel NULL
378 #define elf_info_to_howto moxie_info_to_howto_rela
379 #define elf_backend_relocate_section moxie_elf_relocate_section
380 #define elf_backend_gc_mark_hook moxie_elf_gc_mark_hook
381 #define elf_backend_check_relocs moxie_elf_check_relocs
383 #define elf_backend_can_gc_sections 1
384 #define elf_backend_rela_normal 1
386 #define bfd_elf32_bfd_reloc_type_lookup moxie_reloc_type_lookup
387 #define bfd_elf32_bfd_reloc_name_lookup moxie_reloc_name_lookup
389 #include "elf32-target.h"