2c2fa1
commit 45965137bee4946dca3cd99285f2a7afe6b99aeb
2c2fa1
Author: Alan Modra <amodra@gmail.com>
2c2fa1
Date:   Wed Mar 5 19:57:39 2014 +1030
2c2fa1
2c2fa1
    Support R_PPC64_ADDR64_LOCAL
2c2fa1
    
2c2fa1
    This adds support for "func@localentry", an expression that returns the
2c2fa1
    ELFv2 local entry point address of function "func".  I've excluded
2c2fa1
    dynamic relocation support because that obviously would require glibc
2c2fa1
    changes.
2c2fa1
    
2c2fa1
    include/elf/
2c2fa1
            * ppc64.h (R_PPC64_REL24_NOTOC, R_PPC64_ADDR64_LOCAL): Define.
2c2fa1
    bfd/
2c2fa1
            * elf64-ppc.c (ppc64_elf_howto_raw): Add R_PPC64_ADDR64_LOCAL entry.
2c2fa1
            (ppc64_elf_reloc_type_lookup): Support R_PPC64_ADDR64_LOCAL.
2c2fa1
            (ppc64_elf_check_relocs): Likewise.
2c2fa1
            (ppc64_elf_relocate_section): Likewise.
2c2fa1
            * Add BFD_RELOC_PPC64_ADDR64_LOCAL.
2c2fa1
            * bfd-in2.h: Regenerate.
2c2fa1
            * libbfd.h: Regenerate.
2c2fa1
    gas/
2c2fa1
            * config/tc-ppc.c (ppc_elf_suffix): Support @localentry.
2c2fa1
            (md_apply_fix): Support R_PPC64_ADDR64_LOCAL.
2c2fa1
    ld/testsuite/
2c2fa1
            * ld-powerpc/elfv2-2a.s, ld-powerpc/elfv2-2b.s: New files.
2c2fa1
            * ld-powerpc/elfv2-2exe.d, ld-powerpc/elfv2-2so.d: New files.
2c2fa1
            * ld-powerpc/powerpc.exp: Run new test.
2c2fa1
    elfcpp/
2c2fa1
            * powerpc.h (R_PPC64_REL24_NOTOC, R_PPC64_ADDR64_LOCAL): Define.
2c2fa1
    gold/
2c2fa1
            * powerpc.cc (Target_powerpc::Scan::local, global): Support
2c2fa1
            R_PPC64_ADDR64_LOCAL.
2c2fa1
            (Target_powerpc::Relocate::relocate): Likewise.
2c2fa1
2c2fa1
### a/bfd/ChangeLog
2c2fa1
### b/bfd/ChangeLog
2c2fa1
## -1,3 +1,13 @@
2c2fa1
+2014-03-05  Alan Modra  <amodra@gmail.com>
2c2fa1
+
2c2fa1
+	* elf64-ppc.c (ppc64_elf_howto_raw): Add R_PPC64_ADDR64_LOCAL entry.
2c2fa1
+	(ppc64_elf_reloc_type_lookup): Support R_PPC64_ADDR64_LOCAL.
2c2fa1
+	(ppc64_elf_check_relocs): Likewise.
2c2fa1
+	(ppc64_elf_relocate_section): Likewise.
2c2fa1
+	* Add BFD_RELOC_PPC64_ADDR64_LOCAL.
2c2fa1
+	* bfd-in2.h: Regenerate.
2c2fa1
+	* libbfd.h: Regenerate.
2c2fa1
+
2c2fa1
 2014-03-04  Heiher  <r@hev.cc>
2c2fa1
 
2c2fa1
 	* elfxx-mips.c (mips_set_isa_flags): Use E_MIPS_ARCH_64R2 for
2c2fa1
--- a/bfd/bfd-in2.h
2c2fa1
+++ b/bfd/bfd-in2.h
2c2fa1
@@ -3259,6 +3259,7 @@ instruction.  */
2c2fa1
   BFD_RELOC_PPC64_PLTGOT16_LO_DS,
2c2fa1
   BFD_RELOC_PPC64_ADDR16_HIGH,
