blob: bda43d6994315404065c35d3c1f2569c2dbaeb55 [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"
Zachary Turnera8cfc292017-06-14 15:59:27 +000031#include "llvm/DebugInfo/CodeView/StringsAndChecksums.h"
Zachary Turner1b88f4f2017-05-31 04:17:13 +000032#include "llvm/DebugInfo/CodeView/SymbolRecord.h"
Zachary Turnerdeb39132017-06-09 00:28:08 +000033#include "llvm/DebugInfo/CodeView/SymbolSerializer.h"
34#include "llvm/ObjectYAML/CodeViewYAMLSymbols.h"
35#include "llvm/Support/BinaryStreamWriter.h"
Zachary Turner1b88f4f2017-05-31 04:17:13 +000036using namespace llvm;
37using namespace llvm::codeview;
38using namespace llvm::CodeViewYAML;
39using namespace llvm::CodeViewYAML::detail;
40using namespace llvm::yaml;
41
42LLVM_YAML_IS_SEQUENCE_VECTOR(SourceFileChecksumEntry)
43LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineEntry)
44LLVM_YAML_IS_SEQUENCE_VECTOR(SourceColumnEntry)
45LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineBlock)
46LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineInfo)
47LLVM_YAML_IS_SEQUENCE_VECTOR(InlineeSite)
48LLVM_YAML_IS_SEQUENCE_VECTOR(InlineeInfo)
Zachary Turner349c18f2017-06-05 21:40:33 +000049LLVM_YAML_IS_SEQUENCE_VECTOR(CrossModuleExport)
50LLVM_YAML_IS_SEQUENCE_VECTOR(YAMLCrossModuleImport)
Zachary Turnerdeb39132017-06-09 00:28:08 +000051LLVM_YAML_IS_SEQUENCE_VECTOR(YAMLFrameData)
Zachary Turner1b88f4f2017-05-31 04:17:13 +000052
53LLVM_YAML_DECLARE_SCALAR_TRAITS(HexFormattedString, false)
Zachary Turner92dcdda2017-06-02 19:49:14 +000054LLVM_YAML_DECLARE_ENUM_TRAITS(DebugSubsectionKind)
Zachary Turner1b88f4f2017-05-31 04:17:13 +000055LLVM_YAML_DECLARE_ENUM_TRAITS(FileChecksumKind)
56LLVM_YAML_DECLARE_BITSET_TRAITS(LineFlags)
57
Zachary Turner349c18f2017-06-05 21:40:33 +000058LLVM_YAML_DECLARE_MAPPING_TRAITS(CrossModuleExport)
Zachary Turnerdeb39132017-06-09 00:28:08 +000059LLVM_YAML_DECLARE_MAPPING_TRAITS(YAMLFrameData)
Zachary Turner349c18f2017-06-05 21:40:33 +000060LLVM_YAML_DECLARE_MAPPING_TRAITS(YAMLCrossModuleImport)
61LLVM_YAML_DECLARE_MAPPING_TRAITS(CrossModuleImportItem)
Zachary Turner92dcdda2017-06-02 19:49:14 +000062LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceLineEntry)
63LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceColumnEntry)
64LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceFileChecksumEntry)
65LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceLineBlock)
66LLVM_YAML_DECLARE_MAPPING_TRAITS(InlineeSite)
67
68namespace llvm {
69namespace CodeViewYAML {
70namespace detail {
71struct YAMLSubsectionBase {
72 explicit YAMLSubsectionBase(DebugSubsectionKind Kind) : Kind(Kind) {}
73 DebugSubsectionKind Kind;
74 virtual ~YAMLSubsectionBase() {}
75
76 virtual void map(IO &IO) = 0;
Zachary Turnera8cfc292017-06-14 15:59:27 +000077 virtual std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +000078 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +000079 const codeview::StringsAndChecksums &SC) const = 0;
Zachary Turner92dcdda2017-06-02 19:49:14 +000080};
81}
82}
83}
84
85namespace {
86struct YAMLChecksumsSubsection : public YAMLSubsectionBase {
87 YAMLChecksumsSubsection()
88 : YAMLSubsectionBase(DebugSubsectionKind::FileChecksums) {}
89
90 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +000091 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +000092 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +000093 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner92dcdda2017-06-02 19:49:14 +000094 static Expected<std::shared_ptr<YAMLChecksumsSubsection>>
95 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
96 const DebugChecksumsSubsectionRef &FC);
97
98 std::vector<SourceFileChecksumEntry> Checksums;
99};
100
101struct YAMLLinesSubsection : public YAMLSubsectionBase {
102 YAMLLinesSubsection() : YAMLSubsectionBase(DebugSubsectionKind::Lines) {}
103
104 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000105 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000106 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000107 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000108 static Expected<std::shared_ptr<YAMLLinesSubsection>>
109 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
110 const DebugChecksumsSubsectionRef &Checksums,
111 const DebugLinesSubsectionRef &Lines);
112
113 SourceLineInfo Lines;
114};
115
116struct YAMLInlineeLinesSubsection : public YAMLSubsectionBase {
117 YAMLInlineeLinesSubsection()
118 : YAMLSubsectionBase(DebugSubsectionKind::InlineeLines) {}
119
120 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000121 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000122 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000123 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000124 static Expected<std::shared_ptr<YAMLInlineeLinesSubsection>>
125 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
126 const DebugChecksumsSubsectionRef &Checksums,
127 const DebugInlineeLinesSubsectionRef &Lines);
128
129 InlineeInfo InlineeLines;
130};
Zachary Turner349c18f2017-06-05 21:40:33 +0000131
132struct YAMLCrossModuleExportsSubsection : public YAMLSubsectionBase {
133 YAMLCrossModuleExportsSubsection()
134 : YAMLSubsectionBase(DebugSubsectionKind::CrossScopeExports) {}
135
136 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000137 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000138 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000139 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner349c18f2017-06-05 21:40:33 +0000140 static Expected<std::shared_ptr<YAMLCrossModuleExportsSubsection>>
141 fromCodeViewSubsection(const DebugCrossModuleExportsSubsectionRef &Exports);
142
143 std::vector<CrossModuleExport> Exports;
144};
145
146struct YAMLCrossModuleImportsSubsection : public YAMLSubsectionBase {
147 YAMLCrossModuleImportsSubsection()
148 : YAMLSubsectionBase(DebugSubsectionKind::CrossScopeImports) {}
149
150 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000151 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000152 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000153 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner349c18f2017-06-05 21:40:33 +0000154 static Expected<std::shared_ptr<YAMLCrossModuleImportsSubsection>>
155 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
156 const DebugCrossModuleImportsSubsectionRef &Imports);
157
158 std::vector<YAMLCrossModuleImport> Imports;
159};
Zachary Turnerdeb39132017-06-09 00:28:08 +0000160
161struct YAMLSymbolsSubsection : public YAMLSubsectionBase {
162 YAMLSymbolsSubsection() : YAMLSubsectionBase(DebugSubsectionKind::Symbols) {}
163
164 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000165 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000166 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000167 const codeview::StringsAndChecksums &SC) const override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000168 static Expected<std::shared_ptr<YAMLSymbolsSubsection>>
169 fromCodeViewSubsection(const DebugSymbolsSubsectionRef &Symbols);
170
171 std::vector<CodeViewYAML::SymbolRecord> Symbols;
172};
173
174struct YAMLStringTableSubsection : public YAMLSubsectionBase {
175 YAMLStringTableSubsection()
176 : YAMLSubsectionBase(DebugSubsectionKind::StringTable) {}
177
178 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000179 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000180 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000181 const codeview::StringsAndChecksums &SC) const override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000182 static Expected<std::shared_ptr<YAMLStringTableSubsection>>
183 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings);
184
185 std::vector<StringRef> Strings;
186};
187
188struct YAMLFrameDataSubsection : public YAMLSubsectionBase {
189 YAMLFrameDataSubsection()
190 : YAMLSubsectionBase(DebugSubsectionKind::FrameData) {}
191
192 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000193 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000194 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000195 const codeview::StringsAndChecksums &SC) const override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000196 static Expected<std::shared_ptr<YAMLFrameDataSubsection>>
197 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
198 const DebugFrameDataSubsectionRef &Frames);
199
200 std::vector<YAMLFrameData> Frames;
201};
Zachary Turner3226fe92017-06-09 20:46:52 +0000202
203struct YAMLCoffSymbolRVASubsection : public YAMLSubsectionBase {
204 YAMLCoffSymbolRVASubsection()
205 : YAMLSubsectionBase(DebugSubsectionKind::CoffSymbolRVA) {}
206
207 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000208 std::shared_ptr<DebugSubsection>
Zachary Turner3226fe92017-06-09 20:46:52 +0000209 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000210 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner3226fe92017-06-09 20:46:52 +0000211 static Expected<std::shared_ptr<YAMLCoffSymbolRVASubsection>>
212 fromCodeViewSubsection(const DebugSymbolRVASubsectionRef &RVAs);
213
214 std::vector<uint32_t> RVAs;
215};
Zachary Turner92dcdda2017-06-02 19:49:14 +0000216}
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000217
218void ScalarBitSetTraits<LineFlags>::bitset(IO &io, LineFlags &Flags) {
219 io.bitSetCase(Flags, "HasColumnInfo", LF_HaveColumns);
220 io.enumFallback<Hex16>(Flags);
221}
222
223void ScalarEnumerationTraits<FileChecksumKind>::enumeration(
224 IO &io, FileChecksumKind &Kind) {
225 io.enumCase(Kind, "None", FileChecksumKind::None);
226 io.enumCase(Kind, "MD5", FileChecksumKind::MD5);
227 io.enumCase(Kind, "SHA1", FileChecksumKind::SHA1);
228 io.enumCase(Kind, "SHA256", FileChecksumKind::SHA256);
229}
230
231void ScalarTraits<HexFormattedString>::output(const HexFormattedString &Value,
232 void *ctx, raw_ostream &Out) {
233 StringRef Bytes(reinterpret_cast<const char *>(Value.Bytes.data()),
234 Value.Bytes.size());
235 Out << toHex(Bytes);
236}
237
238StringRef ScalarTraits<HexFormattedString>::input(StringRef Scalar, void *ctxt,
239 HexFormattedString &Value) {
240 std::string H = fromHex(Scalar);
241 Value.Bytes.assign(H.begin(), H.end());
242 return StringRef();
243}
244
245void MappingTraits<SourceLineEntry>::mapping(IO &IO, SourceLineEntry &Obj) {
246 IO.mapRequired("Offset", Obj.Offset);
247 IO.mapRequired("LineStart", Obj.LineStart);
248 IO.mapRequired("IsStatement", Obj.IsStatement);
249 IO.mapRequired("EndDelta", Obj.EndDelta);
250}
251
252void MappingTraits<SourceColumnEntry>::mapping(IO &IO, SourceColumnEntry &Obj) {
253 IO.mapRequired("StartColumn", Obj.StartColumn);
254 IO.mapRequired("EndColumn", Obj.EndColumn);
255}
256
257void MappingTraits<SourceLineBlock>::mapping(IO &IO, SourceLineBlock &Obj) {
258 IO.mapRequired("FileName", Obj.FileName);
259 IO.mapRequired("Lines", Obj.Lines);
260 IO.mapRequired("Columns", Obj.Columns);
261}
262
Zachary Turner349c18f2017-06-05 21:40:33 +0000263void MappingTraits<CrossModuleExport>::mapping(IO &IO, CrossModuleExport &Obj) {
264 IO.mapRequired("LocalId", Obj.Local);
265 IO.mapRequired("GlobalId", Obj.Global);
266}
267
268void MappingTraits<YAMLCrossModuleImport>::mapping(IO &IO,
269 YAMLCrossModuleImport &Obj) {
270 IO.mapRequired("Module", Obj.ModuleName);
271 IO.mapRequired("Imports", Obj.ImportIds);
272}
273
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000274void MappingTraits<SourceFileChecksumEntry>::mapping(
275 IO &IO, SourceFileChecksumEntry &Obj) {
276 IO.mapRequired("FileName", Obj.FileName);
277 IO.mapRequired("Kind", Obj.Kind);
278 IO.mapRequired("Checksum", Obj.ChecksumBytes);
279}
280
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000281void MappingTraits<InlineeSite>::mapping(IO &IO, InlineeSite &Obj) {
282 IO.mapRequired("FileName", Obj.FileName);
283 IO.mapRequired("LineNum", Obj.SourceLineNum);
284 IO.mapRequired("Inlinee", Obj.Inlinee);
285 IO.mapOptional("ExtraFiles", Obj.ExtraFiles);
286}
287
Zachary Turnerdeb39132017-06-09 00:28:08 +0000288void MappingTraits<YAMLFrameData>::mapping(IO &IO, YAMLFrameData &Obj) {
289 IO.mapRequired("CodeSize", Obj.CodeSize);
290 IO.mapRequired("FrameFunc", Obj.FrameFunc);
291 IO.mapRequired("LocalSize", Obj.LocalSize);
292 IO.mapOptional("MaxStackSize", Obj.MaxStackSize);
293 IO.mapOptional("ParamsSize", Obj.ParamsSize);
294 IO.mapOptional("PrologSize", Obj.PrologSize);
295 IO.mapOptional("RvaStart", Obj.RvaStart);
296 IO.mapOptional("SavedRegsSize", Obj.SavedRegsSize);
297}
298
Zachary Turner92dcdda2017-06-02 19:49:14 +0000299void YAMLChecksumsSubsection::map(IO &IO) {
300 IO.mapTag("!FileChecksums", true);
301 IO.mapRequired("Checksums", Checksums);
302}
303
304void YAMLLinesSubsection::map(IO &IO) {
305 IO.mapTag("!Lines", true);
306 IO.mapRequired("CodeSize", Lines.CodeSize);
307
308 IO.mapRequired("Flags", Lines.Flags);
309 IO.mapRequired("RelocOffset", Lines.RelocOffset);
310 IO.mapRequired("RelocSegment", Lines.RelocSegment);
311 IO.mapRequired("Blocks", Lines.Blocks);
312}
313
314void YAMLInlineeLinesSubsection::map(IO &IO) {
315 IO.mapTag("!InlineeLines", true);
316 IO.mapRequired("HasExtraFiles", InlineeLines.HasExtraFiles);
317 IO.mapRequired("Sites", InlineeLines.Sites);
318}
319
Zachary Turner349c18f2017-06-05 21:40:33 +0000320void YAMLCrossModuleExportsSubsection::map(IO &IO) {
321 IO.mapTag("!CrossModuleExports", true);
322 IO.mapOptional("Exports", Exports);
323}
324
325void YAMLCrossModuleImportsSubsection::map(IO &IO) {
326 IO.mapTag("!CrossModuleImports", true);
327 IO.mapOptional("Imports", Imports);
328}
329
Zachary Turnerdeb39132017-06-09 00:28:08 +0000330void YAMLSymbolsSubsection::map(IO &IO) {
331 IO.mapTag("!Symbols", true);
332 IO.mapRequired("Records", Symbols);
333}
334
335void YAMLStringTableSubsection::map(IO &IO) {
336 IO.mapTag("!StringTable", true);
337 IO.mapRequired("Strings", Strings);
338}
339
340void YAMLFrameDataSubsection::map(IO &IO) {
341 IO.mapTag("!FrameData", true);
342 IO.mapRequired("Frames", Frames);
343}
344
Zachary Turner3226fe92017-06-09 20:46:52 +0000345void YAMLCoffSymbolRVASubsection::map(IO &IO) {
346 IO.mapTag("!COFFSymbolRVAs", true);
347 IO.mapRequired("RVAs", RVAs);
348}
349
Zachary Turner92dcdda2017-06-02 19:49:14 +0000350void MappingTraits<YAMLDebugSubsection>::mapping(
351 IO &IO, YAMLDebugSubsection &Subsection) {
352 if (!IO.outputting()) {
353 if (IO.mapTag("!FileChecksums")) {
354 auto SS = std::make_shared<YAMLChecksumsSubsection>();
355 Subsection.Subsection = SS;
356 } else if (IO.mapTag("!Lines")) {
357 Subsection.Subsection = std::make_shared<YAMLLinesSubsection>();
358 } else if (IO.mapTag("!InlineeLines")) {
359 Subsection.Subsection = std::make_shared<YAMLInlineeLinesSubsection>();
Zachary Turner349c18f2017-06-05 21:40:33 +0000360 } else if (IO.mapTag("!CrossModuleExports")) {
361 Subsection.Subsection =
362 std::make_shared<YAMLCrossModuleExportsSubsection>();
363 } else if (IO.mapTag("!CrossModuleImports")) {
364 Subsection.Subsection =
365 std::make_shared<YAMLCrossModuleImportsSubsection>();
Zachary Turnerdeb39132017-06-09 00:28:08 +0000366 } else if (IO.mapTag("!Symbols")) {
367 Subsection.Subsection = std::make_shared<YAMLSymbolsSubsection>();
368 } else if (IO.mapTag("!StringTable")) {
369 Subsection.Subsection = std::make_shared<YAMLStringTableSubsection>();
370 } else if (IO.mapTag("!FrameData")) {
371 Subsection.Subsection = std::make_shared<YAMLFrameDataSubsection>();
Zachary Turner3226fe92017-06-09 20:46:52 +0000372 } else if (IO.mapTag("!COFFSymbolRVAs")) {
373 Subsection.Subsection = std::make_shared<YAMLCoffSymbolRVASubsection>();
Zachary Turner92dcdda2017-06-02 19:49:14 +0000374 } else {
375 llvm_unreachable("Unexpected subsection tag!");
376 }
377 }
378 Subsection.Subsection->map(IO);
379}
380
Zachary Turnera8cfc292017-06-14 15:59:27 +0000381std::shared_ptr<DebugSubsection> YAMLChecksumsSubsection::toCodeViewSubsection(
382 BumpPtrAllocator &Allocator,
383 const codeview::StringsAndChecksums &SC) const {
384 assert(SC.hasStrings());
385 auto Result = std::make_shared<DebugChecksumsSubsection>(*SC.strings());
Zachary Turner92dcdda2017-06-02 19:49:14 +0000386 for (const auto &CS : Checksums) {
387 Result->addChecksum(CS.FileName, CS.Kind, CS.ChecksumBytes.Bytes);
388 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000389 return Result;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000390}
391
Zachary Turnera8cfc292017-06-14 15:59:27 +0000392std::shared_ptr<DebugSubsection> YAMLLinesSubsection::toCodeViewSubsection(
393 BumpPtrAllocator &Allocator,
394 const codeview::StringsAndChecksums &SC) const {
395 assert(SC.hasStrings() && SC.hasChecksums());
Zachary Turner92dcdda2017-06-02 19:49:14 +0000396 auto Result =
Zachary Turnera8cfc292017-06-14 15:59:27 +0000397 std::make_shared<DebugLinesSubsection>(*SC.checksums(), *SC.strings());
Zachary Turner92dcdda2017-06-02 19:49:14 +0000398 Result->setCodeSize(Lines.CodeSize);
399 Result->setRelocationAddress(Lines.RelocSegment, Lines.RelocOffset);
400 Result->setFlags(Lines.Flags);
401 for (const auto &LC : Lines.Blocks) {
402 Result->createBlock(LC.FileName);
403 if (Result->hasColumnInfo()) {
404 for (const auto &Item : zip(LC.Lines, LC.Columns)) {
405 auto &L = std::get<0>(Item);
406 auto &C = std::get<1>(Item);
407 uint32_t LE = L.LineStart + L.EndDelta;
408 Result->addLineAndColumnInfo(L.Offset,
409 LineInfo(L.LineStart, LE, L.IsStatement),
410 C.StartColumn, C.EndColumn);
411 }
412 } else {
413 for (const auto &L : LC.Lines) {
414 uint32_t LE = L.LineStart + L.EndDelta;
415 Result->addLineInfo(L.Offset, LineInfo(L.LineStart, LE, L.IsStatement));
416 }
417 }
418 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000419 return Result;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000420}
421
Zachary Turnera8cfc292017-06-14 15:59:27 +0000422std::shared_ptr<DebugSubsection>
Zachary Turner92dcdda2017-06-02 19:49:14 +0000423YAMLInlineeLinesSubsection::toCodeViewSubsection(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000424 BumpPtrAllocator &Allocator,
425 const codeview::StringsAndChecksums &SC) const {
426 assert(SC.hasChecksums());
427 auto Result = std::make_shared<DebugInlineeLinesSubsection>(
428 *SC.checksums(), InlineeLines.HasExtraFiles);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000429
430 for (const auto &Site : InlineeLines.Sites) {
431 Result->addInlineSite(TypeIndex(Site.Inlinee), Site.FileName,
432 Site.SourceLineNum);
433 if (!InlineeLines.HasExtraFiles)
434 continue;
435
436 for (auto EF : Site.ExtraFiles) {
437 Result->addExtraFile(EF);
438 }
439 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000440 return Result;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000441}
442
Zachary Turnera8cfc292017-06-14 15:59:27 +0000443std::shared_ptr<DebugSubsection>
Zachary Turner349c18f2017-06-05 21:40:33 +0000444YAMLCrossModuleExportsSubsection::toCodeViewSubsection(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000445 BumpPtrAllocator &Allocator,
446 const codeview::StringsAndChecksums &SC) const {
447 auto Result = std::make_shared<DebugCrossModuleExportsSubsection>();
Zachary Turner349c18f2017-06-05 21:40:33 +0000448 for (const auto &M : Exports)
449 Result->addMapping(M.Local, M.Global);
Zachary Turnera8cfc292017-06-14 15:59:27 +0000450 return Result;
Zachary Turner349c18f2017-06-05 21:40:33 +0000451}
452
Zachary Turnera8cfc292017-06-14 15:59:27 +0000453std::shared_ptr<DebugSubsection>
Zachary Turner349c18f2017-06-05 21:40:33 +0000454YAMLCrossModuleImportsSubsection::toCodeViewSubsection(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000455 BumpPtrAllocator &Allocator,
456 const codeview::StringsAndChecksums &SC) const {
457 assert(SC.hasStrings());
458
459 auto Result =
460 std::make_shared<DebugCrossModuleImportsSubsection>(*SC.strings());
Zachary Turner349c18f2017-06-05 21:40:33 +0000461 for (const auto &M : Imports) {
462 for (const auto Id : M.ImportIds)
463 Result->addImport(M.ModuleName, Id);
464 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000465 return Result;
Zachary Turner349c18f2017-06-05 21:40:33 +0000466}
467
Zachary Turnera8cfc292017-06-14 15:59:27 +0000468std::shared_ptr<DebugSubsection> YAMLSymbolsSubsection::toCodeViewSubsection(
469 BumpPtrAllocator &Allocator,
470 const codeview::StringsAndChecksums &SC) const {
471 auto Result = std::make_shared<DebugSymbolsSubsection>();
Zachary Turnerdeb39132017-06-09 00:28:08 +0000472 for (const auto &Sym : Symbols)
473 Result->addSymbol(
474 Sym.toCodeViewSymbol(Allocator, CodeViewContainer::ObjectFile));
Zachary Turnera8cfc292017-06-14 15:59:27 +0000475 return Result;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000476}
477
Zachary Turnera8cfc292017-06-14 15:59:27 +0000478std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000479YAMLStringTableSubsection::toCodeViewSubsection(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000480 BumpPtrAllocator &Allocator,
481 const codeview::StringsAndChecksums &SC) const {
482 auto Result = std::make_shared<DebugStringTableSubsection>();
Zachary Turnerdeb39132017-06-09 00:28:08 +0000483 for (const auto &Str : this->Strings)
484 Result->insert(Str);
Zachary Turnera8cfc292017-06-14 15:59:27 +0000485 return Result;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000486}
487
Zachary Turnera8cfc292017-06-14 15:59:27 +0000488std::shared_ptr<DebugSubsection> YAMLFrameDataSubsection::toCodeViewSubsection(
489 BumpPtrAllocator &Allocator,
490 const codeview::StringsAndChecksums &SC) const {
491 assert(SC.hasStrings());
492
493 auto Result = std::make_shared<DebugFrameDataSubsection>();
Zachary Turnerdeb39132017-06-09 00:28:08 +0000494 for (const auto &YF : Frames) {
495 codeview::FrameData F;
496 F.CodeSize = YF.CodeSize;
497 F.Flags = YF.Flags;
498 F.LocalSize = YF.LocalSize;
499 F.MaxStackSize = YF.MaxStackSize;
500 F.ParamsSize = YF.ParamsSize;
501 F.PrologSize = YF.PrologSize;
502 F.RvaStart = YF.RvaStart;
503 F.SavedRegsSize = YF.SavedRegsSize;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000504 F.FrameFunc = SC.strings()->insert(YF.FrameFunc);
Zachary Turnerdeb39132017-06-09 00:28:08 +0000505 Result->addFrameData(F);
506 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000507 return Result;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000508}
509
Zachary Turnera8cfc292017-06-14 15:59:27 +0000510std::shared_ptr<DebugSubsection>
Zachary Turner3226fe92017-06-09 20:46:52 +0000511YAMLCoffSymbolRVASubsection::toCodeViewSubsection(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000512 BumpPtrAllocator &Allocator,
513 const codeview::StringsAndChecksums &SC) const {
514 auto Result = std::make_shared<DebugSymbolRVASubsection>();
Zachary Turner3226fe92017-06-09 20:46:52 +0000515 for (const auto &RVA : RVAs)
516 Result->addRVA(RVA);
Zachary Turnera8cfc292017-06-14 15:59:27 +0000517 return Result;
Zachary Turner3226fe92017-06-09 20:46:52 +0000518}
519
Zachary Turner92dcdda2017-06-02 19:49:14 +0000520static Expected<SourceFileChecksumEntry>
521convertOneChecksum(const DebugStringTableSubsectionRef &Strings,
522 const FileChecksumEntry &CS) {
523 auto ExpectedString = Strings.getString(CS.FileNameOffset);
524 if (!ExpectedString)
525 return ExpectedString.takeError();
526
527 SourceFileChecksumEntry Result;
528 Result.ChecksumBytes.Bytes = CS.Checksum;
529 Result.Kind = CS.Kind;
530 Result.FileName = *ExpectedString;
531 return Result;
532}
533
534static Expected<StringRef>
535getFileName(const DebugStringTableSubsectionRef &Strings,
536 const DebugChecksumsSubsectionRef &Checksums, uint32_t FileID) {
537 auto Iter = Checksums.getArray().at(FileID);
538 if (Iter == Checksums.getArray().end())
539 return make_error<CodeViewError>(cv_error_code::no_records);
540 uint32_t Offset = Iter->FileNameOffset;
541 return Strings.getString(Offset);
542}
543
544Expected<std::shared_ptr<YAMLChecksumsSubsection>>
545YAMLChecksumsSubsection::fromCodeViewSubsection(
546 const DebugStringTableSubsectionRef &Strings,
547 const DebugChecksumsSubsectionRef &FC) {
548 auto Result = std::make_shared<YAMLChecksumsSubsection>();
549
550 for (const auto &CS : FC) {
551 auto ConvertedCS = convertOneChecksum(Strings, CS);
552 if (!ConvertedCS)
553 return ConvertedCS.takeError();
554 Result->Checksums.push_back(*ConvertedCS);
555 }
556 return Result;
557}
558
559Expected<std::shared_ptr<YAMLLinesSubsection>>
560YAMLLinesSubsection::fromCodeViewSubsection(
561 const DebugStringTableSubsectionRef &Strings,
562 const DebugChecksumsSubsectionRef &Checksums,
563 const DebugLinesSubsectionRef &Lines) {
564 auto Result = std::make_shared<YAMLLinesSubsection>();
565 Result->Lines.CodeSize = Lines.header()->CodeSize;
566 Result->Lines.RelocOffset = Lines.header()->RelocOffset;
567 Result->Lines.RelocSegment = Lines.header()->RelocSegment;
568 Result->Lines.Flags = static_cast<LineFlags>(uint16_t(Lines.header()->Flags));
569 for (const auto &L : Lines) {
570 SourceLineBlock Block;
571 auto EF = getFileName(Strings, Checksums, L.NameIndex);
572 if (!EF)
573 return EF.takeError();
574 Block.FileName = *EF;
575 if (Lines.hasColumnInfo()) {
576 for (const auto &C : L.Columns) {
577 SourceColumnEntry SCE;
578 SCE.EndColumn = C.EndColumn;
579 SCE.StartColumn = C.StartColumn;
580 Block.Columns.push_back(SCE);
581 }
582 }
583 for (const auto &LN : L.LineNumbers) {
584 SourceLineEntry SLE;
585 LineInfo LI(LN.Flags);
586 SLE.Offset = LN.Offset;
587 SLE.LineStart = LI.getStartLine();
588 SLE.EndDelta = LI.getLineDelta();
589 SLE.IsStatement = LI.isStatement();
590 Block.Lines.push_back(SLE);
591 }
592 Result->Lines.Blocks.push_back(Block);
593 }
594 return Result;
595}
596
597Expected<std::shared_ptr<YAMLInlineeLinesSubsection>>
598YAMLInlineeLinesSubsection::fromCodeViewSubsection(
599 const DebugStringTableSubsectionRef &Strings,
600 const DebugChecksumsSubsectionRef &Checksums,
601 const DebugInlineeLinesSubsectionRef &Lines) {
602 auto Result = std::make_shared<YAMLInlineeLinesSubsection>();
603
604 Result->InlineeLines.HasExtraFiles = Lines.hasExtraFiles();
605 for (const auto &IL : Lines) {
606 InlineeSite Site;
607 auto ExpF = getFileName(Strings, Checksums, IL.Header->FileID);
608 if (!ExpF)
609 return ExpF.takeError();
610 Site.FileName = *ExpF;
611 Site.Inlinee = IL.Header->Inlinee.getIndex();
612 Site.SourceLineNum = IL.Header->SourceLineNum;
613 if (Lines.hasExtraFiles()) {
614 for (const auto EF : IL.ExtraFiles) {
615 auto ExpF2 = getFileName(Strings, Checksums, EF);
616 if (!ExpF2)
617 return ExpF2.takeError();
618 Site.ExtraFiles.push_back(*ExpF2);
619 }
620 }
621 Result->InlineeLines.Sites.push_back(Site);
622 }
623 return Result;
624}
625
Zachary Turner349c18f2017-06-05 21:40:33 +0000626Expected<std::shared_ptr<YAMLCrossModuleExportsSubsection>>
627YAMLCrossModuleExportsSubsection::fromCodeViewSubsection(
628 const DebugCrossModuleExportsSubsectionRef &Exports) {
629 auto Result = std::make_shared<YAMLCrossModuleExportsSubsection>();
630 Result->Exports.assign(Exports.begin(), Exports.end());
631 return Result;
632}
633
634Expected<std::shared_ptr<YAMLCrossModuleImportsSubsection>>
635YAMLCrossModuleImportsSubsection::fromCodeViewSubsection(
636 const DebugStringTableSubsectionRef &Strings,
637 const DebugCrossModuleImportsSubsectionRef &Imports) {
638 auto Result = std::make_shared<YAMLCrossModuleImportsSubsection>();
639 for (const auto &CMI : Imports) {
640 YAMLCrossModuleImport YCMI;
641 auto ExpectedStr = Strings.getString(CMI.Header->ModuleNameOffset);
642 if (!ExpectedStr)
643 return ExpectedStr.takeError();
644 YCMI.ModuleName = *ExpectedStr;
645 YCMI.ImportIds.assign(CMI.Imports.begin(), CMI.Imports.end());
646 Result->Imports.push_back(YCMI);
647 }
648 return Result;
649}
650
Zachary Turnerdeb39132017-06-09 00:28:08 +0000651Expected<std::shared_ptr<YAMLSymbolsSubsection>>
652YAMLSymbolsSubsection::fromCodeViewSubsection(
653 const DebugSymbolsSubsectionRef &Symbols) {
654 auto Result = std::make_shared<YAMLSymbolsSubsection>();
655 for (const auto &Sym : Symbols) {
656 auto S = CodeViewYAML::SymbolRecord::fromCodeViewSymbol(Sym);
657 if (!S)
658 return joinErrors(make_error<CodeViewError>(
659 cv_error_code::corrupt_record,
660 "Invalid CodeView Symbol Record in SymbolRecord "
661 "subsection of .debug$S while converting to YAML!"),
662 S.takeError());
663
664 Result->Symbols.push_back(*S);
665 }
666 return Result;
667}
668
669Expected<std::shared_ptr<YAMLStringTableSubsection>>
670YAMLStringTableSubsection::fromCodeViewSubsection(
671 const DebugStringTableSubsectionRef &Strings) {
672 auto Result = std::make_shared<YAMLStringTableSubsection>();
673 BinaryStreamReader Reader(Strings.getBuffer());
674 StringRef S;
675 // First item is a single null string, skip it.
676 if (auto EC = Reader.readCString(S))
677 return std::move(EC);
678 assert(S.empty());
679 while (Reader.bytesRemaining() > 0) {
680 if (auto EC = Reader.readCString(S))
681 return std::move(EC);
682 Result->Strings.push_back(S);
683 }
684 return Result;
685}
686
687Expected<std::shared_ptr<YAMLFrameDataSubsection>>
688YAMLFrameDataSubsection::fromCodeViewSubsection(
689 const DebugStringTableSubsectionRef &Strings,
690 const DebugFrameDataSubsectionRef &Frames) {
691 auto Result = std::make_shared<YAMLFrameDataSubsection>();
692 for (const auto &F : Frames) {
693 YAMLFrameData YF;
694 YF.CodeSize = F.CodeSize;
695 YF.Flags = F.Flags;
696 YF.LocalSize = F.LocalSize;
697 YF.MaxStackSize = F.MaxStackSize;
698 YF.ParamsSize = F.ParamsSize;
699 YF.PrologSize = F.PrologSize;
700 YF.RvaStart = F.RvaStart;
701 YF.SavedRegsSize = F.SavedRegsSize;
702
703 auto ES = Strings.getString(F.FrameFunc);
704 if (!ES)
705 return joinErrors(
706 make_error<CodeViewError>(
707 cv_error_code::no_records,
708 "Could not find string for string id while mapping FrameData!"),
709 ES.takeError());
710 YF.FrameFunc = *ES;
711 Result->Frames.push_back(YF);
712 }
713 return Result;
714}
715
Zachary Turner3226fe92017-06-09 20:46:52 +0000716Expected<std::shared_ptr<YAMLCoffSymbolRVASubsection>>
717YAMLCoffSymbolRVASubsection::fromCodeViewSubsection(
718 const DebugSymbolRVASubsectionRef &Section) {
719 auto Result = std::make_shared<YAMLCoffSymbolRVASubsection>();
720 for (const auto &RVA : Section) {
721 Result->RVAs.push_back(RVA);
722 }
723 return Result;
724}
725
Zachary Turnera8cfc292017-06-14 15:59:27 +0000726Expected<std::vector<std::shared_ptr<DebugSubsection>>>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000727llvm::CodeViewYAML::toCodeViewSubsectionList(
728 BumpPtrAllocator &Allocator, ArrayRef<YAMLDebugSubsection> Subsections,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000729 const codeview::StringsAndChecksums &SC) {
730 std::vector<std::shared_ptr<DebugSubsection>> Result;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000731 if (Subsections.empty())
Zachary Turner64726f22017-06-02 21:00:22 +0000732 return std::move(Result);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000733
Zachary Turner92dcdda2017-06-02 19:49:14 +0000734 for (const auto &SS : Subsections) {
Zachary Turnera8cfc292017-06-14 15:59:27 +0000735 std::shared_ptr<DebugSubsection> CVS;
736 CVS = SS.Subsection->toCodeViewSubsection(Allocator, SC);
Zachary Turner349c18f2017-06-05 21:40:33 +0000737 assert(CVS != nullptr);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000738 Result.push_back(std::move(CVS));
739 }
740 return std::move(Result);
741}
742
743namespace {
744struct SubsectionConversionVisitor : public DebugSubsectionVisitor {
Zachary Turner1bf77622017-06-08 23:49:01 +0000745 SubsectionConversionVisitor() {}
Zachary Turner92dcdda2017-06-02 19:49:14 +0000746
747 Error visitUnknown(DebugUnknownSubsectionRef &Unknown) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000748 Error visitLines(DebugLinesSubsectionRef &Lines,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000749 const StringsAndChecksumsRef &State) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000750 Error visitFileChecksums(DebugChecksumsSubsectionRef &Checksums,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000751 const StringsAndChecksumsRef &State) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000752 Error visitInlineeLines(DebugInlineeLinesSubsectionRef &Inlinees,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000753 const StringsAndChecksumsRef &State) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000754 Error visitCrossModuleExports(DebugCrossModuleExportsSubsectionRef &Checksums,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000755 const StringsAndChecksumsRef &State) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000756 Error visitCrossModuleImports(DebugCrossModuleImportsSubsectionRef &Inlinees,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000757 const StringsAndChecksumsRef &State) override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000758 Error visitStringTable(DebugStringTableSubsectionRef &ST,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000759 const StringsAndChecksumsRef &State) override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000760 Error visitSymbols(DebugSymbolsSubsectionRef &Symbols,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000761 const StringsAndChecksumsRef &State) override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000762 Error visitFrameData(DebugFrameDataSubsectionRef &Symbols,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000763 const StringsAndChecksumsRef &State) override;
Zachary Turner3226fe92017-06-09 20:46:52 +0000764 Error visitCOFFSymbolRVAs(DebugSymbolRVASubsectionRef &Symbols,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000765 const StringsAndChecksumsRef &State) override;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000766
767 YAMLDebugSubsection Subsection;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000768};
769
770Error SubsectionConversionVisitor::visitUnknown(
771 DebugUnknownSubsectionRef &Unknown) {
772 return make_error<CodeViewError>(cv_error_code::operation_unsupported);
773}
774
Zachary Turner1bf77622017-06-08 23:49:01 +0000775Error SubsectionConversionVisitor::visitLines(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000776 DebugLinesSubsectionRef &Lines, const StringsAndChecksumsRef &State) {
Zachary Turner1bf77622017-06-08 23:49:01 +0000777 auto Result = YAMLLinesSubsection::fromCodeViewSubsection(
778 State.strings(), State.checksums(), Lines);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000779 if (!Result)
780 return Result.takeError();
781 Subsection.Subsection = *Result;
782 return Error::success();
783}
784
785Error SubsectionConversionVisitor::visitFileChecksums(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000786 DebugChecksumsSubsectionRef &Checksums,
787 const StringsAndChecksumsRef &State) {
Zachary Turner1bf77622017-06-08 23:49:01 +0000788 auto Result = YAMLChecksumsSubsection::fromCodeViewSubsection(State.strings(),
789 Checksums);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000790 if (!Result)
791 return Result.takeError();
792 Subsection.Subsection = *Result;
793 return Error::success();
794}
795
796Error SubsectionConversionVisitor::visitInlineeLines(
Zachary Turner1bf77622017-06-08 23:49:01 +0000797 DebugInlineeLinesSubsectionRef &Inlinees,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000798 const StringsAndChecksumsRef &State) {
Zachary Turner92dcdda2017-06-02 19:49:14 +0000799 auto Result = YAMLInlineeLinesSubsection::fromCodeViewSubsection(
Zachary Turner1bf77622017-06-08 23:49:01 +0000800 State.strings(), State.checksums(), Inlinees);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000801 if (!Result)
802 return Result.takeError();
803 Subsection.Subsection = *Result;
804 return Error::success();
805}
Zachary Turner349c18f2017-06-05 21:40:33 +0000806
807Error SubsectionConversionVisitor::visitCrossModuleExports(
Zachary Turner1bf77622017-06-08 23:49:01 +0000808 DebugCrossModuleExportsSubsectionRef &Exports,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000809 const StringsAndChecksumsRef &State) {
Zachary Turner349c18f2017-06-05 21:40:33 +0000810 auto Result =
811 YAMLCrossModuleExportsSubsection::fromCodeViewSubsection(Exports);
812 if (!Result)
813 return Result.takeError();
814 Subsection.Subsection = *Result;
815 return Error::success();
816}
817
818Error SubsectionConversionVisitor::visitCrossModuleImports(
Zachary Turner1bf77622017-06-08 23:49:01 +0000819 DebugCrossModuleImportsSubsectionRef &Imports,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000820 const StringsAndChecksumsRef &State) {
Zachary Turner349c18f2017-06-05 21:40:33 +0000821 auto Result = YAMLCrossModuleImportsSubsection::fromCodeViewSubsection(
Zachary Turner1bf77622017-06-08 23:49:01 +0000822 State.strings(), Imports);
Zachary Turner349c18f2017-06-05 21:40:33 +0000823 if (!Result)
824 return Result.takeError();
825 Subsection.Subsection = *Result;
826 return Error::success();
827}
Zachary Turnerdeb39132017-06-09 00:28:08 +0000828
829Error SubsectionConversionVisitor::visitStringTable(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000830 DebugStringTableSubsectionRef &Strings,
831 const StringsAndChecksumsRef &State) {
Zachary Turnerdeb39132017-06-09 00:28:08 +0000832 auto Result = YAMLStringTableSubsection::fromCodeViewSubsection(Strings);
833 if (!Result)
834 return Result.takeError();
835 Subsection.Subsection = *Result;
836 return Error::success();
837}
838
839Error SubsectionConversionVisitor::visitSymbols(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000840 DebugSymbolsSubsectionRef &Symbols, const StringsAndChecksumsRef &State) {
Zachary Turnerdeb39132017-06-09 00:28:08 +0000841 auto Result = YAMLSymbolsSubsection::fromCodeViewSubsection(Symbols);
842 if (!Result)
843 return Result.takeError();
844 Subsection.Subsection = *Result;
845 return Error::success();
846}
847
848Error SubsectionConversionVisitor::visitFrameData(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000849 DebugFrameDataSubsectionRef &Frames, const StringsAndChecksumsRef &State) {
Zachary Turnerdeb39132017-06-09 00:28:08 +0000850 auto Result =
851 YAMLFrameDataSubsection::fromCodeViewSubsection(State.strings(), Frames);
852 if (!Result)
853 return Result.takeError();
854 Subsection.Subsection = *Result;
855 return Error::success();
856}
Zachary Turner3226fe92017-06-09 20:46:52 +0000857
858Error SubsectionConversionVisitor::visitCOFFSymbolRVAs(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000859 DebugSymbolRVASubsectionRef &RVAs, const StringsAndChecksumsRef &State) {
Zachary Turner3226fe92017-06-09 20:46:52 +0000860 auto Result = YAMLCoffSymbolRVASubsection::fromCodeViewSubsection(RVAs);
861 if (!Result)
862 return Result.takeError();
863 Subsection.Subsection = *Result;
864 return Error::success();
865}
Zachary Turner92dcdda2017-06-02 19:49:14 +0000866}
867
Zachary Turnera8cfc292017-06-14 15:59:27 +0000868Expected<YAMLDebugSubsection>
869YAMLDebugSubsection::fromCodeViewSubection(const StringsAndChecksumsRef &SC,
870 const DebugSubsectionRecord &SS) {
Zachary Turner1bf77622017-06-08 23:49:01 +0000871 SubsectionConversionVisitor V;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000872 if (auto EC = visitDebugSubsection(SS, V, SC))
Zachary Turner92dcdda2017-06-02 19:49:14 +0000873 return std::move(EC);
874
875 return V.Subsection;
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000876}
Zachary Turnerdeb39132017-06-09 00:28:08 +0000877
Zachary Turnera8cfc292017-06-14 15:59:27 +0000878std::vector<YAMLDebugSubsection>
879llvm::CodeViewYAML::fromDebugS(ArrayRef<uint8_t> Data,
880 const StringsAndChecksumsRef &SC) {
881 BinaryStreamReader Reader(Data, support::little);
882 uint32_t Magic;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000883
Zachary Turnera8cfc292017-06-14 15:59:27 +0000884 ExitOnError Err("Invalid .debug$S section!");
885 Err(Reader.readInteger(Magic));
886 assert(Magic == COFF::DEBUG_SECTION_MAGIC && "Invalid .debug$S section!");
887
888 DebugSubsectionArray Subsections;
889 Err(Reader.readArray(Subsections, Reader.bytesRemaining()));
890
891 std::vector<YAMLDebugSubsection> Result;
892
893 for (const auto &SS : Subsections) {
894 auto YamlSS = Err(YAMLDebugSubsection::fromCodeViewSubection(SC, SS));
895 Result.push_back(YamlSS);
Zachary Turnerdeb39132017-06-09 00:28:08 +0000896 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000897 return Result;
898}
899
900void llvm::CodeViewYAML::initializeStringsAndChecksums(
901 ArrayRef<YAMLDebugSubsection> Sections, codeview::StringsAndChecksums &SC) {
902 // String Table and Checksums subsections don't use the allocator.
903 BumpPtrAllocator Allocator;
904
905 // It's possible for checksums and strings to even appear in different debug$S
906 // sections, so we have to make this a stateful function that can build up
907 // the strings and checksums field over multiple iterations.
908
909 // File Checksums require the string table, but may become before it, so we
910 // have to scan for strings first, then scan for checksums again from the
911 // beginning.
912 if (!SC.hasStrings()) {
913 for (const auto &SS : Sections) {
914 if (SS.Subsection->Kind != DebugSubsectionKind::StringTable)
915 continue;
916
917 auto Result = SS.Subsection->toCodeViewSubsection(Allocator, SC);
918 SC.setStrings(
919 std::static_pointer_cast<DebugStringTableSubsection>(Result));
920 break;
921 }
922 }
923
924 if (SC.hasStrings() && !SC.hasChecksums()) {
925 for (const auto &SS : Sections) {
926 if (SS.Subsection->Kind != DebugSubsectionKind::FileChecksums)
927 continue;
928
929 auto Result = SS.Subsection->toCodeViewSubsection(Allocator, SC);
930 SC.setChecksums(
931 std::static_pointer_cast<DebugChecksumsSubsection>(Result));
932 break;
933 }
934 }
Zachary Turnerdeb39132017-06-09 00:28:08 +0000935}