2015-02-08 00:29:29 +00:00
|
|
|
//===- PDBSymbolExe.cpp - ---------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2015-02-14 03:53:56 +00:00
|
|
|
#include "llvm/DebugInfo/PDB/PDBSymbolExe.h"
|
|
|
|
|
2015-02-10 22:43:25 +00:00
|
|
|
#include "llvm/DebugInfo/PDB/IPDBEnumChildren.h"
|
|
|
|
#include "llvm/DebugInfo/PDB/PDBExtras.h"
|
2015-02-08 00:29:29 +00:00
|
|
|
#include "llvm/DebugInfo/PDB/PDBSymbol.h"
|
|
|
|
#include "llvm/Support/ConvertUTF.h"
|
|
|
|
#include "llvm/Support/FileSystem.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2015-02-13 09:09:03 +00:00
|
|
|
#include <utility>
|
2015-02-08 00:29:29 +00:00
|
|
|
|
2015-02-14 03:53:56 +00:00
|
|
|
#include <utility>
|
|
|
|
|
2015-02-08 00:29:29 +00:00
|
|
|
using namespace llvm;
|
|
|
|
|
2015-02-15 20:27:53 +00:00
|
|
|
#define SKIP_SYMBOL_IF_FLAG_UNSET(Tag, Flag) \
|
|
|
|
case PDB_SymType::Tag: \
|
|
|
|
if ((Flags & Flag) == 0) \
|
|
|
|
continue; \
|
|
|
|
break;
|
|
|
|
|
2015-02-08 22:53:53 +00:00
|
|
|
PDBSymbolExe::PDBSymbolExe(const IPDBSession &PDBSession,
|
2015-02-08 20:58:09 +00:00
|
|
|
std::unique_ptr<IPDBRawSymbol> Symbol)
|
|
|
|
: PDBSymbol(PDBSession, std::move(Symbol)) {}
|
2015-02-08 00:29:29 +00:00
|
|
|
|
2015-02-10 22:43:25 +00:00
|
|
|
void PDBSymbolExe::dump(raw_ostream &OS, int Indent,
|
2015-02-15 20:27:53 +00:00
|
|
|
PDB_DumpLevel Level, PDB_DumpFlags Flags) const {
|
2015-02-10 22:43:25 +00:00
|
|
|
std::string FileName(getSymbolsFileName());
|
|
|
|
|
2015-02-13 07:40:03 +00:00
|
|
|
OS << stream_indent(Indent) << "Summary for " << FileName << "\n";
|
2015-02-10 22:43:25 +00:00
|
|
|
|
|
|
|
uint64_t FileSize = 0;
|
|
|
|
if (!llvm::sys::fs::file_size(FileName, FileSize))
|
2015-02-13 07:40:03 +00:00
|
|
|
OS << stream_indent(Indent + 2) << "Size: " << FileSize << " bytes\n";
|
2015-02-10 22:43:25 +00:00
|
|
|
else
|
2015-02-13 07:40:03 +00:00
|
|
|
OS << stream_indent(Indent + 2) << "Size: (Unable to obtain file size)\n";
|
2015-02-10 22:43:25 +00:00
|
|
|
PDB_UniqueId Guid = getGuid();
|
2015-02-13 07:40:03 +00:00
|
|
|
OS << stream_indent(Indent + 2) << "Guid: " << Guid << "\n";
|
|
|
|
OS << stream_indent(Indent + 2) << "Age: " << getAge() << "\n";
|
|
|
|
OS << stream_indent(Indent + 2) << "Attributes: ";
|
2015-02-10 22:43:25 +00:00
|
|
|
if (hasCTypes())
|
|
|
|
OS << "HasCTypes ";
|
|
|
|
if (hasPrivateSymbols())
|
|
|
|
OS << "HasPrivateSymbols ";
|
|
|
|
OS << "\n";
|
2015-02-13 01:23:51 +00:00
|
|
|
|
2015-02-15 20:27:53 +00:00
|
|
|
if (Flags & PDB_DF_Children) {
|
2015-02-22 21:45:38 +00:00
|
|
|
OS << stream_indent(Indent + 2) << "Dumping types\n";
|
2015-02-15 20:27:53 +00:00
|
|
|
if (Flags & PDB_DF_Hidden) {
|
|
|
|
// For some reason, for each SymTag T, this dumps more items of type T
|
|
|
|
// than are dumped by calling dumpChildren(T). In other words, there are
|
|
|
|
// "hidden" symbols. For example, it causes functions to be dumped which
|
|
|
|
// have no address information, whereas specifically dumping only
|
|
|
|
// functions will not find those symbols.
|
|
|
|
//
|
|
|
|
// My suspicion is that in the underlying DIA call, when you call
|
|
|
|
// findChildren, passing a value of SymTagNone means all children
|
|
|
|
// recursively, whereas passing a concrete tag value means only immediate
|
|
|
|
// children of the global scope. So perhaps we need to find these
|
|
|
|
// mysterious missing values by recursing through the hierarchy.
|
|
|
|
//
|
|
|
|
// On the other hand, there may just be some symbols that DIA tries to
|
|
|
|
// hide from you because it thinks you don't care about them. However
|
|
|
|
// experimentation shows that even vtables, for example, can't be found
|
|
|
|
// without an exhaustive search.
|
|
|
|
auto ChildrenEnum = findAllChildren();
|
|
|
|
OS << stream_indent(Indent + 2) << ChildrenEnum->getChildCount()
|
|
|
|
<< " symbols";
|
|
|
|
|
|
|
|
while (auto Child = ChildrenEnum->getNext()) {
|
|
|
|
switch (Child->getSymTag()) {
|
|
|
|
SKIP_SYMBOL_IF_FLAG_UNSET(Function, PDB_DF_Functions)
|
|
|
|
SKIP_SYMBOL_IF_FLAG_UNSET(Data, PDB_DF_Data)
|
|
|
|
SKIP_SYMBOL_IF_FLAG_UNSET(Label, PDB_DF_Labels)
|
|
|
|
SKIP_SYMBOL_IF_FLAG_UNSET(PublicSymbol, PDB_DF_PublicSyms)
|
|
|
|
SKIP_SYMBOL_IF_FLAG_UNSET(UDT, PDB_DF_Classes)
|
|
|
|
SKIP_SYMBOL_IF_FLAG_UNSET(Enum, PDB_DF_Enums)
|
|
|
|
SKIP_SYMBOL_IF_FLAG_UNSET(FunctionSig, PDB_DF_Funcsigs)
|
|
|
|
SKIP_SYMBOL_IF_FLAG_UNSET(VTable, PDB_DF_VTables)
|
|
|
|
SKIP_SYMBOL_IF_FLAG_UNSET(Thunk, PDB_DF_Thunks)
|
|
|
|
SKIP_SYMBOL_IF_FLAG_UNSET(Compiland, PDB_DF_ObjFiles)
|
|
|
|
default:
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
PDB_DumpLevel ChildLevel = (Level == PDB_DumpLevel::Detailed)
|
|
|
|
? PDB_DumpLevel::Normal
|
|
|
|
: PDB_DumpLevel::Compact;
|
|
|
|
OS << "\n";
|
|
|
|
Child->dump(OS, Indent + 4, ChildLevel, PDB_DF_Children);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (Flags & PDB_DF_ObjFiles)
|
|
|
|
dumpChildren(OS, "Compilands", PDB_SymType::Compiland, Indent + 4);
|
|
|
|
if (Flags & PDB_DF_Functions)
|
|
|
|
dumpChildren(OS, "Functions", PDB_SymType::Function, Indent + 4);
|
|
|
|
if (Flags & PDB_DF_Data)
|
|
|
|
dumpChildren(OS, "Data", PDB_SymType::Data, Indent + 4);
|
|
|
|
if (Flags & PDB_DF_Labels)
|
|
|
|
dumpChildren(OS, "Labels", PDB_SymType::Label, Indent + 4);
|
|
|
|
if (Flags & PDB_DF_PublicSyms)
|
|
|
|
dumpChildren(OS, "Public Symbols", PDB_SymType::PublicSymbol,
|
|
|
|
Indent + 4);
|
|
|
|
if (Flags & PDB_DF_Classes)
|
|
|
|
dumpChildren(OS, "UDTs", PDB_SymType::UDT, Indent + 4);
|
|
|
|
if (Flags & PDB_DF_Enums)
|
|
|
|
dumpChildren(OS, "Enums", PDB_SymType::Enum, Indent + 4);
|
|
|
|
if (Flags & PDB_DF_Funcsigs)
|
|
|
|
dumpChildren(OS, "Function Signatures", PDB_SymType::FunctionSig,
|
|
|
|
Indent + 4);
|
|
|
|
if (Flags & PDB_DF_Typedefs)
|
|
|
|
dumpChildren(OS, "Typedefs", PDB_SymType::Typedef, Indent + 4);
|
|
|
|
if (Flags & PDB_DF_VTables)
|
|
|
|
dumpChildren(OS, "VTables", PDB_SymType::VTable, Indent + 4);
|
|
|
|
if (Flags & PDB_DF_Thunks)
|
|
|
|
dumpChildren(OS, "Thunks", PDB_SymType::Thunk, Indent + 4);
|
|
|
|
}
|
|
|
|
}
|
2015-02-14 03:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void PDBSymbolExe::dumpChildren(raw_ostream &OS, StringRef Label,
|
|
|
|
PDB_SymType ChildType, int Indent) const {
|
|
|
|
auto ChildrenEnum = findAllChildren(ChildType);
|
|
|
|
OS << stream_indent(Indent) << Label << ": (" << ChildrenEnum->getChildCount()
|
|
|
|
<< " items)\n";
|
2015-02-13 01:23:51 +00:00
|
|
|
while (auto Child = ChildrenEnum->getNext()) {
|
2015-02-15 20:27:53 +00:00
|
|
|
Child->dump(OS, Indent + 2, PDB_DumpLevel::Normal, PDB_DF_None);
|
2015-02-13 07:40:03 +00:00
|
|
|
OS << "\n";
|
2015-02-13 01:23:51 +00:00
|
|
|
}
|
2015-02-08 00:29:29 +00:00
|
|
|
}
|