diff options
-rw-r--r-- | lld/ELF/Driver.cpp | 7 | ||||
-rw-r--r-- | lld/ELF/InputFiles.h | 4 | ||||
-rw-r--r-- | lld/ELF/InputSection.cpp | 1 | ||||
-rw-r--r-- | lld/ELF/InputSection.h | 10 | ||||
-rw-r--r-- | lld/ELF/MarkLive.cpp | 36 | ||||
-rw-r--r-- | lld/test/ELF/linkerscript/comdat-gc.s | 3 | ||||
-rw-r--r-- | lld/test/ELF/linkerscript/debuginfo-gc.s | 14 |
7 files changed, 56 insertions, 19 deletions
diff --git a/lld/ELF/Driver.cpp b/lld/ELF/Driver.cpp index 49812f960dd..57b9146a899 100644 --- a/lld/ELF/Driver.cpp +++ b/lld/ELF/Driver.cpp @@ -1613,11 +1613,8 @@ template <class ELFT> void LinkerDriver::link(opt::InputArgList &Args) { // We do not want to emit debug sections if --strip-all // or -strip-debug are given. - if (Config->Strip != StripPolicy::None) { - llvm::erase_if(InputSections, [](InputSectionBase *S) { - return S->Name.startswith(".debug") || S->Name.startswith(".zdebug"); - }); - } + if (Config->Strip != StripPolicy::None) + llvm::erase_if(InputSections, [](InputSectionBase *S) { return S->Debug; }); // The Target instance handles target-specific stuff, such as applying // relocations or writing a PLT section. It also contains target-dependent diff --git a/lld/ELF/InputFiles.h b/lld/ELF/InputFiles.h index 1b9de7183c1..e60b63c7b61 100644 --- a/lld/ELF/InputFiles.h +++ b/lld/ELF/InputFiles.h @@ -234,6 +234,10 @@ public: // but had one or more functions with the no_split_stack attribute. bool SomeNoSplitStack = false; + // True if the file has any live Regular or Merge sections that aren't + // the LDSA section. + bool HasLiveCodeOrData = false; + // Pointer to this input file's .llvm_addrsig section, if it has one. const Elf_Shdr *AddrsigSec = nullptr; diff --git a/lld/ELF/InputSection.cpp b/lld/ELF/InputSection.cpp index 8acaa2c0e05..58c3513aeff 100644 --- a/lld/ELF/InputSection.cpp +++ b/lld/ELF/InputSection.cpp @@ -71,6 +71,7 @@ InputSectionBase::InputSectionBase(InputFile *File, uint64_t Flags, NumRelocations = 0; AreRelocsRela = false; + Debug = Name.startswith(".debug") || Name.startswith(".zdebug"); // The ELF spec states that a value of 0 means the section has // no alignment constraits. diff --git a/lld/ELF/InputSection.h b/lld/ELF/InputSection.h index b559ff1b202..ee5c7cbf6e3 100644 --- a/lld/ELF/InputSection.h +++ b/lld/ELF/InputSection.h @@ -51,7 +51,7 @@ public: unsigned SectionKind : 3; - // The next three bit fields are only used by InputSectionBase, but we + // The next four bit fields are only used by InputSectionBase, but we // put them here so the struct packs better. // The garbage collector sets sections' Live bits. @@ -73,6 +73,9 @@ public: unsigned Bss : 1; + // True if this is a debuginfo section. + unsigned Debug : 1; + // Set for sections that should not be folded by ICF. unsigned KeepUnique : 1; @@ -100,8 +103,9 @@ protected: uint64_t Entsize, uint64_t Alignment, uint32_t Type, uint32_t Info, uint32_t Link) : Name(Name), Repl(this), SectionKind(SectionKind), Live(false), - Assigned(false), Bss(false), KeepUnique(false), Alignment(Alignment), - Flags(Flags), Entsize(Entsize), Type(Type), Link(Link), Info(Info) {} + Assigned(false), Bss(false), Debug(false), KeepUnique(false), + Alignment(Alignment), Flags(Flags), Entsize(Entsize), Type(Type), + Link(Link), Info(Info) {} }; // This corresponds to a section of an input file. diff --git a/lld/ELF/MarkLive.cpp b/lld/ELF/MarkLive.cpp index d040a214aa6..08f26aed3b1 100644 --- a/lld/ELF/MarkLive.cpp +++ b/lld/ELF/MarkLive.cpp @@ -47,7 +47,7 @@ public: void run(); private: - void enqueue(InputSectionBase *Sec, uint64_t Offset); + void enqueue(InputSectionBase *Sec, uint64_t Offset, bool IsLSDA); void markSymbol(Symbol *Sym); template <class RelTy> @@ -97,7 +97,7 @@ void MarkLive<ELFT>::resolveReloc(InputSectionBase &Sec, RelTy &Rel, Offset += getAddend<ELFT>(Sec, Rel); if (!IsLSDA || !(RelSec->Flags & SHF_EXECINSTR)) - enqueue(RelSec, Offset); + enqueue(RelSec, Offset, IsLSDA); return; } @@ -106,7 +106,7 @@ void MarkLive<ELFT>::resolveReloc(InputSectionBase &Sec, RelTy &Rel, SS->getFile().IsNeeded = true; for (InputSectionBase *Sec : CNamedSections.lookup(Sym.getName())) - enqueue(Sec, 0); + enqueue(Sec, 0, IsLSDA); } // The .eh_frame section is an unfortunate special case. @@ -169,7 +169,8 @@ static bool isReserved(InputSectionBase *Sec) { } template <class ELFT> -void MarkLive<ELFT>::enqueue(InputSectionBase *Sec, uint64_t Offset) { +void MarkLive<ELFT>::enqueue(InputSectionBase *Sec, uint64_t Offset, + bool IsLSDA) { // Skip over discarded sections. This in theory shouldn't happen, because // the ELF spec doesn't allow a relocation to point to a deduplicated // COMDAT section directly. Unfortunately this happens in practice (e.g. @@ -183,19 +184,26 @@ void MarkLive<ELFT>::enqueue(InputSectionBase *Sec, uint64_t Offset) { if (auto *MS = dyn_cast<MergeInputSection>(Sec)) MS->getSectionPiece(Offset)->Live = true; + InputSection *S = dyn_cast<InputSection>(Sec); + // LSDA does not count as "live code or data" in the object file. + // The section may already have been marked live for LSDA in which + // case we still need to mark the file. + if (S && !IsLSDA && Sec->File) + Sec->getFile<ELFT>()->HasLiveCodeOrData = true; + if (Sec->Live) return; - Sec->Live = true; + Sec->Live = true; // Add input section to the queue. - if (InputSection *S = dyn_cast<InputSection>(Sec)) + if (S) Queue.push_back(S); } template <class ELFT> void MarkLive<ELFT>::markSymbol(Symbol *Sym) { if (auto *D = dyn_cast_or_null<Defined>(Sym)) if (auto *IS = dyn_cast_or_null<InputSectionBase>(D->Section)) - enqueue(IS, D->Value); + enqueue(IS, D->Value, false); } // This is the main function of the garbage collector. @@ -239,7 +247,7 @@ template <class ELFT> void MarkLive<ELFT>::run() { continue; if (isReserved(Sec) || Script->shouldKeep(Sec)) { - enqueue(Sec, 0); + enqueue(Sec, 0, false); } else if (isValidCIdentifier(Sec->Name)) { CNamedSections[Saver.save("__start_" + Sec->Name)].push_back(Sec); CNamedSections[Saver.save("__stop_" + Sec->Name)].push_back(Sec); @@ -259,7 +267,7 @@ template <class ELFT> void MarkLive<ELFT>::run() { } for (InputSectionBase *IS : Sec.DependentSections) - enqueue(IS, 0); + enqueue(IS, 0, false); } } @@ -285,7 +293,7 @@ template <class ELFT> void elf::markLive() { // The -gc-sections option works only for SHF_ALLOC sections // (sections that are memory-mapped at runtime). So we can // unconditionally make non-SHF_ALLOC sections alive except - // SHF_LINK_ORDER and SHT_REL/SHT_RELA sections. + // SHF_LINK_ORDER and SHT_REL/SHT_RELA sections and .debug sections. // // Usually, SHF_ALLOC sections are not removed even if they are // unreachable through relocations because reachability is not @@ -306,13 +314,19 @@ template <class ELFT> void elf::markLive() { bool IsLinkOrder = (Sec->Flags & SHF_LINK_ORDER); bool IsRel = (Sec->Type == SHT_REL || Sec->Type == SHT_RELA); - if (!IsAlloc && !IsLinkOrder && !IsRel) + if (!IsAlloc && !IsLinkOrder && !IsRel && !Sec->Debug) Sec->Live = true; } // Follow the graph to mark all live sections. MarkLive<ELFT>().run(); + // Mark debug sections as live in any object file that has a live + // Regular or Merge section. + for (InputSectionBase *Sec : InputSections) + if (Sec->Debug && Sec->getFile<ELFT>()->HasLiveCodeOrData) + Sec->Live = true; + // Report garbage-collected sections. if (Config->PrintGcSections) for (InputSectionBase *Sec : InputSections) diff --git a/lld/test/ELF/linkerscript/comdat-gc.s b/lld/test/ELF/linkerscript/comdat-gc.s index 63dcf85c5a6..223ce93db53 100644 --- a/lld/test/ELF/linkerscript/comdat-gc.s +++ b/lld/test/ELF/linkerscript/comdat-gc.s @@ -8,6 +8,9 @@ # GC1: Name: .debug_line +# Add .ctors section so all debuginfo isn't GCed +.section .ctors,"ax",@progbits + .file 1 "test/ELF/linkerscript/comdat_gc.s" .section .text._Z3fooIiEvv,"axG",@progbits,_Z3fooIiEvv,comdat .loc 1 14 diff --git a/lld/test/ELF/linkerscript/debuginfo-gc.s b/lld/test/ELF/linkerscript/debuginfo-gc.s new file mode 100644 index 00000000000..a991e5f1dca --- /dev/null +++ b/lld/test/ELF/linkerscript/debuginfo-gc.s @@ -0,0 +1,14 @@ +# REQUIRES: x86 + +# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux %s -o %t +# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux %p/Inputs/comdat-gc.s -o %t1 +# RUN: echo "SECTIONS { .text : { *(.text*) } }" > %t.script +# RUN: ld.lld --gc-sections --script %t.script %t %t1 -o %t2 +# RUN: llvm-readobj -sections -symbols %t2 | FileCheck %s + +# CHECK-NOT: Name: .debug_line + +.file 1 "test/ELF/linkerscript/comdat_gc.s" +.section .text._Z3fooIiEvv,"axG",@progbits,_Z3fooIiEvv,comdat +.loc 1 14 + ret |