blob: 21f5a775c55a30d4118f461bba7f50a98c60c304 [file] [log] [blame]
Zachary Turner7120a472016-06-06 20:37:05 +00001//===- PdbYAML.cpp -------------------------------------------- *- C++ --*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "PdbYaml.h"
Zachary Turner2f951ce2016-08-31 21:42:26 +000011
Zachary Turnerc6d54da2016-09-09 17:46:17 +000012#include "YamlSerializationContext.h"
Zachary Turner3b147642016-10-08 01:12:01 +000013#include "YamlSymbolDumper.h"
14#include "YamlTypeDumper.h"
Zachary Turner7120a472016-06-06 20:37:05 +000015
Zachary Turner3b147642016-10-08 01:12:01 +000016#include "llvm/DebugInfo/CodeView/CVSymbolVisitor.h"
Zachary Turnerac5763e2016-08-18 16:49:29 +000017#include "llvm/DebugInfo/CodeView/CVTypeVisitor.h"
Zachary Turner3b147642016-10-08 01:12:01 +000018#include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
Zachary Turner407dec52017-03-13 14:57:45 +000019#include "llvm/DebugInfo/CodeView/SymbolSerializer.h"
Zachary Turner3b147642016-10-08 01:12:01 +000020#include "llvm/DebugInfo/CodeView/SymbolVisitorCallbackPipeline.h"
Zachary Turnerac5763e2016-08-18 16:49:29 +000021#include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
Zachary Turner4efa0a42016-11-08 22:24:53 +000022#include "llvm/DebugInfo/CodeView/TypeSerializer.h"
Zachary Turner2f951ce2016-08-31 21:42:26 +000023#include "llvm/DebugInfo/CodeView/TypeVisitorCallbackPipeline.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000024#include "llvm/DebugInfo/PDB/Native/PDBFile.h"
25#include "llvm/DebugInfo/PDB/Native/TpiHashing.h"
Zachary Turner8848a7a2016-07-06 18:05:57 +000026#include "llvm/DebugInfo/PDB/PDBExtras.h"
Zachary Turnerac5763e2016-08-18 16:49:29 +000027#include "llvm/DebugInfo/PDB/PDBTypes.h"
Zachary Turner7120a472016-06-06 20:37:05 +000028
29using namespace llvm;
30using namespace llvm::pdb;
31using namespace llvm::pdb::yaml;
Zachary Turnerac5763e2016-08-18 16:49:29 +000032using namespace llvm::yaml;
33
34LLVM_YAML_IS_FLOW_SEQUENCE_VECTOR(uint32_t)
35LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::StringRef)
36LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::pdb::yaml::NamedStreamMapping)
37LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::pdb::yaml::PdbDbiModuleInfo)
Zachary Turner3b147642016-10-08 01:12:01 +000038LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::pdb::yaml::PdbSymbolRecord)
Zachary Turnerac5763e2016-08-18 16:49:29 +000039LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::pdb::yaml::PdbTpiRecord)
40LLVM_YAML_IS_SEQUENCE_VECTOR(llvm::pdb::yaml::StreamBlockList)
Zachary Turner7120a472016-06-06 20:37:05 +000041
Zachary Turner8848a7a2016-07-06 18:05:57 +000042namespace llvm {
43namespace yaml {
Zachary Turnerac5763e2016-08-18 16:49:29 +000044
Zachary Turner8848a7a2016-07-06 18:05:57 +000045template <> struct ScalarTraits<llvm::pdb::PDB_UniqueId> {
46 static void output(const llvm::pdb::PDB_UniqueId &S, void *,
47 llvm::raw_ostream &OS) {
48 OS << S;
49 }
50
51 static StringRef input(StringRef Scalar, void *Ctx,
52 llvm::pdb::PDB_UniqueId &S) {
53 if (Scalar.size() != 38)
54 return "GUID strings are 38 characters long";
55 if (Scalar[0] != '{' || Scalar[37] != '}')
56 return "GUID is not enclosed in {}";
57 if (Scalar[9] != '-' || Scalar[14] != '-' || Scalar[19] != '-' ||
58 Scalar[24] != '-')
59 return "GUID sections are not properly delineated with dashes";
60
Saleem Abdulrasoolde9f00e2016-12-30 19:42:13 +000061 uint8_t *OutBuffer = S.Guid;
Zachary Turner8848a7a2016-07-06 18:05:57 +000062 for (auto Iter = Scalar.begin(); Iter != Scalar.end();) {
63 if (*Iter == '-' || *Iter == '{' || *Iter == '}') {
64 ++Iter;
65 continue;
66 }
67 uint8_t Value = (llvm::hexDigitValue(*Iter) << 4);
68 ++Iter;
69 Value |= llvm::hexDigitValue(*Iter);
70 ++Iter;
71 *OutBuffer++ = Value;
72 }
73
74 return "";
75 }
76
77 static bool mustQuote(StringRef Scalar) { return needsQuotes(Scalar); }
78};
Zachary Turnerdbeaea72016-07-11 21:45:26 +000079
80template <> struct ScalarEnumerationTraits<llvm::pdb::PDB_Machine> {
81 static void enumeration(IO &io, llvm::pdb::PDB_Machine &Value) {
82 io.enumCase(Value, "Invalid", PDB_Machine::Invalid);
83 io.enumCase(Value, "Am33", PDB_Machine::Am33);
84 io.enumCase(Value, "Amd64", PDB_Machine::Amd64);
85 io.enumCase(Value, "Arm", PDB_Machine::Arm);
86 io.enumCase(Value, "ArmNT", PDB_Machine::ArmNT);
87 io.enumCase(Value, "Ebc", PDB_Machine::Ebc);
88 io.enumCase(Value, "x86", PDB_Machine::x86);
89 io.enumCase(Value, "Ia64", PDB_Machine::Ia64);
90 io.enumCase(Value, "M32R", PDB_Machine::M32R);
91 io.enumCase(Value, "Mips16", PDB_Machine::Mips16);
92 io.enumCase(Value, "MipsFpu", PDB_Machine::MipsFpu);
93 io.enumCase(Value, "MipsFpu16", PDB_Machine::MipsFpu16);
94 io.enumCase(Value, "PowerPCFP", PDB_Machine::PowerPCFP);
95 io.enumCase(Value, "R4000", PDB_Machine::R4000);
96 io.enumCase(Value, "SH3", PDB_Machine::SH3);
97 io.enumCase(Value, "SH3DSP", PDB_Machine::SH3DSP);
98 io.enumCase(Value, "Thumb", PDB_Machine::Thumb);
99 io.enumCase(Value, "WceMipsV2", PDB_Machine::WceMipsV2);
100 }
101};
102
103template <> struct ScalarEnumerationTraits<llvm::pdb::PdbRaw_DbiVer> {
104 static void enumeration(IO &io, llvm::pdb::PdbRaw_DbiVer &Value) {
105 io.enumCase(Value, "V41", llvm::pdb::PdbRaw_DbiVer::PdbDbiVC41);
106 io.enumCase(Value, "V50", llvm::pdb::PdbRaw_DbiVer::PdbDbiV50);
107 io.enumCase(Value, "V60", llvm::pdb::PdbRaw_DbiVer::PdbDbiV60);
108 io.enumCase(Value, "V70", llvm::pdb::PdbRaw_DbiVer::PdbDbiV70);
109 io.enumCase(Value, "V110", llvm::pdb::PdbRaw_DbiVer::PdbDbiV110);
110 }
111};
112
113template <> struct ScalarEnumerationTraits<llvm::pdb::PdbRaw_ImplVer> {
114 static void enumeration(IO &io, llvm::pdb::PdbRaw_ImplVer &Value) {
115 io.enumCase(Value, "VC2", llvm::pdb::PdbRaw_ImplVer::PdbImplVC2);
116 io.enumCase(Value, "VC4", llvm::pdb::PdbRaw_ImplVer::PdbImplVC4);
117 io.enumCase(Value, "VC41", llvm::pdb::PdbRaw_ImplVer::PdbImplVC41);
118 io.enumCase(Value, "VC50", llvm::pdb::PdbRaw_ImplVer::PdbImplVC50);
119 io.enumCase(Value, "VC98", llvm::pdb::PdbRaw_ImplVer::PdbImplVC98);
120 io.enumCase(Value, "VC70Dep", llvm::pdb::PdbRaw_ImplVer::PdbImplVC70Dep);
121 io.enumCase(Value, "VC70", llvm::pdb::PdbRaw_ImplVer::PdbImplVC70);
122 io.enumCase(Value, "VC80", llvm::pdb::PdbRaw_ImplVer::PdbImplVC80);
123 io.enumCase(Value, "VC110", llvm::pdb::PdbRaw_ImplVer::PdbImplVC110);
124 io.enumCase(Value, "VC140", llvm::pdb::PdbRaw_ImplVer::PdbImplVC140);
125 }
126};
Zachary Turnerac5763e2016-08-18 16:49:29 +0000127
128template <> struct ScalarEnumerationTraits<llvm::pdb::PdbRaw_TpiVer> {
129 static void enumeration(IO &io, llvm::pdb::PdbRaw_TpiVer &Value) {
130 io.enumCase(Value, "VC40", llvm::pdb::PdbRaw_TpiVer::PdbTpiV40);
131 io.enumCase(Value, "VC41", llvm::pdb::PdbRaw_TpiVer::PdbTpiV41);
132 io.enumCase(Value, "VC50", llvm::pdb::PdbRaw_TpiVer::PdbTpiV50);
133 io.enumCase(Value, "VC70", llvm::pdb::PdbRaw_TpiVer::PdbTpiV70);
134 io.enumCase(Value, "VC80", llvm::pdb::PdbRaw_TpiVer::PdbTpiV80);
135 }
136};
Zachary Turner8848a7a2016-07-06 18:05:57 +0000137}
138}
139
Zachary Turnerdbeaea72016-07-11 21:45:26 +0000140void MappingTraits<PdbObject>::mapping(IO &IO, PdbObject &Obj) {
Zachary Turner407dec52017-03-13 14:57:45 +0000141 // Create a single serialization context that will be passed through the
142 // entire process of serializing / deserializing a Tpi Stream. This is
143 // especially important when we are going from Pdb -> Yaml because we need
144 // to maintain state in a TypeTableBuilder across mappings, and at the end of
145 // the entire process, we need to have one TypeTableBuilder that has every
146 // record.
147 pdb::yaml::SerializationContext Context(IO, Obj.Allocator);
148
149
Zachary Turnerdbeaea72016-07-11 21:45:26 +0000150 IO.mapOptional("MSF", Obj.Headers);
151 IO.mapOptional("StreamSizes", Obj.StreamSizes);
152 IO.mapOptional("StreamMap", Obj.StreamMap);
Zachary Turner760ad4d2017-01-20 22:42:09 +0000153 IO.mapOptional("StringTable", Obj.StringTable);
Zachary Turnerdbeaea72016-07-11 21:45:26 +0000154 IO.mapOptional("PdbStream", Obj.PdbStream);
Zachary Turner407dec52017-03-13 14:57:45 +0000155 IO.mapOptionalWithContext("DbiStream", Obj.DbiStream, Context);
156 IO.mapOptionalWithContext("TpiStream", Obj.TpiStream, Context);
157 IO.mapOptionalWithContext("IpiStream", Obj.IpiStream, Context);
Zachary Turnerdbeaea72016-07-11 21:45:26 +0000158}
159
Zachary Turnera3225b02016-07-29 20:56:36 +0000160void MappingTraits<MSFHeaders>::mapping(IO &IO, MSFHeaders &Obj) {
Zachary Turnerea4e6072017-03-15 22:18:53 +0000161 IO.mapOptional("SuperBlock", Obj.SuperBlock);
162 IO.mapOptional("NumDirectoryBlocks", Obj.NumDirectoryBlocks);
163 IO.mapOptional("DirectoryBlocks", Obj.DirectoryBlocks);
164 IO.mapOptional("NumStreams", Obj.NumStreams);
165 IO.mapOptional("FileSize", Obj.FileSize);
Zachary Turnerdbeaea72016-07-11 21:45:26 +0000166}
167
Zachary Turnerfaa554b2016-07-15 22:16:56 +0000168void MappingTraits<msf::SuperBlock>::mapping(IO &IO, msf::SuperBlock &SB) {
Zachary Turner1dc9fd32016-06-14 20:48:36 +0000169 if (!IO.outputting()) {
Zachary Turnerfaa554b2016-07-15 22:16:56 +0000170 ::memcpy(SB.MagicBytes, msf::Magic, sizeof(msf::Magic));
Zachary Turner1dc9fd32016-06-14 20:48:36 +0000171 }
172
Zachary Turnerea4e6072017-03-15 22:18:53 +0000173 using u32 = support::ulittle32_t;
174 IO.mapOptional("BlockSize", SB.BlockSize, u32(4096U));
175 IO.mapOptional("FreeBlockMap", SB.FreeBlockMapBlock, u32(0U));
176 IO.mapOptional("NumBlocks", SB.NumBlocks, u32(0U));
177 IO.mapOptional("NumDirectoryBytes", SB.NumDirectoryBytes, u32(0U));
178 IO.mapOptional("Unknown1", SB.Unknown1, u32(0U));
179 IO.mapOptional("BlockMapAddr", SB.BlockMapAddr, u32(0U));
Zachary Turner1dc9fd32016-06-14 20:48:36 +0000180}
181
182void MappingTraits<StreamBlockList>::mapping(IO &IO, StreamBlockList &SB) {
183 IO.mapRequired("Stream", SB.Blocks);
184}
185
Zachary Turner8848a7a2016-07-06 18:05:57 +0000186void MappingTraits<PdbInfoStream>::mapping(IO &IO, PdbInfoStream &Obj) {
Zachary Turnerea4e6072017-03-15 22:18:53 +0000187 IO.mapOptional("Age", Obj.Age, 1U);
188 IO.mapOptional("Guid", Obj.Guid);
189 IO.mapOptional("Signature", Obj.Signature, 0U);
190 IO.mapOptional("Version", Obj.Version, PdbImplVC70);
Zachary Turnerdbeaea72016-07-11 21:45:26 +0000191}
192
Zachary Turner407dec52017-03-13 14:57:45 +0000193void MappingContextTraits<PdbDbiStream, pdb::yaml::SerializationContext>::mapping(IO &IO, PdbDbiStream &Obj, pdb::yaml::SerializationContext &Context) {
Zachary Turnerea4e6072017-03-15 22:18:53 +0000194 IO.mapOptional("VerHeader", Obj.VerHeader, PdbDbiV70);
195 IO.mapOptional("Age", Obj.Age, 1U);
196 IO.mapOptional("BuildNumber", Obj.BuildNumber, uint16_t(0U));
197 IO.mapOptional("PdbDllVersion", Obj.PdbDllVersion, 0U);
198 IO.mapOptional("PdbDllRbld", Obj.PdbDllRbld, uint16_t(0U));
199 IO.mapOptional("Flags", Obj.Flags, uint16_t(1U));
200 IO.mapOptional("MachineType", Obj.MachineType, PDB_Machine::x86);
Zachary Turner407dec52017-03-13 14:57:45 +0000201 IO.mapOptionalWithContext("Modules", Obj.ModInfos, Context);
Zachary Turnerdbeaea72016-07-11 21:45:26 +0000202}
Zachary Turner5e534c72016-07-15 22:17:08 +0000203
Zachary Turner407dec52017-03-13 14:57:45 +0000204void MappingContextTraits<PdbTpiStream, pdb::yaml::SerializationContext>::mapping(
205 IO &IO, pdb::yaml::PdbTpiStream &Obj, pdb::yaml::SerializationContext &Context) {
Zachary Turnerea4e6072017-03-15 22:18:53 +0000206 IO.mapOptional("Version", Obj.Version, PdbTpiV80);
Zachary Turnerc6d54da2016-09-09 17:46:17 +0000207 IO.mapRequired("Records", Obj.Records, Context);
Zachary Turnerac5763e2016-08-18 16:49:29 +0000208}
209
Zachary Turner5e534c72016-07-15 22:17:08 +0000210void MappingTraits<NamedStreamMapping>::mapping(IO &IO,
211 NamedStreamMapping &Obj) {
212 IO.mapRequired("Name", Obj.StreamName);
213 IO.mapRequired("StreamNum", Obj.StreamNumber);
214}
Zachary Turnerd218c262016-07-22 15:46:37 +0000215
Zachary Turner407dec52017-03-13 14:57:45 +0000216void MappingContextTraits<PdbSymbolRecord, pdb::yaml::SerializationContext>::mapping(IO &IO, PdbSymbolRecord &Obj, pdb::yaml::SerializationContext &Context) {
Zachary Turner3b147642016-10-08 01:12:01 +0000217 codeview::SymbolVisitorCallbackPipeline Pipeline;
Zachary Turner407dec52017-03-13 14:57:45 +0000218 codeview::SymbolSerializer Serializer(Context.Allocator);
Zachary Turner3b147642016-10-08 01:12:01 +0000219 codeview::SymbolDeserializer Deserializer(nullptr);
220 codeview::yaml::YamlSymbolDumper Dumper(IO);
221
222 if (IO.outputting()) {
223 // For PDB to Yaml, deserialize into a high level record type, then dump it.
224 Pipeline.addCallbackToPipeline(Deserializer);
225 Pipeline.addCallbackToPipeline(Dumper);
226 } else {
Zachary Turner407dec52017-03-13 14:57:45 +0000227 // For the other way around, dump it into a concrete structure, and then
228 // serialize it into the CVRecord.
229 Pipeline.addCallbackToPipeline(Dumper);
230 Pipeline.addCallbackToPipeline(Serializer);
Zachary Turner3b147642016-10-08 01:12:01 +0000231 }
232
233 codeview::CVSymbolVisitor Visitor(Pipeline);
234 consumeError(Visitor.visitSymbolRecord(Obj.Record));
235}
236
Zachary Turner407dec52017-03-13 14:57:45 +0000237void MappingContextTraits<PdbModiStream, pdb::yaml::SerializationContext>::mapping(IO &IO, PdbModiStream &Obj, pdb::yaml::SerializationContext &Context) {
Zachary Turnerea4e6072017-03-15 22:18:53 +0000238 IO.mapOptional("Signature", Obj.Signature, 4U);
Zachary Turner407dec52017-03-13 14:57:45 +0000239 IO.mapRequired("Records", Obj.Symbols, Context);
Zachary Turner3b147642016-10-08 01:12:01 +0000240}
241
Zachary Turner407dec52017-03-13 14:57:45 +0000242void MappingContextTraits<PdbDbiModuleInfo, pdb::yaml::SerializationContext>::mapping(IO &IO, PdbDbiModuleInfo &Obj, pdb::yaml::SerializationContext &Context) {
Zachary Turnerd218c262016-07-22 15:46:37 +0000243 IO.mapRequired("Module", Obj.Mod);
Zachary Turnerea4e6072017-03-15 22:18:53 +0000244 IO.mapOptional("ObjFile", Obj.Obj, Obj.Mod);
Zachary Turnerd218c262016-07-22 15:46:37 +0000245 IO.mapOptional("SourceFiles", Obj.SourceFiles);
Zachary Turner407dec52017-03-13 14:57:45 +0000246 IO.mapOptionalWithContext("Modi", Obj.Modi, Context);
Zachary Turnerd218c262016-07-22 15:46:37 +0000247}
Zachary Turnerac5763e2016-08-18 16:49:29 +0000248
Zachary Turnerc6d54da2016-09-09 17:46:17 +0000249void MappingContextTraits<PdbTpiRecord, pdb::yaml::SerializationContext>::
250 mapping(IO &IO, pdb::yaml::PdbTpiRecord &Obj,
251 pdb::yaml::SerializationContext &Context) {
252 codeview::TypeVisitorCallbackPipeline Pipeline;
253 codeview::TypeDeserializer Deserializer;
Zachary Turner4efa0a42016-11-08 22:24:53 +0000254 codeview::TypeSerializer Serializer(Context.Allocator);
Zachary Turner620961d2016-09-14 23:00:02 +0000255 pdb::TpiHashUpdater Hasher;
Zachary Turnerc6d54da2016-09-09 17:46:17 +0000256
Zachary Turnerac5763e2016-08-18 16:49:29 +0000257 if (IO.outputting()) {
Zachary Turnerc6d54da2016-09-09 17:46:17 +0000258 // For PDB to Yaml, deserialize into a high level record type, then dump it.
Zachary Turner2f951ce2016-08-31 21:42:26 +0000259 Pipeline.addCallbackToPipeline(Deserializer);
Zachary Turnerc6d54da2016-09-09 17:46:17 +0000260 Pipeline.addCallbackToPipeline(Context.Dumper);
Zachary Turnerac5763e2016-08-18 16:49:29 +0000261 } else {
Zachary Turnerc6d54da2016-09-09 17:46:17 +0000262 // For Yaml to PDB, extract from the high level record type, then write it
263 // to bytes.
Zachary Turner4efa0a42016-11-08 22:24:53 +0000264
265 // This might be interpreted as a hack, but serializing FieldList
266 // sub-records requires having access to the same serializer being used by
267 // the FieldList itself.
268 Context.ActiveSerializer = &Serializer;
Zachary Turnerc6d54da2016-09-09 17:46:17 +0000269 Pipeline.addCallbackToPipeline(Context.Dumper);
270 Pipeline.addCallbackToPipeline(Serializer);
Zachary Turner620961d2016-09-14 23:00:02 +0000271 Pipeline.addCallbackToPipeline(Hasher);
Zachary Turnerc6d54da2016-09-09 17:46:17 +0000272 }
273
274 codeview::CVTypeVisitor Visitor(Pipeline);
275 consumeError(Visitor.visitTypeRecord(Obj.Record));
Zachary Turner4efa0a42016-11-08 22:24:53 +0000276 Context.ActiveSerializer = nullptr;
Zachary Turnerac5763e2016-08-18 16:49:29 +0000277}