bfd/
[deliverable/binutils-gdb.git] / bfd / elf64-ppc.c
index 7e7e5ba7bfddad58e930c417b127833a11926d64..fca84ac87c50b7684a1eb9055d77ea1593f53faf 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>
@@ -81,6 +82,9 @@ static bfd_reloc_status_type ppc64_elf_unhandled_reloc
 #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 +96,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
@@ -2379,7 +2384,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 +2397,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 +2432,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 +2494,16 @@ 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
 /* The following functions are specific to the ELF linker, while
    functions above are used generally.  Those named ppc64_elf_* are
@@ -2556,7 +2571,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 +2660,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 +2685,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 +2777,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 +2839,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 +2862,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 +2996,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;
     }
 
@@ -3178,9 +3205,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 +3381,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,23 +3472,175 @@ 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_entry *fdh = 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;
+         fh->is_func = 1;
+         fh->oh = fdh;
+       }
+    }
+
+  return fdh;
+}
+
+/* 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.
+
+   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".
+
+   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.  */
+
+/* Fix function descriptor symbols defined in .opd sections to be
+   function 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.  */
+
+static bfd_boolean
+add_symbol_adjust (struct elf_link_hash_entry *h, void *inf)
+{
+  struct bfd_link_info *info;
+  struct ppc_link_hash_table *htab;
+  struct ppc_link_hash_entry *eh;
+  struct ppc_link_hash_entry *fdh;
+
+  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_undefined
+      || 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)
+    {
+      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;
-  struct bfd_sym_chain *sym;
 
   htab = ppc_hash_table (info);
-  for (sym = info->gc_sym_list; sym; sym = sym->next)
+  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 elf_link_hash_entry *h;
+      struct bfd_link_hash_entry **pun;
 
-      h = elf_link_hash_lookup (&htab->elf, sym->name, FALSE, FALSE, FALSE);
-      if (h != NULL)
-       ((struct ppc_link_hash_entry *) h)->is_entry = 1;
+      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;
 }
@@ -3541,31 +3718,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 +3737,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 +3773,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) / 24;
       opd_sym_map = bfd_zalloc (abfd, amt);
       if (opd_sym_map == NULL)
        return FALSE;
@@ -3776,14 +3937,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 +3955,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 +4024,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 +4059,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 / 24] = s;
+               }
            }
          /* Fall through.  */
 
@@ -3942,10 +4115,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 +4174,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 +4247,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 +4411,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 +4453,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 / 24];
+       }
     }
 
   return rsec;
@@ -4161,6 +4479,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);
@@ -4312,10 +4633,6 @@ func_desc_adjust (struct elf_link_hash_entry *h, void *inf)
   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 +4690,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 +4699,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
@@ -4477,21 +4798,16 @@ ppc64_elf_func_desc_adjust (bfd *obfd ATTRIBUTE_UNUSED,
 
   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);
+  htab->sfpr->size = ((MAX_SAVE_FPR + 2 - lowest_savef) * 4
+                     + (MAX_SAVE_FPR + 2 - lowest_restf) * 4);
 
-  if (htab->sfpr->_raw_size == 0)
+  if (htab->sfpr->size == 0)
     {
-      if (!htab->have_undefweak)
-       {
-         _bfd_strip_section_from_output (info, htab->sfpr);
-         return TRUE;
-       }
-
-      htab->sfpr->_raw_size = 4;
+      _bfd_strip_section_from_output (info, htab->sfpr);
+      return TRUE;
     }
 
-  p = bfd_alloc (htab->elf.dynobj, htab->sfpr->_raw_size);
+  p = bfd_alloc (htab->elf.dynobj, htab->sfpr->size);
   if (p == NULL)
     return FALSE;
   htab->sfpr->contents = p;
@@ -4516,11 +4832,8 @@ ppc64_elf_func_desc_adjust (bfd *obfd ATTRIBUTE_UNUSED,
       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 (lowest_restf <= MAX_SAVE_FPR)
+    bfd_put_32 (htab->elf.dynobj, BLR, p);
 
   return TRUE;
 }
@@ -4551,8 +4864,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 +4958,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 +4970,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 +4979,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 +4994,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 +5014,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 +5031,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 +5195,59 @@ 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 / 24];
+      if (adjust == -1)
+       {
+         /* This entry has been deleted.  */
+         eh->elf.root.u.def.value = 0;
+         eh->elf.root.u.def.section = &bfd_abs_section;
+       }
+      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)
 {
   bfd *ibfd;
+  bfd_boolean some_edited = FALSE;
 
   for (ibfd = info->input_bfds; ibfd != NULL; ibfd = ibfd->link_next)
     {
@@ -4889,23 +5258,23 @@ 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;
+      long *opd_adjust;
       bfd_boolean need_edit;
 
       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) / 24;
+      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;
@@ -4950,8 +5319,7 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
                 something silly in .opd with the assembler.  No .opd
                 optimization for them!  */
              (*_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 +5328,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 +5348,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;
            }
@@ -5018,11 +5385,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)
@@ -5057,7 +5424,8 @@ 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)
+                 if (h != NULL
+                     && h->root.root.string[0] == '.')
                    fdh = get_fdh ((struct ppc_link_hash_entry *) h,
                                   ppc_hash_table (info));
 
