xref: /freebsd/contrib/llvm-project/llvm/lib/ObjectYAML/CodeViewYAMLSymbols.cpp (revision 81ad626541db97eb356e2c1d4a20eb2a26a766ab)
10b57cec5SDimitry Andric //===- CodeViewYAMLSymbols.cpp - CodeView YAMLIO Symbol implementation ----===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric //
90b57cec5SDimitry Andric // This file defines classes for handling the YAML representation of CodeView
100b57cec5SDimitry Andric // Debug Info.
110b57cec5SDimitry Andric //
120b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
130b57cec5SDimitry Andric 
140b57cec5SDimitry Andric #include "llvm/ObjectYAML/CodeViewYAMLSymbols.h"
150b57cec5SDimitry Andric #include "llvm/ADT/ArrayRef.h"
160b57cec5SDimitry Andric #include "llvm/ADT/StringRef.h"
170b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/CodeView.h"
180b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/CodeViewError.h"
190b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/EnumTables.h"
200b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/RecordSerialization.h"
210b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
220b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolRecord.h"
230b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolSerializer.h"
240b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/TypeIndex.h"
250b57cec5SDimitry Andric #include "llvm/ObjectYAML/YAML.h"
260b57cec5SDimitry Andric #include "llvm/Support/Allocator.h"
270b57cec5SDimitry Andric #include "llvm/Support/Error.h"
28*81ad6265SDimitry Andric #include "llvm/Support/ScopedPrinter.h"
290b57cec5SDimitry Andric #include "llvm/Support/YAMLTraits.h"
300b57cec5SDimitry Andric #include <algorithm>
310b57cec5SDimitry Andric #include <cstdint>
320b57cec5SDimitry Andric #include <cstring>
330b57cec5SDimitry Andric #include <string>
340b57cec5SDimitry Andric #include <vector>
350b57cec5SDimitry Andric 
360b57cec5SDimitry Andric using namespace llvm;
370b57cec5SDimitry Andric using namespace llvm::codeview;
380b57cec5SDimitry Andric using namespace llvm::CodeViewYAML;
390b57cec5SDimitry Andric using namespace llvm::CodeViewYAML::detail;
400b57cec5SDimitry Andric using namespace llvm::yaml;
410b57cec5SDimitry Andric 
420b57cec5SDimitry Andric LLVM_YAML_IS_FLOW_SEQUENCE_VECTOR(TypeIndex)
430b57cec5SDimitry Andric LLVM_YAML_IS_SEQUENCE_VECTOR(LocalVariableAddrGap)
440b57cec5SDimitry Andric 
450b57cec5SDimitry Andric // We only need to declare these, the definitions are in CodeViewYAMLTypes.cpp
460b57cec5SDimitry Andric LLVM_YAML_DECLARE_SCALAR_TRAITS(APSInt, QuotingType::None)
470b57cec5SDimitry Andric LLVM_YAML_DECLARE_SCALAR_TRAITS(TypeIndex, QuotingType::None)
480b57cec5SDimitry Andric 
490b57cec5SDimitry Andric LLVM_YAML_DECLARE_ENUM_TRAITS(SymbolKind)
500b57cec5SDimitry Andric LLVM_YAML_DECLARE_ENUM_TRAITS(FrameCookieKind)
510b57cec5SDimitry Andric 
520b57cec5SDimitry Andric LLVM_YAML_DECLARE_BITSET_TRAITS(CompileSym2Flags)
530b57cec5SDimitry Andric LLVM_YAML_DECLARE_BITSET_TRAITS(CompileSym3Flags)
540b57cec5SDimitry Andric LLVM_YAML_DECLARE_BITSET_TRAITS(ExportFlags)
550b57cec5SDimitry Andric LLVM_YAML_DECLARE_BITSET_TRAITS(PublicSymFlags)
560b57cec5SDimitry Andric LLVM_YAML_DECLARE_BITSET_TRAITS(LocalSymFlags)
570b57cec5SDimitry Andric LLVM_YAML_DECLARE_BITSET_TRAITS(ProcSymFlags)
580b57cec5SDimitry Andric LLVM_YAML_DECLARE_BITSET_TRAITS(FrameProcedureOptions)
590b57cec5SDimitry Andric LLVM_YAML_DECLARE_ENUM_TRAITS(CPUType)
600b57cec5SDimitry Andric LLVM_YAML_DECLARE_ENUM_TRAITS(RegisterId)
610b57cec5SDimitry Andric LLVM_YAML_DECLARE_ENUM_TRAITS(TrampolineType)
620b57cec5SDimitry Andric LLVM_YAML_DECLARE_ENUM_TRAITS(ThunkOrdinal)
630b57cec5SDimitry Andric 
640b57cec5SDimitry Andric LLVM_YAML_STRONG_TYPEDEF(StringRef, TypeName)
650b57cec5SDimitry Andric 
660b57cec5SDimitry Andric LLVM_YAML_DECLARE_SCALAR_TRAITS(TypeName, QuotingType::Single)
670b57cec5SDimitry Andric 
680b57cec5SDimitry Andric StringRef ScalarTraits<TypeName>::input(StringRef S, void *V, TypeName &T) {
690b57cec5SDimitry Andric   return ScalarTraits<StringRef>::input(S, V, T.value);
700b57cec5SDimitry Andric }
710b57cec5SDimitry Andric 
720b57cec5SDimitry Andric void ScalarTraits<TypeName>::output(const TypeName &T, void *V,
730b57cec5SDimitry Andric                                     raw_ostream &R) {
740b57cec5SDimitry Andric   ScalarTraits<StringRef>::output(T.value, V, R);
750b57cec5SDimitry Andric }
760b57cec5SDimitry Andric 
770b57cec5SDimitry Andric void ScalarEnumerationTraits<SymbolKind>::enumeration(IO &io,
780b57cec5SDimitry Andric                                                       SymbolKind &Value) {
790b57cec5SDimitry Andric   auto SymbolNames = getSymbolTypeNames();
800b57cec5SDimitry Andric   for (const auto &E : SymbolNames)
810b57cec5SDimitry Andric     io.enumCase(Value, E.Name.str().c_str(), E.Value);
820b57cec5SDimitry Andric }
830b57cec5SDimitry Andric 
840b57cec5SDimitry Andric void ScalarBitSetTraits<CompileSym2Flags>::bitset(IO &io,
850b57cec5SDimitry Andric                                                   CompileSym2Flags &Flags) {
860b57cec5SDimitry Andric   auto FlagNames = getCompileSym2FlagNames();
870b57cec5SDimitry Andric   for (const auto &E : FlagNames) {
880b57cec5SDimitry Andric     io.bitSetCase(Flags, E.Name.str().c_str(),
890b57cec5SDimitry Andric                   static_cast<CompileSym2Flags>(E.Value));
900b57cec5SDimitry Andric   }
910b57cec5SDimitry Andric }
920b57cec5SDimitry Andric 
930b57cec5SDimitry Andric void ScalarBitSetTraits<CompileSym3Flags>::bitset(IO &io,
940b57cec5SDimitry Andric                                                   CompileSym3Flags &Flags) {
950b57cec5SDimitry Andric   auto FlagNames = getCompileSym3FlagNames();
960b57cec5SDimitry Andric   for (const auto &E : FlagNames) {
970b57cec5SDimitry Andric     io.bitSetCase(Flags, E.Name.str().c_str(),
980b57cec5SDimitry Andric                   static_cast<CompileSym3Flags>(E.Value));
990b57cec5SDimitry Andric   }
1000b57cec5SDimitry Andric }
1010b57cec5SDimitry Andric 
1020b57cec5SDimitry Andric void ScalarBitSetTraits<ExportFlags>::bitset(IO &io, ExportFlags &Flags) {
1030b57cec5SDimitry Andric   auto FlagNames = getExportSymFlagNames();
1040b57cec5SDimitry Andric   for (const auto &E : FlagNames) {
1050b57cec5SDimitry Andric     io.bitSetCase(Flags, E.Name.str().c_str(),
1060b57cec5SDimitry Andric                   static_cast<ExportFlags>(E.Value));
1070b57cec5SDimitry Andric   }
1080b57cec5SDimitry Andric }
1090b57cec5SDimitry Andric 
1100b57cec5SDimitry Andric void ScalarBitSetTraits<PublicSymFlags>::bitset(IO &io, PublicSymFlags &Flags) {
1110b57cec5SDimitry Andric   auto FlagNames = getPublicSymFlagNames();
1120b57cec5SDimitry Andric   for (const auto &E : FlagNames) {
1130b57cec5SDimitry Andric     io.bitSetCase(Flags, E.Name.str().c_str(),
1140b57cec5SDimitry Andric                   static_cast<PublicSymFlags>(E.Value));
1150b57cec5SDimitry Andric   }
1160b57cec5SDimitry Andric }
1170b57cec5SDimitry Andric 
1180b57cec5SDimitry Andric void ScalarBitSetTraits<LocalSymFlags>::bitset(IO &io, LocalSymFlags &Flags) {
1190b57cec5SDimitry Andric   auto FlagNames = getLocalFlagNames();
1200b57cec5SDimitry Andric   for (const auto &E : FlagNames) {
1210b57cec5SDimitry Andric     io.bitSetCase(Flags, E.Name.str().c_str(),
1220b57cec5SDimitry Andric                   static_cast<LocalSymFlags>(E.Value));
1230b57cec5SDimitry Andric   }
1240b57cec5SDimitry Andric }
1250b57cec5SDimitry Andric 
1260b57cec5SDimitry Andric void ScalarBitSetTraits<ProcSymFlags>::bitset(IO &io, ProcSymFlags &Flags) {
1270b57cec5SDimitry Andric   auto FlagNames = getProcSymFlagNames();
1280b57cec5SDimitry Andric   for (const auto &E : FlagNames) {
1290b57cec5SDimitry Andric     io.bitSetCase(Flags, E.Name.str().c_str(),
1300b57cec5SDimitry Andric                   static_cast<ProcSymFlags>(E.Value));
1310b57cec5SDimitry Andric   }
1320b57cec5SDimitry Andric }
1330b57cec5SDimitry Andric 
1340b57cec5SDimitry Andric void ScalarBitSetTraits<FrameProcedureOptions>::bitset(
1350b57cec5SDimitry Andric     IO &io, FrameProcedureOptions &Flags) {
1360b57cec5SDimitry Andric   auto FlagNames = getFrameProcSymFlagNames();
1370b57cec5SDimitry Andric   for (const auto &E : FlagNames) {
1380b57cec5SDimitry Andric     io.bitSetCase(Flags, E.Name.str().c_str(),
1390b57cec5SDimitry Andric                   static_cast<FrameProcedureOptions>(E.Value));
1400b57cec5SDimitry Andric   }
1410b57cec5SDimitry Andric }
1420b57cec5SDimitry Andric 
1430b57cec5SDimitry Andric void ScalarEnumerationTraits<CPUType>::enumeration(IO &io, CPUType &Cpu) {
1440b57cec5SDimitry Andric   auto CpuNames = getCPUTypeNames();
1450b57cec5SDimitry Andric   for (const auto &E : CpuNames) {
1460b57cec5SDimitry Andric     io.enumCase(Cpu, E.Name.str().c_str(), static_cast<CPUType>(E.Value));
1470b57cec5SDimitry Andric   }
1480b57cec5SDimitry Andric }
1490b57cec5SDimitry Andric 
1500b57cec5SDimitry Andric void ScalarEnumerationTraits<RegisterId>::enumeration(IO &io, RegisterId &Reg) {
151e8d8bef9SDimitry Andric   const auto *Header = static_cast<COFF::header *>(io.getContext());
152e8d8bef9SDimitry Andric   assert(Header && "The IO context is not initialized");
153e8d8bef9SDimitry Andric 
154e8d8bef9SDimitry Andric   Optional<CPUType> CpuType;
155e8d8bef9SDimitry Andric   ArrayRef<EnumEntry<uint16_t>> RegNames;
156e8d8bef9SDimitry Andric 
157e8d8bef9SDimitry Andric   switch (Header->Machine) {
158e8d8bef9SDimitry Andric   case COFF::IMAGE_FILE_MACHINE_I386:
159e8d8bef9SDimitry Andric     CpuType = CPUType::Pentium3;
160e8d8bef9SDimitry Andric     break;
161e8d8bef9SDimitry Andric   case COFF::IMAGE_FILE_MACHINE_AMD64:
162e8d8bef9SDimitry Andric     CpuType = CPUType::X64;
163e8d8bef9SDimitry Andric     break;
164e8d8bef9SDimitry Andric   case COFF::IMAGE_FILE_MACHINE_ARMNT:
165e8d8bef9SDimitry Andric     CpuType = CPUType::ARMNT;
166e8d8bef9SDimitry Andric     break;
167e8d8bef9SDimitry Andric   case COFF::IMAGE_FILE_MACHINE_ARM64:
168e8d8bef9SDimitry Andric     CpuType = CPUType::ARM64;
169e8d8bef9SDimitry Andric     break;
170e8d8bef9SDimitry Andric   }
171e8d8bef9SDimitry Andric 
172e8d8bef9SDimitry Andric   if (CpuType)
173e8d8bef9SDimitry Andric     RegNames = getRegisterNames(*CpuType);
174e8d8bef9SDimitry Andric 
1750b57cec5SDimitry Andric   for (const auto &E : RegNames) {
1760b57cec5SDimitry Andric     io.enumCase(Reg, E.Name.str().c_str(), static_cast<RegisterId>(E.Value));
1770b57cec5SDimitry Andric   }
1780b57cec5SDimitry Andric   io.enumFallback<Hex16>(Reg);
1790b57cec5SDimitry Andric }
1800b57cec5SDimitry Andric 
1810b57cec5SDimitry Andric void ScalarEnumerationTraits<TrampolineType>::enumeration(
1820b57cec5SDimitry Andric     IO &io, TrampolineType &Tramp) {
1830b57cec5SDimitry Andric   auto TrampNames = getTrampolineNames();
1840b57cec5SDimitry Andric   for (const auto &E : TrampNames) {
1850b57cec5SDimitry Andric     io.enumCase(Tramp, E.Name.str().c_str(),
1860b57cec5SDimitry Andric                 static_cast<TrampolineType>(E.Value));
1870b57cec5SDimitry Andric   }
1880b57cec5SDimitry Andric }
1890b57cec5SDimitry Andric 
1900b57cec5SDimitry Andric void ScalarEnumerationTraits<ThunkOrdinal>::enumeration(IO &io,
1910b57cec5SDimitry Andric                                                         ThunkOrdinal &Ord) {
1920b57cec5SDimitry Andric   auto ThunkNames = getThunkOrdinalNames();
1930b57cec5SDimitry Andric   for (const auto &E : ThunkNames) {
1940b57cec5SDimitry Andric     io.enumCase(Ord, E.Name.str().c_str(), static_cast<ThunkOrdinal>(E.Value));
1950b57cec5SDimitry Andric   }
1960b57cec5SDimitry Andric }
1970b57cec5SDimitry Andric 
1980b57cec5SDimitry Andric void ScalarEnumerationTraits<FrameCookieKind>::enumeration(
1990b57cec5SDimitry Andric     IO &io, FrameCookieKind &FC) {
2000b57cec5SDimitry Andric   auto ThunkNames = getFrameCookieKindNames();
2010b57cec5SDimitry Andric   for (const auto &E : ThunkNames) {
2020b57cec5SDimitry Andric     io.enumCase(FC, E.Name.str().c_str(),
2030b57cec5SDimitry Andric                 static_cast<FrameCookieKind>(E.Value));
2040b57cec5SDimitry Andric   }
2050b57cec5SDimitry Andric }
2060b57cec5SDimitry Andric 
2070b57cec5SDimitry Andric namespace llvm {
2080b57cec5SDimitry Andric namespace yaml {
2090b57cec5SDimitry Andric template <> struct MappingTraits<LocalVariableAddrRange> {
2100b57cec5SDimitry Andric   static void mapping(IO &io, LocalVariableAddrRange &Range) {
2110b57cec5SDimitry Andric     io.mapRequired("OffsetStart", Range.OffsetStart);
2120b57cec5SDimitry Andric     io.mapRequired("ISectStart", Range.ISectStart);
2130b57cec5SDimitry Andric     io.mapRequired("Range", Range.Range);
2140b57cec5SDimitry Andric   }
2150b57cec5SDimitry Andric };
2160b57cec5SDimitry Andric template <> struct MappingTraits<LocalVariableAddrGap> {
2170b57cec5SDimitry Andric   static void mapping(IO &io, LocalVariableAddrGap &Gap) {
2180b57cec5SDimitry Andric     io.mapRequired("GapStartOffset", Gap.GapStartOffset);
2190b57cec5SDimitry Andric     io.mapRequired("Range", Gap.Range);
2200b57cec5SDimitry Andric   }
2210b57cec5SDimitry Andric };
2220b57cec5SDimitry Andric } // namespace yaml
2230b57cec5SDimitry Andric } // namespace llvm
2240b57cec5SDimitry Andric 
2250b57cec5SDimitry Andric namespace llvm {
2260b57cec5SDimitry Andric namespace CodeViewYAML {
2270b57cec5SDimitry Andric namespace detail {
2280b57cec5SDimitry Andric 
2290b57cec5SDimitry Andric struct SymbolRecordBase {
2300b57cec5SDimitry Andric   codeview::SymbolKind Kind;
2310b57cec5SDimitry Andric 
2320b57cec5SDimitry Andric   explicit SymbolRecordBase(codeview::SymbolKind K) : Kind(K) {}
2330b57cec5SDimitry Andric   virtual ~SymbolRecordBase() = default;
2340b57cec5SDimitry Andric 
2350b57cec5SDimitry Andric   virtual void map(yaml::IO &io) = 0;
2360b57cec5SDimitry Andric   virtual codeview::CVSymbol
2370b57cec5SDimitry Andric   toCodeViewSymbol(BumpPtrAllocator &Allocator,
2380b57cec5SDimitry Andric                    CodeViewContainer Container) const = 0;
2390b57cec5SDimitry Andric   virtual Error fromCodeViewSymbol(codeview::CVSymbol Type) = 0;
2400b57cec5SDimitry Andric };
2410b57cec5SDimitry Andric 
2420b57cec5SDimitry Andric template <typename T> struct SymbolRecordImpl : public SymbolRecordBase {
2430b57cec5SDimitry Andric   explicit SymbolRecordImpl(codeview::SymbolKind K)
2440b57cec5SDimitry Andric       : SymbolRecordBase(K), Symbol(static_cast<SymbolRecordKind>(K)) {}
2450b57cec5SDimitry Andric 
2460b57cec5SDimitry Andric   void map(yaml::IO &io) override;
2470b57cec5SDimitry Andric 
2480b57cec5SDimitry Andric   codeview::CVSymbol
2490b57cec5SDimitry Andric   toCodeViewSymbol(BumpPtrAllocator &Allocator,
2500b57cec5SDimitry Andric                    CodeViewContainer Container) const override {
2510b57cec5SDimitry Andric     return SymbolSerializer::writeOneSymbol(Symbol, Allocator, Container);
2520b57cec5SDimitry Andric   }
2530b57cec5SDimitry Andric 
2540b57cec5SDimitry Andric   Error fromCodeViewSymbol(codeview::CVSymbol CVS) override {
2550b57cec5SDimitry Andric     return SymbolDeserializer::deserializeAs<T>(CVS, Symbol);
2560b57cec5SDimitry Andric   }
2570b57cec5SDimitry Andric 
2580b57cec5SDimitry Andric   mutable T Symbol;
2590b57cec5SDimitry Andric };
2600b57cec5SDimitry Andric 
2610b57cec5SDimitry Andric struct UnknownSymbolRecord : public SymbolRecordBase {
2620b57cec5SDimitry Andric   explicit UnknownSymbolRecord(codeview::SymbolKind K) : SymbolRecordBase(K) {}
2630b57cec5SDimitry Andric 
2640b57cec5SDimitry Andric   void map(yaml::IO &io) override;
2650b57cec5SDimitry Andric 
2660b57cec5SDimitry Andric   CVSymbol toCodeViewSymbol(BumpPtrAllocator &Allocator,
2670b57cec5SDimitry Andric                             CodeViewContainer Container) const override {
2680b57cec5SDimitry Andric     RecordPrefix Prefix;
2690b57cec5SDimitry Andric     uint32_t TotalLen = sizeof(RecordPrefix) + Data.size();
2700b57cec5SDimitry Andric     Prefix.RecordKind = Kind;
2710b57cec5SDimitry Andric     Prefix.RecordLen = TotalLen - 2;
2720b57cec5SDimitry Andric     uint8_t *Buffer = Allocator.Allocate<uint8_t>(TotalLen);
2730b57cec5SDimitry Andric     ::memcpy(Buffer, &Prefix, sizeof(RecordPrefix));
2740b57cec5SDimitry Andric     ::memcpy(Buffer + sizeof(RecordPrefix), Data.data(), Data.size());
2750b57cec5SDimitry Andric     return CVSymbol(ArrayRef<uint8_t>(Buffer, TotalLen));
2760b57cec5SDimitry Andric   }
2770b57cec5SDimitry Andric 
2780b57cec5SDimitry Andric   Error fromCodeViewSymbol(CVSymbol CVS) override {
2790b57cec5SDimitry Andric     this->Kind = CVS.kind();
2800b57cec5SDimitry Andric     Data = CVS.RecordData.drop_front(sizeof(RecordPrefix));
2810b57cec5SDimitry Andric     return Error::success();
2820b57cec5SDimitry Andric   }
2830b57cec5SDimitry Andric 
2840b57cec5SDimitry Andric   std::vector<uint8_t> Data;
2850b57cec5SDimitry Andric };
2860b57cec5SDimitry Andric 
2870b57cec5SDimitry Andric template <> void SymbolRecordImpl<ScopeEndSym>::map(IO &IO) {}
2880b57cec5SDimitry Andric 
2890b57cec5SDimitry Andric void UnknownSymbolRecord::map(yaml::IO &io) {
2900b57cec5SDimitry Andric   yaml::BinaryRef Binary;
2910b57cec5SDimitry Andric   if (io.outputting())
2920b57cec5SDimitry Andric     Binary = yaml::BinaryRef(Data);
2930b57cec5SDimitry Andric   io.mapRequired("Data", Binary);
2940b57cec5SDimitry Andric   if (!io.outputting()) {
2950b57cec5SDimitry Andric     std::string Str;
2960b57cec5SDimitry Andric     raw_string_ostream OS(Str);
2970b57cec5SDimitry Andric     Binary.writeAsBinary(OS);
2980b57cec5SDimitry Andric     OS.flush();
2990b57cec5SDimitry Andric     Data.assign(Str.begin(), Str.end());
3000b57cec5SDimitry Andric   }
3010b57cec5SDimitry Andric }
3020b57cec5SDimitry Andric 
3030b57cec5SDimitry Andric template <> void SymbolRecordImpl<Thunk32Sym>::map(IO &IO) {
3040b57cec5SDimitry Andric   IO.mapRequired("Parent", Symbol.Parent);
3050b57cec5SDimitry Andric   IO.mapRequired("End", Symbol.End);
3060b57cec5SDimitry Andric   IO.mapRequired("Next", Symbol.Next);
3070b57cec5SDimitry Andric   IO.mapRequired("Off", Symbol.Offset);
3080b57cec5SDimitry Andric   IO.mapRequired("Seg", Symbol.Segment);
3090b57cec5SDimitry Andric   IO.mapRequired("Len", Symbol.Length);
3100b57cec5SDimitry Andric   IO.mapRequired("Ordinal", Symbol.Thunk);
3110b57cec5SDimitry Andric }
3120b57cec5SDimitry Andric 
3130b57cec5SDimitry Andric template <> void SymbolRecordImpl<TrampolineSym>::map(IO &IO) {
3140b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Type);
3150b57cec5SDimitry Andric   IO.mapRequired("Size", Symbol.Size);
3160b57cec5SDimitry Andric   IO.mapRequired("ThunkOff", Symbol.ThunkOffset);
3170b57cec5SDimitry Andric   IO.mapRequired("TargetOff", Symbol.TargetOffset);
3180b57cec5SDimitry Andric   IO.mapRequired("ThunkSection", Symbol.ThunkSection);
3190b57cec5SDimitry Andric   IO.mapRequired("TargetSection", Symbol.TargetSection);
3200b57cec5SDimitry Andric }
3210b57cec5SDimitry Andric 
3220b57cec5SDimitry Andric template <> void SymbolRecordImpl<SectionSym>::map(IO &IO) {
3230b57cec5SDimitry Andric   IO.mapRequired("SectionNumber", Symbol.SectionNumber);
3240b57cec5SDimitry Andric   IO.mapRequired("Alignment", Symbol.Alignment);
3250b57cec5SDimitry Andric   IO.mapRequired("Rva", Symbol.Rva);
3260b57cec5SDimitry Andric   IO.mapRequired("Length", Symbol.Length);
3270b57cec5SDimitry Andric   IO.mapRequired("Characteristics", Symbol.Characteristics);
3280b57cec5SDimitry Andric   IO.mapRequired("Name", Symbol.Name);
3290b57cec5SDimitry Andric }
3300b57cec5SDimitry Andric 
3310b57cec5SDimitry Andric template <> void SymbolRecordImpl<CoffGroupSym>::map(IO &IO) {
3320b57cec5SDimitry Andric   IO.mapRequired("Size", Symbol.Size);
3330b57cec5SDimitry Andric   IO.mapRequired("Characteristics", Symbol.Characteristics);
3340b57cec5SDimitry Andric   IO.mapRequired("Offset", Symbol.Offset);
3350b57cec5SDimitry Andric   IO.mapRequired("Segment", Symbol.Segment);
3360b57cec5SDimitry Andric   IO.mapRequired("Name", Symbol.Name);
3370b57cec5SDimitry Andric }
3380b57cec5SDimitry Andric 
3390b57cec5SDimitry Andric template <> void SymbolRecordImpl<ExportSym>::map(IO &IO) {
3400b57cec5SDimitry Andric   IO.mapRequired("Ordinal", Symbol.Ordinal);
3410b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
3420b57cec5SDimitry Andric   IO.mapRequired("Name", Symbol.Name);
3430b57cec5SDimitry Andric }
3440b57cec5SDimitry Andric 
3450b57cec5SDimitry Andric template <> void SymbolRecordImpl<ProcSym>::map(IO &IO) {
3460b57cec5SDimitry Andric   IO.mapOptional("PtrParent", Symbol.Parent, 0U);
3470b57cec5SDimitry Andric   IO.mapOptional("PtrEnd", Symbol.End, 0U);
3480b57cec5SDimitry Andric   IO.mapOptional("PtrNext", Symbol.Next, 0U);
3490b57cec5SDimitry Andric   IO.mapRequired("CodeSize", Symbol.CodeSize);
3500b57cec5SDimitry Andric   IO.mapRequired("DbgStart", Symbol.DbgStart);
3510b57cec5SDimitry Andric   IO.mapRequired("DbgEnd", Symbol.DbgEnd);
3520b57cec5SDimitry Andric   IO.mapRequired("FunctionType", Symbol.FunctionType);
3530b57cec5SDimitry Andric   IO.mapOptional("Offset", Symbol.CodeOffset, 0U);
3540b57cec5SDimitry Andric   IO.mapOptional("Segment", Symbol.Segment, uint16_t(0));
3550b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
3560b57cec5SDimitry Andric   IO.mapRequired("DisplayName", Symbol.Name);
3570b57cec5SDimitry Andric }
3580b57cec5SDimitry Andric 
3590b57cec5SDimitry Andric template <> void SymbolRecordImpl<RegisterSym>::map(IO &IO) {
3600b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Index);
3610b57cec5SDimitry Andric   IO.mapRequired("Seg", Symbol.Register);
3620b57cec5SDimitry Andric   IO.mapRequired("Name", Symbol.Name);
3630b57cec5SDimitry Andric }
3640b57cec5SDimitry Andric 
3650b57cec5SDimitry Andric template <> void SymbolRecordImpl<PublicSym32>::map(IO &IO) {
3660b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
3670b57cec5SDimitry Andric   IO.mapOptional("Offset", Symbol.Offset, 0U);
3680b57cec5SDimitry Andric   IO.mapOptional("Segment", Symbol.Segment, uint16_t(0));
3690b57cec5SDimitry Andric   IO.mapRequired("Name", Symbol.Name);
3700b57cec5SDimitry Andric }
3710b57cec5SDimitry Andric 
3720b57cec5SDimitry Andric template <> void SymbolRecordImpl<ProcRefSym>::map(IO &IO) {
3730b57cec5SDimitry Andric   IO.mapRequired("SumName", Symbol.SumName);
3740b57cec5SDimitry Andric   IO.mapRequired("SymOffset", Symbol.SymOffset);
3750b57cec5SDimitry Andric   IO.mapRequired("Mod", Symbol.Module);
3760b57cec5SDimitry Andric   IO.mapRequired("Name", Symbol.Name);
3770b57cec5SDimitry Andric }
3780b57cec5SDimitry Andric 
3790b57cec5SDimitry Andric template <> void SymbolRecordImpl<EnvBlockSym>::map(IO &IO) {
3800b57cec5SDimitry Andric   IO.mapRequired("Entries", Symbol.Fields);
3810b57cec5SDimitry Andric }
3820b57cec5SDimitry Andric 
3830b57cec5SDimitry Andric template <> void SymbolRecordImpl<InlineSiteSym>::map(IO &IO) {
3840b57cec5SDimitry Andric   IO.mapOptional("PtrParent", Symbol.Parent, 0U);
3850b57cec5SDimitry Andric   IO.mapOptional("PtrEnd", Symbol.End, 0U);
3860b57cec5SDimitry Andric   IO.mapRequired("Inlinee", Symbol.Inlinee);
3870b57cec5SDimitry Andric   // TODO: The binary annotations
3880b57cec5SDimitry Andric }
3890b57cec5SDimitry Andric 
3900b57cec5SDimitry Andric template <> void SymbolRecordImpl<LocalSym>::map(IO &IO) {
3910b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Type);
3920b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
3930b57cec5SDimitry Andric 
3940b57cec5SDimitry Andric   IO.mapRequired("VarName", Symbol.Name);
3950b57cec5SDimitry Andric }
3960b57cec5SDimitry Andric 
3970b57cec5SDimitry Andric template <> void SymbolRecordImpl<DefRangeSym>::map(IO &IO) {
3980b57cec5SDimitry Andric   IO.mapRequired("Program", Symbol.Program);
3990b57cec5SDimitry Andric   IO.mapRequired("Range", Symbol.Range);
4000b57cec5SDimitry Andric   IO.mapRequired("Gaps", Symbol.Gaps);
4010b57cec5SDimitry Andric }
4020b57cec5SDimitry Andric 
4030b57cec5SDimitry Andric template <> void SymbolRecordImpl<DefRangeSubfieldSym>::map(IO &IO) {
4040b57cec5SDimitry Andric   IO.mapRequired("Program", Symbol.Program);
4050b57cec5SDimitry Andric   IO.mapRequired("OffsetInParent", Symbol.OffsetInParent);
4060b57cec5SDimitry Andric   IO.mapRequired("Range", Symbol.Range);
4070b57cec5SDimitry Andric   IO.mapRequired("Gaps", Symbol.Gaps);
4080b57cec5SDimitry Andric }
4090b57cec5SDimitry Andric 
4100b57cec5SDimitry Andric template <> void SymbolRecordImpl<DefRangeRegisterSym>::map(IO &IO) {
4110b57cec5SDimitry Andric   IO.mapRequired("Register", Symbol.Hdr.Register);
4120b57cec5SDimitry Andric   IO.mapRequired("MayHaveNoName", Symbol.Hdr.MayHaveNoName);
4130b57cec5SDimitry Andric   IO.mapRequired("Range", Symbol.Range);
4140b57cec5SDimitry Andric   IO.mapRequired("Gaps", Symbol.Gaps);
4150b57cec5SDimitry Andric }
4160b57cec5SDimitry Andric 
4170b57cec5SDimitry Andric template <> void SymbolRecordImpl<DefRangeFramePointerRelSym>::map(IO &IO) {
4188bcb0991SDimitry Andric   IO.mapRequired("Offset", Symbol.Hdr.Offset);
4190b57cec5SDimitry Andric   IO.mapRequired("Range", Symbol.Range);
4200b57cec5SDimitry Andric   IO.mapRequired("Gaps", Symbol.Gaps);
4210b57cec5SDimitry Andric }
4220b57cec5SDimitry Andric 
4230b57cec5SDimitry Andric template <> void SymbolRecordImpl<DefRangeSubfieldRegisterSym>::map(IO &IO) {
4240b57cec5SDimitry Andric   IO.mapRequired("Register", Symbol.Hdr.Register);
4250b57cec5SDimitry Andric   IO.mapRequired("MayHaveNoName", Symbol.Hdr.MayHaveNoName);
4260b57cec5SDimitry Andric   IO.mapRequired("OffsetInParent", Symbol.Hdr.OffsetInParent);
4270b57cec5SDimitry Andric   IO.mapRequired("Range", Symbol.Range);
4280b57cec5SDimitry Andric   IO.mapRequired("Gaps", Symbol.Gaps);
4290b57cec5SDimitry Andric }
4300b57cec5SDimitry Andric 
4310b57cec5SDimitry Andric template <>
4320b57cec5SDimitry Andric void SymbolRecordImpl<DefRangeFramePointerRelFullScopeSym>::map(IO &IO) {
4330b57cec5SDimitry Andric   IO.mapRequired("Register", Symbol.Offset);
4340b57cec5SDimitry Andric }
4350b57cec5SDimitry Andric 
4360b57cec5SDimitry Andric template <> void SymbolRecordImpl<DefRangeRegisterRelSym>::map(IO &IO) {
4370b57cec5SDimitry Andric   IO.mapRequired("Register", Symbol.Hdr.Register);
4380b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Hdr.Flags);
4390b57cec5SDimitry Andric   IO.mapRequired("BasePointerOffset", Symbol.Hdr.BasePointerOffset);
4400b57cec5SDimitry Andric   IO.mapRequired("Range", Symbol.Range);
4410b57cec5SDimitry Andric   IO.mapRequired("Gaps", Symbol.Gaps);
4420b57cec5SDimitry Andric }
4430b57cec5SDimitry Andric 
4440b57cec5SDimitry Andric template <> void SymbolRecordImpl<BlockSym>::map(IO &IO) {
4450b57cec5SDimitry Andric   IO.mapOptional("PtrParent", Symbol.Parent, 0U);
4460b57cec5SDimitry Andric   IO.mapOptional("PtrEnd", Symbol.End, 0U);
4470b57cec5SDimitry Andric   IO.mapRequired("CodeSize", Symbol.CodeSize);
4480b57cec5SDimitry Andric   IO.mapOptional("Offset", Symbol.CodeOffset, 0U);
4490b57cec5SDimitry Andric   IO.mapOptional("Segment", Symbol.Segment, uint16_t(0));
4500b57cec5SDimitry Andric   IO.mapRequired("BlockName", Symbol.Name);
4510b57cec5SDimitry Andric }
4520b57cec5SDimitry Andric 
4530b57cec5SDimitry Andric template <> void SymbolRecordImpl<LabelSym>::map(IO &IO) {
4540b57cec5SDimitry Andric   IO.mapOptional("Offset", Symbol.CodeOffset, 0U);
4550b57cec5SDimitry Andric   IO.mapOptional("Segment", Symbol.Segment, uint16_t(0));
4560b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
4570b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
4580b57cec5SDimitry Andric   IO.mapRequired("DisplayName", Symbol.Name);
4590b57cec5SDimitry Andric }
4600b57cec5SDimitry Andric 
4610b57cec5SDimitry Andric template <> void SymbolRecordImpl<ObjNameSym>::map(IO &IO) {
4620b57cec5SDimitry Andric   IO.mapRequired("Signature", Symbol.Signature);
4630b57cec5SDimitry Andric   IO.mapRequired("ObjectName", Symbol.Name);
4640b57cec5SDimitry Andric }
4650b57cec5SDimitry Andric 
4660b57cec5SDimitry Andric template <> void SymbolRecordImpl<Compile2Sym>::map(IO &IO) {
4670b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
4680b57cec5SDimitry Andric   IO.mapRequired("Machine", Symbol.Machine);
4690b57cec5SDimitry Andric   IO.mapRequired("FrontendMajor", Symbol.VersionFrontendMajor);
4700b57cec5SDimitry Andric   IO.mapRequired("FrontendMinor", Symbol.VersionFrontendMinor);
4710b57cec5SDimitry Andric   IO.mapRequired("FrontendBuild", Symbol.VersionFrontendBuild);
4720b57cec5SDimitry Andric   IO.mapRequired("BackendMajor", Symbol.VersionBackendMajor);
4730b57cec5SDimitry Andric   IO.mapRequired("BackendMinor", Symbol.VersionBackendMinor);
4740b57cec5SDimitry Andric   IO.mapRequired("BackendBuild", Symbol.VersionBackendBuild);
4750b57cec5SDimitry Andric   IO.mapRequired("Version", Symbol.Version);
4760b57cec5SDimitry Andric }
4770b57cec5SDimitry Andric 
4780b57cec5SDimitry Andric template <> void SymbolRecordImpl<Compile3Sym>::map(IO &IO) {
4790b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
4800b57cec5SDimitry Andric   IO.mapRequired("Machine", Symbol.Machine);
4810b57cec5SDimitry Andric   IO.mapRequired("FrontendMajor", Symbol.VersionFrontendMajor);
4820b57cec5SDimitry Andric   IO.mapRequired("FrontendMinor", Symbol.VersionFrontendMinor);
4830b57cec5SDimitry Andric   IO.mapRequired("FrontendBuild", Symbol.VersionFrontendBuild);
4840b57cec5SDimitry Andric   IO.mapRequired("FrontendQFE", Symbol.VersionFrontendQFE);
4850b57cec5SDimitry Andric   IO.mapRequired("BackendMajor", Symbol.VersionBackendMajor);
4860b57cec5SDimitry Andric   IO.mapRequired("BackendMinor", Symbol.VersionBackendMinor);
4870b57cec5SDimitry Andric   IO.mapRequired("BackendBuild", Symbol.VersionBackendBuild);
4880b57cec5SDimitry Andric   IO.mapRequired("BackendQFE", Symbol.VersionBackendQFE);
4890b57cec5SDimitry Andric   IO.mapRequired("Version", Symbol.Version);
4900b57cec5SDimitry Andric }
4910b57cec5SDimitry Andric 
4920b57cec5SDimitry Andric template <> void SymbolRecordImpl<FrameProcSym>::map(IO &IO) {
4930b57cec5SDimitry Andric   IO.mapRequired("TotalFrameBytes", Symbol.TotalFrameBytes);
4940b57cec5SDimitry Andric   IO.mapRequired("PaddingFrameBytes", Symbol.PaddingFrameBytes);
4950b57cec5SDimitry Andric   IO.mapRequired("OffsetToPadding", Symbol.OffsetToPadding);
4960b57cec5SDimitry Andric   IO.mapRequired("BytesOfCalleeSavedRegisters",
4970b57cec5SDimitry Andric                  Symbol.BytesOfCalleeSavedRegisters);
4980b57cec5SDimitry Andric   IO.mapRequired("OffsetOfExceptionHandler", Symbol.OffsetOfExceptionHandler);
4990b57cec5SDimitry Andric   IO.mapRequired("SectionIdOfExceptionHandler",
5000b57cec5SDimitry Andric                  Symbol.SectionIdOfExceptionHandler);
5010b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
5020b57cec5SDimitry Andric }
5030b57cec5SDimitry Andric 
5040b57cec5SDimitry Andric template <> void SymbolRecordImpl<CallSiteInfoSym>::map(IO &IO) {
5050b57cec5SDimitry Andric   IO.mapOptional("Offset", Symbol.CodeOffset, 0U);
5060b57cec5SDimitry Andric   IO.mapOptional("Segment", Symbol.Segment, uint16_t(0));
5070b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Type);
5080b57cec5SDimitry Andric }
5090b57cec5SDimitry Andric 
5100b57cec5SDimitry Andric template <> void SymbolRecordImpl<FileStaticSym>::map(IO &IO) {
5110b57cec5SDimitry Andric   IO.mapRequired("Index", Symbol.Index);
5120b57cec5SDimitry Andric   IO.mapRequired("ModFilenameOffset", Symbol.ModFilenameOffset);
5130b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
5140b57cec5SDimitry Andric   IO.mapRequired("Name", Symbol.Name);
5150b57cec5SDimitry Andric }
5160b57cec5SDimitry Andric 
5170b57cec5SDimitry Andric template <> void SymbolRecordImpl<HeapAllocationSiteSym>::map(IO &IO) {
5180b57cec5SDimitry Andric   IO.mapOptional("Offset", Symbol.CodeOffset, 0U);
5190b57cec5SDimitry Andric   IO.mapOptional("Segment", Symbol.Segment, uint16_t(0));
5200b57cec5SDimitry Andric   IO.mapRequired("CallInstructionSize", Symbol.CallInstructionSize);
5210b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Type);
5220b57cec5SDimitry Andric }
5230b57cec5SDimitry Andric 
5240b57cec5SDimitry Andric template <> void SymbolRecordImpl<FrameCookieSym>::map(IO &IO) {
5250b57cec5SDimitry Andric   IO.mapRequired("Register", Symbol.Register);
5260b57cec5SDimitry Andric   IO.mapRequired("CookieKind", Symbol.CookieKind);
5270b57cec5SDimitry Andric   IO.mapRequired("Flags", Symbol.Flags);
5280b57cec5SDimitry Andric }
5290b57cec5SDimitry Andric 
5300b57cec5SDimitry Andric template <> void SymbolRecordImpl<CallerSym>::map(IO &IO) {
5310b57cec5SDimitry Andric   IO.mapRequired("FuncID", Symbol.Indices);
5320b57cec5SDimitry Andric }
5330b57cec5SDimitry Andric 
5340b57cec5SDimitry Andric template <> void SymbolRecordImpl<UDTSym>::map(IO &IO) {
5350b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Type);
5360b57cec5SDimitry Andric   IO.mapRequired("UDTName", Symbol.Name);
5370b57cec5SDimitry Andric }
5380b57cec5SDimitry Andric 
5390b57cec5SDimitry Andric template <> void SymbolRecordImpl<BuildInfoSym>::map(IO &IO) {
5400b57cec5SDimitry Andric   IO.mapRequired("BuildId", Symbol.BuildId);
5410b57cec5SDimitry Andric }
5420b57cec5SDimitry Andric 
5430b57cec5SDimitry Andric template <> void SymbolRecordImpl<BPRelativeSym>::map(IO &IO) {
5440b57cec5SDimitry Andric   IO.mapRequired("Offset", Symbol.Offset);
5450b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Type);
5460b57cec5SDimitry Andric   IO.mapRequired("VarName", Symbol.Name);
5470b57cec5SDimitry Andric }
5480b57cec5SDimitry Andric 
5490b57cec5SDimitry Andric template <> void SymbolRecordImpl<RegRelativeSym>::map(IO &IO) {
5500b57cec5SDimitry Andric   IO.mapRequired("Offset", Symbol.Offset);
5510b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Type);
5520b57cec5SDimitry Andric   IO.mapRequired("Register", Symbol.Register);
5530b57cec5SDimitry Andric   IO.mapRequired("VarName", Symbol.Name);
5540b57cec5SDimitry Andric }
5550b57cec5SDimitry Andric 
5560b57cec5SDimitry Andric template <> void SymbolRecordImpl<ConstantSym>::map(IO &IO) {
5570b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Type);
5580b57cec5SDimitry Andric   IO.mapRequired("Value", Symbol.Value);
5590b57cec5SDimitry Andric   IO.mapRequired("Name", Symbol.Name);
5600b57cec5SDimitry Andric }
5610b57cec5SDimitry Andric 
5620b57cec5SDimitry Andric template <> void SymbolRecordImpl<DataSym>::map(IO &IO) {
5630b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Type);
5640b57cec5SDimitry Andric   IO.mapOptional("Offset", Symbol.DataOffset, 0U);
5650b57cec5SDimitry Andric   IO.mapOptional("Segment", Symbol.Segment, uint16_t(0));
5660b57cec5SDimitry Andric   IO.mapRequired("DisplayName", Symbol.Name);
5670b57cec5SDimitry Andric }
5680b57cec5SDimitry Andric 
5690b57cec5SDimitry Andric template <> void SymbolRecordImpl<ThreadLocalDataSym>::map(IO &IO) {
5700b57cec5SDimitry Andric   IO.mapRequired("Type", Symbol.Type);
5710b57cec5SDimitry Andric   IO.mapOptional("Offset", Symbol.DataOffset, 0U);
5720b57cec5SDimitry Andric   IO.mapOptional("Segment", Symbol.Segment, uint16_t(0));
5730b57cec5SDimitry Andric   IO.mapRequired("DisplayName", Symbol.Name);
5740b57cec5SDimitry Andric }
5750b57cec5SDimitry Andric 
5760b57cec5SDimitry Andric template <> void SymbolRecordImpl<UsingNamespaceSym>::map(IO &IO) {
5770b57cec5SDimitry Andric   IO.mapRequired("Namespace", Symbol.Name);
5780b57cec5SDimitry Andric }
5790b57cec5SDimitry Andric 
5800b57cec5SDimitry Andric template <> void SymbolRecordImpl<AnnotationSym>::map(IO &IO) {
5810b57cec5SDimitry Andric   IO.mapOptional("Offset", Symbol.CodeOffset, 0U);
5820b57cec5SDimitry Andric   IO.mapOptional("Segment", Symbol.Segment, uint16_t(0));
5830b57cec5SDimitry Andric   IO.mapRequired("Strings", Symbol.Strings);
5840b57cec5SDimitry Andric }
5850b57cec5SDimitry Andric 
5860b57cec5SDimitry Andric } // end namespace detail
5870b57cec5SDimitry Andric } // end namespace CodeViewYAML
5880b57cec5SDimitry Andric } // end namespace llvm
5890b57cec5SDimitry Andric 
5900b57cec5SDimitry Andric CVSymbol CodeViewYAML::SymbolRecord::toCodeViewSymbol(
5910b57cec5SDimitry Andric     BumpPtrAllocator &Allocator, CodeViewContainer Container) const {
5920b57cec5SDimitry Andric   return Symbol->toCodeViewSymbol(Allocator, Container);
5930b57cec5SDimitry Andric }
5940b57cec5SDimitry Andric 
5950b57cec5SDimitry Andric namespace llvm {
5960b57cec5SDimitry Andric namespace yaml {
5970b57cec5SDimitry Andric 
5980b57cec5SDimitry Andric template <> struct MappingTraits<SymbolRecordBase> {
5990b57cec5SDimitry Andric   static void mapping(IO &io, SymbolRecordBase &Record) { Record.map(io); }
6000b57cec5SDimitry Andric };
6010b57cec5SDimitry Andric 
6020b57cec5SDimitry Andric } // end namespace yaml
6030b57cec5SDimitry Andric } // end namespace llvm
6040b57cec5SDimitry Andric 
6050b57cec5SDimitry Andric template <typename SymbolType>
6060b57cec5SDimitry Andric static inline Expected<CodeViewYAML::SymbolRecord>
6070b57cec5SDimitry Andric fromCodeViewSymbolImpl(CVSymbol Symbol) {
6080b57cec5SDimitry Andric   CodeViewYAML::SymbolRecord Result;
6090b57cec5SDimitry Andric 
6100b57cec5SDimitry Andric   auto Impl = std::make_shared<SymbolType>(Symbol.kind());
6110b57cec5SDimitry Andric   if (auto EC = Impl->fromCodeViewSymbol(Symbol))
6120b57cec5SDimitry Andric     return std::move(EC);
6130b57cec5SDimitry Andric   Result.Symbol = Impl;
6140b57cec5SDimitry Andric   return Result;
6150b57cec5SDimitry Andric }
6160b57cec5SDimitry Andric 
6170b57cec5SDimitry Andric Expected<CodeViewYAML::SymbolRecord>
6180b57cec5SDimitry Andric CodeViewYAML::SymbolRecord::fromCodeViewSymbol(CVSymbol Symbol) {
6190b57cec5SDimitry Andric #define SYMBOL_RECORD(EnumName, EnumVal, ClassName)                            \
6200b57cec5SDimitry Andric   case EnumName:                                                               \
6210b57cec5SDimitry Andric     return fromCodeViewSymbolImpl<SymbolRecordImpl<ClassName>>(Symbol);
6220b57cec5SDimitry Andric #define SYMBOL_RECORD_ALIAS(EnumName, EnumVal, AliasName, ClassName)           \
6230b57cec5SDimitry Andric   SYMBOL_RECORD(EnumName, EnumVal, ClassName)
6240b57cec5SDimitry Andric   switch (Symbol.kind()) {
6250b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/CodeViewSymbols.def"
6260b57cec5SDimitry Andric   default:
6270b57cec5SDimitry Andric     return fromCodeViewSymbolImpl<UnknownSymbolRecord>(Symbol);
6280b57cec5SDimitry Andric   }
6290b57cec5SDimitry Andric   return make_error<CodeViewError>(cv_error_code::corrupt_record);
6300b57cec5SDimitry Andric }
6310b57cec5SDimitry Andric 
6320b57cec5SDimitry Andric template <typename ConcreteType>
6330b57cec5SDimitry Andric static void mapSymbolRecordImpl(IO &IO, const char *Class, SymbolKind Kind,
6340b57cec5SDimitry Andric                                 CodeViewYAML::SymbolRecord &Obj) {
6350b57cec5SDimitry Andric   if (!IO.outputting())
6360b57cec5SDimitry Andric     Obj.Symbol = std::make_shared<ConcreteType>(Kind);
6370b57cec5SDimitry Andric 
6380b57cec5SDimitry Andric   IO.mapRequired(Class, *Obj.Symbol);
6390b57cec5SDimitry Andric }
6400b57cec5SDimitry Andric 
6410b57cec5SDimitry Andric void MappingTraits<CodeViewYAML::SymbolRecord>::mapping(
6420b57cec5SDimitry Andric     IO &IO, CodeViewYAML::SymbolRecord &Obj) {
6430b57cec5SDimitry Andric   SymbolKind Kind;
6440b57cec5SDimitry Andric   if (IO.outputting())
6450b57cec5SDimitry Andric     Kind = Obj.Symbol->Kind;
6460b57cec5SDimitry Andric   IO.mapRequired("Kind", Kind);
6470b57cec5SDimitry Andric 
6480b57cec5SDimitry Andric #define SYMBOL_RECORD(EnumName, EnumVal, ClassName)                            \
6490b57cec5SDimitry Andric   case EnumName:                                                               \
6500b57cec5SDimitry Andric     mapSymbolRecordImpl<SymbolRecordImpl<ClassName>>(IO, #ClassName, Kind,     \
6510b57cec5SDimitry Andric                                                      Obj);                     \
6520b57cec5SDimitry Andric     break;
6530b57cec5SDimitry Andric #define SYMBOL_RECORD_ALIAS(EnumName, EnumVal, AliasName, ClassName)           \
6540b57cec5SDimitry Andric   SYMBOL_RECORD(EnumName, EnumVal, ClassName)
6550b57cec5SDimitry Andric   switch (Kind) {
6560b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/CodeViewSymbols.def"
6570b57cec5SDimitry Andric   default:
6580b57cec5SDimitry Andric     mapSymbolRecordImpl<UnknownSymbolRecord>(IO, "UnknownSym", Kind, Obj);
6590b57cec5SDimitry Andric   }
6600b57cec5SDimitry Andric }
661