//===-- SourcePrinter.h - source interleaving utilities --------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #ifndef LLVM_TOOLS_LLVM_OBJDUMP_SOURCEPRINTER_H #define LLVM_TOOLS_LLVM_OBJDUMP_SOURCEPRINTER_H #include "llvm/ADT/IndexedMap.h" #include "llvm/ADT/StringSet.h" #include "llvm/DebugInfo/DWARF/DWARFContext.h" #include "llvm/DebugInfo/Symbolize/Symbolize.h" #include "llvm/Support/FormattedStream.h" #include #include namespace llvm { namespace objdump { /// Stores a single expression representing the location of a source-level /// variable, along with the PC range for which that expression is valid. struct LiveVariable { DWARFLocationExpression LocExpr; const char *VarName; DWARFUnit *Unit; const DWARFDie FuncDie; LiveVariable(const DWARFLocationExpression &LocExpr, const char *VarName, DWARFUnit *Unit, const DWARFDie FuncDie) : LocExpr(LocExpr), VarName(VarName), Unit(Unit), FuncDie(FuncDie) {} bool liveAtAddress(object::SectionedAddress Addr); void print(raw_ostream &OS, const MCRegisterInfo &MRI) const; }; /// Helper class for printing source variable locations alongside disassembly. class LiveVariablePrinter { // Information we want to track about one column in which we are printing a // variable live range. struct Column { unsigned VarIdx = NullVarIdx; bool LiveIn = false; bool LiveOut = false; bool MustDrawLabel = false; bool isActive() const { return VarIdx != NullVarIdx; } static constexpr unsigned NullVarIdx = std::numeric_limits::max(); }; // All live variables we know about in the object/image file. std::vector LiveVariables; // The columns we are currently drawing. IndexedMap ActiveCols; const MCRegisterInfo &MRI; const MCSubtargetInfo &STI; void addVariable(DWARFDie FuncDie, DWARFDie VarDie); void addFunction(DWARFDie D); // Get the column number (in characters) at which the first live variable // line should be printed. unsigned getIndentLevel() const; // Indent to the first live-range column to the right of the currently // printed line, and return the index of that column. // TODO: formatted_raw_ostream uses "column" to mean a number of characters // since the last \n, and we use it to mean the number of slots in which we // put live variable lines. Pick a less overloaded word. unsigned moveToFirstVarColumn(formatted_raw_ostream &OS); unsigned findFreeColumn(); public: LiveVariablePrinter(const MCRegisterInfo &MRI, const MCSubtargetInfo &STI) : LiveVariables(), ActiveCols(Column()), MRI(MRI), STI(STI) {} void dump() const; void addCompileUnit(DWARFDie D); /// Update to match the state of the instruction between ThisAddr and /// NextAddr. In the common case, any live range active at ThisAddr is /// live-in to the instruction, and any live range active at NextAddr is /// live-out of the instruction. If IncludeDefinedVars is false, then live /// ranges starting at NextAddr will be ignored. void update(object::SectionedAddress ThisAddr, object::SectionedAddress NextAddr, bool IncludeDefinedVars); enum class LineChar { RangeStart, RangeMid, RangeEnd, LabelVert, LabelCornerNew, LabelCornerActive, LabelHoriz, }; const char *getLineChar(LineChar C) const; /// Print live ranges to the right of an existing line. This assumes the /// line is not an instruction, so doesn't start or end any live ranges, so /// we only need to print active ranges or empty columns. If AfterInst is /// true, this is being printed after the last instruction fed to update(), /// otherwise this is being printed before it. void printAfterOtherLine(formatted_raw_ostream &OS, bool AfterInst); /// Print any live variable range info needed to the right of a /// non-instruction line of disassembly. This is where we print the variable /// names and expressions, with thin line-drawing characters connecting them /// to the live range which starts at the next instruction. If MustPrint is /// true, we have to print at least one line (with the continuation of any /// already-active live ranges) because something has already been printed /// earlier on this line. void printBetweenInsts(formatted_raw_ostream &OS, bool MustPrint); /// Print the live variable ranges to the right of a disassembled instruction. void printAfterInst(formatted_raw_ostream &OS); }; class SourcePrinter { protected: DILineInfo OldLineInfo; const object::ObjectFile *Obj = nullptr; std::unique_ptr Symbolizer; // File name to file contents of source. std::unordered_map> SourceCache; // Mark the line endings of the cached source. std::unordered_map> LineCache; // Keep track of missing sources. StringSet<> MissingSources; // Only emit 'invalid debug info' warning once. bool WarnedInvalidDebugInfo = false; private: bool cacheSource(const DILineInfo &LineInfoFile); void printLines(formatted_raw_ostream &OS, const DILineInfo &LineInfo, StringRef Delimiter, LiveVariablePrinter &LVP); void printSources(formatted_raw_ostream &OS, const DILineInfo &LineInfo, StringRef ObjectFilename, StringRef Delimiter, LiveVariablePrinter &LVP); public: SourcePrinter() = default; SourcePrinter(const object::ObjectFile *Obj, StringRef DefaultArch); virtual ~SourcePrinter() = default; virtual void printSourceLine(formatted_raw_ostream &OS, object::SectionedAddress Address, StringRef ObjectFilename, LiveVariablePrinter &LVP, StringRef Delimiter = "; "); }; } // namespace objdump } // namespace llvm #endif