blob: 4deeae878013aca2e8446a32633d46e009b052ae [file] [log] [blame]
Zachary Turner7b436b02017-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"
Eugene Zelenkocd903442017-07-01 01:35:55 +000016#include "llvm/ADT/STLExtras.h"
Zachary Turner7b436b02017-05-31 04:17:13 +000017#include "llvm/ADT/StringExtras.h"
Eugene Zelenkocd903442017-07-01 01:35:55 +000018#include "llvm/ADT/StringRef.h"
19#include "llvm/BinaryFormat/COFF.h"
20#include "llvm/DebugInfo/CodeView/CodeView.h"
Zachary Turner7b436b02017-05-31 04:17:13 +000021#include "llvm/DebugInfo/CodeView/CodeViewError.h"
Zachary Turner42d60ef2017-06-02 19:49:14 +000022#include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h"
Zachary Turner11d16782017-06-05 21:40:33 +000023#include "llvm/DebugInfo/CodeView/DebugCrossExSubsection.h"
24#include "llvm/DebugInfo/CodeView/DebugCrossImpSubsection.h"
Zachary Turner68ca30a2017-06-09 00:28:08 +000025#include "llvm/DebugInfo/CodeView/DebugFrameDataSubsection.h"
Zachary Turner42d60ef2017-06-02 19:49:14 +000026#include "llvm/DebugInfo/CodeView/DebugInlineeLinesSubsection.h"
27#include "llvm/DebugInfo/CodeView/DebugLinesSubsection.h"
28#include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h"
Eugene Zelenkocd903442017-07-01 01:35:55 +000029#include "llvm/DebugInfo/CodeView/DebugSubsection.h"
Zachary Turner42d60ef2017-06-02 19:49:14 +000030#include "llvm/DebugInfo/CodeView/DebugSubsectionVisitor.h"
Zachary Turnerbb67f7f2017-06-09 20:46:52 +000031#include "llvm/DebugInfo/CodeView/DebugSymbolRVASubsection.h"
Zachary Turner68ca30a2017-06-09 00:28:08 +000032#include "llvm/DebugInfo/CodeView/DebugSymbolsSubsection.h"
Eugene Zelenkocd903442017-07-01 01:35:55 +000033#include "llvm/DebugInfo/CodeView/Line.h"
Zachary Turner63d2fab2017-06-14 15:59:27 +000034#include "llvm/DebugInfo/CodeView/StringsAndChecksums.h"
Eugene Zelenkocd903442017-07-01 01:35:55 +000035#include "llvm/DebugInfo/CodeView/TypeIndex.h"
Zachary Turner68ca30a2017-06-09 00:28:08 +000036#include "llvm/ObjectYAML/CodeViewYAMLSymbols.h"
Eugene Zelenkocd903442017-07-01 01:35:55 +000037#include "llvm/Support/Allocator.h"
38#include "llvm/Support/BinaryStreamReader.h"
39#include "llvm/Support/Endian.h"
40#include "llvm/Support/Error.h"
41#include "llvm/Support/ErrorHandling.h"
42#include "llvm/Support/YAMLTraits.h"
43#include "llvm/Support/raw_ostream.h"
44#include <algorithm>
45#include <cassert>
46#include <cstdint>
47#include <memory>
48#include <string>
49#include <tuple>
50#include <vector>
51
Zachary Turner7b436b02017-05-31 04:17:13 +000052using namespace llvm;
53using namespace llvm::codeview;
54using namespace llvm::CodeViewYAML;
55using namespace llvm::CodeViewYAML::detail;
56using namespace llvm::yaml;
57
58LLVM_YAML_IS_SEQUENCE_VECTOR(SourceFileChecksumEntry)
59LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineEntry)
60LLVM_YAML_IS_SEQUENCE_VECTOR(SourceColumnEntry)
61LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineBlock)
62LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineInfo)
63LLVM_YAML_IS_SEQUENCE_VECTOR(InlineeSite)
64LLVM_YAML_IS_SEQUENCE_VECTOR(InlineeInfo)
Zachary Turner11d16782017-06-05 21:40:33 +000065LLVM_YAML_IS_SEQUENCE_VECTOR(CrossModuleExport)
66LLVM_YAML_IS_SEQUENCE_VECTOR(YAMLCrossModuleImport)
Zachary Turner68ca30a2017-06-09 00:28:08 +000067LLVM_YAML_IS_SEQUENCE_VECTOR(YAMLFrameData)
Zachary Turner7b436b02017-05-31 04:17:13 +000068
Francis Visoiu Mistrih65ad22d2017-12-18 17:38:03 +000069LLVM_YAML_DECLARE_SCALAR_TRAITS(HexFormattedString, QuotingType::None)
Zachary Turner42d60ef2017-06-02 19:49:14 +000070LLVM_YAML_DECLARE_ENUM_TRAITS(DebugSubsectionKind)
Zachary Turner7b436b02017-05-31 04:17:13 +000071LLVM_YAML_DECLARE_ENUM_TRAITS(FileChecksumKind)
72LLVM_YAML_DECLARE_BITSET_TRAITS(LineFlags)
73
Zachary Turner11d16782017-06-05 21:40:33 +000074LLVM_YAML_DECLARE_MAPPING_TRAITS(CrossModuleExport)
Zachary Turner68ca30a2017-06-09 00:28:08 +000075LLVM_YAML_DECLARE_MAPPING_TRAITS(YAMLFrameData)
Zachary Turner11d16782017-06-05 21:40:33 +000076LLVM_YAML_DECLARE_MAPPING_TRAITS(YAMLCrossModuleImport)
77LLVM_YAML_DECLARE_MAPPING_TRAITS(CrossModuleImportItem)
Zachary Turner42d60ef2017-06-02 19:49:14 +000078LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceLineEntry)
79LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceColumnEntry)
80LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceFileChecksumEntry)
81LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceLineBlock)
82LLVM_YAML_DECLARE_MAPPING_TRAITS(InlineeSite)
83
84namespace llvm {
85namespace CodeViewYAML {
86namespace detail {
Eugene Zelenkocd903442017-07-01 01:35:55 +000087
Zachary Turner42d60ef2017-06-02 19:49:14 +000088struct YAMLSubsectionBase {
89 explicit YAMLSubsectionBase(DebugSubsectionKind Kind) : Kind(Kind) {}
Eugene Zelenkocd903442017-07-01 01:35:55 +000090 virtual ~YAMLSubsectionBase() = default;
Zachary Turner42d60ef2017-06-02 19:49:14 +000091
92 virtual void map(IO &IO) = 0;
Zachary Turner63d2fab2017-06-14 15:59:27 +000093 virtual std::shared_ptr<DebugSubsection>
Zachary Turner68ca30a2017-06-09 00:28:08 +000094 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turner63d2fab2017-06-14 15:59:27 +000095 const codeview::StringsAndChecksums &SC) const = 0;
Eugene Zelenkocd903442017-07-01 01:35:55 +000096
97 DebugSubsectionKind Kind;
Zachary Turner42d60ef2017-06-02 19:49:14 +000098};
Eugene Zelenkocd903442017-07-01 01:35:55 +000099
100} // end namespace detail
101} // end namespace CodeViewYAML
102} // end namespace llvm
Zachary Turner42d60ef2017-06-02 19:49:14 +0000103
104namespace {
Eugene Zelenkocd903442017-07-01 01:35:55 +0000105
Zachary Turner42d60ef2017-06-02 19:49:14 +0000106struct YAMLChecksumsSubsection : public YAMLSubsectionBase {
107 YAMLChecksumsSubsection()
108 : YAMLSubsectionBase(DebugSubsectionKind::FileChecksums) {}
109
110 void map(IO &IO) override;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000111 std::shared_ptr<DebugSubsection>
Zachary Turner68ca30a2017-06-09 00:28:08 +0000112 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000113 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner42d60ef2017-06-02 19:49:14 +0000114 static Expected<std::shared_ptr<YAMLChecksumsSubsection>>
115 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
116 const DebugChecksumsSubsectionRef &FC);
117
118 std::vector<SourceFileChecksumEntry> Checksums;
119};
120
121struct YAMLLinesSubsection : public YAMLSubsectionBase {
122 YAMLLinesSubsection() : YAMLSubsectionBase(DebugSubsectionKind::Lines) {}
123
124 void map(IO &IO) override;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000125 std::shared_ptr<DebugSubsection>
Zachary Turner68ca30a2017-06-09 00:28:08 +0000126 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000127 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner42d60ef2017-06-02 19:49:14 +0000128 static Expected<std::shared_ptr<YAMLLinesSubsection>>
129 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
130 const DebugChecksumsSubsectionRef &Checksums,
131 const DebugLinesSubsectionRef &Lines);
132
133 SourceLineInfo Lines;
134};
135
136struct YAMLInlineeLinesSubsection : public YAMLSubsectionBase {
137 YAMLInlineeLinesSubsection()
138 : YAMLSubsectionBase(DebugSubsectionKind::InlineeLines) {}
139
140 void map(IO &IO) override;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000141 std::shared_ptr<DebugSubsection>
Zachary Turner68ca30a2017-06-09 00:28:08 +0000142 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000143 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner42d60ef2017-06-02 19:49:14 +0000144 static Expected<std::shared_ptr<YAMLInlineeLinesSubsection>>
145 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
146 const DebugChecksumsSubsectionRef &Checksums,
147 const DebugInlineeLinesSubsectionRef &Lines);
148
149 InlineeInfo InlineeLines;
150};
Zachary Turner11d16782017-06-05 21:40:33 +0000151
152struct YAMLCrossModuleExportsSubsection : public YAMLSubsectionBase {
153 YAMLCrossModuleExportsSubsection()
154 : YAMLSubsectionBase(DebugSubsectionKind::CrossScopeExports) {}
155
156 void map(IO &IO) override;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000157 std::shared_ptr<DebugSubsection>
Zachary Turner68ca30a2017-06-09 00:28:08 +0000158 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000159 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner11d16782017-06-05 21:40:33 +0000160 static Expected<std::shared_ptr<YAMLCrossModuleExportsSubsection>>
161 fromCodeViewSubsection(const DebugCrossModuleExportsSubsectionRef &Exports);
162
163 std::vector<CrossModuleExport> Exports;
164};
165
166struct YAMLCrossModuleImportsSubsection : public YAMLSubsectionBase {
167 YAMLCrossModuleImportsSubsection()
168 : YAMLSubsectionBase(DebugSubsectionKind::CrossScopeImports) {}
169
170 void map(IO &IO) override;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000171 std::shared_ptr<DebugSubsection>
Zachary Turner68ca30a2017-06-09 00:28:08 +0000172 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000173 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner11d16782017-06-05 21:40:33 +0000174 static Expected<std::shared_ptr<YAMLCrossModuleImportsSubsection>>
175 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
176 const DebugCrossModuleImportsSubsectionRef &Imports);
177
178 std::vector<YAMLCrossModuleImport> Imports;
179};
Zachary Turner68ca30a2017-06-09 00:28:08 +0000180
181struct YAMLSymbolsSubsection : public YAMLSubsectionBase {
182 YAMLSymbolsSubsection() : YAMLSubsectionBase(DebugSubsectionKind::Symbols) {}
183
184 void map(IO &IO) override;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000185 std::shared_ptr<DebugSubsection>
Zachary Turner68ca30a2017-06-09 00:28:08 +0000186 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000187 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner68ca30a2017-06-09 00:28:08 +0000188 static Expected<std::shared_ptr<YAMLSymbolsSubsection>>
189 fromCodeViewSubsection(const DebugSymbolsSubsectionRef &Symbols);
190
191 std::vector<CodeViewYAML::SymbolRecord> Symbols;
192};
193
194struct YAMLStringTableSubsection : public YAMLSubsectionBase {
195 YAMLStringTableSubsection()
196 : YAMLSubsectionBase(DebugSubsectionKind::StringTable) {}
197
198 void map(IO &IO) override;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000199 std::shared_ptr<DebugSubsection>
Zachary Turner68ca30a2017-06-09 00:28:08 +0000200 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000201 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner68ca30a2017-06-09 00:28:08 +0000202 static Expected<std::shared_ptr<YAMLStringTableSubsection>>
203 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings);
204
205 std::vector<StringRef> Strings;
206};
207
208struct YAMLFrameDataSubsection : public YAMLSubsectionBase {
209 YAMLFrameDataSubsection()
210 : YAMLSubsectionBase(DebugSubsectionKind::FrameData) {}
211
212 void map(IO &IO) override;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000213 std::shared_ptr<DebugSubsection>
Zachary Turner68ca30a2017-06-09 00:28:08 +0000214 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000215 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner68ca30a2017-06-09 00:28:08 +0000216 static Expected<std::shared_ptr<YAMLFrameDataSubsection>>
217 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
218 const DebugFrameDataSubsectionRef &Frames);
219
220 std::vector<YAMLFrameData> Frames;
221};
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000222
223struct YAMLCoffSymbolRVASubsection : public YAMLSubsectionBase {
224 YAMLCoffSymbolRVASubsection()
225 : YAMLSubsectionBase(DebugSubsectionKind::CoffSymbolRVA) {}
226
227 void map(IO &IO) override;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000228 std::shared_ptr<DebugSubsection>
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000229 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000230 const codeview::StringsAndChecksums &SC) const override;
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000231 static Expected<std::shared_ptr<YAMLCoffSymbolRVASubsection>>
232 fromCodeViewSubsection(const DebugSymbolRVASubsectionRef &RVAs);
233
234 std::vector<uint32_t> RVAs;
235};
Eugene Zelenkocd903442017-07-01 01:35:55 +0000236
237} // end anonymous namespace
Zachary Turner7b436b02017-05-31 04:17:13 +0000238
239void ScalarBitSetTraits<LineFlags>::bitset(IO &io, LineFlags &Flags) {
240 io.bitSetCase(Flags, "HasColumnInfo", LF_HaveColumns);
241 io.enumFallback<Hex16>(Flags);
242}
243
244void ScalarEnumerationTraits<FileChecksumKind>::enumeration(
245 IO &io, FileChecksumKind &Kind) {
246 io.enumCase(Kind, "None", FileChecksumKind::None);
247 io.enumCase(Kind, "MD5", FileChecksumKind::MD5);
248 io.enumCase(Kind, "SHA1", FileChecksumKind::SHA1);
249 io.enumCase(Kind, "SHA256", FileChecksumKind::SHA256);
250}
251
252void ScalarTraits<HexFormattedString>::output(const HexFormattedString &Value,
253 void *ctx, raw_ostream &Out) {
254 StringRef Bytes(reinterpret_cast<const char *>(Value.Bytes.data()),
255 Value.Bytes.size());
256 Out << toHex(Bytes);
257}
258
259StringRef ScalarTraits<HexFormattedString>::input(StringRef Scalar, void *ctxt,
260 HexFormattedString &Value) {
261 std::string H = fromHex(Scalar);
262 Value.Bytes.assign(H.begin(), H.end());
263 return StringRef();
264}
265
266void MappingTraits<SourceLineEntry>::mapping(IO &IO, SourceLineEntry &Obj) {
267 IO.mapRequired("Offset", Obj.Offset);
268 IO.mapRequired("LineStart", Obj.LineStart);
269 IO.mapRequired("IsStatement", Obj.IsStatement);
270 IO.mapRequired("EndDelta", Obj.EndDelta);
271}
272
273void MappingTraits<SourceColumnEntry>::mapping(IO &IO, SourceColumnEntry &Obj) {
274 IO.mapRequired("StartColumn", Obj.StartColumn);
275 IO.mapRequired("EndColumn", Obj.EndColumn);
276}
277
278void MappingTraits<SourceLineBlock>::mapping(IO &IO, SourceLineBlock &Obj) {
279 IO.mapRequired("FileName", Obj.FileName);
280 IO.mapRequired("Lines", Obj.Lines);
281 IO.mapRequired("Columns", Obj.Columns);
282}
283
Zachary Turner11d16782017-06-05 21:40:33 +0000284void MappingTraits<CrossModuleExport>::mapping(IO &IO, CrossModuleExport &Obj) {
285 IO.mapRequired("LocalId", Obj.Local);
286 IO.mapRequired("GlobalId", Obj.Global);
287}
288
289void MappingTraits<YAMLCrossModuleImport>::mapping(IO &IO,
290 YAMLCrossModuleImport &Obj) {
291 IO.mapRequired("Module", Obj.ModuleName);
292 IO.mapRequired("Imports", Obj.ImportIds);
293}
294
Zachary Turner7b436b02017-05-31 04:17:13 +0000295void MappingTraits<SourceFileChecksumEntry>::mapping(
296 IO &IO, SourceFileChecksumEntry &Obj) {
297 IO.mapRequired("FileName", Obj.FileName);
298 IO.mapRequired("Kind", Obj.Kind);
299 IO.mapRequired("Checksum", Obj.ChecksumBytes);
300}
301
Zachary Turner7b436b02017-05-31 04:17:13 +0000302void MappingTraits<InlineeSite>::mapping(IO &IO, InlineeSite &Obj) {
303 IO.mapRequired("FileName", Obj.FileName);
304 IO.mapRequired("LineNum", Obj.SourceLineNum);
305 IO.mapRequired("Inlinee", Obj.Inlinee);
306 IO.mapOptional("ExtraFiles", Obj.ExtraFiles);
307}
308
Zachary Turner68ca30a2017-06-09 00:28:08 +0000309void MappingTraits<YAMLFrameData>::mapping(IO &IO, YAMLFrameData &Obj) {
310 IO.mapRequired("CodeSize", Obj.CodeSize);
311 IO.mapRequired("FrameFunc", Obj.FrameFunc);
312 IO.mapRequired("LocalSize", Obj.LocalSize);
313 IO.mapOptional("MaxStackSize", Obj.MaxStackSize);
314 IO.mapOptional("ParamsSize", Obj.ParamsSize);
315 IO.mapOptional("PrologSize", Obj.PrologSize);
316 IO.mapOptional("RvaStart", Obj.RvaStart);
317 IO.mapOptional("SavedRegsSize", Obj.SavedRegsSize);
318}
319
Zachary Turner42d60ef2017-06-02 19:49:14 +0000320void YAMLChecksumsSubsection::map(IO &IO) {
321 IO.mapTag("!FileChecksums", true);
322 IO.mapRequired("Checksums", Checksums);
323}
324
325void YAMLLinesSubsection::map(IO &IO) {
326 IO.mapTag("!Lines", true);
327 IO.mapRequired("CodeSize", Lines.CodeSize);
328
329 IO.mapRequired("Flags", Lines.Flags);
330 IO.mapRequired("RelocOffset", Lines.RelocOffset);
331 IO.mapRequired("RelocSegment", Lines.RelocSegment);
332 IO.mapRequired("Blocks", Lines.Blocks);
333}
334
335void YAMLInlineeLinesSubsection::map(IO &IO) {
336 IO.mapTag("!InlineeLines", true);
337 IO.mapRequired("HasExtraFiles", InlineeLines.HasExtraFiles);
338 IO.mapRequired("Sites", InlineeLines.Sites);
339}
340
Zachary Turner11d16782017-06-05 21:40:33 +0000341void YAMLCrossModuleExportsSubsection::map(IO &IO) {
342 IO.mapTag("!CrossModuleExports", true);
343 IO.mapOptional("Exports", Exports);
344}
345
346void YAMLCrossModuleImportsSubsection::map(IO &IO) {
347 IO.mapTag("!CrossModuleImports", true);
348 IO.mapOptional("Imports", Imports);
349}
350
Zachary Turner68ca30a2017-06-09 00:28:08 +0000351void YAMLSymbolsSubsection::map(IO &IO) {
352 IO.mapTag("!Symbols", true);
353 IO.mapRequired("Records", Symbols);
354}
355
356void YAMLStringTableSubsection::map(IO &IO) {
357 IO.mapTag("!StringTable", true);
358 IO.mapRequired("Strings", Strings);
359}
360
361void YAMLFrameDataSubsection::map(IO &IO) {
362 IO.mapTag("!FrameData", true);
363 IO.mapRequired("Frames", Frames);
364}
365
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000366void YAMLCoffSymbolRVASubsection::map(IO &IO) {
367 IO.mapTag("!COFFSymbolRVAs", true);
368 IO.mapRequired("RVAs", RVAs);
369}
370
Zachary Turner42d60ef2017-06-02 19:49:14 +0000371void MappingTraits<YAMLDebugSubsection>::mapping(
372 IO &IO, YAMLDebugSubsection &Subsection) {
373 if (!IO.outputting()) {
374 if (IO.mapTag("!FileChecksums")) {
375 auto SS = std::make_shared<YAMLChecksumsSubsection>();
376 Subsection.Subsection = SS;
377 } else if (IO.mapTag("!Lines")) {
378 Subsection.Subsection = std::make_shared<YAMLLinesSubsection>();
379 } else if (IO.mapTag("!InlineeLines")) {
380 Subsection.Subsection = std::make_shared<YAMLInlineeLinesSubsection>();
Zachary Turner11d16782017-06-05 21:40:33 +0000381 } else if (IO.mapTag("!CrossModuleExports")) {
382 Subsection.Subsection =
383 std::make_shared<YAMLCrossModuleExportsSubsection>();
384 } else if (IO.mapTag("!CrossModuleImports")) {
385 Subsection.Subsection =
386 std::make_shared<YAMLCrossModuleImportsSubsection>();
Zachary Turner68ca30a2017-06-09 00:28:08 +0000387 } else if (IO.mapTag("!Symbols")) {
388 Subsection.Subsection = std::make_shared<YAMLSymbolsSubsection>();
389 } else if (IO.mapTag("!StringTable")) {
390 Subsection.Subsection = std::make_shared<YAMLStringTableSubsection>();
391 } else if (IO.mapTag("!FrameData")) {
392 Subsection.Subsection = std::make_shared<YAMLFrameDataSubsection>();
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000393 } else if (IO.mapTag("!COFFSymbolRVAs")) {
394 Subsection.Subsection = std::make_shared<YAMLCoffSymbolRVASubsection>();
Zachary Turner42d60ef2017-06-02 19:49:14 +0000395 } else {
396 llvm_unreachable("Unexpected subsection tag!");
397 }
398 }
399 Subsection.Subsection->map(IO);
400}
401
Zachary Turner63d2fab2017-06-14 15:59:27 +0000402std::shared_ptr<DebugSubsection> YAMLChecksumsSubsection::toCodeViewSubsection(
403 BumpPtrAllocator &Allocator,
404 const codeview::StringsAndChecksums &SC) const {
405 assert(SC.hasStrings());
406 auto Result = std::make_shared<DebugChecksumsSubsection>(*SC.strings());
Zachary Turner42d60ef2017-06-02 19:49:14 +0000407 for (const auto &CS : Checksums) {
408 Result->addChecksum(CS.FileName, CS.Kind, CS.ChecksumBytes.Bytes);
409 }
Zachary Turner63d2fab2017-06-14 15:59:27 +0000410 return Result;
Zachary Turner42d60ef2017-06-02 19:49:14 +0000411}
412
Zachary Turner63d2fab2017-06-14 15:59:27 +0000413std::shared_ptr<DebugSubsection> YAMLLinesSubsection::toCodeViewSubsection(
414 BumpPtrAllocator &Allocator,
415 const codeview::StringsAndChecksums &SC) const {
416 assert(SC.hasStrings() && SC.hasChecksums());
Zachary Turner42d60ef2017-06-02 19:49:14 +0000417 auto Result =
Zachary Turner63d2fab2017-06-14 15:59:27 +0000418 std::make_shared<DebugLinesSubsection>(*SC.checksums(), *SC.strings());
Zachary Turner42d60ef2017-06-02 19:49:14 +0000419 Result->setCodeSize(Lines.CodeSize);
420 Result->setRelocationAddress(Lines.RelocSegment, Lines.RelocOffset);
421 Result->setFlags(Lines.Flags);
422 for (const auto &LC : Lines.Blocks) {
423 Result->createBlock(LC.FileName);
424 if (Result->hasColumnInfo()) {
425 for (const auto &Item : zip(LC.Lines, LC.Columns)) {
426 auto &L = std::get<0>(Item);
427 auto &C = std::get<1>(Item);
428 uint32_t LE = L.LineStart + L.EndDelta;
429 Result->addLineAndColumnInfo(L.Offset,
430 LineInfo(L.LineStart, LE, L.IsStatement),
431 C.StartColumn, C.EndColumn);
432 }
433 } else {
434 for (const auto &L : LC.Lines) {
435 uint32_t LE = L.LineStart + L.EndDelta;
436 Result->addLineInfo(L.Offset, LineInfo(L.LineStart, LE, L.IsStatement));
437 }
438 }
439 }
Zachary Turner63d2fab2017-06-14 15:59:27 +0000440 return Result;
Zachary Turner42d60ef2017-06-02 19:49:14 +0000441}
442
Zachary Turner63d2fab2017-06-14 15:59:27 +0000443std::shared_ptr<DebugSubsection>
Zachary Turner42d60ef2017-06-02 19:49:14 +0000444YAMLInlineeLinesSubsection::toCodeViewSubsection(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000445 BumpPtrAllocator &Allocator,
446 const codeview::StringsAndChecksums &SC) const {
447 assert(SC.hasChecksums());
448 auto Result = std::make_shared<DebugInlineeLinesSubsection>(
449 *SC.checksums(), InlineeLines.HasExtraFiles);
Zachary Turner42d60ef2017-06-02 19:49:14 +0000450
451 for (const auto &Site : InlineeLines.Sites) {
452 Result->addInlineSite(TypeIndex(Site.Inlinee), Site.FileName,
453 Site.SourceLineNum);
454 if (!InlineeLines.HasExtraFiles)
455 continue;
456
457 for (auto EF : Site.ExtraFiles) {
458 Result->addExtraFile(EF);
459 }
460 }
Zachary Turner63d2fab2017-06-14 15:59:27 +0000461 return Result;
Zachary Turner42d60ef2017-06-02 19:49:14 +0000462}
463
Zachary Turner63d2fab2017-06-14 15:59:27 +0000464std::shared_ptr<DebugSubsection>
Zachary Turner11d16782017-06-05 21:40:33 +0000465YAMLCrossModuleExportsSubsection::toCodeViewSubsection(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000466 BumpPtrAllocator &Allocator,
467 const codeview::StringsAndChecksums &SC) const {
468 auto Result = std::make_shared<DebugCrossModuleExportsSubsection>();
Zachary Turner11d16782017-06-05 21:40:33 +0000469 for (const auto &M : Exports)
470 Result->addMapping(M.Local, M.Global);
Zachary Turner63d2fab2017-06-14 15:59:27 +0000471 return Result;
Zachary Turner11d16782017-06-05 21:40:33 +0000472}
473
Zachary Turner63d2fab2017-06-14 15:59:27 +0000474std::shared_ptr<DebugSubsection>
Zachary Turner11d16782017-06-05 21:40:33 +0000475YAMLCrossModuleImportsSubsection::toCodeViewSubsection(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000476 BumpPtrAllocator &Allocator,
477 const codeview::StringsAndChecksums &SC) const {
478 assert(SC.hasStrings());
479
480 auto Result =
481 std::make_shared<DebugCrossModuleImportsSubsection>(*SC.strings());
Zachary Turner11d16782017-06-05 21:40:33 +0000482 for (const auto &M : Imports) {
483 for (const auto Id : M.ImportIds)
484 Result->addImport(M.ModuleName, Id);
485 }
Zachary Turner63d2fab2017-06-14 15:59:27 +0000486 return Result;
Zachary Turner11d16782017-06-05 21:40:33 +0000487}
488
Zachary Turner63d2fab2017-06-14 15:59:27 +0000489std::shared_ptr<DebugSubsection> YAMLSymbolsSubsection::toCodeViewSubsection(
490 BumpPtrAllocator &Allocator,
491 const codeview::StringsAndChecksums &SC) const {
492 auto Result = std::make_shared<DebugSymbolsSubsection>();
Zachary Turner68ca30a2017-06-09 00:28:08 +0000493 for (const auto &Sym : Symbols)
494 Result->addSymbol(
495 Sym.toCodeViewSymbol(Allocator, CodeViewContainer::ObjectFile));
Zachary Turner63d2fab2017-06-14 15:59:27 +0000496 return Result;
Zachary Turner68ca30a2017-06-09 00:28:08 +0000497}
498
Zachary Turner63d2fab2017-06-14 15:59:27 +0000499std::shared_ptr<DebugSubsection>
Zachary Turner68ca30a2017-06-09 00:28:08 +0000500YAMLStringTableSubsection::toCodeViewSubsection(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000501 BumpPtrAllocator &Allocator,
502 const codeview::StringsAndChecksums &SC) const {
503 auto Result = std::make_shared<DebugStringTableSubsection>();
Zachary Turner68ca30a2017-06-09 00:28:08 +0000504 for (const auto &Str : this->Strings)
505 Result->insert(Str);
Zachary Turner63d2fab2017-06-14 15:59:27 +0000506 return Result;
Zachary Turner68ca30a2017-06-09 00:28:08 +0000507}
508
Zachary Turner63d2fab2017-06-14 15:59:27 +0000509std::shared_ptr<DebugSubsection> YAMLFrameDataSubsection::toCodeViewSubsection(
510 BumpPtrAllocator &Allocator,
511 const codeview::StringsAndChecksums &SC) const {
512 assert(SC.hasStrings());
513
Zachary Turner77a68882018-09-11 22:35:01 +0000514 auto Result = std::make_shared<DebugFrameDataSubsection>(true);
Zachary Turner68ca30a2017-06-09 00:28:08 +0000515 for (const auto &YF : Frames) {
516 codeview::FrameData F;
517 F.CodeSize = YF.CodeSize;
518 F.Flags = YF.Flags;
519 F.LocalSize = YF.LocalSize;
520 F.MaxStackSize = YF.MaxStackSize;
521 F.ParamsSize = YF.ParamsSize;
522 F.PrologSize = YF.PrologSize;
523 F.RvaStart = YF.RvaStart;
524 F.SavedRegsSize = YF.SavedRegsSize;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000525 F.FrameFunc = SC.strings()->insert(YF.FrameFunc);
Zachary Turner68ca30a2017-06-09 00:28:08 +0000526 Result->addFrameData(F);
527 }
Zachary Turner63d2fab2017-06-14 15:59:27 +0000528 return Result;
Zachary Turner68ca30a2017-06-09 00:28:08 +0000529}
530
Zachary Turner63d2fab2017-06-14 15:59:27 +0000531std::shared_ptr<DebugSubsection>
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000532YAMLCoffSymbolRVASubsection::toCodeViewSubsection(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000533 BumpPtrAllocator &Allocator,
534 const codeview::StringsAndChecksums &SC) const {
535 auto Result = std::make_shared<DebugSymbolRVASubsection>();
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000536 for (const auto &RVA : RVAs)
537 Result->addRVA(RVA);
Zachary Turner63d2fab2017-06-14 15:59:27 +0000538 return Result;
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000539}
540
Zachary Turner42d60ef2017-06-02 19:49:14 +0000541static Expected<SourceFileChecksumEntry>
542convertOneChecksum(const DebugStringTableSubsectionRef &Strings,
543 const FileChecksumEntry &CS) {
544 auto ExpectedString = Strings.getString(CS.FileNameOffset);
545 if (!ExpectedString)
546 return ExpectedString.takeError();
547
548 SourceFileChecksumEntry Result;
549 Result.ChecksumBytes.Bytes = CS.Checksum;
550 Result.Kind = CS.Kind;
551 Result.FileName = *ExpectedString;
552 return Result;
553}
554
555static Expected<StringRef>
556getFileName(const DebugStringTableSubsectionRef &Strings,
557 const DebugChecksumsSubsectionRef &Checksums, uint32_t FileID) {
558 auto Iter = Checksums.getArray().at(FileID);
559 if (Iter == Checksums.getArray().end())
560 return make_error<CodeViewError>(cv_error_code::no_records);
561 uint32_t Offset = Iter->FileNameOffset;
562 return Strings.getString(Offset);
563}
564
565Expected<std::shared_ptr<YAMLChecksumsSubsection>>
566YAMLChecksumsSubsection::fromCodeViewSubsection(
567 const DebugStringTableSubsectionRef &Strings,
568 const DebugChecksumsSubsectionRef &FC) {
569 auto Result = std::make_shared<YAMLChecksumsSubsection>();
570
571 for (const auto &CS : FC) {
572 auto ConvertedCS = convertOneChecksum(Strings, CS);
573 if (!ConvertedCS)
574 return ConvertedCS.takeError();
575 Result->Checksums.push_back(*ConvertedCS);
576 }
577 return Result;
578}
579
580Expected<std::shared_ptr<YAMLLinesSubsection>>
581YAMLLinesSubsection::fromCodeViewSubsection(
582 const DebugStringTableSubsectionRef &Strings,
583 const DebugChecksumsSubsectionRef &Checksums,
584 const DebugLinesSubsectionRef &Lines) {
585 auto Result = std::make_shared<YAMLLinesSubsection>();
586 Result->Lines.CodeSize = Lines.header()->CodeSize;
587 Result->Lines.RelocOffset = Lines.header()->RelocOffset;
588 Result->Lines.RelocSegment = Lines.header()->RelocSegment;
589 Result->Lines.Flags = static_cast<LineFlags>(uint16_t(Lines.header()->Flags));
590 for (const auto &L : Lines) {
591 SourceLineBlock Block;
592 auto EF = getFileName(Strings, Checksums, L.NameIndex);
593 if (!EF)
594 return EF.takeError();
595 Block.FileName = *EF;
596 if (Lines.hasColumnInfo()) {
597 for (const auto &C : L.Columns) {
598 SourceColumnEntry SCE;
599 SCE.EndColumn = C.EndColumn;
600 SCE.StartColumn = C.StartColumn;
601 Block.Columns.push_back(SCE);
602 }
603 }
604 for (const auto &LN : L.LineNumbers) {
605 SourceLineEntry SLE;
606 LineInfo LI(LN.Flags);
607 SLE.Offset = LN.Offset;
608 SLE.LineStart = LI.getStartLine();
609 SLE.EndDelta = LI.getLineDelta();
610 SLE.IsStatement = LI.isStatement();
611 Block.Lines.push_back(SLE);
612 }
613 Result->Lines.Blocks.push_back(Block);
614 }
615 return Result;
616}
617
618Expected<std::shared_ptr<YAMLInlineeLinesSubsection>>
619YAMLInlineeLinesSubsection::fromCodeViewSubsection(
620 const DebugStringTableSubsectionRef &Strings,
621 const DebugChecksumsSubsectionRef &Checksums,
622 const DebugInlineeLinesSubsectionRef &Lines) {
623 auto Result = std::make_shared<YAMLInlineeLinesSubsection>();
624
625 Result->InlineeLines.HasExtraFiles = Lines.hasExtraFiles();
626 for (const auto &IL : Lines) {
627 InlineeSite Site;
628 auto ExpF = getFileName(Strings, Checksums, IL.Header->FileID);
629 if (!ExpF)
630 return ExpF.takeError();
631 Site.FileName = *ExpF;
632 Site.Inlinee = IL.Header->Inlinee.getIndex();
633 Site.SourceLineNum = IL.Header->SourceLineNum;
634 if (Lines.hasExtraFiles()) {
635 for (const auto EF : IL.ExtraFiles) {
636 auto ExpF2 = getFileName(Strings, Checksums, EF);
637 if (!ExpF2)
638 return ExpF2.takeError();
639 Site.ExtraFiles.push_back(*ExpF2);
640 }
641 }
642 Result->InlineeLines.Sites.push_back(Site);
643 }
644 return Result;
645}
646
Zachary Turner11d16782017-06-05 21:40:33 +0000647Expected<std::shared_ptr<YAMLCrossModuleExportsSubsection>>
648YAMLCrossModuleExportsSubsection::fromCodeViewSubsection(
649 const DebugCrossModuleExportsSubsectionRef &Exports) {
650 auto Result = std::make_shared<YAMLCrossModuleExportsSubsection>();
651 Result->Exports.assign(Exports.begin(), Exports.end());
652 return Result;
653}
654
655Expected<std::shared_ptr<YAMLCrossModuleImportsSubsection>>
656YAMLCrossModuleImportsSubsection::fromCodeViewSubsection(
657 const DebugStringTableSubsectionRef &Strings,
658 const DebugCrossModuleImportsSubsectionRef &Imports) {
659 auto Result = std::make_shared<YAMLCrossModuleImportsSubsection>();
660 for (const auto &CMI : Imports) {
661 YAMLCrossModuleImport YCMI;
662 auto ExpectedStr = Strings.getString(CMI.Header->ModuleNameOffset);
663 if (!ExpectedStr)
664 return ExpectedStr.takeError();
665 YCMI.ModuleName = *ExpectedStr;
666 YCMI.ImportIds.assign(CMI.Imports.begin(), CMI.Imports.end());
667 Result->Imports.push_back(YCMI);
668 }
669 return Result;
670}
671
Zachary Turner68ca30a2017-06-09 00:28:08 +0000672Expected<std::shared_ptr<YAMLSymbolsSubsection>>
673YAMLSymbolsSubsection::fromCodeViewSubsection(
674 const DebugSymbolsSubsectionRef &Symbols) {
675 auto Result = std::make_shared<YAMLSymbolsSubsection>();
676 for (const auto &Sym : Symbols) {
677 auto S = CodeViewYAML::SymbolRecord::fromCodeViewSymbol(Sym);
678 if (!S)
679 return joinErrors(make_error<CodeViewError>(
680 cv_error_code::corrupt_record,
681 "Invalid CodeView Symbol Record in SymbolRecord "
682 "subsection of .debug$S while converting to YAML!"),
683 S.takeError());
684
685 Result->Symbols.push_back(*S);
686 }
687 return Result;
688}
689
690Expected<std::shared_ptr<YAMLStringTableSubsection>>
691YAMLStringTableSubsection::fromCodeViewSubsection(
692 const DebugStringTableSubsectionRef &Strings) {
693 auto Result = std::make_shared<YAMLStringTableSubsection>();
694 BinaryStreamReader Reader(Strings.getBuffer());
695 StringRef S;
696 // First item is a single null string, skip it.
697 if (auto EC = Reader.readCString(S))
698 return std::move(EC);
699 assert(S.empty());
700 while (Reader.bytesRemaining() > 0) {
701 if (auto EC = Reader.readCString(S))
702 return std::move(EC);
703 Result->Strings.push_back(S);
704 }
705 return Result;
706}
707
708Expected<std::shared_ptr<YAMLFrameDataSubsection>>
709YAMLFrameDataSubsection::fromCodeViewSubsection(
710 const DebugStringTableSubsectionRef &Strings,
711 const DebugFrameDataSubsectionRef &Frames) {
712 auto Result = std::make_shared<YAMLFrameDataSubsection>();
713 for (const auto &F : Frames) {
714 YAMLFrameData YF;
715 YF.CodeSize = F.CodeSize;
716 YF.Flags = F.Flags;
717 YF.LocalSize = F.LocalSize;
718 YF.MaxStackSize = F.MaxStackSize;
719 YF.ParamsSize = F.ParamsSize;
720 YF.PrologSize = F.PrologSize;
721 YF.RvaStart = F.RvaStart;
722 YF.SavedRegsSize = F.SavedRegsSize;
723
724 auto ES = Strings.getString(F.FrameFunc);
725 if (!ES)
726 return joinErrors(
727 make_error<CodeViewError>(
728 cv_error_code::no_records,
729 "Could not find string for string id while mapping FrameData!"),
730 ES.takeError());
731 YF.FrameFunc = *ES;
732 Result->Frames.push_back(YF);
733 }
734 return Result;
735}
736
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000737Expected<std::shared_ptr<YAMLCoffSymbolRVASubsection>>
738YAMLCoffSymbolRVASubsection::fromCodeViewSubsection(
739 const DebugSymbolRVASubsectionRef &Section) {
740 auto Result = std::make_shared<YAMLCoffSymbolRVASubsection>();
741 for (const auto &RVA : Section) {
742 Result->RVAs.push_back(RVA);
743 }
744 return Result;
745}
746
Zachary Turner63d2fab2017-06-14 15:59:27 +0000747Expected<std::vector<std::shared_ptr<DebugSubsection>>>
Zachary Turner68ca30a2017-06-09 00:28:08 +0000748llvm::CodeViewYAML::toCodeViewSubsectionList(
749 BumpPtrAllocator &Allocator, ArrayRef<YAMLDebugSubsection> Subsections,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000750 const codeview::StringsAndChecksums &SC) {
751 std::vector<std::shared_ptr<DebugSubsection>> Result;
Zachary Turner42d60ef2017-06-02 19:49:14 +0000752 if (Subsections.empty())
Zachary Turnerefbf6a82017-06-02 21:00:22 +0000753 return std::move(Result);
Zachary Turner42d60ef2017-06-02 19:49:14 +0000754
Zachary Turner42d60ef2017-06-02 19:49:14 +0000755 for (const auto &SS : Subsections) {
Zachary Turner63d2fab2017-06-14 15:59:27 +0000756 std::shared_ptr<DebugSubsection> CVS;
757 CVS = SS.Subsection->toCodeViewSubsection(Allocator, SC);
Zachary Turner11d16782017-06-05 21:40:33 +0000758 assert(CVS != nullptr);
Zachary Turner42d60ef2017-06-02 19:49:14 +0000759 Result.push_back(std::move(CVS));
760 }
761 return std::move(Result);
762}
763
764namespace {
Eugene Zelenkocd903442017-07-01 01:35:55 +0000765
Zachary Turner42d60ef2017-06-02 19:49:14 +0000766struct SubsectionConversionVisitor : public DebugSubsectionVisitor {
Eugene Zelenkocd903442017-07-01 01:35:55 +0000767 SubsectionConversionVisitor() = default;
Zachary Turner42d60ef2017-06-02 19:49:14 +0000768
769 Error visitUnknown(DebugUnknownSubsectionRef &Unknown) override;
Zachary Turner8f318ce2017-06-08 23:49:01 +0000770 Error visitLines(DebugLinesSubsectionRef &Lines,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000771 const StringsAndChecksumsRef &State) override;
Zachary Turner8f318ce2017-06-08 23:49:01 +0000772 Error visitFileChecksums(DebugChecksumsSubsectionRef &Checksums,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000773 const StringsAndChecksumsRef &State) override;
Zachary Turner8f318ce2017-06-08 23:49:01 +0000774 Error visitInlineeLines(DebugInlineeLinesSubsectionRef &Inlinees,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000775 const StringsAndChecksumsRef &State) override;
Zachary Turner8f318ce2017-06-08 23:49:01 +0000776 Error visitCrossModuleExports(DebugCrossModuleExportsSubsectionRef &Checksums,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000777 const StringsAndChecksumsRef &State) override;
Zachary Turner8f318ce2017-06-08 23:49:01 +0000778 Error visitCrossModuleImports(DebugCrossModuleImportsSubsectionRef &Inlinees,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000779 const StringsAndChecksumsRef &State) override;
Zachary Turner68ca30a2017-06-09 00:28:08 +0000780 Error visitStringTable(DebugStringTableSubsectionRef &ST,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000781 const StringsAndChecksumsRef &State) override;
Zachary Turner68ca30a2017-06-09 00:28:08 +0000782 Error visitSymbols(DebugSymbolsSubsectionRef &Symbols,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000783 const StringsAndChecksumsRef &State) override;
Zachary Turner68ca30a2017-06-09 00:28:08 +0000784 Error visitFrameData(DebugFrameDataSubsectionRef &Symbols,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000785 const StringsAndChecksumsRef &State) override;
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000786 Error visitCOFFSymbolRVAs(DebugSymbolRVASubsectionRef &Symbols,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000787 const StringsAndChecksumsRef &State) override;
Zachary Turner42d60ef2017-06-02 19:49:14 +0000788
789 YAMLDebugSubsection Subsection;
Zachary Turner42d60ef2017-06-02 19:49:14 +0000790};
791
Eugene Zelenkocd903442017-07-01 01:35:55 +0000792} // end anonymous namespace
793
Zachary Turner42d60ef2017-06-02 19:49:14 +0000794Error SubsectionConversionVisitor::visitUnknown(
795 DebugUnknownSubsectionRef &Unknown) {
796 return make_error<CodeViewError>(cv_error_code::operation_unsupported);
797}
798
Zachary Turner8f318ce2017-06-08 23:49:01 +0000799Error SubsectionConversionVisitor::visitLines(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000800 DebugLinesSubsectionRef &Lines, const StringsAndChecksumsRef &State) {
Zachary Turner8f318ce2017-06-08 23:49:01 +0000801 auto Result = YAMLLinesSubsection::fromCodeViewSubsection(
802 State.strings(), State.checksums(), Lines);
Zachary Turner42d60ef2017-06-02 19:49:14 +0000803 if (!Result)
804 return Result.takeError();
805 Subsection.Subsection = *Result;
806 return Error::success();
807}
808
809Error SubsectionConversionVisitor::visitFileChecksums(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000810 DebugChecksumsSubsectionRef &Checksums,
811 const StringsAndChecksumsRef &State) {
Zachary Turner8f318ce2017-06-08 23:49:01 +0000812 auto Result = YAMLChecksumsSubsection::fromCodeViewSubsection(State.strings(),
813 Checksums);
Zachary Turner42d60ef2017-06-02 19:49:14 +0000814 if (!Result)
815 return Result.takeError();
816 Subsection.Subsection = *Result;
817 return Error::success();
818}
819
820Error SubsectionConversionVisitor::visitInlineeLines(
Zachary Turner8f318ce2017-06-08 23:49:01 +0000821 DebugInlineeLinesSubsectionRef &Inlinees,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000822 const StringsAndChecksumsRef &State) {
Zachary Turner42d60ef2017-06-02 19:49:14 +0000823 auto Result = YAMLInlineeLinesSubsection::fromCodeViewSubsection(
Zachary Turner8f318ce2017-06-08 23:49:01 +0000824 State.strings(), State.checksums(), Inlinees);
Zachary Turner42d60ef2017-06-02 19:49:14 +0000825 if (!Result)
826 return Result.takeError();
827 Subsection.Subsection = *Result;
828 return Error::success();
829}
Zachary Turner11d16782017-06-05 21:40:33 +0000830
831Error SubsectionConversionVisitor::visitCrossModuleExports(
Zachary Turner8f318ce2017-06-08 23:49:01 +0000832 DebugCrossModuleExportsSubsectionRef &Exports,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000833 const StringsAndChecksumsRef &State) {
Zachary Turner11d16782017-06-05 21:40:33 +0000834 auto Result =
835 YAMLCrossModuleExportsSubsection::fromCodeViewSubsection(Exports);
836 if (!Result)
837 return Result.takeError();
838 Subsection.Subsection = *Result;
839 return Error::success();
840}
841
842Error SubsectionConversionVisitor::visitCrossModuleImports(
Zachary Turner8f318ce2017-06-08 23:49:01 +0000843 DebugCrossModuleImportsSubsectionRef &Imports,
Zachary Turner63d2fab2017-06-14 15:59:27 +0000844 const StringsAndChecksumsRef &State) {
Zachary Turner11d16782017-06-05 21:40:33 +0000845 auto Result = YAMLCrossModuleImportsSubsection::fromCodeViewSubsection(
Zachary Turner8f318ce2017-06-08 23:49:01 +0000846 State.strings(), Imports);
Zachary Turner11d16782017-06-05 21:40:33 +0000847 if (!Result)
848 return Result.takeError();
849 Subsection.Subsection = *Result;
850 return Error::success();
851}
Zachary Turner68ca30a2017-06-09 00:28:08 +0000852
853Error SubsectionConversionVisitor::visitStringTable(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000854 DebugStringTableSubsectionRef &Strings,
855 const StringsAndChecksumsRef &State) {
Zachary Turner68ca30a2017-06-09 00:28:08 +0000856 auto Result = YAMLStringTableSubsection::fromCodeViewSubsection(Strings);
857 if (!Result)
858 return Result.takeError();
859 Subsection.Subsection = *Result;
860 return Error::success();
861}
862
863Error SubsectionConversionVisitor::visitSymbols(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000864 DebugSymbolsSubsectionRef &Symbols, const StringsAndChecksumsRef &State) {
Zachary Turner68ca30a2017-06-09 00:28:08 +0000865 auto Result = YAMLSymbolsSubsection::fromCodeViewSubsection(Symbols);
866 if (!Result)
867 return Result.takeError();
868 Subsection.Subsection = *Result;
869 return Error::success();
870}
871
872Error SubsectionConversionVisitor::visitFrameData(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000873 DebugFrameDataSubsectionRef &Frames, const StringsAndChecksumsRef &State) {
Zachary Turner68ca30a2017-06-09 00:28:08 +0000874 auto Result =
875 YAMLFrameDataSubsection::fromCodeViewSubsection(State.strings(), Frames);
876 if (!Result)
877 return Result.takeError();
878 Subsection.Subsection = *Result;
879 return Error::success();
880}
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000881
882Error SubsectionConversionVisitor::visitCOFFSymbolRVAs(
Zachary Turner63d2fab2017-06-14 15:59:27 +0000883 DebugSymbolRVASubsectionRef &RVAs, const StringsAndChecksumsRef &State) {
Zachary Turnerbb67f7f2017-06-09 20:46:52 +0000884 auto Result = YAMLCoffSymbolRVASubsection::fromCodeViewSubsection(RVAs);
885 if (!Result)
886 return Result.takeError();
887 Subsection.Subsection = *Result;
888 return Error::success();
889}
Zachary Turner42d60ef2017-06-02 19:49:14 +0000890
Zachary Turner63d2fab2017-06-14 15:59:27 +0000891Expected<YAMLDebugSubsection>
892YAMLDebugSubsection::fromCodeViewSubection(const StringsAndChecksumsRef &SC,
893 const DebugSubsectionRecord &SS) {
Zachary Turner8f318ce2017-06-08 23:49:01 +0000894 SubsectionConversionVisitor V;
Zachary Turner63d2fab2017-06-14 15:59:27 +0000895 if (auto EC = visitDebugSubsection(SS, V, SC))
Zachary Turner42d60ef2017-06-02 19:49:14 +0000896 return std::move(EC);
897
898 return V.Subsection;
Zachary Turner7b436b02017-05-31 04:17:13 +0000899}
Zachary Turner68ca30a2017-06-09 00:28:08 +0000900
Zachary Turner63d2fab2017-06-14 15:59:27 +0000901std::vector<YAMLDebugSubsection>
902llvm::CodeViewYAML::fromDebugS(ArrayRef<uint8_t> Data,
903 const StringsAndChecksumsRef &SC) {
904 BinaryStreamReader Reader(Data, support::little);
905 uint32_t Magic;
Zachary Turner68ca30a2017-06-09 00:28:08 +0000906
Zachary Turner63d2fab2017-06-14 15:59:27 +0000907 ExitOnError Err("Invalid .debug$S section!");
908 Err(Reader.readInteger(Magic));
909 assert(Magic == COFF::DEBUG_SECTION_MAGIC && "Invalid .debug$S section!");
910
911 DebugSubsectionArray Subsections;
912 Err(Reader.readArray(Subsections, Reader.bytesRemaining()));
913
914 std::vector<YAMLDebugSubsection> Result;
915
916 for (const auto &SS : Subsections) {
917 auto YamlSS = Err(YAMLDebugSubsection::fromCodeViewSubection(SC, SS));
918 Result.push_back(YamlSS);
Zachary Turner68ca30a2017-06-09 00:28:08 +0000919 }
Zachary Turner63d2fab2017-06-14 15:59:27 +0000920 return Result;
921}
922
923void llvm::CodeViewYAML::initializeStringsAndChecksums(
924 ArrayRef<YAMLDebugSubsection> Sections, codeview::StringsAndChecksums &SC) {
925 // String Table and Checksums subsections don't use the allocator.
926 BumpPtrAllocator Allocator;
927
928 // It's possible for checksums and strings to even appear in different debug$S
929 // sections, so we have to make this a stateful function that can build up
930 // the strings and checksums field over multiple iterations.
931
932 // File Checksums require the string table, but may become before it, so we
933 // have to scan for strings first, then scan for checksums again from the
934 // beginning.
935 if (!SC.hasStrings()) {
936 for (const auto &SS : Sections) {
937 if (SS.Subsection->Kind != DebugSubsectionKind::StringTable)
938 continue;
939
940 auto Result = SS.Subsection->toCodeViewSubsection(Allocator, SC);
941 SC.setStrings(
942 std::static_pointer_cast<DebugStringTableSubsection>(Result));
943 break;
944 }
945 }
946
947 if (SC.hasStrings() && !SC.hasChecksums()) {
948 for (const auto &SS : Sections) {
949 if (SS.Subsection->Kind != DebugSubsectionKind::FileChecksums)
950 continue;
951
952 auto Result = SS.Subsection->toCodeViewSubsection(Allocator, SC);
953 SC.setChecksums(
954 std::static_pointer_cast<DebugChecksumsSubsection>(Result));
955 break;
956 }
957 }
Zachary Turner68ca30a2017-06-09 00:28:08 +0000958}