blob: 08a4bb715fac4f18d0f0fc34633838915f375aad [file] [log] [blame]
Zachary Turner1b88f4f2017-05-31 04:17:13 +00001//===- CodeViewYAMLDebugSections.cpp - CodeView YAMLIO debug sections -----===//
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// This file defines classes for handling the YAML representation of CodeView
11// Debug Info.
12//
13//===----------------------------------------------------------------------===//
14
15#include "llvm/ObjectYAML/CodeViewYAMLDebugSections.h"
16
17#include "llvm/ADT/StringExtras.h"
18#include "llvm/ADT/StringSwitch.h"
19#include "llvm/DebugInfo/CodeView/CodeViewError.h"
Zachary Turner92dcdda2017-06-02 19:49:14 +000020#include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h"
Zachary Turner349c18f2017-06-05 21:40:33 +000021#include "llvm/DebugInfo/CodeView/DebugCrossExSubsection.h"
22#include "llvm/DebugInfo/CodeView/DebugCrossImpSubsection.h"
Zachary Turnerdeb39132017-06-09 00:28:08 +000023#include "llvm/DebugInfo/CodeView/DebugFrameDataSubsection.h"
Zachary Turner92dcdda2017-06-02 19:49:14 +000024#include "llvm/DebugInfo/CodeView/DebugInlineeLinesSubsection.h"
25#include "llvm/DebugInfo/CodeView/DebugLinesSubsection.h"
26#include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h"
27#include "llvm/DebugInfo/CodeView/DebugSubsectionVisitor.h"
Zachary Turner3226fe92017-06-09 20:46:52 +000028#include "llvm/DebugInfo/CodeView/DebugSymbolRVASubsection.h"
Zachary Turnerdeb39132017-06-09 00:28:08 +000029#include "llvm/DebugInfo/CodeView/DebugSymbolsSubsection.h"
Zachary Turner1b88f4f2017-05-31 04:17:13 +000030#include "llvm/DebugInfo/CodeView/EnumTables.h"
31#include "llvm/DebugInfo/CodeView/SymbolRecord.h"
Zachary Turnerdeb39132017-06-09 00:28:08 +000032#include "llvm/DebugInfo/CodeView/SymbolSerializer.h"
33#include "llvm/ObjectYAML/CodeViewYAMLSymbols.h"
34#include "llvm/Support/BinaryStreamWriter.h"
Zachary Turner1b88f4f2017-05-31 04:17:13 +000035using namespace llvm;
36using namespace llvm::codeview;
37using namespace llvm::CodeViewYAML;
38using namespace llvm::CodeViewYAML::detail;
39using namespace llvm::yaml;
40
41LLVM_YAML_IS_SEQUENCE_VECTOR(SourceFileChecksumEntry)
42LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineEntry)
43LLVM_YAML_IS_SEQUENCE_VECTOR(SourceColumnEntry)
44LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineBlock)
45LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineInfo)
46LLVM_YAML_IS_SEQUENCE_VECTOR(InlineeSite)
47LLVM_YAML_IS_SEQUENCE_VECTOR(InlineeInfo)
Zachary Turner349c18f2017-06-05 21:40:33 +000048LLVM_YAML_IS_SEQUENCE_VECTOR(CrossModuleExport)
49LLVM_YAML_IS_SEQUENCE_VECTOR(YAMLCrossModuleImport)
Zachary Turner1b88f4f2017-05-31 04:17:13 +000050LLVM_YAML_IS_SEQUENCE_VECTOR(StringRef)
Zachary Turnerdeb39132017-06-09 00:28:08 +000051LLVM_YAML_IS_SEQUENCE_VECTOR(YAMLFrameData)
Zachary Turner349c18f2017-06-05 21:40:33 +000052LLVM_YAML_IS_FLOW_SEQUENCE_VECTOR(uint32_t)
Zachary Turner1b88f4f2017-05-31 04:17:13 +000053
54LLVM_YAML_DECLARE_SCALAR_TRAITS(HexFormattedString, false)
Zachary Turner92dcdda2017-06-02 19:49:14 +000055LLVM_YAML_DECLARE_ENUM_TRAITS(DebugSubsectionKind)
Zachary Turner1b88f4f2017-05-31 04:17:13 +000056LLVM_YAML_DECLARE_ENUM_TRAITS(FileChecksumKind)
57LLVM_YAML_DECLARE_BITSET_TRAITS(LineFlags)
58
Zachary Turner349c18f2017-06-05 21:40:33 +000059LLVM_YAML_DECLARE_MAPPING_TRAITS(CrossModuleExport)
Zachary Turnerdeb39132017-06-09 00:28:08 +000060LLVM_YAML_DECLARE_MAPPING_TRAITS(YAMLFrameData)
Zachary Turner349c18f2017-06-05 21:40:33 +000061LLVM_YAML_DECLARE_MAPPING_TRAITS(YAMLCrossModuleImport)
62LLVM_YAML_DECLARE_MAPPING_TRAITS(CrossModuleImportItem)
Zachary Turner92dcdda2017-06-02 19:49:14 +000063LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceLineEntry)
64LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceColumnEntry)
65LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceFileChecksumEntry)
66LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceLineBlock)
67LLVM_YAML_DECLARE_MAPPING_TRAITS(InlineeSite)
68
69namespace llvm {
70namespace CodeViewYAML {
71namespace detail {
72struct YAMLSubsectionBase {
73 explicit YAMLSubsectionBase(DebugSubsectionKind Kind) : Kind(Kind) {}
74 DebugSubsectionKind Kind;
75 virtual ~YAMLSubsectionBase() {}
76
77 virtual void map(IO &IO) = 0;
78 virtual std::unique_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +000079 toCodeViewSubsection(BumpPtrAllocator &Allocator,
80 DebugStringTableSubsection *UseStrings,
Zachary Turner92dcdda2017-06-02 19:49:14 +000081 DebugChecksumsSubsection *UseChecksums) const = 0;
82};
83}
84}
85}
86
87namespace {
88struct YAMLChecksumsSubsection : public YAMLSubsectionBase {
89 YAMLChecksumsSubsection()
90 : YAMLSubsectionBase(DebugSubsectionKind::FileChecksums) {}
91
92 void map(IO &IO) override;
93 std::unique_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +000094 toCodeViewSubsection(BumpPtrAllocator &Allocator,
95 DebugStringTableSubsection *Strings,
Zachary Turner92dcdda2017-06-02 19:49:14 +000096 DebugChecksumsSubsection *Checksums) const override;
97 static Expected<std::shared_ptr<YAMLChecksumsSubsection>>
98 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
99 const DebugChecksumsSubsectionRef &FC);
100
101 std::vector<SourceFileChecksumEntry> Checksums;
102};
103
104struct YAMLLinesSubsection : public YAMLSubsectionBase {
105 YAMLLinesSubsection() : YAMLSubsectionBase(DebugSubsectionKind::Lines) {}
106
107 void map(IO &IO) override;
108 std::unique_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000109 toCodeViewSubsection(BumpPtrAllocator &Allocator,
110 DebugStringTableSubsection *Strings,
Zachary Turner92dcdda2017-06-02 19:49:14 +0000111 DebugChecksumsSubsection *Checksums) const override;
112 static Expected<std::shared_ptr<YAMLLinesSubsection>>
113 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
114 const DebugChecksumsSubsectionRef &Checksums,
115 const DebugLinesSubsectionRef &Lines);
116
117 SourceLineInfo Lines;
118};
119
120struct YAMLInlineeLinesSubsection : public YAMLSubsectionBase {
121 YAMLInlineeLinesSubsection()
122 : YAMLSubsectionBase(DebugSubsectionKind::InlineeLines) {}
123
124 void map(IO &IO) override;
125 std::unique_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000126 toCodeViewSubsection(BumpPtrAllocator &Allocator,
127 DebugStringTableSubsection *Strings,
Zachary Turner92dcdda2017-06-02 19:49:14 +0000128 DebugChecksumsSubsection *Checksums) const override;
129 static Expected<std::shared_ptr<YAMLInlineeLinesSubsection>>
130 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
131 const DebugChecksumsSubsectionRef &Checksums,
132 const DebugInlineeLinesSubsectionRef &Lines);
133
134 InlineeInfo InlineeLines;
135};
Zachary Turner349c18f2017-06-05 21:40:33 +0000136
137struct YAMLCrossModuleExportsSubsection : public YAMLSubsectionBase {
138 YAMLCrossModuleExportsSubsection()
139 : YAMLSubsectionBase(DebugSubsectionKind::CrossScopeExports) {}
140
141 void map(IO &IO) override;
142 std::unique_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000143 toCodeViewSubsection(BumpPtrAllocator &Allocator,
144 DebugStringTableSubsection *Strings,
Zachary Turner349c18f2017-06-05 21:40:33 +0000145 DebugChecksumsSubsection *Checksums) const override;
146 static Expected<std::shared_ptr<YAMLCrossModuleExportsSubsection>>
147 fromCodeViewSubsection(const DebugCrossModuleExportsSubsectionRef &Exports);
148
149 std::vector<CrossModuleExport> Exports;
150};
151
152struct YAMLCrossModuleImportsSubsection : public YAMLSubsectionBase {
153 YAMLCrossModuleImportsSubsection()
154 : YAMLSubsectionBase(DebugSubsectionKind::CrossScopeImports) {}
155
156 void map(IO &IO) override;
157 std::unique_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000158 toCodeViewSubsection(BumpPtrAllocator &Allocator,
159 DebugStringTableSubsection *Strings,
Zachary Turner349c18f2017-06-05 21:40:33 +0000160 DebugChecksumsSubsection *Checksums) const override;
161 static Expected<std::shared_ptr<YAMLCrossModuleImportsSubsection>>
162 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
163 const DebugCrossModuleImportsSubsectionRef &Imports);
164
165 std::vector<YAMLCrossModuleImport> Imports;
166};
Zachary Turnerdeb39132017-06-09 00:28:08 +0000167
168struct YAMLSymbolsSubsection : public YAMLSubsectionBase {
169 YAMLSymbolsSubsection() : YAMLSubsectionBase(DebugSubsectionKind::Symbols) {}
170
171 void map(IO &IO) override;
172 std::unique_ptr<DebugSubsection>
173 toCodeViewSubsection(BumpPtrAllocator &Allocator,
174 DebugStringTableSubsection *Strings,
175 DebugChecksumsSubsection *Checksums) const override;
176 static Expected<std::shared_ptr<YAMLSymbolsSubsection>>
177 fromCodeViewSubsection(const DebugSymbolsSubsectionRef &Symbols);
178
179 std::vector<CodeViewYAML::SymbolRecord> Symbols;
180};
181
182struct YAMLStringTableSubsection : public YAMLSubsectionBase {
183 YAMLStringTableSubsection()
184 : YAMLSubsectionBase(DebugSubsectionKind::StringTable) {}
185
186 void map(IO &IO) override;
187 std::unique_ptr<DebugSubsection>
188 toCodeViewSubsection(BumpPtrAllocator &Allocator,
189 DebugStringTableSubsection *Strings,
190 DebugChecksumsSubsection *Checksums) const override;
191 static Expected<std::shared_ptr<YAMLStringTableSubsection>>
192 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings);
193
194 std::vector<StringRef> Strings;
195};
196
197struct YAMLFrameDataSubsection : public YAMLSubsectionBase {
198 YAMLFrameDataSubsection()
199 : YAMLSubsectionBase(DebugSubsectionKind::FrameData) {}
200
201 void map(IO &IO) override;
202 std::unique_ptr<DebugSubsection>
203 toCodeViewSubsection(BumpPtrAllocator &Allocator,
204 DebugStringTableSubsection *Strings,
205 DebugChecksumsSubsection *Checksums) const override;
206 static Expected<std::shared_ptr<YAMLFrameDataSubsection>>
207 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
208 const DebugFrameDataSubsectionRef &Frames);
209
210 std::vector<YAMLFrameData> Frames;
211};
Zachary Turner3226fe92017-06-09 20:46:52 +0000212
213struct YAMLCoffSymbolRVASubsection : public YAMLSubsectionBase {
214 YAMLCoffSymbolRVASubsection()
215 : YAMLSubsectionBase(DebugSubsectionKind::CoffSymbolRVA) {}
216
217 void map(IO &IO) override;
218 std::unique_ptr<DebugSubsection>
219 toCodeViewSubsection(BumpPtrAllocator &Allocator,
220 DebugStringTableSubsection *Strings,
221 DebugChecksumsSubsection *Checksums) const override;
222 static Expected<std::shared_ptr<YAMLCoffSymbolRVASubsection>>
223 fromCodeViewSubsection(const DebugSymbolRVASubsectionRef &RVAs);
224
225 std::vector<uint32_t> RVAs;
226};
Zachary Turner92dcdda2017-06-02 19:49:14 +0000227}
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000228
229void ScalarBitSetTraits<LineFlags>::bitset(IO &io, LineFlags &Flags) {
230 io.bitSetCase(Flags, "HasColumnInfo", LF_HaveColumns);
231 io.enumFallback<Hex16>(Flags);
232}
233
234void ScalarEnumerationTraits<FileChecksumKind>::enumeration(
235 IO &io, FileChecksumKind &Kind) {
236 io.enumCase(Kind, "None", FileChecksumKind::None);
237 io.enumCase(Kind, "MD5", FileChecksumKind::MD5);
238 io.enumCase(Kind, "SHA1", FileChecksumKind::SHA1);
239 io.enumCase(Kind, "SHA256", FileChecksumKind::SHA256);
240}
241
242void ScalarTraits<HexFormattedString>::output(const HexFormattedString &Value,
243 void *ctx, raw_ostream &Out) {
244 StringRef Bytes(reinterpret_cast<const char *>(Value.Bytes.data()),
245 Value.Bytes.size());
246 Out << toHex(Bytes);
247}
248
249StringRef ScalarTraits<HexFormattedString>::input(StringRef Scalar, void *ctxt,
250 HexFormattedString &Value) {
251 std::string H = fromHex(Scalar);
252 Value.Bytes.assign(H.begin(), H.end());
253 return StringRef();
254}
255
256void MappingTraits<SourceLineEntry>::mapping(IO &IO, SourceLineEntry &Obj) {
257 IO.mapRequired("Offset", Obj.Offset);
258 IO.mapRequired("LineStart", Obj.LineStart);
259 IO.mapRequired("IsStatement", Obj.IsStatement);
260 IO.mapRequired("EndDelta", Obj.EndDelta);
261}
262
263void MappingTraits<SourceColumnEntry>::mapping(IO &IO, SourceColumnEntry &Obj) {
264 IO.mapRequired("StartColumn", Obj.StartColumn);
265 IO.mapRequired("EndColumn", Obj.EndColumn);
266}
267
268void MappingTraits<SourceLineBlock>::mapping(IO &IO, SourceLineBlock &Obj) {
269 IO.mapRequired("FileName", Obj.FileName);
270 IO.mapRequired("Lines", Obj.Lines);
271 IO.mapRequired("Columns", Obj.Columns);
272}
273
Zachary Turner349c18f2017-06-05 21:40:33 +0000274void MappingTraits<CrossModuleExport>::mapping(IO &IO, CrossModuleExport &Obj) {
275 IO.mapRequired("LocalId", Obj.Local);
276 IO.mapRequired("GlobalId", Obj.Global);
277}
278
279void MappingTraits<YAMLCrossModuleImport>::mapping(IO &IO,
280 YAMLCrossModuleImport &Obj) {
281 IO.mapRequired("Module", Obj.ModuleName);
282 IO.mapRequired("Imports", Obj.ImportIds);
283}
284
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000285void MappingTraits<SourceFileChecksumEntry>::mapping(
286 IO &IO, SourceFileChecksumEntry &Obj) {
287 IO.mapRequired("FileName", Obj.FileName);
288 IO.mapRequired("Kind", Obj.Kind);
289 IO.mapRequired("Checksum", Obj.ChecksumBytes);
290}
291
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000292void MappingTraits<InlineeSite>::mapping(IO &IO, InlineeSite &Obj) {
293 IO.mapRequired("FileName", Obj.FileName);
294 IO.mapRequired("LineNum", Obj.SourceLineNum);
295 IO.mapRequired("Inlinee", Obj.Inlinee);
296 IO.mapOptional("ExtraFiles", Obj.ExtraFiles);
297}
298
Zachary Turnerdeb39132017-06-09 00:28:08 +0000299void MappingTraits<YAMLFrameData>::mapping(IO &IO, YAMLFrameData &Obj) {
300 IO.mapRequired("CodeSize", Obj.CodeSize);
301 IO.mapRequired("FrameFunc", Obj.FrameFunc);
302 IO.mapRequired("LocalSize", Obj.LocalSize);
303 IO.mapOptional("MaxStackSize", Obj.MaxStackSize);
304 IO.mapOptional("ParamsSize", Obj.ParamsSize);
305 IO.mapOptional("PrologSize", Obj.PrologSize);
306 IO.mapOptional("RvaStart", Obj.RvaStart);
307 IO.mapOptional("SavedRegsSize", Obj.SavedRegsSize);
308}
309
Zachary Turner92dcdda2017-06-02 19:49:14 +0000310void YAMLChecksumsSubsection::map(IO &IO) {
311 IO.mapTag("!FileChecksums", true);
312 IO.mapRequired("Checksums", Checksums);
313}
314
315void YAMLLinesSubsection::map(IO &IO) {
316 IO.mapTag("!Lines", true);
317 IO.mapRequired("CodeSize", Lines.CodeSize);
318
319 IO.mapRequired("Flags", Lines.Flags);
320 IO.mapRequired("RelocOffset", Lines.RelocOffset);
321 IO.mapRequired("RelocSegment", Lines.RelocSegment);
322 IO.mapRequired("Blocks", Lines.Blocks);
323}
324
325void YAMLInlineeLinesSubsection::map(IO &IO) {
326 IO.mapTag("!InlineeLines", true);
327 IO.mapRequired("HasExtraFiles", InlineeLines.HasExtraFiles);
328 IO.mapRequired("Sites", InlineeLines.Sites);
329}
330
Zachary Turner349c18f2017-06-05 21:40:33 +0000331void YAMLCrossModuleExportsSubsection::map(IO &IO) {
332 IO.mapTag("!CrossModuleExports", true);
333 IO.mapOptional("Exports", Exports);
334}
335
336void YAMLCrossModuleImportsSubsection::map(IO &IO) {
337 IO.mapTag("!CrossModuleImports", true);
338 IO.mapOptional("Imports", Imports);
339}
340
Zachary Turnerdeb39132017-06-09 00:28:08 +0000341void YAMLSymbolsSubsection::map(IO &IO) {
342 IO.mapTag("!Symbols", true);
343 IO.mapRequired("Records", Symbols);
344}
345
346void YAMLStringTableSubsection::map(IO &IO) {
347 IO.mapTag("!StringTable", true);
348 IO.mapRequired("Strings", Strings);
349}
350
351void YAMLFrameDataSubsection::map(IO &IO) {
352 IO.mapTag("!FrameData", true);
353 IO.mapRequired("Frames", Frames);
354}
355
Zachary Turner3226fe92017-06-09 20:46:52 +0000356void YAMLCoffSymbolRVASubsection::map(IO &IO) {
357 IO.mapTag("!COFFSymbolRVAs", true);
358 IO.mapRequired("RVAs", RVAs);
359}
360
Zachary Turner92dcdda2017-06-02 19:49:14 +0000361void MappingTraits<YAMLDebugSubsection>::mapping(
362 IO &IO, YAMLDebugSubsection &Subsection) {
363 if (!IO.outputting()) {
364 if (IO.mapTag("!FileChecksums")) {
365 auto SS = std::make_shared<YAMLChecksumsSubsection>();
366 Subsection.Subsection = SS;
367 } else if (IO.mapTag("!Lines")) {
368 Subsection.Subsection = std::make_shared<YAMLLinesSubsection>();
369 } else if (IO.mapTag("!InlineeLines")) {
370 Subsection.Subsection = std::make_shared<YAMLInlineeLinesSubsection>();
Zachary Turner349c18f2017-06-05 21:40:33 +0000371 } else if (IO.mapTag("!CrossModuleExports")) {
372 Subsection.Subsection =
373 std::make_shared<YAMLCrossModuleExportsSubsection>();
374 } else if (IO.mapTag("!CrossModuleImports")) {
375 Subsection.Subsection =
376 std::make_shared<YAMLCrossModuleImportsSubsection>();
Zachary Turnerdeb39132017-06-09 00:28:08 +0000377 } else if (IO.mapTag("!Symbols")) {
378 Subsection.Subsection = std::make_shared<YAMLSymbolsSubsection>();
379 } else if (IO.mapTag("!StringTable")) {
380 Subsection.Subsection = std::make_shared<YAMLStringTableSubsection>();
381 } else if (IO.mapTag("!FrameData")) {
382 Subsection.Subsection = std::make_shared<YAMLFrameDataSubsection>();
Zachary Turner3226fe92017-06-09 20:46:52 +0000383 } else if (IO.mapTag("!COFFSymbolRVAs")) {
384 Subsection.Subsection = std::make_shared<YAMLCoffSymbolRVASubsection>();
Zachary Turner92dcdda2017-06-02 19:49:14 +0000385 } else {
386 llvm_unreachable("Unexpected subsection tag!");
387 }
388 }
389 Subsection.Subsection->map(IO);
390}
391
Zachary Turner349c18f2017-06-05 21:40:33 +0000392static std::shared_ptr<YAMLChecksumsSubsection>
Zachary Turner92dcdda2017-06-02 19:49:14 +0000393findChecksums(ArrayRef<YAMLDebugSubsection> Subsections) {
394 for (const auto &SS : Subsections) {
395 if (SS.Subsection->Kind == DebugSubsectionKind::FileChecksums) {
Zachary Turner349c18f2017-06-05 21:40:33 +0000396 return std::static_pointer_cast<YAMLChecksumsSubsection>(SS.Subsection);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000397 }
398 }
Zachary Turner349c18f2017-06-05 21:40:33 +0000399
400 return nullptr;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000401}
402
403std::unique_ptr<DebugSubsection> YAMLChecksumsSubsection::toCodeViewSubsection(
Zachary Turnerdeb39132017-06-09 00:28:08 +0000404 BumpPtrAllocator &Allocator, DebugStringTableSubsection *UseStrings,
Zachary Turner92dcdda2017-06-02 19:49:14 +0000405 DebugChecksumsSubsection *UseChecksums) const {
406 assert(UseStrings && !UseChecksums);
407 auto Result = llvm::make_unique<DebugChecksumsSubsection>(*UseStrings);
408 for (const auto &CS : Checksums) {
409 Result->addChecksum(CS.FileName, CS.Kind, CS.ChecksumBytes.Bytes);
410 }
411 return std::move(Result);
412}
413
414std::unique_ptr<DebugSubsection> YAMLLinesSubsection::toCodeViewSubsection(
Zachary Turnerdeb39132017-06-09 00:28:08 +0000415 BumpPtrAllocator &Allocator, DebugStringTableSubsection *UseStrings,
Zachary Turner92dcdda2017-06-02 19:49:14 +0000416 DebugChecksumsSubsection *UseChecksums) const {
417 assert(UseStrings && UseChecksums);
418 auto Result =
419 llvm::make_unique<DebugLinesSubsection>(*UseChecksums, *UseStrings);
420 Result->setCodeSize(Lines.CodeSize);
421 Result->setRelocationAddress(Lines.RelocSegment, Lines.RelocOffset);
422 Result->setFlags(Lines.Flags);
423 for (const auto &LC : Lines.Blocks) {
424 Result->createBlock(LC.FileName);
425 if (Result->hasColumnInfo()) {
426 for (const auto &Item : zip(LC.Lines, LC.Columns)) {
427 auto &L = std::get<0>(Item);
428 auto &C = std::get<1>(Item);
429 uint32_t LE = L.LineStart + L.EndDelta;
430 Result->addLineAndColumnInfo(L.Offset,
431 LineInfo(L.LineStart, LE, L.IsStatement),
432 C.StartColumn, C.EndColumn);
433 }
434 } else {
435 for (const auto &L : LC.Lines) {
436 uint32_t LE = L.LineStart + L.EndDelta;
437 Result->addLineInfo(L.Offset, LineInfo(L.LineStart, LE, L.IsStatement));
438 }
439 }
440 }
Zachary Turner4bedb5f2017-06-02 20:00:10 +0000441 return llvm::cast<DebugSubsection>(std::move(Result));
Zachary Turner92dcdda2017-06-02 19:49:14 +0000442}
443
444std::unique_ptr<DebugSubsection>
445YAMLInlineeLinesSubsection::toCodeViewSubsection(
Zachary Turnerdeb39132017-06-09 00:28:08 +0000446 BumpPtrAllocator &Allocator, DebugStringTableSubsection *UseStrings,
Zachary Turner92dcdda2017-06-02 19:49:14 +0000447 DebugChecksumsSubsection *UseChecksums) const {
448 assert(UseChecksums);
449 auto Result = llvm::make_unique<DebugInlineeLinesSubsection>(
450 *UseChecksums, InlineeLines.HasExtraFiles);
451
452 for (const auto &Site : InlineeLines.Sites) {
453 Result->addInlineSite(TypeIndex(Site.Inlinee), Site.FileName,
454 Site.SourceLineNum);
455 if (!InlineeLines.HasExtraFiles)
456 continue;
457
458 for (auto EF : Site.ExtraFiles) {
459 Result->addExtraFile(EF);
460 }
461 }
Zachary Turner4bedb5f2017-06-02 20:00:10 +0000462 return llvm::cast<DebugSubsection>(std::move(Result));
Zachary Turner92dcdda2017-06-02 19:49:14 +0000463}
464
Zachary Turner349c18f2017-06-05 21:40:33 +0000465std::unique_ptr<DebugSubsection>
466YAMLCrossModuleExportsSubsection::toCodeViewSubsection(
Zachary Turnerdeb39132017-06-09 00:28:08 +0000467 BumpPtrAllocator &Allocator, DebugStringTableSubsection *Strings,
Zachary Turner349c18f2017-06-05 21:40:33 +0000468 DebugChecksumsSubsection *Checksums) const {
469 auto Result = llvm::make_unique<DebugCrossModuleExportsSubsection>();
470 for (const auto &M : Exports)
471 Result->addMapping(M.Local, M.Global);
472 return llvm::cast<DebugSubsection>(std::move(Result));
473}
474
475std::unique_ptr<DebugSubsection>
476YAMLCrossModuleImportsSubsection::toCodeViewSubsection(
Zachary Turnerdeb39132017-06-09 00:28:08 +0000477 BumpPtrAllocator &Allocator, DebugStringTableSubsection *Strings,
Zachary Turner349c18f2017-06-05 21:40:33 +0000478 DebugChecksumsSubsection *Checksums) const {
479 auto Result = llvm::make_unique<DebugCrossModuleImportsSubsection>(*Strings);
480 for (const auto &M : Imports) {
481 for (const auto Id : M.ImportIds)
482 Result->addImport(M.ModuleName, Id);
483 }
484 return llvm::cast<DebugSubsection>(std::move(Result));
485}
486
Zachary Turnerdeb39132017-06-09 00:28:08 +0000487std::unique_ptr<DebugSubsection> YAMLSymbolsSubsection::toCodeViewSubsection(
488 BumpPtrAllocator &Allocator, DebugStringTableSubsection *Strings,
489 DebugChecksumsSubsection *Checksums) const {
490 auto Result = llvm::make_unique<DebugSymbolsSubsection>();
491 for (const auto &Sym : Symbols)
492 Result->addSymbol(
493 Sym.toCodeViewSymbol(Allocator, CodeViewContainer::ObjectFile));
494 return std::move(Result);
495}
496
497std::unique_ptr<DebugSubsection>
498YAMLStringTableSubsection::toCodeViewSubsection(
499 BumpPtrAllocator &Allocator, DebugStringTableSubsection *Strings,
500 DebugChecksumsSubsection *Checksums) const {
501 auto Result = llvm::make_unique<DebugStringTableSubsection>();
502 for (const auto &Str : this->Strings)
503 Result->insert(Str);
504 return std::move(Result);
505}
506
507std::unique_ptr<DebugSubsection> YAMLFrameDataSubsection::toCodeViewSubsection(
508 BumpPtrAllocator &Allocator, DebugStringTableSubsection *Strings,
509 DebugChecksumsSubsection *Checksums) const {
510 assert(Strings);
511 auto Result = llvm::make_unique<DebugFrameDataSubsection>();
512 for (const auto &YF : Frames) {
513 codeview::FrameData F;
514 F.CodeSize = YF.CodeSize;
515 F.Flags = YF.Flags;
516 F.LocalSize = YF.LocalSize;
517 F.MaxStackSize = YF.MaxStackSize;
518 F.ParamsSize = YF.ParamsSize;
519 F.PrologSize = YF.PrologSize;
520 F.RvaStart = YF.RvaStart;
521 F.SavedRegsSize = YF.SavedRegsSize;
522 F.FrameFunc = Strings->insert(YF.FrameFunc);
523 Result->addFrameData(F);
524 }
525 return std::move(Result);
526}
527
Zachary Turner3226fe92017-06-09 20:46:52 +0000528std::unique_ptr<DebugSubsection>
529YAMLCoffSymbolRVASubsection::toCodeViewSubsection(
530 BumpPtrAllocator &Allocator, DebugStringTableSubsection *Strings,
531 DebugChecksumsSubsection *Checksums) const {
532 auto Result = llvm::make_unique<DebugSymbolRVASubsection>();
533 for (const auto &RVA : RVAs)
534 Result->addRVA(RVA);
535 return std::move(Result);
536}
537
Zachary Turner92dcdda2017-06-02 19:49:14 +0000538static Expected<SourceFileChecksumEntry>
539convertOneChecksum(const DebugStringTableSubsectionRef &Strings,
540 const FileChecksumEntry &CS) {
541 auto ExpectedString = Strings.getString(CS.FileNameOffset);
542 if (!ExpectedString)
543 return ExpectedString.takeError();
544
545 SourceFileChecksumEntry Result;
546 Result.ChecksumBytes.Bytes = CS.Checksum;
547 Result.Kind = CS.Kind;
548 Result.FileName = *ExpectedString;
549 return Result;
550}
551
552static Expected<StringRef>
553getFileName(const DebugStringTableSubsectionRef &Strings,
554 const DebugChecksumsSubsectionRef &Checksums, uint32_t FileID) {
555 auto Iter = Checksums.getArray().at(FileID);
556 if (Iter == Checksums.getArray().end())
557 return make_error<CodeViewError>(cv_error_code::no_records);
558 uint32_t Offset = Iter->FileNameOffset;
559 return Strings.getString(Offset);
560}
561
562Expected<std::shared_ptr<YAMLChecksumsSubsection>>
563YAMLChecksumsSubsection::fromCodeViewSubsection(
564 const DebugStringTableSubsectionRef &Strings,
565 const DebugChecksumsSubsectionRef &FC) {
566 auto Result = std::make_shared<YAMLChecksumsSubsection>();
567
568 for (const auto &CS : FC) {
569 auto ConvertedCS = convertOneChecksum(Strings, CS);
570 if (!ConvertedCS)
571 return ConvertedCS.takeError();
572 Result->Checksums.push_back(*ConvertedCS);
573 }
574 return Result;
575}
576
577Expected<std::shared_ptr<YAMLLinesSubsection>>
578YAMLLinesSubsection::fromCodeViewSubsection(
579 const DebugStringTableSubsectionRef &Strings,
580 const DebugChecksumsSubsectionRef &Checksums,
581 const DebugLinesSubsectionRef &Lines) {
582 auto Result = std::make_shared<YAMLLinesSubsection>();
583 Result->Lines.CodeSize = Lines.header()->CodeSize;
584 Result->Lines.RelocOffset = Lines.header()->RelocOffset;
585 Result->Lines.RelocSegment = Lines.header()->RelocSegment;
586 Result->Lines.Flags = static_cast<LineFlags>(uint16_t(Lines.header()->Flags));
587 for (const auto &L : Lines) {
588 SourceLineBlock Block;
589 auto EF = getFileName(Strings, Checksums, L.NameIndex);
590 if (!EF)
591 return EF.takeError();
592 Block.FileName = *EF;
593 if (Lines.hasColumnInfo()) {
594 for (const auto &C : L.Columns) {
595 SourceColumnEntry SCE;
596 SCE.EndColumn = C.EndColumn;
597 SCE.StartColumn = C.StartColumn;
598 Block.Columns.push_back(SCE);
599 }
600 }
601 for (const auto &LN : L.LineNumbers) {
602 SourceLineEntry SLE;
603 LineInfo LI(LN.Flags);
604 SLE.Offset = LN.Offset;
605 SLE.LineStart = LI.getStartLine();
606 SLE.EndDelta = LI.getLineDelta();
607 SLE.IsStatement = LI.isStatement();
608 Block.Lines.push_back(SLE);
609 }
610 Result->Lines.Blocks.push_back(Block);
611 }
612 return Result;
613}
614
615Expected<std::shared_ptr<YAMLInlineeLinesSubsection>>
616YAMLInlineeLinesSubsection::fromCodeViewSubsection(
617 const DebugStringTableSubsectionRef &Strings,
618 const DebugChecksumsSubsectionRef &Checksums,
619 const DebugInlineeLinesSubsectionRef &Lines) {
620 auto Result = std::make_shared<YAMLInlineeLinesSubsection>();
621
622 Result->InlineeLines.HasExtraFiles = Lines.hasExtraFiles();
623 for (const auto &IL : Lines) {
624 InlineeSite Site;
625 auto ExpF = getFileName(Strings, Checksums, IL.Header->FileID);
626 if (!ExpF)
627 return ExpF.takeError();
628 Site.FileName = *ExpF;
629 Site.Inlinee = IL.Header->Inlinee.getIndex();
630 Site.SourceLineNum = IL.Header->SourceLineNum;
631 if (Lines.hasExtraFiles()) {
632 for (const auto EF : IL.ExtraFiles) {
633 auto ExpF2 = getFileName(Strings, Checksums, EF);
634 if (!ExpF2)
635 return ExpF2.takeError();
636 Site.ExtraFiles.push_back(*ExpF2);
637 }
638 }
639 Result->InlineeLines.Sites.push_back(Site);
640 }
641 return Result;
642}
643
Zachary Turner349c18f2017-06-05 21:40:33 +0000644Expected<std::shared_ptr<YAMLCrossModuleExportsSubsection>>
645YAMLCrossModuleExportsSubsection::fromCodeViewSubsection(
646 const DebugCrossModuleExportsSubsectionRef &Exports) {
647 auto Result = std::make_shared<YAMLCrossModuleExportsSubsection>();
648 Result->Exports.assign(Exports.begin(), Exports.end());
649 return Result;
650}
651
652Expected<std::shared_ptr<YAMLCrossModuleImportsSubsection>>
653YAMLCrossModuleImportsSubsection::fromCodeViewSubsection(
654 const DebugStringTableSubsectionRef &Strings,
655 const DebugCrossModuleImportsSubsectionRef &Imports) {
656 auto Result = std::make_shared<YAMLCrossModuleImportsSubsection>();
657 for (const auto &CMI : Imports) {
658 YAMLCrossModuleImport YCMI;
659 auto ExpectedStr = Strings.getString(CMI.Header->ModuleNameOffset);
660 if (!ExpectedStr)
661 return ExpectedStr.takeError();
662 YCMI.ModuleName = *ExpectedStr;
663 YCMI.ImportIds.assign(CMI.Imports.begin(), CMI.Imports.end());
664 Result->Imports.push_back(YCMI);
665 }
666 return Result;
667}
668
Zachary Turnerdeb39132017-06-09 00:28:08 +0000669Expected<std::shared_ptr<YAMLSymbolsSubsection>>
670YAMLSymbolsSubsection::fromCodeViewSubsection(
671 const DebugSymbolsSubsectionRef &Symbols) {
672 auto Result = std::make_shared<YAMLSymbolsSubsection>();
673 for (const auto &Sym : Symbols) {
674 auto S = CodeViewYAML::SymbolRecord::fromCodeViewSymbol(Sym);
675 if (!S)
676 return joinErrors(make_error<CodeViewError>(
677 cv_error_code::corrupt_record,
678 "Invalid CodeView Symbol Record in SymbolRecord "
679 "subsection of .debug$S while converting to YAML!"),
680 S.takeError());
681
682 Result->Symbols.push_back(*S);
683 }
684 return Result;
685}
686
687Expected<std::shared_ptr<YAMLStringTableSubsection>>
688YAMLStringTableSubsection::fromCodeViewSubsection(
689 const DebugStringTableSubsectionRef &Strings) {
690 auto Result = std::make_shared<YAMLStringTableSubsection>();
691 BinaryStreamReader Reader(Strings.getBuffer());
692 StringRef S;
693 // First item is a single null string, skip it.
694 if (auto EC = Reader.readCString(S))
695 return std::move(EC);
696 assert(S.empty());
697 while (Reader.bytesRemaining() > 0) {
698 if (auto EC = Reader.readCString(S))
699 return std::move(EC);
700 Result->Strings.push_back(S);
701 }
702 return Result;
703}
704
705Expected<std::shared_ptr<YAMLFrameDataSubsection>>
706YAMLFrameDataSubsection::fromCodeViewSubsection(
707 const DebugStringTableSubsectionRef &Strings,
708 const DebugFrameDataSubsectionRef &Frames) {
709 auto Result = std::make_shared<YAMLFrameDataSubsection>();
710 for (const auto &F : Frames) {
711 YAMLFrameData YF;
712 YF.CodeSize = F.CodeSize;
713 YF.Flags = F.Flags;
714 YF.LocalSize = F.LocalSize;
715 YF.MaxStackSize = F.MaxStackSize;
716 YF.ParamsSize = F.ParamsSize;
717 YF.PrologSize = F.PrologSize;
718 YF.RvaStart = F.RvaStart;
719 YF.SavedRegsSize = F.SavedRegsSize;
720
721 auto ES = Strings.getString(F.FrameFunc);
722 if (!ES)
723 return joinErrors(
724 make_error<CodeViewError>(
725 cv_error_code::no_records,
726 "Could not find string for string id while mapping FrameData!"),
727 ES.takeError());
728 YF.FrameFunc = *ES;
729 Result->Frames.push_back(YF);
730 }
731 return Result;
732}
733
Zachary Turner3226fe92017-06-09 20:46:52 +0000734Expected<std::shared_ptr<YAMLCoffSymbolRVASubsection>>
735YAMLCoffSymbolRVASubsection::fromCodeViewSubsection(
736 const DebugSymbolRVASubsectionRef &Section) {
737 auto Result = std::make_shared<YAMLCoffSymbolRVASubsection>();
738 for (const auto &RVA : Section) {
739 Result->RVAs.push_back(RVA);
740 }
741 return Result;
742}
743
Zachary Turner92dcdda2017-06-02 19:49:14 +0000744Expected<std::vector<std::unique_ptr<DebugSubsection>>>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000745llvm::CodeViewYAML::toCodeViewSubsectionList(
746 BumpPtrAllocator &Allocator, ArrayRef<YAMLDebugSubsection> Subsections,
Zachary Turner92dcdda2017-06-02 19:49:14 +0000747 DebugStringTableSubsection &Strings) {
748 std::vector<std::unique_ptr<DebugSubsection>> Result;
749 if (Subsections.empty())
Zachary Turner64726f22017-06-02 21:00:22 +0000750 return std::move(Result);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000751
752 auto Checksums = findChecksums(Subsections);
Zachary Turner349c18f2017-06-05 21:40:33 +0000753 std::unique_ptr<DebugSubsection> ChecksumsBase;
754 if (Checksums)
Zachary Turnerdeb39132017-06-09 00:28:08 +0000755 ChecksumsBase =
756 Checksums->toCodeViewSubsection(Allocator, &Strings, nullptr);
Zachary Turner349c18f2017-06-05 21:40:33 +0000757 DebugChecksumsSubsection *CS =
758 static_cast<DebugChecksumsSubsection *>(ChecksumsBase.get());
Zachary Turner92dcdda2017-06-02 19:49:14 +0000759 for (const auto &SS : Subsections) {
760 // We've already converted the checksums subsection, don't do it
761 // twice.
762 std::unique_ptr<DebugSubsection> CVS;
763 if (SS.Subsection->Kind == DebugSubsectionKind::FileChecksums)
764 CVS = std::move(ChecksumsBase);
765 else
Zachary Turnerdeb39132017-06-09 00:28:08 +0000766 CVS = SS.Subsection->toCodeViewSubsection(Allocator, &Strings, CS);
767 assert(CVS != nullptr);
768 Result.push_back(std::move(CVS));
769 }
770 return std::move(Result);
771}
772
773Expected<std::vector<std::unique_ptr<codeview::DebugSubsection>>>
774llvm::CodeViewYAML::toCodeViewSubsectionList(
775 BumpPtrAllocator &Allocator, ArrayRef<YAMLDebugSubsection> Subsections,
776 std::unique_ptr<DebugStringTableSubsection> &TakeStrings,
777 DebugStringTableSubsection *StringsRef) {
778 std::vector<std::unique_ptr<DebugSubsection>> Result;
779 if (Subsections.empty())
780 return std::move(Result);
781
782 auto Checksums = findChecksums(Subsections);
783
784 std::unique_ptr<DebugSubsection> ChecksumsBase;
785 if (Checksums)
786 ChecksumsBase =
787 Checksums->toCodeViewSubsection(Allocator, StringsRef, nullptr);
788 DebugChecksumsSubsection *CS =
789 static_cast<DebugChecksumsSubsection *>(ChecksumsBase.get());
790 for (const auto &SS : Subsections) {
791 // We've already converted the checksums and string table subsection, don't
792 // do it twice.
793 std::unique_ptr<DebugSubsection> CVS;
794 if (SS.Subsection->Kind == DebugSubsectionKind::FileChecksums)
795 CVS = std::move(ChecksumsBase);
796 else if (SS.Subsection->Kind == DebugSubsectionKind::StringTable) {
797 assert(TakeStrings && "No string table!");
798 CVS = std::move(TakeStrings);
799 } else
800 CVS = SS.Subsection->toCodeViewSubsection(Allocator, StringsRef, CS);
Zachary Turner349c18f2017-06-05 21:40:33 +0000801 assert(CVS != nullptr);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000802 Result.push_back(std::move(CVS));
803 }
804 return std::move(Result);
805}
806
807namespace {
808struct SubsectionConversionVisitor : public DebugSubsectionVisitor {
Zachary Turner1bf77622017-06-08 23:49:01 +0000809 SubsectionConversionVisitor() {}
Zachary Turner92dcdda2017-06-02 19:49:14 +0000810
811 Error visitUnknown(DebugUnknownSubsectionRef &Unknown) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000812 Error visitLines(DebugLinesSubsectionRef &Lines,
813 const DebugSubsectionState &State) override;
814 Error visitFileChecksums(DebugChecksumsSubsectionRef &Checksums,
815 const DebugSubsectionState &State) override;
816 Error visitInlineeLines(DebugInlineeLinesSubsectionRef &Inlinees,
817 const DebugSubsectionState &State) override;
818 Error visitCrossModuleExports(DebugCrossModuleExportsSubsectionRef &Checksums,
819 const DebugSubsectionState &State) override;
820 Error visitCrossModuleImports(DebugCrossModuleImportsSubsectionRef &Inlinees,
821 const DebugSubsectionState &State) override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000822 Error visitStringTable(DebugStringTableSubsectionRef &ST,
823 const DebugSubsectionState &State) override;
824 Error visitSymbols(DebugSymbolsSubsectionRef &Symbols,
825 const DebugSubsectionState &State) override;
826 Error visitFrameData(DebugFrameDataSubsectionRef &Symbols,
827 const DebugSubsectionState &State) override;
Zachary Turner3226fe92017-06-09 20:46:52 +0000828 Error visitCOFFSymbolRVAs(DebugSymbolRVASubsectionRef &Symbols,
829 const DebugSubsectionState &State) override;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000830
831 YAMLDebugSubsection Subsection;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000832};
833
834Error SubsectionConversionVisitor::visitUnknown(
835 DebugUnknownSubsectionRef &Unknown) {
836 return make_error<CodeViewError>(cv_error_code::operation_unsupported);
837}
838
Zachary Turner1bf77622017-06-08 23:49:01 +0000839Error SubsectionConversionVisitor::visitLines(
840 DebugLinesSubsectionRef &Lines, const DebugSubsectionState &State) {
841 auto Result = YAMLLinesSubsection::fromCodeViewSubsection(
842 State.strings(), State.checksums(), Lines);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000843 if (!Result)
844 return Result.takeError();
845 Subsection.Subsection = *Result;
846 return Error::success();
847}
848
849Error SubsectionConversionVisitor::visitFileChecksums(
Zachary Turner1bf77622017-06-08 23:49:01 +0000850 DebugChecksumsSubsectionRef &Checksums, const DebugSubsectionState &State) {
851 auto Result = YAMLChecksumsSubsection::fromCodeViewSubsection(State.strings(),
852 Checksums);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000853 if (!Result)
854 return Result.takeError();
855 Subsection.Subsection = *Result;
856 return Error::success();
857}
858
859Error SubsectionConversionVisitor::visitInlineeLines(
Zachary Turner1bf77622017-06-08 23:49:01 +0000860 DebugInlineeLinesSubsectionRef &Inlinees,
861 const DebugSubsectionState &State) {
Zachary Turner92dcdda2017-06-02 19:49:14 +0000862 auto Result = YAMLInlineeLinesSubsection::fromCodeViewSubsection(
Zachary Turner1bf77622017-06-08 23:49:01 +0000863 State.strings(), State.checksums(), Inlinees);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000864 if (!Result)
865 return Result.takeError();
866 Subsection.Subsection = *Result;
867 return Error::success();
868}
Zachary Turner349c18f2017-06-05 21:40:33 +0000869
870Error SubsectionConversionVisitor::visitCrossModuleExports(
Zachary Turner1bf77622017-06-08 23:49:01 +0000871 DebugCrossModuleExportsSubsectionRef &Exports,
872 const DebugSubsectionState &State) {
Zachary Turner349c18f2017-06-05 21:40:33 +0000873 auto Result =
874 YAMLCrossModuleExportsSubsection::fromCodeViewSubsection(Exports);
875 if (!Result)
876 return Result.takeError();
877 Subsection.Subsection = *Result;
878 return Error::success();
879}
880
881Error SubsectionConversionVisitor::visitCrossModuleImports(
Zachary Turner1bf77622017-06-08 23:49:01 +0000882 DebugCrossModuleImportsSubsectionRef &Imports,
883 const DebugSubsectionState &State) {
Zachary Turner349c18f2017-06-05 21:40:33 +0000884 auto Result = YAMLCrossModuleImportsSubsection::fromCodeViewSubsection(
Zachary Turner1bf77622017-06-08 23:49:01 +0000885 State.strings(), Imports);
Zachary Turner349c18f2017-06-05 21:40:33 +0000886 if (!Result)
887 return Result.takeError();
888 Subsection.Subsection = *Result;
889 return Error::success();
890}
Zachary Turnerdeb39132017-06-09 00:28:08 +0000891
892Error SubsectionConversionVisitor::visitStringTable(
893 DebugStringTableSubsectionRef &Strings, const DebugSubsectionState &State) {
894 auto Result = YAMLStringTableSubsection::fromCodeViewSubsection(Strings);
895 if (!Result)
896 return Result.takeError();
897 Subsection.Subsection = *Result;
898 return Error::success();
899}
900
901Error SubsectionConversionVisitor::visitSymbols(
902 DebugSymbolsSubsectionRef &Symbols, const DebugSubsectionState &State) {
903 auto Result = YAMLSymbolsSubsection::fromCodeViewSubsection(Symbols);
904 if (!Result)
905 return Result.takeError();
906 Subsection.Subsection = *Result;
907 return Error::success();
908}
909
910Error SubsectionConversionVisitor::visitFrameData(
911 DebugFrameDataSubsectionRef &Frames, const DebugSubsectionState &State) {
912 auto Result =
913 YAMLFrameDataSubsection::fromCodeViewSubsection(State.strings(), Frames);
914 if (!Result)
915 return Result.takeError();
916 Subsection.Subsection = *Result;
917 return Error::success();
918}
Zachary Turner3226fe92017-06-09 20:46:52 +0000919
920Error SubsectionConversionVisitor::visitCOFFSymbolRVAs(
921 DebugSymbolRVASubsectionRef &RVAs, const DebugSubsectionState &State) {
922 auto Result = YAMLCoffSymbolRVASubsection::fromCodeViewSubsection(RVAs);
923 if (!Result)
924 return Result.takeError();
925 Subsection.Subsection = *Result;
926 return Error::success();
927}
Zachary Turner92dcdda2017-06-02 19:49:14 +0000928}
929
930Expected<YAMLDebugSubsection> YAMLDebugSubsection::fromCodeViewSubection(
931 const DebugStringTableSubsectionRef &Strings,
932 const DebugChecksumsSubsectionRef &Checksums,
933 const DebugSubsectionRecord &SS) {
Zachary Turner1bf77622017-06-08 23:49:01 +0000934 DebugSubsectionState State(Strings, Checksums);
935 SubsectionConversionVisitor V;
936 if (auto EC = visitDebugSubsection(SS, V, State))
Zachary Turner92dcdda2017-06-02 19:49:14 +0000937 return std::move(EC);
938
939 return V.Subsection;
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000940}
Zachary Turnerdeb39132017-06-09 00:28:08 +0000941
942std::unique_ptr<DebugStringTableSubsection>
943llvm::CodeViewYAML::findStringTable(ArrayRef<YAMLDebugSubsection> Sections) {
944 for (const auto &SS : Sections) {
945 if (SS.Subsection->Kind != DebugSubsectionKind::StringTable)
946 continue;
947
948 // String Table doesn't use the allocator.
949 BumpPtrAllocator Allocator;
950 auto Result =
951 SS.Subsection->toCodeViewSubsection(Allocator, nullptr, nullptr);
952 return llvm::cast<DebugStringTableSubsection>(std::move(Result));
953 }
954 return nullptr;
955}