forked from OSchip/llvm-project
[LLD] [COFF] Fix mingw comdat associativity for leader symbols with a different name
For a weak symbol func in a comdat, the actual leader symbol ends up named like .weak.func.default*. Likewise, for stdcall on i386, the symbol may be named _func@4, while the section suffix only is "func", which the previous implementation didn't handle. This fixes unwinding through weak functions when using -ffunction-sections in mingw environments. Differential Revision: https://reviews.llvm.org/D84607
This commit is contained in:
parent
a7044edde7
commit
343ffa70fc
|
@ -348,13 +348,13 @@ void ObjFile::recordPrevailingSymbolForMingw(
|
|||
// of the section chunk we actually include instead of discarding it,
|
||||
// add the symbol to a map to allow using it for implicitly
|
||||
// associating .[px]data$<func> sections to it.
|
||||
// Use the suffix from the .text$<func> instead of the leader symbol
|
||||
// name, for cases where the names differ (i386 mangling/decorations,
|
||||
// cases where the leader is a weak symbol named .weak.func.default*).
|
||||
int32_t sectionNumber = sym.getSectionNumber();
|
||||
SectionChunk *sc = sparseChunks[sectionNumber];
|
||||
if (sc && sc->getOutputCharacteristics() & IMAGE_SCN_MEM_EXECUTE) {
|
||||
StringRef name;
|
||||
name = check(coffObj->getSymbolName(sym));
|
||||
if (getMachineType() == I386)
|
||||
name.consume_front("_");
|
||||
StringRef name = sc->getSectionName().split('$').second;
|
||||
prevailingSectionMap[name] = sectionNumber;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,10 +1,14 @@
|
|||
# REQUIRES: x86
|
||||
|
||||
# RUN: llvm-mc -triple=i686-windows-gnu %s -filetype=obj -o %t.obj
|
||||
# RUN: llvm-mc -triple=i686-windows-gnu %s -defsym stdcall=0 -filetype=obj -o %t.obj
|
||||
|
||||
# RUN: lld-link -lldmingw -entry:main %t.obj -out:%t.exe
|
||||
# RUN: llvm-objdump -s %t.exe | FileCheck %s
|
||||
|
||||
# RUN: llvm-mc -triple=i686-windows-gnu %s -defsym stdcall=1 -filetype=obj -o %t.stdcall.obj
|
||||
# RUN: lld-link -lldmingw -entry:main %t.stdcall.obj -out:%t.stdcall.exe
|
||||
# RUN: llvm-objdump -s %t.stdcall.exe | FileCheck %s
|
||||
|
||||
# Check that the .eh_frame comdat was included, even if it had no symbols,
|
||||
# due to associativity with the symbol _foo.
|
||||
|
||||
|
@ -19,19 +23,34 @@
|
|||
.globl _main
|
||||
.p2align 4, 0x90
|
||||
_main:
|
||||
.if stdcall==0
|
||||
call _foo
|
||||
.else
|
||||
call _foo@0
|
||||
.endif
|
||||
ret
|
||||
|
||||
.section .eh_frame$foo,"dr"
|
||||
.linkonce discard
|
||||
.byte 0x42
|
||||
|
||||
.if stdcall==0
|
||||
.def _foo;
|
||||
.else
|
||||
.def _foo@0;
|
||||
.endif
|
||||
.scl 2;
|
||||
.type 32;
|
||||
.endef
|
||||
.if stdcall==0
|
||||
.section .text$foo,"xr",discard,_foo
|
||||
.globl _foo
|
||||
.p2align 4
|
||||
_foo:
|
||||
.else
|
||||
.section .text$foo,"xr",discard,_foo@0
|
||||
.globl _foo@0
|
||||
.p2align 4
|
||||
_foo@0:
|
||||
.endif
|
||||
ret
|
||||
|
|
|
@ -0,0 +1,63 @@
|
|||
# REQUIRES: x86
|
||||
|
||||
# RUN: llvm-mc -triple=x86_64-windows-gnu %s -filetype=obj -o %t.obj
|
||||
# RUN: llvm-readobj --symbols %t.obj | FileCheck %s --check-prefix=SYMBOL
|
||||
|
||||
# RUN: lld-link -lldmingw -entry:main %t.obj -out:%t.exe -lldmap:%t.map -verbose
|
||||
# RUN: llvm-readobj --sections %t.exe | FileCheck %s
|
||||
|
||||
# CHECK: Sections [
|
||||
# CHECK: Section {
|
||||
# CHECK: Number: 2
|
||||
# CHECK-LABEL: Name: .rdata (2E 72 64 61 74 61 00 00)
|
||||
# This is the critical check to show that .xdata$foo was
|
||||
# retained, while .xdata$bar wasn't. This *must* be 0x24
|
||||
# (0x4 for the .xdata section and 0x20 for the
|
||||
# .ctors/.dtors headers/ends).
|
||||
# CHECK-NEXT: VirtualSize: 0x24
|
||||
|
||||
# Check that the weak symbols still are emitted as it was when the test was
|
||||
# written, to make sure the test still actually tests what was intended.
|
||||
|
||||
# SYMBOL: Symbol {
|
||||
# SYMBOL: Name: foo
|
||||
# SYMBOL-NEXT: Value: 0
|
||||
# SYMBOL-NEXT: Section: IMAGE_SYM_UNDEFINED (0)
|
||||
# SYMBOL-NEXT: BaseType: Null (0x0)
|
||||
# SYMBOL-NEXT: ComplexType: Null (0x0)
|
||||
# SYMBOL-NEXT: StorageClass: WeakExternal (0x69)
|
||||
# SYMBOL-NEXT: AuxSymbolCount: 1
|
||||
# SYMBOL-NEXT: AuxWeakExternal {
|
||||
# SYMBOL-NEXT: Linked: .weak.foo.default.main (19)
|
||||
# SYMBOL-NEXT: Search: Alias (0x3)
|
||||
# SYMBOL-NEXT: }
|
||||
# SYMBOL-NEXT: }
|
||||
|
||||
.text
|
||||
.globl main
|
||||
main:
|
||||
call foo
|
||||
retq
|
||||
|
||||
# See associative-comdat-mingw.s for the general setup. Here, the leader
|
||||
# symbols are weak, which causes the functions foo and bar to be undefined
|
||||
# weak externals, while the actual leader symbols are named like
|
||||
# .weak.foo.default.main.
|
||||
|
||||
.section .xdata$foo,"dr"
|
||||
.linkonce discard
|
||||
.long 42
|
||||
|
||||
.section .xdata$bar,"dr"
|
||||
.linkonce discard
|
||||
.long 43
|
||||
|
||||
.section .text$foo,"xr",discard,foo
|
||||
.weak foo
|
||||
foo:
|
||||
ret
|
||||
|
||||
.section .text$bar,"xr",discard,bar
|
||||
.weak bar
|
||||
bar:
|
||||
ret
|
Loading…
Reference in New Issue