322 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			322 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
//===--- unittests/DebugInfo/DWARF/DwarfGenerator.h -------------*- C++ -*-===//
 | 
						|
//
 | 
						|
//                     The LLVM Compiler Infrastructure
 | 
						|
//
 | 
						|
// This file is distributed under the University of Illinois Open Source
 | 
						|
// License. See LICENSE.TXT for details.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
//
 | 
						|
// A file that can generate DWARF debug info for unit tests.
 | 
						|
//
 | 
						|
//===----------------------------------------------------------------------===//
 | 
						|
 | 
						|
#ifndef LLVM_UNITTESTS_DEBUG_INFO_DWARF_DWARFGENERATOR_H
 | 
						|
#define LLVM_UNITTESTS_DEBUG_INFO_DWARF_DWARFGENERATOR_H
 | 
						|
 | 
						|
#include "llvm/ADT/StringRef.h"
 | 
						|
#include "llvm/CodeGen/DIE.h"
 | 
						|
#include "llvm/DebugInfo/DWARF/DWARFDebugLine.h"
 | 
						|
#include "llvm/Support/Error.h"
 | 
						|
 | 
						|
#include <memory>
 | 
						|
#include <string>
 | 
						|
#include <tuple>
 | 
						|
#include <vector>
 | 
						|
 | 
						|
