1 /* MIPS-specific support for 64-bit ELF
2 Copyright 1996, 1997, 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005,
3 2006, 2007, 2008 Free Software Foundation, Inc.
4 Ian Lance Taylor, Cygnus Support
5 Linker support added by Mark Mitchell, CodeSourcery, LLC.
6 <mark@codesourcery.com>
8 This file is part of BFD, the Binary File Descriptor library.
10 This program is free software; you can redistribute it and/or modify
11 it under the terms of the GNU General Public License as published by
12 the Free Software Foundation; either version 3 of the License, or
13 (at your option) any later version.
15 This program is distributed in the hope that it will be useful,
16 but WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18 GNU General Public License for more details.
20 You should have received a copy of the GNU General Public License
21 along with this program; if not, write to the Free Software
22 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
23 MA 02110-1301, USA. */
26 /* This file supports the 64-bit MIPS ELF ABI.
28 The MIPS 64-bit ELF ABI uses an unusual reloc format. This file
29 overrides the usual ELF reloc handling, and handles reading and
30 writing the relocations here. */
32 /* TODO: Many things are unsupported, even if there is some code for it
33 . (which was mostly stolen from elf32-mips.c and slightly adapted).
35 . - Relocation handling for REL relocs is wrong in many cases and
37 . - Relocation handling for RELA relocs related to GOT support are
38 . also likely to be wrong.
39 . - Support for MIPS16 is untested.
40 . - Combined relocs with RSS_* entries are unsupported.
41 . - The whole GOT handling for NewABI is missing, some parts of
42 . the OldABI version is still lying around and should be removed.
52 #include "elfxx-mips.h"
55 /* Get the ECOFF swapping routines. The 64-bit ABI is not supposed to
56 use ECOFF. However, we support it anyhow for an easier changeover. */
58 #include "coff/symconst.h"
59 #include "coff/internal.h"
60 #include "coff/ecoff.h"
61 /* The 64 bit versions of the mdebug data structures are in alpha.h. */
62 #include "coff/alpha.h"
63 #define ECOFF_SIGNED_64
64 #include "ecoffswap.h"
66 static void mips_elf64_swap_reloc_in
67 (bfd
*, const Elf64_Mips_External_Rel
*, Elf64_Mips_Internal_Rela
*);
68 static void mips_elf64_swap_reloca_in
69 (bfd
*, const Elf64_Mips_External_Rela
*, Elf64_Mips_Internal_Rela
*);
70 static void mips_elf64_swap_reloc_out
71 (bfd
*, const Elf64_Mips_Internal_Rela
*, Elf64_Mips_External_Rel
*);
72 static void mips_elf64_swap_reloca_out
73 (bfd
*, const Elf64_Mips_Internal_Rela
*, Elf64_Mips_External_Rela
*);
74 static void mips_elf64_be_swap_reloc_in
75 (bfd
*, const bfd_byte
*, Elf_Internal_Rela
*);
76 static void mips_elf64_be_swap_reloc_out
77 (bfd
*, const Elf_Internal_Rela
*, bfd_byte
*);
78 static void mips_elf64_be_swap_reloca_in
79 (bfd
*, const bfd_byte
*, Elf_Internal_Rela
*);
80 static void mips_elf64_be_swap_reloca_out
81 (bfd
*, const Elf_Internal_Rela
*, bfd_byte
*);
82 static reloc_howto_type
*bfd_elf64_bfd_reloc_type_lookup
83 (bfd
*, bfd_reloc_code_real_type
);
84 static reloc_howto_type
*mips_elf64_rtype_to_howto
85 (unsigned int, bfd_boolean
);
86 static void mips_elf64_info_to_howto_rel
87 (bfd
*, arelent
*, Elf_Internal_Rela
*);
88 static void mips_elf64_info_to_howto_rela
89 (bfd
*, arelent
*, Elf_Internal_Rela
*);
90 static long mips_elf64_get_reloc_upper_bound
92 static long mips_elf64_canonicalize_reloc
93 (bfd
*, asection
*, arelent
**, asymbol
**);
94 static long mips_elf64_get_dynamic_reloc_upper_bound
96 static long mips_elf64_canonicalize_dynamic_reloc
97 (bfd
*, arelent
**, asymbol
**);
98 static bfd_boolean mips_elf64_slurp_one_reloc_table
99 (bfd
*, asection
*, Elf_Internal_Shdr
*, bfd_size_type
, arelent
*,
100 asymbol
**, bfd_boolean
);
101 static bfd_boolean mips_elf64_slurp_reloc_table
102 (bfd
*, asection
*, asymbol
**, bfd_boolean
);
103 static void mips_elf64_write_relocs
104 (bfd
*, asection
*, void *);
105 static void mips_elf64_write_rel
106 (bfd
*, asection
*, Elf_Internal_Shdr
*, int *, void *);
107 static void mips_elf64_write_rela
108 (bfd
*, asection
*, Elf_Internal_Shdr
*, int *, void *);
109 static bfd_reloc_status_type mips_elf64_gprel16_reloc
110 (bfd
*, arelent
*, asymbol
*, void *, asection
*, bfd
*, char **);
111 static bfd_reloc_status_type mips_elf64_literal_reloc
112 (bfd
*, arelent
*, asymbol
*, void *, asection
*, bfd
*, char **);
113 static bfd_reloc_status_type mips_elf64_gprel32_reloc
114 (bfd
*, arelent
*, asymbol
*, void *, asection
*, bfd
*, char **);
115 static bfd_reloc_status_type mips_elf64_shift6_reloc
116 (bfd
*, arelent
*, asymbol
*, void *, asection
*, bfd
*, char **);
117 static bfd_reloc_status_type mips16_gprel_reloc
118 (bfd
*, arelent
*, asymbol
*, void *, asection
*, bfd
*, char **);
119 static bfd_boolean mips_elf64_assign_gp
121 static bfd_reloc_status_type mips_elf64_final_gp
122 (bfd
*, asymbol
*, bfd_boolean
, char **, bfd_vma
*);
123 static bfd_boolean mips_elf64_object_p
125 static irix_compat_t elf64_mips_irix_compat
127 static bfd_boolean elf64_mips_grok_prstatus
128 (bfd
*, Elf_Internal_Note
*);
129 static bfd_boolean elf64_mips_grok_psinfo
130 (bfd
*, Elf_Internal_Note
*);
132 extern const bfd_target bfd_elf64_bigmips_vec
;
133 extern const bfd_target bfd_elf64_littlemips_vec
;
135 /* In case we're on a 32-bit machine, construct a 64-bit "-1" value
136 from smaller values. Start with zero, widen, *then* decrement. */
137 #define MINUS_ONE (((bfd_vma)0) - 1)
139 /* The number of local .got entries we reserve. */
140 #define MIPS_RESERVED_GOTNO (2)
142 /* The relocation table used for SHT_REL sections. */
144 static reloc_howto_type mips_elf64_howto_table_rel
[] =
147 HOWTO (R_MIPS_NONE
, /* type */
149 0, /* size (0 = byte, 1 = short, 2 = long) */
151 FALSE
, /* pc_relative */
153 complain_overflow_dont
, /* complain_on_overflow */
154 _bfd_mips_elf_generic_reloc
, /* special_function */
155 "R_MIPS_NONE", /* name */
156 FALSE
, /* partial_inplace */
159 FALSE
), /* pcrel_offset */
161 /* 16 bit relocation. */
162 HOWTO (R_MIPS_16
, /* type */
164 2, /* size (0 = byte, 1 = short, 2 = long) */
166 FALSE
, /* pc_relative */
168 complain_overflow_signed
, /* complain_on_overflow */
169 _bfd_mips_elf_generic_reloc
, /* special_function */
170 "R_MIPS_16", /* name */
171 TRUE
, /* partial_inplace */
172 0x0000ffff, /* src_mask */
173 0x0000ffff, /* dst_mask */
174 FALSE
), /* pcrel_offset */
176 /* 32 bit relocation. */
177 HOWTO (R_MIPS_32
, /* type */
179 2, /* size (0 = byte, 1 = short, 2 = long) */
181 FALSE
, /* pc_relative */
183 complain_overflow_dont
, /* complain_on_overflow */
184 _bfd_mips_elf_generic_reloc
, /* special_function */
185 "R_MIPS_32", /* name */
186 TRUE
, /* partial_inplace */
187 0xffffffff, /* src_mask */
188 0xffffffff, /* dst_mask */
189 FALSE
), /* pcrel_offset */
191 /* 32 bit symbol relative relocation. */
192 HOWTO (R_MIPS_REL32
, /* type */
194 2, /* size (0 = byte, 1 = short, 2 = long) */
196 FALSE
, /* pc_relative */
198 complain_overflow_dont
, /* complain_on_overflow */
199 _bfd_mips_elf_generic_reloc
, /* special_function */
200 "R_MIPS_REL32", /* name */
201 TRUE
, /* partial_inplace */
202 0xffffffff, /* src_mask */
203 0xffffffff, /* dst_mask */
204 FALSE
), /* pcrel_offset */
206 /* 26 bit jump address. */
207 HOWTO (R_MIPS_26
, /* type */
209 2, /* size (0 = byte, 1 = short, 2 = long) */
211 FALSE
, /* pc_relative */
213 complain_overflow_dont
, /* complain_on_overflow */
214 /* This needs complex overflow
215 detection, because the upper 36
216 bits must match the PC + 4. */
217 _bfd_mips_elf_generic_reloc
, /* special_function */
218 "R_MIPS_26", /* name */
219 TRUE
, /* partial_inplace */
220 0x03ffffff, /* src_mask */
221 0x03ffffff, /* dst_mask */
222 FALSE
), /* pcrel_offset */
224 /* R_MIPS_HI16 and R_MIPS_LO16 are unsupported for NewABI REL.
225 However, the native IRIX6 tools use them, so we try our best. */
227 /* High 16 bits of symbol value. */
228 HOWTO (R_MIPS_HI16
, /* type */
230 2, /* size (0 = byte, 1 = short, 2 = long) */
232 FALSE
, /* pc_relative */
234 complain_overflow_dont
, /* complain_on_overflow */
235 _bfd_mips_elf_hi16_reloc
, /* special_function */
236 "R_MIPS_HI16", /* name */
237 TRUE
, /* partial_inplace */
238 0x0000ffff, /* src_mask */
239 0x0000ffff, /* dst_mask */
240 FALSE
), /* pcrel_offset */
242 /* Low 16 bits of symbol value. */
243 HOWTO (R_MIPS_LO16
, /* type */
245 2, /* size (0 = byte, 1 = short, 2 = long) */
247 FALSE
, /* pc_relative */
249 complain_overflow_dont
, /* complain_on_overflow */
250 _bfd_mips_elf_lo16_reloc
, /* special_function */
251 "R_MIPS_LO16", /* name */
252 TRUE
, /* partial_inplace */
253 0x0000ffff, /* src_mask */
254 0x0000ffff, /* dst_mask */
255 FALSE
), /* pcrel_offset */
257 /* GP relative reference. */
258 HOWTO (R_MIPS_GPREL16
, /* type */
260 2, /* size (0 = byte, 1 = short, 2 = long) */
262 FALSE
, /* pc_relative */
264 complain_overflow_signed
, /* complain_on_overflow */
265 mips_elf64_gprel16_reloc
, /* special_function */
266 "R_MIPS_GPREL16", /* name */
267 TRUE
, /* partial_inplace */
268 0x0000ffff, /* src_mask */
269 0x0000ffff, /* dst_mask */
270 FALSE
), /* pcrel_offset */
272 /* Reference to literal section. */
273 HOWTO (R_MIPS_LITERAL
, /* type */
275 2, /* size (0 = byte, 1 = short, 2 = long) */
277 FALSE
, /* pc_relative */
279 complain_overflow_signed
, /* complain_on_overflow */
280 mips_elf64_literal_reloc
, /* special_function */
281 "R_MIPS_LITERAL", /* name */
282 TRUE
, /* partial_inplace */
283 0x0000ffff, /* src_mask */
284 0x0000ffff, /* dst_mask */
285 FALSE
), /* pcrel_offset */
287 /* Reference to global offset table. */
288 HOWTO (R_MIPS_GOT16
, /* type */
290 2, /* size (0 = byte, 1 = short, 2 = long) */
292 FALSE
, /* pc_relative */
294 complain_overflow_signed
, /* complain_on_overflow */
295 _bfd_mips_elf_got16_reloc
, /* special_function */
296 "R_MIPS_GOT16", /* name */
297 TRUE
, /* partial_inplace */
298 0x0000ffff, /* src_mask */
299 0x0000ffff, /* dst_mask */
300 FALSE
), /* pcrel_offset */
302 /* 16 bit PC relative reference. Note that the ABI document has a typo
303 and claims R_MIPS_PC16 to be not rightshifted, rendering it useless.
304 We do the right thing here. */
305 HOWTO (R_MIPS_PC16
, /* type */
307 2, /* size (0 = byte, 1 = short, 2 = long) */
309 TRUE
, /* pc_relative */
311 complain_overflow_signed
, /* complain_on_overflow */
312 _bfd_mips_elf_generic_reloc
, /* special_function */
313 "R_MIPS_PC16", /* name */
314 TRUE
, /* partial_inplace */
315 0x0000ffff, /* src_mask */
316 0x0000ffff, /* dst_mask */
317 TRUE
), /* pcrel_offset */
319 /* 16 bit call through global offset table. */
320 HOWTO (R_MIPS_CALL16
, /* type */
322 2, /* size (0 = byte, 1 = short, 2 = long) */
324 FALSE
, /* pc_relative */
326 complain_overflow_signed
, /* complain_on_overflow */
327 _bfd_mips_elf_generic_reloc
, /* special_function */
328 "R_MIPS_CALL16", /* name */
329 TRUE
, /* partial_inplace */
330 0x0000ffff, /* src_mask */
331 0x0000ffff, /* dst_mask */
332 FALSE
), /* pcrel_offset */
334 /* 32 bit GP relative reference. */
335 HOWTO (R_MIPS_GPREL32
, /* type */
337 2, /* size (0 = byte, 1 = short, 2 = long) */
339 FALSE
, /* pc_relative */
341 complain_overflow_dont
, /* complain_on_overflow */
342 mips_elf64_gprel32_reloc
, /* special_function */
343 "R_MIPS_GPREL32", /* name */
344 TRUE
, /* partial_inplace */
345 0xffffffff, /* src_mask */
346 0xffffffff, /* dst_mask */
347 FALSE
), /* pcrel_offset */
353 /* A 5 bit shift field. */
354 HOWTO (R_MIPS_SHIFT5
, /* type */
356 2, /* size (0 = byte, 1 = short, 2 = long) */
358 FALSE
, /* pc_relative */
360 complain_overflow_bitfield
, /* complain_on_overflow */
361 _bfd_mips_elf_generic_reloc
, /* special_function */
362 "R_MIPS_SHIFT5", /* name */
363 TRUE
, /* partial_inplace */
364 0x000007c0, /* src_mask */
365 0x000007c0, /* dst_mask */
366 FALSE
), /* pcrel_offset */
368 /* A 6 bit shift field. */
369 HOWTO (R_MIPS_SHIFT6
, /* type */
371 2, /* size (0 = byte, 1 = short, 2 = long) */
373 FALSE
, /* pc_relative */
375 complain_overflow_bitfield
, /* complain_on_overflow */
376 mips_elf64_shift6_reloc
, /* special_function */
377 "R_MIPS_SHIFT6", /* name */
378 TRUE
, /* partial_inplace */
379 0x000007c4, /* src_mask */
380 0x000007c4, /* dst_mask */
381 FALSE
), /* pcrel_offset */
383 /* 64 bit relocation. */
384 HOWTO (R_MIPS_64
, /* type */
386 4, /* size (0 = byte, 1 = short, 2 = long) */
388 FALSE
, /* pc_relative */
390 complain_overflow_dont
, /* complain_on_overflow */
391 _bfd_mips_elf_generic_reloc
, /* special_function */
392 "R_MIPS_64", /* name */
393 TRUE
, /* partial_inplace */
394 MINUS_ONE
, /* src_mask */
395 MINUS_ONE
, /* dst_mask */
396 FALSE
), /* pcrel_offset */
398 /* Displacement in the global offset table. */
399 HOWTO (R_MIPS_GOT_DISP
, /* type */
401 2, /* size (0 = byte, 1 = short, 2 = long) */
403 FALSE
, /* pc_relative */
405 complain_overflow_signed
, /* complain_on_overflow */
406 _bfd_mips_elf_generic_reloc
, /* special_function */
407 "R_MIPS_GOT_DISP", /* name */
408 TRUE
, /* partial_inplace */
409 0x0000ffff, /* src_mask */
410 0x0000ffff, /* dst_mask */
411 FALSE
), /* pcrel_offset */
413 /* Displacement to page pointer in the global offset table. */
414 HOWTO (R_MIPS_GOT_PAGE
, /* type */
416 2, /* size (0 = byte, 1 = short, 2 = long) */
418 FALSE
, /* pc_relative */
420 complain_overflow_signed
, /* complain_on_overflow */
421 _bfd_mips_elf_generic_reloc
, /* special_function */
422 "R_MIPS_GOT_PAGE", /* name */
423 TRUE
, /* partial_inplace */
424 0x0000ffff, /* src_mask */
425 0x0000ffff, /* dst_mask */
426 FALSE
), /* pcrel_offset */
428 /* Offset from page pointer in the global offset table. */
429 HOWTO (R_MIPS_GOT_OFST
, /* type */
431 2, /* size (0 = byte, 1 = short, 2 = long) */
433 FALSE
, /* pc_relative */
435 complain_overflow_signed
, /* complain_on_overflow */
436 _bfd_mips_elf_generic_reloc
, /* special_function */
437 "R_MIPS_GOT_OFST", /* name */
438 TRUE
, /* partial_inplace */
439 0x0000ffff, /* src_mask */
440 0x0000ffff, /* dst_mask */
441 FALSE
), /* pcrel_offset */
443 /* High 16 bits of displacement in global offset table. */
444 HOWTO (R_MIPS_GOT_HI16
, /* type */
446 2, /* size (0 = byte, 1 = short, 2 = long) */
448 FALSE
, /* pc_relative */
450 complain_overflow_dont
, /* complain_on_overflow */
451 _bfd_mips_elf_generic_reloc
, /* special_function */
452 "R_MIPS_GOT_HI16", /* name */
453 TRUE
, /* partial_inplace */
454 0x0000ffff, /* src_mask */
455 0x0000ffff, /* dst_mask */
456 FALSE
), /* pcrel_offset */
458 /* Low 16 bits of displacement in global offset table. */
459 HOWTO (R_MIPS_GOT_LO16
, /* type */
461 2, /* size (0 = byte, 1 = short, 2 = long) */
463 FALSE
, /* pc_relative */
465 complain_overflow_dont
, /* complain_on_overflow */
466 _bfd_mips_elf_generic_reloc
, /* special_function */
467 "R_MIPS_GOT_LO16", /* name */
468 TRUE
, /* partial_inplace */
469 0x0000ffff, /* src_mask */
470 0x0000ffff, /* dst_mask */
471 FALSE
), /* pcrel_offset */
473 /* 64 bit subtraction. */
474 HOWTO (R_MIPS_SUB
, /* type */
476 4, /* size (0 = byte, 1 = short, 2 = long) */
478 FALSE
, /* pc_relative */
480 complain_overflow_dont
, /* complain_on_overflow */
481 _bfd_mips_elf_generic_reloc
, /* special_function */
482 "R_MIPS_SUB", /* name */
483 TRUE
, /* partial_inplace */
484 MINUS_ONE
, /* src_mask */
485 MINUS_ONE
, /* dst_mask */
486 FALSE
), /* pcrel_offset */
488 /* Insert the addend as an instruction. */
489 /* FIXME: Not handled correctly. */
490 HOWTO (R_MIPS_INSERT_A
, /* type */
492 2, /* size (0 = byte, 1 = short, 2 = long) */
494 FALSE
, /* pc_relative */
496 complain_overflow_dont
, /* complain_on_overflow */
497 _bfd_mips_elf_generic_reloc
, /* special_function */
498 "R_MIPS_INSERT_A", /* name */
499 TRUE
, /* partial_inplace */
500 0xffffffff, /* src_mask */
501 0xffffffff, /* dst_mask */
502 FALSE
), /* pcrel_offset */
504 /* Insert the addend as an instruction, and change all relocations
505 to refer to the old instruction at the address. */
506 /* FIXME: Not handled correctly. */
507 HOWTO (R_MIPS_INSERT_B
, /* type */
509 2, /* size (0 = byte, 1 = short, 2 = long) */
511 FALSE
, /* pc_relative */
513 complain_overflow_dont
, /* complain_on_overflow */
514 _bfd_mips_elf_generic_reloc
, /* special_function */
515 "R_MIPS_INSERT_B", /* name */
516 TRUE
, /* partial_inplace */
517 0xffffffff, /* src_mask */
518 0xffffffff, /* dst_mask */
519 FALSE
), /* pcrel_offset */
521 /* Delete a 32 bit instruction. */
522 /* FIXME: Not handled correctly. */
523 HOWTO (R_MIPS_DELETE
, /* type */
525 2, /* size (0 = byte, 1 = short, 2 = long) */
527 FALSE
, /* pc_relative */
529 complain_overflow_dont
, /* complain_on_overflow */
530 _bfd_mips_elf_generic_reloc
, /* special_function */
531 "R_MIPS_DELETE", /* name */
532 TRUE
, /* partial_inplace */
533 0xffffffff, /* src_mask */
534 0xffffffff, /* dst_mask */
535 FALSE
), /* pcrel_offset */
537 /* The MIPS ELF64 ABI Draft wants us to support these for REL relocations.
539 a) It means building the addend from a R_MIPS_HIGHEST/R_MIPS_HIGHER/
540 R_MIPS_HI16/R_MIPS_LO16 sequence with varying ordering, using
542 b) No other NewABI toolchain actually emits such relocations. */
543 EMPTY_HOWTO (R_MIPS_HIGHER
),
544 EMPTY_HOWTO (R_MIPS_HIGHEST
),
546 /* High 16 bits of displacement in global offset table. */
547 HOWTO (R_MIPS_CALL_HI16
, /* type */
549 2, /* size (0 = byte, 1 = short, 2 = long) */
551 FALSE
, /* pc_relative */
553 complain_overflow_dont
, /* complain_on_overflow */
554 _bfd_mips_elf_generic_reloc
, /* special_function */
555 "R_MIPS_CALL_HI16", /* name */
556 TRUE
, /* partial_inplace */
557 0x0000ffff, /* src_mask */
558 0x0000ffff, /* dst_mask */
559 FALSE
), /* pcrel_offset */
561 /* Low 16 bits of displacement in global offset table. */
562 HOWTO (R_MIPS_CALL_LO16
, /* type */
564 2, /* size (0 = byte, 1 = short, 2 = long) */
566 FALSE
, /* pc_relative */
568 complain_overflow_dont
, /* complain_on_overflow */
569 _bfd_mips_elf_generic_reloc
, /* special_function */
570 "R_MIPS_CALL_LO16", /* name */
571 TRUE
, /* partial_inplace */
572 0x0000ffff, /* src_mask */
573 0x0000ffff, /* dst_mask */
574 FALSE
), /* pcrel_offset */
576 /* Section displacement, used by an associated event location section. */
577 HOWTO (R_MIPS_SCN_DISP
, /* type */
579 2, /* size (0 = byte, 1 = short, 2 = long) */
581 FALSE
, /* pc_relative */
583 complain_overflow_dont
, /* complain_on_overflow */
584 _bfd_mips_elf_generic_reloc
, /* special_function */
585 "R_MIPS_SCN_DISP", /* name */
586 TRUE
, /* partial_inplace */
587 0xffffffff, /* src_mask */
588 0xffffffff, /* dst_mask */
589 FALSE
), /* pcrel_offset */
591 HOWTO (R_MIPS_REL16
, /* type */
593 1, /* size (0 = byte, 1 = short, 2 = long) */
595 FALSE
, /* pc_relative */
597 complain_overflow_signed
, /* complain_on_overflow */
598 _bfd_mips_elf_generic_reloc
, /* special_function */
599 "R_MIPS_REL16", /* name */
600 TRUE
, /* partial_inplace */
601 0xffff, /* src_mask */
602 0xffff, /* dst_mask */
603 FALSE
), /* pcrel_offset */
605 /* These two are obsolete. */
606 EMPTY_HOWTO (R_MIPS_ADD_IMMEDIATE
),
607 EMPTY_HOWTO (R_MIPS_PJUMP
),
609 /* Similiar to R_MIPS_REL32, but used for relocations in a GOT section.
610 It must be used for multigot GOT's (and only there). */
611 HOWTO (R_MIPS_RELGOT
, /* type */
613 2, /* size (0 = byte, 1 = short, 2 = long) */
615 FALSE
, /* pc_relative */
617 complain_overflow_dont
, /* complain_on_overflow */
618 _bfd_mips_elf_generic_reloc
, /* special_function */
619 "R_MIPS_RELGOT", /* name */
620 TRUE
, /* partial_inplace */
621 0xffffffff, /* src_mask */
622 0xffffffff, /* dst_mask */
623 FALSE
), /* pcrel_offset */
625 /* Protected jump conversion. This is an optimization hint. No
626 relocation is required for correctness. */
627 HOWTO (R_MIPS_JALR
, /* type */
629 2, /* size (0 = byte, 1 = short, 2 = long) */
631 FALSE
, /* pc_relative */
633 complain_overflow_dont
, /* complain_on_overflow */
634 _bfd_mips_elf_generic_reloc
, /* special_function */
635 "R_MIPS_JALR", /* name */
636 FALSE
, /* partial_inplace */
638 0x00000000, /* dst_mask */
639 FALSE
), /* pcrel_offset */
641 /* TLS relocations. */
642 EMPTY_HOWTO (R_MIPS_TLS_DTPMOD32
),
643 EMPTY_HOWTO (R_MIPS_TLS_DTPREL32
),
645 HOWTO (R_MIPS_TLS_DTPMOD64
, /* type */
647 4, /* size (0 = byte, 1 = short, 2 = long) */
649 FALSE
, /* pc_relative */
651 complain_overflow_dont
, /* complain_on_overflow */
652 _bfd_mips_elf_generic_reloc
, /* special_function */
653 "R_MIPS_TLS_DTPMOD64", /* name */
654 TRUE
, /* partial_inplace */
655 MINUS_ONE
, /* src_mask */
656 MINUS_ONE
, /* dst_mask */
657 FALSE
), /* pcrel_offset */
659 HOWTO (R_MIPS_TLS_DTPREL64
, /* type */
661 4, /* size (0 = byte, 1 = short, 2 = long) */
663 FALSE
, /* pc_relative */
665 complain_overflow_dont
, /* complain_on_overflow */
666 _bfd_mips_elf_generic_reloc
, /* special_function */
667 "R_MIPS_TLS_DTPREL64", /* name */
668 TRUE
, /* partial_inplace */
669 MINUS_ONE
, /* src_mask */
670 MINUS_ONE
, /* dst_mask */
671 FALSE
), /* pcrel_offset */
673 /* TLS general dynamic variable reference. */
674 HOWTO (R_MIPS_TLS_GD
, /* type */
676 2, /* size (0 = byte, 1 = short, 2 = long) */
678 FALSE
, /* pc_relative */
680 complain_overflow_signed
, /* complain_on_overflow */
681 _bfd_mips_elf_generic_reloc
, /* special_function */
682 "R_MIPS_TLS_GD", /* name */
683 TRUE
, /* partial_inplace */
684 0x0000ffff, /* src_mask */
685 0x0000ffff, /* dst_mask */
686 FALSE
), /* pcrel_offset */
688 /* TLS local dynamic variable reference. */
689 HOWTO (R_MIPS_TLS_LDM
, /* type */
691 2, /* size (0 = byte, 1 = short, 2 = long) */
693 FALSE
, /* pc_relative */
695 complain_overflow_signed
, /* complain_on_overflow */
696 _bfd_mips_elf_generic_reloc
, /* special_function */
697 "R_MIPS_TLS_LDM", /* name */
698 TRUE
, /* partial_inplace */
699 0x0000ffff, /* src_mask */
700 0x0000ffff, /* dst_mask */
701 FALSE
), /* pcrel_offset */
703 /* TLS local dynamic offset. */
704 HOWTO (R_MIPS_TLS_DTPREL_HI16
, /* type */
706 2, /* size (0 = byte, 1 = short, 2 = long) */
708 FALSE
, /* pc_relative */
710 complain_overflow_signed
, /* complain_on_overflow */
711 _bfd_mips_elf_generic_reloc
, /* special_function */
712 "R_MIPS_TLS_DTPREL_HI16", /* name */
713 TRUE
, /* partial_inplace */
714 0x0000ffff, /* src_mask */
715 0x0000ffff, /* dst_mask */
716 FALSE
), /* pcrel_offset */
718 /* TLS local dynamic offset. */
719 HOWTO (R_MIPS_TLS_DTPREL_LO16
, /* type */
721 2, /* size (0 = byte, 1 = short, 2 = long) */
723 FALSE
, /* pc_relative */
725 complain_overflow_signed
, /* complain_on_overflow */
726 _bfd_mips_elf_generic_reloc
, /* special_function */
727 "R_MIPS_TLS_DTPREL_LO16", /* name */
728 TRUE
, /* partial_inplace */
729 0x0000ffff, /* src_mask */
730 0x0000ffff, /* dst_mask */
731 FALSE
), /* pcrel_offset */
733 /* TLS thread pointer offset. */
734 HOWTO (R_MIPS_TLS_GOTTPREL
, /* type */
736 2, /* size (0 = byte, 1 = short, 2 = long) */
738 FALSE
, /* pc_relative */
740 complain_overflow_signed
, /* complain_on_overflow */
741 _bfd_mips_elf_generic_reloc
, /* special_function */
742 "R_MIPS_TLS_GOTTPREL", /* name */
743 TRUE
, /* partial_inplace */
744 0x0000ffff, /* src_mask */
745 0x0000ffff, /* dst_mask */
746 FALSE
), /* pcrel_offset */
748 /* TLS IE dynamic relocations. */
749 EMPTY_HOWTO (R_MIPS_TLS_TPREL32
),
751 HOWTO (R_MIPS_TLS_TPREL64
, /* type */
753 4, /* size (0 = byte, 1 = short, 2 = long) */
755 FALSE
, /* pc_relative */
757 complain_overflow_dont
, /* complain_on_overflow */
758 _bfd_mips_elf_generic_reloc
, /* special_function */
759 "R_MIPS_TLS_TPREL64", /* name */
760 TRUE
, /* partial_inplace */
761 MINUS_ONE
, /* src_mask */
762 MINUS_ONE
, /* dst_mask */
763 FALSE
), /* pcrel_offset */
765 /* TLS thread pointer offset. */
766 HOWTO (R_MIPS_TLS_TPREL_HI16
, /* type */
768 2, /* size (0 = byte, 1 = short, 2 = long) */
770 FALSE
, /* pc_relative */
772 complain_overflow_signed
, /* complain_on_overflow */
773 _bfd_mips_elf_generic_reloc
, /* special_function */
774 "R_MIPS_TLS_TPREL_HI16", /* name */
775 TRUE
, /* partial_inplace */
776 0x0000ffff, /* src_mask */
777 0x0000ffff, /* dst_mask */
778 FALSE
), /* pcrel_offset */
780 /* TLS thread pointer offset. */
781 HOWTO (R_MIPS_TLS_TPREL_LO16
, /* type */
783 2, /* size (0 = byte, 1 = short, 2 = long) */
785 FALSE
, /* pc_relative */
787 complain_overflow_signed
, /* complain_on_overflow */
788 _bfd_mips_elf_generic_reloc
, /* special_function */
789 "R_MIPS_TLS_TPREL_LO16", /* name */
790 TRUE
, /* partial_inplace */
791 0x0000ffff, /* src_mask */
792 0x0000ffff, /* dst_mask */
793 FALSE
), /* pcrel_offset */
795 /* 32 bit relocation with no addend. */
796 HOWTO (R_MIPS_GLOB_DAT
, /* type */
798 2, /* size (0 = byte, 1 = short, 2 = long) */
800 FALSE
, /* pc_relative */
802 complain_overflow_dont
, /* complain_on_overflow */
803 _bfd_mips_elf_generic_reloc
, /* special_function */
804 "R_MIPS_GLOB_DAT", /* name */
805 FALSE
, /* partial_inplace */
807 0xffffffff, /* dst_mask */
808 FALSE
), /* pcrel_offset */
811 /* The relocation table used for SHT_RELA sections. */
813 static reloc_howto_type mips_elf64_howto_table_rela
[] =
816 HOWTO (R_MIPS_NONE
, /* type */
818 0, /* size (0 = byte, 1 = short, 2 = long) */
820 FALSE
, /* pc_relative */
822 complain_overflow_dont
, /* complain_on_overflow */
823 _bfd_mips_elf_generic_reloc
, /* special_function */
824 "R_MIPS_NONE", /* name */
825 FALSE
, /* partial_inplace */
828 FALSE
), /* pcrel_offset */
830 /* 16 bit relocation. */
831 HOWTO (R_MIPS_16
, /* type */
833 2, /* size (0 = byte, 1 = short, 2 = long) */
835 FALSE
, /* pc_relative */
837 complain_overflow_signed
, /* complain_on_overflow */
838 _bfd_mips_elf_generic_reloc
, /* special_function */
839 "R_MIPS_16", /* name */
840 FALSE
, /* partial_inplace */
842 0x0000ffff, /* dst_mask */
843 FALSE
), /* pcrel_offset */
845 /* 32 bit relocation. */
846 HOWTO (R_MIPS_32
, /* type */
848 2, /* size (0 = byte, 1 = short, 2 = long) */
850 FALSE
, /* pc_relative */
852 complain_overflow_dont
, /* complain_on_overflow */
853 _bfd_mips_elf_generic_reloc
, /* special_function */
854 "R_MIPS_32", /* name */
855 FALSE
, /* partial_inplace */
857 0xffffffff, /* dst_mask */
858 FALSE
), /* pcrel_offset */
860 /* 32 bit symbol relative relocation. */
861 HOWTO (R_MIPS_REL32
, /* type */
863 2, /* size (0 = byte, 1 = short, 2 = long) */
865 FALSE
, /* pc_relative */
867 complain_overflow_dont
, /* complain_on_overflow */
868 _bfd_mips_elf_generic_reloc
, /* special_function */
869 "R_MIPS_REL32", /* name */
870 FALSE
, /* partial_inplace */
872 0xffffffff, /* dst_mask */
873 FALSE
), /* pcrel_offset */
875 /* 26 bit jump address. */
876 HOWTO (R_MIPS_26
, /* type */
878 2, /* size (0 = byte, 1 = short, 2 = long) */
880 FALSE
, /* pc_relative */
882 complain_overflow_dont
, /* complain_on_overflow */
883 /* This needs complex overflow
884 detection, because the upper 36
885 bits must match the PC + 4. */
886 _bfd_mips_elf_generic_reloc
, /* special_function */
887 "R_MIPS_26", /* name */
888 FALSE
, /* partial_inplace */
890 0x03ffffff, /* dst_mask */
891 FALSE
), /* pcrel_offset */
893 /* High 16 bits of symbol value. */
894 HOWTO (R_MIPS_HI16
, /* type */
896 2, /* size (0 = byte, 1 = short, 2 = long) */
898 FALSE
, /* pc_relative */
900 complain_overflow_dont
, /* complain_on_overflow */
901 _bfd_mips_elf_generic_reloc
, /* special_function */
902 "R_MIPS_HI16", /* name */
903 FALSE
, /* partial_inplace */
905 0x0000ffff, /* dst_mask */
906 FALSE
), /* pcrel_offset */
908 /* Low 16 bits of symbol value. */
909 HOWTO (R_MIPS_LO16
, /* type */
911 2, /* size (0 = byte, 1 = short, 2 = long) */
913 FALSE
, /* pc_relative */
915 complain_overflow_dont
, /* complain_on_overflow */
916 _bfd_mips_elf_generic_reloc
, /* special_function */
917 "R_MIPS_LO16", /* name */
918 FALSE
, /* partial_inplace */
920 0x0000ffff, /* dst_mask */
921 FALSE
), /* pcrel_offset */
923 /* GP relative reference. */
924 HOWTO (R_MIPS_GPREL16
, /* type */
926 2, /* size (0 = byte, 1 = short, 2 = long) */
928 FALSE
, /* pc_relative */
930 complain_overflow_signed
, /* complain_on_overflow */
931 mips_elf64_gprel16_reloc
, /* special_function */
932 "R_MIPS_GPREL16", /* name */
933 FALSE
, /* partial_inplace */
935 0x0000ffff, /* dst_mask */
936 FALSE
), /* pcrel_offset */
938 /* Reference to literal section. */
939 HOWTO (R_MIPS_LITERAL
, /* type */
941 2, /* size (0 = byte, 1 = short, 2 = long) */
943 FALSE
, /* pc_relative */
945 complain_overflow_signed
, /* complain_on_overflow */
946 mips_elf64_literal_reloc
, /* special_function */
947 "R_MIPS_LITERAL", /* name */
948 FALSE
, /* partial_inplace */
950 0x0000ffff, /* dst_mask */
951 FALSE
), /* pcrel_offset */
953 /* Reference to global offset table. */
954 HOWTO (R_MIPS_GOT16
, /* type */
956 2, /* size (0 = byte, 1 = short, 2 = long) */
958 FALSE
, /* pc_relative */
960 complain_overflow_signed
, /* complain_on_overflow */
961 _bfd_mips_elf_generic_reloc
, /* special_function */
962 "R_MIPS_GOT16", /* name */
963 FALSE
, /* partial_inplace */
965 0x0000ffff, /* dst_mask */
966 FALSE
), /* pcrel_offset */
968 /* 16 bit PC relative reference. Note that the ABI document has a typo
969 and claims R_MIPS_PC16 to be not rightshifted, rendering it useless.
970 We do the right thing here. */
971 HOWTO (R_MIPS_PC16
, /* type */
973 2, /* size (0 = byte, 1 = short, 2 = long) */
975 TRUE
, /* pc_relative */
977 complain_overflow_signed
, /* complain_on_overflow */
978 _bfd_mips_elf_generic_reloc
, /* special_function */
979 "R_MIPS_PC16", /* name */
980 FALSE
, /* partial_inplace */
982 0x0000ffff, /* dst_mask */
983 TRUE
), /* pcrel_offset */
985 /* 16 bit call through global offset table. */
986 HOWTO (R_MIPS_CALL16
, /* type */
988 2, /* size (0 = byte, 1 = short, 2 = long) */
990 FALSE
, /* pc_relative */
992 complain_overflow_signed
, /* complain_on_overflow */
993 _bfd_mips_elf_generic_reloc
, /* special_function */
994 "R_MIPS_CALL16", /* name */
995 FALSE
, /* partial_inplace */
997 0x0000ffff, /* dst_mask */
998 FALSE
), /* pcrel_offset */
1000 /* 32 bit GP relative reference. */
1001 HOWTO (R_MIPS_GPREL32
, /* type */
1003 2, /* size (0 = byte, 1 = short, 2 = long) */
1005 FALSE
, /* pc_relative */
1007 complain_overflow_dont
, /* complain_on_overflow */
1008 mips_elf64_gprel32_reloc
, /* special_function */
1009 "R_MIPS_GPREL32", /* name */
1010 FALSE
, /* partial_inplace */
1012 0xffffffff, /* dst_mask */
1013 FALSE
), /* pcrel_offset */
1019 /* A 5 bit shift field. */
1020 HOWTO (R_MIPS_SHIFT5
, /* type */
1022 2, /* size (0 = byte, 1 = short, 2 = long) */
1024 FALSE
, /* pc_relative */
1026 complain_overflow_bitfield
, /* complain_on_overflow */
1027 _bfd_mips_elf_generic_reloc
, /* special_function */
1028 "R_MIPS_SHIFT5", /* name */
1029 FALSE
, /* partial_inplace */
1031 0x000007c0, /* dst_mask */
1032 FALSE
), /* pcrel_offset */
1034 /* A 6 bit shift field. */
1035 HOWTO (R_MIPS_SHIFT6
, /* type */
1037 2, /* size (0 = byte, 1 = short, 2 = long) */
1039 FALSE
, /* pc_relative */
1041 complain_overflow_bitfield
, /* complain_on_overflow */
1042 mips_elf64_shift6_reloc
, /* special_function */
1043 "R_MIPS_SHIFT6", /* name */
1044 FALSE
, /* partial_inplace */
1046 0x000007c4, /* dst_mask */
1047 FALSE
), /* pcrel_offset */
1049 /* 64 bit relocation. */
1050 HOWTO (R_MIPS_64
, /* type */
1052 4, /* size (0 = byte, 1 = short, 2 = long) */
1054 FALSE
, /* pc_relative */
1056 complain_overflow_dont
, /* complain_on_overflow */
1057 _bfd_mips_elf_generic_reloc
, /* special_function */
1058 "R_MIPS_64", /* name */
1059 FALSE
, /* partial_inplace */
1061 MINUS_ONE
, /* dst_mask */
1062 FALSE
), /* pcrel_offset */
1064 /* Displacement in the global offset table. */
1065 HOWTO (R_MIPS_GOT_DISP
, /* type */
1067 2, /* size (0 = byte, 1 = short, 2 = long) */
1069 FALSE
, /* pc_relative */
1071 complain_overflow_signed
, /* complain_on_overflow */
1072 _bfd_mips_elf_generic_reloc
, /* special_function */
1073 "R_MIPS_GOT_DISP", /* name */
1074 FALSE
, /* partial_inplace */
1076 0x0000ffff, /* dst_mask */
1077 FALSE
), /* pcrel_offset */
1079 /* Displacement to page pointer in the global offset table. */
1080 HOWTO (R_MIPS_GOT_PAGE
, /* type */
1082 2, /* size (0 = byte, 1 = short, 2 = long) */
1084 FALSE
, /* pc_relative */
1086 complain_overflow_signed
, /* complain_on_overflow */
1087 _bfd_mips_elf_generic_reloc
, /* special_function */
1088 "R_MIPS_GOT_PAGE", /* name */
1089 FALSE
, /* partial_inplace */
1091 0x0000ffff, /* dst_mask */
1092 FALSE
), /* pcrel_offset */
1094 /* Offset from page pointer in the global offset table. */
1095 HOWTO (R_MIPS_GOT_OFST
, /* type */
1097 2, /* size (0 = byte, 1 = short, 2 = long) */
1099 FALSE
, /* pc_relative */
1101 complain_overflow_signed
, /* complain_on_overflow */
1102 _bfd_mips_elf_generic_reloc
, /* special_function */
1103 "R_MIPS_GOT_OFST", /* name */
1104 FALSE
, /* partial_inplace */
1106 0x0000ffff, /* dst_mask */
1107 FALSE
), /* pcrel_offset */
1109 /* High 16 bits of displacement in global offset table. */
1110 HOWTO (R_MIPS_GOT_HI16
, /* type */
1112 2, /* size (0 = byte, 1 = short, 2 = long) */
1114 FALSE
, /* pc_relative */
1116 complain_overflow_dont
, /* complain_on_overflow */
1117 _bfd_mips_elf_generic_reloc
, /* special_function */
1118 "R_MIPS_GOT_HI16", /* name */
1119 FALSE
, /* partial_inplace */
1121 0x0000ffff, /* dst_mask */
1122 FALSE
), /* pcrel_offset */
1124 /* Low 16 bits of displacement in global offset table. */
1125 HOWTO (R_MIPS_GOT_LO16
, /* type */
1127 2, /* size (0 = byte, 1 = short, 2 = long) */
1129 FALSE
, /* pc_relative */
1131 complain_overflow_dont
, /* complain_on_overflow */
1132 _bfd_mips_elf_generic_reloc
, /* special_function */
1133 "R_MIPS_GOT_LO16", /* name */
1134 FALSE
, /* partial_inplace */
1136 0x0000ffff, /* dst_mask */
1137 FALSE
), /* pcrel_offset */
1139 /* 64 bit subtraction. */
1140 HOWTO (R_MIPS_SUB
, /* type */
1142 4, /* size (0 = byte, 1 = short, 2 = long) */
1144 FALSE
, /* pc_relative */
1146 complain_overflow_dont
, /* complain_on_overflow */
1147 _bfd_mips_elf_generic_reloc
, /* special_function */
1148 "R_MIPS_SUB", /* name */
1149 FALSE
, /* partial_inplace */
1151 MINUS_ONE
, /* dst_mask */
1152 FALSE
), /* pcrel_offset */
1154 /* Insert the addend as an instruction. */
1155 /* FIXME: Not handled correctly. */
1156 HOWTO (R_MIPS_INSERT_A
, /* type */
1158 2, /* size (0 = byte, 1 = short, 2 = long) */
1160 FALSE
, /* pc_relative */
1162 complain_overflow_dont
, /* complain_on_overflow */
1163 _bfd_mips_elf_generic_reloc
, /* special_function */
1164 "R_MIPS_INSERT_A", /* name */
1165 FALSE
, /* partial_inplace */
1167 0xffffffff, /* dst_mask */
1168 FALSE
), /* pcrel_offset */
1170 /* Insert the addend as an instruction, and change all relocations
1171 to refer to the old instruction at the address. */
1172 /* FIXME: Not handled correctly. */
1173 HOWTO (R_MIPS_INSERT_B
, /* type */
1175 2, /* size (0 = byte, 1 = short, 2 = long) */
1177 FALSE
, /* pc_relative */
1179 complain_overflow_dont
, /* complain_on_overflow */
1180 _bfd_mips_elf_generic_reloc
, /* special_function */
1181 "R_MIPS_INSERT_B", /* name */
1182 FALSE
, /* partial_inplace */
1184 0xffffffff, /* dst_mask */
1185 FALSE
), /* pcrel_offset */
1187 /* Delete a 32 bit instruction. */
1188 /* FIXME: Not handled correctly. */
1189 HOWTO (R_MIPS_DELETE
, /* type */
1191 2, /* size (0 = byte, 1 = short, 2 = long) */
1193 FALSE
, /* pc_relative */
1195 complain_overflow_dont
, /* complain_on_overflow */
1196 _bfd_mips_elf_generic_reloc
, /* special_function */
1197 "R_MIPS_DELETE", /* name */
1198 FALSE
, /* partial_inplace */
1200 0xffffffff, /* dst_mask */
1201 FALSE
), /* pcrel_offset */
1203 /* Get the higher value of a 64 bit addend. */
1204 HOWTO (R_MIPS_HIGHER
, /* type */
1206 2, /* size (0 = byte, 1 = short, 2 = long) */
1208 FALSE
, /* pc_relative */
1210 complain_overflow_dont
, /* complain_on_overflow */
1211 _bfd_mips_elf_generic_reloc
, /* special_function */
1212 "R_MIPS_HIGHER", /* name */
1213 FALSE
, /* partial_inplace */
1215 0x0000ffff, /* dst_mask */
1216 FALSE
), /* pcrel_offset */
1218 /* Get the highest value of a 64 bit addend. */
1219 HOWTO (R_MIPS_HIGHEST
, /* type */
1221 2, /* size (0 = byte, 1 = short, 2 = long) */
1223 FALSE
, /* pc_relative */
1225 complain_overflow_dont
, /* complain_on_overflow */
1226 _bfd_mips_elf_generic_reloc
, /* special_function */
1227 "R_MIPS_HIGHEST", /* name */
1228 FALSE
, /* partial_inplace */
1230 0x0000ffff, /* dst_mask */
1231 FALSE
), /* pcrel_offset */
1233 /* High 16 bits of displacement in global offset table. */
1234 HOWTO (R_MIPS_CALL_HI16
, /* type */
1236 2, /* size (0 = byte, 1 = short, 2 = long) */
1238 FALSE
, /* pc_relative */
1240 complain_overflow_dont
, /* complain_on_overflow */
1241 _bfd_mips_elf_generic_reloc
, /* special_function */
1242 "R_MIPS_CALL_HI16", /* name */
1243 FALSE
, /* partial_inplace */
1245 0x0000ffff, /* dst_mask */
1246 FALSE
), /* pcrel_offset */
1248 /* Low 16 bits of displacement in global offset table. */
1249 HOWTO (R_MIPS_CALL_LO16
, /* type */
1251 2, /* size (0 = byte, 1 = short, 2 = long) */
1253 FALSE
, /* pc_relative */
1255 complain_overflow_dont
, /* complain_on_overflow */
1256 _bfd_mips_elf_generic_reloc
, /* special_function */
1257 "R_MIPS_CALL_LO16", /* name */
1258 FALSE
, /* partial_inplace */
1260 0x0000ffff, /* dst_mask */
1261 FALSE
), /* pcrel_offset */
1263 /* Section displacement, used by an associated event location section. */
1264 HOWTO (R_MIPS_SCN_DISP
, /* type */
1266 2, /* size (0 = byte, 1 = short, 2 = long) */
1268 FALSE
, /* pc_relative */
1270 complain_overflow_dont
, /* complain_on_overflow */
1271 _bfd_mips_elf_generic_reloc
, /* special_function */
1272 "R_MIPS_SCN_DISP", /* name */
1273 FALSE
, /* partial_inplace */
1275 0xffffffff, /* dst_mask */
1276 FALSE
), /* pcrel_offset */
1278 HOWTO (R_MIPS_REL16
, /* type */
1280 1, /* size (0 = byte, 1 = short, 2 = long) */
1282 FALSE
, /* pc_relative */
1284 complain_overflow_signed
, /* complain_on_overflow */
1285 _bfd_mips_elf_generic_reloc
, /* special_function */
1286 "R_MIPS_REL16", /* name */
1287 FALSE
, /* partial_inplace */
1289 0xffff, /* dst_mask */
1290 FALSE
), /* pcrel_offset */
1292 /* These two are obsolete. */
1293 EMPTY_HOWTO (R_MIPS_ADD_IMMEDIATE
),
1294 EMPTY_HOWTO (R_MIPS_PJUMP
),
1296 /* Similiar to R_MIPS_REL32, but used for relocations in a GOT section.
1297 It must be used for multigot GOT's (and only there). */
1298 HOWTO (R_MIPS_RELGOT
, /* type */
1300 2, /* size (0 = byte, 1 = short, 2 = long) */
1302 FALSE
, /* pc_relative */
1304 complain_overflow_dont
, /* complain_on_overflow */
1305 _bfd_mips_elf_generic_reloc
, /* special_function */
1306 "R_MIPS_RELGOT", /* name */
1307 FALSE
, /* partial_inplace */
1309 0xffffffff, /* dst_mask */
1310 FALSE
), /* pcrel_offset */
1312 /* Protected jump conversion. This is an optimization hint. No
1313 relocation is required for correctness. */
1314 HOWTO (R_MIPS_JALR
, /* type */
1316 2, /* size (0 = byte, 1 = short, 2 = long) */
1318 FALSE
, /* pc_relative */
1320 complain_overflow_dont
, /* complain_on_overflow */
1321 _bfd_mips_elf_generic_reloc
, /* special_function */
1322 "R_MIPS_JALR", /* name */
1323 FALSE
, /* partial_inplace */
1325 0x00000000, /* dst_mask */
1326 FALSE
), /* pcrel_offset */
1328 /* TLS relocations. */
1329 EMPTY_HOWTO (R_MIPS_TLS_DTPMOD32
),
1330 EMPTY_HOWTO (R_MIPS_TLS_DTPREL32
),
1332 HOWTO (R_MIPS_TLS_DTPMOD64
, /* type */
1334 4, /* size (0 = byte, 1 = short, 2 = long) */
1336 FALSE
, /* pc_relative */
1338 complain_overflow_dont
, /* complain_on_overflow */
1339 _bfd_mips_elf_generic_reloc
, /* special_function */
1340 "R_MIPS_TLS_DTPMOD64", /* name */
1341 TRUE
, /* partial_inplace */
1342 MINUS_ONE
, /* src_mask */
1343 MINUS_ONE
, /* dst_mask */
1344 FALSE
), /* pcrel_offset */
1346 HOWTO (R_MIPS_TLS_DTPREL64
, /* type */
1348 4, /* size (0 = byte, 1 = short, 2 = long) */
1350 FALSE
, /* pc_relative */
1352 complain_overflow_dont
, /* complain_on_overflow */
1353 _bfd_mips_elf_generic_reloc
, /* special_function */
1354 "R_MIPS_TLS_DTPREL64", /* name */
1355 TRUE
, /* partial_inplace */
1356 MINUS_ONE
, /* src_mask */
1357 MINUS_ONE
, /* dst_mask */
1358 FALSE
), /* pcrel_offset */
1360 /* TLS general dynamic variable reference. */
1361 HOWTO (R_MIPS_TLS_GD
, /* type */
1363 2, /* size (0 = byte, 1 = short, 2 = long) */
1365 FALSE
, /* pc_relative */
1367 complain_overflow_signed
, /* complain_on_overflow */
1368 _bfd_mips_elf_generic_reloc
, /* special_function */
1369 "R_MIPS_TLS_GD", /* name */
1370 TRUE
, /* partial_inplace */
1371 0x0000ffff, /* src_mask */
1372 0x0000ffff, /* dst_mask */
1373 FALSE
), /* pcrel_offset */
1375 /* TLS local dynamic variable reference. */
1376 HOWTO (R_MIPS_TLS_LDM
, /* type */
1378 2, /* size (0 = byte, 1 = short, 2 = long) */
1380 FALSE
, /* pc_relative */
1382 complain_overflow_signed
, /* complain_on_overflow */
1383 _bfd_mips_elf_generic_reloc
, /* special_function */
1384 "R_MIPS_TLS_LDM", /* name */
1385 TRUE
, /* partial_inplace */
1386 0x0000ffff, /* src_mask */
1387 0x0000ffff, /* dst_mask */
1388 FALSE
), /* pcrel_offset */
1390 /* TLS local dynamic offset. */
1391 HOWTO (R_MIPS_TLS_DTPREL_HI16
, /* type */
1393 2, /* size (0 = byte, 1 = short, 2 = long) */
1395 FALSE
, /* pc_relative */
1397 complain_overflow_signed
, /* complain_on_overflow */
1398 _bfd_mips_elf_generic_reloc
, /* special_function */
1399 "R_MIPS_TLS_DTPREL_HI16", /* name */
1400 TRUE
, /* partial_inplace */
1401 0x0000ffff, /* src_mask */
1402 0x0000ffff, /* dst_mask */
1403 FALSE
), /* pcrel_offset */
1405 /* TLS local dynamic offset. */
1406 HOWTO (R_MIPS_TLS_DTPREL_LO16
, /* type */
1408 2, /* size (0 = byte, 1 = short, 2 = long) */
1410 FALSE
, /* pc_relative */
1412 complain_overflow_signed
, /* complain_on_overflow */
1413 _bfd_mips_elf_generic_reloc
, /* special_function */
1414 "R_MIPS_TLS_DTPREL_LO16", /* name */
1415 TRUE
, /* partial_inplace */
1416 0x0000ffff, /* src_mask */
1417 0x0000ffff, /* dst_mask */
1418 FALSE
), /* pcrel_offset */
1420 /* TLS thread pointer offset. */
1421 HOWTO (R_MIPS_TLS_GOTTPREL
, /* type */
1423 2, /* size (0 = byte, 1 = short, 2 = long) */
1425 FALSE
, /* pc_relative */
1427 complain_overflow_signed
, /* complain_on_overflow */
1428 _bfd_mips_elf_generic_reloc
, /* special_function */
1429 "R_MIPS_TLS_GOTTPREL", /* name */
1430 TRUE
, /* partial_inplace */
1431 0x0000ffff, /* src_mask */
1432 0x0000ffff, /* dst_mask */
1433 FALSE
), /* pcrel_offset */
1435 /* TLS IE dynamic relocations. */
1436 EMPTY_HOWTO (R_MIPS_TLS_TPREL32
),
1438 HOWTO (R_MIPS_TLS_TPREL64
, /* type */
1440 4, /* size (0 = byte, 1 = short, 2 = long) */
1442 FALSE
, /* pc_relative */
1444 complain_overflow_dont
, /* complain_on_overflow */
1445 _bfd_mips_elf_generic_reloc
, /* special_function */
1446 "R_MIPS_TLS_TPREL64", /* name */
1447 TRUE
, /* partial_inplace */
1448 MINUS_ONE
, /* src_mask */
1449 MINUS_ONE
, /* dst_mask */
1450 FALSE
), /* pcrel_offset */
1452 /* TLS thread pointer offset. */
1453 HOWTO (R_MIPS_TLS_TPREL_HI16
, /* type */
1455 2, /* size (0 = byte, 1 = short, 2 = long) */
1457 FALSE
, /* pc_relative */
1459 complain_overflow_signed
, /* complain_on_overflow */
1460 _bfd_mips_elf_generic_reloc
, /* special_function */
1461 "R_MIPS_TLS_TPREL_HI16", /* name */
1462 TRUE
, /* partial_inplace */
1463 0x0000ffff, /* src_mask */
1464 0x0000ffff, /* dst_mask */
1465 FALSE
), /* pcrel_offset */
1467 /* TLS thread pointer offset. */
1468 HOWTO (R_MIPS_TLS_TPREL_LO16
, /* type */
1470 2, /* size (0 = byte, 1 = short, 2 = long) */
1472 FALSE
, /* pc_relative */
1474 complain_overflow_signed
, /* complain_on_overflow */
1475 _bfd_mips_elf_generic_reloc
, /* special_function */
1476 "R_MIPS_TLS_TPREL_LO16", /* name */
1477 TRUE
, /* partial_inplace */
1478 0x0000ffff, /* src_mask */
1479 0x0000ffff, /* dst_mask */
1480 FALSE
), /* pcrel_offset */
1482 /* 32 bit relocation with no addend. */
1483 HOWTO (R_MIPS_GLOB_DAT
, /* type */
1485 2, /* size (0 = byte, 1 = short, 2 = long) */
1487 FALSE
, /* pc_relative */
1489 complain_overflow_dont
, /* complain_on_overflow */
1490 _bfd_mips_elf_generic_reloc
, /* special_function */
1491 "R_MIPS_GLOB_DAT", /* name */
1492 FALSE
, /* partial_inplace */
1494 0xffffffff, /* dst_mask */
1495 FALSE
), /* pcrel_offset */
1498 static reloc_howto_type mips16_elf64_howto_table_rel
[] =
1500 /* The reloc used for the mips16 jump instruction. */
1501 HOWTO (R_MIPS16_26
, /* type */
1503 2, /* size (0 = byte, 1 = short, 2 = long) */
1505 FALSE
, /* pc_relative */
1507 complain_overflow_dont
, /* complain_on_overflow */
1508 /* This needs complex overflow
1509 detection, because the upper four
1510 bits must match the PC. */
1511 _bfd_mips_elf_generic_reloc
, /* special_function */
1512 "R_MIPS16_26", /* name */
1513 TRUE
, /* partial_inplace */
1514 0x3ffffff, /* src_mask */
1515 0x3ffffff, /* dst_mask */
1516 FALSE
), /* pcrel_offset */
1518 /* The reloc used for the mips16 gprel instruction. */
1519 HOWTO (R_MIPS16_GPREL
, /* type */
1521 2, /* size (0 = byte, 1 = short, 2 = long) */
1523 FALSE
, /* pc_relative */
1525 complain_overflow_signed
, /* complain_on_overflow */
1526 mips16_gprel_reloc
, /* special_function */
1527 "R_MIPS16_GPREL", /* name */
1528 TRUE
, /* partial_inplace */
1529 0x0000ffff, /* src_mask */
1530 0x0000ffff, /* dst_mask */
1531 FALSE
), /* pcrel_offset */
1533 /* A MIPS16 reference to the global offset table. */
1534 HOWTO (R_MIPS16_GOT16
, /* type */
1536 2, /* size (0 = byte, 1 = short, 2 = long) */
1538 FALSE
, /* pc_relative */
1540 complain_overflow_dont
, /* complain_on_overflow */
1541 _bfd_mips_elf_got16_reloc
, /* special_function */
1542 "R_MIPS16_GOT16", /* name */
1543 TRUE
, /* partial_inplace */
1544 0x0000ffff, /* src_mask */
1545 0x0000ffff, /* dst_mask */
1546 FALSE
), /* pcrel_offset */
1548 /* A MIPS16 call through the global offset table. */
1549 HOWTO (R_MIPS16_CALL16
, /* type */
1551 2, /* size (0 = byte, 1 = short, 2 = long) */
1553 FALSE
, /* pc_relative */
1555 complain_overflow_dont
, /* complain_on_overflow */
1556 _bfd_mips_elf_generic_reloc
, /* special_function */
1557 "R_MIPS16_CALL16", /* name */
1558 TRUE
, /* partial_inplace */
1559 0x0000ffff, /* src_mask */
1560 0x0000ffff, /* dst_mask */
1561 FALSE
), /* pcrel_offset */
1563 /* MIPS16 high 16 bits of symbol value. */
1564 HOWTO (R_MIPS16_HI16
, /* type */
1565 16, /* rightshift */
1566 2, /* size (0 = byte, 1 = short, 2 = long) */
1568 FALSE
, /* pc_relative */
1570 complain_overflow_dont
, /* complain_on_overflow */
1571 _bfd_mips_elf_hi16_reloc
, /* special_function */
1572 "R_MIPS16_HI16", /* name */
1573 TRUE
, /* partial_inplace */
1574 0x0000ffff, /* src_mask */
1575 0x0000ffff, /* dst_mask */
1576 FALSE
), /* pcrel_offset */
1578 /* MIPS16 low 16 bits of symbol value. */
1579 HOWTO (R_MIPS16_LO16
, /* type */
1581 2, /* size (0 = byte, 1 = short, 2 = long) */
1583 FALSE
, /* pc_relative */
1585 complain_overflow_dont
, /* complain_on_overflow */
1586 _bfd_mips_elf_lo16_reloc
, /* special_function */
1587 "R_MIPS16_LO16", /* name */
1588 TRUE
, /* partial_inplace */
1589 0x0000ffff, /* src_mask */
1590 0x0000ffff, /* dst_mask */
1591 FALSE
), /* pcrel_offset */
1594 static reloc_howto_type mips16_elf64_howto_table_rela
[] =
1596 /* The reloc used for the mips16 jump instruction. */
1597 HOWTO (R_MIPS16_26
, /* type */
1599 2, /* size (0 = byte, 1 = short, 2 = long) */
1601 FALSE
, /* pc_relative */
1603 complain_overflow_dont
, /* complain_on_overflow */
1604 /* This needs complex overflow
1605 detection, because the upper four
1606 bits must match the PC. */
1607 _bfd_mips_elf_generic_reloc
, /* special_function */
1608 "R_MIPS16_26", /* name */
1609 FALSE
, /* partial_inplace */
1610 0x3ffffff, /* src_mask */
1611 0x3ffffff, /* dst_mask */
1612 FALSE
), /* pcrel_offset */
1614 /* The reloc used for the mips16 gprel instruction. */
1615 HOWTO (R_MIPS16_GPREL
, /* type */
1617 2, /* size (0 = byte, 1 = short, 2 = long) */
1619 FALSE
, /* pc_relative */
1621 complain_overflow_signed
, /* complain_on_overflow */
1622 mips16_gprel_reloc
, /* special_function */
1623 "R_MIPS16_GPREL", /* name */
1624 FALSE
, /* partial_inplace */
1625 0x0000ffff, /* src_mask */
1626 0x0000ffff, /* dst_mask */
1627 FALSE
), /* pcrel_offset */
1629 /* A MIPS16 reference to the global offset table. */
1630 HOWTO (R_MIPS16_GOT16
, /* type */
1632 2, /* size (0 = byte, 1 = short, 2 = long) */
1634 FALSE
, /* pc_relative */
1636 complain_overflow_dont
, /* complain_on_overflow */
1637 _bfd_mips_elf_got16_reloc
, /* special_function */
1638 "R_MIPS16_GOT16", /* name */
1639 FALSE
, /* partial_inplace */
1640 0x0000ffff, /* src_mask */
1641 0x0000ffff, /* dst_mask */
1642 FALSE
), /* pcrel_offset */
1644 /* A MIPS16 call through the global offset table. */
1645 HOWTO (R_MIPS16_CALL16
, /* type */
1647 2, /* size (0 = byte, 1 = short, 2 = long) */
1649 FALSE
, /* pc_relative */
1651 complain_overflow_dont
, /* complain_on_overflow */
1652 _bfd_mips_elf_generic_reloc
, /* special_function */
1653 "R_MIPS16_CALL16", /* name */
1654 FALSE
, /* partial_inplace */
1655 0x0000ffff, /* src_mask */
1656 0x0000ffff, /* dst_mask */
1657 FALSE
), /* pcrel_offset */
1659 /* MIPS16 high 16 bits of symbol value. */
1660 HOWTO (R_MIPS16_HI16
, /* type */
1661 16, /* rightshift */
1662 2, /* size (0 = byte, 1 = short, 2 = long) */
1664 FALSE
, /* pc_relative */
1666 complain_overflow_dont
, /* complain_on_overflow */
1667 _bfd_mips_elf_hi16_reloc
, /* special_function */
1668 "R_MIPS16_HI16", /* name */
1669 FALSE
, /* partial_inplace */
1670 0x0000ffff, /* src_mask */
1671 0x0000ffff, /* dst_mask */
1672 FALSE
), /* pcrel_offset */
1674 /* MIPS16 low 16 bits of symbol value. */
1675 HOWTO (R_MIPS16_LO16
, /* type */
1677 2, /* size (0 = byte, 1 = short, 2 = long) */
1679 FALSE
, /* pc_relative */
1681 complain_overflow_dont
, /* complain_on_overflow */
1682 _bfd_mips_elf_lo16_reloc
, /* special_function */
1683 "R_MIPS16_LO16", /* name */
1684 FALSE
, /* partial_inplace */
1685 0x0000ffff, /* src_mask */
1686 0x0000ffff, /* dst_mask */
1687 FALSE
), /* pcrel_offset */
1690 /* GNU extension to record C++ vtable hierarchy */
1691 static reloc_howto_type elf_mips_gnu_vtinherit_howto
=
1692 HOWTO (R_MIPS_GNU_VTINHERIT
, /* type */
1694 2, /* size (0 = byte, 1 = short, 2 = long) */
1696 FALSE
, /* pc_relative */
1698 complain_overflow_dont
, /* complain_on_overflow */
1699 NULL
, /* special_function */
1700 "R_MIPS_GNU_VTINHERIT", /* name */
1701 FALSE
, /* partial_inplace */
1704 FALSE
); /* pcrel_offset */
1706 /* GNU extension to record C++ vtable member usage */
1707 static reloc_howto_type elf_mips_gnu_vtentry_howto
=
1708 HOWTO (R_MIPS_GNU_VTENTRY
, /* type */
1710 2, /* size (0 = byte, 1 = short, 2 = long) */
1712 FALSE
, /* pc_relative */
1714 complain_overflow_dont
, /* complain_on_overflow */
1715 _bfd_elf_rel_vtable_reloc_fn
, /* special_function */
1716 "R_MIPS_GNU_VTENTRY", /* name */
1717 FALSE
, /* partial_inplace */
1720 FALSE
); /* pcrel_offset */
1722 /* 16 bit offset for pc-relative branches. */
1723 static reloc_howto_type elf_mips_gnu_rel16_s2
=
1724 HOWTO (R_MIPS_GNU_REL16_S2
, /* type */
1726 2, /* size (0 = byte, 1 = short, 2 = long) */
1728 TRUE
, /* pc_relative */
1730 complain_overflow_signed
, /* complain_on_overflow */
1731 _bfd_mips_elf_generic_reloc
, /* special_function */
1732 "R_MIPS_GNU_REL16_S2", /* name */
1733 TRUE
, /* partial_inplace */
1734 0x0000ffff, /* src_mask */
1735 0x0000ffff, /* dst_mask */
1736 TRUE
); /* pcrel_offset */
1738 /* 16 bit offset for pc-relative branches. */
1739 static reloc_howto_type elf_mips_gnu_rela16_s2
=
1740 HOWTO (R_MIPS_GNU_REL16_S2
, /* type */
1742 2, /* size (0 = byte, 1 = short, 2 = long) */
1744 TRUE
, /* pc_relative */
1746 complain_overflow_signed
, /* complain_on_overflow */
1747 _bfd_mips_elf_generic_reloc
, /* special_function */
1748 "R_MIPS_GNU_REL16_S2", /* name */
1749 FALSE
, /* partial_inplace */
1751 0x0000ffff, /* dst_mask */
1752 TRUE
); /* pcrel_offset */
1754 /* Originally a VxWorks extension, but now used for other systems too. */
1755 static reloc_howto_type elf_mips_copy_howto
=
1756 HOWTO (R_MIPS_COPY
, /* type */
1758 0, /* this one is variable size */
1760 FALSE
, /* pc_relative */
1762 complain_overflow_bitfield
, /* complain_on_overflow */
1763 bfd_elf_generic_reloc
, /* special_function */
1764 "R_MIPS_COPY", /* name */
1765 FALSE
, /* partial_inplace */
1768 FALSE
); /* pcrel_offset */
1770 /* Originally a VxWorks extension, but now used for other systems too. */
1771 static reloc_howto_type elf_mips_jump_slot_howto
=
1772 HOWTO (R_MIPS_JUMP_SLOT
, /* type */
1774 4, /* size (0 = byte, 1 = short, 2 = long) */
1776 FALSE
, /* pc_relative */
1778 complain_overflow_bitfield
, /* complain_on_overflow */
1779 bfd_elf_generic_reloc
, /* special_function */
1780 "R_MIPS_JUMP_SLOT", /* name */
1781 FALSE
, /* partial_inplace */
1784 FALSE
); /* pcrel_offset */
1786 /* Swap in a MIPS 64-bit Rel reloc. */
1789 mips_elf64_swap_reloc_in (bfd
*abfd
, const Elf64_Mips_External_Rel
*src
,
1790 Elf64_Mips_Internal_Rela
*dst
)
1792 dst
->r_offset
= H_GET_64 (abfd
, src
->r_offset
);
1793 dst
->r_sym
= H_GET_32 (abfd
, src
->r_sym
);
1794 dst
->r_ssym
= H_GET_8 (abfd
, src
->r_ssym
);
1795 dst
->r_type3
= H_GET_8 (abfd
, src
->r_type3
);
1796 dst
->r_type2
= H_GET_8 (abfd
, src
->r_type2
);
1797 dst
->r_type
= H_GET_8 (abfd
, src
->r_type
);
1801 /* Swap in a MIPS 64-bit Rela reloc. */
1804 mips_elf64_swap_reloca_in (bfd
*abfd
, const Elf64_Mips_External_Rela
*src
,
1805 Elf64_Mips_Internal_Rela
*dst
)
1807 dst
->r_offset
= H_GET_64 (abfd
, src
->r_offset
);
1808 dst
->r_sym
= H_GET_32 (abfd
, src
->r_sym
);
1809 dst
->r_ssym
= H_GET_8 (abfd
, src
->r_ssym
);
1810 dst
->r_type3
= H_GET_8 (abfd
, src
->r_type3
);
1811 dst
->r_type2
= H_GET_8 (abfd
, src
->r_type2
);
1812 dst
->r_type
= H_GET_8 (abfd
, src
->r_type
);
1813 dst
->r_addend
= H_GET_S64 (abfd
, src
->r_addend
);
1816 /* Swap out a MIPS 64-bit Rel reloc. */
1819 mips_elf64_swap_reloc_out (bfd
*abfd
, const Elf64_Mips_Internal_Rela
*src
,
1820 Elf64_Mips_External_Rel
*dst
)
1822 H_PUT_64 (abfd
, src
->r_offset
, dst
->r_offset
);
1823 H_PUT_32 (abfd
, src
->r_sym
, dst
->r_sym
);
1824 H_PUT_8 (abfd
, src
->r_ssym
, dst
->r_ssym
);
1825 H_PUT_8 (abfd
, src
->r_type3
, dst
->r_type3
);
1826 H_PUT_8 (abfd
, src
->r_type2
, dst
->r_type2
);
1827 H_PUT_8 (abfd
, src
->r_type
, dst
->r_type
);
1830 /* Swap out a MIPS 64-bit Rela reloc. */
1833 mips_elf64_swap_reloca_out (bfd
*abfd
, const Elf64_Mips_Internal_Rela
*src
,
1834 Elf64_Mips_External_Rela
*dst
)
1836 H_PUT_64 (abfd
, src
->r_offset
, dst
->r_offset
);
1837 H_PUT_32 (abfd
, src
->r_sym
, dst
->r_sym
);
1838 H_PUT_8 (abfd
, src
->r_ssym
, dst
->r_ssym
);
1839 H_PUT_8 (abfd
, src
->r_type3
, dst
->r_type3
);
1840 H_PUT_8 (abfd
, src
->r_type2
, dst
->r_type2
);
1841 H_PUT_8 (abfd
, src
->r_type
, dst
->r_type
);
1842 H_PUT_S64 (abfd
, src
->r_addend
, dst
->r_addend
);
1845 /* Swap in a MIPS 64-bit Rel reloc. */
1848 mips_elf64_be_swap_reloc_in (bfd
*abfd
, const bfd_byte
*src
,
1849 Elf_Internal_Rela
*dst
)
1851 Elf64_Mips_Internal_Rela mirel
;
1853 mips_elf64_swap_reloc_in (abfd
,
1854 (const Elf64_Mips_External_Rel
*) src
,
1857 dst
[0].r_offset
= mirel
.r_offset
;
1858 dst
[0].r_info
= ELF64_R_INFO (mirel
.r_sym
, mirel
.r_type
);
1859 dst
[0].r_addend
= 0;
1860 dst
[1].r_offset
= mirel
.r_offset
;
1861 dst
[1].r_info
= ELF64_R_INFO (mirel
.r_ssym
, mirel
.r_type2
);
1862 dst
[1].r_addend
= 0;
1863 dst
[2].r_offset
= mirel
.r_offset
;
1864 dst
[2].r_info
= ELF64_R_INFO (STN_UNDEF
, mirel
.r_type3
);
1865 dst
[2].r_addend
= 0;
1868 /* Swap in a MIPS 64-bit Rela reloc. */
1871 mips_elf64_be_swap_reloca_in (bfd
*abfd
, const bfd_byte
*src
,
1872 Elf_Internal_Rela
*dst
)
1874 Elf64_Mips_Internal_Rela mirela
;
1876 mips_elf64_swap_reloca_in (abfd
,
1877 (const Elf64_Mips_External_Rela
*) src
,
1880 dst
[0].r_offset
= mirela
.r_offset
;
1881 dst
[0].r_info
= ELF64_R_INFO (mirela
.r_sym
, mirela
.r_type
);
1882 dst
[0].r_addend
= mirela
.r_addend
;
1883 dst
[1].r_offset
= mirela
.r_offset
;
1884 dst
[1].r_info
= ELF64_R_INFO (mirela
.r_ssym
, mirela
.r_type2
);
1885 dst
[1].r_addend
= 0;
1886 dst
[2].r_offset
= mirela
.r_offset
;
1887 dst
[2].r_info
= ELF64_R_INFO (STN_UNDEF
, mirela
.r_type3
);
1888 dst
[2].r_addend
= 0;
1891 /* Swap out a MIPS 64-bit Rel reloc. */
1894 mips_elf64_be_swap_reloc_out (bfd
*abfd
, const Elf_Internal_Rela
*src
,
1897 Elf64_Mips_Internal_Rela mirel
;
1899 mirel
.r_offset
= src
[0].r_offset
;
1900 BFD_ASSERT(src
[0].r_offset
== src
[1].r_offset
);
1902 mirel
.r_type
= ELF64_MIPS_R_TYPE (src
[0].r_info
);
1903 mirel
.r_sym
= ELF64_R_SYM (src
[0].r_info
);
1904 mirel
.r_type2
= ELF64_MIPS_R_TYPE (src
[1].r_info
);
1905 mirel
.r_ssym
= ELF64_MIPS_R_SSYM (src
[1].r_info
);
1906 mirel
.r_type3
= ELF64_MIPS_R_TYPE (src
[2].r_info
);
1908 mips_elf64_swap_reloc_out (abfd
, &mirel
,
1909 (Elf64_Mips_External_Rel
*) dst
);
1912 /* Swap out a MIPS 64-bit Rela reloc. */
1915 mips_elf64_be_swap_reloca_out (bfd
*abfd
, const Elf_Internal_Rela
*src
,
1918 Elf64_Mips_Internal_Rela mirela
;
1920 mirela
.r_offset
= src
[0].r_offset
;
1921 BFD_ASSERT(src
[0].r_offset
== src
[1].r_offset
);
1922 BFD_ASSERT(src
[0].r_offset
== src
[2].r_offset
);
1924 mirela
.r_type
= ELF64_MIPS_R_TYPE (src
[0].r_info
);
1925 mirela
.r_sym
= ELF64_R_SYM (src
[0].r_info
);
1926 mirela
.r_addend
= src
[0].r_addend
;
1927 BFD_ASSERT(src
[1].r_addend
== 0);
1928 BFD_ASSERT(src
[2].r_addend
== 0);
1930 mirela
.r_type2
= ELF64_MIPS_R_TYPE (src
[1].r_info
);
1931 mirela
.r_ssym
= ELF64_MIPS_R_SSYM (src
[1].r_info
);
1932 mirela
.r_type3
= ELF64_MIPS_R_TYPE (src
[2].r_info
);
1934 mips_elf64_swap_reloca_out (abfd
, &mirela
,
1935 (Elf64_Mips_External_Rela
*) dst
);
1938 /* Set the GP value for OUTPUT_BFD. Returns FALSE if this is a
1939 dangerous relocation. */
1942 mips_elf64_assign_gp (bfd
*output_bfd
, bfd_vma
*pgp
)
1948 /* If we've already figured out what GP will be, just return it. */
1949 *pgp
= _bfd_get_gp_value (output_bfd
);
1953 count
= bfd_get_symcount (output_bfd
);
1954 sym
= bfd_get_outsymbols (output_bfd
);
1956 /* The linker script will have created a symbol named `_gp' with the
1957 appropriate value. */
1962 for (i
= 0; i
< count
; i
++, sym
++)
1964 register const char *name
;
1966 name
= bfd_asymbol_name (*sym
);
1967 if (*name
== '_' && strcmp (name
, "_gp") == 0)
1969 *pgp
= bfd_asymbol_value (*sym
);
1970 _bfd_set_gp_value (output_bfd
, *pgp
);
1978 /* Only get the error once. */
1980 _bfd_set_gp_value (output_bfd
, *pgp
);
1987 /* We have to figure out the gp value, so that we can adjust the
1988 symbol value correctly. We look up the symbol _gp in the output
1989 BFD. If we can't find it, we're stuck. We cache it in the ELF
1990 target data. We don't need to adjust the symbol value for an
1991 external symbol if we are producing relocatable output. */
1993 static bfd_reloc_status_type
1994 mips_elf64_final_gp (bfd
*output_bfd
, asymbol
*symbol
, bfd_boolean relocatable
,
1995 char **error_message
, bfd_vma
*pgp
)
1997 if (bfd_is_und_section (symbol
->section
)
2001 return bfd_reloc_undefined
;
2004 *pgp
= _bfd_get_gp_value (output_bfd
);
2007 || (symbol
->flags
& BSF_SECTION_SYM
) != 0))
2011 /* Make up a value. */
2012 *pgp
= symbol
->section
->output_section
->vma
/*+ 0x4000*/;
2013 _bfd_set_gp_value (output_bfd
, *pgp
);
2015 else if (!mips_elf64_assign_gp (output_bfd
, pgp
))
2018 (char *) _("GP relative relocation when _gp not defined");
2019 return bfd_reloc_dangerous
;
2023 return bfd_reloc_ok
;
2026 /* Do a R_MIPS_GPREL16 relocation. This is a 16 bit value which must
2027 become the offset from the gp register. */
2029 static bfd_reloc_status_type
2030 mips_elf64_gprel16_reloc (bfd
*abfd
, arelent
*reloc_entry
, asymbol
*symbol
,
2031 void *data
, asection
*input_section
, bfd
*output_bfd
,
2032 char **error_message
)
2034 bfd_boolean relocatable
;
2035 bfd_reloc_status_type ret
;
2038 /* If we're relocating, and this is an external symbol, we don't want
2039 to change anything. */
2040 if (output_bfd
!= NULL
2041 && (symbol
->flags
& BSF_SECTION_SYM
) == 0
2042 && (symbol
->flags
& BSF_LOCAL
) != 0)
2044 reloc_entry
->address
+= input_section
->output_offset
;
2045 return bfd_reloc_ok
;
2048 if (output_bfd
!= NULL
)
2052 relocatable
= FALSE
;
2053 output_bfd
= symbol
->section
->output_section
->owner
;
2056 ret
= mips_elf64_final_gp (output_bfd
, symbol
, relocatable
, error_message
,
2058 if (ret
!= bfd_reloc_ok
)
2061 return _bfd_mips_elf_gprel16_with_gp (abfd
, symbol
, reloc_entry
,
2062 input_section
, relocatable
,
2066 /* Do a R_MIPS_LITERAL relocation. */
2068 static bfd_reloc_status_type
2069 mips_elf64_literal_reloc (bfd
*abfd
, arelent
*reloc_entry
, asymbol
*symbol
,
2070 void *data
, asection
*input_section
, bfd
*output_bfd
,
2071 char **error_message
)
2073 bfd_boolean relocatable
;
2074 bfd_reloc_status_type ret
;
2077 /* R_MIPS_LITERAL relocations are defined for local symbols only. */
2078 if (output_bfd
!= NULL
2079 && (symbol
->flags
& BSF_SECTION_SYM
) == 0
2080 && (symbol
->flags
& BSF_LOCAL
) != 0)
2082 *error_message
= (char *)
2083 _("literal relocation occurs for an external symbol");
2084 return bfd_reloc_outofrange
;
2087 /* FIXME: The entries in the .lit8 and .lit4 sections should be merged. */
2088 if (output_bfd
!= NULL
)
2092 relocatable
= FALSE
;
2093 output_bfd
= symbol
->section
->output_section
->owner
;
2096 ret
= mips_elf64_final_gp (output_bfd
, symbol
, relocatable
, error_message
,
2098 if (ret
!= bfd_reloc_ok
)
2101 return _bfd_mips_elf_gprel16_with_gp (abfd
, symbol
, reloc_entry
,
2102 input_section
, relocatable
,
2106 /* Do a R_MIPS_GPREL32 relocation. This is a 32 bit value which must
2107 become the offset from the gp register. */
2109 static bfd_reloc_status_type
2110 mips_elf64_gprel32_reloc (bfd
*abfd
, arelent
*reloc_entry
, asymbol
*symbol
,
2111 void *data
, asection
*input_section
, bfd
*output_bfd
,
2112 char **error_message
)
2114 bfd_boolean relocatable
;
2115 bfd_reloc_status_type ret
;
2120 /* R_MIPS_GPREL32 relocations are defined for local symbols only. */
2121 if (output_bfd
!= NULL
2122 && (symbol
->flags
& BSF_SECTION_SYM
) == 0
2123 && (symbol
->flags
& BSF_LOCAL
) != 0)
2125 *error_message
= (char *)
2126 _("32bits gp relative relocation occurs for an external symbol");
2127 return bfd_reloc_outofrange
;
2130 if (output_bfd
!= NULL
)
2134 relocatable
= FALSE
;
2135 output_bfd
= symbol
->section
->output_section
->owner
;
2138 ret
= mips_elf64_final_gp (output_bfd
, symbol
, relocatable
,
2139 error_message
, &gp
);
2140 if (ret
!= bfd_reloc_ok
)
2143 if (bfd_is_com_section (symbol
->section
))
2146 relocation
= symbol
->value
;
2148 relocation
+= symbol
->section
->output_section
->vma
;
2149 relocation
+= symbol
->section
->output_offset
;
2151 if (reloc_entry
->address
> bfd_get_section_limit (abfd
, input_section
))
2152 return bfd_reloc_outofrange
;
2154 /* Set val to the offset into the section or symbol. */
2155 val
= reloc_entry
->addend
;
2157 if (reloc_entry
->howto
->partial_inplace
)
2158 val
+= bfd_get_32 (abfd
, (bfd_byte
*) data
+ reloc_entry
->address
);
2160 /* Adjust val for the final section location and GP value. If we
2161 are producing relocatable output, we don't want to do this for
2162 an external symbol. */
2164 || (symbol
->flags
& BSF_SECTION_SYM
) != 0)
2165 val
+= relocation
- gp
;
2167 if (reloc_entry
->howto
->partial_inplace
)
2168 bfd_put_32 (abfd
, val
, (bfd_byte
*) data
+ reloc_entry
->address
);
2170 reloc_entry
->addend
= val
;
2173 reloc_entry
->address
+= input_section
->output_offset
;
2175 return bfd_reloc_ok
;
2178 /* Do a R_MIPS_SHIFT6 relocation. The MSB of the shift is stored at bit 2,
2179 the rest is at bits 6-10. The bitpos already got right by the howto. */
2181 static bfd_reloc_status_type
2182 mips_elf64_shift6_reloc (bfd
*abfd
, arelent
*reloc_entry
, asymbol
*symbol
,
2183 void *data
, asection
*input_section
, bfd
*output_bfd
,
2184 char **error_message
)
2186 if (reloc_entry
->howto
->partial_inplace
)
2188 reloc_entry
->addend
= ((reloc_entry
->addend
& 0x00007c0)
2189 | (reloc_entry
->addend
& 0x00000800) >> 9);
2192 return _bfd_mips_elf_generic_reloc (abfd
, reloc_entry
, symbol
, data
,
2193 input_section
, output_bfd
,
2197 /* Handle a mips16 GP relative reloc. */
2199 static bfd_reloc_status_type
2200 mips16_gprel_reloc (bfd
*abfd
, arelent
*reloc_entry
, asymbol
*symbol
,
2201 void *data
, asection
*input_section
, bfd
*output_bfd
,
2202 char **error_message
)
2204 bfd_boolean relocatable
;
2205 bfd_reloc_status_type ret
;
2209 /* If we're relocating, and this is an external symbol, we don't want
2210 to change anything. */
2211 if (output_bfd
!= NULL
2212 && (symbol
->flags
& BSF_SECTION_SYM
) == 0
2213 && (symbol
->flags
& BSF_LOCAL
) != 0)
2215 reloc_entry
->address
+= input_section
->output_offset
;
2216 return bfd_reloc_ok
;
2219 if (output_bfd
!= NULL
)
2223 relocatable
= FALSE
;
2224 output_bfd
= symbol
->section
->output_section
->owner
;
2227 ret
= mips_elf64_final_gp (output_bfd
, symbol
, relocatable
, error_message
,
2229 if (ret
!= bfd_reloc_ok
)
2232 location
= (bfd_byte
*) data
+ reloc_entry
->address
;
2233 _bfd_mips16_elf_reloc_unshuffle (abfd
, reloc_entry
->howto
->type
, FALSE
,
2235 ret
= _bfd_mips_elf_gprel16_with_gp (abfd
, symbol
, reloc_entry
,
2236 input_section
, relocatable
,
2238 _bfd_mips16_elf_reloc_shuffle (abfd
, reloc_entry
->howto
->type
, !relocatable
,
2244 /* A mapping from BFD reloc types to MIPS ELF reloc types. */
2246 struct elf_reloc_map
{
2247 bfd_reloc_code_real_type bfd_val
;
2248 enum elf_mips_reloc_type elf_val
;
2251 static const struct elf_reloc_map mips_reloc_map
[] =
2253 { BFD_RELOC_NONE
, R_MIPS_NONE
},
2254 { BFD_RELOC_16
, R_MIPS_16
},
2255 { BFD_RELOC_32
, R_MIPS_32
},
2256 /* There is no BFD reloc for R_MIPS_REL32. */
2257 { BFD_RELOC_64
, R_MIPS_64
},
2258 { BFD_RELOC_CTOR
, R_MIPS_64
},
2259 { BFD_RELOC_16_PCREL_S2
, R_MIPS_PC16
},
2260 { BFD_RELOC_HI16_S
, R_MIPS_HI16
},
2261 { BFD_RELOC_LO16
, R_MIPS_LO16
},
2262 { BFD_RELOC_GPREL16
, R_MIPS_GPREL16
},
2263 { BFD_RELOC_GPREL32
, R_MIPS_GPREL32
},
2264 { BFD_RELOC_MIPS_JMP
, R_MIPS_26
},
2265 { BFD_RELOC_MIPS_LITERAL
, R_MIPS_LITERAL
},
2266 { BFD_RELOC_MIPS_GOT16
, R_MIPS_GOT16
},
2267 { BFD_RELOC_MIPS_CALL16
, R_MIPS_CALL16
},
2268 { BFD_RELOC_MIPS_SHIFT5
, R_MIPS_SHIFT5
},
2269 { BFD_RELOC_MIPS_SHIFT6
, R_MIPS_SHIFT6
},
2270 { BFD_RELOC_MIPS_GOT_DISP
, R_MIPS_GOT_DISP
},
2271 { BFD_RELOC_MIPS_GOT_PAGE
, R_MIPS_GOT_PAGE
},
2272 { BFD_RELOC_MIPS_GOT_OFST
, R_MIPS_GOT_OFST
},
2273 { BFD_RELOC_MIPS_GOT_HI16
, R_MIPS_GOT_HI16
},
2274 { BFD_RELOC_MIPS_GOT_LO16
, R_MIPS_GOT_LO16
},
2275 { BFD_RELOC_MIPS_SUB
, R_MIPS_SUB
},
2276 { BFD_RELOC_MIPS_INSERT_A
, R_MIPS_INSERT_A
},
2277 { BFD_RELOC_MIPS_INSERT_B
, R_MIPS_INSERT_B
},
2278 { BFD_RELOC_MIPS_DELETE
, R_MIPS_DELETE
},
2279 { BFD_RELOC_MIPS_HIGHEST
, R_MIPS_HIGHEST
},
2280 { BFD_RELOC_MIPS_HIGHER
, R_MIPS_HIGHER
},
2281 { BFD_RELOC_MIPS_CALL_HI16
, R_MIPS_CALL_HI16
},
2282 { BFD_RELOC_MIPS_CALL_LO16
, R_MIPS_CALL_LO16
},
2283 { BFD_RELOC_MIPS_SCN_DISP
, R_MIPS_SCN_DISP
},
2284 { BFD_RELOC_MIPS_REL16
, R_MIPS_REL16
},
2285 /* Use of R_MIPS_ADD_IMMEDIATE and R_MIPS_PJUMP is deprecated. */
2286 { BFD_RELOC_MIPS_RELGOT
, R_MIPS_RELGOT
},
2287 { BFD_RELOC_MIPS_JALR
, R_MIPS_JALR
},
2288 { BFD_RELOC_MIPS_TLS_DTPMOD32
, R_MIPS_TLS_DTPMOD32
},
2289 { BFD_RELOC_MIPS_TLS_DTPREL32
, R_MIPS_TLS_DTPREL32
},
2290 { BFD_RELOC_MIPS_TLS_DTPMOD64
, R_MIPS_TLS_DTPMOD64
},
2291 { BFD_RELOC_MIPS_TLS_DTPREL64
, R_MIPS_TLS_DTPREL64
},
2292 { BFD_RELOC_MIPS_TLS_GD
, R_MIPS_TLS_GD
},
2293 { BFD_RELOC_MIPS_TLS_LDM
, R_MIPS_TLS_LDM
},
2294 { BFD_RELOC_MIPS_TLS_DTPREL_HI16
, R_MIPS_TLS_DTPREL_HI16
},
2295 { BFD_RELOC_MIPS_TLS_DTPREL_LO16
, R_MIPS_TLS_DTPREL_LO16
},
2296 { BFD_RELOC_MIPS_TLS_GOTTPREL
, R_MIPS_TLS_GOTTPREL
},
2297 { BFD_RELOC_MIPS_TLS_TPREL32
, R_MIPS_TLS_TPREL32
},
2298 { BFD_RELOC_MIPS_TLS_TPREL64
, R_MIPS_TLS_TPREL64
},
2299 { BFD_RELOC_MIPS_TLS_TPREL_HI16
, R_MIPS_TLS_TPREL_HI16
},
2300 { BFD_RELOC_MIPS_TLS_TPREL_LO16
, R_MIPS_TLS_TPREL_LO16
}
2303 static const struct elf_reloc_map mips16_reloc_map
[] =
2305 { BFD_RELOC_MIPS16_JMP
, R_MIPS16_26
- R_MIPS16_min
},
2306 { BFD_RELOC_MIPS16_GPREL
, R_MIPS16_GPREL
- R_MIPS16_min
},
2307 { BFD_RELOC_MIPS16_GOT16
, R_MIPS16_GOT16
- R_MIPS16_min
},
2308 { BFD_RELOC_MIPS16_CALL16
, R_MIPS16_CALL16
- R_MIPS16_min
},
2309 { BFD_RELOC_MIPS16_HI16_S
, R_MIPS16_HI16
- R_MIPS16_min
},
2310 { BFD_RELOC_MIPS16_LO16
, R_MIPS16_LO16
- R_MIPS16_min
},
2313 /* Given a BFD reloc type, return a howto structure. */
2315 static reloc_howto_type
*
2316 bfd_elf64_bfd_reloc_type_lookup (bfd
*abfd ATTRIBUTE_UNUSED
,
2317 bfd_reloc_code_real_type code
)
2320 /* FIXME: We default to RELA here instead of choosing the right
2321 relocation variant. */
2322 reloc_howto_type
*howto_table
= mips_elf64_howto_table_rela
;
2323 reloc_howto_type
*howto16_table
= mips16_elf64_howto_table_rela
;
2325 for (i
= 0; i
< sizeof (mips_reloc_map
) / sizeof (struct elf_reloc_map
);
2328 if (mips_reloc_map
[i
].bfd_val
== code
)
2329 return &howto_table
[(int) mips_reloc_map
[i
].elf_val
];
2332 for (i
= 0; i
< sizeof (mips16_reloc_map
) / sizeof (struct elf_reloc_map
);
2335 if (mips16_reloc_map
[i
].bfd_val
== code
)
2336 return &howto16_table
[(int) mips16_reloc_map
[i
].elf_val
];
2341 case BFD_RELOC_VTABLE_INHERIT
:
2342 return &elf_mips_gnu_vtinherit_howto
;
2343 case BFD_RELOC_VTABLE_ENTRY
:
2344 return &elf_mips_gnu_vtentry_howto
;
2345 case BFD_RELOC_MIPS_COPY
:
2346 return &elf_mips_copy_howto
;
2347 case BFD_RELOC_MIPS_JUMP_SLOT
:
2348 return &elf_mips_jump_slot_howto
;
2350 bfd_set_error (bfd_error_bad_value
);
2355 static reloc_howto_type
*
2356 bfd_elf64_bfd_reloc_name_lookup (bfd
*abfd ATTRIBUTE_UNUSED
,
2362 i
< (sizeof (mips_elf64_howto_table_rela
)
2363 / sizeof (mips_elf64_howto_table_rela
[0])); i
++)
2364 if (mips_elf64_howto_table_rela
[i
].name
!= NULL
2365 && strcasecmp (mips_elf64_howto_table_rela
[i
].name
, r_name
) == 0)
2366 return &mips_elf64_howto_table_rela
[i
];
2369 i
< (sizeof (mips16_elf64_howto_table_rela
)
2370 / sizeof (mips16_elf64_howto_table_rela
[0]));
2372 if (mips16_elf64_howto_table_rela
[i
].name
!= NULL
2373 && strcasecmp (mips16_elf64_howto_table_rela
[i
].name
, r_name
) == 0)
2374 return &mips16_elf64_howto_table_rela
[i
];
2376 if (strcasecmp (elf_mips_gnu_vtinherit_howto
.name
, r_name
) == 0)
2377 return &elf_mips_gnu_vtinherit_howto
;
2378 if (strcasecmp (elf_mips_gnu_vtentry_howto
.name
, r_name
) == 0)
2379 return &elf_mips_gnu_vtentry_howto
;
2380 if (strcasecmp (elf_mips_gnu_rel16_s2
.name
, r_name
) == 0)
2381 return &elf_mips_gnu_rel16_s2
;
2382 if (strcasecmp (elf_mips_gnu_rela16_s2
.name
, r_name
) == 0)
2383 return &elf_mips_gnu_rela16_s2
;
2384 if (strcasecmp (elf_mips_copy_howto
.name
, r_name
) == 0)
2385 return &elf_mips_copy_howto
;
2386 if (strcasecmp (elf_mips_jump_slot_howto
.name
, r_name
) == 0)
2387 return &elf_mips_jump_slot_howto
;
2392 /* Given a MIPS Elf_Internal_Rel, fill in an arelent structure. */
2394 static reloc_howto_type
*
2395 mips_elf64_rtype_to_howto (unsigned int r_type
, bfd_boolean rela_p
)
2399 case R_MIPS_GNU_VTINHERIT
:
2400 return &elf_mips_gnu_vtinherit_howto
;
2401 case R_MIPS_GNU_VTENTRY
:
2402 return &elf_mips_gnu_vtentry_howto
;
2403 case R_MIPS_GNU_REL16_S2
:
2405 return &elf_mips_gnu_rela16_s2
;
2407 return &elf_mips_gnu_rel16_s2
;
2409 return &elf_mips_copy_howto
;
2410 case R_MIPS_JUMP_SLOT
:
2411 return &elf_mips_jump_slot_howto
;
2413 if (r_type
>= R_MIPS16_min
&& r_type
< R_MIPS16_max
)
2416 return &mips16_elf64_howto_table_rela
[r_type
- R_MIPS16_min
];
2418 return &mips16_elf64_howto_table_rel
[r_type
- R_MIPS16_min
];
2420 BFD_ASSERT (r_type
< (unsigned int) R_MIPS_max
);
2422 return &mips_elf64_howto_table_rela
[r_type
];
2424 return &mips_elf64_howto_table_rel
[r_type
];
2429 /* Prevent relocation handling by bfd for MIPS ELF64. */
2432 mips_elf64_info_to_howto_rel (bfd
*abfd ATTRIBUTE_UNUSED
,
2433 arelent
*cache_ptr ATTRIBUTE_UNUSED
,
2434 Elf_Internal_Rela
*dst ATTRIBUTE_UNUSED
)
2440 mips_elf64_info_to_howto_rela (bfd
*abfd ATTRIBUTE_UNUSED
,
2441 arelent
*cache_ptr ATTRIBUTE_UNUSED
,
2442 Elf_Internal_Rela
*dst ATTRIBUTE_UNUSED
)
2447 /* Since each entry in an SHT_REL or SHT_RELA section can represent up
2448 to three relocs, we must tell the user to allocate more space. */
2451 mips_elf64_get_reloc_upper_bound (bfd
*abfd ATTRIBUTE_UNUSED
, asection
*sec
)
2453 return (sec
->reloc_count
* 3 + 1) * sizeof (arelent
*);
2457 mips_elf64_get_dynamic_reloc_upper_bound (bfd
*abfd
)
2459 return _bfd_elf_get_dynamic_reloc_upper_bound (abfd
) * 3;
2462 /* We must also copy more relocations than the corresponding functions
2463 in elf.c would, so the two following functions are slightly
2464 modified from elf.c, that multiply the external relocation count by
2465 3 to obtain the internal relocation count. */
2468 mips_elf64_canonicalize_reloc (bfd
*abfd
, sec_ptr section
,
2469 arelent
**relptr
, asymbol
**symbols
)
2473 const struct elf_backend_data
*bed
= get_elf_backend_data (abfd
);
2475 if (! bed
->s
->slurp_reloc_table (abfd
, section
, symbols
, FALSE
))
2478 tblptr
= section
->relocation
;
2479 for (i
= 0; i
< section
->reloc_count
* 3; i
++)
2480 *relptr
++ = tblptr
++;
2484 return section
->reloc_count
* 3;
2488 mips_elf64_canonicalize_dynamic_reloc (bfd
*abfd
, arelent
**storage
,
2491 bfd_boolean (*slurp_relocs
) (bfd
*, asection
*, asymbol
**, bfd_boolean
);
2495 if (elf_dynsymtab (abfd
) == 0)
2497 bfd_set_error (bfd_error_invalid_operation
);
2501 slurp_relocs
= get_elf_backend_data (abfd
)->s
->slurp_reloc_table
;
2503 for (s
= abfd
->sections
; s
!= NULL
; s
= s
->next
)
2505 if (elf_section_data (s
)->this_hdr
.sh_link
== elf_dynsymtab (abfd
)
2506 && (elf_section_data (s
)->this_hdr
.sh_type
== SHT_REL
2507 || elf_section_data (s
)->this_hdr
.sh_type
== SHT_RELA
))
2512 if (! (*slurp_relocs
) (abfd
, s
, syms
, TRUE
))
2514 count
= s
->size
/ elf_section_data (s
)->this_hdr
.sh_entsize
* 3;
2516 for (i
= 0; i
< count
; i
++)
2527 /* Read the relocations from one reloc section. This is mostly copied
2528 from elfcode.h, except for the changes to expand one external
2529 relocation to 3 internal ones. We must unfortunately set
2530 reloc_count to the number of external relocations, because a lot of
2531 generic code seems to depend on this. */
2534 mips_elf64_slurp_one_reloc_table (bfd
*abfd
, asection
*asect
,
2535 Elf_Internal_Shdr
*rel_hdr
,
2536 bfd_size_type reloc_count
,
2537 arelent
*relents
, asymbol
**symbols
,
2538 bfd_boolean dynamic
)
2541 bfd_byte
*native_relocs
;
2547 allocated
= bfd_malloc (rel_hdr
->sh_size
);
2548 if (allocated
== NULL
)
2551 if (bfd_seek (abfd
, rel_hdr
->sh_offset
, SEEK_SET
) != 0
2552 || (bfd_bread (allocated
, rel_hdr
->sh_size
, abfd
)
2553 != rel_hdr
->sh_size
))
2556 native_relocs
= allocated
;
2558 entsize
= rel_hdr
->sh_entsize
;
2559 BFD_ASSERT (entsize
== sizeof (Elf64_Mips_External_Rel
)
2560 || entsize
== sizeof (Elf64_Mips_External_Rela
));
2562 if (entsize
== sizeof (Elf64_Mips_External_Rel
))
2567 for (i
= 0, relent
= relents
;
2569 i
++, native_relocs
+= entsize
)
2571 Elf64_Mips_Internal_Rela rela
;
2572 bfd_boolean used_sym
, used_ssym
;
2575 if (entsize
== sizeof (Elf64_Mips_External_Rela
))
2576 mips_elf64_swap_reloca_in (abfd
,
2577 (Elf64_Mips_External_Rela
*) native_relocs
,
2580 mips_elf64_swap_reloc_in (abfd
,
2581 (Elf64_Mips_External_Rel
*) native_relocs
,
2584 /* Each entry represents exactly three actual relocations. */
2588 for (ir
= 0; ir
< 3; ir
++)
2590 enum elf_mips_reloc_type type
;
2597 type
= (enum elf_mips_reloc_type
) rela
.r_type
;
2600 type
= (enum elf_mips_reloc_type
) rela
.r_type2
;
2603 type
= (enum elf_mips_reloc_type
) rela
.r_type3
;
2607 /* Some types require symbols, whereas some do not. */
2611 case R_MIPS_LITERAL
:
2612 case R_MIPS_INSERT_A
:
2613 case R_MIPS_INSERT_B
:
2615 relent
->sym_ptr_ptr
= bfd_abs_section_ptr
->symbol_ptr_ptr
;
2621 if (rela
.r_sym
== 0)
2622 relent
->sym_ptr_ptr
= bfd_abs_section_ptr
->symbol_ptr_ptr
;
2627 ps
= symbols
+ rela
.r_sym
- 1;
2629 if ((s
->flags
& BSF_SECTION_SYM
) == 0)
2630 relent
->sym_ptr_ptr
= ps
;
2632 relent
->sym_ptr_ptr
= s
->section
->symbol_ptr_ptr
;
2637 else if (! used_ssym
)
2639 switch (rela
.r_ssym
)
2642 relent
->sym_ptr_ptr
=
2643 bfd_abs_section_ptr
->symbol_ptr_ptr
;
2649 /* FIXME: I think these need to be handled using
2650 special howto structures. */
2662 relent
->sym_ptr_ptr
= bfd_abs_section_ptr
->symbol_ptr_ptr
;
2667 /* The address of an ELF reloc is section relative for an
2668 object file, and absolute for an executable file or
2669 shared library. The address of a BFD reloc is always
2670 section relative. */
2671 if ((abfd
->flags
& (EXEC_P
| DYNAMIC
)) == 0 || dynamic
)
2672 relent
->address
= rela
.r_offset
;
2674 relent
->address
= rela
.r_offset
- asect
->vma
;
2676 relent
->addend
= rela
.r_addend
;
2678 relent
->howto
= mips_elf64_rtype_to_howto (type
, rela_p
);
2684 asect
->reloc_count
+= (relent
- relents
) / 3;
2686 if (allocated
!= NULL
)
2692 if (allocated
!= NULL
)
2697 /* Read the relocations. On Irix 6, there can be two reloc sections
2698 associated with a single data section. This is copied from
2699 elfcode.h as well, with changes as small as accounting for 3
2700 internal relocs per external reloc and resetting reloc_count to
2701 zero before processing the relocs of a section. */
2704 mips_elf64_slurp_reloc_table (bfd
*abfd
, asection
*asect
,
2705 asymbol
**symbols
, bfd_boolean dynamic
)
2707 struct bfd_elf_section_data
* const d
= elf_section_data (asect
);
2708 Elf_Internal_Shdr
*rel_hdr
;
2709 Elf_Internal_Shdr
*rel_hdr2
;
2710 bfd_size_type reloc_count
;
2711 bfd_size_type reloc_count2
;
2715 if (asect
->relocation
!= NULL
)
2720 if ((asect
->flags
& SEC_RELOC
) == 0
2721 || asect
->reloc_count
== 0)
2724 rel_hdr
= &d
->rel_hdr
;
2725 reloc_count
= NUM_SHDR_ENTRIES (rel_hdr
);
2726 rel_hdr2
= d
->rel_hdr2
;
2727 reloc_count2
= (rel_hdr2
? NUM_SHDR_ENTRIES (rel_hdr2
) : 0);
2729 BFD_ASSERT (asect
->reloc_count
== reloc_count
+ reloc_count2
);
2730 BFD_ASSERT (asect
->rel_filepos
== rel_hdr
->sh_offset
2731 || (rel_hdr2
&& asect
->rel_filepos
== rel_hdr2
->sh_offset
));
2736 /* Note that ASECT->RELOC_COUNT tends not to be accurate in this
2737 case because relocations against this section may use the
2738 dynamic symbol table, and in that case bfd_section_from_shdr
2739 in elf.c does not update the RELOC_COUNT. */
2740 if (asect
->size
== 0)
2743 rel_hdr
= &d
->this_hdr
;
2744 reloc_count
= NUM_SHDR_ENTRIES (rel_hdr
);
2749 /* Allocate space for 3 arelent structures for each Rel structure. */
2750 amt
= (reloc_count
+ reloc_count2
) * 3 * sizeof (arelent
);
2751 relents
= bfd_alloc (abfd
, amt
);
2752 if (relents
== NULL
)
2755 /* The slurp_one_reloc_table routine increments reloc_count. */
2756 asect
->reloc_count
= 0;
2758 if (! mips_elf64_slurp_one_reloc_table (abfd
, asect
,
2759 rel_hdr
, reloc_count
,
2763 if (d
->rel_hdr2
!= NULL
)
2765 if (! mips_elf64_slurp_one_reloc_table (abfd
, asect
,
2766 rel_hdr2
, reloc_count2
,
2767 relents
+ reloc_count
* 3,
2772 asect
->relocation
= relents
;
2776 /* Write out the relocations. */
2779 mips_elf64_write_relocs (bfd
*abfd
, asection
*sec
, void *data
)
2781 bfd_boolean
*failedp
= data
;
2783 Elf_Internal_Shdr
*rel_hdr
;
2786 /* If we have already failed, don't do anything. */
2790 if ((sec
->flags
& SEC_RELOC
) == 0)
2793 /* The linker backend writes the relocs out itself, and sets the
2794 reloc_count field to zero to inhibit writing them here. Also,
2795 sometimes the SEC_RELOC flag gets set even when there aren't any
2797 if (sec
->reloc_count
== 0)
2800 /* We can combine up to three relocs that refer to the same address
2801 if the latter relocs have no associated symbol. */
2803 for (idx
= 0; idx
< sec
->reloc_count
; idx
++)
2810 addr
= sec
->orelocation
[idx
]->address
;
2811 for (i
= 0; i
< 2; i
++)
2815 if (idx
+ 1 >= sec
->reloc_count
)
2817 r
= sec
->orelocation
[idx
+ 1];
2818 if (r
->address
!= addr
2819 || ! bfd_is_abs_section ((*r
->sym_ptr_ptr
)->section
)
2820 || (*r
->sym_ptr_ptr
)->value
!= 0)
2823 /* We can merge the reloc at IDX + 1 with the reloc at IDX. */
2829 rel_hdr
= &elf_section_data (sec
)->rel_hdr
;
2831 /* Do the actual relocation. */
2833 if (rel_hdr
->sh_entsize
== sizeof(Elf64_Mips_External_Rel
))
2834 mips_elf64_write_rel (abfd
, sec
, rel_hdr
, &count
, data
);
2835 else if (rel_hdr
->sh_entsize
== sizeof(Elf64_Mips_External_Rela
))
2836 mips_elf64_write_rela (abfd
, sec
, rel_hdr
, &count
, data
);
2842 mips_elf64_write_rel (bfd
*abfd
, asection
*sec
,
2843 Elf_Internal_Shdr
*rel_hdr
,
2844 int *count
, void *data
)
2846 bfd_boolean
*failedp
= data
;
2847 Elf64_Mips_External_Rel
*ext_rel
;
2849 asymbol
*last_sym
= 0;
2850 int last_sym_idx
= 0;
2852 rel_hdr
->sh_size
= rel_hdr
->sh_entsize
* *count
;
2853 rel_hdr
->contents
= bfd_alloc (abfd
, rel_hdr
->sh_size
);
2854 if (rel_hdr
->contents
== NULL
)
2860 ext_rel
= (Elf64_Mips_External_Rel
*) rel_hdr
->contents
;
2861 for (idx
= 0; idx
< sec
->reloc_count
; idx
++, ext_rel
++)
2864 Elf64_Mips_Internal_Rela int_rel
;
2869 ptr
= sec
->orelocation
[idx
];
2871 /* The address of an ELF reloc is section relative for an object
2872 file, and absolute for an executable file or shared library.
2873 The address of a BFD reloc is always section relative. */
2874 if ((abfd
->flags
& (EXEC_P
| DYNAMIC
)) == 0)
2875 int_rel
.r_offset
= ptr
->address
;
2877 int_rel
.r_offset
= ptr
->address
+ sec
->vma
;
2879 sym
= *ptr
->sym_ptr_ptr
;
2880 if (sym
== last_sym
)
2882 else if (bfd_is_abs_section (sym
->section
) && sym
->value
== 0)
2887 n
= _bfd_elf_symbol_from_bfd_symbol (abfd
, &sym
);
2897 int_rel
.r_ssym
= RSS_UNDEF
;
2899 if ((*ptr
->sym_ptr_ptr
)->the_bfd
->xvec
!= abfd
->xvec
2900 && ! _bfd_elf_validate_reloc (abfd
, ptr
))
2906 int_rel
.r_type
= ptr
->howto
->type
;
2907 int_rel
.r_type2
= (int) R_MIPS_NONE
;
2908 int_rel
.r_type3
= (int) R_MIPS_NONE
;
2910 for (i
= 0; i
< 2; i
++)
2914 if (idx
+ 1 >= sec
->reloc_count
)
2916 r
= sec
->orelocation
[idx
+ 1];
2917 if (r
->address
!= ptr
->address
2918 || ! bfd_is_abs_section ((*r
->sym_ptr_ptr
)->section
)
2919 || (*r
->sym_ptr_ptr
)->value
!= 0)
2922 /* We can merge the reloc at IDX + 1 with the reloc at IDX. */
2925 int_rel
.r_type2
= r
->howto
->type
;
2927 int_rel
.r_type3
= r
->howto
->type
;
2932 mips_elf64_swap_reloc_out (abfd
, &int_rel
, ext_rel
);
2935 BFD_ASSERT (ext_rel
- (Elf64_Mips_External_Rel
*) rel_hdr
->contents
2940 mips_elf64_write_rela (bfd
*abfd
, asection
*sec
,
2941 Elf_Internal_Shdr
*rela_hdr
,
2942 int *count
, void *data
)
2944 bfd_boolean
*failedp
= data
;
2945 Elf64_Mips_External_Rela
*ext_rela
;
2947 asymbol
*last_sym
= 0;
2948 int last_sym_idx
= 0;
2950 rela_hdr
->sh_size
= rela_hdr
->sh_entsize
* *count
;
2951 rela_hdr
->contents
= bfd_alloc (abfd
, rela_hdr
->sh_size
);
2952 if (rela_hdr
->contents
== NULL
)
2958 ext_rela
= (Elf64_Mips_External_Rela
*) rela_hdr
->contents
;
2959 for (idx
= 0; idx
< sec
->reloc_count
; idx
++, ext_rela
++)
2962 Elf64_Mips_Internal_Rela int_rela
;
2967 ptr
= sec
->orelocation
[idx
];
2969 /* The address of an ELF reloc is section relative for an object
2970 file, and absolute for an executable file or shared library.
2971 The address of a BFD reloc is always section relative. */
2972 if ((abfd
->flags
& (EXEC_P
| DYNAMIC
)) == 0)
2973 int_rela
.r_offset
= ptr
->address
;
2975 int_rela
.r_offset
= ptr
->address
+ sec
->vma
;
2977 sym
= *ptr
->sym_ptr_ptr
;
2978 if (sym
== last_sym
)
2980 else if (bfd_is_abs_section (sym
->section
) && sym
->value
== 0)
2985 n
= _bfd_elf_symbol_from_bfd_symbol (abfd
, &sym
);
2995 int_rela
.r_addend
= ptr
->addend
;
2996 int_rela
.r_ssym
= RSS_UNDEF
;
2998 if ((*ptr
->sym_ptr_ptr
)->the_bfd
->xvec
!= abfd
->xvec
2999 && ! _bfd_elf_validate_reloc (abfd
, ptr
))
3005 int_rela
.r_type
= ptr
->howto
->type
;
3006 int_rela
.r_type2
= (int) R_MIPS_NONE
;
3007 int_rela
.r_type3
= (int) R_MIPS_NONE
;
3009 for (i
= 0; i
< 2; i
++)
3013 if (idx
+ 1 >= sec
->reloc_count
)
3015 r
= sec
->orelocation
[idx
+ 1];
3016 if (r
->address
!= ptr
->address
3017 || ! bfd_is_abs_section ((*r
->sym_ptr_ptr
)->section
)
3018 || (*r
->sym_ptr_ptr
)->value
!= 0)
3021 /* We can merge the reloc at IDX + 1 with the reloc at IDX. */
3024 int_rela
.r_type2
= r
->howto
->type
;
3026 int_rela
.r_type3
= r
->howto
->type
;
3031 mips_elf64_swap_reloca_out (abfd
, &int_rela
, ext_rela
);
3034 BFD_ASSERT (ext_rela
- (Elf64_Mips_External_Rela
*) rela_hdr
->contents
3038 /* Set the right machine number for a MIPS ELF file. */
3041 mips_elf64_object_p (bfd
*abfd
)
3045 /* Irix 6 is broken. Object file symbol tables are not always
3046 sorted correctly such that local symbols precede global symbols,
3047 and the sh_info field in the symbol table is not always right. */
3048 if (elf64_mips_irix_compat (abfd
) != ict_none
)
3049 elf_bad_symtab (abfd
) = TRUE
;
3051 mach
= _bfd_elf_mips_mach (elf_elfheader (abfd
)->e_flags
);
3052 bfd_default_set_arch_mach (abfd
, bfd_arch_mips
, mach
);
3056 /* Depending on the target vector we generate some version of Irix
3057 executables or "normal" MIPS ELF ABI executables. */
3058 static irix_compat_t
3059 elf64_mips_irix_compat (bfd
*abfd
)
3061 if ((abfd
->xvec
== &bfd_elf64_bigmips_vec
)
3062 || (abfd
->xvec
== &bfd_elf64_littlemips_vec
))
3068 /* Support for core dump NOTE sections. */
3070 elf64_mips_grok_prstatus (bfd
*abfd
, Elf_Internal_Note
*note
)
3075 switch (note
->descsz
)
3080 case 480: /* Linux/MIPS - N64 kernel */
3082 elf_tdata (abfd
)->core_signal
= bfd_get_16 (abfd
, note
->descdata
+ 12);
3085 elf_tdata (abfd
)->core_pid
= bfd_get_32 (abfd
, note
->descdata
+ 32);
3094 /* Make a ".reg/999" section. */
3095 return _bfd_elfcore_make_pseudosection (abfd
, ".reg",
3096 size
, note
->descpos
+ offset
);
3100 elf64_mips_grok_psinfo (bfd
*abfd
, Elf_Internal_Note
*note
)
3102 switch (note
->descsz
)
3107 case 136: /* Linux/MIPS - N64 kernel elf_prpsinfo */
3108 elf_tdata (abfd
)->core_program
3109 = _bfd_elfcore_strndup (abfd
, note
->descdata
+ 40, 16);
3110 elf_tdata (abfd
)->core_command
3111 = _bfd_elfcore_strndup (abfd
, note
->descdata
+ 56, 80);
3114 /* Note that for some reason, a spurious space is tacked
3115 onto the end of the args in some (at least one anyway)
3116 implementations, so strip it off if it exists. */
3119 char *command
= elf_tdata (abfd
)->core_command
;
3120 int n
= strlen (command
);
3122 if (0 < n
&& command
[n
- 1] == ' ')
3123 command
[n
- 1] = '\0';
3129 /* ECOFF swapping routines. These are used when dealing with the
3130 .mdebug section, which is in the ECOFF debugging format. */
3131 static const struct ecoff_debug_swap mips_elf64_ecoff_debug_swap
=
3133 /* Symbol table magic number. */
3135 /* Alignment of debugging information. E.g., 4. */
3137 /* Sizes of external symbolic information. */
3138 sizeof (struct hdr_ext
),
3139 sizeof (struct dnr_ext
),
3140 sizeof (struct pdr_ext
),
3141 sizeof (struct sym_ext
),
3142 sizeof (struct opt_ext
),
3143 sizeof (struct fdr_ext
),
3144 sizeof (struct rfd_ext
),
3145 sizeof (struct ext_ext
),
3146 /* Functions to swap in external symbolic data. */
3155 _bfd_ecoff_swap_tir_in
,
3156 _bfd_ecoff_swap_rndx_in
,
3157 /* Functions to swap out external symbolic data. */
3166 _bfd_ecoff_swap_tir_out
,
3167 _bfd_ecoff_swap_rndx_out
,
3168 /* Function to read in symbolic data. */
3169 _bfd_mips_elf_read_ecoff_info
3172 /* Relocations in the 64 bit MIPS ELF ABI are more complex than in
3173 standard ELF. This structure is used to redirect the relocation
3174 handling routines. */
3176 const struct elf_size_info mips_elf64_size_info
=
3178 sizeof (Elf64_External_Ehdr
),
3179 sizeof (Elf64_External_Phdr
),
3180 sizeof (Elf64_External_Shdr
),
3181 sizeof (Elf64_Mips_External_Rel
),
3182 sizeof (Elf64_Mips_External_Rela
),
3183 sizeof (Elf64_External_Sym
),
3184 sizeof (Elf64_External_Dyn
),
3185 sizeof (Elf_External_Note
),
3186 4, /* hash-table entry size */
3187 3, /* internal relocations per external relocations */
3189 3, /* log_file_align */
3192 bfd_elf64_write_out_phdrs
,
3193 bfd_elf64_write_shdrs_and_ehdr
,
3194 bfd_elf64_checksum_contents
,
3195 mips_elf64_write_relocs
,
3196 bfd_elf64_swap_symbol_in
,
3197 bfd_elf64_swap_symbol_out
,
3198 mips_elf64_slurp_reloc_table
,
3199 bfd_elf64_slurp_symbol_table
,
3200 bfd_elf64_swap_dyn_in
,
3201 bfd_elf64_swap_dyn_out
,
3202 mips_elf64_be_swap_reloc_in
,
3203 mips_elf64_be_swap_reloc_out
,
3204 mips_elf64_be_swap_reloca_in
,
3205 mips_elf64_be_swap_reloca_out
3208 #define ELF_ARCH bfd_arch_mips
3209 #define ELF_MACHINE_CODE EM_MIPS
3211 #define elf_backend_collect TRUE
3212 #define elf_backend_type_change_ok TRUE
3213 #define elf_backend_can_gc_sections TRUE
3214 #define elf_info_to_howto mips_elf64_info_to_howto_rela
3215 #define elf_info_to_howto_rel mips_elf64_info_to_howto_rel
3216 #define elf_backend_object_p mips_elf64_object_p
3217 #define elf_backend_symbol_processing _bfd_mips_elf_symbol_processing
3218 #define elf_backend_section_processing _bfd_mips_elf_section_processing
3219 #define elf_backend_section_from_shdr _bfd_mips_elf_section_from_shdr
3220 #define elf_backend_fake_sections _bfd_mips_elf_fake_sections
3221 #define elf_backend_section_from_bfd_section \
3222 _bfd_mips_elf_section_from_bfd_section
3223 #define elf_backend_add_symbol_hook _bfd_mips_elf_add_symbol_hook
3224 #define elf_backend_link_output_symbol_hook \
3225 _bfd_mips_elf_link_output_symbol_hook
3226 #define elf_backend_create_dynamic_sections \
3227 _bfd_mips_elf_create_dynamic_sections
3228 #define elf_backend_check_relocs _bfd_mips_elf_check_relocs
3229 #define elf_backend_merge_symbol_attribute \
3230 _bfd_mips_elf_merge_symbol_attribute
3231 #define elf_backend_get_target_dtag _bfd_mips_elf_get_target_dtag
3232 #define elf_backend_adjust_dynamic_symbol \
3233 _bfd_mips_elf_adjust_dynamic_symbol
3234 #define elf_backend_always_size_sections \
3235 _bfd_mips_elf_always_size_sections
3236 #define elf_backend_size_dynamic_sections \
3237 _bfd_mips_elf_size_dynamic_sections
3238 #define elf_backend_init_index_section _bfd_elf_init_1_index_section
3239 #define elf_backend_relocate_section _bfd_mips_elf_relocate_section
3240 #define elf_backend_finish_dynamic_symbol \
3241 _bfd_mips_elf_finish_dynamic_symbol
3242 #define elf_backend_finish_dynamic_sections \
3243 _bfd_mips_elf_finish_dynamic_sections
3244 #define elf_backend_final_write_processing \
3245 _bfd_mips_elf_final_write_processing
3246 #define elf_backend_additional_program_headers \
3247 _bfd_mips_elf_additional_program_headers
3248 #define elf_backend_modify_segment_map _bfd_mips_elf_modify_segment_map
3249 #define elf_backend_gc_mark_hook _bfd_mips_elf_gc_mark_hook
3250 #define elf_backend_gc_sweep_hook _bfd_mips_elf_gc_sweep_hook
3251 #define elf_backend_copy_indirect_symbol \
3252 _bfd_mips_elf_copy_indirect_symbol
3253 #define elf_backend_ignore_discarded_relocs \
3254 _bfd_mips_elf_ignore_discarded_relocs
3255 #define elf_backend_mips_irix_compat elf64_mips_irix_compat
3256 #define elf_backend_mips_rtype_to_howto mips_elf64_rtype_to_howto
3257 #define elf_backend_ecoff_debug_swap &mips_elf64_ecoff_debug_swap
3258 #define elf_backend_size_info mips_elf64_size_info
3260 #define elf_backend_grok_prstatus elf64_mips_grok_prstatus
3261 #define elf_backend_grok_psinfo elf64_mips_grok_psinfo
3263 #define elf_backend_got_header_size (4 * MIPS_RESERVED_GOTNO)
3265 /* MIPS ELF64 can use a mixture of REL and RELA, but some Relocations
3266 work better/work only in RELA, so we default to this. */
3267 #define elf_backend_may_use_rel_p 1
3268 #define elf_backend_may_use_rela_p 1
3269 #define elf_backend_default_use_rela_p 1
3270 #define elf_backend_rela_plts_and_copies_p 0
3271 #define elf_backend_plt_readonly 1
3272 #define elf_backend_plt_sym_val _bfd_mips_elf_plt_sym_val
3274 #define elf_backend_sign_extend_vma TRUE
3276 #define elf_backend_write_section _bfd_mips_elf_write_section
3278 /* We don't set bfd_elf64_bfd_is_local_label_name because the 32-bit
3279 MIPS-specific function only applies to IRIX5, which had no 64-bit
3281 #define bfd_elf64_find_nearest_line _bfd_mips_elf_find_nearest_line
3282 #define bfd_elf64_find_inliner_info _bfd_mips_elf_find_inliner_info
3283 #define bfd_elf64_new_section_hook _bfd_mips_elf_new_section_hook
3284 #define bfd_elf64_set_section_contents _bfd_mips_elf_set_section_contents
3285 #define bfd_elf64_bfd_get_relocated_section_contents \
3286 _bfd_elf_mips_get_relocated_section_contents
3287 #define bfd_elf64_mkobject _bfd_mips_elf_mkobject
3288 #define bfd_elf64_bfd_link_hash_table_create \
3289 _bfd_mips_elf_link_hash_table_create
3290 #define bfd_elf64_bfd_final_link _bfd_mips_elf_final_link
3291 #define bfd_elf64_bfd_merge_private_bfd_data \
3292 _bfd_mips_elf_merge_private_bfd_data
3293 #define bfd_elf64_bfd_set_private_flags _bfd_mips_elf_set_private_flags
3294 #define bfd_elf64_bfd_print_private_bfd_data \
3295 _bfd_mips_elf_print_private_bfd_data
3297 #define bfd_elf64_get_reloc_upper_bound mips_elf64_get_reloc_upper_bound
3298 #define bfd_elf64_canonicalize_reloc mips_elf64_canonicalize_reloc
3299 #define bfd_elf64_get_dynamic_reloc_upper_bound mips_elf64_get_dynamic_reloc_upper_bound
3300 #define bfd_elf64_canonicalize_dynamic_reloc mips_elf64_canonicalize_dynamic_reloc
3301 #define bfd_elf64_bfd_relax_section _bfd_mips_relax_section
3303 /* MIPS ELF64 archive functions. */
3304 #define bfd_elf64_archive_functions
3305 extern bfd_boolean bfd_elf64_archive_slurp_armap
3307 extern bfd_boolean bfd_elf64_archive_write_armap
3308 (bfd
*, unsigned int, struct orl
*, unsigned int, int);
3309 #define bfd_elf64_archive_slurp_extended_name_table \
3310 _bfd_archive_coff_slurp_extended_name_table
3311 #define bfd_elf64_archive_construct_extended_name_table \
3312 _bfd_archive_coff_construct_extended_name_table
3313 #define bfd_elf64_archive_truncate_arname \
3314 _bfd_archive_coff_truncate_arname
3315 #define bfd_elf64_archive_read_ar_hdr _bfd_archive_coff_read_ar_hdr
3316 #define bfd_elf64_archive_write_ar_hdr _bfd_archive_coff_write_ar_hdr
3317 #define bfd_elf64_archive_openr_next_archived_file \
3318 _bfd_archive_coff_openr_next_archived_file
3319 #define bfd_elf64_archive_get_elt_at_index \
3320 _bfd_archive_coff_get_elt_at_index
3321 #define bfd_elf64_archive_generic_stat_arch_elt \
3322 _bfd_archive_coff_generic_stat_arch_elt
3323 #define bfd_elf64_archive_update_armap_timestamp \
3324 _bfd_archive_coff_update_armap_timestamp
3326 /* The SGI style (n)64 NewABI. */
3327 #define TARGET_LITTLE_SYM bfd_elf64_littlemips_vec
3328 #define TARGET_LITTLE_NAME "elf64-littlemips"
3329 #define TARGET_BIG_SYM bfd_elf64_bigmips_vec
3330 #define TARGET_BIG_NAME "elf64-bigmips"
3332 #define ELF_MAXPAGESIZE 0x10000
3333 #define ELF_COMMONPAGESIZE 0x1000
3335 #include "elf64-target.h"
3337 /* The SYSV-style 'traditional' (n)64 NewABI. */
3338 #undef TARGET_LITTLE_SYM
3339 #undef TARGET_LITTLE_NAME
3340 #undef TARGET_BIG_SYM
3341 #undef TARGET_BIG_NAME
3343 #undef ELF_MAXPAGESIZE
3344 #undef ELF_COMMONPAGESIZE
3346 #define TARGET_LITTLE_SYM bfd_elf64_tradlittlemips_vec
3347 #define TARGET_LITTLE_NAME "elf64-tradlittlemips"
3348 #define TARGET_BIG_SYM bfd_elf64_tradbigmips_vec
3349 #define TARGET_BIG_NAME "elf64-tradbigmips"
3351 #define ELF_MAXPAGESIZE 0x10000
3352 #define ELF_COMMONPAGESIZE 0x1000
3353 #define elf64_bed elf64_tradbed
3355 /* Include the target file again for this target. */
3356 #include "elf64-target.h"