mirror of
https://github.com/bebbo/amigaos-binutils-2.14.git
synced 2025-12-08 22:38:24 +00:00
2240 lines
70 KiB
C
2240 lines
70 KiB
C
/* MIPS-specific support for 32-bit ELF
|
||
Copyright 1993, 1994, 1995, 1996, 1997, 1998, 1999, 2000, 2001, 2002,
|
||
2003 Free Software Foundation, Inc.
|
||
|
||
Most of the information added by Ian Lance Taylor, Cygnus Support,
|
||
<ian@cygnus.com>.
|
||
N32/64 ABI support added by Mark Mitchell, CodeSourcery, LLC.
|
||
<mark@codesourcery.com>
|
||
Traditional MIPS targets support added by Koundinya.K, Dansk Data
|
||
Elektronik & Operations Research Group. <kk@ddeorg.soft.net>
|
||
|
||
This file is part of BFD, the Binary File Descriptor library.
|
||
|
||
This program is free software; you can redistribute it and/or modify
|
||
it under the terms of the GNU General Public License as published by
|
||
the Free Software Foundation; either version 2 of the License, or
|
||
(at your option) any later version.
|
||
|
||
This program is distributed in the hope that it will be useful,
|
||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
GNU General Public License for more details.
|
||
|
||
You should have received a copy of the GNU General Public License
|
||
along with this program; if not, write to the Free Software
|
||
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */
|
||
|
||
/* This file handles MIPS ELF targets. SGI Irix 5 uses a slightly
|
||
different MIPS ELF from other targets. This matters when linking.
|
||
This file supports both, switching at runtime. */
|
||
|
||
#include "bfd.h"
|
||
#include "sysdep.h"
|
||
#include "libbfd.h"
|
||
#include "bfdlink.h"
|
||
#include "genlink.h"
|
||
#include "elf-bfd.h"
|
||
#include "elfxx-mips.h"
|
||
#include "elf/mips.h"
|
||
|
||
/* Get the ECOFF swapping routines. */
|
||
#include "coff/sym.h"
|
||
#include "coff/symconst.h"
|
||
#include "coff/internal.h"
|
||
#include "coff/ecoff.h"
|
||
#include "coff/mips.h"
|
||
#define ECOFF_SIGNED_32
|
||
#include "ecoffswap.h"
|
||
|
||
static bfd_reloc_status_type mips_elf_generic_reloc
|
||
PARAMS ((bfd *, arelent *, asymbol *, PTR, asection *, bfd *, char **));
|
||
static bfd_reloc_status_type mips_elf_hi16_reloc
|
||
PARAMS ((bfd *, arelent *, asymbol *, PTR, asection *, bfd *, char **));
|
||
static bfd_reloc_status_type mips_elf_lo16_reloc
|
||
PARAMS ((bfd *, arelent *, asymbol *, PTR, asection *, bfd *, char **));
|
||
static bfd_reloc_status_type mips_elf_got16_reloc
|
||
PARAMS ((bfd *, arelent *, asymbol *, PTR, asection *, bfd *, char **));
|
||
static bfd_boolean mips_elf_assign_gp
|
||
PARAMS ((bfd *, bfd_vma *));
|
||
static bfd_reloc_status_type mips_elf_final_gp
|
||
PARAMS ((bfd *, asymbol *, bfd_boolean, char **, bfd_vma *));
|
||
static bfd_reloc_status_type mips_elf_gprel16_reloc
|
||
PARAMS ((bfd *, arelent *, asymbol *, PTR, asection *, bfd *, char **));
|
||
static bfd_reloc_status_type mips_elf_literal_reloc
|
||
PARAMS ((bfd *, arelent *, asymbol *, PTR, asection *, bfd *, char **));
|
||
static bfd_reloc_status_type mips_elf_gprel32_reloc
|
||
PARAMS ((bfd *, arelent *, asymbol *, PTR, asection *, bfd *, char **));
|
||
static bfd_reloc_status_type gprel32_with_gp
|
||
PARAMS ((bfd *, asymbol *, arelent *, asection *, bfd_boolean, PTR,
|
||
bfd_vma));
|
||
static bfd_reloc_status_type mips_elf_shift6_reloc
|
||
PARAMS ((bfd *, arelent *, asymbol *, PTR, asection *, bfd *, char **));
|
||
static bfd_reloc_status_type mips16_jump_reloc
|
||
PARAMS ((bfd *, arelent *, asymbol *, PTR, asection *, bfd *, char **));
|
||
static bfd_reloc_status_type mips16_gprel_reloc
|
||
PARAMS ((bfd *, arelent *, asymbol *, PTR, asection *, bfd *, char **));
|
||
static reloc_howto_type *bfd_elf32_bfd_reloc_type_lookup
|
||
PARAMS ((bfd *, bfd_reloc_code_real_type));
|
||
static reloc_howto_type *mips_elf_n32_rtype_to_howto
|
||
PARAMS ((unsigned int, bfd_boolean));
|
||
static void mips_info_to_howto_rel
|
||
PARAMS ((bfd *, arelent *, Elf_Internal_Rela *));
|
||
static void mips_info_to_howto_rela
|
||
PARAMS ((bfd *, arelent *, Elf_Internal_Rela *));
|
||
static bfd_boolean mips_elf_sym_is_global
|
||
PARAMS ((bfd *, asymbol *));
|
||
static bfd_boolean mips_elf_n32_object_p
|
||
PARAMS ((bfd *));
|
||
static bfd_boolean elf32_mips_grok_prstatus
|
||
PARAMS ((bfd *, Elf_Internal_Note *));
|
||
static bfd_boolean elf32_mips_grok_psinfo
|
||
PARAMS ((bfd *, Elf_Internal_Note *));
|
||
static irix_compat_t elf_n32_mips_irix_compat
|
||
PARAMS ((bfd *));
|
||
|
||
extern const bfd_target bfd_elf32_nbigmips_vec;
|
||
extern const bfd_target bfd_elf32_nlittlemips_vec;
|
||
|
||
static asection *prev_reloc_section = NULL;
|
||
static bfd_vma prev_reloc_address = -1;
|
||
static bfd_vma prev_reloc_addend = 0;
|
||
|
||
/* Nonzero if ABFD is using the N32 ABI. */
|
||
#define ABI_N32_P(abfd) \
|
||
((elf_elfheader (abfd)->e_flags & EF_MIPS_ABI2) != 0)
|
||
|
||
/* Whether we are trying to be compatible with IRIX at all. */
|
||
#define SGI_COMPAT(abfd) \
|
||
(elf_n32_mips_irix_compat (abfd) != ict_none)
|
||
|
||
/* The number of local .got entries we reserve. */
|
||
#define MIPS_RESERVED_GOTNO (2)
|
||
|
||
/* In case we're on a 32-bit machine, construct a 64-bit "-1" value
|
||
from smaller values. Start with zero, widen, *then* decrement. */
|
||
#define MINUS_ONE (((bfd_vma)0) - 1)
|
||
|
||
/* The relocation table used for SHT_REL sections. */
|
||
|
||
static reloc_howto_type elf_mips_howto_table_rel[] =
|
||
{
|
||
/* No relocation. */
|
||
HOWTO (R_MIPS_NONE, /* type */
|
||
0, /* rightshift */
|
||
0, /* size (0 = byte, 1 = short, 2 = long) */
|
||
0, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_NONE", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 16 bit relocation. */
|
||
HOWTO (R_MIPS_16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 32 bit relocation. */
|
||
HOWTO (R_MIPS_32, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_32", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0xffffffff, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 32 bit symbol relative relocation. */
|
||
HOWTO (R_MIPS_REL32, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_REL32", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0xffffffff, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 26 bit jump address. */
|
||
HOWTO (R_MIPS_26, /* type */
|
||
2, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
26, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
/* This needs complex overflow
|
||
detection, because the upper four
|
||
bits must match the PC + 4. */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_26", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x03ffffff, /* src_mask */
|
||
0x03ffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* R_MIPS_HI16 and R_MIPS_LO16 are unsupported for NewABI REL.
|
||
However, the native IRIX6 tools use them, so we try our best. */
|
||
|
||
/* High 16 bits of symbol value. */
|
||
HOWTO (R_MIPS_HI16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_hi16_reloc, /* special_function */
|
||
"R_MIPS_HI16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Low 16 bits of symbol value. */
|
||
HOWTO (R_MIPS_LO16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_lo16_reloc, /* special_function */
|
||
"R_MIPS_LO16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* GP relative reference. */
|
||
HOWTO (R_MIPS_GPREL16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_gprel16_reloc, /* special_function */
|
||
"R_MIPS_GPREL16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Reference to literal section. */
|
||
HOWTO (R_MIPS_LITERAL, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_literal_reloc, /* special_function */
|
||
"R_MIPS_LITERAL", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Reference to global offset table. */
|
||
HOWTO (R_MIPS_GOT16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_got16_reloc, /* special_function */
|
||
"R_MIPS_GOT16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 16 bit PC relative reference. */
|
||
HOWTO (R_MIPS_PC16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
TRUE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_PC16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
TRUE), /* pcrel_offset */
|
||
|
||
/* 16 bit call through global offset table. */
|
||
HOWTO (R_MIPS_CALL16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_CALL16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 32 bit GP relative reference. */
|
||
HOWTO (R_MIPS_GPREL32, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_gprel32_reloc, /* special_function */
|
||
"R_MIPS_GPREL32", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0xffffffff, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* The remaining relocs are defined on Irix 5, although they are
|
||
not defined by the ABI. */
|
||
EMPTY_HOWTO (13),
|
||
EMPTY_HOWTO (14),
|
||
EMPTY_HOWTO (15),
|
||
|
||
/* A 5 bit shift field. */
|
||
HOWTO (R_MIPS_SHIFT5, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
5, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
6, /* bitpos */
|
||
complain_overflow_bitfield, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_SHIFT5", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x000007c0, /* src_mask */
|
||
0x000007c0, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* A 6 bit shift field. */
|
||
HOWTO (R_MIPS_SHIFT6, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
6, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
6, /* bitpos */
|
||
complain_overflow_bitfield, /* complain_on_overflow */
|
||
mips_elf_shift6_reloc, /* special_function */
|
||
"R_MIPS_SHIFT6", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x000007c4, /* src_mask */
|
||
0x000007c4, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* A 64 bit relocation. */
|
||
HOWTO (R_MIPS_64, /* type */
|
||
0, /* rightshift */
|
||
4, /* size (0 = byte, 1 = short, 2 = long) */
|
||
64, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_64", /* name */
|
||
TRUE, /* partial_inplace */
|
||
MINUS_ONE, /* src_mask */
|
||
MINUS_ONE, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Displacement in the global offset table. */
|
||
HOWTO (R_MIPS_GOT_DISP, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_GOT_DISP", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Displacement to page pointer in the global offset table. */
|
||
HOWTO (R_MIPS_GOT_PAGE, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_GOT_PAGE", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Offset from page pointer in the global offset table. */
|
||
HOWTO (R_MIPS_GOT_OFST, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_GOT_OFST", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* High 16 bits of displacement in global offset table. */
|
||
HOWTO (R_MIPS_GOT_HI16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_GOT_HI16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Low 16 bits of displacement in global offset table. */
|
||
HOWTO (R_MIPS_GOT_LO16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_GOT_LO16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 64 bit subtraction. */
|
||
HOWTO (R_MIPS_SUB, /* type */
|
||
0, /* rightshift */
|
||
4, /* size (0 = byte, 1 = short, 2 = long) */
|
||
64, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_SUB", /* name */
|
||
TRUE, /* partial_inplace */
|
||
MINUS_ONE, /* src_mask */
|
||
MINUS_ONE, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Insert the addend as an instruction. */
|
||
/* FIXME: Not handled correctly. */
|
||
HOWTO (R_MIPS_INSERT_A, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_INSERT_A", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0xffffffff, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Insert the addend as an instruction, and change all relocations
|
||
to refer to the old instruction at the address. */
|
||
/* FIXME: Not handled correctly. */
|
||
HOWTO (R_MIPS_INSERT_B, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_INSERT_B", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0xffffffff, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Delete a 32 bit instruction. */
|
||
/* FIXME: Not handled correctly. */
|
||
HOWTO (R_MIPS_DELETE, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_DELETE", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0xffffffff, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* The MIPS ELF64 ABI Draft wants us to support these for REL relocations.
|
||
We don't, because
|
||
a) It means building the addend from a R_MIPS_HIGHEST/R_MIPS_HIGHER/
|
||
R_MIPS_HI16/R_MIPS_LO16 sequence with varying ordering, using
|
||
fallable heuristics.
|
||
b) No other NEwABI toolchain actually emits such relocations. */
|
||
EMPTY_HOWTO (R_MIPS_HIGHER),
|
||
EMPTY_HOWTO (R_MIPS_HIGHEST),
|
||
|
||
/* High 16 bits of displacement in global offset table. */
|
||
HOWTO (R_MIPS_CALL_HI16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_CALL_HI16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Low 16 bits of displacement in global offset table. */
|
||
HOWTO (R_MIPS_CALL_LO16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_CALL_LO16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0x0000ffff, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Section displacement. */
|
||
HOWTO (R_MIPS_SCN_DISP, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_SCN_DISP", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0xffffffff, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
HOWTO (R_MIPS_REL16, /* type */
|
||
0, /* rightshift */
|
||
1, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_REL16", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0xffff, /* src_mask */
|
||
0xffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* These two are obsolete. */
|
||
EMPTY_HOWTO (R_MIPS_ADD_IMMEDIATE),
|
||
EMPTY_HOWTO (R_MIPS_PJUMP),
|
||
|
||
/* Similiar to R_MIPS_REL32, but used for relocations in a GOT section.
|
||
It must be used for multigot GOT's (and only there). */
|
||
HOWTO (R_MIPS_RELGOT, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_RELGOT", /* name */
|
||
TRUE, /* partial_inplace */
|
||
0xffffffff, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Protected jump conversion. This is an optimization hint. No
|
||
relocation is required for correctness. */
|
||
HOWTO (R_MIPS_JALR, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_JALR", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0x00000000, /* src_mask */
|
||
0x00000000, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
};
|
||
|
||
/* The relocation table used for SHT_RELA sections. */
|
||
|
||
static reloc_howto_type elf_mips_howto_table_rela[] =
|
||
{
|
||
/* No relocation. */
|
||
HOWTO (R_MIPS_NONE, /* type */
|
||
0, /* rightshift */
|
||
0, /* size (0 = byte, 1 = short, 2 = long) */
|
||
0, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_NONE", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 16 bit relocation. */
|
||
HOWTO (R_MIPS_16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_16", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0x0000, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 32 bit relocation. */
|
||
HOWTO (R_MIPS_32, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_32", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 32 bit symbol relative relocation. */
|
||
HOWTO (R_MIPS_REL32, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_REL32", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 26 bit jump address. */
|
||
HOWTO (R_MIPS_26, /* type */
|
||
2, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
26, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
/* This needs complex overflow
|
||
detection, because the upper 36
|
||
bits must match the PC + 4. */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_26", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0x03ffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* High 16 bits of symbol value. */
|
||
HOWTO (R_MIPS_HI16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_HI16", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Low 16 bits of symbol value. */
|
||
HOWTO (R_MIPS_LO16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_LO16", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* GP relative reference. */
|
||
HOWTO (R_MIPS_GPREL16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_gprel16_reloc, /* special_function */
|
||
"R_MIPS_GPREL16", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Reference to literal section. */
|
||
HOWTO (R_MIPS_LITERAL, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_literal_reloc, /* special_function */
|
||
"R_MIPS_LITERAL", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* Reference to global offset table. */
|
||
HOWTO (R_MIPS_GOT16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_got16_reloc, /* special_function */
|
||
"R_MIPS_GOT16", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 16 bit PC relative reference. */
|
||
HOWTO (R_MIPS_PC16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
TRUE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_PC16", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
TRUE), /* pcrel_offset */
|
||
|
||
/* 16 bit call through global offset table. */
|
||
HOWTO (R_MIPS_CALL16, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
16, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_signed, /* complain_on_overflow */
|
||
mips_elf_generic_reloc, /* special_function */
|
||
"R_MIPS_CALL16", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0x0000ffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||
/* 32 bit GP relative reference. */
|
||
HOWTO (R_MIPS_GPREL32, /* type */
|
||
0, /* rightshift */
|
||
2, /* size (0 = byte, 1 = short, 2 = long) */
|
||
32, /* bitsize */
|
||
FALSE, /* pc_relative */
|
||
0, /* bitpos */
|
||
complain_overflow_dont, /* complain_on_overflow */
|
||
mips_elf_gprel32_reloc, /* special_function */
|
||
"R_MIPS_GPREL32", /* name */
|
||
FALSE, /* partial_inplace */
|
||
0, /* src_mask */
|
||
0xffffffff, /* dst_mask */
|
||
FALSE), /* pcrel_offset */
|
||
|
||