You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			259 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			259 lines
		
	
	
		
			8.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
|   | //===- PrettyFunctionDumper.cpp --------------------------------- *- C++ *-===//
 | ||
|  | //
 | ||
|  | //                     The LLVM Compiler Infrastructure
 | ||
|  | //
 | ||
|  | // This file is distributed under the University of Illinois Open Source
 | ||
|  | // License. See LICENSE.TXT for details.
 | ||
|  | //
 | ||
|  | //===----------------------------------------------------------------------===//
 | ||
|  | 
 | ||
|  | #include "PrettyFunctionDumper.h"
 | ||
|  | #include "LinePrinter.h"
 | ||
|  | #include "PrettyBuiltinDumper.h"
 | ||
|  | 
 | ||
|  | #include "llvm/DebugInfo/PDB/IPDBSession.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/PDBExtras.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/PDBSymbolData.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/PDBSymbolFunc.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/PDBSymbolFuncDebugEnd.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/PDBSymbolFuncDebugStart.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeArray.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeEnum.h"
 | ||
|  | #include "llvm/DebugInfo/PDB/PDBSymbolTypeFunctionArg.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/Support/Format.h"
 | ||
|  | #include "llvm/Support/FormatVariadic.h"
 | ||
|  | 
 | ||
|  | using namespace llvm; | ||
|  | using namespace llvm::codeview; | ||
|  | using namespace llvm::pdb; | ||
|  | 
 | ||
|  | namespace { | ||
|  | template <class T> | ||
|  | void dumpClassParentWithScopeOperator(const T &Symbol, LinePrinter &Printer, | ||
|  |                                       FunctionDumper &Dumper) { | ||
|  |   uint32_t ClassParentId = Symbol.getClassParentId(); | ||
|  |   auto ClassParent = | ||
|  |       Symbol.getSession().template getConcreteSymbolById<PDBSymbolTypeUDT>( | ||
|  |           ClassParentId); | ||
|  |   if (!ClassParent) | ||
|  |     return; | ||
|  | 
 | ||
|  |   WithColor(Printer, PDB_ColorItem::Type).get() << ClassParent->getName(); | ||
|  |   Printer << "::"; | ||
|  | } | ||
|  | } | ||
|  | 
 | ||
|  | FunctionDumper::FunctionDumper(LinePrinter &P) | ||
|  |     : PDBSymDumper(true), Printer(P) {} | ||
|  | 
 | ||
|  | void FunctionDumper::start(const PDBSymbolTypeFunctionSig &Symbol, | ||
|  |                            const char *Name, PointerType Pointer) { | ||
|  |   auto ReturnType = Symbol.getReturnType(); | ||
|  |   ReturnType->dump(*this); | ||
|  |   Printer << " "; | ||
|  |   uint32_t ClassParentId = Symbol.getClassParentId(); | ||
|  |   auto ClassParent = | ||
|  |       Symbol.getSession().getConcreteSymbolById<PDBSymbolTypeUDT>( | ||
|  |           ClassParentId); | ||
|  | 
 | ||
|  |   PDB_CallingConv CC = Symbol.getCallingConvention(); | ||
|  |   bool ShouldDumpCallingConvention = true; | ||
|  |   if ((ClassParent && CC == CallingConvention::ThisCall) || | ||
|  |       (!ClassParent && CC == CallingConvention::NearStdCall)) { | ||
|  |     ShouldDumpCallingConvention = false; | ||
|  |   } | ||
|  | 
 | ||
|  |   if (Pointer == PointerType::None) { | ||
|  |     if (ShouldDumpCallingConvention) | ||
|  |       WithColor(Printer, PDB_ColorItem::Keyword).get() << CC << " "; | ||
|  |     if (ClassParent) { | ||
|  |       Printer << "("; | ||
|  |       WithColor(Printer, PDB_ColorItem::Identifier).get() | ||
|  |           << ClassParent->getName(); | ||
|  |       Printer << "::)"; | ||
|  |     } | ||
|  |   } else { | ||
|  |     Printer << "("; | ||
|  |     if (ShouldDumpCallingConvention) | ||
|  |       WithColor(Printer, PDB_ColorItem::Keyword).get() << CC << " "; | ||
|  |     if (ClassParent) { | ||
|  |       WithColor(Printer, PDB_ColorItem::Identifier).get() | ||
|  |           << ClassParent->getName(); | ||
|  |       Printer << "::"; | ||
|  |     } | ||
|  |     if (Pointer == PointerType::Reference) | ||
|  |       Printer << "&"; | ||
|  |     else | ||
|  |       Printer << "*"; | ||
|  |     if (Name) | ||
|  |       WithColor(Printer, PDB_ColorItem::Identifier).get() << Name; | ||
|  |     Printer << ")"; | ||
|  |   } | ||
|  | 
 | ||
|  |   Printer << "("; | ||
|  |   if (auto ChildEnum = Symbol.getArguments()) { | ||
|  |     uint32_t Index = 0; | ||
|  |     while (auto Arg = ChildEnum->getNext()) { | ||
|  |       Arg->dump(*this); | ||
|  |       if (++Index < ChildEnum->getChildCount()) | ||
|  |         Printer << ", "; | ||
|  |     } | ||
|  |   } | ||
|  |   Printer << ")"; | ||
|  | 
 | ||
|  |   if (Symbol.isConstType()) | ||
|  |     WithColor(Printer, PDB_ColorItem::Keyword).get() << " const"; | ||
|  |   if (Symbol.isVolatileType()) | ||
|  |     WithColor(Printer, PDB_ColorItem::Keyword).get() << " volatile"; | ||
|  | } | ||
|  | 
 | ||
