aboutsummaryrefslogtreecommitdiff
path: root/bfd/elf64-ppc.c
diff options
context:
space:
mode:
authorAlan Modra <amodra@gmail.com>2016-08-27 08:59:29 +0930
committerAlan Modra <amodra@gmail.com>2016-08-27 09:42:09 +0930
commit8a9e8e72fe88095043d16f8a56b5a1e150ee288b (patch)
treed7fff3fcb432159e06b8e51d44780716e36d254c /bfd/elf64-ppc.c
parente55c2fc0ef151054f4a1603799d8ecd6a3f407bd (diff)
downloadgdb-8a9e8e72fe88095043d16f8a56b5a1e150ee288b.zip
gdb-8a9e8e72fe88095043d16f8a56b5a1e150ee288b.tar.gz
gdb-8a9e8e72fe88095043d16f8a56b5a1e150ee288b.tar.bz2
Fix commit 980aa3e6
Commit 980aa3e6 was supposed to cure dyn_reloc counting problems, but did the opposite. For PIC we count two types of dyn_reloc, those on pc-relative relocs, and the total. If a sym needs pc-relative dyn relocs then all the relocs are dynamic. If not, then only those that are must_be_dyn_reloc are dynamic. PR 20519 * elf64-ppc.c (pc_dynrelocs): New function. (ppc64_elf_relocate_section): Use it and must_be_dyn_reloc to handle pic dynamic relocs.
Diffstat (limited to 'bfd/elf64-ppc.c')
-rw-r--r--bfd/elf64-ppc.c22
1 files changed, 18 insertions, 4 deletions
diff --git a/bfd/elf64-ppc.c b/bfd/elf64-ppc.c
index 286130c..a9cedb5 100644
--- a/bfd/elf64-ppc.c
+++ b/bfd/elf64-ppc.c
@@ -7173,6 +7173,19 @@ alias_readonly_dynrelocs (struct elf_link_hash_entry *h)
return FALSE;
}
+/* Return whether EH has pc-relative dynamic relocs. */
+
+static bfd_boolean
+pc_dynrelocs (struct ppc_link_hash_entry *eh)
+{
+ struct elf_dyn_relocs *p;
+
+ for (p = eh->dyn_relocs; p != NULL; p = p->next)
+ if (p->pc_count != 0)
+ return TRUE;
+ return FALSE;
+}
+
/* Return true if a global entry stub will be created for H. Valid
for ELFv2 before plt entries have been allocated. */
@@ -14745,10 +14758,11 @@ ppc64_elf_relocate_section (bfd *output_bfd,
if (NO_OPD_RELOCS && is_opd)
break;
- if (h != NULL
- ? h->dyn_relocs != NULL
- : (bfd_link_pic (info)
- ? must_be_dyn_reloc (info, r_type)
+ if (bfd_link_pic (info)
+ ? ((h != NULL && pc_dynrelocs (h))
+ || must_be_dyn_reloc (info, r_type))
+ : (h != NULL
+ ? h->dyn_relocs != NULL
: ELF_ST_TYPE (sym->st_info) == STT_GNU_IFUNC))
{
bfd_boolean skip, relocate;