//===-- LVLine.h ------------------------------------------------*- 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 // //===----------------------------------------------------------------------===// // // This file defines the LVLine class, which is used to describe a debug // information line. // //===----------------------------------------------------------------------===// #ifndef LLVM_DEBUGINFO_LOGICALVIEW_CORE_LVLINE_H #define LLVM_DEBUGINFO_LOGICALVIEW_CORE_LVLINE_H #include "llvm/DebugInfo/LogicalView/Core/LVElement.h" namespace llvm { namespace logicalview { enum class LVLineKind { IsBasicBlock, IsDiscriminator, IsEndSequence, IsEpilogueBegin, IsLineDebug, IsLineAssembler, IsNewStatement, // Shared with CodeView 'IsStatement' flag. IsPrologueEnd, IsAlwaysStepInto, // CodeView IsNeverStepInto, // CodeView LastEntry }; using LVLineKindSet = std::set; using LVLineDispatch = std::map; using LVLineRequest = std::vector; // Class to represent a logical line. class LVLine : public LVElement { // Typed bitvector with kinds for this line. LVProperties Kinds; static LVLineDispatch Dispatch; // Find the current line in the given 'Targets'. LVLine *findIn(const LVLines *Targets) const; public: LVLine() : LVElement(LVSubclassID::LV_LINE) { setIsLine(); setIncludeInPrint(); } LVLine(const LVLine &) = delete; LVLine &operator=(const LVLine &) = delete; virtual ~LVLine() = default; static bool classof(const LVElement *Element) { return Element->getSubclassID() == LVSubclassID::LV_LINE; } KIND(LVLineKind, IsBasicBlock); KIND(LVLineKind, IsDiscriminator); KIND(LVLineKind, IsEndSequence); KIND(LVLineKind, IsEpilogueBegin); KIND(LVLineKind, IsLineDebug); KIND(LVLineKind, IsLineAssembler); KIND(LVLineKind, IsNewStatement); KIND(LVLineKind, IsPrologueEnd); KIND(LVLineKind, IsAlwaysStepInto); KIND(LVLineKind, IsNeverStepInto); const char *kind() const override; // Use the offset to store the line address. uint64_t getAddress() const { return getOffset(); } void setAddress(uint64_t address) { setOffset(address); } // String used for printing objects with no line number. std::string noLineAsString(bool ShowZero = false) const override; // Line number for display; in the case of Inlined Functions, we use the // DW_AT_call_line attribute; otherwise use DW_AT_decl_line attribute. std::string lineNumberAsString(bool ShowZero = false) const override { return lineAsString(getLineNumber(), getDiscriminator(), ShowZero); } static LVLineDispatch &getDispatch() { return Dispatch; } // Iterate through the 'References' set and check that all its elements // are present in the 'Targets' set. For a missing element, mark its // parents as missing. static void markMissingParents(const LVLines *References, const LVLines *Targets); // Returns true if current line is logically equal to the given 'Line'. virtual bool equals(const LVLine *Line) const; // Returns true if the given 'References' are logically equal to the // given 'Targets'. static bool equals(const LVLines *References, const LVLines *Targets); // Report the current line as missing or added during comparison. void report(LVComparePass Pass) override; void print(raw_ostream &OS, bool Full = true) const override; void printExtra(raw_ostream &OS, bool Full = true) const override {} #if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP) void dump() const override { print(dbgs()); } #endif }; // Class to represent a DWARF line record object. class LVLineDebug final : public LVLine { // Discriminator value (DW_LNE_set_discriminator). The DWARF standard // defines the discriminator as an unsigned LEB128 integer. uint32_t Discriminator = 0; public: LVLineDebug() : LVLine() { setIsLineDebug(); } LVLineDebug(const LVLineDebug &) = delete; LVLineDebug &operator=(const LVLineDebug &) = delete; ~LVLineDebug() = default; // Additional line information. It includes attributes that describes // states in the machine instructions (basic block, end prologue, etc). std::string statesInfo(bool Formatted) const; // Access DW_LNE_set_discriminator attribute. uint32_t getDiscriminator() const override { return Discriminator; } void setDiscriminator(uint32_t Value) override { Discriminator = Value; setIsDiscriminator(); } // Returns true if current line is logically equal to the given 'Line'. bool equals(const LVLine *Line) const override; void printExtra(raw_ostream &OS, bool Full = true) const override; }; // Class to represent an assembler line extracted from the text section. class LVLineAssembler final : public LVLine { public: LVLineAssembler() : LVLine() { setIsLineAssembler(); } LVLineAssembler(const LVLineAssembler &) = delete; LVLineAssembler &operator=(const LVLineAssembler &) = delete; ~LVLineAssembler() = default; // Print blanks as the line number. std::string noLineAsString(bool ShowZero) const override { return std::string(8, ' '); }; // Returns true if current line is logically equal to the given 'Line'. bool equals(const LVLine *Line) const override; void printExtra(raw_ostream &OS, bool Full = true) const override; }; } // end namespace logicalview } // end namespace llvm #endif // LLVM_DEBUGINFO_LOGICALVIEW_CORE_LVLINE_H