mirror of
https://github.com/llvm/llvm-project.git
synced 2025-05-08 03:16:07 +00:00

After some internal discussions, we agreed that the raw output style had outlived its usefulness. It was originally created before we had even thought of dumping to YAML, and it was intended to give us some insight into the internals of a PDB file. Now we have YAML mode which does almost exactly this but is more powerful in that it can round-trip back to a PDB, which the raw mode could not do. So the raw mode had become purely a maintenance burden. One option was to just delete it. However, its original goal was to be as readable as possible while staying close to the "metal" - i.e. presenting the output in a way that maps directly to the underlying file format. We don't actually need that last requirement anymore since it's covered by the yaml mode, so we could repurpose "raw" mode to actually just be as readable as possible. This patch implements about 80% of the functionality previously in raw mode, but in a completely different style that is more akin to what cvdump outputs. Records are very compressed, often times appearing on just one line. One nice thing about this is that it makes full record matching easier, because you can grep for indices, names, and leaf types on a single line often. See the tests for some examples of what the new output looks like. Note that this patch actually regresses the functionality of raw mode in a few areas, but only because the patch was already unreasonably large and going 100% would have been even worse. Specifically, this patch is missing: The ability to dump module debug subsections (checksums, lines, etc) The ability to dump section headers Aside from that everything is here. While goign through the tests fixing them all up, I found many duplicate tests. They've been deleted. In subsequent patches I will go through and re-add the missing functionality. Differential Revision: https://reviews.llvm.org/D34191 llvm-svn: 305495
564 lines
19 KiB
C++
564 lines
19 KiB
C++
//===- CodeViewYAMLSymbols.cpp - CodeView YAMLIO Symbol implementation ----===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file defines classes for handling the YAML representation of CodeView
|
|
// Debug Info.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "llvm/ObjectYAML/CodeViewYAMLSymbols.h"
|
|
#include "llvm/ADT/StringExtras.h"
|
|
#include "llvm/ADT/StringSwitch.h"
|
|
#include "llvm/DebugInfo/CodeView/CVTypeVisitor.h"
|
|
#include "llvm/DebugInfo/CodeView/CodeViewError.h"
|
|
#include "llvm/DebugInfo/CodeView/EnumTables.h"
|
|
#include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
|
|
#include "llvm/DebugInfo/CodeView/SymbolSerializer.h"
|
|
|
|
using namespace llvm;
|
|
using namespace llvm::codeview;
|
|
using namespace llvm::CodeViewYAML;
|
|
using namespace llvm::CodeViewYAML::detail;
|
|
using namespace llvm::yaml;
|
|
|
|
LLVM_YAML_IS_SEQUENCE_VECTOR(StringRef)
|
|
LLVM_YAML_IS_FLOW_SEQUENCE_VECTOR(TypeIndex)
|
|
|
|
// We only need to declare these, the definitions are in CodeViewYAMLTypes.cpp
|
|
LLVM_YAML_DECLARE_SCALAR_TRAITS(APSInt, false)
|
|
LLVM_YAML_DECLARE_SCALAR_TRAITS(TypeIndex, false)
|
|
|
|
LLVM_YAML_DECLARE_ENUM_TRAITS(SymbolKind)
|
|
LLVM_YAML_DECLARE_ENUM_TRAITS(FrameCookieKind)
|
|
|
|
LLVM_YAML_DECLARE_BITSET_TRAITS(CompileSym2Flags)
|
|
LLVM_YAML_DECLARE_BITSET_TRAITS(CompileSym3Flags)
|
|
LLVM_YAML_DECLARE_BITSET_TRAITS(ExportFlags)
|
|
LLVM_YAML_DECLARE_BITSET_TRAITS(LocalSymFlags)
|
|
LLVM_YAML_DECLARE_BITSET_TRAITS(ProcSymFlags)
|
|
LLVM_YAML_DECLARE_BITSET_TRAITS(FrameProcedureOptions)
|
|
LLVM_YAML_DECLARE_ENUM_TRAITS(CPUType)
|
|
LLVM_YAML_DECLARE_ENUM_TRAITS(RegisterId)
|
|
LLVM_YAML_DECLARE_ENUM_TRAITS(TrampolineType)
|
|
LLVM_YAML_DECLARE_ENUM_TRAITS(ThunkOrdinal)
|
|
|
|
LLVM_YAML_STRONG_TYPEDEF(llvm::StringRef, TypeName)
|
|
|
|
LLVM_YAML_DECLARE_SCALAR_TRAITS(TypeName, true)
|
|
|
|
StringRef ScalarTraits<TypeName>::input(StringRef S, void *V, TypeName &T) {
|
|
return ScalarTraits<StringRef>::input(S, V, T.value);
|
|
}
|
|
void ScalarTraits<TypeName>::output(const TypeName &T, void *V,
|
|
llvm::raw_ostream &R) {
|
|
ScalarTraits<StringRef>::output(T.value, V, R);
|
|
}
|
|
|
|
void ScalarEnumerationTraits<SymbolKind>::enumeration(IO &io,
|
|
SymbolKind &Value) {
|
|
auto SymbolNames = getSymbolTypeNames();
|
|
for (const auto &E : SymbolNames)
|
|
io.enumCase(Value, E.Name.str().c_str(), E.Value);
|
|
}
|
|
|
|
void ScalarBitSetTraits<CompileSym2Flags>::bitset(IO &io,
|
|
CompileSym2Flags &Flags) {
|
|
auto FlagNames = getCompileSym2FlagNames();
|
|
for (const auto &E : FlagNames) {
|
|
io.bitSetCase(Flags, E.Name.str().c_str(),
|
|
static_cast<CompileSym2Flags>(E.Value));
|
|
}
|
|
}
|
|
|
|
void ScalarBitSetTraits<CompileSym3Flags>::bitset(IO &io,
|
|
CompileSym3Flags &Flags) {
|
|
auto FlagNames = getCompileSym3FlagNames();
|
|
for (const auto &E : FlagNames) {
|
|
io.bitSetCase(Flags, E.Name.str().c_str(),
|
|
static_cast<CompileSym3Flags>(E.Value));
|
|
}
|
|
}
|
|
|
|
void ScalarBitSetTraits<ExportFlags>::bitset(IO &io, ExportFlags &Flags) {
|
|
auto FlagNames = getExportSymFlagNames();
|
|
for (const auto &E : FlagNames) {
|
|
io.bitSetCase(Flags, E.Name.str().c_str(),
|
|
static_cast<ExportFlags>(E.Value));
|
|
}
|
|
}
|
|
|
|
void ScalarBitSetTraits<LocalSymFlags>::bitset(IO &io, LocalSymFlags &Flags) {
|
|
auto FlagNames = getLocalFlagNames();
|
|
for (const auto &E : FlagNames) {
|
|
io.bitSetCase(Flags, E.Name.str().c_str(),
|
|
static_cast<LocalSymFlags>(E.Value));
|
|
}
|
|
}
|
|
|
|
void ScalarBitSetTraits<ProcSymFlags>::bitset(IO &io, ProcSymFlags &Flags) {
|
|
auto FlagNames = getProcSymFlagNames();
|
|
for (const auto &E : FlagNames) {
|
|
io.bitSetCase(Flags, E.Name.str().c_str(),
|
|
static_cast<ProcSymFlags>(E.Value));
|
|
}
|
|
}
|
|
|
|
void ScalarBitSetTraits<FrameProcedureOptions>::bitset(
|
|
IO &io, FrameProcedureOptions &Flags) {
|
|
auto FlagNames = getFrameProcSymFlagNames();
|
|
for (const auto &E : FlagNames) {
|
|
io.bitSetCase(Flags, E.Name.str().c_str(),
|
|
static_cast<FrameProcedureOptions>(E.Value));
|
|
}
|
|
}
|
|
|
|
void ScalarEnumerationTraits<CPUType>::enumeration(IO &io, CPUType &Cpu) {
|
|
auto CpuNames = getCPUTypeNames();
|
|
for (const auto &E : CpuNames) {
|
|
io.enumCase(Cpu, E.Name.str().c_str(), static_cast<CPUType>(E.Value));
|
|
}
|
|
}
|
|
|
|
void ScalarEnumerationTraits<RegisterId>::enumeration(IO &io, RegisterId &Reg) {
|
|
auto RegNames = getRegisterNames();
|
|
for (const auto &E : RegNames) {
|
|
io.enumCase(Reg, E.Name.str().c_str(), static_cast<RegisterId>(E.Value));
|
|
}
|
|
io.enumFallback<Hex16>(Reg);
|
|
}
|
|
|
|
void ScalarEnumerationTraits<TrampolineType>::enumeration(
|
|
IO &io, TrampolineType &Tramp) {
|
|
auto TrampNames = getTrampolineNames();
|
|
for (const auto &E : TrampNames) {
|
|
io.enumCase(Tramp, E.Name.str().c_str(),
|
|
static_cast<TrampolineType>(E.Value));
|
|
}
|
|
}
|
|
|
|
void ScalarEnumerationTraits<ThunkOrdinal>::enumeration(IO &io,
|
|
ThunkOrdinal &Ord) {
|
|
auto ThunkNames = getThunkOrdinalNames();
|
|
for (const auto &E : ThunkNames) {
|
|
io.enumCase(Ord, E.Name.str().c_str(), static_cast<ThunkOrdinal>(E.Value));
|
|
}
|
|
}
|
|
|
|
void ScalarEnumerationTraits<FrameCookieKind>::enumeration(
|
|
IO &io, FrameCookieKind &FC) {
|
|
auto ThunkNames = getFrameCookieKindNames();
|
|
for (const auto &E : ThunkNames) {
|
|
io.enumCase(FC, E.Name.str().c_str(),
|
|
static_cast<FrameCookieKind>(E.Value));
|
|
}
|
|
}
|
|
|
|
namespace llvm {
|
|
namespace CodeViewYAML {
|
|
namespace detail {
|
|
|
|
struct SymbolRecordBase {
|
|
codeview::SymbolKind Kind;
|
|
explicit SymbolRecordBase(codeview::SymbolKind K) : Kind(K) {}
|
|
|
|
virtual ~SymbolRecordBase() {}
|
|
virtual void map(yaml::IO &io) = 0;
|
|
virtual codeview::CVSymbol
|
|
toCodeViewSymbol(BumpPtrAllocator &Allocator,
|
|
CodeViewContainer Container) const = 0;
|
|
virtual Error fromCodeViewSymbol(codeview::CVSymbol Type) = 0;
|
|
};
|
|
|
|
template <typename T> struct SymbolRecordImpl : public SymbolRecordBase {
|
|
explicit SymbolRecordImpl(codeview::SymbolKind K)
|
|
: SymbolRecordBase(K), Symbol(static_cast<SymbolRecordKind>(K)) {}
|
|
|
|
void map(yaml::IO &io) override;
|
|
|
|
codeview::CVSymbol
|
|
toCodeViewSymbol(BumpPtrAllocator &Allocator,
|
|
CodeViewContainer Container) const override {
|
|
return SymbolSerializer::writeOneSymbol(Symbol, Allocator, Container);
|
|
}
|
|
Error fromCodeViewSymbol(codeview::CVSymbol CVS) override {
|
|
return SymbolDeserializer::deserializeAs<T>(CVS, Symbol);
|
|
}
|
|
|
|
mutable T Symbol;
|
|
};
|
|
|
|
struct UnknownSymbolRecord : public SymbolRecordBase {
|
|
explicit UnknownSymbolRecord(codeview::SymbolKind K) : SymbolRecordBase(K) {}
|
|
|
|
void map(yaml::IO &io) override;
|
|
|
|
CVSymbol toCodeViewSymbol(BumpPtrAllocator &Allocator,
|
|
CodeViewContainer Container) const override {
|
|
RecordPrefix Prefix;
|
|
uint32_t TotalLen = sizeof(RecordPrefix) + Data.size();
|
|
Prefix.RecordKind = Kind;
|
|
Prefix.RecordLen = TotalLen - 2;
|
|
uint8_t *Buffer = Allocator.Allocate<uint8_t>(TotalLen);
|
|
::memcpy(Buffer, &Prefix, sizeof(RecordPrefix));
|
|
::memcpy(Buffer + sizeof(RecordPrefix), Data.data(), Data.size());
|
|
return CVSymbol(Kind, ArrayRef<uint8_t>(Buffer, TotalLen));
|
|
}
|
|
Error fromCodeViewSymbol(CVSymbol CVS) override {
|
|
this->Kind = CVS.kind();
|
|
Data = CVS.RecordData.drop_front(sizeof(RecordPrefix));
|
|
return Error::success();
|
|
}
|
|
|
|
std::vector<uint8_t> Data;
|
|
};
|
|
|
|
template <> void SymbolRecordImpl<ScopeEndSym>::map(IO &IO) {}
|
|
|
|
void UnknownSymbolRecord::map(yaml::IO &io) {
|
|
yaml::BinaryRef Binary;
|
|
if (io.outputting())
|
|
Binary = yaml::BinaryRef(Data);
|
|
io.mapRequired("Data", Binary);
|
|
if (!io.outputting()) {
|
|
std::string Str;
|
|
raw_string_ostream OS(Str);
|
|
Binary.writeAsBinary(OS);
|
|
OS.flush();
|
|
Data.assign(Str.begin(), Str.end());
|
|
}
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<Thunk32Sym>::map(IO &IO) {
|
|
IO.mapRequired("Parent", Symbol.Parent);
|
|
IO.mapRequired("End", Symbol.End);
|
|
IO.mapRequired("Next", Symbol.Next);
|
|
IO.mapRequired("Off", Symbol.Offset);
|
|
IO.mapRequired("Seg", Symbol.Segment);
|
|
IO.mapRequired("Len", Symbol.Length);
|
|
IO.mapRequired("Ordinal", Symbol.Thunk);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<TrampolineSym>::map(IO &IO) {
|
|
IO.mapRequired("Type", Symbol.Type);
|
|
IO.mapRequired("Size", Symbol.Size);
|
|
IO.mapRequired("ThunkOff", Symbol.ThunkOffset);
|
|
IO.mapRequired("TargetOff", Symbol.TargetOffset);
|
|
IO.mapRequired("ThunkSection", Symbol.ThunkSection);
|
|
IO.mapRequired("TargetSection", Symbol.TargetSection);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<SectionSym>::map(IO &IO) {
|
|
IO.mapRequired("SectionNumber", Symbol.SectionNumber);
|
|
IO.mapRequired("Alignment", Symbol.Alignment);
|
|
IO.mapRequired("Rva", Symbol.Rva);
|
|
IO.mapRequired("Length", Symbol.Length);
|
|
IO.mapRequired("Characteristics", Symbol.Characteristics);
|
|
IO.mapRequired("Name", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<CoffGroupSym>::map(IO &IO) {
|
|
IO.mapRequired("Size", Symbol.Size);
|
|
IO.mapRequired("Characteristics", Symbol.Characteristics);
|
|
IO.mapRequired("Offset", Symbol.Offset);
|
|
IO.mapRequired("Segment", Symbol.Segment);
|
|
IO.mapRequired("Name", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<ExportSym>::map(IO &IO) {
|
|
IO.mapRequired("Ordinal", Symbol.Ordinal);
|
|
IO.mapRequired("Flags", Symbol.Flags);
|
|
IO.mapRequired("Name", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<ProcSym>::map(IO &IO) {
|
|
// TODO: Print the linkage name
|
|
|
|
IO.mapRequired("PtrParent", Symbol.Parent);
|
|
IO.mapRequired("PtrEnd", Symbol.End);
|
|
IO.mapRequired("PtrNext", Symbol.Next);
|
|
IO.mapRequired("CodeSize", Symbol.CodeSize);
|
|
IO.mapRequired("DbgStart", Symbol.DbgStart);
|
|
IO.mapRequired("DbgEnd", Symbol.DbgEnd);
|
|
IO.mapRequired("FunctionType", Symbol.FunctionType);
|
|
IO.mapRequired("Segment", Symbol.Segment);
|
|
IO.mapRequired("Flags", Symbol.Flags);
|
|
IO.mapRequired("DisplayName", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<RegisterSym>::map(IO &IO) {
|
|
IO.mapRequired("Type", Symbol.Index);
|
|
IO.mapRequired("Seg", Symbol.Register);
|
|
IO.mapRequired("Name", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<PublicSym32>::map(IO &IO) {
|
|
IO.mapRequired("Type", Symbol.Index);
|
|
IO.mapRequired("Seg", Symbol.Segment);
|
|
IO.mapRequired("Off", Symbol.Offset);
|
|
IO.mapRequired("Name", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<ProcRefSym>::map(IO &IO) {
|
|
IO.mapRequired("SumName", Symbol.SumName);
|
|
IO.mapRequired("SymOffset", Symbol.SymOffset);
|
|
IO.mapRequired("Mod", Symbol.Module);
|
|
IO.mapRequired("Name", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<EnvBlockSym>::map(IO &IO) {
|
|
IO.mapRequired("Entries", Symbol.Fields);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<InlineSiteSym>::map(IO &IO) {
|
|
IO.mapRequired("PtrParent", Symbol.Parent);
|
|
IO.mapRequired("PtrEnd", Symbol.End);
|
|
IO.mapRequired("Inlinee", Symbol.Inlinee);
|
|
// TODO: The binary annotations
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<LocalSym>::map(IO &IO) {
|
|
IO.mapRequired("Type", Symbol.Type);
|
|
IO.mapRequired("Flags", Symbol.Flags);
|
|
|
|
IO.mapRequired("VarName", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<DefRangeSym>::map(IO &IO) {
|
|
// TODO: Print the subfields
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<DefRangeSubfieldSym>::map(IO &IO) {
|
|
// TODO: Print the subfields
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<DefRangeRegisterSym>::map(IO &IO) {
|
|
// TODO: Print the subfields
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<DefRangeFramePointerRelSym>::map(IO &IO) {
|
|
// TODO: Print the subfields
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<DefRangeSubfieldRegisterSym>::map(IO &IO) {
|
|
// TODO: Print the subfields
|
|
}
|
|
|
|
template <>
|
|
void SymbolRecordImpl<DefRangeFramePointerRelFullScopeSym>::map(IO &IO) {
|
|
// TODO: Print the subfields
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<DefRangeRegisterRelSym>::map(IO &IO) {
|
|
// TODO: Print the subfields
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<BlockSym>::map(IO &IO) {
|
|
// TODO: Print the linkage name
|
|
IO.mapRequired("PtrParent", Symbol.Parent);
|
|
IO.mapRequired("PtrEnd", Symbol.End);
|
|
IO.mapRequired("CodeSize", Symbol.CodeSize);
|
|
IO.mapRequired("Segment", Symbol.Segment);
|
|
IO.mapRequired("BlockName", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<LabelSym>::map(IO &IO) {
|
|
// TODO: Print the linkage name
|
|
IO.mapRequired("Segment", Symbol.Segment);
|
|
IO.mapRequired("Flags", Symbol.Flags);
|
|
IO.mapRequired("Flags", Symbol.Flags);
|
|
IO.mapRequired("DisplayName", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<ObjNameSym>::map(IO &IO) {
|
|
IO.mapRequired("Signature", Symbol.Signature);
|
|
IO.mapRequired("ObjectName", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<Compile2Sym>::map(IO &IO) {
|
|
IO.mapRequired("Flags", Symbol.Flags);
|
|
IO.mapRequired("Machine", Symbol.Machine);
|
|
IO.mapRequired("FrontendMajor", Symbol.VersionFrontendMajor);
|
|
IO.mapRequired("FrontendMinor", Symbol.VersionFrontendMinor);
|
|
IO.mapRequired("FrontendBuild", Symbol.VersionFrontendBuild);
|
|
IO.mapRequired("BackendMajor", Symbol.VersionBackendMajor);
|
|
IO.mapRequired("BackendMinor", Symbol.VersionBackendMinor);
|
|
IO.mapRequired("BackendBuild", Symbol.VersionBackendBuild);
|
|
IO.mapRequired("Version", Symbol.Version);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<Compile3Sym>::map(IO &IO) {
|
|
IO.mapRequired("Flags", Symbol.Flags);
|
|
IO.mapRequired("Machine", Symbol.Machine);
|
|
IO.mapRequired("FrontendMajor", Symbol.VersionFrontendMajor);
|
|
IO.mapRequired("FrontendMinor", Symbol.VersionFrontendMinor);
|
|
IO.mapRequired("FrontendBuild", Symbol.VersionFrontendBuild);
|
|
IO.mapRequired("FrontendQFE", Symbol.VersionFrontendQFE);
|
|
IO.mapRequired("BackendMajor", Symbol.VersionBackendMajor);
|
|
IO.mapRequired("BackendMinor", Symbol.VersionBackendMinor);
|
|
IO.mapRequired("BackendBuild", Symbol.VersionBackendBuild);
|
|
IO.mapRequired("BackendQFE", Symbol.VersionBackendQFE);
|
|
IO.mapRequired("Version", Symbol.Version);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<FrameProcSym>::map(IO &IO) {
|
|
IO.mapRequired("TotalFrameBytes", Symbol.TotalFrameBytes);
|
|
IO.mapRequired("PaddingFrameBytes", Symbol.PaddingFrameBytes);
|
|
IO.mapRequired("OffsetToPadding", Symbol.OffsetToPadding);
|
|
IO.mapRequired("BytesOfCalleeSavedRegisters",
|
|
Symbol.BytesOfCalleeSavedRegisters);
|
|
IO.mapRequired("OffsetOfExceptionHandler", Symbol.OffsetOfExceptionHandler);
|
|
IO.mapRequired("SectionIdOfExceptionHandler",
|
|
Symbol.SectionIdOfExceptionHandler);
|
|
IO.mapRequired("Flags", Symbol.Flags);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<CallSiteInfoSym>::map(IO &IO) {
|
|
// TODO: Map Linkage Name
|
|
IO.mapRequired("Segment", Symbol.Segment);
|
|
IO.mapRequired("Type", Symbol.Type);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<FileStaticSym>::map(IO &IO) {
|
|
IO.mapRequired("Index", Symbol.Index);
|
|
IO.mapRequired("ModFilenameOffset", Symbol.ModFilenameOffset);
|
|
IO.mapRequired("Flags", Symbol.Flags);
|
|
IO.mapRequired("Name", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<HeapAllocationSiteSym>::map(IO &IO) {
|
|
// TODO: Map Linkage Name
|
|
IO.mapRequired("Segment", Symbol.Segment);
|
|
IO.mapRequired("CallInstructionSize", Symbol.CallInstructionSize);
|
|
IO.mapRequired("Type", Symbol.Type);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<FrameCookieSym>::map(IO &IO) {
|
|
// TODO: Map Linkage Name
|
|
IO.mapRequired("Register", Symbol.Register);
|
|
IO.mapRequired("CookieKind", Symbol.CookieKind);
|
|
IO.mapRequired("Flags", Symbol.Flags);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<CallerSym>::map(IO &IO) {
|
|
IO.mapRequired("FuncID", Symbol.Indices);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<UDTSym>::map(IO &IO) {
|
|
IO.mapRequired("Type", Symbol.Type);
|
|
IO.mapRequired("UDTName", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<BuildInfoSym>::map(IO &IO) {
|
|
IO.mapRequired("BuildId", Symbol.BuildId);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<BPRelativeSym>::map(IO &IO) {
|
|
IO.mapRequired("Offset", Symbol.Offset);
|
|
IO.mapRequired("Type", Symbol.Type);
|
|
IO.mapRequired("VarName", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<RegRelativeSym>::map(IO &IO) {
|
|
IO.mapRequired("Offset", Symbol.Offset);
|
|
IO.mapRequired("Type", Symbol.Type);
|
|
IO.mapRequired("Register", Symbol.Register);
|
|
IO.mapRequired("VarName", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<ConstantSym>::map(IO &IO) {
|
|
IO.mapRequired("Type", Symbol.Type);
|
|
IO.mapRequired("Value", Symbol.Value);
|
|
IO.mapRequired("Name", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<DataSym>::map(IO &IO) {
|
|
// TODO: Map linkage name
|
|
IO.mapRequired("Type", Symbol.Type);
|
|
IO.mapRequired("DisplayName", Symbol.Name);
|
|
}
|
|
|
|
template <> void SymbolRecordImpl<ThreadLocalDataSym>::map(IO &IO) {
|
|
// TODO: Map linkage name
|
|
IO.mapRequired("Type", Symbol.Type);
|
|
IO.mapRequired("DisplayName", Symbol.Name);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
CVSymbol CodeViewYAML::SymbolRecord::toCodeViewSymbol(
|
|
BumpPtrAllocator &Allocator, CodeViewContainer Container) const {
|
|
return Symbol->toCodeViewSymbol(Allocator, Container);
|
|
}
|
|
|
|
namespace llvm {
|
|
namespace yaml {
|
|
template <> struct MappingTraits<SymbolRecordBase> {
|
|
static void mapping(IO &io, SymbolRecordBase &Record) { Record.map(io); }
|
|
};
|
|
}
|
|
}
|
|
|
|
template <typename SymbolType>
|
|
static inline Expected<CodeViewYAML::SymbolRecord>
|
|
fromCodeViewSymbolImpl(CVSymbol Symbol) {
|
|
CodeViewYAML::SymbolRecord Result;
|
|
|
|
auto Impl = std::make_shared<SymbolType>(Symbol.kind());
|
|
if (auto EC = Impl->fromCodeViewSymbol(Symbol))
|
|
return std::move(EC);
|
|
Result.Symbol = Impl;
|
|
return Result;
|
|
}
|
|
|
|
Expected<CodeViewYAML::SymbolRecord>
|
|
CodeViewYAML::SymbolRecord::fromCodeViewSymbol(CVSymbol Symbol) {
|
|
#define SYMBOL_RECORD(EnumName, EnumVal, ClassName) \
|
|
case EnumName: \
|
|
return fromCodeViewSymbolImpl<SymbolRecordImpl<ClassName>>(Symbol);
|
|
#define SYMBOL_RECORD_ALIAS(EnumName, EnumVal, AliasName, ClassName) \
|
|
SYMBOL_RECORD(EnumName, EnumVal, ClassName)
|
|
switch (Symbol.kind()) {
|
|
#include "llvm/DebugInfo/CodeView/CodeViewSymbols.def"
|
|
default:
|
|
return fromCodeViewSymbolImpl<UnknownSymbolRecord>(Symbol);
|
|
}
|
|
return make_error<CodeViewError>(cv_error_code::corrupt_record);
|
|
}
|
|
|
|
template <typename ConcreteType>
|
|
static void mapSymbolRecordImpl(IO &IO, const char *Class, SymbolKind Kind,
|
|
CodeViewYAML::SymbolRecord &Obj) {
|
|
if (!IO.outputting())
|
|
Obj.Symbol = std::make_shared<ConcreteType>(Kind);
|
|
|
|
IO.mapRequired(Class, *Obj.Symbol);
|
|
}
|
|
|
|
void MappingTraits<CodeViewYAML::SymbolRecord>::mapping(
|
|
IO &IO, CodeViewYAML::SymbolRecord &Obj) {
|
|
SymbolKind Kind;
|
|
if (IO.outputting())
|
|
Kind = Obj.Symbol->Kind;
|
|
IO.mapRequired("Kind", Kind);
|
|
|
|
#define SYMBOL_RECORD(EnumName, EnumVal, ClassName) \
|
|
case EnumName: \
|
|
mapSymbolRecordImpl<SymbolRecordImpl<ClassName>>(IO, #ClassName, Kind, \
|
|
Obj); \
|
|
break;
|
|
#define SYMBOL_RECORD_ALIAS(EnumName, EnumVal, AliasName, ClassName) \
|
|
SYMBOL_RECORD(EnumName, EnumVal, ClassName)
|
|
switch (Kind) {
|
|
#include "llvm/DebugInfo/CodeView/CodeViewSymbols.def"
|
|
default:
|
|
mapSymbolRecordImpl<UnknownSymbolRecord>(IO, "UnknownSym", Kind, Obj);
|
|
}
|
|
}
|