From 7afd84dc1f020ccc7e867f41ef49656314ec4338 Mon Sep 17 00:00:00 2001 From: "H.J. Lu" Date: Tue, 2 Jun 2009 17:31:42 +0000 Subject: [PATCH] bfd/ 2009-06-02 H.J. Lu * elf32-i386.c (elf_i386_check_relocs): Increment got.refcount for R_386_GOT32/R_386_GOTOFF relocations against STT_GNU_IFUNC symbol. (elf_i386_allocate_dynrelocs): Set got.refcount to 0 if local STT_GNU_IFUNC definition is used. (elf_i386_relocate_section): Handle got.offset != -1 for R_386_GOT32/R_386_GOTOFF relocations against STT_GNU_IFUNC symbol. * elf64-x86-64.c (elf64_x86_64_check_relocs): Increment got.refcount for R_X86_64_GOTPCREL/R_X86_64_GOTPCREL64 relocations against STT_GNU_IFUNC symbol. (elf64_x86_64_allocate_dynrelocs): Set got.refcount to 0 if local STT_GNU_IFUNC definition is used. (elf64_x86_64_relocate_section): Handle got.offset != -1 for R_X86_64_GOTPCREL/R_X86_64_GOTPCREL64 relocations against STT_GNU_IFUNC symbol. ld/testsuite/ 2009-06-02 H.J. Lu * ld-ifunc/ifunc-5-i386.d: Renamed to ... * ld-ifunc/ifunc-5a-i386.d: This. * ld-ifunc/ifunc-5-x86-64.d: Renamed to ... * ld-ifunc/ifunc-5a-x86-64.d: This. * ld-ifunc/ifunc-5b-i386.d: New. * ld-ifunc/ifunc-5b-x86-64.d: Likewise. * ld-ifunc/ifunc-6a-i386.d: Likewise. * ld-ifunc/ifunc-6a-x86-64.d: Likewise. * ld-ifunc/ifunc-6b-i386.d: Likewise. * ld-ifunc/ifunc-6b-x86-64.d: Likewise. * ld-ifunc/ifunc-6-i386.s: Likewise. * ld-ifunc/ifunc-6-x86-64.s: Likewise. * ld-ifunc/ifunc-7a-i386.d: Likewise. * ld-ifunc/ifunc-7a-x86-64.d: Likewise. * ld-ifunc/ifunc-7b-i386.d: Likewise. * ld-ifunc/ifunc-7b-x86-64.d: Likewise. * ld-ifunc/ifunc-7-i386.s: Likewise. * ld-ifunc/ifunc-7-x86-64.s: Likewise. --- bfd/ChangeLog | 20 ++++ bfd/elf32-i386.c | 99 +++++++++++-------- bfd/elf64-x86-64.c | 86 ++++++++-------- ld/testsuite/ChangeLog | 23 +++++ .../{ifunc-5-i386.d => ifunc-5a-i386.d} | 5 +- .../{ifunc-5-x86-64.d => ifunc-5a-x86-64.d} | 5 +- ld/testsuite/ld-ifunc/ifunc-5b-i386.d | 13 +++ ld/testsuite/ld-ifunc/ifunc-5b-x86-64.d | 12 +++ ld/testsuite/ld-ifunc/ifunc-6-i386.s | 24 +++++ ld/testsuite/ld-ifunc/ifunc-6-x86-64.s | 20 ++++ ld/testsuite/ld-ifunc/ifunc-6a-i386.d | 9 ++ ld/testsuite/ld-ifunc/ifunc-6a-x86-64.d | 8 ++ ld/testsuite/ld-ifunc/ifunc-6b-i386.d | 13 +++ ld/testsuite/ld-ifunc/ifunc-6b-x86-64.d | 12 +++ ld/testsuite/ld-ifunc/ifunc-7-i386.s | 24 +++++ ld/testsuite/ld-ifunc/ifunc-7-x86-64.s | 20 ++++ ld/testsuite/ld-ifunc/ifunc-7a-i386.d | 9 ++ ld/testsuite/ld-ifunc/ifunc-7a-x86-64.d | 8 ++ ld/testsuite/ld-ifunc/ifunc-7b-i386.d | 9 ++ ld/testsuite/ld-ifunc/ifunc-7b-x86-64.d | 8 ++ 20 files changed, 341 insertions(+), 86 deletions(-) rename ld/testsuite/ld-ifunc/{ifunc-5-i386.d => ifunc-5a-i386.d} (56%) rename ld/testsuite/ld-ifunc/{ifunc-5-x86-64.d => ifunc-5a-x86-64.d} (52%) create mode 100644 ld/testsuite/ld-ifunc/ifunc-5b-i386.d create mode 100644 ld/testsuite/ld-ifunc/ifunc-5b-x86-64.d create mode 100644 ld/testsuite/ld-ifunc/ifunc-6-i386.s create mode 100644 ld/testsuite/ld-ifunc/ifunc-6-x86-64.s create mode 100644 ld/testsuite/ld-ifunc/ifunc-6a-i386.d create mode 100644 ld/testsuite/ld-ifunc/ifunc-6a-x86-64.d create mode 100644 ld/testsuite/ld-ifunc/ifunc-6b-i386.d create mode 100644 ld/testsuite/ld-ifunc/ifunc-6b-x86-64.d create mode 100644 ld/testsuite/ld-ifunc/ifunc-7-i386.s create mode 100644 ld/testsuite/ld-ifunc/ifunc-7-x86-64.s create mode 100644 ld/testsuite/ld-ifunc/ifunc-7a-i386.d create mode 100644 ld/testsuite/ld-ifunc/ifunc-7a-x86-64.d create mode 100644 ld/testsuite/ld-ifunc/ifunc-7b-i386.d create mode 100644 ld/testsuite/ld-ifunc/ifunc-7b-x86-64.d diff --git a/bfd/ChangeLog b/bfd/ChangeLog index b7cf0effe02..77f080c9278 100644 --- a/bfd/ChangeLog +++ b/bfd/ChangeLog @@ -1,3 +1,23 @@ +2009-06-02 H.J. Lu + + * elf32-i386.c (elf_i386_check_relocs): Increment + got.refcount for R_386_GOT32/R_386_GOTOFF relocations + against STT_GNU_IFUNC symbol. + (elf_i386_allocate_dynrelocs): Set got.refcount to 0 if + local STT_GNU_IFUNC definition is used. + (elf_i386_relocate_section): Handle got.offset != -1 for + R_386_GOT32/R_386_GOTOFF relocations against STT_GNU_IFUNC + symbol. + + * elf64-x86-64.c (elf64_x86_64_check_relocs): Increment + got.refcount for R_X86_64_GOTPCREL/R_X86_64_GOTPCREL64 + relocations against STT_GNU_IFUNC symbol. + (elf64_x86_64_allocate_dynrelocs): Set got.refcount to 0 if + local STT_GNU_IFUNC definition is used. + (elf64_x86_64_relocate_section): Handle got.offset != -1 + for R_X86_64_GOTPCREL/R_X86_64_GOTPCREL64 relocations against + STT_GNU_IFUNC symbol. + 2009-06-01 H.J. Lu PR ld/10205 diff --git a/bfd/elf32-i386.c b/bfd/elf32-i386.c index 70c579e76fc..ade5aa8ccee 100644 --- a/bfd/elf32-i386.c +++ b/bfd/elf32-i386.c @@ -1336,6 +1336,7 @@ elf_i386_check_relocs (bfd *abfd, case R_386_GOT32: case R_386_GOTOFF: + h->got.refcount += 1; if (htab->sgot == NULL && !elf_i386_create_got_section (htab->elf.dynobj, info)) @@ -1999,8 +2000,13 @@ elf_i386_allocate_dynrelocs (struct elf_link_hash_entry *h, void *inf) if (h->dynindx == -1 || h->forced_local) eh->dyn_relocs = NULL; - /* STT_GNU_IFUNC symbol uses .got.plt, not .got. */ - h->got.refcount = 0; + /* STT_GNU_IFUNC symbol uses .got.plt, not .got. But for + shared library, we must go through GOT and we can't + use R_386_IRELATIVE unless it is forced local. */ + if (!info->shared + || info->symbolic + || h->forced_local) + h->got.refcount = 0; } else if (htab->elf.dynamic_sections_created && h->plt.refcount > 0) @@ -2908,55 +2914,62 @@ elf_i386_relocate_section (bfd *output_bfd, base_got = htab->sgot; off = h->got.offset; - if (base_got == NULL - || off != (bfd_vma) -1) + if (base_got == NULL) abort (); - /* We can't use h->got.offset here to save state, or - even just remember the offset, as finish_dynamic_symbol - would use that as offset into .got. */ - - if (htab->splt != NULL) - { - plt_index = h->plt.offset / PLT_ENTRY_SIZE - 1; - off = (plt_index + 3) * 4; - base_got = htab->sgotplt; - } - else + if (off == (bfd_vma) -1) { - plt_index = h->plt.offset / PLT_ENTRY_SIZE; - off = plt_index * 4; - base_got = htab->igotplt; - } - - if (h->dynindx == -1 - || h->forced_local - || info->symbolic) - { - /* This references the local defitionion. We must - initialize this entry in the global offset table. - Since the offset must always be a multiple of 8, we - use the least significant bit to record whether we - have initialized it already. - - When doing a dynamic link, we create a .rela.got - relocation entry to initialize the value. This is - done in the finish_dynamic_symbol routine. */ - if ((off & 1) != 0) - off &= ~1; + /* We can't use h->got.offset here to save state, or + even just remember the offset, as finish_dynamic_symbol + would use that as offset into .got. */ + + if (htab->splt != NULL) + { + plt_index = h->plt.offset / PLT_ENTRY_SIZE - 1; + off = (plt_index + 3) * 4; + base_got = htab->sgotplt; + } else { - bfd_put_32 (output_bfd, relocation, - base_got->contents + off); - h->got.offset |= 1; + plt_index = h->plt.offset / PLT_ENTRY_SIZE; + off = plt_index * 4; + base_got = htab->igotplt; } - } - relocation = off; + if (h->dynindx == -1 + || h->forced_local + || info->symbolic) + { + /* This references the local defitionion. We must + initialize this entry in the global offset table. + Since the offset must always be a multiple of 8, + we use the least significant bit to record + whether we have initialized it already. + + When doing a dynamic link, we create a .rela.got + relocation entry to initialize the value. This + is done in the finish_dynamic_symbol routine. */ + if ((off & 1) != 0) + off &= ~1; + else + { + bfd_put_32 (output_bfd, relocation, + base_got->contents + off); + h->got.offset |= 1; + } + } + + relocation = off; - /* Adjust for static executables. */ - if (htab->splt == NULL) - relocation += gotplt->output_offset; + /* Adjust for static executables. */ + if (htab->splt == NULL) + relocation += gotplt->output_offset; + } + else + relocation = (base_got->output_section->vma + + base_got->output_offset + off + - gotplt->output_section->vma + - gotplt->output_offset); goto do_relocation; diff --git a/bfd/elf64-x86-64.c b/bfd/elf64-x86-64.c index 86715865b6a..e85832519ea 100644 --- a/bfd/elf64-x86-64.c +++ b/bfd/elf64-x86-64.c @@ -1126,6 +1126,7 @@ elf64_x86_64_check_relocs (bfd *abfd, struct bfd_link_info *info, case R_X86_64_GOTPCREL: case R_X86_64_GOTPCREL64: + h->got.refcount += 1; if (htab->sgot == NULL && !elf64_x86_64_create_got_section (htab->elf.dynobj, info)) @@ -1837,8 +1838,13 @@ elf64_x86_64_allocate_dynrelocs (struct elf_link_hash_entry *h, void * inf) if (h->dynindx == -1 || h->forced_local) eh->dyn_relocs = NULL; - /* STT_GNU_IFUNC symbol uses .got.plt, not .got. */ - h->got.refcount = 0; + /* STT_GNU_IFUNC symbol uses .got.plt, not .got. But for + shared library, we must go through GOT and we can't + use R_X86_64_IRELATIVE unless it is forced local. */ + if (!info->shared + || info->symbolic + || h->forced_local) + h->got.refcount = 0; } else if (htab->elf.dynamic_sections_created && h->plt.refcount > 0) @@ -2616,49 +2622,51 @@ elf64_x86_64_relocate_section (bfd *output_bfd, struct bfd_link_info *info, base_got = htab->sgot; off = h->got.offset; - if (base_got == NULL - || off != (bfd_vma) -1) + if (base_got == NULL) abort (); - /* We can't use h->got.offset here to save state, or - even just remember the offset, as finish_dynamic_symbol - would use that as offset into .got. */ - - if (htab->splt != NULL) - { - plt_index = h->plt.offset / PLT_ENTRY_SIZE - 1; - off = (plt_index + 3) * GOT_ENTRY_SIZE; - base_got = htab->sgotplt; - } - else - { - plt_index = h->plt.offset / PLT_ENTRY_SIZE; - off = plt_index * GOT_ENTRY_SIZE; - base_got = htab->igotplt; - } - - if (h->dynindx == -1 - || h->forced_local - || info->symbolic) + if (off == (bfd_vma) -1) { - /* This references the local defitionion. We must - initialize this entry in the global offset table. - Since the offset must always be a multiple of 8, we - use the least significant bit to record whether we - have initialized it already. + /* We can't use h->got.offset here to save state, or + even just remember the offset, as finish_dynamic_symbol + would use that as offset into .got. */ - When doing a dynamic link, we create a .rela.got - relocation entry to initialize the value. This is - done in the finish_dynamic_symbol routine. */ - if ((off & 1) != 0) - off &= ~1; + if (htab->splt != NULL) + { + plt_index = h->plt.offset / PLT_ENTRY_SIZE - 1; + off = (plt_index + 3) * GOT_ENTRY_SIZE; + base_got = htab->sgotplt; + } else { - bfd_put_64 (output_bfd, relocation, - base_got->contents + off); - /* Note that this is harmless for the GOTPLT64 case, - as -1 | 1 still is -1. */ - h->got.offset |= 1; + plt_index = h->plt.offset / PLT_ENTRY_SIZE; + off = plt_index * GOT_ENTRY_SIZE; + base_got = htab->igotplt; + } + + if (h->dynindx == -1 + || h->forced_local + || info->symbolic) + { + /* This references the local defitionion. We must + initialize this entry in the global offset table. + Since the offset must always be a multiple of 8, + we use the least significant bit to record + whether we have initialized it already. + + When doing a dynamic link, we create a .rela.got + relocation entry to initialize the value. This + is done in the finish_dynamic_symbol routine. */ + if ((off & 1) != 0) + off &= ~1; + else + { + bfd_put_64 (output_bfd, relocation, + base_got->contents + off); + /* Note that this is harmless for the GOTPLT64 + case, as -1 | 1 still is -1. */ + h->got.offset |= 1; + } } } diff --git a/ld/testsuite/ChangeLog b/ld/testsuite/ChangeLog index 03fb645a0cd..6c73acfe214 100644 --- a/ld/testsuite/ChangeLog +++ b/ld/testsuite/ChangeLog @@ -1,3 +1,26 @@ +2009-06-02 H.J. Lu + + * ld-ifunc/ifunc-5-i386.d: Renamed to ... + * ld-ifunc/ifunc-5a-i386.d: This. + + * ld-ifunc/ifunc-5-x86-64.d: Renamed to ... + * ld-ifunc/ifunc-5a-x86-64.d: This. + + * ld-ifunc/ifunc-5b-i386.d: New. + * ld-ifunc/ifunc-5b-x86-64.d: Likewise. + * ld-ifunc/ifunc-6a-i386.d: Likewise. + * ld-ifunc/ifunc-6a-x86-64.d: Likewise. + * ld-ifunc/ifunc-6b-i386.d: Likewise. + * ld-ifunc/ifunc-6b-x86-64.d: Likewise. + * ld-ifunc/ifunc-6-i386.s: Likewise. + * ld-ifunc/ifunc-6-x86-64.s: Likewise. + * ld-ifunc/ifunc-7a-i386.d: Likewise. + * ld-ifunc/ifunc-7a-x86-64.d: Likewise. + * ld-ifunc/ifunc-7b-i386.d: Likewise. + * ld-ifunc/ifunc-7b-x86-64.d: Likewise. + * ld-ifunc/ifunc-7-i386.s: Likewise. + * ld-ifunc/ifunc-7-x86-64.s: Likewise. + 2009-06-01 H.J. Lu PR ld/10205 diff --git a/ld/testsuite/ld-ifunc/ifunc-5-i386.d b/ld/testsuite/ld-ifunc/ifunc-5a-i386.d similarity index 56% rename from ld/testsuite/ld-ifunc/ifunc-5-i386.d rename to ld/testsuite/ld-ifunc/ifunc-5a-i386.d index eb3fc01e7bc..8fba080218f 100644 --- a/ld/testsuite/ld-ifunc/ifunc-5-i386.d +++ b/ld/testsuite/ld-ifunc/ifunc-5a-i386.d @@ -1,8 +1,9 @@ +#source: ifunc-5-i386.s #ld: -m elf_i386 #as: --32 #readelf: -r --wide #target: x86_64-*-* i?86-*-* -#... +Relocation section '.rel.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* [0-9a-f]+[ ]+[0-9a-f]+[ ]+R_386_IRELATIVE[ ]* -#pass diff --git a/ld/testsuite/ld-ifunc/ifunc-5-x86-64.d b/ld/testsuite/ld-ifunc/ifunc-5a-x86-64.d similarity index 52% rename from ld/testsuite/ld-ifunc/ifunc-5-x86-64.d rename to ld/testsuite/ld-ifunc/ifunc-5a-x86-64.d index 84347cc4f9c..b635099b409 100644 --- a/ld/testsuite/ld-ifunc/ifunc-5-x86-64.d +++ b/ld/testsuite/ld-ifunc/ifunc-5a-x86-64.d @@ -1,7 +1,8 @@ +#source: ifunc-5-x86-64.s #ld: #readelf: -r --wide #target: x86_64-*-* -#... +Relocation section '.rela.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* [0-9a-f]+[ ]+[0-9a-f]+[ ]+R_X86_64_IRELATIVE[ ]+[0-9a-f]* -#pass diff --git a/ld/testsuite/ld-ifunc/ifunc-5b-i386.d b/ld/testsuite/ld-ifunc/ifunc-5b-i386.d new file mode 100644 index 00000000000..e00401bb9be --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-5b-i386.d @@ -0,0 +1,13 @@ +#source: ifunc-5-i386.s +#ld: -shared -m elf_i386 -z nocombreloc +#as: --32 +#readelf: -r --wide +#target: x86_64-*-* i?86-*-* + +Relocation section '.rel.got' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_386_GLOB_DAT[ ]+foo\(\)[ ]+foo +#... +Relocation section '.rel.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_386_JUMP_SLOT[ ]+foo\(\)[ ]+foo diff --git a/ld/testsuite/ld-ifunc/ifunc-5b-x86-64.d b/ld/testsuite/ld-ifunc/ifunc-5b-x86-64.d new file mode 100644 index 00000000000..d8b2e913a70 --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-5b-x86-64.d @@ -0,0 +1,12 @@ +#source: ifunc-5-x86-64.s +#ld: -shared -z nocombreloc +#readelf: -r --wide +#target: x86_64-*-* + +Relocation section '.rela.got' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_X86_64_GLOB_DAT[ ]+foo\(\)[ ]+foo \+ 0 +#... +Relocation section '.rela.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_X86_64_JUMP_SLOT[ ]+foo\(\)[ ]+foo \+ 0 diff --git a/ld/testsuite/ld-ifunc/ifunc-6-i386.s b/ld/testsuite/ld-ifunc/ifunc-6-i386.s new file mode 100644 index 00000000000..bc632c9a7ab --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-6-i386.s @@ -0,0 +1,24 @@ + .text + .type foo, %gnu_indirect_function +.globl foo + .type foo, @function +foo: + ret + .size foo, .-foo + .protected foo + .type start,"function" + .global start +start: + .type _start,"function" + .global _start +_start: + .type __start,"function" + .global __start +__start: + .type __start,"function" + call .L6 +.L6: + popl %ebx + addl $_GLOBAL_OFFSET_TABLE_+[.-.L6], %ebx + call foo@PLT + leal foo@GOT(%ebx), %eax diff --git a/ld/testsuite/ld-ifunc/ifunc-6-x86-64.s b/ld/testsuite/ld-ifunc/ifunc-6-x86-64.s new file mode 100644 index 00000000000..954d67d9023 --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-6-x86-64.s @@ -0,0 +1,20 @@ + .text + .type foo, %gnu_indirect_function +.globl foo + .type foo, @function +foo: + ret + .size foo, .-foo + .protected foo + .type start,"function" + .global start +start: + .type _start,"function" + .global _start +_start: + .type __start,"function" + .global __start +__start: + .type __start,"function" + call foo@PLT + movq foo@GOTPCREL(%rip), %rax diff --git a/ld/testsuite/ld-ifunc/ifunc-6a-i386.d b/ld/testsuite/ld-ifunc/ifunc-6a-i386.d new file mode 100644 index 00000000000..a83e62bd9ba --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-6a-i386.d @@ -0,0 +1,9 @@ +#source: ifunc-6-i386.s +#ld: -m elf_i386 +#as: --32 +#readelf: -r --wide +#target: x86_64-*-* i?86-*-* + +Relocation section '.rel.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_386_IRELATIVE[ ]* diff --git a/ld/testsuite/ld-ifunc/ifunc-6a-x86-64.d b/ld/testsuite/ld-ifunc/ifunc-6a-x86-64.d new file mode 100644 index 00000000000..169cf1c40a7 --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-6a-x86-64.d @@ -0,0 +1,8 @@ +#source: ifunc-6-x86-64.s +#ld: +#readelf: -r --wide +#target: x86_64-*-* + +Relocation section '.rela.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_X86_64_IRELATIVE[ ]+[0-9a-f]* diff --git a/ld/testsuite/ld-ifunc/ifunc-6b-i386.d b/ld/testsuite/ld-ifunc/ifunc-6b-i386.d new file mode 100644 index 00000000000..e53fc72ff31 --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-6b-i386.d @@ -0,0 +1,13 @@ +#source: ifunc-6-i386.s +#ld: -shared -m elf_i386 -z nocombreloc +#as: --32 +#readelf: -r --wide +#target: x86_64-*-* i?86-*-* + +Relocation section '.rel.got' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_386_GLOB_DAT[ ]+foo\(\)[ ]+foo +#... +Relocation section '.rel.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_386_IRELATIVE[ ]* diff --git a/ld/testsuite/ld-ifunc/ifunc-6b-x86-64.d b/ld/testsuite/ld-ifunc/ifunc-6b-x86-64.d new file mode 100644 index 00000000000..5a948e9ca7f --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-6b-x86-64.d @@ -0,0 +1,12 @@ +#source: ifunc-6-x86-64.s +#ld: -shared -z nocombreloc +#readelf: -r --wide +#target: x86_64-*-* + +Relocation section '.rela.got' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_X86_64_GLOB_DAT[ ]+foo\(\)[ ]+foo \+ 0 +#... +Relocation section '.rela.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_X86_64_IRELATIVE[ ]+[0-9a-f]* diff --git a/ld/testsuite/ld-ifunc/ifunc-7-i386.s b/ld/testsuite/ld-ifunc/ifunc-7-i386.s new file mode 100644 index 00000000000..8616c4e5d42 --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-7-i386.s @@ -0,0 +1,24 @@ + .text + .type foo, %gnu_indirect_function +.globl foo + .type foo, @function +foo: + ret + .size foo, .-foo + .hidden foo + .type start,"function" + .global start +start: + .type _start,"function" + .global _start +_start: + .type __start,"function" + .global __start +__start: + .type __start,"function" + call .L6 +.L6: + popl %ebx + addl $_GLOBAL_OFFSET_TABLE_+[.-.L6], %ebx + call foo@PLT + leal foo@GOT(%ebx), %eax diff --git a/ld/testsuite/ld-ifunc/ifunc-7-x86-64.s b/ld/testsuite/ld-ifunc/ifunc-7-x86-64.s new file mode 100644 index 00000000000..24fce11493c --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-7-x86-64.s @@ -0,0 +1,20 @@ + .text + .type foo, %gnu_indirect_function +.globl foo + .type foo, @function +foo: + ret + .size foo, .-foo + .hidden foo + .type start,"function" + .global start +start: + .type _start,"function" + .global _start +_start: + .type __start,"function" + .global __start +__start: + .type __start,"function" + call foo@PLT + movq foo@GOTPCREL(%rip), %rax diff --git a/ld/testsuite/ld-ifunc/ifunc-7a-i386.d b/ld/testsuite/ld-ifunc/ifunc-7a-i386.d new file mode 100644 index 00000000000..6b4afe156eb --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-7a-i386.d @@ -0,0 +1,9 @@ +#source: ifunc-7-i386.s +#ld: -m elf_i386 +#as: --32 +#readelf: -r --wide +#target: x86_64-*-* i?86-*-* + +Relocation section '.rel.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_386_IRELATIVE[ ]* diff --git a/ld/testsuite/ld-ifunc/ifunc-7a-x86-64.d b/ld/testsuite/ld-ifunc/ifunc-7a-x86-64.d new file mode 100644 index 00000000000..350614ab5ab --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-7a-x86-64.d @@ -0,0 +1,8 @@ +#source: ifunc-7-x86-64.s +#ld: +#readelf: -r --wide +#target: x86_64-*-* + +Relocation section '.rela.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_X86_64_IRELATIVE[ ]+[0-9a-f]* diff --git a/ld/testsuite/ld-ifunc/ifunc-7b-i386.d b/ld/testsuite/ld-ifunc/ifunc-7b-i386.d new file mode 100644 index 00000000000..31cdd974772 --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-7b-i386.d @@ -0,0 +1,9 @@ +#source: ifunc-7-i386.s +#ld: -shared -m elf_i386 +#as: --32 +#readelf: -r --wide +#target: x86_64-*-* i?86-*-* + +Relocation section '.rel.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_386_IRELATIVE[ ]* diff --git a/ld/testsuite/ld-ifunc/ifunc-7b-x86-64.d b/ld/testsuite/ld-ifunc/ifunc-7b-x86-64.d new file mode 100644 index 00000000000..cd66d58d2d5 --- /dev/null +++ b/ld/testsuite/ld-ifunc/ifunc-7b-x86-64.d @@ -0,0 +1,8 @@ +#source: ifunc-7-x86-64.s +#ld: -shared +#readelf: -r --wide +#target: x86_64-*-* + +Relocation section '.rela.plt' at .* +[ ]+Offset[ ]+Info[ ]+Type[ ]+.* +[0-9a-f]+[ ]+[0-9a-f]+[ ]+R_X86_64_IRELATIVE[ ]+[0-9a-f]* -- 2.39.2