diff --git a/lld/MachO/UnwindInfoSection.cpp b/lld/MachO/UnwindInfoSection.cpp index 507cca0fb601..10304327ed2f 100644 --- a/lld/MachO/UnwindInfoSection.cpp +++ b/lld/MachO/UnwindInfoSection.cpp @@ -100,6 +100,9 @@ bool UnwindInfoSection::isNeeded() const { return (compactUnwindSection != nullptr); } +SmallDenseMap, macho::Symbol *> + personalityTable; + // Compact unwind relocations have different semantics, so we handle them in a // separate code path from regular relocations. First, we do not wish to add // rebase opcodes for __LD,__compact_unwind, because that section doesn't @@ -109,25 +112,39 @@ void macho::prepareCompactUnwind(InputSection *isec) { assert(isec->segname == segment_names::ld && isec->name == section_names::compactUnwind); - DenseMap, macho::Symbol *> - anonPersonalitySymbols; for (Reloc &r : isec->relocs) { - // TODO: generalize for other archs - assert(r.type == X86_64_RELOC_UNSIGNED); + assert(target->hasAttr(r.type, RelocAttrBits::UNSIGNED)); if (r.offset % sizeof(CompactUnwindEntry64) != offsetof(struct CompactUnwindEntry64, personality)) continue; if (auto *s = r.referent.dyn_cast()) { - if (auto *undefined = dyn_cast(s)) + if (auto *undefined = dyn_cast(s)) { treatUndefinedSymbol(*undefined); - else - in.got->addEntry(s); - } else if (auto *referentIsec = r.referent.dyn_cast()) { + continue; + } + if (auto *defined = dyn_cast(s)) { + // Check if we have created a synthetic symbol at the same address. + macho::Symbol *&personality = + personalityTable[{defined->isec, defined->value}]; + if (personality == nullptr) { + personality = defined; + in.got->addEntry(defined); + } else if (personality != defined) { + r.referent = personality; + } + continue; + } + assert(isa(s)); + in.got->addEntry(s); + continue; + } + + if (auto *referentIsec = r.referent.dyn_cast()) { // Personality functions can be referenced via section relocations - // if they live in an object file (instead of a dylib). Create - // placeholder synthetic symbols for them in the GOT. - macho::Symbol *&s = anonPersonalitySymbols[{referentIsec, r.addend}]; + // if they live in the same object file. Create placeholder synthetic + // symbols for them in the GOT. + macho::Symbol *&s = personalityTable[{referentIsec, r.addend}]; if (s == nullptr) { s = make("", nullptr, referentIsec, r.addend, false, false, false); @@ -225,7 +242,6 @@ void UnwindInfoSection::finalize() { size_t cuCount = compactUnwindSection->getSize() / sizeof(CompactUnwindEntry64); cuVector.resize(cuCount); - // Relocate all __LD,__compact_unwind entries relocateCompactUnwind(compactUnwindSection, cuVector); // Rather than sort & fold the 32-byte entries directly, we create a diff --git a/lld/test/MachO/compact-unwind.s b/lld/test/MachO/compact-unwind.s index 6cddda9adfc9..3d853f8e9aa6 100644 --- a/lld/test/MachO/compact-unwind.s +++ b/lld/test/MachO/compact-unwind.s @@ -1,39 +1,48 @@ # REQUIRES: x86 -# RUN: llvm-mc -filetype=obj -triple=x86_64-apple-darwin19.0.0 %s -o %t.o -# RUN: %lld -pie -lSystem -lc++ %t.o -o %t -# RUN: llvm-objdump --macho --unwind-info --syms --indirect-symbols --rebase %t | FileCheck %s +# RUN: rm -rf %t; split-file %s %t +# RUN: llvm-mc -filetype=obj -triple=x86_64-apple-darwin19.0.0 %t/my_personality.s -o %t/my_personality.o +# RUN: llvm-mc -filetype=obj -triple=x86_64-apple-darwin19.0.0 %t/main.s -o %t/main.o +# RUN: %lld -pie -lSystem -lc++ %t/my_personality.o %t/main.o -o %t/personality-first +# RUN: llvm-objdump --macho --unwind-info --syms --indirect-symbols --rebase %t/personality-first | FileCheck %s --check-prefixes=FIRST,CHECK +# RUN: %lld -pie -lSystem -lc++ %t/main.o %t/my_personality.o -o %t/personality-second +# RUN: llvm-objdump --macho --unwind-info --syms --indirect-symbols --rebase %t/personality-second | FileCheck %s --check-prefixes=SECOND,CHECK -# CHECK: Indirect symbols for (__DATA_CONST,__got) -# CHECK-NEXT: address index name -# CHECK-DAG: 0x[[#%x,GXX_PERSONALITY:]] [[#]] ___gxx_personality_v0 -# CHECK-DAG: 0x[[#%x,MY_PERSONALITY:]] LOCAL +# FIRST: Indirect symbols for (__DATA_CONST,__got) +# FIRST-NEXT: address index name +# FIRST-DAG: 0x[[#%x,GXX_PERSONALITY:]] [[#]] ___gxx_personality_v0 +# FIRST-DAG: 0x[[#%x,MY_PERSONALITY:]] LOCAL + +# SECOND: Indirect symbols for (__DATA_CONST,__got) +# SECOND-NEXT: address index name +# SECOND-DAG: 0x[[#%x,GXX_PERSONALITY:]] [[#]] ___gxx_personality_v0 +# SECOND-DAG: 0x[[#%x,MY_PERSONALITY:]] [[#]] _my_personality # CHECK: SYMBOL TABLE: # CHECK-DAG: [[#%x,MAIN:]] g F __TEXT,__text _main -# CHECK-DAG: [[#%x,FOO:]] g F __TEXT,__text _foo +# CHECK-DAG: [[#%x,FOO:]] l F __TEXT,__text _foo # CHECK-DAG: [[#%x,EXCEPTION0:]] g O __TEXT,__gcc_except_tab _exception0 # CHECK-DAG: [[#%x,EXCEPTION1:]] g O __TEXT,__gcc_except_tab _exception1 # CHECK: Contents of __unwind_info section: # CHECK: Personality functions: (count = 2) -# CHECK-NEXT: personality[1]: 0x{{0*}}[[#MY_PERSONALITY-0x100000000]] -# CHECK-NEXT: personality[2]: 0x{{0*}}[[#GXX_PERSONALITY-0x100000000]] +# CHECK-DAG: personality[{{[0-9]+}}]: 0x{{0*}}[[#MY_PERSONALITY-0x100000000]] +# CHECK-DAG: personality[{{[0-9]+}}]: 0x{{0*}}[[#GXX_PERSONALITY-0x100000000]] # CHECK: LSDA descriptors: -# CHECK-NEXT: [0]: function offset=0x{{0*}}[[#FOO-0x100000000]], LSDA offset=0x{{0*}}[[#EXCEPTION0-0x100000000]] -# CHECK-NEXT: [1]: function offset=0x{{0*}}[[#MAIN-0x100000000]], LSDA offset=0x{{0*}}[[#EXCEPTION1-0x100000000]] +# CHECK-DAG: function offset=0x{{0*}}[[#FOO-0x100000000]], LSDA offset=0x{{0*}}[[#EXCEPTION0-0x100000000]] +# CHECK-DAG: function offset=0x{{0*}}[[#MAIN-0x100000000]], LSDA offset=0x{{0*}}[[#EXCEPTION1-0x100000000]] ## Check that we do not add rebase opcodes to the compact unwind section. # CHECK: Rebase table: # CHECK-NEXT: segment section address type # CHECK-NEXT: __DATA_CONST __got 0x{{[0-9a-f]*}} pointer -# CHECK-NEXT: __DATA_CONST __got 0x{{[0-9a-f]*}} pointer -# CHECK-EMPTY: - -.globl _main, _foo, _my_personality, _bar, _exception0, _exception1 +# CHECK-NOT: __TEXT +#--- my_personality.s +.globl _my_personality, _exception0 .text _foo: .cfi_startproc +## This will generate a section relocation. .cfi_personality 155, _my_personality .cfi_lsda 16, _exception0 .cfi_def_cfa_offset 16 @@ -49,6 +58,17 @@ _bar: retq .cfi_endproc +_my_personality: + retq + +.section __TEXT,__gcc_except_tab +_exception0: + .space 1 + +#--- main.s +.globl _main, _my_personality, _exception1 + +.text _main: .cfi_startproc .cfi_personality 155, ___gxx_personality_v0 @@ -57,11 +77,17 @@ _main: retq .cfi_endproc -_my_personality: +_baz: + .cfi_startproc +## This will generate a symbol relocation. Check that we reuse the personality +## referenced by the section relocation in my_personality.s. + .cfi_personality 155, _my_personality + .cfi_lsda 16, _exception1 + .cfi_def_cfa_offset 16 retq + .cfi_endproc + .section __TEXT,__gcc_except_tab -_exception0: - .space 1 _exception1: .space 1