2c2fa1
   BFD_RELOC_PPC64_ADDR16_HIGHA,
2c2fa1
+  BFD_RELOC_PPC64_ADDR64_LOCAL,
2c2fa1
 
2c2fa1
 /* PowerPC and PowerPC64 thread-local storage relocations.  */
2c2fa1
   BFD_RELOC_PPC_TLS,
2c2fa1
--- a/bfd/elf64-ppc.c
2c2fa1
+++ b/bfd/elf64-ppc.c
2c2fa1
@@ -2095,6 +2095,21 @@ static reloc_howto_type ppc64_elf_howto_raw[] = {
2c2fa1
 	 0xffff,		/* dst_mask */
2c2fa1
 	 FALSE),		/* pcrel_offset */
2c2fa1
 
2c2fa1
+  /* Like ADDR64, but use local entry point of function.  */
2c2fa1
+  HOWTO (R_PPC64_ADDR64_LOCAL,	/* type */
2c2fa1
+	 0,			/* rightshift */
2c2fa1
+	 4,			/* size (0=byte, 1=short, 2=long, 4=64 bits) */
2c2fa1
+	 64,			/* bitsize */
2c2fa1
+	 FALSE,			/* pc_relative */
2c2fa1
+	 0,			/* bitpos */
2c2fa1
+	 complain_overflow_dont, /* complain_on_overflow */
2c2fa1
+	 bfd_elf_generic_reloc,	/* special_function */
2c2fa1
+	 "R_PPC64_ADDR64_LOCAL", /* name */
2c2fa1
+	 FALSE,			/* partial_inplace */
2c2fa1
+	 0,			/* src_mask */
2c2fa1
+	 ONES (64),		/* dst_mask */
2c2fa1
+	 FALSE),		/* pcrel_offset */
2c2fa1
+
2c2fa1
   /* GNU extension to record C++ vtable hierarchy.  */
