Index: lld/ELF/InputFiles.h =================================================================== --- lld/ELF/InputFiles.h +++ lld/ELF/InputFiles.h @@ -184,8 +184,9 @@ void initializeSymbols(); void initializeDwarfLine(); InputSectionBase *getRelocTarget(const Elf_Shdr &Sec); - InputSectionBase *createInputSection(const Elf_Shdr &Sec, - StringRef SectionStringTable); + InputSectionBase * + createInputSection(const Elf_Shdr &Sec, StringRef SectionStringTable, + llvm::DenseSet &ComdatGroups); bool shouldMerge(const Elf_Shdr &Sec); SymbolBody *createSymbolBody(const Elf_Sym *Sym); Index: lld/ELF/InputFiles.cpp =================================================================== --- lld/ELF/InputFiles.cpp +++ lld/ELF/InputFiles.cpp @@ -294,7 +294,7 @@ case SHT_NULL: break; default: - Sections[I] = createInputSection(Sec, SectionStringTable); + Sections[I] = createInputSection(Sec, SectionStringTable, ComdatGroups); } // .ARM.exidx sections have a reverse dependency on the InputSection they @@ -329,12 +329,11 @@ } template -InputSectionBase * -elf::ObjectFile::createInputSection(const Elf_Shdr &Sec, - StringRef SectionStringTable) { +InputSectionBase *elf::ObjectFile::createInputSection( + const Elf_Shdr &Sec, StringRef SectionStringTable, + DenseSet &ComdatGroups) { StringRef Name = check(this->getObj().getSectionName(&Sec, SectionStringTable)); - switch (Sec.sh_type) { case SHT_ARM_ATTRIBUTES: // FIXME: ARM meta-data section. Retain the first attribute section @@ -399,6 +398,15 @@ if (Config->Strip != StripPolicy::None && Name.startswith(".debug")) return &InputSection::Discarded; + // The linkonce feature is a sort of proto-comdat. bfd and gold appear to + // treat linkonce sections as if they were single-element comdat groups named + // after the section name suffix (which means they can cause comdat groups to + // be discarded), and this feature is necessary to link parts of the x86-32 + // CRT on Linux. + if (Name.startswith(".gnu.linkonce.t.") && + !ComdatGroups.insert(CachedHashStringRef(Name.substr(16))).second) + return &InputSection::Discarded; + // The linker merges EH (exception handling) frames and creates a // .eh_frame_hdr section for runtime. So we handle them with a special // class. For relocatable outputs, they are just passed through. Index: lld/test/ELF/comdat-linkonce.s =================================================================== --- /dev/null +++ lld/test/ELF/comdat-linkonce.s @@ -0,0 +1,9 @@ +// RUN: llvm-mc -filetype=obj -triple=x86_64-pc-linux %s -o %t.o +// RUN: llvm-mc -filetype=obj -triple=x86_64-pc-linux %p/Inputs/comdat.s -o %t2.o +// RUN: ld.lld -shared %t.o %t2.o -o %t +// RUN: ld.lld -shared %t2.o %t.o -o %t + +.section .gnu.linkonce.t.zed +.globl abc +abc: +nop