* elf64-ppc.c (is_ppc64_target): New function, extracted from..
[deliverable/binutils-gdb.git] / bfd / elf64-ppc.c
index 96e1d230fc9584302b97255dbddb1f82700ffd4d..865460575360e087545dacf9d67f62b56c15cfb3 100644 (file)
@@ -1,5 +1,6 @@
 /* PowerPC64-specific support for 64-bit ELF.
-   Copyright 1999, 2000, 2001, 2002, 2003 Free Software Foundation, Inc.
+   Copyright 1999, 2000, 2001, 2002, 2003, 2004
+   Free Software Foundation, Inc.
    Written by Linus Nordberg, Swox AB <info@swox.com>,
    based on elf32-ppc.c by Ian Lance Taylor.
    Largely rewritten by Alan Modra <amodra@bigpond.net.au>
@@ -34,6 +35,8 @@
 
 static bfd_reloc_status_type ppc64_elf_ha_reloc
   (bfd *, arelent *, asymbol *, void *, asection *, bfd *, char **);
+static bfd_reloc_status_type ppc64_elf_branch_reloc
+  (bfd *, arelent *, asymbol *, void *, asection *, bfd *, char **);
 static bfd_reloc_status_type ppc64_elf_brtaken_reloc
   (bfd *, arelent *, asymbol *, void *, asection *, bfd *, char **);
 static bfd_reloc_status_type ppc64_elf_sectoff_reloc
@@ -48,7 +51,8 @@ static bfd_reloc_status_type ppc64_elf_toc64_reloc
   (bfd *, arelent *, asymbol *, void *, asection *, bfd *, char **);
 static bfd_reloc_status_type ppc64_elf_unhandled_reloc
   (bfd *, arelent *, asymbol *, void *, asection *, bfd *, char **);
-
+static bfd_vma opd_entry_value
+  (asection *, bfd_vma, asection **, bfd_vma *);
 
 #define TARGET_LITTLE_SYM      bfd_elf64_powerpcle_vec
 #define TARGET_LITTLE_NAME     "elf64-powerpcle"
@@ -75,12 +79,16 @@ static bfd_reloc_status_type ppc64_elf_unhandled_reloc
 #define bfd_elf64_new_section_hook           ppc64_elf_new_section_hook
 #define bfd_elf64_bfd_link_hash_table_create  ppc64_elf_link_hash_table_create
 #define bfd_elf64_bfd_link_hash_table_free    ppc64_elf_link_hash_table_free
+#define bfd_elf64_get_synthetic_symtab       ppc64_elf_get_synthetic_symtab
 
 #define elf_backend_object_p                 ppc64_elf_object_p
 #define elf_backend_grok_prstatus            ppc64_elf_grok_prstatus
 #define elf_backend_grok_psinfo                      ppc64_elf_grok_psinfo
 #define elf_backend_create_dynamic_sections   ppc64_elf_create_dynamic_sections
 #define elf_backend_copy_indirect_symbol      ppc64_elf_copy_indirect_symbol
+#define elf_backend_add_symbol_hook          ppc64_elf_add_symbol_hook
+#define elf_backend_check_directives         ppc64_elf_check_directives
+#define elf_backend_archive_symbol_lookup     ppc64_elf_archive_symbol_lookup
 #define elf_backend_check_relocs             ppc64_elf_check_relocs
 #define elf_backend_gc_mark_hook             ppc64_elf_gc_mark_hook
 #define elf_backend_gc_sweep_hook            ppc64_elf_gc_sweep_hook
@@ -92,6 +100,7 @@ static bfd_reloc_status_type ppc64_elf_unhandled_reloc
 #define elf_backend_finish_dynamic_symbol     ppc64_elf_finish_dynamic_symbol
 #define elf_backend_reloc_type_class         ppc64_elf_reloc_type_class
 #define elf_backend_finish_dynamic_sections   ppc64_elf_finish_dynamic_sections
+#define elf_backend_link_output_symbol_hook   ppc64_elf_output_symbol_hook
 #define elf_backend_special_sections         ppc64_elf_special_sections
 
 /* The name of the dynamic interpreter.  This is put in the .interp
@@ -165,9 +174,17 @@ static bfd_reloc_status_type ppc64_elf_unhandled_reloc
 #define LIS_R0_0       0x3c000000      /* lis   %r0,0          */
 #define ORI_R0_R0_0    0x60000000      /* ori   %r0,%r0,0      */
 
-/* Instructions to save and restore floating point regs.  */
+/* Instructions used by the save and restore reg functions.  */
+#define STD_R0_0R1     0xf8010000      /* std   %r0,0(%r1)     */
+#define STD_R0_0R12    0xf80c0000      /* std   %r0,0(%r12)    */
+#define LD_R0_0R1      0xe8010000      /* ld    %r0,0(%r1)     */
+#define LD_R0_0R12     0xe80c0000      /* ld    %r0,0(%r12)    */
 #define STFD_FR0_0R1   0xd8010000      /* stfd  %fr0,0(%r1)    */
 #define LFD_FR0_0R1    0xc8010000      /* lfd   %fr0,0(%r1)    */
+#define LI_R12_0       0x39800000      /* li    %r12,0         */
+#define STVX_VR0_R12_R0        0x7c0c01ce      /* stvx  %v0,%r12,%r0   */
+#define LVX_VR0_R12_R0 0x7c0c00ce      /* lvx   %v0,%r12,%r0   */
+#define MTLR_R0                0x7c0803a6      /* mtlr  %r0            */
 #define BLR            0x4e800020      /* blr                  */
 
 /* Since .opd is an array of descriptors and each entry will end up
@@ -300,7 +317,7 @@ static reloc_howto_type ppc64_elf_howto_raw[] = {
         FALSE,                 /* pc_relative */
         0,                     /* bitpos */
         complain_overflow_bitfield, /* complain_on_overflow */
-        bfd_elf_generic_reloc, /* special_function */
+        ppc64_elf_branch_reloc, /* special_function */
         "R_PPC64_ADDR14",      /* name */
         FALSE,                 /* partial_inplace */
         0,                     /* src_mask */