namespace llvm {
 | 
						|
 | 
						|
class AsmPrinter;
 | 
						|
class DIE;
 | 
						|
class DIEAbbrev;
 | 
						|
class DwarfStringPool;
 | 
						|
class MCAsmBackend;
 | 
						|
class MCAsmInfo;
 | 
						|
class MCCodeEmitter;
 | 
						|
class MCContext;
 | 
						|
struct MCDwarfLineTableParams;
 | 
						|
class MCInstrInfo;
 | 
						|
class MCRegisterInfo;
 | 
						|
class MCStreamer;
 | 
						|
class MCSubtargetInfo;
 | 
						|
class raw_fd_ostream;
 | 
						|
class TargetLoweringObjectFile;
 | 
						|
class TargetMachine;
 | 
						|
class Triple;
 | 
						|
 | 
						|
namespace dwarfgen {
 | 
						|
 | 
						|
class Generator;
 | 
						|
class CompileUnit;
 | 
						|
 | 
						|
/// A DWARF debug information entry class used to generate DWARF DIEs.
 | 
						|
///
 | 
						|
/// This class is used to quickly generate DWARF debug information by creating
 | 
						|
/// child DIEs or adding attributes to the current DIE. Instances of this class
 | 
						|
/// are created from the compile unit (dwarfgen::CompileUnit::getUnitDIE()) or
 | 
						|
/// by calling dwarfgen::DIE::addChild(...) and using the returned DIE object.
 | 
						|
class DIE {
 | 
						|
  dwarfgen::CompileUnit *CU;
 | 
						|
  llvm::DIE *Die;
 | 
						|
 | 
						|
protected:
 | 
						|
  friend class Generator;
 | 
						|
  friend class CompileUnit;
 | 
						|
 | 
						|
  DIE(CompileUnit *U = nullptr, llvm::DIE *D = nullptr) : CU(U), Die(D) {}
 | 
						|
 | 
						|
  /// Called with a compile/type unit relative offset prior to generating the
 | 
						|
  /// DWARF debug info.
 | 
						|
  ///
 | 
						|
  /// \param CUOffset the compile/type unit relative offset where the
 | 
						|
  /// abbreviation code for this DIE will be encoded.
 | 
						|
  unsigned computeSizeAndOffsets(unsigned CUOffset);
 | 
						|
 | 
						|
public:
 | 
						|
  /// Add an attribute value that has no value.
 | 
						|
  ///
 | 
						|
  /// \param Attr a dwarf::Attribute enumeration value or any uint16_t that
 | 
						|
  /// represents a user defined DWARF attribute.
 | 
						|
  /// \param Form the dwarf::Form to use when encoding the attribute. This is
 | 
						|
  /// only used with the DW_FORM_flag_present form encoding.
 | 
						|
  void addAttribute(uint16_t Attr, dwarf::Form Form);
 | 
						|
 | 
						|
  /// Add an attribute value to be encoded as a DIEInteger
 | 
						|
  ///
 | 
						|
  /// \param Attr a dwarf::Attribute enumeration value or any uint16_t that
 | 
						|
  /// represents a user defined DWARF attribute.
 | 
						|
  /// \param Form the dwarf::Form to use when encoding the attribute.
 | 
						|
  /// \param U the unsigned integer to encode.
 | 
						|
  void addAttribute(uint16_t Attr, dwarf::Form Form, uint64_t U);
 | 
						|
 | 
						|
  /// Add an attribute value to be encoded as a DIEExpr
 | 
						|
  ///
 | 
						|
  /// \param Attr a dwarf::Attribute enumeration value or any uint16_t that
 | 
						|
  /// represents a user defined DWARF attribute.
 | 
						|
  /// \param Form the dwarf::Form to use when encoding the attribute.
 | 
						|
  /// \param Expr the MC expression used to compute the value.
 | 
						|
  void addAttribute(uint16_t Attr, dwarf::Form Form, const MCExpr &Expr);
 | 
						|
 | 
						|
  /// Add an attribute value to be encoded as a DIEString or DIEInlinedString.
 | 
						|
  ///
 | 
						|
  /// \param Attr a dwarf::Attribute enumeration value or any uint16_t that
 | 
						|
  /// represents a user defined DWARF attribute.
 | 
						|
  /// \param Form the dwarf::Form to use when encoding the attribute. The form
 | 
						|
  /// must be one of DW_FORM_strp or DW_FORM_string.
 | 
						|
  /// \param String the string to encode.
 | 
						|
  void addAttribute(uint16_t Attr, dwarf::Form Form, StringRef String);
 | 
						|
 | 
						|
  /// Add an attribute value to be encoded as a DIEEntry.
 | 
						|
  ///
 | 
						|
  /// DIEEntry attributes refer to other llvm::DIE objects that have been
 | 
						|
  /// created.
 | 
						|
  ///
 | 
						|
  /// \param Attr a dwarf::Attribute enumeration value or any uint16_t that
 | 
						|
  /// represents a user defined DWARF attribute.
 | 
						|
  /// \param Form the dwarf::Form to use when encoding the attribute. The form
 | 
						|
  /// must be one of DW_FORM_strp or DW_FORM_string.
 | 
						|
  /// \param RefDie the DIE that this attriute refers to.
 | 
						|
  void addAttribute(uint16_t Attr, dwarf::Form Form, dwarfgen::DIE &RefDie);
 | 
						|
 | 
						|
  /// Add an attribute value to be encoded as a DIEBlock.
 | 
						|
  ///
 | 
						|
  /// DIEBlock attributes refers to binary data that is stored as the
 | 
						|
  /// attribute's value.
 | 
						|
  ///
 | 
						|
  /// \param Attr a dwarf::Attribute enumeration value or any uint16_t that
 | 
						|
  /// represents a user defined DWARF attribute.
 | 
						|
  /// \param Form the dwarf::Form to use when encoding the attribute. The form
 | 
						|
  /// must be one of DW_FORM_strp or DW_FORM_string.
 | 
						|
  /// \param P a pointer to the data to store as the attribute value.
 | 
						|
  /// \param S the size in bytes of the data pointed to by P .
 | 
						|
  void addAttribute(uint16_t Attr, dwarf::Form Form, const void *P, size_t S);
 | 
						|
 | 
						|
  /// Add a DW_AT_str_offsets_base attribute to this DIE.
 | 
						|
  void addStrOffsetsBaseAttribute();
 | 
						|
 | 
						|
  /// Add a new child to this DIE object.
 | 
						|
  ///
 | 
						|
  /// \param Tag the dwarf::Tag to assing to the llvm::DIE object.
 | 
						|
  /// \returns the newly created DIE object that is now a child owned by this
 | 
						|
  /// object.
 | 
						|
  dwarfgen::DIE addChild(dwarf::Tag Tag);
 | 
						|
};
 | 
						|
 | 
						|
/// A DWARF compile unit used to generate DWARF compile/type units.
 | 
						|
///
 | 
						|
/// Instances of these classes are created by instances of the Generator
 | 
						|
/// class. All information required to generate a DWARF compile unit is
 | 
						|
/// contained inside this class.
 | 
						|
class CompileUnit {
 | 
						|
  Generator &DG;
 | 
						|
  BasicDIEUnit DU;
 | 
						|
 | 
						|
public:
 | 
						|
  CompileUnit(Generator &D, uint16_t V, uint8_t A)
 | 
						|
      : DG(D), DU(V, A, dwarf::DW_TAG_compile_unit) {}
 | 
						|
  DIE getUnitDIE();
 | 
						|
  Generator &getGenerator() { return DG; }
 | 
						|
  uint64_t getOffset() const { return DU.getDebugSectionOffset(); }
 | 
						|
  uint64_t getLength() const { return DU.getLength(); }
 | 
						|
  uint16_t getVersion() const { return DU.getDwarfVersion(); }
 | 
						|
  uint16_t getAddressSize() const { return DU.getAddressSize(); }
 | 
						|
  void setOffset(uint64_t Offset) { DU.setDebugSectionOffset(Offset); }
 | 
						|
  void setLength(uint64_t Length) { DU.setLength(Length); }
 | 
						|
};
 | 
						|
 | 
						|
/// A DWARF line unit-like class used to generate DWARF line units.
 | 
						|
///
 | 
						|
/// Instances of this class are created by instances of the Generator class.
 | 
						|
class LineTable {
 | 
						|
public:
 | 
						|
  enum ValueLength { Byte = 1, Half = 2, Long = 4, Quad = 8, ULEB, SLEB };
 | 
						|
 | 
						|
  struct ValueAndLength {
 | 
						|
    uint64_t Value;
 | 
						|
    ValueLength Length;
 | 
						|
  };
 | 
						|
 | 
						|
  LineTable(uint16_t Version, dwarf::DwarfFormat Format, uint8_t AddrSize,
 | 
						|
            uint8_t SegSize = 0)
 | 
						|
      : Version(Version), Format(Format), AddrSize(AddrSize), SegSize(SegSize) {
 | 
						|
    assert(Version >= 2 && Version <= 5 && "unsupported version");
 | 
						|
  }
 | 
						|
 | 
						|
  // Create a Prologue suitable to pass to setPrologue, with a single file and
 | 
						|
  // include directory entry.
 | 
						|
  DWARFDebugLine::Prologue createBasicPrologue() const;
 | 
						|
 | 
						|
  // Set or replace the current prologue with the specified prologue. If no
 | 
						|
  // prologue is set, a default one will be used when generating.
 | 
						|
  void setPrologue(DWARFDebugLine::Prologue NewPrologue);
 | 
						|
  // Used to write an arbitrary payload instead of the standard prologue. This
 | 
						|
  // is useful if you wish to test handling of corrupt .debug_line sections.
 | 
						|
  void setCustomPrologue(ArrayRef<ValueAndLength> NewPrologue);
 | 
						|
 | 
						|
  // Add a byte to the program, with the given value. This can be used to
 | 
						|
  // specify a special opcode, or to add arbitrary contents to the section.
 | 
						|
  void addByte(uint8_t Value);
 | 
						|
  // Add a standard opcode to the program. The opcode and operands do not have
 | 
						|
  // to be valid.
 | 
						|
  void addStandardOpcode(uint8_t Opcode, ArrayRef<ValueAndLength> Operands);
 | 
						|
  // Add an extended opcode to the program with the specified length, opcode,
 | 
						|
  // and operands. These values do not have to be valid.
 | 
						|
  void addExtendedOpcode(uint64_t Length, uint8_t Opcode,
 | 
						|
                         ArrayRef<ValueAndLength> Operands);
 | 
						|
 | 
						|
  // Write the contents of the LineUnit to the current section in the generator.
 | 
						|
  void generate(MCContext &MC, AsmPrinter &Asm) const;
 | 
						|
 | 
						|
private:
 | 
						|
  void writeData(ArrayRef<ValueAndLength> Data, AsmPrinter &Asm) const;
 | 
						|
  MCSymbol *writeDefaultPrologue(AsmPrinter &Asm) const;
 | 
						|
  void writePrologue(AsmPrinter &Asm) const;
 | 
						|
 | 
						|
  void writeProloguePayload(const DWARFDebugLine::Prologue &Prologue,
 | 
						|
                            AsmPrinter &Asm) const;
 | 
						|
 | 
						|
  llvm::Optional<DWARFDebugLine::Prologue> Prologue;
 | 
						|
  std::vector<ValueAndLength> CustomPrologue;
 | 
						|
  std::vector<ValueAndLength> Contents;
 | 
						|
 | 
						|
  // The Version field is used for determining how to write the Prologue, if a
 | 
						|
  // non-custom prologue is used. The version value actually written, will be
 | 
						|
  // that specified in the Prologue, if a custom prologue has been passed in.
 | 
						|
  // Otherwise, it will be this value.
 | 
						|
  uint16_t Version;
 | 
						|
 | 
						|
  dwarf::DwarfFormat Format;
 | 
						|
  uint8_t AddrSize;
 | 
						|
  uint8_t SegSize;
 | 
						|
};
 | 
						|
 | 
						|
/// A DWARF generator.
 | 
						|
///
 | 
						|
/// Generate DWARF for unit tests by creating any instance of this class and
 | 
						|
/// calling Generator::addCompileUnit(), and then getting the dwarfgen::DIE from
 | 
						|
/// the returned compile unit and adding attributes and children to each DIE.
 | 
						|
class Generator {
 | 
						|
  std::unique_ptr<MCRegisterInfo> MRI;
 | 
						|
  std::unique_ptr<MCAsmInfo> MAI;
 | 
						|
  std::unique_ptr<MCContext> MC;
 | 
						|
  MCAsmBackend *MAB; // Owned by MCStreamer
 | 
						|
  std::unique_ptr<MCInstrInfo> MII;
 | 
						|
  std::unique_ptr<MCSubtargetInfo> MSTI;
 | 
						|
  MCCodeEmitter *MCE; // Owned by MCStreamer
 | 
						|
  MCStreamer *MS;     // Owned by AsmPrinter
 | 
						|
  std::unique_ptr<TargetMachine> TM;
 | 
						|
  TargetLoweringObjectFile *TLOF; // Owned by TargetMachine;
 | 
						|
  std::unique_ptr<AsmPrinter> Asm;
 | 
						|
  BumpPtrAllocator Allocator;
 | 
						|
  std::unique_ptr<DwarfStringPool> StringPool; // Entries owned by Allocator.
 | 
						|
  std::vector<std::unique_ptr<CompileUnit>> CompileUnits;
 | 
						|
  std::vector<std::unique_ptr<LineTable>> LineTables;
 | 
						|
  DIEAbbrevSet Abbreviations;
 | 
						|
 | 
						|
  MCSymbol *StringOffsetsStartSym;
 | 
						|
 | 
						|
  SmallString<4096> FileBytes;
 | 
						|
  /// The stream we use to generate the DWARF into as an ELF file.
 | 
						|
  std::unique_ptr<raw_svector_ostream> Stream;
 | 
						|
  /// The DWARF version to generate.
 | 
						|
  uint16_t Version;
 | 
						|
 | 
						|
  /// Private constructor, call Generator::Create(...) to get a DWARF generator
 | 
						|
  /// expected.
 | 
						|
  Generator();
 | 
						|
 | 
						|
  /// Create the streamer and setup the output buffer.
 | 
						|
  llvm::Error init(Triple TheTriple, uint16_t DwarfVersion);
 | 
						|
 | 
						|
public:
 | 
						|
  /// Create a DWARF generator or get an appropriate error.
 | 
						|
  ///
 | 
						|
  /// \param TheTriple the triple to use when creating any required support
 | 
						|
  /// classes needed to emit the DWARF.
 | 
						|
  /// \param DwarfVersion the version of DWARF to emit.
 | 
						|
  ///
 | 
						|
  /// \returns a llvm::Expected that either contains a unique_ptr to a Generator
 | 
						|
  /// or a llvm::Error.
 | 
						|
  static llvm::Expected<std::unique_ptr<Generator>>
 | 
						|
  create(Triple TheTriple, uint16_t DwarfVersion);
 | 
						|
 | 
						|
  ~Generator();
 | 
						|
 | 
						|
  /// Generate all DWARF sections and return a memory buffer that
 | 
						|
  /// contains an ELF file that contains the DWARF.
 | 
						|
  StringRef generate();
 | 
						|
 | 
						|
  /// Add a compile unit to be generated.
 | 
						|
  ///
 | 
						|
  /// \returns a dwarfgen::CompileUnit that can be used to retrieve the compile
 | 
						|
  /// unit dwarfgen::DIE that can be used to add attributes and add child DIE
 | 
						|
  /// objects to.
 | 
						|
  dwarfgen::CompileUnit &addCompileUnit();
 | 
						|
 | 
						|
  /// Add a line table unit to be generated.
 | 
						|
  /// \param DwarfFormat the DWARF format to use (DWARF32 or DWARF64).
 | 
						|
  ///
 | 
						|
  /// \returns a dwarfgen::LineTable that can be used to customise the contents
 | 
						|
  /// of the line table.
 | 
						|
  LineTable &
 | 
						|
  addLineTable(dwarf::DwarfFormat DwarfFormat = dwarf::DwarfFormat::DWARF32);
 | 
						|
 | 
						|
  BumpPtrAllocator &getAllocator() { return Allocator; }
 | 
						|
  AsmPrinter *getAsmPrinter() const { return Asm.get(); }
 | 
						|
  MCContext *getMCContext() const { return MC.get(); }
 | 
						|
  DIEAbbrevSet &getAbbrevSet() { return Abbreviations; }
 | 
						|
  DwarfStringPool &getStringPool() { return *StringPool; }
 | 
						|
  MCSymbol *getStringOffsetsStartSym() const { return StringOffsetsStartSym; }
 | 
						|
 | 
						|
  /// Save the generated DWARF file to disk.
 | 
						|
  ///
 | 
						|
  /// \param Path the path to save the ELF file to.
 | 
						|
  bool saveFile(StringRef Path);
 | 
						|
};
 | 
						|
 | 
						|
} // end namespace dwarfgen
 | 
						|
 | 
						|
} // end namespace llvm
 | 
						|
 | 
						|
#endif // LLVM_UNITTESTS_DEBUG_INFO_DWARF_DWARFGENERATOR_H
 |