aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--bfd/ChangeLog3
-rw-r--r--bfd/elf64-ppc.c46
2 files changed, 22 insertions, 27 deletions
diff --git a/bfd/ChangeLog b/bfd/ChangeLog
index 288c815..ecfbeb0 100644
--- a/bfd/ChangeLog
+++ b/bfd/ChangeLog
@@ -1,5 +1,8 @@
2003-06-17 Alan Modra <amodra@bigpond.net.au>
+ * elf64-ppc.c (ppc64_elf_relocate_section): Do unaligned reloc
+ optimizations earlier.
+
* elf64-ppc.c (struct ppc_link_hash_table): Reinstate top_index.
Restore previous input_list type.
(ppc64_elf_link_hash_table_create): Undo last change.
diff --git a/bfd/elf64-ppc.c b/bfd/elf64-ppc.c
index 97fd482..6ac58cc 100644
--- a/bfd/elf64-ppc.c
+++ b/bfd/elf64-ppc.c
@@ -8212,6 +8212,7 @@ ppc64_elf_relocate_section (output_bfd, info, input_bfd, input_section,
bfd_boolean skip, relocate;
asection *sreloc;
bfd_byte *loc;
+ bfd_vma out_off;
/* When generating a dynamic object, these relocations
are copied into the output file to be resolved at run
@@ -8220,17 +8221,28 @@ ppc64_elf_relocate_section (output_bfd, info, input_bfd, input_section,
skip = FALSE;
relocate = FALSE;
- outrel.r_offset =
- _bfd_elf_section_offset (output_bfd, info, input_section,
- rel->r_offset);
- if (outrel.r_offset == (bfd_vma) -1)
+ out_off = _bfd_elf_section_offset (output_bfd, info,
+ input_section, rel->r_offset);
+ if (out_off == (bfd_vma) -1)
skip = TRUE;
- else if (outrel.r_offset == (bfd_vma) -2)
+ else if (out_off == (bfd_vma) -2)
skip = TRUE, relocate = TRUE;
- outrel.r_offset += (input_section->output_section->vma
- + input_section->output_offset);
+ out_off += (input_section->output_section->vma
+ + input_section->output_offset);
+ outrel.r_offset = out_off;
outrel.r_addend = rel->r_addend;
+ /* Optimize unaligned reloc use. */
+ if ((r_type == R_PPC64_ADDR64 && (out_off & 7) != 0)
+ || (r_type == R_PPC64_UADDR64 && (out_off & 7) == 0))
+ r_type ^= R_PPC64_ADDR64 ^ R_PPC64_UADDR64;
+ else if ((r_type == R_PPC64_ADDR32 && (out_off & 3) != 0)
+ || (r_type == R_PPC64_UADDR32 && (out_off & 3) == 0))
+ r_type ^= R_PPC64_ADDR32 ^ R_PPC64_UADDR32;
+ else if ((r_type == R_PPC64_ADDR16 && (out_off & 1) != 0)
+ || (r_type == R_PPC64_UADDR16 && (out_off & 1) == 0))
+ r_type ^= R_PPC64_ADDR16 ^ R_PPC64_UADDR16;
+
if (skip)
memset (&outrel, 0, sizeof outrel);
else if (h != NULL
@@ -8297,26 +8309,6 @@ ppc64_elf_relocate_section (output_bfd, info, input_bfd, input_section,
}
}
- /* Optimize unaligned reloc use. */
- if ((ELF64_R_TYPE (outrel.r_info) == R_PPC64_ADDR64
- && (outrel.r_offset & 7) != 0)
- || (ELF64_R_TYPE (outrel.r_info) == R_PPC64_UADDR64
- && (outrel.r_offset & 7) == 0))
- outrel.r_info ^= (ELF64_R_INFO (0, R_PPC64_ADDR64)
- ^ ELF64_R_INFO (0, R_PPC64_UADDR64));
- else if ((ELF64_R_TYPE (outrel.r_info) == R_PPC64_ADDR32
- && (outrel.r_offset & 3) != 0)
- || (ELF64_R_TYPE (outrel.r_info) == R_PPC64_UADDR32
- && (outrel.r_offset & 3) == 0))
- outrel.r_info ^= (ELF64_R_INFO (0, R_PPC64_ADDR32)
- ^ ELF64_R_INFO (0, R_PPC64_UADDR32));
- else if ((ELF64_R_TYPE (outrel.r_info) == R_PPC64_ADDR16
- && (outrel.r_offset & 1) != 0)
- || (ELF64_R_TYPE (outrel.r_info) == R_PPC64_UADDR16
- && (outrel.r_offset & 1) == 0))
- outrel.r_info ^= (ELF64_R_INFO (0, R_PPC64_ADDR16)
- ^ ELF64_R_INFO (0, R_PPC64_UADDR16));
-
sreloc = elf_section_data (input_section)->sreloc;
if (sreloc == NULL)
abort ();