@@ -349,7 +366,7 @@ static reloc_howto_type ppc64_elf_howto_raw[] = {
         TRUE,                  /* pc_relative */
         0,                     /* bitpos */
         complain_overflow_signed, /* complain_on_overflow */
-        bfd_elf_generic_reloc, /* special_function */
+        ppc64_elf_branch_reloc, /* special_function */
         "R_PPC64_REL24",       /* name */
         FALSE,                 /* partial_inplace */
         0,                     /* src_mask */
@@ -364,7 +381,7 @@ static reloc_howto_type ppc64_elf_howto_raw[] = {
         TRUE,                  /* pc_relative */
         0,                     /* bitpos */
         complain_overflow_signed, /* complain_on_overflow */
-        bfd_elf_generic_reloc, /* special_function */
+        ppc64_elf_branch_reloc, /* special_function */
         "R_PPC64_REL14",       /* name */
         FALSE,                 /* partial_inplace */
         0,                     /* src_mask */
@@ -2127,6 +2144,29 @@ ppc64_elf_ha_reloc (bfd *abfd, arelent *reloc_entry, asymbol *symbol,
   return bfd_reloc_continue;
 }
 
+static bfd_reloc_status_type
+ppc64_elf_branch_reloc (bfd *abfd, arelent *reloc_entry, asymbol *symbol,
+                       void *data, asection *input_section,
+                       bfd *output_bfd, char **error_message)
+{
+  if (output_bfd != NULL)
+    return bfd_elf_generic_reloc (abfd, reloc_entry, symbol, data,
+                                 input_section, output_bfd, error_message);
+
+  if (strcmp (symbol->section->name, ".opd") == 0
+      && (symbol->section->owner->flags & DYNAMIC) == 0)
+    {
+      bfd_vma dest = opd_entry_value (symbol->section,
+                                     symbol->value + reloc_entry->addend,
+                                     NULL, NULL);
+      if (dest != (bfd_vma) -1)
+       reloc_entry->addend = dest - (symbol->value
+                                     + symbol->section->output_section->vma
+                                     + symbol->section->output_offset);
+    }
+  return bfd_reloc_continue;
+}
+
 static bfd_reloc_status_type
 ppc64_elf_brtaken_reloc (bfd *abfd, arelent *reloc_entry, asymbol *symbol,
                         void *data, asection *input_section,
@@ -2163,7 +2203,7 @@ ppc64_elf_brtaken_reloc (bfd *abfd, arelent *reloc_entry, asymbol *symbol,
       else if ((insn & (0x14 << 21)) == (0x10 << 21))
        insn |= 0x08 << 21;
       else
-       return bfd_reloc_continue;
+       goto out;
     }
   else
     {
@@ -2185,7 +2225,9 @@ ppc64_elf_brtaken_reloc (bfd *abfd, arelent *reloc_entry, asymbol *symbol,
        insn ^= 0x01 << 21;
     }
   bfd_put_32 (abfd, insn, (bfd_byte *) data + octets);
-  return bfd_reloc_continue;
+ out:
+  return ppc64_elf_branch_reloc (abfd, reloc_entry, symbol, data,
+                                input_section, output_bfd, error_message);
 }
 
 static bfd_reloc_status_type
@@ -2328,6 +2370,10 @@ struct ppc64_elf_obj_tdata
   asection *got;
   asection *relgot;
 
+  /* Used during garbage collection.  We attach global symbols defined
+     on removed .opd entries to this section so that the sym is removed.  */
+  asection *deleted_section;
+
   /* TLS local dynamic got entry handling.  Suppose for multiple GOT
      sections means we potentially need one of these for each input bfd.  */
   union {
@@ -2354,6 +2400,15 @@ ppc64_elf_mkobject (bfd *abfd)
   return TRUE;
 }
 
+static bfd_boolean
+is_ppc64_target (const struct bfd_target *targ)
+{
+  extern const bfd_target bfd_elf64_powerpc_vec;
+  extern const bfd_target bfd_elf64_powerpcle_vec;
+
+  return targ == &bfd_elf64_powerpc_vec || targ == &bfd_elf64_powerpcle_vec;
+}
+
 /* Fix bad default arch selected for a 64 bit input bfd when the
    default is 32 bit.  */
 
@@ -2379,7 +2434,7 @@ ppc64_elf_object_p (bfd *abfd)
 static bfd_boolean
 ppc64_elf_grok_prstatus (bfd *abfd, Elf_Internal_Note *note)
 {
-  size_t offset, raw_size;
+  size_t offset, size;
 
   if (note->descsz != 504)
     return FALSE;
@@ -2392,11 +2447,11 @@ ppc64_elf_grok_prstatus (bfd *abfd, Elf_Internal_Note *note)
 
   /* pr_reg */
   offset = 112;
-  raw_size = 384;
+  size = 384;
 
   /* Make a ".reg/999" section.  */
   return _bfd_elfcore_make_pseudosection (abfd, ".reg",
-                                         raw_size, note->descpos + offset);
+                                         size, note->descpos + offset);
 }
 
 static bfd_boolean
@@ -2427,13 +2482,13 @@ ppc64_elf_merge_private_bfd_data (bfd *ibfd, bfd *obfd)
       const char *msg;
 
       if (bfd_big_endian (ibfd))
-       msg = _("%s: compiled for a big endian system "
+       msg = _("%B: compiled for a big endian system "
                "and target is little endian");
       else
-       msg = _("%s: compiled for a little endian system "
+       msg = _("%B: compiled for a little endian system "
                "and target is big endian");
 
-      (*_bfd_error_handler) (msg, bfd_archive_filename (ibfd));
+      (*_bfd_error_handler) (msg, ibfd);
 
       bfd_set_error (bfd_error_wrong_format);
       return FALSE;
@@ -2489,6 +2544,387 @@ ppc64_elf_new_section_hook (bfd *abfd, asection *sec)
 
   return _bfd_elf_new_section_hook (abfd, sec);
 }
+
+static void *
+get_opd_info (asection * sec)
+{
+  if (sec != NULL
+      && ppc64_elf_section_data (sec) != NULL
+      && ppc64_elf_section_data (sec)->opd.adjust != NULL)
+    return ppc64_elf_section_data (sec)->opd.adjust;
+  return NULL;
+}
+\f
+/* Parameters for the qsort hook.  */
+static asection *synthetic_opd;
+static bfd_boolean synthetic_relocatable;
+
+/* qsort comparison function for ppc64_elf_get_synthetic_symtab.  */
+
+static int
+compare_symbols (const void *ap, const void *bp)
+{
+  const asymbol *a = * (const asymbol **) ap;
+  const asymbol *b = * (const asymbol **) bp;
+
+  /* Section symbols first.  */
+  if ((a->flags & BSF_SECTION_SYM) && !(b->flags & BSF_SECTION_SYM))
+    return -1;
+  if (!(a->flags & BSF_SECTION_SYM) && (b->flags & BSF_SECTION_SYM))
+    return 1;
+
+  /* then .opd symbols.  */
+  if (a->section == synthetic_opd && b->section != synthetic_opd)
+    return -1;
+  if (a->section != synthetic_opd && b->section == synthetic_opd)
+    return 1;
+
+  /* then other code symbols.  */
+  if ((a->section->flags & (SEC_CODE | SEC_ALLOC | SEC_THREAD_LOCAL))
+      == (SEC_CODE | SEC_ALLOC)
+      && (b->section->flags & (SEC_CODE | SEC_ALLOC | SEC_THREAD_LOCAL))
+        != (SEC_CODE | SEC_ALLOC))
+    return -1;
+
+  if ((a->section->flags & (SEC_CODE | SEC_ALLOC | SEC_THREAD_LOCAL))
+      != (SEC_CODE | SEC_ALLOC)
+      && (b->section->flags & (SEC_CODE | SEC_ALLOC | SEC_THREAD_LOCAL))
+        == (SEC_CODE | SEC_ALLOC))
+    return 1;
+
+  if (synthetic_relocatable)
+    {
+      if (a->section->id < b->section->id)
+       return -1;
+
+      if (a->section->id > b->section->id)
+       return 1;
+    }
+
+  if (a->value + a->section->vma < b->value + b->section->vma)
+    return -1;
+
+  if (a->value + a->section->vma > b->value + b->section->vma)
+    return 1;
+
+  return 0;
+}
+
+/* Search SYMS for a symbol of the given VALUE.  */
+
+static asymbol *
+sym_exists_at (asymbol **syms, long lo, long hi, int id, bfd_vma value)
+{
+  long mid;
+
+  if (id == -1)
+    {
+      while (lo < hi)
+       {
+         mid = (lo + hi) >> 1;
+         if (syms[mid]->value + syms[mid]->section->vma < value)
+           lo = mid + 1;
+         else if (syms[mid]->value + syms[mid]->section->vma > value)
+           hi = mid;
+         else
+           return syms[mid];
+       }
+    }
+  else
+    {
+      while (lo < hi)
+       {
+         mid = (lo + hi) >> 1;
+         if (syms[mid]->section->id < id)
+           lo = mid + 1;
+         else if (syms[mid]->section->id > id)
+           hi = mid;
+         else if (syms[mid]->value < value)
+           lo = mid + 1;
+         else if (syms[mid]->value > value)
+           hi = mid;
+         else
+           return syms[mid];
+       }
+    }
+  return NULL;
+}
+
+/* Create synthetic symbols, effectively restoring "dot-symbol" function
+   entry syms.  */
+
+static long
+ppc64_elf_get_synthetic_symtab (bfd *abfd,
+                               long static_count, asymbol **static_syms,
+                               long dyn_count, asymbol **dyn_syms,
+                               asymbol **ret)
+{
+  asymbol *s;
+  long i;
+  long count;
+  char *names;
+  long symcount, codesecsym, codesecsymend, secsymend, opdsymend;
+  asection *opd;
+  bfd_boolean relocatable = (abfd->flags & (EXEC_P | DYNAMIC)) == 0;
+  asymbol **syms;
+
+  *ret = NULL;
+
+  opd = bfd_get_section_by_name (abfd, ".opd");
+  if (opd == NULL)
+    return 0;
+
+  symcount = static_count;
+  if (!relocatable)
+    symcount += dyn_count;
+  if (symcount == 0)
+    return 0;
+
+  syms = bfd_malloc ((symcount + 1) * sizeof (*syms));
+  if (syms == NULL)
+    return 0;
+
+  if (!relocatable && static_count != 0 && dyn_count != 0)
+    {
+      /* Use both symbol tables.  */
+      memcpy (syms, static_syms, static_count * sizeof (*syms));
+      memcpy (syms + static_count, dyn_syms, (dyn_count + 1) * sizeof (*syms));
+    }
+  else if (!relocatable && static_count == 0)
+    memcpy (syms, dyn_syms, (symcount + 1) * sizeof (*syms));
+  else
+    memcpy (syms, static_syms, (symcount + 1) * sizeof (*syms));
+
+  synthetic_opd = opd;
+  synthetic_relocatable = relocatable;
+  qsort (syms, symcount, sizeof (*syms), compare_symbols);
+
+  if (!relocatable && symcount > 1)
+    {
+      long j;
+      /* Trim duplicate syms, since we may have merged the normal and
+        dynamic symbols.  Actually, we only care about syms that have
+        different values, so trim any with the same value.  */ 
+      for (i = 1, j = 1; i < symcount; ++i)
+       if (syms[i - 1]->value + syms[i - 1]->section->vma
+           != syms[i]->value + syms[i]->section->vma)
+         syms[j++] = syms[i];
+      symcount = j;
+    }
+
+  i = 0;
+  if (syms[i]->section == opd)
+    ++i;
+  codesecsym = i;
+
+  for (; i < symcount; ++i)
+    if (((syms[i]->section->flags & (SEC_CODE | SEC_ALLOC | SEC_THREAD_LOCAL))
+        != (SEC_CODE | SEC_ALLOC))
+       || (syms[i]->flags & BSF_SECTION_SYM) == 0)
+      break;
+  codesecsymend = i;
+
+  for (; i < symcount; ++i)
+    if ((syms[i]->flags & BSF_SECTION_SYM) == 0)
+      break;
+  secsymend = i;
+
+  for (; i < symcount; ++i)
+    if (syms[i]->section != opd)
+      break;
+  opdsymend = i;
+
+  for (; i < symcount; ++i)
+    if ((syms[i]->section->flags & (SEC_CODE | SEC_ALLOC | SEC_THREAD_LOCAL))
+       != (SEC_CODE | SEC_ALLOC))
+      break;
+  symcount = i;
+
+  count = 0;
+  if (opdsymend == secsymend)
+    goto done;
+
+  if (relocatable)
+    {
+      bfd_boolean (*slurp_relocs) (bfd *, asection *, asymbol **, bfd_boolean);
+      arelent *r;
+      size_t size;
+      long relcount;
+
+      slurp_relocs = get_elf_backend_data (abfd)->s->slurp_reloc_table;
+      relcount = (opd->flags & SEC_RELOC) ? opd->reloc_count : 0;
+
+      if (! relcount
+         || ! (*slurp_relocs) (abfd, opd, static_syms, FALSE))
+       goto done;
+
+      size = 0;
+      for (i = secsymend, r = opd->relocation; i < opdsymend; ++i)
+       {
+         asymbol *sym;
+
+         while (r < opd->relocation + relcount
+                && r->address < syms[i]->value + opd->vma)
+           ++r;
+
+         if (r == opd->relocation + relcount)
+           break;
+
+         if (r->address != syms[i]->value + opd->vma)
+           continue;
+
+         if (r->howto->type != R_PPC64_ADDR64)
+           continue;
+
+         sym = *r->sym_ptr_ptr;
+         if (!sym_exists_at (syms, opdsymend, symcount,
+                             sym->section->id, sym->value + r->addend))
+           {
+             ++count;
+             size += sizeof (asymbol);
+             size += strlen (syms[i]->name) + 2;
+           }
+       }
+
+      s = *ret = bfd_malloc (size);
+      if (s == NULL)
+       {
+         count = 0;
+         goto done;
+       }
+
+      names = (char *) (s + count);
+
+      for (i = secsymend, r = opd->relocation; i < opdsymend; ++i)
+       {
+         asymbol *sym;
+
+         while (r < opd->relocation + relcount
+                && r->address < syms[i]->value + opd->vma)
+           ++r;
+
+         if (r == opd->relocation + relcount)
+           break;
+
+         if (r->address != syms[i]->value + opd->vma)
+           continue;
+
+         if (r->howto->type != R_PPC64_ADDR64)
+           continue;
+
+         sym = *r->sym_ptr_ptr;
+         if (!sym_exists_at (syms, opdsymend, symcount,
+                             sym->section->id, sym->value + r->addend))
+           {
+             size_t len;
+
+             *s = *syms[i];
+             s->section = sym->section;
+             s->value = sym->value + r->addend;
+             s->name = names;
+             *names++ = '.';
+             len = strlen (syms[i]->name);
+             memcpy (names, syms[i]->name, len + 1);
+             names += len + 1;
+             s++;
+           }
+       }
+    }
+  else
+    {
+      bfd_byte *contents;
+      size_t size;
+
+      if (!bfd_malloc_and_get_section (abfd, opd, &contents))
+       {
+         if (contents)
+           {
+           free_contents_and_exit:
+             free (contents);
+           }
+         goto done;
+       }
+
+      size = 0;
+      for (i = secsymend; i < opdsymend; ++i)
+       {
+         bfd_vma ent;
+
+         ent = bfd_get_64 (abfd, contents + syms[i]->value);
+         if (!sym_exists_at (syms, opdsymend, symcount, -1, ent))
+           {
+             ++count;
+             size += sizeof (asymbol);
+             size += strlen (syms[i]->name) + 2;
+           }
+       }
+
+      s = *ret = bfd_malloc (size);
+      if (s == NULL)
+       {
+         count = 0;
+         goto free_contents_and_exit;
+       }
+
+      names = (char *) (s + count);
+
+      for (i = secsymend; i < opdsymend; ++i)
+       {
+         bfd_vma ent;
+
+         ent = bfd_get_64 (abfd, contents + syms[i]->value);
+         if (!sym_exists_at (syms, opdsymend, symcount, -1, ent))
+           {
+             long lo, hi;
+             size_t len;
+             asection *sec = abfd->sections;
+
+             *s = *syms[i];
+             lo = codesecsym;
+             hi = codesecsymend;
+             while (lo < hi)
+               {
+                 long mid = (lo + hi) >> 1;
+                 if (syms[mid]->section->vma < ent)
+                   lo = mid + 1;
+                 else if (syms[mid]->section->vma > ent)
+                   hi = mid;
+                 else
+                   {
+                     sec = syms[mid]->section;
+                     break;
+                   }
+               }
+
+             if (lo >= hi && lo > codesecsym)
+               sec = syms[lo - 1]->section;
+
+             for (; sec != NULL; sec = sec->next)
+               {
+                 if (sec->vma > ent)
+                   break;
+                 if ((sec->flags & SEC_ALLOC) == 0
+                     || (sec->flags & SEC_LOAD) == 0)
+                   break;
+                 if ((sec->flags & SEC_CODE) != 0)
+                   s->section = sec;
+               }
+             s->value = ent - s->section->vma;
+             s->name = names;
+             *names++ = '.';
+             len = strlen (syms[i]->name);
+             memcpy (names, syms[i]->name, len + 1);
+             names += len + 1;
+             s++;
+           }
+       }
+      free (contents);
+    }
+
+ done:
+  free (syms);
+  return count;
+}
 \f
 /* The following functions are specific to the ELF linker, while
    functions above are used generally.  Those named ppc64_elf_* are
@@ -2556,7 +2992,11 @@ ppc64_elf_new_section_hook (bfd *abfd, asection *sec)
    creating a shared library containing foo, we need to have both symbols
    dynamic so that references to .foo are satisfied during the early
    stages of linking.  Otherwise the linker might decide to pull in a
-   definition from some other object, eg. a static library.  */
+   definition from some other object, eg. a static library.
+
+   Update: As of August 2004, we support a new convention.  Function
+   calls may use the function descriptor symbol, ie. "bl foo".  This
+   behaves exactly as "bl .foo".  */
 
 /* The linker needs to keep track of the number of relocs that it
    decides to copy as dynamic relocs in check_relocs for each symbol.
@@ -2641,7 +3081,7 @@ struct plt_entry
    pointers must reference the descriptor.  Thus, a function pointer
    initialized to the address of a function in a shared library will
    either require a copy reloc, or a dynamic reloc.  Using a copy reloc
-   redefines the function desctriptor symbol to point to the copy.  This
+   redefines the function descriptor symbol to point to the copy.  This
    presents a problem as a plt entry for that function is also
    initialized from the function descriptor symbol and the copy reloc
    may not be initialized first.  */
@@ -2666,7 +3106,10 @@ struct plt_entry
    .   bctr
 
    ppc_stub_plt_call:
-   Used to call a function in a shared library.
+   Used to call a function in a shared library.  If it so happens that
+   the plt entry referenced crosses a 64k boundary, then an extra
+   "addis %r12,%r12,1" will be inserted before the load at xxx+8 or
+   xxx+16 as appropriate.
    .   addis   %r12,%r2,xxx@toc@ha
    .   std     %r2,40(%r1)
    .   ld      %r11,xxx+0@toc@l(%r12)
@@ -2755,12 +3198,19 @@ struct ppc_link_hash_entry
   struct ppc_dyn_relocs *dyn_relocs;
 
   /* Link between function code and descriptor symbols.  */
-  struct elf_link_hash_entry *oh;
+  struct ppc_link_hash_entry *oh;
 
   /* Flag function code and descriptor symbols.  */
   unsigned int is_func:1;
   unsigned int is_func_descriptor:1;
-  unsigned int is_entry:1;
+
+  /* Whether global opd sym has been adjusted or not.
+     After ppc64_elf_edit_opd has run, this flag should be set for all
+     globals defined in any opd section.  */
+  unsigned int adjust_done:1;
+
+  /* Set if we twiddled this symbol to weak at some stage.  */
+  unsigned int was_undefined:1;
 
   /* Contexts in which symbol is used in the GOT (or TOC).
      TLS_GD .. TLS_EXPLICIT bits are or'd into the mask as the
@@ -2810,10 +3260,6 @@ struct ppc_link_hash_table
     bfd_vma toc_off;
   } *stub_group;
 
-  /* Support for multiple toc sections.  */
-  unsigned int no_multi_toc;
-  unsigned int multi_toc_needed;
-
   /* Temp used when calculating TOC pointers.  */
   bfd_vma toc_curr;
 
@@ -2837,24 +3283,25 @@ struct ppc_link_hash_table
   asection *brlt;
   asection *relbrlt;
 
-  /* Shortcut to .__tls_get_addr.  */
-  struct elf_link_hash_entry *tls_get_addr;
+  /* Shortcut to .__tls_get_addr and __tls_get_addr.  */
+  struct ppc_link_hash_entry *tls_get_addr;
+  struct ppc_link_hash_entry *tls_get_addr_fd;
 
   /* Statistics.  */
   unsigned long stub_count[ppc_stub_plt_call];
 
   /* Set if we should emit symbols for stubs.  */
-  unsigned int emit_stub_syms;
+  unsigned int emit_stub_syms:1;
 
   /* Set on error.  */
-  unsigned int stub_error;
+  unsigned int stub_error:1;
 
   /* Flag set when small branches are detected.  Used to
      select suitable defaults for the stub group size.  */
-  unsigned int has_14bit_branch;
+  unsigned int has_14bit_branch:1;
 
-  /* Set if we detect a reference undefined weak symbol.  */
-  unsigned int have_undefweak;
+  /* Temp used by ppc64_elf_check_directives.  */
+  unsigned int twiddled_syms:1;
 
   /* Incremented every time we size stubs.  */
   unsigned int stub_iteration;
@@ -2970,7 +3417,8 @@ link_hash_newfunc (struct bfd_hash_entry *entry,
       eh->oh = NULL;
       eh->is_func = 0;
       eh->is_func_descriptor = 0;
-      eh->is_entry = 0;
+      eh->adjust_done = 0;
+      eh->was_undefined = 0;
       eh->tls_mask = 0;
     }
 
@@ -3094,12 +3542,11 @@ ppc_stub_name (const asection *input_section,
 static struct ppc_stub_hash_entry *
 ppc_get_stub_entry (const asection *input_section,
                    const asection *sym_sec,
-                   struct elf_link_hash_entry *hash,
+                   struct ppc_link_hash_entry *h,
                    const Elf_Internal_Rela *rel,
                    struct ppc_link_hash_table *htab)
 {
   struct ppc_stub_hash_entry *stub_entry;
-  struct ppc_link_hash_entry *h = (struct ppc_link_hash_entry *) hash;
   const asection *id_sec;
 
   /* If this input section is part of a group of sections sharing one
@@ -3178,9 +3625,8 @@ ppc_add_stub (const char *stub_name,
                                     TRUE, FALSE);
   if (stub_entry == NULL)
     {
-      (*_bfd_error_handler) (_("%s: cannot create stub entry %s"),
-                            bfd_archive_filename (section->owner),
-                            stub_name);
+      (*_bfd_error_handler) (_("%B: cannot create stub entry %s"),
+                            section->owner, stub_name);
       return NULL;
     }
 
@@ -3355,7 +3801,6 @@ ppc64_elf_copy_indirect_symbol
 
   edir->is_func |= eind->is_func;
   edir->is_func_descriptor |= eind->is_func_descriptor;
-  edir->is_entry |= eind->is_entry;
   edir->tls_mask |= eind->tls_mask;
 
   mask = (ELF_LINK_HASH_REF_DYNAMIC | ELF_LINK_HASH_REF_REGULAR
@@ -3447,83 +3892,250 @@ ppc64_elf_copy_indirect_symbol
     BFD_ASSERT (eind->elf.dynindx == -1);
 }
 
-/* Set a flag, used by ppc64_elf_gc_mark_hook, on the entry symbol and
-   symbols undefined on the command-line.  */
+/* Find the function descriptor hash entry from the given function code
+   hash entry FH.  Link the entries via their OH fields.  */
 
-bfd_boolean
-ppc64_elf_mark_entry_syms (struct bfd_link_info *info)
+static struct ppc_link_hash_entry *
+get_fdh (struct ppc_link_hash_entry *fh, struct ppc_link_hash_table *htab)
 {
-  struct ppc_link_hash_table *htab;
-  struct bfd_sym_chain *sym;
+  struct ppc_link_hash_entry *fdh = fh->oh;
 
-  htab = ppc_hash_table (info);
-  for (sym = info->gc_sym_list; sym; sym = sym->next)
+  if (fdh == NULL)
     {
-      struct elf_link_hash_entry *h;
+      const char *fd_name = fh->elf.root.root.string + 1;
 
-      h = elf_link_hash_lookup (&htab->elf, sym->name, FALSE, FALSE, FALSE);
-      if (h != NULL)
-       ((struct ppc_link_hash_entry *) h)->is_entry = 1;
+      fdh = (struct ppc_link_hash_entry *)
+       elf_link_hash_lookup (&htab->elf, fd_name, FALSE, FALSE, FALSE);
+      if (fdh != NULL)
+       {
+         fdh->is_func_descriptor = 1;
+         fdh->oh = fh;
+         fh->is_func = 1;
+         fh->oh = fdh;
+       }
     }
-  return TRUE;
-}
 
-static bfd_boolean
-update_local_sym_info (bfd *abfd, Elf_Internal_Shdr *symtab_hdr,
-                      unsigned long r_symndx, bfd_vma r_addend, int tls_type)
-{
-  struct got_entry **local_got_ents = elf_local_got_ents (abfd);
-  char *local_got_tls_masks;
+  return fdh;
+}
 
-  if (local_got_ents == NULL)
-    {
-      bfd_size_type size = symtab_hdr->sh_info;
+/* Hacks to support old ABI code.
+   When making function calls, old ABI code references function entry
+   points (dot symbols), while new ABI code references the function
+   descriptor symbol.  We need to make any combination of reference and
+   definition work together, without breaking archive linking.
 
-      size *= sizeof (*local_got_ents) + sizeof (*local_got_tls_masks);
-      local_got_ents = bfd_zalloc (abfd, size);
-      if (local_got_ents == NULL)
-       return FALSE;
-      elf_local_got_ents (abfd) = local_got_ents;
-    }
+   For a defined function "foo" and an undefined call to "bar":
+   An old object defines "foo" and ".foo", references ".bar" (possibly
+   "bar" too).
+   A new object defines "foo" and references "bar".
 
-  if ((tls_type & TLS_EXPLICIT) == 0)
-    {
-      struct got_entry *ent;
+   A new object thus has no problem with its undefined symbols being
+   satisfied by definitions in an old object.  On the other hand, the
+   old object won't have ".bar" satisfied by a new object.  */
 
-      for (ent = local_got_ents[r_symndx]; ent != NULL; ent = ent->next)
-       if (ent->addend == r_addend
-           && ent->owner == abfd
-           && ent->tls_type == tls_type)
-         break;
-      if (ent == NULL)
-       {
-         bfd_size_type amt = sizeof (*ent);
-         ent = bfd_alloc (abfd, amt);
-         if (ent == NULL)
-           return FALSE;
-         ent->next = local_got_ents[r_symndx];
-         ent->addend = r_addend;
-         ent->owner = abfd;
-         ent->tls_type = tls_type;
-         ent->got.refcount = 0;
-         local_got_ents[r_symndx] = ent;
-       }
-      ent->got.refcount += 1;
-    }
+/* Fix function descriptor symbols defined in .opd sections to be
+   function type.  */
 
-  local_got_tls_masks = (char *) (local_got_ents + symtab_hdr->sh_info);
-  local_got_tls_masks[r_symndx] |= tls_type;
+static bfd_boolean
+ppc64_elf_add_symbol_hook (bfd *ibfd ATTRIBUTE_UNUSED,
+                          struct bfd_link_info *info ATTRIBUTE_UNUSED,
+                          Elf_Internal_Sym *isym,
+                          const char **name ATTRIBUTE_UNUSED,
+                          flagword *flags ATTRIBUTE_UNUSED,
+                          asection **sec,
+                          bfd_vma *value ATTRIBUTE_UNUSED)
+{
+  if (*sec != NULL
+      && strcmp (bfd_get_section_name (ibfd, *sec), ".opd") == 0)
+    isym->st_info = ELF_ST_INFO (ELF_ST_BIND (isym->st_info), STT_FUNC);
   return TRUE;
 }
 
+/* This function makes an old ABI object reference to ".bar" cause the
+   inclusion of a new ABI object archive that defines "bar".  */
+
+static struct elf_link_hash_entry *
+ppc64_elf_archive_symbol_lookup (bfd *abfd,
+                                struct bfd_link_info *info,
+                                const char *name)
+{
+  struct elf_link_hash_entry *h;
+  char *dot_name;
+  size_t len;
+
+  h = _bfd_elf_archive_symbol_lookup (abfd, info, name);
+  if (h != NULL)
+    return h;
+
+  if (name[0] == '.')
+    return h;
+
+  len = strlen (name);
+  dot_name = bfd_alloc (abfd, len + 2);
+  if (dot_name == NULL)
+    return (struct elf_link_hash_entry *) 0 - 1;
+  dot_name[0] = '.';
+  memcpy (dot_name + 1, name, len + 1);
+  h = _bfd_elf_archive_symbol_lookup (abfd, info, dot_name);
+  bfd_release (abfd, dot_name);
+  return h;
+}
+
+/* This function satisfies all old ABI object references to ".bar" if a
+   new ABI object defines "bar".  Well, at least, undefined dot symbols
+   are made weak.  This stops later archive searches from including an
+   object if we already have a function descriptor definition.  It also
+   prevents the linker complaining about undefined symbols.
+   We also check and correct mismatched symbol visibility here.  The
+   most restrictive visibility of the function descriptor and the
+   function entry symbol is used.  */
+
 static bfd_boolean
-update_plt_info (bfd *abfd, struct ppc_link_hash_entry *eh, bfd_vma addend)
+add_symbol_adjust (struct elf_link_hash_entry *h, void *inf)
 {
-  struct plt_entry *ent;
+  struct bfd_link_info *info;
+  struct ppc_link_hash_table *htab;
+  struct ppc_link_hash_entry *eh;
+  struct ppc_link_hash_entry *fdh;
 
-  for (ent = eh->elf.plt.plist; ent != NULL; ent = ent->next)
-    if (ent->addend == addend)
-      break;
+  if (h->root.type == bfd_link_hash_indirect)
+    return TRUE;
+
+  if (h->root.type == bfd_link_hash_warning)
+    h = (struct elf_link_hash_entry *) h->root.u.i.link;
+
+  if (h->root.root.string[0] != '.')
+    return TRUE;
+
+  info = inf;
+  htab = ppc_hash_table (info);
+  eh = (struct ppc_link_hash_entry *) h;
+  fdh = get_fdh (eh, htab);
+  if (fdh != NULL)
+    {
+      unsigned entry_vis = ELF_ST_VISIBILITY (eh->elf.other) - 1;
+      unsigned descr_vis = ELF_ST_VISIBILITY (fdh->elf.other) - 1;
+      if (entry_vis < descr_vis)
+       fdh->elf.other += entry_vis - descr_vis;
+      else if (entry_vis > descr_vis)
+       eh->elf.other += descr_vis - entry_vis;
+
+      if (eh->elf.root.type == bfd_link_hash_undefined)
+       {
+         eh->elf.root.type = bfd_link_hash_undefweak;
+         eh->was_undefined = 1;
+         htab->twiddled_syms = 1;
+       }
+    }
+
+  return TRUE;
+}
+
+static bfd_boolean
+ppc64_elf_check_directives (bfd *abfd ATTRIBUTE_UNUSED,
+                           struct bfd_link_info *info)
+{
+  struct ppc_link_hash_table *htab;
+
+  htab = ppc_hash_table (info);
+  if (!is_ppc64_target (htab->elf.root.creator))
+    return TRUE;
+
+  elf_link_hash_traverse (&htab->elf, add_symbol_adjust, info);
+
+  /* We need to fix the undefs list for any syms we have twiddled to
+     undef_weak.  */
+  if (htab->twiddled_syms)
+    {
+      struct bfd_link_hash_entry **pun;
+
+      pun = &htab->elf.root.undefs;
+      while (*pun != NULL)
+       {
+         struct bfd_link_hash_entry *h = *pun;
+
+         if (h->type != bfd_link_hash_undefined
+             && h->type != bfd_link_hash_common)
+           {
+             *pun = h->und_next;
+             h->und_next = NULL;
+             if (h == htab->elf.root.undefs_tail)
+               {
+                 if (pun == &htab->elf.root.undefs)
+                   htab->elf.root.undefs_tail = NULL;
+                 else
+                   /* pun points at an und_next field.  Go back to
+                      the start of the link_hash_entry.  */
+                   htab->elf.root.undefs_tail = (struct bfd_link_hash_entry *)
+                     ((char *) pun - ((char *) &h->und_next - (char *) h));
+                 break;
+               }
+           }
+         else
+           pun = &h->und_next;
+       }
+
+      htab->twiddled_syms = 0;
+    }
+  return TRUE;
+}
+
+static bfd_boolean
+update_local_sym_info (bfd *abfd, Elf_Internal_Shdr *symtab_hdr,
+                      unsigned long r_symndx, bfd_vma r_addend, int tls_type)
+{
+  struct got_entry **local_got_ents = elf_local_got_ents (abfd);
+  char *local_got_tls_masks;
+
+  if (local_got_ents == NULL)
+    {
+      bfd_size_type size = symtab_hdr->sh_info;
+
+      size *= sizeof (*local_got_ents) + sizeof (*local_got_tls_masks);
+      local_got_ents = bfd_zalloc (abfd, size);
+      if (local_got_ents == NULL)
+       return FALSE;
+      elf_local_got_ents (abfd) = local_got_ents;
+    }
+
+  if ((tls_type & TLS_EXPLICIT) == 0)
+    {
+      struct got_entry *ent;
+
+      for (ent = local_got_ents[r_symndx]; ent != NULL; ent = ent->next)
+       if (ent->addend == r_addend
+           && ent->owner == abfd
+           && ent->tls_type == tls_type)
+         break;
+      if (ent == NULL)
+       {
+         bfd_size_type amt = sizeof (*ent);
+         ent = bfd_alloc (abfd, amt);
+         if (ent == NULL)
+           return FALSE;
+         ent->next = local_got_ents[r_symndx];
+         ent->addend = r_addend;
+         ent->owner = abfd;
+         ent->tls_type = tls_type;
+         ent->got.refcount = 0;
+         local_got_ents[r_symndx] = ent;
+       }
+      ent->got.refcount += 1;
+    }
+
+  local_got_tls_masks = (char *) (local_got_ents + symtab_hdr->sh_info);
+  local_got_tls_masks[r_symndx] |= tls_type;
+  return TRUE;
+}
+
+static bfd_boolean
+update_plt_info (bfd *abfd, struct ppc_link_hash_entry *eh, bfd_vma addend)
+{
+  struct plt_entry *ent;
+
+  for (ent = eh->elf.plt.plist; ent != NULL; ent = ent->next)
+    if (ent->addend == addend)
+      break;
   if (ent == NULL)
     {
       bfd_size_type amt = sizeof (*ent);
@@ -3541,31 +4153,6 @@ update_plt_info (bfd *abfd, struct ppc_link_hash_entry *eh, bfd_vma addend)
   return TRUE;
 }
 
-/* Find the function descriptor hash entry from the given function code
-   hash entry FH.  Link the entries via their OH fields.  */
-static struct ppc_link_hash_entry *
-get_fdh (struct ppc_link_hash_entry *fh, struct ppc_link_hash_table *htab)
-{
-  struct ppc_link_hash_entry *fdh = (struct ppc_link_hash_entry *) fh->oh;
-
-  if (fdh == NULL)
-    {
-      const char *fd_name = fh->elf.root.root.string + 1;
-
-      fdh = (struct ppc_link_hash_entry *)
-       elf_link_hash_lookup (&htab->elf, fd_name, FALSE, FALSE, FALSE);
-      if (fdh != NULL)
-       {
-         fdh->is_func_descriptor = 1;
-         fdh->oh = &fh->elf;
-         fh->is_func = 1;
-         fh->oh = &fdh->elf;
-       }
-    }
-
-  return fdh;
-}
-
 /* Look through the relocs for a section during the first phase, and
    calculate needed space in the global offset table, procedure
    linkage table, and dynamic reloc sections.  */
@@ -3585,6 +4172,15 @@ ppc64_elf_check_relocs (bfd *abfd, struct bfd_link_info *info,
   if (info->relocatable)
     return TRUE;
 
+  /* Don't do anything special with non-loaded, non-alloced sections.
+     In particular, any relocs in such sections should not affect GOT
+     and PLT reference counting (ie. we don't allow them to create GOT
+     or PLT entries), there's no possibility or desire to optimize TLS
+     relocs, and there's not much point in propagating relocs to shared
+     libs that the dynamic linker won't relocate.  */
+  if ((sec->flags & SEC_ALLOC) == 0)
+    return TRUE;
+
   htab = ppc_hash_table (info);
   symtab_hdr = &elf_tdata (abfd)->symtab_hdr;
 
@@ -3612,7 +4208,7 @@ ppc64_elf_check_relocs (bfd *abfd, struct bfd_link_info *info,
       };
       bfd_size_type amt;
 
-      amt = sec->_raw_size * sizeof (union opd_info) / 24;
+      amt = sec->size * sizeof (union opd_info) / 8;
       opd_sym_map = bfd_zalloc (abfd, amt);
       if (opd_sym_map == NULL)
        return FALSE;
@@ -3776,14 +4372,14 @@ ppc64_elf_check_relocs (bfd *abfd, struct bfd_link_info *info,
          /* This relocation describes the C++ object vtable hierarchy.
             Reconstruct it for later use during GC.  */
        case R_PPC64_GNU_VTINHERIT:
-         if (!_bfd_elf64_gc_record_vtinherit (abfd, sec, h, rel->r_offset))
+         if (!bfd_elf_gc_record_vtinherit (abfd, sec, h, rel->r_offset))
            return FALSE;
          break;
 
          /* This relocation describes which C++ vtable entries are actually
             used.  Record for later use during GC.  */
        case R_PPC64_GNU_VTENTRY:
-         if (!_bfd_elf64_gc_record_vtentry (abfd, sec, h, rel->r_addend))
+         if (!bfd_elf_gc_record_vtentry (abfd, sec, h, rel->r_addend))
            return FALSE;
          break;
 
@@ -3794,23 +4390,30 @@ ppc64_elf_check_relocs (bfd *abfd, struct bfd_link_info *info,
          /* Fall through.  */
 
        case R_PPC64_REL24:
-         if (h != NULL
-             && h->root.root.string[0] == '.'
-             && h->root.root.string[1] != 0)
+         if (h != NULL)
            {
              /* We may need a .plt entry if the function this reloc
                 refers to is in a shared lib.  */
              if (!update_plt_info (abfd, (struct ppc_link_hash_entry *) h,
                                    rel->r_addend))
                return FALSE;
-             if (h == htab->tls_get_addr)
+             if (h == &htab->tls_get_addr->elf
+                 || h == &htab->tls_get_addr_fd->elf)
                sec->has_tls_reloc = 1;
-             else if ((strncmp (h->root.root.string, ".__tls_get_addr", 15)
-                       == 0)
+             else if (htab->tls_get_addr == NULL
+                      && !strncmp (h->root.root.string, ".__tls_get_addr", 15)
                       && (h->root.root.string[15] == 0
                           || h->root.root.string[15] == '@'))
                {
-                 htab->tls_get_addr = h;
+                 htab->tls_get_addr = (struct ppc_link_hash_entry *) h;
+                 sec->has_tls_reloc = 1;
+               }
+             else if (htab->tls_get_addr_fd == NULL
+                      && !strncmp (h->root.root.string, "__tls_get_addr", 14)
+                      && (h->root.root.string[14] == 0
+                          || h->root.root.string[14] == '@'))
+               {
+                 htab->tls_get_addr_fd = (struct ppc_link_hash_entry *) h;
                  sec->has_tls_reloc = 1;
                }
            }
@@ -3856,7 +4459,7 @@ ppc64_elf_check_relocs (bfd *abfd, struct bfd_link_info *info,
          if (ppc64_elf_section_data (sec)->t_symndx == NULL)
            {
              /* One extra to simplify get_tls_mask.  */
-             bfd_size_type amt = sec->_raw_size * sizeof (unsigned) / 8 + 1;
+             bfd_size_type amt = sec->size * sizeof (unsigned) / 8 + 1;
              ppc64_elf_section_data (sec)->t_symndx = bfd_zalloc (abfd, amt);
              if (ppc64_elf_section_data (sec)->t_symndx == NULL)
                return FALSE;
@@ -3891,24 +4494,29 @@ ppc64_elf_check_relocs (bfd *abfd, struct bfd_link_info *info,
 
        case R_PPC64_ADDR64:
          if (opd_sym_map != NULL
-             && h != NULL
-             && h->root.root.string[0] == '.'
-             && h->root.root.string[1] != 0)
-           get_fdh ((struct ppc_link_hash_entry *) h, htab);
-
-         if (opd_sym_map != NULL
-             && h == NULL
              && rel + 1 < rel_end
              && ELF64_R_TYPE ((rel + 1)->r_info) == R_PPC64_TOC)
            {
-             asection *s;
+             if (h != NULL)
+               {
+                 if (h->root.root.string[0] == '.'
+                     && h->root.root.string[1] != 0
+                     && get_fdh ((struct ppc_link_hash_entry *) h, htab))
+                   ;
+                 else
+                   ((struct ppc_link_hash_entry *) h)->is_func = 1;
+               }
+             else
+               {
+                 asection *s;
 
-             s = bfd_section_from_r_symndx (abfd, &htab->sym_sec, sec,
-                                            r_symndx);
-             if (s == NULL)
-               return FALSE;
-             else if (s != sec)
-               opd_sym_map[rel->r_offset / 24] = s;
+                 s = bfd_section_from_r_symndx (abfd, &htab->sym_sec, sec,
+                                                r_symndx);
+                 if (s == NULL)
+                   return FALSE;
+                 else if (s != sec)
+                   opd_sym_map[rel->r_offset / 8] = s;
+               }
            }
          /* Fall through.  */
 
@@ -3942,10 +4550,6 @@ ppc64_elf_check_relocs (bfd *abfd, struct bfd_link_info *info,
          if (NO_OPD_RELOCS && opd_sym_map != NULL)
            break;
 
-         /* Don't propagate relocs that the dynamic linker won't relocate.  */
-         if ((sec->flags & SEC_ALLOC) == 0)
-           break;
-
          /* If we are creating a shared library, and this is a reloc
             against a global symbol, or a non PC relative reloc
             against a local symbol, then we need to copy the reloc
@@ -4005,8 +4609,8 @@ ppc64_elf_check_relocs (bfd *abfd, struct bfd_link_info *info,
                                 name + 5) != 0)
                    {
                      (*_bfd_error_handler)
-                       (_("%s: bad relocation section name `%s\'"),
-                        bfd_archive_filename (abfd), name);
+                       (_("%B: bad relocation section name `%s\'"),
+                        abfd, name);
                      bfd_set_error (bfd_error_bad_value);
                    }
 
@@ -4078,22 +4682,157 @@ ppc64_elf_check_relocs (bfd *abfd, struct bfd_link_info *info,
   return TRUE;
 }
 
+/* OFFSET in OPD_SEC specifies a function descriptor.  Return the address
+   of the code entry point, and its section.  */
+
+static bfd_vma
+opd_entry_value (asection *opd_sec,
+                bfd_vma offset,
+                asection **code_sec,
+                bfd_vma *code_off)
+{
+  bfd *opd_bfd = opd_sec->owner;
+  Elf_Internal_Rela *lo, *hi, *look;
+
+  /* Go find the opd reloc at the sym address.  */
+  lo = _bfd_elf_link_read_relocs (opd_bfd, opd_sec, NULL, NULL, TRUE);
+  BFD_ASSERT (lo != NULL);
+  hi = lo + opd_sec->reloc_count - 1; /* ignore last reloc */
+
+  while (lo < hi)
+    {
+      look = lo + (hi - lo) / 2;
+      if (look->r_offset < offset)
+       lo = look + 1;
+      else if (look->r_offset > offset)
+       hi = look;
+      else
+       {
+         Elf_Internal_Shdr *symtab_hdr = &elf_tdata (opd_bfd)->symtab_hdr;
+         if (ELF64_R_TYPE (look->r_info) == R_PPC64_ADDR64
+             && ELF64_R_TYPE ((look + 1)->r_info) == R_PPC64_TOC)
+           {
+             unsigned long symndx = ELF64_R_SYM (look->r_info);
+             bfd_vma val;
+             asection *sec;
+
+             if (symndx < symtab_hdr->sh_info)
+               {
+                 Elf_Internal_Sym *sym;
+
+                 sym = (Elf_Internal_Sym *) symtab_hdr->contents;
+                 if (sym == NULL)
+                   {
+                     sym = bfd_elf_get_elf_syms (opd_bfd, symtab_hdr,
+                                                 symtab_hdr->sh_info,
+                                                 0, NULL, NULL, NULL);
+                     if (sym == NULL)
+                       return (bfd_vma) -1;
+                     symtab_hdr->contents = (bfd_byte *) sym;
+                   }
+
+                 sym += symndx;
+                 val = sym->st_value;
+                 sec = NULL;
+                 if ((sym->st_shndx != SHN_UNDEF
+                      && sym->st_shndx < SHN_LORESERVE)
+                     || sym->st_shndx > SHN_HIRESERVE)
+                   sec = bfd_section_from_elf_index (opd_bfd, sym->st_shndx);
+                 BFD_ASSERT ((sec->flags & SEC_MERGE) == 0);
+               }
+             else
+               {
+                 struct elf_link_hash_entry **sym_hashes;
+                 struct elf_link_hash_entry *rh;
+
+                 sym_hashes = elf_sym_hashes (opd_bfd);
+                 rh = sym_hashes[symndx - symtab_hdr->sh_info];
+                 while (rh->root.type == bfd_link_hash_indirect
+                        || rh->root.type == bfd_link_hash_warning)
+                   rh = ((struct elf_link_hash_entry *) rh->root.u.i.link);
+                 BFD_ASSERT (rh->root.type == bfd_link_hash_defined
+                             || rh->root.type == bfd_link_hash_defweak);
+                 val = rh->root.u.def.value;
+                 sec = rh->root.u.def.section;
+               }
+             val += look->r_addend;
+             if (code_off != NULL)
+               *code_off = val;
+             if (code_sec != NULL)
+               *code_sec = sec;
+             if (sec != NULL && sec->output_section != NULL)
+               val += sec->output_section->vma + sec->output_offset;
+             return val;
+           }
+         break;
+       }
+    }
+  return (bfd_vma) -1;
+}
+
 /* Return the section that should be marked against GC for a given
    relocation.  */
 
 static asection *
 ppc64_elf_gc_mark_hook (asection *sec,
-                       struct bfd_link_info *info ATTRIBUTE_UNUSED,
+                       struct bfd_link_info *info,
                        Elf_Internal_Rela *rel,
                        struct elf_link_hash_entry *h,
                        Elf_Internal_Sym *sym)
 {
-  asection *rsec = NULL;
+  asection *rsec;
+
+  /* First mark all our entry sym sections.  */
+  if (info->gc_sym_list != NULL)
+    {
+      struct ppc_link_hash_table *htab = ppc_hash_table (info);
+      struct bfd_sym_chain *sym = info->gc_sym_list;
+
+      info->gc_sym_list = NULL;
+      do
+       {
+         struct ppc_link_hash_entry *eh;
+
+         eh = (struct ppc_link_hash_entry *)
+           elf_link_hash_lookup (&htab->elf, sym->name, FALSE, FALSE, FALSE);
+         if (eh == NULL)
+           continue;
+         if (eh->elf.root.type != bfd_link_hash_defined
+             && eh->elf.root.type != bfd_link_hash_defweak)
+           continue;
+
+         if (eh->is_func_descriptor)
+           rsec = eh->oh->elf.root.u.def.section;
+         else if (get_opd_info (eh->elf.root.u.def.section) != NULL
+                  && opd_entry_value (eh->elf.root.u.def.section,
+                                      eh->elf.root.u.def.value,
+                                      &rsec, NULL) != (bfd_vma) -1)
+           ;
+         else
+           continue;
+
+         if (!rsec->gc_mark)
+           _bfd_elf_gc_mark (info, rsec, ppc64_elf_gc_mark_hook);
+
+         rsec = eh->elf.root.u.def.section;
+         if (!rsec->gc_mark)
+           _bfd_elf_gc_mark (info, rsec, ppc64_elf_gc_mark_hook);
+
+         sym = sym->next;
+       }
+      while (sym != NULL);
+    }
+
+  /* Syms return NULL if we're marking .opd, so we avoid marking all
+     function sections, as all functions are referenced in .opd.  */
+  rsec = NULL;
+  if (get_opd_info (sec) != NULL)
+    return rsec;
 
   if (h != NULL)
     {
       enum elf_ppc64_reloc_type r_type;
-      struct ppc_link_hash_entry *fdh;
+      struct ppc_link_hash_entry *eh;
 
       r_type = ELF64_R_TYPE (rel->r_info);
       switch (r_type)
@@ -4107,20 +4846,31 @@ ppc64_elf_gc_mark_hook (asection *sec,
            {
            case bfd_link_hash_defined:
            case bfd_link_hash_defweak:
-             fdh = (struct ppc_link_hash_entry *) h;
+             eh = (struct ppc_link_hash_entry *) h;
+             if (eh->oh != NULL && eh->oh->is_func_descriptor)
+               eh = eh->oh;
 
              /* Function descriptor syms cause the associated
                 function code sym section to be marked.  */
-             if (fdh->is_func_descriptor)
-               rsec = fdh->oh->root.u.def.section;
-
-             /* Function entry syms return NULL if they are in .opd
-                and are not ._start (or others undefined on the ld
-                command line).  Thus we avoid marking all function
-                sections, as all functions are referenced in .opd.  */
-             else if ((fdh->oh != NULL
-                       && ((struct ppc_link_hash_entry *) fdh->oh)->is_entry)
-                      || ppc64_elf_section_data (sec)->opd.func_sec == NULL)
+             if (eh->is_func_descriptor)
+               {
+                 /* They also mark their opd section.  */
+                 if (!eh->elf.root.u.def.section->gc_mark)
+                   _bfd_elf_gc_mark (info, eh->elf.root.u.def.section,
+                                     ppc64_elf_gc_mark_hook);
+
+                 rsec = eh->oh->elf.root.u.def.section;
+               }
+             else if (get_opd_info (eh->elf.root.u.def.section) != NULL
+                      && opd_entry_value (eh->elf.root.u.def.section,
+                                          eh->elf.root.u.def.value,
+                                          &rsec, NULL) != (bfd_vma) -1)
+               {
+                 if (!eh->elf.root.u.def.section->gc_mark)
+                   _bfd_elf_gc_mark (info, eh->elf.root.u.def.section,
+                                     ppc64_elf_gc_mark_hook);
+               }
+             else
                rsec = h->root.u.def.section;
              break;
 
@@ -4138,11 +4888,14 @@ ppc64_elf_gc_mark_hook (asection *sec,
       asection **opd_sym_section;
 
       rsec = bfd_section_from_elf_index (sec->owner, sym->st_shndx);
-      opd_sym_section = ppc64_elf_section_data (rsec)->opd.func_sec;
+      opd_sym_section = get_opd_info (rsec);
       if (opd_sym_section != NULL)
-       rsec = opd_sym_section[sym->st_value / 24];
-      else if (ppc64_elf_section_data (sec)->opd.func_sec != NULL)
-       rsec = NULL;
+       {
+         if (!rsec->gc_mark)
+           _bfd_elf_gc_mark (info, rsec, ppc64_elf_gc_mark_hook);
+
+         rsec = opd_sym_section[sym->st_value / 8];
+       }
     }
 
   return rsec;
@@ -4161,6 +4914,9 @@ ppc64_elf_gc_sweep_hook (bfd *abfd, struct bfd_link_info *info,
   struct got_entry **local_got_ents;
   const Elf_Internal_Rela *rel, *relend;
 
+  if ((sec->flags & SEC_ALLOC) == 0)
+    return TRUE;
+
   elf_section_data (sec)->local_dynrel = NULL;
 
   htab = ppc_hash_table (info);
@@ -4284,9 +5040,235 @@ ppc64_elf_gc_sweep_hook (bfd *abfd, struct bfd_link_info *info,
   return TRUE;
 }
 
+/* The maximum size of .sfpr.  */
+#define SFPR_MAX (218*4)
+
+struct sfpr_def_parms
+{
+  const char name[12];
+  unsigned char lo, hi;
+  bfd_byte * (*write_ent) (bfd *, bfd_byte *, int);
+  bfd_byte * (*write_tail) (bfd *, bfd_byte *, int);
+};
+
+/* Auto-generate _save*, _rest* functions in .sfpr.  */
+
+static unsigned int
+sfpr_define (struct bfd_link_info *info, const struct sfpr_def_parms *parm)
+{
+  struct ppc_link_hash_table *htab = ppc_hash_table (info);
+  unsigned int i;
+  size_t len = strlen (parm->name);
+  bfd_boolean writing = FALSE;
+  char sym[16];
+
+  memcpy (sym, parm->name, len);
+  sym[len + 2] = 0;
+
+  for (i = parm->lo; i <= parm->hi; i++)
+    {
+      struct elf_link_hash_entry *h;
+
+      sym[len + 0] = i / 10 + '0';
+      sym[len + 1] = i % 10 + '0';
+      h = elf_link_hash_lookup (&htab->elf, sym, FALSE, FALSE, TRUE);
+      if (h != NULL
+         && (h->elf_link_hash_flags & ELF_LINK_HASH_DEF_REGULAR) == 0)
+       {
+         h->root.type = bfd_link_hash_defined;
+         h->root.u.def.section = htab->sfpr;
+         h->root.u.def.value = htab->sfpr->size;
+         h->type = STT_FUNC;
+         h->elf_link_hash_flags |= ELF_LINK_HASH_DEF_REGULAR;
+         _bfd_elf_link_hash_hide_symbol (info, h, TRUE);
+         writing = TRUE;
+         if (htab->sfpr->contents == NULL)
+           {
+             htab->sfpr->contents = bfd_alloc (htab->elf.dynobj, SFPR_MAX);
+             if (htab->sfpr->contents == NULL)
+               return FALSE;
+           }
+       }
+      if (writing)
+       {
+         bfd_byte *p = htab->sfpr->contents + htab->sfpr->size;
+         if (i != parm->hi)
+           p = (*parm->write_ent) (htab->elf.dynobj, p, i);
+         else
+           p = (*parm->write_tail) (htab->elf.dynobj, p, i);
+         htab->sfpr->size = p - htab->sfpr->contents;
+       }
+    }
+
+  return TRUE;
+}
+
+static bfd_byte *
+savegpr0 (bfd *abfd, bfd_byte *p, int r)
+{
+  bfd_put_32 (abfd, STD_R0_0R1 + (r << 21) + (1 << 16) - (32 - r) * 8, p);
+  return p + 4;
+}
+
+static bfd_byte *
+savegpr0_tail (bfd *abfd, bfd_byte *p, int r)
+{
+  p = savegpr0 (abfd, p, r);
+  bfd_put_32 (abfd, STD_R0_0R1 + 16, p);
+  p = p + 4;
+  bfd_put_32 (abfd, BLR, p);
+  return p + 4;
+}
+
+static bfd_byte *
+restgpr0 (bfd *abfd, bfd_byte *p, int r)
+{
+  bfd_put_32 (abfd, LD_R0_0R1 + (r << 21) + (1 << 16) - (32 - r) * 8, p);
+  return p + 4;
+}
+
+static bfd_byte *
+restgpr0_tail (bfd *abfd, bfd_byte *p, int r)
+{
+  bfd_put_32 (abfd, LD_R0_0R1 + 16, p);
+  p = p + 4;
+  p = restgpr0 (abfd, p, r);
+  bfd_put_32 (abfd, MTLR_R0, p);
+  p = p + 4;
+  if (r == 29)
+    {
+      p = restgpr0 (abfd, p, 30);
+      p = restgpr0 (abfd, p, 31);
+    }
+  bfd_put_32 (abfd, BLR, p);
+  return p + 4;
+}
+
+static bfd_byte *
+savegpr1 (bfd *abfd, bfd_byte *p, int r)
+{
+  bfd_put_32 (abfd, STD_R0_0R12 + (r << 21) + (1 << 16) - (32 - r) * 8, p);
+  return p + 4;
+}
+
+static bfd_byte *
+savegpr1_tail (bfd *abfd, bfd_byte *p, int r)
+{
+  p = savegpr1 (abfd, p, r);
+  bfd_put_32 (abfd, BLR, p);
+  return p + 4;
+}
+
+static bfd_byte *
+restgpr1 (bfd *abfd, bfd_byte *p, int r)
+{
+  bfd_put_32 (abfd, LD_R0_0R12 + (r << 21) + (1 << 16) - (32 - r) * 8, p);
+  return p + 4;
+}
+
+static bfd_byte *
+restgpr1_tail (bfd *abfd, bfd_byte *p, int r)
+{
+  p = restgpr1 (abfd, p, r);
+  bfd_put_32 (abfd, BLR, p);
+  return p + 4;
+}
+
+static bfd_byte *
+savefpr (bfd *abfd, bfd_byte *p, int r)
+{
+  bfd_put_32 (abfd, STFD_FR0_0R1 + (r << 21) + (1 << 16) - (32 - r) * 8, p);
+  return p + 4;
+}
+
+static bfd_byte *
+savefpr0_tail (bfd *abfd, bfd_byte *p, int r)
+{
+  p = savefpr (abfd, p, r);
+  bfd_put_32 (abfd, STD_R0_0R1 + 16, p);
+  p = p + 4;
+  bfd_put_32 (abfd, BLR, p);
+  return p + 4;
+}
+
+static bfd_byte *
+restfpr (bfd *abfd, bfd_byte *p, int r)
+{
+  bfd_put_32 (abfd, LFD_FR0_0R1 + (r << 21) + (1 << 16) - (32 - r) * 8, p);
+  return p + 4;
+}
+
+static bfd_byte *
+restfpr0_tail (bfd *abfd, bfd_byte *p, int r)
+{
+  bfd_put_32 (abfd, LD_R0_0R1 + 16, p);
+  p = p + 4;
+  p = restfpr (abfd, p, r);
+  bfd_put_32 (abfd, MTLR_R0, p);
+  p = p + 4;
+  if (r == 29)
+    {
+      p = restfpr (abfd, p, 30);
+      p = restfpr (abfd, p, 31);
+    }
+  bfd_put_32 (abfd, BLR, p);
+  return p + 4;
+}
+
+static bfd_byte *
+savefpr1_tail (bfd *abfd, bfd_byte *p, int r)
+{
+  p = savefpr (abfd, p, r);
+  bfd_put_32 (abfd, BLR, p);
+  return p + 4;
+}
+
+static bfd_byte *
+restfpr1_tail (bfd *abfd, bfd_byte *p, int r)
+{
+  p = restfpr (abfd, p, r);
+  bfd_put_32 (abfd, BLR, p);
+  return p + 4;
+}
+
+static bfd_byte *
+savevr (bfd *abfd, bfd_byte *p, int r)
+{
+  bfd_put_32 (abfd, LI_R12_0 + (1 << 16) - (32 - r) * 16, p);
+  p = p + 4;
+  bfd_put_32 (abfd, STVX_VR0_R12_R0 + (r << 21), p);
+  return p + 4;
+}
+
+static bfd_byte *
+savevr_tail (bfd *abfd, bfd_byte *p, int r)
+{
+  p = savevr (abfd, p, r);
+  bfd_put_32 (abfd, BLR, p);
+  return p + 4;
+}
+
+static bfd_byte *
+restvr (bfd *abfd, bfd_byte *p, int r)
+{
+  bfd_put_32 (abfd, LI_R12_0 + (1 << 16) - (32 - r) * 16, p);
+  p = p + 4;
+  bfd_put_32 (abfd, LVX_VR0_R12_R0 + (r << 21), p);
+  return p + 4;
+}
+
+static bfd_byte *
+restvr_tail (bfd *abfd, bfd_byte *p, int r)
+{
+  p = restvr (abfd, p, r);
+  bfd_put_32 (abfd, BLR, p);
+  return p + 4;
+}
+
 /* Called via elf_link_hash_traverse to transfer dynamic linking
    information on function code symbol entries to their corresponding
    function descriptor symbol entries.  */
+
 static bfd_boolean
 func_desc_adjust (struct elf_link_hash_entry *h, void *inf)
 {
@@ -4307,15 +5289,29 @@ func_desc_adjust (struct elf_link_hash_entry *h, void *inf)
   info = inf;
   htab = ppc_hash_table (info);
 
+  /* Resolve undefined references to dot-symbols as the value
+     in the function descriptor, if we have one in a regular object.
+     This is to satisfy cases like ".quad .foo".  Calls to functions
+     in dynamic objects are handled elsewhere.  */
+  if (fh->elf.root.type == bfd_link_hash_undefweak
+      && fh->was_undefined
+      && (fh->oh->elf.root.type == bfd_link_hash_defined
+         || fh->oh->elf.root.type == bfd_link_hash_defweak)
+      && get_opd_info (fh->oh->elf.root.u.def.section) != NULL
+      && opd_entry_value (fh->oh->elf.root.u.def.section,
+                         fh->oh->elf.root.u.def.value,
+                         &fh->elf.root.u.def.section,
+                         &fh->elf.root.u.def.value) != (bfd_vma) -1)
+    {
+      fh->elf.root.type = fh->oh->elf.root.type;
+      fh->elf.elf_link_hash_flags |= ELF_LINK_FORCED_LOCAL;
+    }
+
   /* If this is a function code symbol, transfer dynamic linking
      information to the function descriptor symbol.  */
   if (!fh->is_func)
     return TRUE;
 
-  if (fh->elf.root.type == bfd_link_hash_undefweak
-      && (fh->elf.elf_link_hash_flags & ELF_LINK_HASH_REF_REGULAR))
-    htab->have_undefweak = TRUE;
-
   for (ent = fh->elf.plt.plist; ent != NULL; ent = ent->next)
     if (ent->plt.refcount > 0)
       break;
@@ -4373,7 +5369,7 @@ func_desc_adjust (struct elf_link_hash_entry *h, void *inf)
              && ELF_ST_VISIBILITY (fdh->elf.other) == STV_DEFAULT)))
     {
       if (fdh->elf.dynindx == -1)
-       if (! bfd_elf64_link_record_dynamic_symbol (info, &fdh->elf))
+       if (! bfd_elf_link_record_dynamic_symbol (info, &fdh->elf))
          return FALSE;
       fdh->elf.elf_link_hash_flags
        |= (fh->elf.elf_link_hash_flags & (ELF_LINK_HASH_REF_REGULAR
@@ -4382,12 +5378,16 @@ func_desc_adjust (struct elf_link_hash_entry *h, void *inf)
                                      | ELF_LINK_NON_GOT_REF));
       if (ELF_ST_VISIBILITY (fh->elf.other) == STV_DEFAULT)
        {
-         fdh->elf.plt.plist = fh->elf.plt.plist;
+         struct plt_entry **ep = &fdh->elf.plt.plist;
+         while (*ep != NULL)
+           ep = &(*ep)->next;
+         *ep = fh->elf.plt.plist;
+         fh->elf.plt.plist = NULL;
          fdh->elf.elf_link_hash_flags |= ELF_LINK_HASH_NEEDS_PLT;
        }
       fdh->is_func_descriptor = 1;
-      fdh->oh = &fh->elf;
-      fh->oh = &fdh->elf;
+      fdh->oh = fh;
+      fh->oh = fdh;
     }
 
   /* Now that the info is on the function descriptor, clear the
@@ -4408,119 +5408,48 @@ func_desc_adjust (struct elf_link_hash_entry *h, void *inf)
   return TRUE;
 }
 
-#define MIN_SAVE_FPR 14
-#define MAX_SAVE_FPR 31
-
 /* Called near the start of bfd_elf_size_dynamic_sections.  We use
    this hook to a) provide some gcc support functions, and b) transfer
    dynamic linking information gathered so far on function code symbol
    entries, to their corresponding function descriptor symbol entries.  */
+
 static bfd_boolean
 ppc64_elf_func_desc_adjust (bfd *obfd ATTRIBUTE_UNUSED,
                            struct bfd_link_info *info)
 {
   struct ppc_link_hash_table *htab;
-  unsigned int lowest_savef = MAX_SAVE_FPR + 2;
-  unsigned int lowest_restf = MAX_SAVE_FPR + 2;
   unsigned int i;
-  struct elf_link_hash_entry *h;
-  bfd_byte *p;
-  char sym[10];
+  const struct sfpr_def_parms funcs[] =
+    {
+      { "_savegpr0_", 14, 31, savegpr0, savegpr0_tail },
+      { "_restgpr0_", 14, 29, restgpr0, restgpr0_tail },
+      { "_restgpr0_", 30, 31, restgpr0, restgpr0_tail },
+      { "_savegpr1_", 14, 31, savegpr1, savegpr1_tail },
+      { "_restgpr1_", 14, 31, restgpr1, restgpr1_tail },
+      { "_savefpr_", 14, 31, savefpr, savefpr0_tail },
+      { "_restfpr_", 14, 29, restfpr, restfpr0_tail },
+      { "_restfpr_", 30, 31, restfpr, restfpr0_tail },
+      { "._savef", 14, 31, savefpr, savefpr1_tail },
+      { "._restf", 14, 31, restfpr, restfpr1_tail },
+      { "_savevr_", 20, 31, savevr, savevr_tail },
+      { "_restvr_", 20, 31, restvr, restvr_tail }
+    };
 
   htab = ppc_hash_table (info);
-
   if (htab->sfpr == NULL)
     /* We don't have any relocs.  */
     return TRUE;
 
-  /* First provide any missing ._savef* and ._restf* functions.  */
-  memcpy (sym, "._savef14", 10);
-  for (i = MIN_SAVE_FPR; i <= MAX_SAVE_FPR; i++)
-    {
-      sym[7] = i / 10 + '0';
-      sym[8] = i % 10 + '0';
-      h = elf_link_hash_lookup (&htab->elf, sym, FALSE, FALSE, TRUE);
-      if (h != NULL
-         && (h->elf_link_hash_flags & ELF_LINK_HASH_DEF_REGULAR) == 0)
-       {
-         if (lowest_savef > i)
-           lowest_savef = i;
-         h->root.type = bfd_link_hash_defined;
-         h->root.u.def.section = htab->sfpr;
-         h->root.u.def.value = (i - lowest_savef) * 4;
-         h->type = STT_FUNC;
-         h->elf_link_hash_flags |= ELF_LINK_HASH_DEF_REGULAR;
-         _bfd_elf_link_hash_hide_symbol (info, h, TRUE);
-       }
-    }
-
-  memcpy (sym, "._restf14", 10);
-  for (i = MIN_SAVE_FPR; i <= MAX_SAVE_FPR; i++)
-    {
-      sym[7] = i / 10 + '0';
-      sym[8] = i % 10 + '0';
-      h = elf_link_hash_lookup (&htab->elf, sym, FALSE, FALSE, TRUE);
-      if (h != NULL
-         && (h->elf_link_hash_flags & ELF_LINK_HASH_DEF_REGULAR) == 0)
-       {
-         if (lowest_restf > i)
-           lowest_restf = i;
-         h->root.type = bfd_link_hash_defined;
-         h->root.u.def.section = htab->sfpr;
-         h->root.u.def.value = ((MAX_SAVE_FPR + 2 - lowest_savef) * 4
-                                + (i - lowest_restf) * 4);
-         h->type = STT_FUNC;
-         h->elf_link_hash_flags |= ELF_LINK_HASH_DEF_REGULAR;
-         _bfd_elf_link_hash_hide_symbol (info, h, TRUE);
-       }
-    }
+  /* Provide any missing _save* and _rest* functions.  */
+  htab->sfpr->size = 0;
+  for (i = 0; i < sizeof (funcs) / sizeof (funcs[0]); i++)
+    if (!sfpr_define (info, &funcs[i]))
+      return FALSE;
 
   elf_link_hash_traverse (&htab->elf, func_desc_adjust, info);
 
-  htab->sfpr->_raw_size = ((MAX_SAVE_FPR + 2 - lowest_savef) * 4
-                          + (MAX_SAVE_FPR + 2 - lowest_restf) * 4);
-
-  if (htab->sfpr->_raw_size == 0)
-    {
-      if (!htab->have_undefweak)
-       {
-         _bfd_strip_section_from_output (info, htab->sfpr);
-         return TRUE;
-       }
-
-      htab->sfpr->_raw_size = 4;
-    }
-
-  p = bfd_alloc (htab->elf.dynobj, htab->sfpr->_raw_size);
-  if (p == NULL)
-    return FALSE;
-  htab->sfpr->contents = p;
-
-  for (i = lowest_savef; i <= MAX_SAVE_FPR; i++)
-    {
-      unsigned int fpr = i << 21;
-      unsigned int stackoff = (1 << 16) - (MAX_SAVE_FPR + 1 - i) * 8;
-      bfd_put_32 (htab->elf.dynobj, STFD_FR0_0R1 + fpr + stackoff, p);
-      p += 4;
-    }
-  if (lowest_savef <= MAX_SAVE_FPR)
-    {
-      bfd_put_32 (htab->elf.dynobj, BLR, p);
-      p += 4;
-    }
-
-  for (i = lowest_restf; i <= MAX_SAVE_FPR; i++)
-    {
-      unsigned int fpr = i << 21;
-      unsigned int stackoff = (1 << 16) - (MAX_SAVE_FPR + 1 - i) * 8;
-      bfd_put_32 (htab->elf.dynobj, LFD_FR0_0R1 + fpr + stackoff, p);
-      p += 4;
-    }
-  if (lowest_restf <= MAX_SAVE_FPR
-      || htab->sfpr->_raw_size == 4)
-    {
-      bfd_put_32 (htab->elf.dynobj, BLR, p);
-    }
+  if (htab->sfpr->size == 0)
+    _bfd_strip_section_from_output (info, htab->sfpr);
 
   return TRUE;
 }
@@ -4551,8 +5480,7 @@ ppc64_elf_adjust_dynamic_symbol (struct bfd_link_info *info,
       for (ent = h->plt.plist; ent != NULL; ent = ent->next)
        if (ent->plt.refcount > 0)
          break;
-      if (!((struct ppc_link_hash_entry *) h)->is_func_descriptor
-         || ent == NULL
+      if (ent == NULL
          || SYMBOL_CALLS_LOCAL (info, h)
          || (ELF_ST_VISIBILITY (h->other) != STV_DEFAULT
              && h->root.type == bfd_link_hash_undefweak))
@@ -4646,7 +5574,7 @@ ppc64_elf_adjust_dynamic_symbol (struct bfd_link_info *info,
      .rela.bss section we are going to use.  */
   if ((h->root.u.def.section->flags & SEC_ALLOC) != 0)
     {
-      htab->relbss->_raw_size += sizeof (Elf64_External_Rela);
+      htab->relbss->size += sizeof (Elf64_External_Rela);
       h->elf_link_hash_flags |= ELF_LINK_HASH_NEEDS_COPY;
     }
 
@@ -4658,7 +5586,7 @@ ppc64_elf_adjust_dynamic_symbol (struct bfd_link_info *info,
 
   /* Apply the required alignment.  */
   s = htab->dynbss;
-  s->_raw_size = BFD_ALIGN (s->_raw_size, (bfd_size_type) (1 << power_of_two));
+  s->size = BFD_ALIGN (s->size, (bfd_size_type) (1 << power_of_two));
   if (power_of_two > bfd_get_section_alignment (htab->elf.dynobj, s))
     {
       if (! bfd_set_section_alignment (htab->elf.dynobj, s, power_of_two))
@@ -4667,10 +5595,10 @@ ppc64_elf_adjust_dynamic_symbol (struct bfd_link_info *info,
 
   /* Define the symbol as being at this point in the section.  */
   h->root.u.def.section = s;
-  h->root.u.def.value = s->_raw_size;
+  h->root.u.def.value = s->size;
 
   /* Increment the section size to make room for the symbol.  */
-  s->_raw_size += h->size;
+  s->size += h->size;
 
   return TRUE;
 }
@@ -4682,11 +5610,13 @@ ppc64_elf_hide_symbol (struct bfd_link_info *info,
                       struct elf_link_hash_entry *h,
                       bfd_boolean force_local)
 {
+  struct ppc_link_hash_entry *eh;
   _bfd_elf_link_hash_hide_symbol (info, h, force_local);
 
-  if (((struct ppc_link_hash_entry *) h)->is_func_descriptor)
+  eh = (struct ppc_link_hash_entry *) h;
+  if (eh->is_func_descriptor)
     {
-      struct elf_link_hash_entry *fh = ((struct ppc_link_hash_entry *) h)->oh;
+      struct ppc_link_hash_entry *fh = eh->oh;
 
       if (fh == NULL)
        {
@@ -4700,14 +5630,15 @@ ppc64_elf_hide_symbol (struct bfd_link_info *info,
             when it runs out of memory.  This function doesn't have a
             return status, so there's no way to gracefully return an
             error.  So cheat.  We know that string[-1] can be safely
-            dereferenced;  It's either a string in an ELF string
-            table, or allocated in an objalloc structure.  */
+            accessed;  It's either a string in an ELF string table,
+            or allocated in an objalloc structure.  */
 
-         p = h->root.root.string - 1;
+         p = eh->elf.root.root.string - 1;
          save = *p;
          *(char *) p = '.';
          htab = ppc_hash_table (info);
-         fh = elf_link_hash_lookup (&htab->elf, p, FALSE, FALSE, FALSE);
+         fh = (struct ppc_link_hash_entry *)
+           elf_link_hash_lookup (&htab->elf, p, FALSE, FALSE, FALSE);
          *(char *) p = save;
 
          /* Unfortunately, if it so happens that the string we were
@@ -4716,27 +5647,32 @@ ppc64_elf_hide_symbol (struct bfd_link_info *info,
             reason the lookup should fail.  */
          if (fh == NULL)
            {
-             q = h->root.root.string + strlen (h->root.root.string);
-             while (q >= h->root.root.string && *q == *p)
+             q = eh->elf.root.root.string + strlen (eh->elf.root.root.string);
+             while (q >= eh->elf.root.root.string && *q == *p)
                --q, --p;
-             if (q < h->root.root.string && *p == '.')
-               fh = elf_link_hash_lookup (&htab->elf, p, FALSE, FALSE, FALSE);
+             if (q < eh->elf.root.root.string && *p == '.')
+               fh = (struct ppc_link_hash_entry *)
+                 elf_link_hash_lookup (&htab->elf, p, FALSE, FALSE, FALSE);
            }
          if (fh != NULL)
            {
-             ((struct ppc_link_hash_entry *) h)->oh = fh;
-             ((struct ppc_link_hash_entry *) fh)->oh = h;
+             eh->oh = fh;
+             fh->oh = eh;
            }
        }
       if (fh != NULL)
-       _bfd_elf_link_hash_hide_symbol (info, fh, force_local);
+       _bfd_elf_link_hash_hide_symbol (info, &fh->elf, force_local);
     }
 }
 
 static bfd_boolean
-get_sym_h (struct elf_link_hash_entry **hp, Elf_Internal_Sym **symp,
-          asection **symsecp, char **tls_maskp, Elf_Internal_Sym **locsymsp,
-          unsigned long r_symndx, bfd *ibfd)
+get_sym_h (struct elf_link_hash_entry **hp,
+          Elf_Internal_Sym **symp,
+          asection **symsecp,
+          char **tls_maskp,
+          Elf_Internal_Sym **locsymsp,
+          unsigned long r_symndx,
+          bfd *ibfd)
 {
   Elf_Internal_Shdr *symtab_hdr = &elf_tdata (ibfd)->symtab_hdr;
 
@@ -4875,10 +5811,71 @@ get_tls_mask (char **tls_maskp, unsigned long *toc_symndx,
   return 1;
 }
 
+/* Adjust all global syms defined in opd sections.  In gcc generated
+   code for the old ABI, these will already have been done.  */
+
+static bfd_boolean
+adjust_opd_syms (struct elf_link_hash_entry *h, void *inf ATTRIBUTE_UNUSED)
+{
+  struct ppc_link_hash_entry *eh;
+  asection *sym_sec;
+  long *opd_adjust;
+
+  if (h->root.type == bfd_link_hash_indirect)
+    return TRUE;
+
+  if (h->root.type == bfd_link_hash_warning)
+    h = (struct elf_link_hash_entry *) h->root.u.i.link;
+
+  if (h->root.type != bfd_link_hash_defined
+      && h->root.type != bfd_link_hash_defweak)
+    return TRUE;
+
+  eh = (struct ppc_link_hash_entry *) h;
+  if (eh->adjust_done)
+    return TRUE;
+
+  sym_sec = eh->elf.root.u.def.section;
+  opd_adjust = get_opd_info (sym_sec);
+  if (opd_adjust != NULL)
+    {
+      long adjust = opd_adjust[eh->elf.root.u.def.value / 8];
+      if (adjust == -1)
+       {
+         /* This entry has been deleted.  */
+         asection *dsec = ppc64_elf_tdata (sym_sec->owner)->deleted_section;
+         if (dsec == NULL)
+           {
+             for (dsec = sym_sec->owner->sections; dsec; dsec = dsec->next)
+               if (elf_discarded_section (dsec))
+                 {
+                   ppc64_elf_tdata (sym_sec->owner)->deleted_section = dsec;
+                   break;
+                 }
+           }
+         eh->elf.root.u.def.value = 0;
+         eh->elf.root.u.def.section = dsec;
+       }
+      else
+       eh->elf.root.u.def.value += adjust;
+      eh->adjust_done = 1;
+    }
+  return TRUE;
+}
+
+/* Remove unused Official Procedure Descriptor entries.  Currently we
+   only remove those associated with functions in discarded link-once
+   sections, or weakly defined functions that have been overridden.  It
+   would be possible to remove many more entries for statically linked
+   applications.  */
+
 bfd_boolean
-ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
+ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info,
+                   bfd_boolean non_overlapping)
 {
   bfd *ibfd;
+  bfd_boolean some_edited = FALSE;
+  asection *need_pad = NULL;
 
   for (ibfd = info->input_bfds; ibfd != NULL; ibfd = ibfd->link_next)
     {
@@ -4889,23 +5886,24 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
       struct elf_link_hash_entry **sym_hashes;
       bfd_vma offset;
       bfd_size_type amt;
-      long *adjust;
-      bfd_boolean need_edit;
+      long *opd_adjust;
+      bfd_boolean need_edit, add_aux_fields;
+      bfd_size_type cnt_16b = 0;
 
       sec = bfd_get_section_by_name (ibfd, ".opd");
       if (sec == NULL)
        continue;
 
-      amt = sec->_raw_size * sizeof (long) / 24;
-      adjust = ppc64_elf_section_data (sec)->opd.adjust;
-      if (adjust == NULL)
+      amt = sec->size * sizeof (long) / 8;
+      opd_adjust = get_opd_info (sec);
+      if (opd_adjust == NULL)
        {
          /* Must be a ld -r link.  ie. check_relocs hasn't been
             called.  */
-         adjust = bfd_zalloc (obfd, amt);
-         ppc64_elf_section_data (sec)->opd.adjust = adjust;
+         opd_adjust = bfd_zalloc (obfd, amt);
+         ppc64_elf_section_data (sec)->opd.adjust = opd_adjust;
        }
-      memset (adjust, 0, amt);
+      memset (opd_adjust, 0, amt);
 
       if (sec->output_section == bfd_abs_section_ptr)
        continue;
@@ -4927,6 +5925,7 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
       /* First run through the relocs to check they are sane, and to
         determine whether we need to edit this opd section.  */
       need_edit = FALSE;
+      need_pad = sec;
       offset = 0;
       relend = relstart + sec->reloc_count;
       for (rel = relstart; rel < relend; )
@@ -4937,7 +5936,7 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
          struct elf_link_hash_entry *h;
          Elf_Internal_Sym *sym;
 
-         /* .opd contains a regular array of 24 byte entries.  We're
+         /* .opd contains a regular array of 16 or 24 byte entries.  We're
             only interested in the reloc pointing to a function entry
             point.  */
          if (rel->r_offset != offset
@@ -4949,9 +5948,9 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
                 Also, there's nothing to prevent someone putting
                 something silly in .opd with the assembler.  No .opd
                 optimization for them!  */
+           broken_opd:
              (*_bfd_error_handler)
-               (_("%s: .opd is not a regular array of opd entries"),
-                bfd_archive_filename (ibfd));
+               (_("%B: .opd is not a regular array of opd entries"), ibfd);
              need_edit = FALSE;
              break;
            }
@@ -4960,8 +5959,8 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
              || (r_type = ELF64_R_TYPE ((rel + 1)->r_info)) != R_PPC64_TOC)
            {
              (*_bfd_error_handler)
-               (_("%s: unexpected reloc type %u in .opd section"),
-                bfd_archive_filename (ibfd), r_type);
+               (_("%B: unexpected reloc type %u in .opd section"),
+                ibfd, r_type);
              need_edit = FALSE;
              break;
            }
@@ -4980,9 +5979,8 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
                sym_name = bfd_elf_local_sym_name (ibfd, sym);
 
              (*_bfd_error_handler)
-               (_("%s: undefined sym `%s' in .opd section"),
-                bfd_archive_filename (ibfd),
-                sym_name);
+               (_("%B: undefined sym `%s' in .opd section"),
+                ibfd, sym_name);
              need_edit = FALSE;
              break;
            }
@@ -4998,19 +5996,54 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
              || sym_sec->output_section == bfd_abs_section_ptr)
            need_edit = TRUE;
 
-         offset += 24;
          rel += 2;
-         /* Allow for the possibility of a reloc on the third word.  */
-         if (rel < relend
-             && rel->r_offset == offset - 8)
-           rel += 1;
+         if (rel == relend
+             || (rel + 1 == relend && rel->r_offset == offset + 16))
+           {
+             if (sec->size == offset + 24)
+               {
+                 need_pad = NULL;
+                 break;
+               }
+             if (rel == relend && sec->size == offset + 16)
+               {
+                 cnt_16b++;
+                 break;
+               }
+             goto broken_opd;
+           }
+
+         if (rel->r_offset == offset + 24)
+           offset += 24;
+         else if (rel->r_offset != offset + 16)
+           goto broken_opd;
+         else if (rel + 1 < relend
+                  && ELF64_R_TYPE (rel[0].r_info) == R_PPC64_ADDR64
+                  && ELF64_R_TYPE (rel[1].r_info) == R_PPC64_TOC)
+           {
+             offset += 16;
+             cnt_16b++;
+           }
+         else if (rel + 2 < relend
+                  && ELF64_R_TYPE (rel[1].r_info) == R_PPC64_ADDR64
+                  && ELF64_R_TYPE (rel[2].r_info) == R_PPC64_TOC)
+           {
+             offset += 24;
+             rel += 1;
+           }
+         else
+           goto broken_opd;
        }
 
-      if (need_edit)
+      add_aux_fields = non_overlapping && cnt_16b > 0;
+
+      if (need_edit || add_aux_fields)
        {
          Elf_Internal_Rela *write_rel;
          bfd_byte *rptr, *wptr;
+         bfd_byte *new_contents = NULL;
          bfd_boolean skip;
+         long opd_ent_size;
 
          /* This seems a waste of time as input .opd sections are all
             zeros as generated by gcc, but I suppose there's no reason
@@ -5018,11 +6051,11 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
             the third word of .opd entries.  */
          if ((sec->flags & SEC_IN_MEMORY) == 0)
            {
-             bfd_byte *loc = bfd_alloc (ibfd, sec->_raw_size);
-             if (loc == NULL
-                 || !bfd_get_section_contents (ibfd, sec, loc, 0,
-                                               sec->_raw_size))
+             bfd_byte *loc;
+             if (!bfd_malloc_and_get_section (ibfd, sec, &loc))
                {
+                 if (loc != NULL)
+                   free (loc);
                error_ret:
                  if (local_syms != NULL
                      && symtab_hdr->contents != (unsigned char *) local_syms)
@@ -5039,9 +6072,21 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
 
          wptr = sec->contents;
          rptr = sec->contents;
+         new_contents = sec->contents;
+
+         if (add_aux_fields)
+           {
+             new_contents = bfd_malloc (sec->size + cnt_16b * 8);
+             if (new_contents == NULL)
+               return FALSE;
+             need_pad = FALSE;
+             wptr = new_contents;
+           }
+
          write_rel = relstart;
          skip = FALSE;
          offset = 0;
+         opd_ent_size = 0;
          for (rel = relstart; rel < relend; rel++)
            {
              unsigned long r_symndx;
@@ -5057,7 +6102,21 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
              if (rel->r_offset == offset)
                {
                  struct ppc_link_hash_entry *fdh = NULL;
-                 if (h != NULL)
+
+                 /* See if the .opd entry is full 24 byte or
+                    16 byte (with fd_aux entry overlapped with next
+                    fd_func).  */
+                 opd_ent_size = 24;
+                 if ((rel + 2 == relend && sec->size == offset + 16)
+                     || (rel + 3 < relend
+                         && rel[2].r_offset == offset + 16
+                         && rel[3].r_offset == offset + 24
+                         && ELF64_R_TYPE (rel[2].r_info) == R_PPC64_ADDR64
+                         && ELF64_R_TYPE (rel[3].r_info) == R_PPC64_TOC))
+                   opd_ent_size = 16;
+
+                 if (h != NULL
+                     && h->root.root.string[0] == '.')
                    fdh = get_fdh ((struct ppc_link_hash_entry *) h,
                                   ppc_hash_table (info));
 
@@ -5065,43 +6124,52 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
                          || sym_sec->output_section == bfd_abs_section_ptr);
                  if (skip)
                    {
-                     if (h != NULL && sym_sec->owner == ibfd)
+                     if (fdh != NULL && sym_sec->owner == ibfd)
                        {
                          /* Arrange for the function descriptor sym
                             to be dropped.  */
                          fdh->elf.root.u.def.value = 0;
                          fdh->elf.root.u.def.section = sym_sec;
                        }
+                     opd_adjust[rel->r_offset / 8] = -1;
                    }
                  else
                    {
                      /* We'll be keeping this opd entry.  */
 
-                     if (h != NULL)
-                       {
-                         /* Redefine the function descriptor symbol
-                            to this location in the opd section.
-                            We've checked above that opd relocs are
-                            ordered.  */
-                         fdh->elf.root.u.def.value = wptr - sec->contents;
-                       }
-                     else
+                     if (fdh != NULL)
                        {
-                         /* Local syms are a bit tricky.  We could
-                            tweak them as they can be cached, but
-                            we'd need to look through the local syms
-                            for the function descriptor sym which we
-                            don't have at the moment.  So keep an
-                            array of adjustments.  */
-                         adjust[rel->r_offset / 24] = wptr - rptr;
+                         /* Redefine the function descriptor symbol to
+                            this location in the opd section.  It is
+                            necessary to update the value here rather
+                            than using an array of adjustments as we do
+                            for local symbols, because various places
+                            in the generic ELF code use the value
+                            stored in u.def.value.  */
+                         fdh->elf.root.u.def.value = wptr - new_contents;
+                         fdh->adjust_done = 1;
                        }
 
+                     /* Local syms are a bit tricky.  We could
+                        tweak them as they can be cached, but
+                        we'd need to look through the local syms
+                        for the function descriptor sym which we
+                        don't have at the moment.  So keep an
+                        array of adjustments.  */
+                     opd_adjust[rel->r_offset / 8]
+                       = (wptr - new_contents) - (rptr - sec->contents);
+
                      if (wptr != rptr)
-                       memcpy (wptr, rptr, 24);
-                     wptr += 24;
+                       memcpy (wptr, rptr, opd_ent_size);
+                     wptr += opd_ent_size;
+                     if (add_aux_fields && opd_ent_size == 16)
+                       {
+                         memset (wptr, '\0', 8);
+                         wptr += 8;
+                       }
                    }
-                 rptr += 24;
-                 offset += 24;
+                 rptr += opd_ent_size;
+                 offset += opd_ent_size;
                }
 
              if (skip)
@@ -5139,20 +6207,27 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
                  /* We need to adjust any reloc offsets to point to the
                     new opd entries.  While we're at it, we may as well
                     remove redundant relocs.  */
-                 rel->r_offset += wptr - rptr;
+                 rel->r_offset += opd_adjust[(offset - opd_ent_size) / 8];
                  if (write_rel != rel)
                    memcpy (write_rel, rel, sizeof (*rel));
                  ++write_rel;
                }
            }
 
-         sec->_cooked_size = wptr - sec->contents;
+         sec->size = wptr - new_contents;
          sec->reloc_count = write_rel - relstart;
+         if (add_aux_fields)
+           {
+             free (sec->contents);
+             sec->contents = new_contents;
+           }
+
          /* Fudge the size too, as this is used later in
             elf_bfd_final_link if we are emitting relocs.  */
          elf_section_data (sec)->rel_hdr.sh_size
            = sec->reloc_count * elf_section_data (sec)->rel_hdr.sh_entsize;
          BFD_ASSERT (elf_section_data (sec)->rel_hdr2 == NULL);
+         some_edited = TRUE;
        }
       else if (elf_section_data (sec)->relocs != relstart)
        free (relstart);
@@ -5167,6 +6242,43 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
        }
     }
 
+  if (some_edited)
+    elf_link_hash_traverse (elf_hash_table (info), adjust_opd_syms, NULL);
+
+  /* If we are doing a final link and the last .opd entry is just 16 byte
+     long, add a 8 byte padding after it.  */
+  if (need_pad != NULL && !info->relocatable)
+    {
+      bfd_byte *p;
+
+      if ((need_pad->flags & SEC_IN_MEMORY) == 0)
+       {
+         BFD_ASSERT (need_pad->size > 0);
+
+         p = bfd_malloc (need_pad->size + 8);
+         if (p == NULL)
+           return FALSE;
+
+         if (! bfd_get_section_contents (need_pad->owner, need_pad,
+                                         p, 0, need_pad->size))
+           return FALSE;
+
+         need_pad->contents = p;
+         need_pad->flags |= (SEC_IN_MEMORY | SEC_HAS_CONTENTS);
+       }
+      else
+       {
+         p = bfd_realloc (need_pad->contents, need_pad->size + 8);
+         if (p == NULL)
+           return FALSE;
+
+         need_pad->contents = p;
+       }
+
+      memset (need_pad->contents + need_pad->size, 0, 8);
+      need_pad->size += 8;
+    }
+
   return TRUE;
 }
 
@@ -5180,13 +6292,29 @@ ppc64_elf_tls_setup (bfd *obfd, struct bfd_link_info *info)
   htab = ppc_hash_table (info);
   if (htab->tls_get_addr != NULL)
     {
-      struct elf_link_hash_entry *h = htab->tls_get_addr;
+      struct ppc_link_hash_entry *h = htab->tls_get_addr;
 
-      while (h->root.type == bfd_link_hash_indirect
-            || h->root.type == bfd_link_hash_warning)
-       h = (struct elf_link_hash_entry *) h->root.u.i.link;
+      while (h->elf.root.type == bfd_link_hash_indirect
+            || h->elf.root.type == bfd_link_hash_warning)
+       h = (struct ppc_link_hash_entry *) h->elf.root.u.i.link;
 
       htab->tls_get_addr = h;
+
+      if (htab->tls_get_addr_fd == NULL
+         && h->oh != NULL
+         && h->oh->is_func_descriptor)
+       htab->tls_get_addr_fd = h->oh;
+    }
+
+  if (htab->tls_get_addr_fd != NULL)
+    {
+      struct ppc_link_hash_entry *h = htab->tls_get_addr_fd;
+
+      while (h->elf.root.type == bfd_link_hash_indirect
+            || h->elf.root.type == bfd_link_hash_warning)
+       h = (struct ppc_link_hash_entry *) h->elf.root.u.i.link;
+
+      htab->tls_get_addr_fd = h;
     }
 
   return _bfd_elf_tls_setup (obfd, info);
@@ -5262,6 +6390,8 @@ ppc64_elf_tls_optimize (bfd *obfd ATTRIBUTE_UNUSED, struct bfd_link_info *info)
                    value = h->root.u.def.value;
                  }
                else
+                 /* Symbols referenced by TLS relocs must be of type
+                    STT_TLS.  So no need for .opd local sym adjust.  */
                  value = sym->st_value;
 
                ok_tprel = FALSE;
@@ -5334,7 +6464,8 @@ ppc64_elf_tls_optimize (bfd *obfd ATTRIBUTE_UNUSED, struct bfd_link_info *info)
                  case R_PPC64_REL14_BRNTAKEN:
                  case R_PPC64_REL24:
                    if (h != NULL
-                       && h == htab->tls_get_addr)
+                       && (h == &htab->tls_get_addr->elf
+                           || h == &htab->tls_get_addr_fd->elf))
                      {
                        if (!expecting_tls_get_addr
                            && rel != relstart
@@ -5484,17 +6615,6 @@ ppc64_elf_tls_optimize (bfd *obfd ATTRIBUTE_UNUSED, struct bfd_link_info *info)
   return TRUE;
 }
 
-/* This is the condition under which ppc64_elf_finish_dynamic_symbol
-   will be called from elflink.h.  If elflink.h doesn't call our
-   finish_dynamic_symbol routine, we'll need to do something about
-   initializing any .plt and .got entries in ppc64_elf_relocate_section.  */
-#define WILL_CALL_FINISH_DYNAMIC_SYMBOL(DYN, SHARED, H) \
-  ((DYN)                                                               \
-   && ((SHARED)                                                                \
-       || ((H)->elf_link_hash_flags & ELF_LINK_FORCED_LOCAL) == 0)     \
-   && ((H)->dynindx != -1                                              \
-       || ((H)->elf_link_hash_flags & ELF_LINK_FORCED_LOCAL) != 0))
-
 /* Allocate space in .plt, .got and associated reloc sections for
    dynamic relocs.  */
 
@@ -5526,31 +6646,29 @@ allocate_dynrelocs (struct elf_link_hash_entry *h, void *inf)
       for (pent = h->plt.plist; pent != NULL; pent = pent->next)
        if (pent->plt.refcount > 0)
          {
-           BFD_ASSERT (((struct ppc_link_hash_entry *) h)->is_func_descriptor);
-
            /* If this is the first .plt entry, make room for the special
               first entry.  */
            s = htab->plt;
-           if (s->_raw_size == 0)
-             s->_raw_size += PLT_INITIAL_ENTRY_SIZE;
+           if (s->size == 0)
+             s->size += PLT_INITIAL_ENTRY_SIZE;
 
-           pent->plt.offset = s->_raw_size;
+           pent->plt.offset = s->size;
 
            /* Make room for this entry.  */
-           s->_raw_size += PLT_ENTRY_SIZE;
+           s->size += PLT_ENTRY_SIZE;
 
            /* Make room for the .glink code.  */
            s = htab->glink;
-           if (s->_raw_size == 0)
-             s->_raw_size += GLINK_CALL_STUB_SIZE;
+           if (s->size == 0)
+             s->size += GLINK_CALL_STUB_SIZE;
            /* We need bigger stubs past index 32767.  */
-           if (s->_raw_size >= GLINK_CALL_STUB_SIZE + 32768*2*4)
-             s->_raw_size += 4;
-           s->_raw_size += 2*4;
+           if (s->size >= GLINK_CALL_STUB_SIZE + 32768*2*4)
+             s->size += 4;
+           s->size += 2*4;
 
            /* We also need to make an entry in the .rela.plt section.  */
            s = htab->relplt;
-           s->_raw_size += sizeof (Elf64_External_Rela);
+           s->size += sizeof (Elf64_External_Rela);
            doneone = TRUE;
          }
        else
@@ -5604,7 +6722,7 @@ allocate_dynrelocs (struct elf_link_hash_entry *h, void *inf)
        if (h->dynindx == -1
            && (h->elf_link_hash_flags & ELF_LINK_FORCED_LOCAL) == 0)
          {
-           if (! bfd_elf64_link_record_dynamic_symbol (info, h))
+           if (! bfd_elf_link_record_dynamic_symbol (info, h))
              return FALSE;
          }
 
@@ -5616,15 +6734,15 @@ allocate_dynrelocs (struct elf_link_hash_entry *h, void *inf)
          }
 
        s = ppc64_elf_tdata (gent->owner)->got;
-       gent->got.offset = s->_raw_size;
-       s->_raw_size
+       gent->got.offset = s->size;
+       s->size
          += (gent->tls_type & eh->tls_mask & (TLS_GD | TLS_LD)) ? 16 : 8;
        dyn = htab->elf.dynamic_sections_created;
        if ((info->shared
             || WILL_CALL_FINISH_DYNAMIC_SYMBOL (dyn, 0, h))
            && (ELF_ST_VISIBILITY (h->other) == STV_DEFAULT
                || h->root.type != bfd_link_hash_undefweak))
-         ppc64_elf_tdata (gent->owner)->relgot->_raw_size
+         ppc64_elf_tdata (gent->owner)->relgot->size
            += (gent->tls_type & eh->tls_mask & TLS_GD
                ? 2 * sizeof (Elf64_External_Rela)
                : sizeof (Elf64_External_Rela));
@@ -5685,7 +6803,7 @@ allocate_dynrelocs (struct elf_link_hash_entry *h, void *inf)
          if (h->dynindx == -1
              && (h->elf_link_hash_flags & ELF_LINK_FORCED_LOCAL) == 0)
            {
-             if (! bfd_elf64_link_record_dynamic_symbol (info, h))
+             if (! bfd_elf_link_record_dynamic_symbol (info, h))
                return FALSE;
            }
 
@@ -5704,7 +6822,7 @@ allocate_dynrelocs (struct elf_link_hash_entry *h, void *inf)
   for (p = eh->dyn_relocs; p != NULL; p = p->next)
     {
       asection *sreloc = elf_section_data (p->sec)->sreloc;
-      sreloc->_raw_size += p->count * sizeof (Elf64_External_Rela);
+      sreloc->size += p->count * sizeof (Elf64_External_Rela);
     }
 
   return TRUE;
@@ -5764,7 +6882,7 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
          s = bfd_get_section_by_name (dynobj, ".interp");
          if (s == NULL)
            abort ();
-         s->_raw_size = sizeof ELF_DYNAMIC_INTERPRETER;
+         s->size = sizeof ELF_DYNAMIC_INTERPRETER;
          s->contents = (unsigned char *) ELF_DYNAMIC_INTERPRETER;
        }
     }
@@ -5780,18 +6898,18 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
       Elf_Internal_Shdr *symtab_hdr;
       asection *srel;
 
-      if (bfd_get_flavour (ibfd) != bfd_target_elf_flavour)
+      if (!is_ppc64_target (ibfd->xvec))
        continue;
 
       if (ppc64_tlsld_got (ibfd)->refcount > 0)
        {
          s = ppc64_elf_tdata (ibfd)->got;
-         ppc64_tlsld_got (ibfd)->offset = s->_raw_size;
-         s->_raw_size += 16;
+         ppc64_tlsld_got (ibfd)->offset = s->size;
+         s->size += 16;
          if (info->shared)
            {
              srel = ppc64_elf_tdata (ibfd)->relgot;
-             srel->_raw_size += sizeof (Elf64_External_Rela);
+             srel->size += sizeof (Elf64_External_Rela);
            }
        }
       else
@@ -5817,7 +6935,7 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
              else if (p->count != 0)
                {
                  srel = elf_section_data (p->sec)->sreloc;
-                 srel->_raw_size += p->count * sizeof (Elf64_External_Rela);
+                 srel->size += p->count * sizeof (Elf64_External_Rela);
                  if ((p->sec->output_section->flags & SEC_READONLY) != 0)
                    info->flags |= DF_TEXTREL;
                }
@@ -5845,27 +6963,27 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
                  {
                    if (ppc64_tlsld_got (ibfd)->offset == (bfd_vma) -1)
                      {
-                       ppc64_tlsld_got (ibfd)->offset = s->_raw_size;
-                       s->_raw_size += 16;
+                       ppc64_tlsld_got (ibfd)->offset = s->size;
+                       s->size += 16;
                        if (info->shared)
-                         srel->_raw_size += sizeof (Elf64_External_Rela);
+                         srel->size += sizeof (Elf64_External_Rela);
                      }
                    ent->got.offset = ppc64_tlsld_got (ibfd)->offset;
                  }
                else
                  {
-                   ent->got.offset = s->_raw_size;
+                   ent->got.offset = s->size;
                    if ((ent->tls_type & *lgot_masks & TLS_GD) != 0)
                      {
-                       s->_raw_size += 16;
+                       s->size += 16;
                        if (info->shared)
-                         srel->_raw_size += 2 * sizeof (Elf64_External_Rela);
+                         srel->size += 2 * sizeof (Elf64_External_Rela);
                      }
                    else
                      {
-                       s->_raw_size += 8;
+                       s->size += 8;
                        if (info->shared)
-                         srel->_raw_size += sizeof (Elf64_External_Rela);
+                         srel->size += sizeof (Elf64_External_Rela);
                      }
                  }
              }
@@ -5886,10 +7004,6 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
       if ((s->flags & SEC_LINKER_CREATED) == 0)
        continue;
 
-      /* Reset _cooked_size since prelim layout will set it wrongly,
-        and a non-zero _cooked_size sticks.  */
-      s->_cooked_size = 0;
-
       if (s == htab->brlt || s == htab->relbrlt)
        /* These haven't been allocated yet;  don't strip.  */
        continue;
@@ -5902,7 +7016,7 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
        }
       else if (strncmp (bfd_get_section_name (dynobj, s), ".rela", 5) == 0)
        {
-         if (s->_raw_size == 0)
+         if (s->size == 0)
            {
              /* If we don't need this section, strip it from the
                 output file.  This is mostly to handle .rela.bss and
@@ -5930,14 +7044,14 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
          continue;
        }
 
-      if (s->_raw_size == 0)
+      if (s->size == 0)
        {
          _bfd_strip_section_from_output (info, s);
          continue;
        }
 
       /* .plt is in the bss section.  We don't initialise it.  */
-      if ((s->flags & SEC_LOAD) == 0)
+      if (s == htab->plt)
        continue;
 
       /* Allocate memory for the section contents.  We use bfd_zalloc
@@ -5947,22 +7061,24 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
         sections instead of garbage.
         We also rely on the section contents being zero when writing
         the GOT.  */
-      s->contents = bfd_zalloc (dynobj, s->_raw_size);
+      s->contents = bfd_zalloc (dynobj, s->size);
       if (s->contents == NULL)
        return FALSE;
     }
 
   for (ibfd = info->input_bfds; ibfd != NULL; ibfd = ibfd->link_next)
     {
+      if (!is_ppc64_target (ibfd->xvec))
+       continue;
+
       s = ppc64_elf_tdata (ibfd)->got;
       if (s != NULL && s != htab->got)
        {
-         s->_cooked_size = 0;
-         if (s->_raw_size == 0)
+         if (s->size == 0)
            _bfd_strip_section_from_output (info, s);
          else
            {
-             s->contents = bfd_zalloc (ibfd, s->_raw_size);
+             s->contents = bfd_zalloc (ibfd, s->size);
              if (s->contents == NULL)
                return FALSE;
            }
@@ -5970,12 +7086,11 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
       s = ppc64_elf_tdata (ibfd)->relgot;
       if (s != NULL)
        {
-         s->_cooked_size = 0;
-         if (s->_raw_size == 0)
+         if (s->size == 0)
            _bfd_strip_section_from_output (info, s);
          else
            {
-             s->contents = bfd_zalloc (ibfd, s->_raw_size);
+             s->contents = bfd_zalloc (ibfd, s->size);
              if (s->contents == NULL)
                return FALSE;
              relocs = TRUE;
@@ -5992,7 +7107,7 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
         the .dynamic section.  The DT_DEBUG entry is filled in by the
         dynamic linker and used by the debugger.  */
 #define add_dynamic_entry(TAG, VAL) \
-  bfd_elf64_add_dynamic_entry (info, (TAG), (VAL))
+  _bfd_elf_add_dynamic_entry (info, TAG, VAL)
 
       if (info->executable)
        {
@@ -6000,7 +7115,7 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
            return FALSE;
        }
 
-      if (htab->plt != NULL && htab->plt->_raw_size != 0)
+      if (htab->plt != NULL && htab->plt->size != 0)
        {
          if (!add_dynamic_entry (DT_PLTGOT, 0)
              || !add_dynamic_entry (DT_PLTRELSZ, 0)
@@ -6058,20 +7173,25 @@ ppc_type_of_stub (asection *input_sec,
   if (h != NULL)
     {
       if (h->oh != NULL
-         && h->oh->dynindx != -1)
+         && h->oh->is_func_descriptor)
+       h = h->oh;
+
+      if (h->elf.dynindx != -1)
        {
          struct plt_entry *ent;
-         for (ent = h->oh->plt.plist; ent != NULL; ent = ent->next)
+
+         for (ent = h->elf.plt.plist; ent != NULL; ent = ent->next)
            if (ent->addend == rel->r_addend
                && ent->plt.offset != (bfd_vma) -1)
              {
-               *hash = (struct ppc_link_hash_entry *) h->oh;
+               *hash = h;
                return ppc_stub_plt_call;
              }
        }
 
-      if (h->elf.root.type != bfd_link_hash_defined
-         && h->elf.root.type != bfd_link_hash_defweak)
+      if (!(h->elf.root.type == bfd_link_hash_defined
+           || h->elf.root.type == bfd_link_hash_defweak)
+         || h->elf.root.u.def.section->output_section == NULL)
        return ppc_stub_none;
     }
 
@@ -6142,7 +7262,7 @@ ppc_build_one_stub (struct bfd_hash_entry *gen_entry, void *in_arg)
   htab = ppc_hash_table (info);
 
   /* Make a note of the offset within the stubs for this entry.  */
-  stub_entry->stub_offset = stub_entry->stub_sec->_cooked_size;
+  stub_entry->stub_offset = stub_entry->stub_sec->size;
   loc = stub_entry->stub_sec->contents + stub_entry->stub_offset;
 
   htab->stub_count[stub_entry->stub_type - 1] += 1;
@@ -6270,17 +7390,18 @@ ppc_build_one_stub (struct bfd_hash_entry *gen_entry, void *in_arg)
       /* Do the best we can for shared libraries built without
         exporting ".foo" for each "foo".  This can happen when symbol
         versioning scripts strip all bar a subset of symbols.  */
-      if (stub_entry->h->oh->root.type != bfd_link_hash_defined
-         && stub_entry->h->oh->root.type != bfd_link_hash_defweak)
+      if (stub_entry->h->oh != NULL
+         && stub_entry->h->oh->elf.root.type != bfd_link_hash_defined
+         && stub_entry->h->oh->elf.root.type != bfd_link_hash_defweak)
        {
          /* Point the symbol at the stub.  There may be multiple stubs,
             we don't really care;  The main thing is to make this sym
             defined somewhere.  Maybe defining the symbol in the stub
             section is a silly idea.  If we didn't do this, htab->top_id
             could disappear.  */
-         stub_entry->h->oh->root.type = bfd_link_hash_defined;
-         stub_entry->h->oh->root.u.def.section = stub_entry->stub_sec;
-         stub_entry->h->oh->root.u.def.value = stub_entry->stub_offset;
+         stub_entry->h->oh->elf.root.type = bfd_link_hash_defined;
+         stub_entry->h->oh->elf.root.u.def.section = stub_entry->stub_sec;
+         stub_entry->h->oh->elf.root.u.def.value = stub_entry->stub_offset;
        }
 
       /* Now build the stub.  */
@@ -6319,13 +7440,14 @@ ppc_build_one_stub (struct bfd_hash_entry *gen_entry, void *in_arg)
       return FALSE;
     }
 
-  stub_entry->stub_sec->_cooked_size += size;
+  stub_entry->stub_sec->size += size;
 
   if (htab->emit_stub_syms
       && !(stub_entry->stub_type == ppc_stub_plt_call
-          && stub_entry->h->oh->root.type == bfd_link_hash_defined
-          && stub_entry->h->oh->root.u.def.section == stub_entry->stub_sec
-          && stub_entry->h->oh->root.u.def.value == stub_entry->stub_offset))
+          && stub_entry->h->oh != NULL
+          && stub_entry->h->oh->elf.root.type == bfd_link_hash_defined
+          && stub_entry->h->oh->elf.root.u.def.section == stub_entry->stub_sec
+          && stub_entry->h->oh->elf.root.u.def.value == stub_entry->stub_offset))
     {
       struct elf_link_hash_entry *h;
       h = elf_link_hash_lookup (&htab->elf, stub_entry->root.string,
@@ -6394,7 +7516,7 @@ ppc_size_one_stub (struct bfd_hash_entry *gen_entry, void *in_arg)
       off = (stub_entry->target_value
             + stub_entry->target_section->output_offset
             + stub_entry->target_section->output_section->vma);
-      off -= (stub_entry->stub_sec->_raw_size
+      off -= (stub_entry->stub_sec->size
              + stub_entry->stub_sec->output_offset
              + stub_entry->stub_sec->output_section->vma);
 
@@ -6429,11 +7551,11 @@ ppc_size_one_stub (struct bfd_hash_entry *gen_entry, void *in_arg)
          if (br_entry->iter != htab->stub_iteration)
            {
              br_entry->iter = htab->stub_iteration;
-             br_entry->offset = htab->brlt->_raw_size;
-             htab->brlt->_raw_size += 8;
+             br_entry->offset = htab->brlt->size;
+             htab->brlt->size += 8;
 
              if (info->shared)
-               htab->relbrlt->_raw_size += sizeof (Elf64_External_Rela);
+               htab->relbrlt->size += sizeof (Elf64_External_Rela);
            }
 
          stub_entry->stub_type += ppc_stub_plt_branch - ppc_stub_long_branch;
@@ -6443,7 +7565,7 @@ ppc_size_one_stub (struct bfd_hash_entry *gen_entry, void *in_arg)
        }
     }
 
-  stub_entry->stub_sec->_raw_size += size;
+  stub_entry->stub_sec->size += size;
   return TRUE;
 }
 
@@ -6520,20 +7642,15 @@ void
 ppc64_elf_next_toc_section (struct bfd_link_info *info, asection *isec)
 {
   struct ppc_link_hash_table *htab = ppc_hash_table (info);
+  bfd_vma addr = isec->output_offset + isec->output_section->vma;
+  bfd_vma off = addr - htab->toc_curr;
 
-  if (!htab->no_multi_toc)
-    {
-      bfd_vma addr = isec->output_offset + isec->output_section->vma;
-      bfd_vma off = addr - htab->toc_curr;
-      if (off + isec->_raw_size > 0x10000)
-       {
-         htab->toc_curr = addr;
-         htab->multi_toc_needed = 1;
-       }
-      elf_gp (isec->owner) = (htab->toc_curr
-                             - elf_gp (isec->output_section->owner)
-                             + TOC_BASE_OFF);
-    }
+  if (off + isec->size > 0x10000)
+    htab->toc_curr = addr;
+
+  elf_gp (isec->owner) = (htab->toc_curr
+                         - elf_gp (isec->output_section->owner)
+                         + TOC_BASE_OFF);
 }
 
 /* Called after the last call to the above function.  */
@@ -6566,7 +7683,7 @@ toc_adjusting_stub_needed (struct bfd_link_info *info, asection *isec)
   if ((isec->flags & SEC_LINKER_CREATED) != 0)
     return 0;
 
-  if (isec->_raw_size == 0)
+  if (isec->size == 0)
     return 0;
 
   /* Hack for linux kernel.  .fixup contains branches, but only back to
@@ -6576,13 +7693,10 @@ toc_adjusting_stub_needed (struct bfd_link_info *info, asection *isec)
   contents = elf_section_data (isec)->this_hdr.contents;
   if (contents == NULL)
     {
-      contents = bfd_malloc (isec->_raw_size);
-      if (contents == NULL)
-       return -1;
-      if (! bfd_get_section_contents (isec->owner, isec, contents,
-                                     0, isec->_raw_size))
+      if (!bfd_malloc_and_get_section (isec->owner, isec, &contents))
        {
-         free (contents);
+         if (contents != NULL)
+           free (contents);
          return -1;
        }
       if (info->keep_memory)
@@ -6592,7 +7706,7 @@ toc_adjusting_stub_needed (struct bfd_link_info *info, asection *isec)
   /* Code scan, because we don't necessarily have relocs on calls to
      static functions.  */
   ret = 0;
-  for (i = 0; i < isec->_raw_size; i += 4)
+  for (i = 0; i < isec->size; i += 4)
     {
       unsigned long insn = bfd_get_32 (isec->owner, contents + i);
       /* Is this a branch?  */
@@ -6682,10 +7796,7 @@ group_sections (struct ppc_link_hash_table *htab,
          bfd_vma curr_toc;
 
          curr = tail;
-         if (tail->_cooked_size)
-           total = tail->_cooked_size;
-         else
-           total = tail->_raw_size;
+         total = tail->size;
          big_sec = total >= stub_group_size;
          curr_toc = htab->stub_group[tail->id].toc_off;
 
@@ -6841,14 +7952,17 @@ ppc64_elf_size_stubs (bfd *output_bfd,
                  unsigned int r_indx;
                  enum ppc_stub_type stub_type;
                  struct ppc_stub_hash_entry *stub_entry;
-                 asection *sym_sec;
+                 asection *sym_sec, *code_sec;
                  bfd_vma sym_value;
                  bfd_vma destination;
+                 bfd_boolean ok_dest;
                  struct ppc_link_hash_entry *hash;
+                 struct ppc_link_hash_entry *fdh;
                  struct elf_link_hash_entry *h;
                  Elf_Internal_Sym *sym;
                  char *stub_name;
                  const asection *id_sec;
+                 long *opd_adjust;
 
                  r_type = ELF64_R_TYPE (irela->r_info);
                  r_indx = ELF64_R_SYM (irela->r_info);
@@ -6868,32 +7982,44 @@ ppc64_elf_size_stubs (bfd *output_bfd,
 
                  /* Now determine the call target, its name, value,
                     section.  */
-                 destination = 0;
                  if (!get_sym_h (&h, &sym, &sym_sec, NULL, &local_syms,
                                  r_indx, input_bfd))
                    goto error_ret_free_internal;
                  hash = (struct ppc_link_hash_entry *) h;
 
+                 ok_dest = FALSE;
+                 fdh = NULL;
                  if (hash == NULL)
                    {
-                     /* It's a local symbol.  */
                      sym_value = sym->st_value;
-                     destination = (sym_value + irela->r_addend
-                                    + sym_sec->output_offset
-                                    + sym_sec->output_section->vma);
+                     ok_dest = TRUE;
                    }
                  else
                    {
-                     /* It's an external symbol.  */
                      sym_value = 0;
-                     if (hash->elf.root.type == bfd_link_hash_defined
-                         || hash->elf.root.type == bfd_link_hash_defweak)
+                     /* Recognise an old ABI func code entry sym, and
+                        use the func descriptor sym instead.  */
+                     if (hash->elf.root.type == bfd_link_hash_undefweak
+                         && hash->elf.root.root.string[0] == '.'
+                         && (fdh = get_fdh (hash, htab)) != NULL)
+                       {
+                         if (fdh->elf.root.type == bfd_link_hash_defined
+                             || fdh->elf.root.type == bfd_link_hash_defweak)
+                           {
+                             sym_sec = fdh->elf.root.u.def.section;
+                             sym_value = fdh->elf.root.u.def.value;
+                             if (sym_sec->output_section != NULL)
+                               ok_dest = TRUE;
+                           }
+                         else
+                           fdh = NULL;
+                       }
+                     else if (hash->elf.root.type == bfd_link_hash_defined
+                              || hash->elf.root.type == bfd_link_hash_defweak)
                        {
                          sym_value = hash->elf.root.u.def.value;
                          if (sym_sec->output_section != NULL)
-                           destination = (sym_value + irela->r_addend
-                                          + sym_sec->output_offset
-                                          + sym_sec->output_section->vma);
+                           ok_dest = TRUE;
                        }
                      else if (hash->elf.root.type == bfd_link_hash_undefweak)
                        ;
@@ -6906,6 +8032,44 @@ ppc64_elf_size_stubs (bfd *output_bfd,
                        }
                    }
 
+                 destination = 0;
+                 if (ok_dest)
+                   {
+                     sym_value += irela->r_addend;
+                     destination = (sym_value
+                                    + sym_sec->output_offset
+                                    + sym_sec->output_section->vma);
+                   }
+
+                 code_sec = sym_sec;
+                 opd_adjust = get_opd_info (sym_sec);
+                 if (opd_adjust != NULL)
+                   {
+                     bfd_vma dest;
+
+                     if (hash == NULL)
+                       {
+                         long adjust = opd_adjust[sym_value / 8];
+                         if (adjust == -1)
+                           continue;
+                         sym_value += adjust;
+                       }
+                     dest = opd_entry_value (sym_sec, sym_value,
+                                             &code_sec, &sym_value);
+                     if (dest != (bfd_vma) -1)
+                       {
+                         destination = dest;
+                         if (fdh != NULL)
+                           {
+                             /* Fixup old ABI sym to point at code
+                                entry.  */
+                             hash->elf.root.type = bfd_link_hash_defweak;
+                             hash->elf.root.u.def.section = code_sec;
+                             hash->elf.root.u.def.value = sym_value;
+                           }
+                       }
+                   }
+
                  /* Determine what (if any) linker stub is needed.  */
                  stub_type = ppc_type_of_stub (section, irela, &hash,
                                                destination);
@@ -6918,11 +8082,11 @@ ppc64_elf_size_stubs (bfd *output_bfd,
                         _init and _fini functions, it may be that a
                         call to what looks like a local sym is in
                         fact a call needing a TOC adjustment.  */
-                     if (sym_sec != NULL
-                         && sym_sec->output_section != NULL
-                         && (htab->stub_group[sym_sec->id].toc_off
+                     if (code_sec != NULL
+                         && code_sec->output_section != NULL
+                         && (htab->stub_group[code_sec->id].toc_off
                              != htab->stub_group[section->id].toc_off)
-                         && sym_sec->has_gp_reloc
+                         && code_sec->has_gp_reloc
                          && section->has_gp_reloc)
                        stub_type = ppc_stub_long_branch_r2off;
                    }
@@ -6933,7 +8097,8 @@ ppc64_elf_size_stubs (bfd *output_bfd,
                  /* __tls_get_addr calls might be eliminated.  */
                  if (stub_type != ppc_stub_plt_call
                      && hash != NULL
-                     && &hash->elf == htab->tls_get_addr
+                     && (hash == htab->tls_get_addr
+                         || hash == htab->tls_get_addr_fd)
                      && section->has_tls_reloc
                      && irela != internal_relocs)
                    {
@@ -6981,7 +8146,7 @@ ppc64_elf_size_stubs (bfd *output_bfd,
 
                  stub_entry->stub_type = stub_type;
                  stub_entry->target_value = sym_value;
-                 stub_entry->target_section = sym_sec;
+                 stub_entry->target_section = code_sec;
                  stub_entry->h = hash;
                  stub_entry->addend = irela->r_addend;
                  stub_changed = TRUE;
@@ -7011,17 +8176,11 @@ ppc64_elf_size_stubs (bfd *output_bfd,
           stub_sec != NULL;
           stub_sec = stub_sec->next)
        if ((stub_sec->flags & SEC_LINKER_CREATED) == 0)
-         {
-           stub_sec->_raw_size = 0;
-           stub_sec->_cooked_size = 0;
-         }
-      htab->brlt->_raw_size = 0;
-      htab->brlt->_cooked_size = 0;
+         stub_sec->size = 0;
+
+      htab->brlt->size = 0;
       if (info->shared)
-       {
-         htab->relbrlt->_raw_size = 0;
-         htab->relbrlt->_cooked_size = 0;
-       }
+       htab->relbrlt->size = 0;
 
       bfd_hash_traverse (&htab->stub_hash_table, ppc_size_one_stub, info);
 
@@ -7108,22 +8267,21 @@ ppc64_elf_build_stubs (bfd_boolean emit_stub_syms,
   int stub_sec_count = 0;
 
   htab->emit_stub_syms = emit_stub_syms;
+
+  /* Allocate memory to hold the linker stubs.  */
   for (stub_sec = htab->stub_bfd->sections;
        stub_sec != NULL;
        stub_sec = stub_sec->next)
-    if ((stub_sec->flags & SEC_LINKER_CREATED) == 0)
+    if ((stub_sec->flags & SEC_LINKER_CREATED) == 0
+       && stub_sec->size != 0)
       {
-       bfd_size_type size;
-
-       /* Allocate memory to hold the linker stubs.  */
-       size = stub_sec->_raw_size;
-       if (size != 0)
-         {
-           stub_sec->contents = bfd_zalloc (htab->stub_bfd, size);
-           if (stub_sec->contents == NULL)
-             return FALSE;
-         }
-       stub_sec->_cooked_size = 0;
+       stub_sec->contents = bfd_zalloc (htab->stub_bfd, stub_sec->size);
+       if (stub_sec->contents == NULL)
+         return FALSE;
+       /* We want to check that built size is the same as calculated
+          size.  rawsize is a convenient location to use.  */
+       stub_sec->rawsize = stub_sec->size;
+       stub_sec->size = 0;
       }
 
   if (htab->plt != NULL)
@@ -7197,7 +8355,7 @@ ppc64_elf_build_stubs (bfd_boolean emit_stub_syms,
 
       /* Build the .glink lazy link call stubs.  */
       indx = 0;
-      while (p < htab->glink->contents + htab->glink->_raw_size)
+      while (p < htab->glink->contents + htab->glink->size)
        {
          if (indx < 0x8000)
            {
@@ -7216,20 +8374,20 @@ ppc64_elf_build_stubs (bfd_boolean emit_stub_syms,
          indx++;
          p += 4;
        }
-      htab->glink->_cooked_size = p - htab->glink->contents;
+      htab->glink->rawsize = p - htab->glink->contents;
     }
 
-  if (htab->brlt->_raw_size != 0)
+  if (htab->brlt->size != 0)
     {
       htab->brlt->contents = bfd_zalloc (htab->brlt->owner,
-                                        htab->brlt->_raw_size);
+                                        htab->brlt->size);
       if (htab->brlt->contents == NULL)
        return FALSE;
     }
-  if (info->shared && htab->relbrlt->_raw_size != 0)
+  if (info->shared && htab->relbrlt->size != 0)
     {
       htab->relbrlt->contents = bfd_zalloc (htab->relbrlt->owner,
-                                           htab->relbrlt->_raw_size);
+                                           htab->relbrlt->size);
       if (htab->relbrlt->contents == NULL)
        return FALSE;
     }
@@ -7243,12 +8401,12 @@ ppc64_elf_build_stubs (bfd_boolean emit_stub_syms,
     if ((stub_sec->flags & SEC_LINKER_CREATED) == 0)
       {
        stub_sec_count += 1;
-       if (stub_sec->_raw_size != stub_sec->_cooked_size)
+       if (stub_sec->rawsize != stub_sec->size)
          break;
       }
 
   if (stub_sec != NULL
-      || htab->glink->_raw_size != htab->glink->_cooked_size)
+      || htab->glink->rawsize != htab->glink->size)
     {
       htab->stub_error = TRUE;
       (*_bfd_error_handler) (_("stubs don't match calculated size"));
@@ -7279,6 +8437,34 @@ ppc64_elf_build_stubs (bfd_boolean emit_stub_syms,
   return TRUE;
 }
 
+/* This function undoes the changes made by add_symbol_adjust.  */
+
+static bfd_boolean
+undo_symbol_twiddle (struct elf_link_hash_entry *h, void *inf ATTRIBUTE_UNUSED)
+{
+  struct ppc_link_hash_entry *eh;
+
+  if (h->root.type == bfd_link_hash_indirect)
+    return TRUE;
+
+  if (h->root.type == bfd_link_hash_warning)
+    h = (struct elf_link_hash_entry *) h->root.u.i.link;
+
+  eh = (struct ppc_link_hash_entry *) h;
+  if (eh->elf.root.type != bfd_link_hash_undefweak || !eh->was_undefined)
+    return TRUE;
+
+  eh->elf.root.type = bfd_link_hash_undefined;
+  return TRUE;
+}
+
+void
+ppc64_elf_restore_symbols (struct bfd_link_info *info)
+{
+  struct ppc_link_hash_table *htab = ppc_hash_table (info);
+  elf_link_hash_traverse (&htab->elf, undo_symbol_twiddle, info);
+}
+
 /* The RELOCATE_SECTION function is called by the ELF backend linker
    to handle the relocations for a section.
 
@@ -7355,8 +8541,9 @@ ppc64_elf_relocate_section (bfd *output_bfd,
       bfd_reloc_status_type r;
       Elf_Internal_Sym *sym;
       asection *sec;
-      struct elf_link_hash_entry *h;
-      struct elf_link_hash_entry *fdh;
+      struct elf_link_hash_entry *h_elf;
+      struct ppc_link_hash_entry *h;
+      struct ppc_link_hash_entry *fdh;
       const char *sym_name;
       unsigned long r_symndx, toc_symndx;
       char tls_mask, tls_gd, tls_type;
@@ -7383,7 +8570,7 @@ ppc64_elf_relocate_section (bfd *output_bfd,
 
       sym = NULL;
       sec = NULL;
-      h = NULL;
+      h_elf = NULL;
       sym_name = NULL;
       unresolved_reloc = FALSE;
       warned = FALSE;
@@ -7391,29 +8578,33 @@ ppc64_elf_relocate_section (bfd *output_bfd,
       if (r_symndx < symtab_hdr->sh_info)
        {
          /* It's a local symbol.  */
+         long *opd_adjust;
+
          sym = local_syms + r_symndx;
          sec = local_sections[r_symndx];
          sym_name = bfd_elf_local_sym_name (input_bfd, sym);
          sym_type = ELF64_ST_TYPE (sym->st_info);
          relocation = _bfd_elf_rela_local_sym (output_bfd, sym, &sec, rel);
-         if (elf_section_data (sec) != NULL)
+         opd_adjust = get_opd_info (sec);
+         if (opd_adjust != NULL)
            {
-             long *opd_sym_adjust;
-
-             opd_sym_adjust = ppc64_elf_section_data (sec)->opd.adjust;
-             if (opd_sym_adjust != NULL && sym->st_value % 24 == 0)
-               relocation += opd_sym_adjust[sym->st_value / 24];
+             long adjust = opd_adjust[(sym->st_value + rel->r_addend) / 8];
+             if (adjust == -1)
+               relocation = 0;
+             else
+               relocation += adjust;
            }
        }
       else
        {
-         RELOC_FOR_GLOBAL_SYMBOL (h, sym_hashes, r_symndx,
-                                  symtab_hdr, relocation, sec,
-                                  unresolved_reloc, info,
-                                  warned);
-         sym_name = h->root.root.string;
-         sym_type = h->type;
+         RELOC_FOR_GLOBAL_SYMBOL (info, input_bfd, input_section, rel,
+                                  r_symndx, symtab_hdr, sym_hashes,
+                                  h_elf, sec, relocation,
+                                  unresolved_reloc, warned);
+         sym_name = h_elf->root.root.string;
+         sym_type = h_elf->type;
        }
+      h = (struct ppc_link_hash_entry *) h_elf;
 
       /* TLS optimizations.  Replace instruction sequences and relocs
         based on information we collected in tls_optimize.  We edit
@@ -7425,7 +8616,7 @@ ppc64_elf_relocate_section (bfd *output_bfd,
       if (IS_PPC64_TLS_RELOC (r_type))
        {
          if (h != NULL)
-           tls_mask = ((struct ppc_link_hash_entry *) h)->tls_mask;
+           tls_mask = h->tls_mask;
          else if (local_got_ents != NULL)
            {
              char *lgot_masks;
@@ -7451,8 +8642,8 @@ ppc64_elf_relocate_section (bfd *output_bfd,
       if (r_symndx != 0
          && r_type != R_PPC64_NONE
          && (h == NULL
-             || h->root.type == bfd_link_hash_defined
-             || h->root.type == bfd_link_hash_defweak)
+             || h->elf.root.type == bfd_link_hash_defined
+             || h->elf.root.type == bfd_link_hash_defweak)
          && IS_PPC64_TLS_RELOC (r_type) != (sym_type == STT_TLS))
        {
          if (r_type == R_PPC64_TLS && tls_mask != 0)
@@ -7461,10 +8652,10 @@ ppc64_elf_relocate_section (bfd *output_bfd,
          else
            (*_bfd_error_handler)
              (sym_type == STT_TLS
-              ? _("%s(%s+0x%lx): %s used with TLS symbol %s")
-              : _("%s(%s+0x%lx): %s used with non-TLS symbol %s"),
-              bfd_archive_filename (input_bfd),
-              input_section->name,
+              ? _("%B(%A+0x%lx): %s used with TLS symbol %s")
+              : _("%B(%A+0x%lx): %s used with non-TLS symbol %s"),
+              input_bfd,
+              input_section,
               (long) rel->r_offset,
               ppc64_elf_howto_table[r_type]->name,
               sym_name);
@@ -7669,7 +8860,8 @@ ppc64_elf_relocate_section (bfd *output_bfd,
                  while (h2->root.type == bfd_link_hash_indirect
                         || h2->root.type == bfd_link_hash_warning)
                    h2 = (struct elf_link_hash_entry *) h2->root.u.i.link;
-                 if (h2 == NULL || h2 != htab->tls_get_addr)
+                 if (h2 == NULL || (h2 != &htab->tls_get_addr->elf
+                                    && h2 != &htab->tls_get_addr_fd->elf))
                    break;
 
                  /* OK, it checks out.  Replace the call.  */
@@ -7773,6 +8965,8 @@ ppc64_elf_relocate_section (bfd *output_bfd,
 
       /* Handle other relocations that tweak non-addend part of insn.  */
       insn = 0;
+      max_br_offset = 1 << 25;
+      addend = rel->r_addend;
       switch (r_type)
        {
        default:
@@ -7789,31 +8983,11 @@ ppc64_elf_relocate_section (bfd *output_bfd,
        case R_PPC64_REL14_BRNTAKEN:
          insn |= bfd_get_32 (output_bfd,
                              contents + rel->r_offset) & ~(0x01 << 21);
-         if (is_power4)
-           {
-             /* Set 'a' bit.  This is 0b00010 in BO field for branch
-                on CR(BI) insns (BO == 001at or 011at), and 0b01000
-                for branch on CTR insns (BO == 1a00t or 1a01t).  */
-             if ((insn & (0x14 << 21)) == (0x04 << 21))
-               insn |= 0x02 << 21;
-             else if ((insn & (0x14 << 21)) == (0x10 << 21))
-               insn |= 0x08 << 21;
-             else
-               break;
-           }
-         else
-           {
-             from = (rel->r_offset
-                     + input_section->output_offset
-                     + input_section->output_section->vma);
-
-             /* Invert 'y' bit if not the default.  */
-             if ((bfd_signed_vma) (relocation + rel->r_addend - from) < 0)
-               insn ^= 0x01 << 21;
-           }
+         /* Fall thru.  */
 
-         bfd_put_32 (output_bfd, insn, contents + rel->r_offset);
-         break;
+       case R_PPC64_REL14:
+         max_br_offset = 1 << 15;
+         /* Fall thru.  */
 
        case R_PPC64_REL24:
          /* Calls to functions with a different TOC, such as calls to
@@ -7822,11 +8996,15 @@ ppc64_elf_relocate_section (bfd *output_bfd,
             linkage stubs needs to be followed by a nop, as the nop
             will be replaced with an instruction to restore the TOC
             base pointer.  */
+         stub_entry = NULL;
+         fdh = h;
          if (((h != NULL
-               && (fdh = ((struct ppc_link_hash_entry *) h)->oh) != NULL
-               && fdh->plt.plist != NULL)
-              || ((fdh = h, sec) != NULL
+               && (((fdh = h->oh) != NULL
+                    && fdh->elf.plt.plist != NULL)
+                   || (fdh = h)->elf.plt.plist != NULL))
+              || (sec != NULL
                   && sec->output_section != NULL
+                  && sec->id <= htab->top_id
                   && (htab->stub_group[sec->id].toc_off
                       != htab->stub_group[input_section->id].toc_off)))
              && (stub_entry = ppc_get_stub_entry (input_section, sec, fdh,
@@ -7835,17 +9013,18 @@ ppc64_elf_relocate_section (bfd *output_bfd,
                  || stub_entry->stub_type == ppc_stub_plt_branch_r2off
                  || stub_entry->stub_type == ppc_stub_long_branch_r2off))
            {
-             bfd_boolean can_plt_call = 0;
+             bfd_boolean can_plt_call = FALSE;
 
-             if (rel->r_offset + 8 <= input_section->_cooked_size)
+             if (rel->r_offset + 8 <= input_section->size)
                {
-                 insn = bfd_get_32 (input_bfd, contents + rel->r_offset + 4);
-                 if (insn == NOP
-                     || insn == CROR_151515 || insn == CROR_313131)
+                 unsigned long nop;
+                 nop = bfd_get_32 (input_bfd, contents + rel->r_offset + 4);
+                 if (nop == NOP
+                     || nop == CROR_151515 || nop == CROR_313131)
                    {
                      bfd_put_32 (input_bfd, LD_R2_40R1,
                                  contents + rel->r_offset + 4);
-                     can_plt_call = 1;
+                     can_plt_call = TRUE;
                    }
                }
 
@@ -7855,16 +9034,17 @@ ppc64_elf_relocate_section (bfd *output_bfd,
                    {
                      /* If this is a plain branch rather than a branch
                         and link, don't require a nop.  */
-                     insn = bfd_get_32 (input_bfd, contents + rel->r_offset);
-                     if ((insn & 1) == 0)
-                       can_plt_call = 1;
+                     unsigned long br;
+                     br = bfd_get_32 (input_bfd, contents + rel->r_offset);
+                     if ((br & 1) == 0)
+                       can_plt_call = TRUE;
                    }
                  else if (h != NULL
-                          && strcmp (h->root.root.string,
+                          && strcmp (h->elf.root.root.string,
                                      ".__libc_start_main") == 0)
                    {
                      /* Allow crt1 branch to go via a toc adjusting stub.  */
-                     can_plt_call = 1;
+                     can_plt_call = TRUE;
                    }
                  else
                    {
@@ -7873,21 +9053,21 @@ ppc64_elf_relocate_section (bfd *output_bfd,
                          || strcmp (input_section->output_section->name,
                                     ".fini") == 0)
                        (*_bfd_error_handler)
-                         (_("%s(%s+0x%lx): automatic multiple TOCs "
+                         (_("%B(%A+0x%lx): automatic multiple TOCs "
                             "not supported using your crt files; "
                             "recompile with -mminimal-toc or upgrade gcc"),
-                          bfd_archive_filename (input_bfd),
-                          input_section->name,
+                          input_bfd,
+                          input_section,
                           (long) rel->r_offset);
                      else
                        (*_bfd_error_handler)
-                         (_("%s(%s+0x%lx): sibling call optimization to `%s' "
+                         (_("%B(%A+0x%lx): sibling call optimization to `%s' "
                             "does not allow automatic multiple TOCs; "
                             "recompile with -mminimal-toc or "
                             "-fno-optimize-sibling-calls, "
                             "or make `%s' extern"),
-                          bfd_archive_filename (input_bfd),
-                          input_section->name,
+                          input_bfd,
+                          input_section,
                           (long) rel->r_offset,
                           sym_name,
                           sym_name);
@@ -7896,51 +9076,96 @@ ppc64_elf_relocate_section (bfd *output_bfd,
                    }
                }
 
-             if (can_plt_call)
+             if (can_plt_call
+                 && stub_entry->stub_type == ppc_stub_plt_call)
+               unresolved_reloc = FALSE;
+           }
+
+         if (stub_entry == NULL
+             && get_opd_info (sec) != NULL)
+           {
+             /* The branch destination is the value of the opd entry. */
+             bfd_vma off = (relocation - sec->output_section->vma
+                            - sec->output_offset + rel->r_addend);
+             bfd_vma dest = opd_entry_value (sec, off, NULL, NULL);
+             if (dest != (bfd_vma) -1)
                {
-                 relocation = (stub_entry->stub_offset
-                               + stub_entry->stub_sec->output_offset
-                               + stub_entry->stub_sec->output_section->vma);
-                 if (stub_entry->stub_type == ppc_stub_plt_call)
-                   unresolved_reloc = FALSE;
+                 relocation = dest;
+                 addend = 0;
+               }
+           }
+
+         /* If the branch is out of reach we ought to have a long
+            branch stub.  */
+         from = (rel->r_offset
+                 + input_section->output_offset
+                 + input_section->output_section->vma);
+
+         if (stub_entry == NULL
+             && (relocation + rel->r_addend - from + max_br_offset
+                 >= 2 * max_br_offset)
+             && r_type != R_PPC64_ADDR14_BRTAKEN
+             && r_type != R_PPC64_ADDR14_BRNTAKEN)
+           stub_entry = ppc_get_stub_entry (input_section, sec, h, rel,
+                                            htab);
+
+         if (stub_entry != NULL)
+           {
+             /* Munge up the value and addend so that we call the stub
+                rather than the procedure directly.  */
+             relocation = (stub_entry->stub_offset
+                           + stub_entry->stub_sec->output_offset
+                           + stub_entry->stub_sec->output_section->vma);
+             addend = 0;
+           }
+
+         if (insn != 0)
+           {
+             if (is_power4)
+               {
+                 /* Set 'a' bit.  This is 0b00010 in BO field for branch
+                    on CR(BI) insns (BO == 001at or 011at), and 0b01000
+                    for branch on CTR insns (BO == 1a00t or 1a01t).  */
+                 if ((insn & (0x14 << 21)) == (0x04 << 21))
+                   insn |= 0x02 << 21;
+                 else if ((insn & (0x14 << 21)) == (0x10 << 21))
+                   insn |= 0x08 << 21;
+                 else
+                   break;
+               }
+             else
+               {
+                 /* Invert 'y' bit if not the default.  */
+                 if ((bfd_signed_vma) (relocation + rel->r_addend - from) < 0)
+                   insn ^= 0x01 << 21;
                }
+
+             bfd_put_32 (output_bfd, insn, contents + rel->r_offset);
            }
 
-         if (h != NULL
-             && h->root.type == bfd_link_hash_undefweak
-             && relocation == 0
-             && rel->r_addend == 0)
+         /* NOP out calls to undefined weak functions.
+            We can thus call a weak function without first
+            checking whether the function is defined.  */
+         else if (h != NULL
+                  && h->elf.root.type == bfd_link_hash_undefweak
+                  && r_type == R_PPC64_REL24
+                  && relocation == 0
+                  && rel->r_addend == 0)
            {
-             /* Tweak calls to undefined weak functions to point at a
-                blr.  We can thus call a weak function without first
-                checking whether the function is defined.  We have a
-                blr at the end of .sfpr.  */
-             BFD_ASSERT (htab->sfpr->_raw_size != 0);
-             relocation = (htab->sfpr->_raw_size - 4
-                           + htab->sfpr->output_offset
-                           + htab->sfpr->output_section->vma);
-             from = (rel->r_offset
-                     + input_section->output_offset
-                     + input_section->output_section->vma);
-
-             /* But let's not be silly about it.  If the blr isn't in
-                reach, just go to the next instruction.  */
-             if (relocation - from + (1 << 25) >= (1 << 26)
-                 || htab->sfpr->_raw_size == 0)
-               relocation = from + 4;
+             bfd_put_32 (output_bfd, NOP, contents + rel->r_offset);
+             continue;
            }
          break;
        }
 
       /* Set `addend'.  */
       tls_type = 0;
-      addend = rel->r_addend;
       switch (r_type)
        {
        default:
          (*_bfd_error_handler)
-           (_("%s: unknown relocation type %d for symbol %s"),
-            bfd_archive_filename (input_bfd), (int) r_type, sym_name);
+           (_("%B: unknown relocation type %d for symbol %s"),
+            input_bfd, (int) r_type, sym_name);
 
          bfd_set_error (bfd_error_bad_value);
          ret = FALSE;
@@ -8001,7 +9226,8 @@ ppc64_elf_relocate_section (bfd *output_bfd,
 
            if (tls_type == (TLS_TLS | TLS_LD)
                && (h == NULL
-                   || !(h->elf_link_hash_flags & ELF_LINK_HASH_DEF_DYNAMIC)))
+                   || (h->elf.elf_link_hash_flags
+                       & ELF_LINK_HASH_DEF_DYNAMIC) == 0))
              offp = &ppc64_tlsld_got (input_bfd)->offset;
            else
              {
@@ -8010,9 +9236,10 @@ ppc64_elf_relocate_section (bfd *output_bfd,
                if (h != NULL)
                  {
                    bfd_boolean dyn = htab->elf.dynamic_sections_created;
-                   if (!WILL_CALL_FINISH_DYNAMIC_SYMBOL (dyn, info->shared, h)
+                   if (!WILL_CALL_FINISH_DYNAMIC_SYMBOL (dyn, info->shared,
+                                                         &h->elf)
                        || (info->shared
-                           && SYMBOL_REFERENCES_LOCAL (info, h)))
+                           && SYMBOL_REFERENCES_LOCAL (info, &h->elf)))
                      /* This is actually a static link, or it is a
                         -Bsymbolic link and the symbol is defined
                         locally, or the symbol was forced to be local
@@ -8020,10 +9247,10 @@ ppc64_elf_relocate_section (bfd *output_bfd,
                      ;
                    else
                      {
-                       indx = h->dynindx;
+                       indx = h->elf.dynindx;
                        unresolved_reloc = FALSE;
                      }
-                   ent = h->got.glist;
+                   ent = h->elf.got.glist;
                  }
                else
                  {
@@ -8062,8 +9289,8 @@ ppc64_elf_relocate_section (bfd *output_bfd,
                *offp = off | 1;
                if ((info->shared || indx != 0)
                    && (h == NULL
-                       || ELF_ST_VISIBILITY (h->other) == STV_DEFAULT
-                       || h->root.type != bfd_link_hash_undefweak))
+                       || ELF_ST_VISIBILITY (h->elf.other) == STV_DEFAULT
+                       || h->elf.root.type != bfd_link_hash_undefweak))
                  {
                    outrel.r_offset = (got->output_section->vma
                                       + got->output_offset
@@ -8171,7 +9398,7 @@ ppc64_elf_relocate_section (bfd *output_bfd,
          if (htab->plt != NULL)
            {
              struct plt_entry *ent;
-             for (ent = h->plt.plist; ent != NULL; ent = ent->next)
+             for (ent = h->elf.plt.plist; ent != NULL; ent = ent->next)
                if (ent->addend == rel->r_addend
                    && ent->plt.offset != (bfd_vma) -1)
                  {
@@ -8311,17 +9538,17 @@ ppc64_elf_relocate_section (bfd *output_bfd,
 
          if ((info->shared
               && (h == NULL
-                  || ELF_ST_VISIBILITY (h->other) == STV_DEFAULT
-                  || h->root.type != bfd_link_hash_undefweak)
+                  || ELF_ST_VISIBILITY (h->elf.other) == STV_DEFAULT
+                  || h->elf.root.type != bfd_link_hash_undefweak)
               && (MUST_BE_DYN_RELOC (r_type)
-                  || !SYMBOL_CALLS_LOCAL (info, h)))
+                  || !SYMBOL_CALLS_LOCAL (info, &h->elf)))
              || (ELIMINATE_COPY_RELOCS
                  && !info->shared
                  && h != NULL
-                 && h->dynindx != -1
-                 && (h->elf_link_hash_flags & ELF_LINK_NON_GOT_REF) == 0
-                 && (h->elf_link_hash_flags & ELF_LINK_HASH_DEF_DYNAMIC) != 0
-                 && (h->elf_link_hash_flags & ELF_LINK_HASH_DEF_REGULAR) == 0))
+                 && h->elf.dynindx != -1
+                 && !(h->elf.elf_link_hash_flags & ELF_LINK_NON_GOT_REF)
+                 && (h->elf.elf_link_hash_flags & ELF_LINK_HASH_DEF_DYNAMIC)
+                 && !(h->elf.elf_link_hash_flags & ELF_LINK_HASH_DEF_REGULAR)))
            {
              Elf_Internal_Rela outrel;
              bfd_boolean skip, relocate;
@@ -8360,10 +9587,10 @@ ppc64_elf_relocate_section (bfd *output_bfd,
 
              if (skip)
                memset (&outrel, 0, sizeof outrel);
-             else if (!SYMBOL_REFERENCES_LOCAL (info, h)
+             else if (!SYMBOL_REFERENCES_LOCAL (info, &h->elf)
                       && !is_opd
                       && r_type != R_PPC64_TOC)
-               outrel.r_info = ELF64_R_INFO (h->dynindx, r_type);
+               outrel.r_info = ELF64_R_INFO (h->elf.dynindx, r_type);
              else
                {
                  /* This symbol is local, or marked to become local,
@@ -8479,8 +9706,8 @@ ppc64_elf_relocate_section (bfd *output_bfd,
          /* These ones haven't been implemented yet.  */
 
          (*_bfd_error_handler)
-           (_("%s: relocation %s is not supported for symbol %s."),
-            bfd_archive_filename (input_bfd),
+           (_("%B: relocation %s is not supported for symbol %s."),
+            input_bfd,
             ppc64_elf_howto_table[r_type]->name, sym_name);
 
          bfd_set_error (bfd_error_invalid_operation);
@@ -8558,8 +9785,8 @@ ppc64_elf_relocate_section (bfd *output_bfd,
          if (((relocation + addend) & mask) != 0)
            {
              (*_bfd_error_handler)
-               (_("%s: error: relocation %s not a multiple of %d"),
-                bfd_archive_filename (input_bfd),
+               (_("%B: error: relocation %s not a multiple of %d"),
+                input_bfd,
                 ppc64_elf_howto_table[r_type]->name,
                 mask + 1);
              bfd_set_error (bfd_error_bad_value);
@@ -8567,40 +9794,6 @@ ppc64_elf_relocate_section (bfd *output_bfd,
              continue;
            }
          break;
-
-       case R_PPC64_REL14:
-       case R_PPC64_REL14_BRNTAKEN:
-       case R_PPC64_REL14_BRTAKEN:
-         max_br_offset = 1 << 15;
-         goto branch_check;
-
-       case R_PPC64_REL24:
-         max_br_offset = 1 << 25;
-
-       branch_check:
-         /* If the branch is out of reach or the TOC register needs
-            adjusting, then redirect the call to the local stub for
-            this function.  */
-         from = (rel->r_offset
-                 + input_section->output_offset
-                 + input_section->output_section->vma);
-         if ((relocation + addend - from + max_br_offset >= 2 * max_br_offset
-              || (sec != NULL
-                  && sec->output_section != NULL
-                  && sec->id <= htab->top_id
-                  && (htab->stub_group[sec->id].toc_off
-                      != htab->stub_group[input_section->id].toc_off)))
-             && (stub_entry = ppc_get_stub_entry (input_section, sec, h,
-                                                  rel, htab)) != NULL)
-           {
-             /* Munge up the value and addend so that we call the stub
-                rather than the procedure directly.  */
-             relocation = (stub_entry->stub_offset
-                           + stub_entry->stub_sec->output_offset
-                           + stub_entry->stub_sec->output_section->vma);
-             addend = 0;
-           }
-         break;
        }
 
       /* Dynamic relocs are not propagated for SEC_DEBUGGING sections
@@ -8608,15 +9801,15 @@ ppc64_elf_relocate_section (bfd *output_bfd,
         not process them.  */
       if (unresolved_reloc
          && !((input_section->flags & SEC_DEBUGGING) != 0
-              && (h->elf_link_hash_flags & ELF_LINK_HASH_DEF_DYNAMIC) != 0))
+              && (h->elf.elf_link_hash_flags & ELF_LINK_HASH_DEF_DYNAMIC)))
        {
          (*_bfd_error_handler)
-           (_("%s(%s+0x%lx): unresolvable %s relocation against symbol `%s'"),
-            bfd_archive_filename (input_bfd),
-            bfd_get_section_name (input_bfd, input_section),
+           (_("%B(%A+0x%lx): unresolvable %s relocation against symbol `%s'"),
+            input_bfd,
+            input_section,
             (long) rel->r_offset,
             ppc64_elf_howto_table[(int) r_type]->name,
-            h->root.root.string);
+            h->elf.root.root.string);
          ret = FALSE;
        }
 
@@ -8637,7 +9830,7 @@ ppc64_elf_relocate_section (bfd *output_bfd,
              if (warned)
                continue;
              if (h != NULL
-                 && h->root.type == bfd_link_hash_undefweak
+                 && h->elf.root.type == bfd_link_hash_undefweak
                  && ppc64_elf_howto_table[r_type]->pc_relative)
                {
                  /* Assume this is a call protected by other code that
@@ -8657,9 +9850,9 @@ ppc64_elf_relocate_section (bfd *output_bfd,
          else
            {
              (*_bfd_error_handler)
-               (_("%s(%s+0x%lx): %s reloc against `%s': error %d"),
-                bfd_archive_filename (input_bfd),
-                bfd_get_section_name (input_bfd, input_section),
+               (_("%B(%A+0x%lx): %s reloc against `%s': error %d"),
+                input_bfd,
+                input_section,
                 (long) rel->r_offset,
                 ppc64_elf_howto_table[r_type]->name,
                 sym_name,
@@ -8672,6 +9865,37 @@ ppc64_elf_relocate_section (bfd *output_bfd,
   return ret;
 }
 
+/* Adjust the value of any local symbols in opd sections.  */
+
+static bfd_boolean
+ppc64_elf_output_symbol_hook (struct bfd_link_info *info,
+                             const char *name ATTRIBUTE_UNUSED,
+                             Elf_Internal_Sym *elfsym,
+                             asection *input_sec,
+                             struct elf_link_hash_entry *h)
+{
+  long *opd_adjust, adjust;
+  bfd_vma value;
+
+  if (h != NULL)
+    return TRUE;
+
+  opd_adjust = get_opd_info (input_sec);
+  if (opd_adjust == NULL)
+    return TRUE;
+
+  value = elfsym->st_value - input_sec->output_offset;
+  if (!info->relocatable)
+    value -= input_sec->output_section->vma;
+
+  adjust = opd_adjust[value / 8];
+  if (adjust == -1)
+    elfsym->st_value = 0;
+  else
+    elfsym->st_value += adjust;
+  return TRUE;
+}
+
 /* Finish up dynamic symbol handling.  We set the contents of various
    dynamic sections here.  */
 
@@ -8683,41 +9907,37 @@ ppc64_elf_finish_dynamic_symbol (bfd *output_bfd,
 {
   struct ppc_link_hash_table *htab;
   bfd *dynobj;
+  struct plt_entry *ent;
+  Elf_Internal_Rela rela;
+  bfd_byte *loc;
 
   htab = ppc_hash_table (info);
   dynobj = htab->elf.dynobj;
 
-  if (((struct ppc_link_hash_entry *) h)->is_func_descriptor)
-    {
-      struct plt_entry *ent;
-      Elf_Internal_Rela rela;
-      bfd_byte *loc;
-
-      for (ent = h->plt.plist; ent != NULL; ent = ent->next)
-       if (ent->plt.offset != (bfd_vma) -1)
-         {
-           /* This symbol has an entry in the procedure linkage
-              table.  Set it up.  */
-
-           if (htab->plt == NULL
-               || htab->relplt == NULL
-               || htab->glink == NULL)
-             abort ();
-
-           /* Create a JMP_SLOT reloc to inform the dynamic linker to
-              fill in the PLT entry.  */
-           rela.r_offset = (htab->plt->output_section->vma
-                            + htab->plt->output_offset
-                            + ent->plt.offset);
-           rela.r_info = ELF64_R_INFO (h->dynindx, R_PPC64_JMP_SLOT);
-           rela.r_addend = ent->addend;
-
-           loc = htab->relplt->contents;
-           loc += ((ent->plt.offset - PLT_INITIAL_ENTRY_SIZE) / PLT_ENTRY_SIZE
-                   * sizeof (Elf64_External_Rela));
-           bfd_elf64_swap_reloca_out (output_bfd, &rela, loc);
-         }
-    }
+  for (ent = h->plt.plist; ent != NULL; ent = ent->next)
+    if (ent->plt.offset != (bfd_vma) -1)
+      {
+       /* This symbol has an entry in the procedure linkage
+          table.  Set it up.  */
+
+       if (htab->plt == NULL
+           || htab->relplt == NULL
+           || htab->glink == NULL)
+         abort ();
+
+       /* Create a JMP_SLOT reloc to inform the dynamic linker to
+          fill in the PLT entry.  */
+       rela.r_offset = (htab->plt->output_section->vma
+                        + htab->plt->output_offset
+                        + ent->plt.offset);
+       rela.r_info = ELF64_R_INFO (h->dynindx, R_PPC64_JMP_SLOT);
+       rela.r_addend = ent->addend;
+
+       loc = htab->relplt->contents;
+       loc += ((ent->plt.offset - PLT_INITIAL_ENTRY_SIZE) / PLT_ENTRY_SIZE
+               * sizeof (Elf64_External_Rela));
+       bfd_elf64_swap_reloca_out (output_bfd, &rela, loc);
+      }
 
   if ((h->elf_link_hash_flags & ELF_LINK_HASH_NEEDS_COPY) != 0)
     {
@@ -8793,7 +10013,7 @@ ppc64_elf_finish_dynamic_sections (bfd *output_bfd,
        abort ();
 
       dyncon = (Elf64_External_Dyn *) sdyn->contents;
-      dynconend = (Elf64_External_Dyn *) (sdyn->contents + sdyn->_raw_size);
+      dynconend = (Elf64_External_Dyn *) (sdyn->contents + sdyn->size);
       for (; dyncon < dynconend; dyncon++)
        {
          Elf_Internal_Dyn dyn;
@@ -8827,7 +10047,7 @@ ppc64_elf_finish_dynamic_sections (bfd *output_bfd,
              s = bfd_get_section_by_name (output_bfd, ".opd");
              if (s == NULL)
                continue;
-             dyn.d_un.d_val = s->_raw_size;
+             dyn.d_un.d_val = s->size;
              break;
 
            case DT_PLTGOT:
@@ -8841,7 +10061,7 @@ ppc64_elf_finish_dynamic_sections (bfd *output_bfd,
              break;
 
            case DT_PLTRELSZ:
-             dyn.d_un.d_val = htab->relplt->_raw_size;
+             dyn.d_un.d_val = htab->relplt->size;
              break;
 
            case DT_RELASZ:
@@ -8850,7 +10070,7 @@ ppc64_elf_finish_dynamic_sections (bfd *output_bfd,
              s = htab->relplt;
              if (s == NULL)
                continue;
-             dyn.d_un.d_val -= s->_raw_size;
+             dyn.d_un.d_val -= s->size;
              break;
 
            case DT_RELA:
@@ -8862,7 +10082,7 @@ ppc64_elf_finish_dynamic_sections (bfd *output_bfd,
                continue;
              if (dyn.d_un.d_ptr != s->output_section->vma + s->output_offset)
                continue;
-             dyn.d_un.d_ptr += s->_raw_size;
+             dyn.d_un.d_ptr += s->size;
              break;
            }
 
@@ -8870,7 +10090,7 @@ ppc64_elf_finish_dynamic_sections (bfd *output_bfd,
        }
     }
 
-  if (htab->got != NULL && htab->got->_raw_size != 0)
+  if (htab->got != NULL && htab->got->size != 0)
     {
       /* Fill in the first entry in the global offset table.
         We use it to hold the link-time TOCbase.  */
@@ -8882,7 +10102,7 @@ ppc64_elf_finish_dynamic_sections (bfd *output_bfd,
       elf_section_data (htab->got->output_section)->this_hdr.sh_entsize = 8;
     }
 
-  if (htab->plt != NULL && htab->plt->_raw_size != 0)
+  if (htab->plt != NULL && htab->plt->size != 0)
     {
       /* Set .plt entry size.  */
       elf_section_data (htab->plt->output_section)->this_hdr.sh_entsize
@@ -8890,25 +10110,30 @@ ppc64_elf_finish_dynamic_sections (bfd *output_bfd,
     }
 
   /* We need to handle writing out multiple GOT sections ourselves,
-     since we didn't add them to DYNOBJ.  */
+     since we didn't add them to DYNOBJ.  We know dynobj is the first
+     bfd.  */
   while ((dynobj = dynobj->link_next) != NULL)
     {
       asection *s;
+
+      if (!is_ppc64_target (dynobj->xvec))
+       continue;
+
       s = ppc64_elf_tdata (dynobj)->got;
       if (s != NULL
-         && s->_raw_size != 0
+         && s->size != 0
          && s->output_section != bfd_abs_section_ptr
          && !bfd_set_section_contents (output_bfd, s->output_section,
                                        s->contents, s->output_offset,
-                                       s->_raw_size))
+                                       s->size))
        return FALSE;
       s = ppc64_elf_tdata (dynobj)->relgot;
       if (s != NULL
-         && s->_raw_size != 0
+         && s->size != 0
          && s->output_section != bfd_abs_section_ptr
          && !bfd_set_section_contents (output_bfd, s->output_section,
                                        s->contents, s->output_offset,
-                                       s->_raw_size))
+                                       s->size))
        return FALSE;
     }
 
This page took 0.107754 seconds and 4 git commands to generate.