|  | //===- PrettyVariableDumper.cpp ---------------------------------*- 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 | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "PrettyVariableDumper.h" | 
|  |  | 
|  | #include "PrettyBuiltinDumper.h" | 
|  | #include "PrettyFunctionDumper.h" | 
|  | #include "llvm-pdbutil.h" | 
|  |  | 
|  | #include "llvm/DebugInfo/PDB/IPDBLineNumber.h" | 
|  | #include "llvm/DebugInfo/PDB/IPDBSession.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolData.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolFunc.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeArray.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeBuiltin.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeEnum.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeFunctionSig.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypePointer.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeTypedef.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeUDT.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeVTable.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeVTableShape.h" | 
|  | #include "llvm/DebugInfo/PDB/PDBTypes.h" | 
|  |  | 
|  | #include "llvm/Support/Format.h" | 
|  |  | 
|  | using namespace llvm; | 
|  | using namespace llvm::codeview; | 
|  | using namespace llvm::pdb; | 
|  |  | 
|  | VariableDumper::VariableDumper(LinePrinter &P) | 
|  | : PDBSymDumper(true), Printer(P) {} | 
|  |  | 
|  | void VariableDumper::start(const PDBSymbolData &Var, uint32_t Offset) { | 
|  | if (Var.isCompilerGenerated() && opts::pretty::ExcludeCompilerGenerated) | 
|  | return; | 
|  | if (Printer.IsSymbolExcluded(Var.getName())) | 
|  | return; | 
|  |  | 
|  | auto VarType = Var.getType(); | 
|  |  | 
|  | uint64_t Length = VarType->getRawSymbol().getLength(); | 
|  |  | 
|  | switch (auto LocType = Var.getLocationType()) { | 
|  | case PDB_LocType::Static: | 
|  | Printer.NewLine(); | 
|  | Printer << "data ["; | 
|  | WithColor(Printer, PDB_ColorItem::Address).get() | 
|  | << format_hex(Var.getVirtualAddress(), 10); | 
|  | Printer << ", sizeof=" << Length << "] "; | 
|  | WithColor(Printer, PDB_ColorItem::Keyword).get() << "static "; | 
|  | dumpSymbolTypeAndName(*VarType, Var.getName()); | 
|  | break; | 
|  | case PDB_LocType::Constant: | 
|  | if (isa<PDBSymbolTypeEnum>(*VarType)) | 
|  | break; | 
|  | Printer.NewLine(); | 
|  | Printer << "data [sizeof=" << Length << "] "; | 
|  | dumpSymbolTypeAndName(*VarType, Var.getName()); | 
|  | Printer << " = "; | 
|  | WithColor(Printer, PDB_ColorItem::LiteralValue).get() << Var.getValue(); | 
|  | break; | 
|  | case PDB_LocType::ThisRel: | 
|  | Printer.NewLine(); | 
|  | Printer << "data "; | 
|  | WithColor(Printer, PDB_ColorItem::Offset).get() | 
|  | << "+" << format_hex(Offset + Var.getOffset(), 4) | 
|  | << " [sizeof=" << Length << "] "; | 
|  | dumpSymbolTypeAndName(*VarType, Var.getName()); | 
|  | break; | 
|  | case PDB_LocType::BitField: | 
|  | Printer.NewLine(); | 
|  | Printer << "data "; | 
|  | WithColor(Printer, PDB_ColorItem::Offset).get() | 
|  | << "+" << format_hex(Offset + Var.getOffset(), 4) | 
|  | << " [sizeof=" << Length << "] "; | 
|  | dumpSymbolTypeAndName(*VarType, Var.getName()); | 
|  | Printer << " : "; | 
|  | WithColor(Printer, PDB_ColorItem::LiteralValue).get() << Var.getLength(); | 
|  | break; | 
|  | default: | 
|  | Printer.NewLine(); | 
|  | Printer << "data [sizeof=" << Length << "] "; | 
|  | Printer << "unknown(" << LocType << ") "; | 
|  | WithColor(Printer, PDB_ColorItem::Identifier).get() << Var.getName(); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | void VariableDumper::startVbptr(uint32_t Offset, uint32_t Size) { | 
|  | Printer.NewLine(); | 
|  | Printer << "vbptr "; | 
|  |  | 
|  | WithColor(Printer, PDB_ColorItem::Offset).get() | 
|  | << "+" << format_hex(Offset, 4) << " [sizeof=" << Size << "] "; | 
|  | } | 
|  |  | 
|  | void VariableDumper::start(const PDBSymbolTypeVTable &Var, uint32_t Offset) { | 
|  | Printer.NewLine(); | 
|  | Printer << "vfptr "; | 
|  | auto VTableType = cast<PDBSymbolTypePointer>(Var.getType()); | 
|  | uint32_t PointerSize = VTableType->getLength(); | 
|  |  | 
|  | WithColor(Printer, PDB_ColorItem::Offset).get() | 
|  | << "+" << format_hex(Offset + Var.getOffset(), 4) | 
|  | << " [sizeof=" << PointerSize << "] "; | 
|  | } | 
|  |  | 
|  | void VariableDumper::dump(const PDBSymbolTypeArray &Symbol) { | 
|  | auto ElementType = Symbol.getElementType(); | 
|  | assert(ElementType); | 
|  | if (!ElementType) | 
|  | return; | 
|  | ElementType->dump(*this); | 
|  | } | 
|  |  | 
|  | void VariableDumper::dumpRight(const PDBSymbolTypeArray &Symbol) { | 
|  | auto ElementType = Symbol.getElementType(); | 
|  | assert(ElementType); | 
|  | if (!ElementType) | 
|  | return; | 
|  | Printer << '[' << Symbol.getCount() << ']'; | 
|  | ElementType->dumpRight(*this); | 
|  | } | 
|  |  | 
|  | void VariableDumper::dump(const PDBSymbolTypeBuiltin &Symbol) { | 
|  | BuiltinDumper Dumper(Printer); | 
|  | Dumper.start(Symbol); | 
|  | } | 
|  |  | 
|  | void VariableDumper::dump(const PDBSymbolTypeEnum &Symbol) { | 
|  | WithColor(Printer, PDB_ColorItem::Type).get() << Symbol.getName(); | 
|  | } | 
|  |  | 
|  | void VariableDumper::dump(const PDBSymbolTypeFunctionSig &Symbol) { | 
|  | auto ReturnType = Symbol.getReturnType(); | 
|  | ReturnType->dump(*this); | 
|  | Printer << " "; | 
|  |  | 
|  | uint32_t ClassParentId = Symbol.getClassParentId(); | 
|  | auto ClassParent = | 
|  | Symbol.getSession().getConcreteSymbolById<PDBSymbolTypeUDT>( | 
|  | ClassParentId); | 
|  |  | 
|  | if (ClassParent) { | 
|  | WithColor(Printer, PDB_ColorItem::Identifier).get() | 
|  | << ClassParent->getName(); | 
|  | Printer << "::"; | 
|  | } | 
|  | } | 
|  |  | 
|  | void VariableDumper::dumpRight(const PDBSymbolTypeFunctionSig &Symbol) { | 
|  | Printer << "("; | 
|  | if (auto Arguments = Symbol.getArguments()) { | 
|  | uint32_t Index = 0; | 
|  | while (auto Arg = Arguments->getNext()) { | 
|  | Arg->dump(*this); | 
|  | if (++Index < Arguments->getChildCount()) | 
|  | Printer << ", "; | 
|  | } | 
|  | } | 
|  | Printer << ")"; | 
|  |  | 
|  | if (Symbol.isConstType()) | 
|  | WithColor(Printer, PDB_ColorItem::Keyword).get() << " const"; | 
|  | if (Symbol.isVolatileType()) | 
|  | WithColor(Printer, PDB_ColorItem::Keyword).get() << " volatile"; | 
|  |  | 
|  | if (Symbol.getRawSymbol().isRestrictedType()) | 
|  | WithColor(Printer, PDB_ColorItem::Keyword).get() << " __restrict"; | 
|  | } | 
|  |  | 
|  | void VariableDumper::dump(const PDBSymbolTypePointer &Symbol) { | 
|  | auto PointeeType = Symbol.getPointeeType(); | 
|  | if (!PointeeType) | 
|  | return; | 
|  | PointeeType->dump(*this); | 
|  | if (auto FuncSig = unique_dyn_cast<PDBSymbolTypeFunctionSig>(PointeeType)) { | 
|  | // A hack to get the calling convention in the right spot. | 
|  | Printer << " ("; | 
|  | PDB_CallingConv CC = FuncSig->getCallingConvention(); | 
|  | WithColor(Printer, PDB_ColorItem::Keyword).get() << CC << " "; | 
|  | } else if (isa<PDBSymbolTypeArray>(PointeeType)) { | 
|  | Printer << " ("; | 
|  | } | 
|  | Printer << (Symbol.isReference() ? "&" : "*"); | 
|  | if (Symbol.isConstType()) | 
|  | WithColor(Printer, PDB_ColorItem::Keyword).get() << " const "; | 
|  | if (Symbol.isVolatileType()) | 
|  | WithColor(Printer, PDB_ColorItem::Keyword).get() << " volatile "; | 
|  |  | 
|  | if (Symbol.getRawSymbol().isRestrictedType()) | 
|  | WithColor(Printer, PDB_ColorItem::Keyword).get() << " __restrict "; | 
|  | } | 
|  |  | 
|  | void VariableDumper::dumpRight(const PDBSymbolTypePointer &Symbol) { | 
|  | auto PointeeType = Symbol.getPointeeType(); | 
|  | assert(PointeeType); | 
|  | if (!PointeeType) | 
|  | return; | 
|  | if (isa<PDBSymbolTypeFunctionSig>(PointeeType) || | 
|  | isa<PDBSymbolTypeArray>(PointeeType)) { | 
|  | Printer << ")"; | 
|  | } | 
|  | PointeeType->dumpRight(*this); | 
|  | } | 
|  |  | 
|  | void VariableDumper::dump(const PDBSymbolTypeTypedef &Symbol) { | 
|  | WithColor(Printer, PDB_ColorItem::Keyword).get() << "typedef "; | 
|  | WithColor(Printer, PDB_ColorItem::Type).get() << Symbol.getName(); | 
|  | } | 
|  |  | 
|  | void VariableDumper::dump(const PDBSymbolTypeUDT &Symbol) { | 
|  | WithColor(Printer, PDB_ColorItem::Type).get() << Symbol.getName(); | 
|  | } | 
|  |  | 
|  | void VariableDumper::dumpSymbolTypeAndName(const PDBSymbol &Type, | 
|  | StringRef Name) { | 
|  | Type.dump(*this); | 
|  | WithColor(Printer, PDB_ColorItem::Identifier).get() << " " << Name; | 
|  | Type.dumpRight(*this); | 
|  | } |