23#define DEBUG_TYPE "correlator"
37 std::string ExpectedSectionName =
40 ExpectedSectionName = StripSuffix(ExpectedSectionName);
41 for (
auto &Section : Obj.sections()) {
48 "could not find section (" +
Twine(ExpectedSectionName) +
")");
60 auto C = std::make_unique<Context>();
62 if (
auto Err = CountersSection.takeError())
63 return std::move(Err);
67 if (
auto Err = DataSection.takeError())
68 return std::move(Err);
69 auto DataOrErr = DataSection->getContents();
71 return DataOrErr.takeError();
73 if (
auto Err = NameSection.takeError())
74 return std::move(Err);
75 auto NameOrErr = NameSection->getContents();
77 return NameOrErr.takeError();
78 C->DataStart = DataOrErr->data();
79 C->DataEnd = DataOrErr->data() + DataOrErr->size();
80 C->NameStart = NameOrErr->data();
81 C->NameSize = NameOrErr->size();
84 std::string FullSectionName =
91 MachO->fixupTable(Err)) {
92 if (Entry.isRebase() && Entry.segmentName() == SegmentAndSection[0] &&
93 Entry.sectionName() == SegmentAndSection[1]) {
94 C->MachOFixups[Entry.address() - DataSection->getAddress()] =
99 return std::move(Err);
103 C->CountersSectionStart = CountersSection->getAddress();
104 C->CountersSectionEnd =
C->CountersSectionStart + CountersSection->getSize();
107 if (
auto E = BitmapSection.takeError()) {
110 C->BitmapSectionStart = 0;
111 C->BitmapSectionEnd = 0;
113 C->BitmapSectionStart = BitmapSection->getAddress();
114 C->BitmapSectionEnd =
C->BitmapSectionStart + BitmapSection->getSize();
119 ++
C->CountersSectionStart;
120 if (
C->BitmapSectionStart)
121 ++
C->BitmapSectionStart;
132 std::optional<std::string> Path;
137 "unsupported profile binary correlation when there is no build ID "
142 "unsupported profile binary correlation when there are multiple "
143 "build IDs in a profile");
155 auto DsymObjectsOrErr =
157 if (
auto Err = DsymObjectsOrErr.takeError())
158 return std::move(Err);
159 if (!DsymObjectsOrErr->empty()) {
162 if (DsymObjectsOrErr->size() > 1)
165 "using multiple objects is not yet supported");
166 Filename = *DsymObjectsOrErr->begin();
169 if (
auto Err = BufferOrErr.takeError())
170 return std::move(Err);
172 return get(std::move(*BufferOrErr), FileKind);
176 if (
auto Err = BufferOrErr.takeError())
177 return std::move(Err);
179 return get(std::move(*BufferOrErr), FileKind);
183 "unsupported correlation kind (only DWARF debug info and Binary format "
184 "(ELF/COFF) are supported)");
189 ProfCorrelatorKind FileKind) {
191 if (
auto Err = BinOrErr.takeError())
192 return std::move(Err);
195 auto CtxOrErr =
Context::get(std::move(Buffer), *Obj, FileKind);
196 if (
auto Err = CtxOrErr.takeError())
197 return std::move(Err);
198 auto T = Obj->makeTriple();
212 return C->getDataSize();
214 return C->getDataSize();
222 std::unique_ptr<InstrProfCorrelator::Context>
Ctx)
227 std::unique_ptr<InstrProfCorrelator::Context>
Ctx)
241template <
class IntPtrT>
244 std::unique_ptr<InstrProfCorrelator::Context>
Ctx,
247 if (Obj.isELF() || Obj.isMachO()) {
249 return std::make_unique<DwarfInstrProfCorrelator<IntPtrT>>(
250 std::move(DICtx), std::move(
Ctx));
254 "unsupported debug info format (only DWARF is supported)");
256 if (Obj.isELF() || Obj.isCOFF() || Obj.isMachO())
257 return std::make_unique<BinaryInstrProfCorrelator<IntPtrT>>(std::move(
Ctx));
260 "unsupported binary format (only ELF, COFF, and Mach-O are supported)");
263template <
class IntPtrT>
267 if (this->
Data.empty())
270 "could not find any profile data metadata in correlated file");
272 this->CounterOffsets.clear();
273 this->BitmapOffsets.clear();
300 static const bool flow =
false;
303template <
class IntPtrT>
308 if (
Data.Probes.empty())
311 "could not find any profile data metadata in debug info");
317template <
class IntPtrT>
320 IntPtrT BitmapOffset, IntPtrT FunctionPtr,
uint32_t NumCounters,
323 if (NumCounters && !CounterOffsets.insert(CounterOffset).second)
346template <
class IntPtrT>
347std::optional<uint64_t>
348DwarfInstrProfCorrelator<IntPtrT>::getLocation(
const DWARFDie &Die)
const {
349 auto Locations = Die.
getLocations(dwarf::DW_AT_location);
356 for (
auto &Location : *Locations) {
359 for (
auto &
Op : Expr) {
360 if (
Op.getCode() == dwarf::DW_OP_addr)
361 return Op.getRawOperand(0);
362 if (
Op.getCode() == dwarf::DW_OP_addrx) {
364 if (
auto SA = DU.getAddrOffsetSectionItem(Index))
372template <
class IntPtrT>
373bool DwarfInstrProfCorrelator<IntPtrT>::isDIEOfProbe(
const DWARFDie &Die,
378 if (Die.
getTag() != dwarf::DW_TAG_variable)
380 if (!ParentDie.isSubprogramDIE())
389template <
class IntPtrT>
390std::optional<std::pair<InstrProfCorrelator::Probe, IntPtrT>>
391DwarfInstrProfCorrelator<IntPtrT>::addCountersToDataProbe(
392 const DWARFDie &Die,
const bool UnlimitedWarnings,
393 int &NumSuppressedWarnings) {
394 std::optional<const char *> FunctionName;
395 std::optional<uint64_t> CFGHash;
396 std::optional<uint64_t> CounterPtr =
getLocation(Die);
399 std::optional<uint64_t> NumCounters;
401 if (Child.getTag() != dwarf::DW_TAG_LLVM_annotation)
403 auto AnnotationFormName = Child.find(dwarf::DW_AT_name);
404 auto AnnotationFormValue = Child.find(dwarf::DW_AT_const_value);
405 if (!AnnotationFormName || !AnnotationFormValue)
407 auto AnnotationNameOrErr = AnnotationFormName->getAsCString();
408 if (
auto Err = AnnotationNameOrErr.takeError()) {
412 StringRef AnnotationName = *AnnotationNameOrErr;
414 if (
auto EC = AnnotationFormValue->getAsCString().moveInto(FunctionName))
417 CFGHash = AnnotationFormValue->getAsUnsignedConstant();
418 }
else if (AnnotationName ==
420 NumCounters = AnnotationFormValue->getAsUnsignedConstant();
424 if (!FunctionPtr && !CounterPtr)
426 if (!FunctionName || !CFGHash || !CounterPtr || !NumCounters) {
427 if (UnlimitedWarnings || ++NumSuppressedWarnings < 1) {
429 <<
": CFGHash=" << CFGHash
430 <<
" CounterPtr=" << CounterPtr
431 <<
" NumCounters=" << NumCounters <<
"\n";
437 uint64_t CountersEnd = this->Ctx->CountersSectionEnd;
438 if (*CounterPtr < CountersStart || *CounterPtr >= CountersEnd) {
439 if (UnlimitedWarnings || ++NumSuppressedWarnings < 1) {
441 "CounterPtr out of range for function %s: Actual=0x%x "
442 "Expected=[0x%x, 0x%x)\n",
448 if (!FunctionPtr && (UnlimitedWarnings || ++NumSuppressedWarnings < 1)) {
458 P.FunctionName = *FunctionName;
460 P.LinkageName =
Name;
461 P.CFGHash = *CFGHash;
462 P.CounterOffset = CounterOffset;
463 P.NumCounters = *NumCounters;
464 auto FilePath = FnDie.getDeclFile(
466 if (!FilePath.empty())
467 P.FilePath = FilePath;
468 if (
auto LineNumber = FnDie.getDeclLine())
469 P.LineNumber = LineNumber;
471 return std::optional<std::pair<InstrProfCorrelator::Probe, IntPtrT>>(
472 {
P, FunctionPtr.value_or(0)});
475template <
class IntPtrT>
476std::optional<std::pair<InstrProfCorrelator::Probe, IntPtrT>>
477DwarfInstrProfCorrelator<IntPtrT>::addBitmapToDataProbe(
478 const DWARFDie &Die,
const bool UnlimitedWarnings,
479 int &NumSuppressedWarnings) {
480 std::optional<const char *> FunctionName;
481 std::optional<uint64_t> BitmapPtr =
getLocation(Die);
484 if (Child.getTag() != dwarf::DW_TAG_LLVM_annotation)
486 auto AnnotationFormName = Child.find(dwarf::DW_AT_name);
487 auto AnnotationFormValue = Child.find(dwarf::DW_AT_const_value);
488 if (!AnnotationFormName || !AnnotationFormValue)
490 auto AnnotationNameOrErr = AnnotationFormName->getAsCString();
491 if (
auto Err = AnnotationNameOrErr.takeError()) {
495 StringRef AnnotationName = *AnnotationNameOrErr;
497 if (
auto EC = AnnotationFormValue->getAsCString().moveInto(FunctionName))
499 }
else if (AnnotationName ==
501 std::optional<uint64_t> NumBitmapBits =
502 AnnotationFormValue->getAsUnsignedConstant();
507 if (UnlimitedWarnings || ++NumSuppressedWarnings < 1) {
509 <<
" BitmapPtr=" << BitmapPtr
515 uint64_t BitmapStart = this->Ctx->BitmapSectionStart;
516 uint64_t BitmapEnd = this->Ctx->BitmapSectionEnd;
520 "could not find profile bitmap section in correlated file");
523 if (*BitmapPtr < BitmapStart || (*BitmapPtr >= BitmapEnd &&
NumBitmapBytes)) {
524 if (UnlimitedWarnings || ++NumSuppressedWarnings < 1) {
526 "BitmapPtr out of range for function %s: Actual=0x%x "
527 "Expected=[0x%x, 0x%x)\n",
528 *FunctionName, *BitmapPtr, BitmapStart, BitmapEnd);
535 IntPtrT BitmapOffset = *BitmapPtr - BitmapStart;
537 P.FunctionName = *FunctionName;
538 P.BitmapOffset = BitmapOffset;
541 return std::optional<std::pair<InstrProfCorrelator::Probe, IntPtrT>>({
P, 0});
544template <
class IntPtrT>
545void DwarfInstrProfCorrelator<IntPtrT>::correlateProfileDataImpl(
552 bool UnlimitedWarnings = (MaxWarnings == 0);
554 int NumSuppressedWarnings = -MaxWarnings;
556 auto MaybeAddProbe = [&](
DWARFDie Die) {
557 std::optional<std::pair<InstrProfCorrelator::Probe, IntPtrT>> ProbeData;
560 addCountersToDataProbe(Die, UnlimitedWarnings, NumSuppressedWarnings);
563 addBitmapToDataProbe(Die, UnlimitedWarnings, NumSuppressedWarnings);
566 auto [Probe, FunctionPtr] = *ProbeData;
569 Probes.
try_emplace(Probe.FunctionName, Probe, FunctionPtr);
571 auto &
P = It->second.first;
573 P.LinkageName = Probe.LinkageName;
574 P.CFGHash = Probe.CFGHash;
575 P.CounterOffset = Probe.CounterOffset;
576 P.NumCounters = Probe.NumCounters;
577 P.FilePath = Probe.FilePath;
578 P.LineNumber = Probe.LineNumber;
580 P.BitmapOffset = Probe.BitmapOffset;
581 P.NumBitmapBytes = Probe.NumBitmapBytes;
585 for (
auto &
CU : DICtx->normal_units())
586 for (
const auto &Entry :
CU->dies())
588 for (
auto &
CU : DICtx->dwo_units())
589 for (
const auto &Entry :
CU->dies())
592 for (
const auto &[FunctionName, ProbeData] : Probes) {
593 const auto &[Probe, FunctionPtr] = ProbeData;
595 Data->Probes.push_back(Probe);
597 this->NamesVec.push_back(FunctionName);
599 this->addDataProbe(NameRef, Probe.CFGHash, Probe.CounterOffset,
600 Probe.BitmapOffset, FunctionPtr, Probe.NumCounters,
601 Probe.NumBitmapBytes);
604 if (!UnlimitedWarnings && NumSuppressedWarnings > 0)
606 NumSuppressedWarnings);
609template <
class IntPtrT>
610Error DwarfInstrProfCorrelator<IntPtrT>::correlateProfileNameImpl() {
611 if (this->NamesVec.empty()) {
614 "could not find any profile name metadata in debug info");
622template <
class IntPtrT>
623void BinaryInstrProfCorrelator<IntPtrT>::correlateProfileDataImpl(
626 bool UnlimitedWarnings = (MaxWarnings == 0);
628 int NumSuppressedWarnings = -MaxWarnings;
630 const RawProfData *
DataStart = (
const RawProfData *)this->Ctx->DataStart;
631 const RawProfData *DataEnd = (
const RawProfData *)this->Ctx->DataEnd;
633 for (
const RawProfData *
I =
DataStart;
I < DataEnd; ++
I) {
634 uint64_t CounterPtr = this->
template maybeSwap<IntPtrT>(
I->CounterPtr);
636 uint64_t CountersEnd = this->Ctx->CountersSectionEnd;
638 uint64_t BitmapPtr = this->
template maybeSwap<IntPtrT>(
I->BitmapPtr);
639 uint64_t BitmapStart = this->Ctx->BitmapSectionStart;
640 uint64_t BitmapEnd = this->Ctx->BitmapSectionEnd;
641 if (!BitmapStart && !BitmapEnd &&
I->NumBitmapBytes) {
644 "could not find profile bitmap section in correlated file");
647 if (!this->Ctx->MachOFixups.empty()) {
649 auto It = this->Ctx->MachOFixups.find(
Offset);
650 if (It != this->Ctx->MachOFixups.end()) {
652 }
else if (UnlimitedWarnings || ++NumSuppressedWarnings < 1) {
654 "Mach-O fixup not found for covdata offset 0x%llx\n",
Offset);
659 GetPtrByOffset(CounterOffset, CounterPtr);
660 GetPtrByOffset(BitmapOffset, BitmapPtr);
662 if (CounterPtr < CountersStart || CounterPtr >= CountersEnd) {
663 if (UnlimitedWarnings || ++NumSuppressedWarnings < 1) {
665 <<
format(
"CounterPtr out of range for function: Actual=0x%x "
666 "Expected=[0x%x, 0x%x) at data offset=0x%x\n",
671 if (
I->NumBitmapBytes &&
672 (BitmapPtr < BitmapStart || BitmapPtr >= BitmapEnd)) {
673 if (UnlimitedWarnings || ++NumSuppressedWarnings < 1) {
675 <<
format(
"BitmapPtr out of range for function: Actual=0x%x "
676 "Expected=[0x%x, 0x%x) at data offset=0x%x\n",
677 BitmapPtr, BitmapStart, BitmapEnd,
684 IntPtrT BitmapOffset = BitmapPtr - BitmapStart;
685 this->addDataProbe(
I->NameRef,
I->FuncHash, CounterOffset, BitmapOffset,
686 I->FunctionPointer,
I->NumCounters,
I->NumBitmapBytes);
690template <
class IntPtrT>
691Error BinaryInstrProfCorrelator<IntPtrT>::correlateProfileNameImpl() {
692 if (this->Ctx->NameSize == 0) {
695 "could not find any profile data metadata in object file");
697 this->Names.append(this->Ctx->NameStart, this->Ctx->NameSize);
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
static Expected< object::SectionRef > getInstrProfSection(const object::ObjectFile &Obj, InstrProfSectKind IPSK)
Get profile section.
static constexpr StringLiteral Filename
static MemoryLocation getLocation(Instruction *I)
Represent a constant reference to an array (0 or more elements consecutively in memory),...
const T & front() const
Get the first element.
size_t size() const
Get the array size.
bool empty() const
Check if the array is empty.
static std::unique_ptr< DWARFContext > create(const object::ObjectFile &Obj, ProcessDebugRelocations RelocAction=ProcessDebugRelocations::Process, const LoadedObjectInfo *L=nullptr, std::string DWPName="", std::function< void(Error)> RecoverableErrorHandler=WithColor::defaultErrorHandler, std::function< void(Error)> WarningHandler=WithColor::defaultWarningHandler, bool ThreadSafe=false)
Utility class that carries the DWARF compile/type unit and the debug info entry in an object.
iterator_range< iterator > children() const
LLVM_ABI DWARFDie getParent() const
Get the parent of this DIE object.
DWARFUnit * getDwarfUnit() const
LLVM_ABI const char * getName(DINameKind Kind) const
Return the DIE name resolving DW_AT_specification or DW_AT_abstract_origin references if necessary.
dwarf::Tag getTag() const
LLVM_ABI Expected< DWARFLocationExpressionsVector > getLocations(dwarf::Attribute Attr) const
bool isNULL() const
Returns true for a valid DIE that terminates a sibling chain.
LLVM_ABI void dump(raw_ostream &OS, unsigned indent=0, DIDumpOptions DumpOpts=DIDumpOptions()) const
Dump the DIE and all of its attributes to the supplied stream.
uint8_t getAddressByteSize() const
Lightweight error class with error context and mandatory checking.
static ErrorSuccess success()
Create a success value.
Tagged union holding either a T or a Error.
InstrProfCorrelatorImpl - A child of InstrProfCorrelator with a template pointer type so that the Pro...
static llvm::Expected< std::unique_ptr< InstrProfCorrelatorImpl< IntPtrT > > > get(std::unique_ptr< InstrProfCorrelator::Context > Ctx, const object::ObjectFile &Obj, ProfCorrelatorKind FileKind)
virtual Error correlateProfileNameImpl()=0
virtual void correlateProfileDataImpl(int MaxWarnings, InstrProfCorrelator::CorrelationData *Data=nullptr)=0
std::vector< RawInstrProf::ProfileData< IntPtrT > > Data
Error correlateProfileData(int MaxWarnings) override
Construct a ProfileData vector used to correlate raw instrumentation data to their functions.
static bool classof(const InstrProfCorrelator *C)
InstrProfCorrelatorImpl(std::unique_ptr< InstrProfCorrelator::Context > Ctx)
void addDataProbe(uint64_t FunctionName, uint64_t CFGHash, IntPtrT CounterOffset, IntPtrT BitmapOffset, IntPtrT FunctionPtr, uint32_t NumCounters, uint32_t NumBitmapBytes)
T maybeSwap(T Value) const
Error dumpYaml(int MaxWarnings, raw_ostream &OS) override
Process debug info and dump the correlation data.
InstrProfCorrelator - A base class used to create raw instrumentation data to their functions.
static LLVM_ABI const char * FunctionNameAttributeName
static LLVM_ABI const char * CFGHashAttributeName
InstrProfCorrelator(InstrProfCorrelatorKind K, std::unique_ptr< Context > Ctx)
std::vector< std::string > NamesVec
static LLVM_ABI const char * NumCountersAttributeName
ProfCorrelatorKind
Indicate if we should use the debug info or profile metadata sections to correlate.
const std::unique_ptr< Context > Ctx
LLVM_ABI std::optional< size_t > getDataSize() const
Return the number of ProfileData elements.
static LLVM_ABI llvm::Expected< std::unique_ptr< InstrProfCorrelator > > get(StringRef Filename, ProfCorrelatorKind FileKind, const object::BuildIDFetcher *BIDFetcher=nullptr, const ArrayRef< llvm::object::BuildID > BIs={})
static LLVM_ABI const char * NumBitmapBitsAttributeName
This class implements a map that also provides access to all stored values in a deterministic order.
std::pair< iterator, bool > try_emplace(const KeyT &Key, Ts &&...Args)
static ErrorOr< std::unique_ptr< MemoryBuffer > > getFile(const Twine &Filename, bool IsText=false, bool RequiresNullTerminator=true, bool IsVolatile=false, std::optional< Align > Alignment=std::nullopt)
Open the specified file as a MemoryBuffer, returning a new MemoryBuffer if successful,...
This is a 'vector' (really, a variable-sized array), optimized for the case when the array is small.
StringMap - This is an unconventional map that is specialized for handling keys that are "strings",...
Represent a constant reference to a string, i.e.
std::pair< StringRef, StringRef > split(char Separator) const
Split into two substrings around the first occurrence of a separator character.
bool starts_with(StringRef Prefix) const
Check if this string starts with the given Prefix.
Twine - A lightweight data structure for efficiently representing the concatenation of temporary valu...
static LLVM_ABI raw_ostream & warning()
Convenience method for printing "warning: " to stderr.
BuildIDFetcher searches local cache directories for debug info.
virtual std::optional< std::string > fetch(BuildIDRef BuildID) const
Returns the path to the debug file with the given build ID.
static Expected< std::vector< std::string > > findDsymObjectMembers(StringRef Path)
If the input path is a .dSYM bundle (as created by the dsymutil tool), return the paths to the object...
This class is the base class for all object file types.
This class implements an extremely fast bulk output stream that can only output to a stream.
void mapOptional(StringRef Key, T &Val)
void mapRequired(StringRef Key, T &Val)
The Output class is used to generate a yaml document from in-memory structs and vectors.
@ C
The default llvm calling convention, compatible with C.
uint64_t ComputeHash(StringRef K)
std::optional< uint64_t > toAddress(const std::optional< DWARFFormValue > &V)
Take an optional DWARFFormValue and try to extract an address.
LLVM_ABI Expected< std::unique_ptr< Binary > > createBinary(MemoryBufferRef Source, LLVMContext *Context=nullptr, bool InitContent=true)
Create a Binary from Source, autodetecting the file type.
constexpr bool IsLittleEndianHost
This is an optimization pass for GlobalISel generic memory operations.
RelativeUniformCounterPtr ValuesPtrExpr NumBitmapBytes
decltype(auto) dyn_cast(const From &Val)
dyn_cast<X> - Return the argument parameter cast to the specified type.
StringRef getInstrProfBitmapVarPrefix()
Return the name prefix of profile bitmap variables.
LLVM_ABI std::string getInstrProfSectionName(InstrProfSectKind IPSK, Triple::ObjectFormatType OF, bool AddSegmentInfo=true)
Return the name of the profile section corresponding to IPSK.
StringRef getInstrProfCountersVarPrefix()
Return the name prefix of profile counter variables.
LLVM_ABI raw_ostream & dbgs()
dbgs() - This returns a reference to a raw_ostream for debugging messages.
constexpr uint64_t alignTo(uint64_t Size, Align A)
Returns a multiple of A needed to store Size bytes.
RelativeUniformCounterPtr ValuesPtrExpr VTableAddr CountersStart
format_object< Ts... > format(const char *Fmt, const Ts &... Vals)
These are helper functions used to produce formatted output.
Error make_error(ArgTs &&... Args)
Make a Error instance representing failure using the given error info type.
@ unable_to_correlate_profile
DWARFExpression::Operation Op
Expected< T > errorOrToExpected(ErrorOr< T > &&EO)
Convert an ErrorOr<T> to an Expected<T>.
LLVM_ABI Error collectGlobalObjectNameStrings(ArrayRef< std::string > NameStrs, bool doCompression, std::string &Result)
Given a vector of strings (names of global objects like functions or, virtual tables) NameStrs,...
void toHex(ArrayRef< uint8_t > Input, bool LowerCase, SmallVectorImpl< char > &Output)
Convert buffer Input to its hexadecimal representation. The returned string is double the size of Inp...
OutputIt move(R &&Range, OutputIt Out)
Provide wrappers to std::move which take ranges instead of having to pass begin/end explicitly.
RelativeUniformCounterPtr ValuesPtrExpr VTableAddr DataStart
void consumeError(Error Err)
Consume a Error without doing anything.
Implement std::hash so that hash_code can be used in STL containers.
std::unique_ptr< MemoryBuffer > Buffer
static LLVM_ABI llvm::Expected< std::unique_ptr< Context > > get(std::unique_ptr< MemoryBuffer > Buffer, object::ObjectFile &Obj, ProfCorrelatorKind FileKind)
This class should be specialized by any type that needs to be converted to/from a YAML mapping.
This class should be specialized by any type for which vectors of that type need to be converted to/f...
static void mapping(yaml::IO &io, InstrProfCorrelator::CorrelationData &Data)
static void mapping(yaml::IO &io, InstrProfCorrelator::Probe &P)