2012-02-17 Andreas Krebbel <Andreas.Krebbel@de.ibm.com>

* elf32-s390.c (elf_s390_relocate_section): Support basr in the
	GD->LE and LD->LE optimizations.
This commit is contained in:
Andreas Krebbel 2012-02-17 08:38:06 +00:00
parent 980bb25303
commit d0ddfe2750
2 changed files with 31 additions and 6 deletions

View File

@ -1,3 +1,8 @@
2012-02-17 Andreas Krebbel <Andreas.Krebbel@de.ibm.com>
* elf32-s390.c (elf_s390_relocate_section): Support basr in the
GD->LE and LD->LE optimizations.
2012-02-16 Alan Modra <amodra@gmail.com> 2012-02-16 Alan Modra <amodra@gmail.com>
PR binutils/13355 PR binutils/13355

View File

@ -2919,11 +2919,18 @@ elf_s390_relocate_section (output_bfd, info, input_bfd, input_section,
insn = bfd_get_32 (input_bfd, contents + rel->r_offset); insn = bfd_get_32 (input_bfd, contents + rel->r_offset);
if ((insn & 0xff000fff) != 0x4d000000 && if ((insn & 0xff000fff) != 0x4d000000 &&
(insn & 0xffff0000) != 0xc0e50000) (insn & 0xffff0000) != 0xc0e50000 &&
(insn & 0xff000000) != 0x0d000000)
invalid_tls_insn (input_bfd, input_section, rel); invalid_tls_insn (input_bfd, input_section, rel);
if (!info->shared && (h == NULL || h->dynindx == -1)) if (!info->shared && (h == NULL || h->dynindx == -1))
{ {
if ((insn & 0xff000000) == 0x4d000000) if ((insn & 0xff000000) == 0x0d000000)
{
/* GD->LE transition.
basr rx, ry -> nopr r7 */
insn = 0x07070000 | (insn & 0xffff);
}
else if ((insn & 0xff000000) == 0x4d000000)
{ {
/* GD->LE transition. /* GD->LE transition.
bas %r14,0(%rx,%r13) -> bc 0,0 */ bas %r14,0(%rx,%r13) -> bc 0,0 */
@ -2932,7 +2939,7 @@ elf_s390_relocate_section (output_bfd, info, input_bfd, input_section,
else else
{ {
/* GD->LE transition. /* GD->LE transition.
brasl %r14,_tls_get_addr@plt -> brcl 0,. */ brasl %r14,_tls_get_offset@plt -> brcl 0,. */
insn = 0xc0040000; insn = 0xc0040000;
bfd_put_16 (output_bfd, 0x0000, bfd_put_16 (output_bfd, 0x0000,
contents + rel->r_offset + 4); contents + rel->r_offset + 4);
@ -2940,6 +2947,11 @@ elf_s390_relocate_section (output_bfd, info, input_bfd, input_section,
} }
else else
{ {
/* If basr is used in the pic case to invoke
_tls_get_offset, something went wrong before. */
if ((insn & 0xff000000) == 0x0d000000)
invalid_tls_insn (input_bfd, input_section, rel);
if ((insn & 0xff000000) == 0x4d000000) if ((insn & 0xff000000) == 0x4d000000)
{ {
/* GD->IE transition. /* GD->IE transition.
@ -2966,9 +2978,17 @@ elf_s390_relocate_section (output_bfd, info, input_bfd, input_section,
insn = bfd_get_32 (input_bfd, contents + rel->r_offset); insn = bfd_get_32 (input_bfd, contents + rel->r_offset);
if ((insn & 0xff000fff) != 0x4d000000 && if ((insn & 0xff000fff) != 0x4d000000 &&
(insn & 0xffff0000) != 0xc0e50000) (insn & 0xffff0000) != 0xc0e50000 &&
(insn & 0xff000000) != 0x0d000000)
invalid_tls_insn (input_bfd, input_section, rel); invalid_tls_insn (input_bfd, input_section, rel);
if ((insn & 0xff000000) == 0x4d000000)
if ((insn & 0xff000000) == 0x0d000000)
{
/* LD->LE transition.
basr rx, ry -> nopr r7 */
insn = 0x07070000 | (insn & 0xffff);
}
else if ((insn & 0xff000000) == 0x4d000000)
{ {
/* LD->LE transition. /* LD->LE transition.
bas %r14,0(%rx,%r13) -> bc 0,0 */ bas %r14,0(%rx,%r13) -> bc 0,0 */
@ -2977,7 +2997,7 @@ elf_s390_relocate_section (output_bfd, info, input_bfd, input_section,
else else
{ {
/* LD->LE transition. /* LD->LE transition.
brasl %r14,__tls_get_addr@plt -> brcl 0,. */ brasl %r14,__tls_get_offset@plt -> brcl 0,. */
insn = 0xc0040000; insn = 0xc0040000;
bfd_put_16 (output_bfd, 0x0000, bfd_put_16 (output_bfd, 0x0000,
contents + rel->r_offset + 4); contents + rel->r_offset + 4);