Correct a MIPS/BFD linker issue with dynamic symbol and corresponding
GOT entry values being redirected to lazy binding stubs where the stubs
section has been discarded by assigning to the `/DISCARD/' output
section in the linker script used. The issue manifests itself by the
values entered being relative to the absolute section, which is what any
discarded sections are internally assigned in the linker.
For the `stub-dynsym-2.s' piece of code included as a test case with
this change this issue results in the dynamic symbol table and the GOT
looking like:
Symbol table '.dynsym' contains 3 entries:
Num: Value Size Type Bind Vis Ndx Name
0:
00000000 0 NOTYPE LOCAL DEFAULT UND
1:
00000010 0 FUNC GLOBAL DEFAULT UND bar
2:
00000000 0 FUNC GLOBAL DEFAULT UND foo
Primary GOT:
Canonical gp value:
00097ff0
Reserved entries:
Address Access Initial Purpose
00090000 -32752(gp)
00000000 Lazy resolver
00090004 -32748(gp)
80000000 Module pointer (GNU extension)
Global entries:
Address Access Initial Sym.Val. Type Ndx Name
00090008 -32744(gp)
00000010 00000010 FUNC UND bar
0009000c -32740(gp)
00000000 00000000 FUNC UND foo
if assembled to regular MIPS code, or:
Symbol table '.dynsym' contains 3 entries:
Num: Value Size Type Bind Vis Ndx Name
0:
00000000 0 NOTYPE LOCAL DEFAULT UND
1:
0000000d 0 FUNC GLOBAL DEFAULT UND bar
2:
00000001 0 FUNC GLOBAL DEFAULT UND foo
Primary GOT:
Canonical gp value:
00097ff0
Reserved entries:
Address Access Initial Purpose
00090000 -32752(gp)
00000000 Lazy resolver
00090004 -32748(gp)
80000000 Module pointer (GNU extension)
Global entries:
Address Access Initial Sym.Val. Type Ndx Name
00090008 -32744(gp)
0000000d 0000000d FUNC UND bar
0009000c -32740(gp)
00000001 00000001 FUNC UND foo
if assembled to microMIPS code. Symbol values and GOT entries record
the offset into the inexistent stubs section and the ISA bit rather than
zero, which would be the case if a lazy binding stub was not used for
other reasons, such as the value of the symbol being taken for a purpose
other than making a function call (e.g. an R_MIPS_GOT16 relocation).
Correct the issue by refraining from redirecting symbols to lazy binding
stubs if the stubs section is going to be discarded.
bfd/
* elfxx-mips.c (_bfd_mips_elf_adjust_dynamic_symbol): Don't set
`->needs_lazy_stub' if the stubs output section is the absolute
section.
ld/
* testsuite/ld-mips-elf/stub-dynsym-2.dd: New test.
* testsuite/ld-mips-elf/stub-dynsym-2.gd: New test.
* testsuite/ld-mips-elf/stub-dynsym-2.sd: New test.
* testsuite/ld-mips-elf/stub-dynsym-discard-2.gd: New test.
* testsuite/ld-mips-elf/stub-dynsym-discard-2.sd: New test.
* testsuite/ld-mips-elf/stub-dynsym-micromips-2.dd: New test.
* testsuite/ld-mips-elf/stub-dynsym-micromips-2.gd: New test.
* testsuite/ld-mips-elf/stub-dynsym-micromips-2.sd: New test.
* testsuite/ld-mips-elf/stub-dynsym-micromips-insn32-2.dd: New
test.
* testsuite/ld-mips-elf/stub-dynsym-micromips-insn32-2.gd: New
test.
* testsuite/ld-mips-elf/stub-dynsym-micromips-insn32-2.sd: New
test.
* testsuite/ld-mips-elf/stub-dynsym-2.ld: New test linker
script.
* testsuite/ld-mips-elf/stub-dynsym-discard-2.ld: New test
linker script.
* testsuite/ld-mips-elf/mips-elf.exp: Run the new tests.
+2018-07-09 Maciej W. Rozycki <macro@mips.com>
+
+ * elfxx-mips.c (_bfd_mips_elf_adjust_dynamic_symbol): Don't set
+ `->needs_lazy_stub' if the stubs output section is the absolute
+ section.
+
2018-07-09 H.J. Lu <hongjiu.lu@intel.com>
PR ld/23388
the symbol to the stub location. This is required to make
function pointers compare as equal between the normal
executable and the shared library. */
- if (!h->def_regular)
+ if (!h->def_regular
+ && !bfd_is_abs_section (htab->sstubs->output_section))
{
hmips->needs_lazy_stub = TRUE;
htab->lazy_stub_count++;
+2018-07-09 Maciej W. Rozycki <macro@mips.com>
+
+ * testsuite/ld-mips-elf/stub-dynsym-2.dd: New test.
+ * testsuite/ld-mips-elf/stub-dynsym-2.gd: New test.
+ * testsuite/ld-mips-elf/stub-dynsym-2.sd: New test.
+ * testsuite/ld-mips-elf/stub-dynsym-discard-2.gd: New test.
+ * testsuite/ld-mips-elf/stub-dynsym-discard-2.sd: New test.
+ * testsuite/ld-mips-elf/stub-dynsym-micromips-2.dd: New test.
+ * testsuite/ld-mips-elf/stub-dynsym-micromips-2.gd: New test.
+ * testsuite/ld-mips-elf/stub-dynsym-micromips-2.sd: New test.
+ * testsuite/ld-mips-elf/stub-dynsym-micromips-insn32-2.dd: New
+ test.
+ * testsuite/ld-mips-elf/stub-dynsym-micromips-insn32-2.gd: New
+ test.
+ * testsuite/ld-mips-elf/stub-dynsym-micromips-insn32-2.sd: New
+ test.
+ * testsuite/ld-mips-elf/stub-dynsym-2.ld: New test linker
+ script.
+ * testsuite/ld-mips-elf/stub-dynsym-discard-2.ld: New test
+ linker script.
+ * testsuite/ld-mips-elf/mips-elf.exp: Run the new tests.
+
2018-07-09 Maciej W. Rozycki <macro@mips.com>
* testsuite/lib/ld-lib.exp (run_dump_test): Call `unsupported'
"stub-dynsym$suffix-1-$dynsym.d"]] \
"stub-dynsym$suffix-1-$dynsym"]]
}
+ run_ld_link_tests [list \
+ [list \
+ "Retained stubs for dynsyms ($isa)" \
+ "-shared -melf32btsmip -T stub-dynsym-2.ld $lflag" "" \
+ "-EB $aflag -32 -KPIC" \
+ [list "stub-dynsym-2.s"] \
+ [list \
+ [list \
+ "objdump" \
+ "-dz -j .MIPS.stubs" \
+ "stub-dynsym$suffix-2.dd"] \
+ [list \
+ "readelf" \
+ "--dyn-syms" \
+ "stub-dynsym$suffix-2.sd"] \
+ [list \
+ "readelf" \
+ "-A" \
+ "stub-dynsym$suffix-2.gd"]] \
+ "stub-dynsym$suffix-2"] \
+ [list \
+ "Discarded stubs for dynsyms ($isa)" \
+ "-shared -melf32btsmip -T stub-dynsym-discard-2.ld $lflag" \
+ "" \
+ "-EB $aflag -32 -KPIC" \
+ [list "stub-dynsym-2.s"] \
+ [list \
+ [list \
+ "readelf" \
+ "--dyn-syms" \
+ "stub-dynsym-discard-2.sd"] \
+ [list \
+ "readelf" \
+ "-A" \
+ "stub-dynsym-discard-2.gd"]] \
+ "stub-dynsym-discard$suffix-2"]]
}
}
--- /dev/null
+.*: +file format .*mips.*
+
+Disassembly of section \.MIPS\.stubs:
+
+000800ec <_MIPS_STUBS_>:
+ 800ec: 8f998010 lw t9,-32752\(gp\)
+ 800f0: 03e07825 move t7,ra
+ 800f4: 0320f809 jalr t9
+ 800f8: 24180002 li t8,2
+ 800fc: 8f998010 lw t9,-32752\(gp\)
+ 80100: 03e07825 move t7,ra
+ 80104: 0320f809 jalr t9
+ 80108: 24180001 li t8,1
+ 8010c: 00000000 nop
+ 80110: 00000000 nop
+ 80114: 00000000 nop
+ 80118: 00000000 nop
--- /dev/null
+Primary GOT:
+ Canonical gp value: 00097ff0
+
+ Reserved entries:
+ Address Access Initial Purpose
+ 00090000 -32752\(gp\) 00000000 Lazy resolver
+ 00090004 -32748\(gp\) 80000000 Module pointer \(GNU extension\)
+
+ Global entries:
+ Address Access Initial Sym\.Val\. Type Ndx Name
+ 00090008 -32744\(gp\) 000800fc 000800fc FUNC UND bar
+ 0009000c -32740\(gp\) 000800ec 000800ec FUNC UND foo
--- /dev/null
+SECTIONS
+{
+ . = 0x80000;
+ .interp : { *(.interp) }
+ .hash : { *(.hash) }
+ .dynsym : { *(.dynsym) }
+ .dynstr : { *(.dynstr) }
+ .rel.dyn : { *(.rel.dyn) }
+ .MIPS.stubs : { *(.MIPS.stubs) }
+ .text : { *(.text) }
+
+ . = ALIGN (0x10000);
+ HIDDEN (_gp = . + 0x7ff0);
+ .got : { *(.got) }
+
+ .symtab : { *(.symtab) }
+ .strtab : { *(.strtab) }
+ .shstrtab : { *(.shstrtab) }
+
+ /DISCARD/ : { *(.reginfo) *(.MIPS.abiflags) *(.gnu.attributes) }
+}
--- /dev/null
+ lw $25,%call16(foo)($gp)
+ lw $25,%call16(bar)($gp)
--- /dev/null
+Symbol table '\.dynsym' contains 3 entries:
+ Num: Value Size Type Bind Vis Ndx Name
+ 0: 00000000 0 NOTYPE LOCAL DEFAULT UND
+ 1: 000800fc 0 FUNC GLOBAL DEFAULT UND bar
+ 2: 000800ec 0 FUNC GLOBAL DEFAULT UND foo
--- /dev/null
+Primary GOT:
+ Canonical gp value: 00097ff0
+
+ Reserved entries:
+ Address Access Initial Purpose
+ 00090000 -32752\(gp\) 00000000 Lazy resolver
+ 00090004 -32748\(gp\) 80000000 Module pointer \(GNU extension\)
+
+ Global entries:
+ Address Access Initial Sym\.Val\. Type Ndx Name
+ 00090008 -32744\(gp\) 00000000 00000000 FUNC UND bar
+ 0009000c -32740\(gp\) 00000000 00000000 FUNC UND foo
--- /dev/null
+SECTIONS
+{
+ . = 0x80000;
+ .interp : { *(.interp) }
+ .hash : { *(.hash) }
+ .dynsym : { *(.dynsym) }
+ .dynstr : { *(.dynstr) }
+ .rel.dyn : { *(.rel.dyn) }
+ .text : { *(.text) }
+
+ . = ALIGN (0x10000);
+ HIDDEN (_gp = . + 0x7ff0);
+ .got : { *(.got) }
+
+ .symtab : { *(.symtab) }
+ .strtab : { *(.strtab) }
+ .shstrtab : { *(.shstrtab) }
+
+ /DISCARD/ : {
+ *(.reginfo) *(.MIPS.abiflags) *(.MIPS.stubs) *(.gnu.attributes)
+ }
+}
--- /dev/null
+Symbol table '\.dynsym' contains 3 entries:
+ Num: Value Size Type Bind Vis Ndx Name
+ 0: 00000000 0 NOTYPE LOCAL DEFAULT UND
+ 1: 00000000 0 FUNC GLOBAL DEFAULT UND bar
+ 2: 00000000 0 FUNC GLOBAL DEFAULT UND foo
--- /dev/null
+.*: +file format .*mips.*
+
+Disassembly of section \.MIPS\.stubs:
+
+000800ec <_MIPS_STUBS_>:
+ 800ec: ff3c 8010 lw t9,-32752\(gp\)
+ 800f0: 0dff move t7,ra
+ 800f2: 45d9 jalr t9
+ 800f4: 3300 0002 li t8,2
+ 800f8: ff3c 8010 lw t9,-32752\(gp\)
+ 800fc: 0dff move t7,ra
+ 800fe: 45d9 jalr t9
+ 80100: 3300 0001 li t8,1
+ 80104: 0000 0000 nop
+ 80108: 0000 0000 nop
+ 8010c: 0000 0000 nop
--- /dev/null
+Primary GOT:
+ Canonical gp value: 00097ff0
+
+ Reserved entries:
+ Address Access Initial Purpose
+ 00090000 -32752\(gp\) 00000000 Lazy resolver
+ 00090004 -32748\(gp\) 80000000 Module pointer \(GNU extension\)
+
+ Global entries:
+ Address Access Initial Sym\.Val\. Type Ndx Name
+ 00090008 -32744\(gp\) 000800f9 000800f9 FUNC UND bar
+ 0009000c -32740\(gp\) 000800ed 000800ed FUNC UND foo
--- /dev/null
+Symbol table '\.dynsym' contains 3 entries:
+ Num: Value Size Type Bind Vis Ndx Name
+ 0: 00000000 0 NOTYPE LOCAL DEFAULT UND
+ 1: 000800f9 0 FUNC GLOBAL DEFAULT UND bar
+ 2: 000800ed 0 FUNC GLOBAL DEFAULT UND foo
--- /dev/null
+.*: +file format .*mips.*
+
+Disassembly of section \.MIPS\.stubs:
+
+000800ec <_MIPS_STUBS_>:
+ 800ec: ff3c 8010 lw t9,-32752\(gp\)
+ 800f0: 001f 7a90 move t7,ra
+ 800f4: 03f9 0f3c jalr t9
+ 800f8: 3300 0002 li t8,2
+ 800fc: ff3c 8010 lw t9,-32752\(gp\)
+ 80100: 001f 7a90 move t7,ra
+ 80104: 03f9 0f3c jalr t9
+ 80108: 3300 0001 li t8,1
+ 8010c: 0000 0000 nop
+ 80110: 0000 0000 nop
+ 80114: 0000 0000 nop
+ 80118: 0000 0000 nop
--- /dev/null
+Primary GOT:
+ Canonical gp value: 00097ff0
+
+ Reserved entries:
+ Address Access Initial Purpose
+ 00090000 -32752\(gp\) 00000000 Lazy resolver
+ 00090004 -32748\(gp\) 80000000 Module pointer \(GNU extension\)
+
+ Global entries:
+ Address Access Initial Sym\.Val\. Type Ndx Name
+ 00090008 -32744\(gp\) 000800fd 000800fd FUNC UND bar
+ 0009000c -32740\(gp\) 000800ed 000800ed FUNC UND foo
--- /dev/null
+Symbol table '\.dynsym' contains 3 entries:
+ Num: Value Size Type Bind Vis Ndx Name
+ 0: 00000000 0 NOTYPE LOCAL DEFAULT UND
+ 1: 000800fd 0 FUNC GLOBAL DEFAULT UND bar
+ 2: 000800ed 0 FUNC GLOBAL DEFAULT UND foo