Index: ELF/Config.h =================================================================== --- ELF/Config.h +++ ELF/Config.h @@ -25,6 +25,7 @@ class InputFile; class InputSectionBase; +class Symbol; enum ELFKind { ELFNoneKind, @@ -108,6 +109,8 @@ llvm::MapVector, uint64_t> CallGraphProfile; + llvm::MapVector, uint64_t> + CallGraphProfileSymbols; bool AllowMultipleDefinition; bool AndroidPackDynRelocs = false; bool ARMHasBlx = false; Index: ELF/Driver.cpp =================================================================== --- ELF/Driver.cpp +++ ELF/Driver.cpp @@ -572,6 +572,23 @@ return {BuildIdKind::None, {}}; } +static void addCallGraphEdge(const Symbol *FromSym, const Symbol *ToSym, + uint64_t Count) { + if (!FromSym || !ToSym || Count == 0) + return; + warnUnorderableSymbol(FromSym); + warnUnorderableSymbol(ToSym); + const Defined *FromSymD = dyn_cast(FromSym); + const Defined *ToSymD = dyn_cast(ToSym); + if (!FromSymD || !ToSymD) + return; + const auto *FromSB = dyn_cast_or_null(FromSymD->Section); + const auto *ToSB = dyn_cast_or_null(ToSymD->Section); + if (!FromSB || !ToSB) + return; + Config->CallGraphProfile[std::make_pair(FromSB, ToSB)] += Count; +} + static void readCallGraph(MemoryBufferRef MB) { // Build a map from symbol name to section DenseMap SymbolNameToSymbol; @@ -595,19 +612,7 @@ if (!ToSym) warn("call graph file: no such symbol: " + Fields[1]); } - if (!FromSym || !ToSym || Count == 0) - continue; - warnUnorderableSymbol(FromSym); - warnUnorderableSymbol(ToSym); - const Defined *FromSymD = dyn_cast(FromSym); - const Defined *ToSymD = dyn_cast(ToSym); - if (!FromSymD || !ToSymD) - continue; - const auto *FromSB = dyn_cast_or_null(FromSymD->Section); - const auto *ToSB = dyn_cast_or_null(ToSymD->Section); - if (!FromSB || !ToSB) - continue; - Config->CallGraphProfile[std::make_pair(FromSB, ToSB)] += Count; + addCallGraphEdge(FromSym, ToSym, Count); } } @@ -1268,6 +1273,11 @@ if (Optional Buffer = readFile(Arg->getValue())) readCallGraph(*Buffer); + // Now that symbol to section mapping is complete, convert the symbol edges + // read from object files to section edges. + for (auto &Edge : Config->CallGraphProfileSymbols) + addCallGraphEdge(Edge.first.first, Edge.first.second, Edge.second); + // Write the result to the file. writeResult(); } Index: ELF/InputFiles.h =================================================================== --- ELF/InputFiles.h +++ ELF/InputFiles.h @@ -167,6 +167,7 @@ typedef typename ELFT::Sym Elf_Sym; typedef typename ELFT::Shdr Elf_Shdr; typedef typename ELFT::Word Elf_Word; + typedef typename ELFT::CGProfile Elf_CGProfile; StringRef getShtGroupSignature(ArrayRef Sections, const Elf_Shdr &Sec); @@ -214,6 +215,7 @@ void initializeSymbols(); void initializeJustSymbols(); void initializeDwarf(); + void parseCGProfile(); InputSectionBase *getRelocTarget(const Elf_Shdr &Sec); InputSectionBase *createInputSection(const Elf_Shdr &Sec); StringRef getSectionName(const Elf_Shdr &Sec); @@ -238,6 +240,7 @@ }; llvm::DenseMap VariableLoc; llvm::once_flag InitDwarfLine; + ArrayRef CGProfile; }; // LazyObjFile is analogous to ArchiveFile in the sense that Index: ELF/InputFiles.cpp =================================================================== --- ELF/InputFiles.cpp +++ ELF/InputFiles.cpp @@ -214,6 +214,14 @@ return ""; } +template void lld::elf::ObjFile::parseCGProfile() { + for (const Elf_CGProfile &CGPE : CGProfile) { + uint64_t &C = Config->CallGraphProfileSymbols[std::make_pair( + &getSymbol(CGPE.cgp_from), &getSymbol(CGPE.cgp_to))]; + C = std::max(C, (uint64_t)CGPE.cgp_weight); + } +} + // Returns "", "foo.a(bar.o)" or "baz.o". std::string lld::toString(const InputFile *F) { if (!F) @@ -287,6 +295,8 @@ // Read a symbol table. initializeSymbols(); + + parseCGProfile(); } // Sections with SHT_GROUP and comdat bits define comdat section groups. @@ -409,6 +419,11 @@ continue; const Elf_Shdr &Sec = ObjSections[I]; + if (Sec.sh_type == ELF::SHT_LLVM_CALL_GRAPH_PROFILE) + CGProfile = check( + this->getObj().template getSectionContentsAsArray( + &Sec)); + // SHF_EXCLUDE'ed sections are discarded by the linker. However, // if -r is given, we'll let the final link discard such sections. // This is compatible with GNU. Index: test/ELF/cgprofile-obj.s =================================================================== --- /dev/null +++ test/ELF/cgprofile-obj.s @@ -0,0 +1,41 @@ +# REQUIRES: x86 + +# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux %s -o %t +# RUN: ld.lld -e A %t -o %t2 +# RUN: llvm-readobj -symbols %t2 | FileCheck %s + + .section .text.D,"ax",@progbits +D: + retq + + .section .text.C,"ax",@progbits + .globl C +C: + retq + + .section .text.B,"ax",@progbits + .globl B +B: + retq + + .section .text.A,"ax",@progbits + .globl A +A: +Aa: + retq + + .cg_profile A, B, 10 + .cg_profile A, B, 10 + .cg_profile Aa, B, 80 + .cg_profile A, C, 40 + .cg_profile B, C, 30 + .cg_profile C, D, 90 + +# CHECK: Name: D +# CHECK-NEXT: Value: 0x201003 +# CHECK: Name: A +# CHECK-NEXT: Value: 0x201000 +# CHECK: Name: B +# CHECK-NEXT: Value: 0x201001 +# CHECK: Name: C +# CHECK-NEXT: Value: 0x201002