2c2fa1
   HOWTO (R_PPC64_GNU_VTINHERIT,	/* type */
2c2fa1
 	 0,			/* rightshift */
2c2fa1
@@ -2383,6 +2398,8 @@ ppc64_elf_reloc_type_lookup (bfd *abfd ATTRIBUTE_UNUSED,
2c2fa1
       break;
2c2fa1
     case BFD_RELOC_HI16_S_PCREL:		r = R_PPC64_REL16_HA;
2c2fa1
       break;
2c2fa1
+    case BFD_RELOC_PPC64_ADDR64_LOCAL:		r = R_PPC64_ADDR64_LOCAL;
2c2fa1
+      break;
2c2fa1
     case BFD_RELOC_VTABLE_INHERIT:		r = R_PPC64_GNU_VTINHERIT;
2c2fa1
       break;
2c2fa1
     case BFD_RELOC_VTABLE_ENTRY:		r = R_PPC64_GNU_VTENTRY;
2c2fa1
@@ -5400,6 +5417,21 @@ ppc64_elf_check_relocs (bfd *abfd, struct bfd_link_info *info,
2c2fa1
 	case R_PPC64_REL16_HA:
2c2fa1
 	  break;
2c2fa1
 
2c2fa1
+	  /* Not supported as a dynamic relocation.  */
2c2fa1
+	case R_PPC64_ADDR64_LOCAL:
2c2fa1
+	  if (info->shared)
2c2fa1
+	    {
2c2fa1
+	      if (!ppc64_elf_howto_table[R_PPC64_ADDR32])
2c2fa1
+		ppc_howto_init ();
2c2fa1
+	      info->callbacks->einfo (_("%P: %H: %s reloc unsupported "
2c2fa1
+					"in shared libraries and PIEs.\n"),
2c2fa1
+				      abfd, sec, rel->r_offset,
2c2fa1
+				      ppc64_elf_howto_table[r_type]->name);
2c2fa1
+	      bfd_set_error (bfd_error_bad_value);
2c2fa1
+	      return FALSE;
2c2fa1
+	    }
2c2fa1
+	  break;
2c2fa1
+
2c2fa1
 	case R_PPC64_TOC16:
2c2fa1
 	case R_PPC64_TOC16_DS:
2c2fa1
 	  htab->do_multi_toc = 1;
2c2fa1
@@ -14134,6 +14166,12 @@ ppc64_elf_relocate_section (bfd *output_bfd,
2c2fa1
 	  addend -= htab->elf.tls_sec->vma + DTP_OFFSET;
2c2fa1
 	  break;
2c2fa1
 
2c2fa1
+	case R_PPC64_ADDR64_LOCAL:
2c2fa1
+	  addend += PPC64_LOCAL_ENTRY_OFFSET (h != NULL
2c2fa1
+					      ? h->elf.other
2c2fa1
+					      : sym->st_other);
2c2fa1
+	  break;
2c2fa1
+
2c2fa1
 	case R_PPC64_DTPMOD64:
2c2fa1
 	  relocation = 1;
2c2fa1
 	  addend = 0;
2c2fa1
--- a/bfd/libbfd.h
2c2fa1
+++ b/bfd/libbfd.h
2c2fa1
@@ -1401,6 +1401,7 @@ static const char *const bfd_reloc_code_real_names[] = { "@@uninitialized@@",
2c2fa1
   "BFD_RELOC_PPC64_PLTGOT16_LO_DS",
2c2fa1
   "BFD_RELOC_PPC64_ADDR16_HIGH",
2c2fa1
   "BFD_RELOC_PPC64_ADDR16_HIGHA",
2c2fa1
+  "BFD_RELOC_PPC64_ADDR64_LOCAL",
2c2fa1
   "BFD_RELOC_PPC_TLS",
2c2fa1
   "BFD_RELOC_PPC_TLSGD",
2c2fa1
   "BFD_RELOC_PPC_TLSLD",
2c2fa1
--- a/bfd/reloc.c
2c2fa1
+++ b/bfd/reloc.c
2c2fa1
@@ -2899,6 +2899,8 @@ ENUMX
2c2fa1
   BFD_RELOC_PPC64_ADDR16_HIGH
2c2fa1
 ENUMX
2c2fa1
   BFD_RELOC_PPC64_ADDR16_HIGHA
2c2fa1
+ENUMX
2c2fa1
+  BFD_RELOC_PPC64_ADDR64_LOCAL
2c2fa1
 ENUMDOC
2c2fa1
   Power(rs6000) and PowerPC relocations.
2c2fa1
 
2c2fa1
### a/include/elf/ChangeLog
2c2fa1
### b/include/elf/ChangeLog
2c2fa1
## -1,3 +1,7 @@
2c2fa1
+2014-03-05  Alan Modra  <amodra@gmail.com>
2c2fa1
+
2c2fa1
+	* ppc64.h (R_PPC64_REL24_NOTOC, R_PPC64_ADDR64_LOCAL): Define.
2c2fa1
+
2c2fa1
 2014-02-06  Andrew Pinski  <apinski@cavium.com>
2c2fa1
 
2c2fa1
 	* mips.h (E_MIPS_MACH_OCTEON3): New machine flag.
2c2fa1
--- a/include/elf/ppc64.h
2c2fa1
+++ b/include/elf/ppc64.h
2c2fa1
@@ -149,6 +149,10 @@ START_RELOC_NUMBERS (elf_ppc64_reloc_type)
2c2fa1
   RELOC_NUMBER (R_PPC64_DTPREL16_HIGH,	   114)
2c2fa1
   RELOC_NUMBER (R_PPC64_DTPREL16_HIGHA,	   115)
2c2fa1
 
2c2fa1
+/* Added for ELFv2.  */
2c2fa1
+  RELOC_NUMBER (R_PPC64_REL24_NOTOC,	   116)
2c2fa1
+  RELOC_NUMBER (R_PPC64_ADDR64_LOCAL,	   117)
2c2fa1
+
2c2fa1
 #ifndef RELOC_MACROS_GEN_FUNC
2c2fa1
 /* Fake relocation only used internally by ld.  */
2c2fa1
   RELOC_NUMBER (R_PPC64_LO_DS_OPT,	   128)