@@ -5065,36 +5433,39 @@ 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 / 24] = -1;
                    }
                  else
                    {
                      /* We'll be keeping this opd entry.  */
 
-                     if (h != NULL)
+                     if (fdh != NULL)
                        {
-                         /* Redefine the function descriptor symbol
-                            to this location in the opd section.
-                            We've checked above that opd relocs are
-                            ordered.  */
+                         /* 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 - sec->contents;
+                         fdh->adjust_done = 1;
                        }
-                     else
-                       {
-                         /* 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;
-                       }
+
+                     /* 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 / 24] = wptr - rptr;
 
                      if (wptr != rptr)
                        memcpy (wptr, rptr, 24);
@@ -5146,13 +5517,14 @@ ppc64_elf_edit_opd (bfd *obfd, struct bfd_link_info *info)
                }
            }
 
-         sec->_cooked_size = wptr - sec->contents;
+         sec->size = wptr - sec->contents;
          sec->reloc_count = write_rel - relstart;
          /* 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 +5539,9 @@ 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);
+
   return TRUE;
 }
 
@@ -5180,13 +5555,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 +5653,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 +5727,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 +5878,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 +5909,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 +5985,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 +5997,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 +6066,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 +6085,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 +6145,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;
        }
     }
@@ -5786,12 +6167,12 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
       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 +6198,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 +6226,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 +6267,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 +6279,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 +6307,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,7 +6324,7 @@ 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;
     }
@@ -5957,12 +6334,11 @@ ppc64_elf_size_dynamic_sections (bfd *output_bfd ATTRIBUTE_UNUSED,
       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 +6346,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 +6367,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 +6375,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,14 +6433,18 @@ 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;
              }
        }
@@ -6143,7 +6522,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;
@@ -6271,17 +6650,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.  */
@@ -6320,13 +6700,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,
@@ -6395,7 +6776,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);
 
@@ -6430,11 +6811,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;
@@ -6444,7 +6825,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;
 }
 
@@ -6521,20 +6902,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.  */
@@ -6567,7 +6943,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
@@ -6577,13 +6953,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)
@@ -6593,7 +6966,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?  */
@@ -6683,10 +7056,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;
 
@@ -6842,14 +7212,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);
@@ -6869,32 +7242,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)
                        ;
@@ -6907,6 +7292,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 / 24];
+                         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);
@@ -6919,11 +7342,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;
                    }
@@ -6934,7 +7357,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)
                    {
@@ -6982,7 +7406,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;
@@ -7012,17 +7436,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);
 
@@ -7109,22 +7527,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)
@@ -7198,7 +7615,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)
            {
@@ -7217,20 +7634,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;
     }
@@ -7244,12 +7661,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"));
@@ -7280,6 +7697,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.
 
@@ -7392,26 +7837,29 @@ 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) / 24];
+             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);
+         RELOC_FOR_GLOBAL_SYMBOL (info, input_bfd, input_section, rel,
+                                  r_symndx, symtab_hdr, sym_hashes,
+                                  h, sec, relocation,
+                                  unresolved_reloc, warned);
          sym_name = h->root.root.string;
          sym_type = h->type;
        }
@@ -7462,10 +7910,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);
@@ -7670,7 +8118,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.  */
@@ -7774,6 +8223,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:
@@ -7790,31 +8241,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
@@ -7823,11 +8254,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 = &((struct ppc_link_hash_entry *) h)->oh->elf) != NULL
+                    && fdh->plt.plist != NULL)
+                   || (fdh = h)->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,
@@ -7836,17 +8271,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;
                    }
                }
 
@@ -7856,16 +8292,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,
                                      ".__libc_start_main") == 0)
                    {
                      /* Allow crt1 branch to go via a toc adjusting stub.  */
-                     can_plt_call = 1;
+                     can_plt_call = TRUE;
                    }
                  else
                    {
@@ -7874,21 +8311,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);
@@ -7897,51 +8334,95 @@ 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->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;
@@ -8480,8 +8961,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);
@@ -8559,8 +9040,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);
@@ -8568,40 +9049,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
@@ -8612,9 +9059,9 @@ ppc64_elf_relocate_section (bfd *output_bfd,
               && (h->elf_link_hash_flags & ELF_LINK_HASH_DEF_DYNAMIC) != 0))
        {
          (*_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);
@@ -8658,9 +9105,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,
@@ -8673,6 +9120,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 / 24];
+  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.  */
 
@@ -8684,41 +9162,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)
     {
@@ -8794,7 +9268,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;
@@ -8828,7 +9302,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:
@@ -8842,7 +9316,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:
@@ -8851,7 +9325,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:
@@ -8863,7 +9337,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;
            }
 
@@ -8871,7 +9345,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.  */
@@ -8883,7 +9357,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
@@ -8897,19 +9371,19 @@ ppc64_elf_finish_dynamic_sections (bfd *output_bfd,
       asection *s;
       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.064496 seconds and 4 git commands to generate.