|  | void FunctionDumper::start(const PDBSymbolFunc &Symbol, PointerType Pointer) { | ||
|  |   uint64_t FuncStart = Symbol.getVirtualAddress(); | ||
|  |   uint64_t FuncEnd = FuncStart + Symbol.getLength(); | ||
|  | 
 | ||
|  |   Printer << "func ["; | ||
|  |   WithColor(Printer, PDB_ColorItem::Address).get() << format_hex(FuncStart, 10); | ||
|  |   if (auto DebugStart = Symbol.findOneChild<PDBSymbolFuncDebugStart>()) { | ||
|  |     uint64_t Prologue = DebugStart->getVirtualAddress() - FuncStart; | ||
|  |     WithColor(Printer, PDB_ColorItem::Offset).get() | ||
|  |         << formatv("+{0,2}", Prologue); | ||
|  |   } | ||
|  |   Printer << " - "; | ||
|  |   WithColor(Printer, PDB_ColorItem::Address).get() << format_hex(FuncEnd, 10); | ||
|  |   if (auto DebugEnd = Symbol.findOneChild<PDBSymbolFuncDebugEnd>()) { | ||
|  |     uint64_t Epilogue = FuncEnd - DebugEnd->getVirtualAddress(); | ||
|  |     WithColor(Printer, PDB_ColorItem::Offset).get() | ||
|  |         << formatv("-{0,2}", Epilogue); | ||
|  |   } | ||
|  | 
 | ||
|  |   WithColor(Printer, PDB_ColorItem::Comment).get() | ||
|  |       << formatv(" | sizeof={0,3}", Symbol.getLength()); | ||
|  |   Printer << "] ("; | ||
|  | 
 | ||
|  |   if (Symbol.hasFramePointer()) { | ||
|  |     WithColor(Printer, PDB_ColorItem::Register).get() | ||
|  |         << Symbol.getLocalBasePointerRegisterId(); | ||
|  |   } else { | ||
|  |     WithColor(Printer, PDB_ColorItem::Register).get() << "FPO"; | ||
|  |   } | ||
|  |   Printer << ") "; | ||
|  | 
 | ||
|  |   if (Symbol.isVirtual() || Symbol.isPureVirtual()) | ||
|  |     WithColor(Printer, PDB_ColorItem::Keyword).get() << "virtual "; | ||
|  | 
 | ||
|  |   auto Signature = Symbol.getSignature(); | ||
|  |   if (!Signature) { | ||
|  |     WithColor(Printer, PDB_ColorItem::Identifier).get() << Symbol.getName(); | ||
|  |     if (Pointer == PointerType::Pointer) | ||
|  |       Printer << "*"; | ||
|  |     else if (Pointer == FunctionDumper::PointerType::Reference) | ||
|  |       Printer << "&"; | ||
|  |     return; | ||
|  |   } | ||
|  | 
 | ||
|  |   auto ReturnType = Signature->getReturnType(); | ||
|  |   ReturnType->dump(*this); | ||
|  |   Printer << " "; | ||
|  | 
 | ||
|  |   auto ClassParent = Symbol.getClassParent(); | ||
|  |   CallingConvention CC = Signature->getCallingConvention(); | ||
|  |   if (Pointer != FunctionDumper::PointerType::None) | ||
|  |     Printer << "("; | ||
|  | 
 | ||
|  |   if ((ClassParent && CC != CallingConvention::ThisCall) || | ||
|  |       (!ClassParent && CC != CallingConvention::NearStdCall)) { | ||
|  |     WithColor(Printer, PDB_ColorItem::Keyword).get() | ||
|  |         << Signature->getCallingConvention() << " "; | ||
|  |   } | ||
|  |   WithColor(Printer, PDB_ColorItem::Identifier).get() << Symbol.getName(); | ||
|  |   if (Pointer != FunctionDumper::PointerType::None) { | ||
|  |     if (Pointer == PointerType::Pointer) | ||
|  |       Printer << "*"; | ||
|  |     else if (Pointer == FunctionDumper::PointerType::Reference) | ||
|  |       Printer << "&"; | ||
|  |     Printer << ")"; | ||
|  |   } | ||
|  | 
 | ||
|  |   Printer << "("; | ||
|  |   if (auto Arguments = Symbol.getArguments()) { | ||
|  |     uint32_t Index = 0; | ||
|  |     while (auto Arg = Arguments->getNext()) { | ||
|  |       auto ArgType = Arg->getType(); | ||
|  |       ArgType->dump(*this); | ||
|  |       WithColor(Printer, PDB_ColorItem::Identifier).get() << " " | ||
|  |                                                           << Arg->getName(); | ||
|  |       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.isPureVirtual()) | ||
|  |     Printer << " = 0"; | ||
|  | } | ||
|  | 
 | ||
|  | void FunctionDumper::dump(const PDBSymbolTypeArray &Symbol) { | ||
|  |   auto ElementType = Symbol.getElementType(); | ||
|  | 
 | ||
|  |   ElementType->dump(*this); | ||
|  |   Printer << "["; | ||
|  |   WithColor(Printer, PDB_ColorItem::LiteralValue).get() << Symbol.getLength(); | ||
|  |   Printer << "]"; | ||
|  | } | ||
|  | 
 | ||
|  | void FunctionDumper::dump(const PDBSymbolTypeBuiltin &Symbol) { | ||
|  |   BuiltinDumper Dumper(Printer); | ||
|  |   Dumper.start(Symbol); | ||
|  | } | ||
|  | 
 | ||
|  | void FunctionDumper::dump(const PDBSymbolTypeEnum &Symbol) { | ||
|  |   dumpClassParentWithScopeOperator(Symbol, Printer, *this); | ||
|  |   WithColor(Printer, PDB_ColorItem::Type).get() << Symbol.getName(); | ||
|  | } | ||
|  | 
 | ||
|  | void FunctionDumper::dump(const PDBSymbolTypeFunctionArg &Symbol) { | ||
|  |   // PDBSymbolTypeFunctionArg is just a shim over the real argument.  Just drill
 | ||
|  |   // through to the real thing and dump it.
 | ||
|  |   uint32_t TypeId = Symbol.getTypeId(); | ||
|  |   auto Type = Symbol.getSession().getSymbolById(TypeId); | ||
|  |   if (!Type) | ||
|  |     return; | ||
|  |   Type->dump(*this); | ||
|  | } | ||
|  | 
 | ||
|  | void FunctionDumper::dump(const PDBSymbolTypeTypedef &Symbol) { | ||
|  |   dumpClassParentWithScopeOperator(Symbol, Printer, *this); | ||
|  |   WithColor(Printer, PDB_ColorItem::Type).get() << Symbol.getName(); | ||
|  | } | ||
|  | 
 | ||
|  | void FunctionDumper::dump(const PDBSymbolTypePointer &Symbol) { | ||
|  |   auto PointeeType = Symbol.getPointeeType(); | ||
|  |   if (!PointeeType) | ||
|  |     return; | ||
|  | 
 | ||
|  |   if (auto FuncSig = unique_dyn_cast<PDBSymbolTypeFunctionSig>(PointeeType)) { | ||
|  |     FunctionDumper NestedDumper(Printer); | ||
|  |     PointerType Pointer = | ||
|  |         Symbol.isReference() ? PointerType::Reference : PointerType::Pointer; | ||
|  |     NestedDumper.start(*FuncSig, nullptr, Pointer); | ||
|  |   } else { | ||
|  |     if (Symbol.isConstType()) | ||
|  |       WithColor(Printer, PDB_ColorItem::Keyword).get() << "const "; | ||
|  |     if (Symbol.isVolatileType()) | ||
|  |       WithColor(Printer, PDB_ColorItem::Keyword).get() << "volatile "; | ||
|  |     PointeeType->dump(*this); | ||
|  |     Printer << (Symbol.isReference() ? "&" : "*"); | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | void FunctionDumper::dump(const PDBSymbolTypeUDT &Symbol) { | ||
|  |   WithColor(Printer, PDB_ColorItem::Type).get() << Symbol.getName(); | ||
|  | } |