blob: eeebb694589b28d34a030c8d5da02a3ebac8d18f [file] [log] [blame]
Zachary Turner1b88f4f2017-05-31 04:17:13 +00001//===- CodeViewYAMLDebugSections.cpp - CodeView YAMLIO debug sections -----===//
2//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Zachary Turner1b88f4f2017-05-31 04:17:13 +00006//
7//===----------------------------------------------------------------------===//
8//
9// This file defines classes for handling the YAML representation of CodeView
10// Debug Info.
11//
12//===----------------------------------------------------------------------===//
13
14#include "llvm/ObjectYAML/CodeViewYAMLDebugSections.h"
Eugene Zelenko28082ab2017-07-01 01:35:55 +000015#include "llvm/ADT/STLExtras.h"
Zachary Turner1b88f4f2017-05-31 04:17:13 +000016#include "llvm/ADT/StringExtras.h"
Eugene Zelenko28082ab2017-07-01 01:35:55 +000017#include "llvm/ADT/StringRef.h"
18#include "llvm/BinaryFormat/COFF.h"
19#include "llvm/DebugInfo/CodeView/CodeView.h"
Zachary Turner1b88f4f2017-05-31 04:17:13 +000020#include "llvm/DebugInfo/CodeView/CodeViewError.h"
Zachary Turner92dcdda2017-06-02 19:49:14 +000021#include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h"
Zachary Turner349c18f2017-06-05 21:40:33 +000022#include "llvm/DebugInfo/CodeView/DebugCrossExSubsection.h"
23#include "llvm/DebugInfo/CodeView/DebugCrossImpSubsection.h"
Zachary Turnerdeb39132017-06-09 00:28:08 +000024#include "llvm/DebugInfo/CodeView/DebugFrameDataSubsection.h"
Zachary Turner92dcdda2017-06-02 19:49:14 +000025#include "llvm/DebugInfo/CodeView/DebugInlineeLinesSubsection.h"
26#include "llvm/DebugInfo/CodeView/DebugLinesSubsection.h"
27#include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h"
Eugene Zelenko28082ab2017-07-01 01:35:55 +000028#include "llvm/DebugInfo/CodeView/DebugSubsection.h"
Zachary Turner92dcdda2017-06-02 19:49:14 +000029#include "llvm/DebugInfo/CodeView/DebugSubsectionVisitor.h"
Zachary Turner3226fe92017-06-09 20:46:52 +000030#include "llvm/DebugInfo/CodeView/DebugSymbolRVASubsection.h"
Zachary Turnerdeb39132017-06-09 00:28:08 +000031#include "llvm/DebugInfo/CodeView/DebugSymbolsSubsection.h"
Eugene Zelenko28082ab2017-07-01 01:35:55 +000032#include "llvm/DebugInfo/CodeView/Line.h"
Zachary Turnera8cfc292017-06-14 15:59:27 +000033#include "llvm/DebugInfo/CodeView/StringsAndChecksums.h"
Eugene Zelenko28082ab2017-07-01 01:35:55 +000034#include "llvm/DebugInfo/CodeView/TypeIndex.h"
Zachary Turnerdeb39132017-06-09 00:28:08 +000035#include "llvm/ObjectYAML/CodeViewYAMLSymbols.h"
Eugene Zelenko28082ab2017-07-01 01:35:55 +000036#include "llvm/Support/Allocator.h"
37#include "llvm/Support/BinaryStreamReader.h"
38#include "llvm/Support/Endian.h"
39#include "llvm/Support/Error.h"
40#include "llvm/Support/ErrorHandling.h"
41#include "llvm/Support/YAMLTraits.h"
42#include "llvm/Support/raw_ostream.h"
43#include <algorithm>
44#include <cassert>
45#include <cstdint>
46#include <memory>
47#include <string>
48#include <tuple>
49#include <vector>
50
Zachary Turner1b88f4f2017-05-31 04:17:13 +000051using namespace llvm;
52using namespace llvm::codeview;
53using namespace llvm::CodeViewYAML;
54using namespace llvm::CodeViewYAML::detail;
55using namespace llvm::yaml;
56
57LLVM_YAML_IS_SEQUENCE_VECTOR(SourceFileChecksumEntry)
58LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineEntry)
59LLVM_YAML_IS_SEQUENCE_VECTOR(SourceColumnEntry)
60LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineBlock)
61LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineInfo)
62LLVM_YAML_IS_SEQUENCE_VECTOR(InlineeSite)
63LLVM_YAML_IS_SEQUENCE_VECTOR(InlineeInfo)
Zachary Turner349c18f2017-06-05 21:40:33 +000064LLVM_YAML_IS_SEQUENCE_VECTOR(CrossModuleExport)
65LLVM_YAML_IS_SEQUENCE_VECTOR(YAMLCrossModuleImport)
Zachary Turnerdeb39132017-06-09 00:28:08 +000066LLVM_YAML_IS_SEQUENCE_VECTOR(YAMLFrameData)
Zachary Turner1b88f4f2017-05-31 04:17:13 +000067
Francis Visoiu Mistrihb213b272017-12-18 17:38:03 +000068LLVM_YAML_DECLARE_SCALAR_TRAITS(HexFormattedString, QuotingType::None)
Zachary Turner92dcdda2017-06-02 19:49:14 +000069LLVM_YAML_DECLARE_ENUM_TRAITS(DebugSubsectionKind)
Zachary Turner1b88f4f2017-05-31 04:17:13 +000070LLVM_YAML_DECLARE_ENUM_TRAITS(FileChecksumKind)
71LLVM_YAML_DECLARE_BITSET_TRAITS(LineFlags)
72
Zachary Turner349c18f2017-06-05 21:40:33 +000073LLVM_YAML_DECLARE_MAPPING_TRAITS(CrossModuleExport)
Zachary Turnerdeb39132017-06-09 00:28:08 +000074LLVM_YAML_DECLARE_MAPPING_TRAITS(YAMLFrameData)
Zachary Turner349c18f2017-06-05 21:40:33 +000075LLVM_YAML_DECLARE_MAPPING_TRAITS(YAMLCrossModuleImport)
76LLVM_YAML_DECLARE_MAPPING_TRAITS(CrossModuleImportItem)
Zachary Turner92dcdda2017-06-02 19:49:14 +000077LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceLineEntry)
78LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceColumnEntry)
79LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceFileChecksumEntry)
80LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceLineBlock)
81LLVM_YAML_DECLARE_MAPPING_TRAITS(InlineeSite)
82
83namespace llvm {
84namespace CodeViewYAML {
85namespace detail {
Eugene Zelenko28082ab2017-07-01 01:35:55 +000086
Zachary Turner92dcdda2017-06-02 19:49:14 +000087struct YAMLSubsectionBase {
88 explicit YAMLSubsectionBase(DebugSubsectionKind Kind) : Kind(Kind) {}
Eugene Zelenko28082ab2017-07-01 01:35:55 +000089 virtual ~YAMLSubsectionBase() = default;
Zachary Turner92dcdda2017-06-02 19:49:14 +000090
91 virtual void map(IO &IO) = 0;
Zachary Turnera8cfc292017-06-14 15:59:27 +000092 virtual std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +000093 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +000094 const codeview::StringsAndChecksums &SC) const = 0;
Eugene Zelenko28082ab2017-07-01 01:35:55 +000095
96 DebugSubsectionKind Kind;
Zachary Turner92dcdda2017-06-02 19:49:14 +000097};
Eugene Zelenko28082ab2017-07-01 01:35:55 +000098
99} // end namespace detail
100} // end namespace CodeViewYAML
101} // end namespace llvm
Zachary Turner92dcdda2017-06-02 19:49:14 +0000102
103namespace {
Eugene Zelenko28082ab2017-07-01 01:35:55 +0000104
Zachary Turner92dcdda2017-06-02 19:49:14 +0000105struct YAMLChecksumsSubsection : public YAMLSubsectionBase {
106 YAMLChecksumsSubsection()
107 : YAMLSubsectionBase(DebugSubsectionKind::FileChecksums) {}
108
109 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000110 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000111 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000112 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000113 static Expected<std::shared_ptr<YAMLChecksumsSubsection>>
114 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
115 const DebugChecksumsSubsectionRef &FC);
116
117 std::vector<SourceFileChecksumEntry> Checksums;
118};
119
120struct YAMLLinesSubsection : public YAMLSubsectionBase {
121 YAMLLinesSubsection() : YAMLSubsectionBase(DebugSubsectionKind::Lines) {}
122
123 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000124 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000125 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000126 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000127 static Expected<std::shared_ptr<YAMLLinesSubsection>>
128 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
129 const DebugChecksumsSubsectionRef &Checksums,
130 const DebugLinesSubsectionRef &Lines);
131
132 SourceLineInfo Lines;
133};
134
135struct YAMLInlineeLinesSubsection : public YAMLSubsectionBase {
136 YAMLInlineeLinesSubsection()
137 : YAMLSubsectionBase(DebugSubsectionKind::InlineeLines) {}
138
139 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000140 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000141 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000142 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000143 static Expected<std::shared_ptr<YAMLInlineeLinesSubsection>>
144 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
145 const DebugChecksumsSubsectionRef &Checksums,
146 const DebugInlineeLinesSubsectionRef &Lines);
147
148 InlineeInfo InlineeLines;
149};
Zachary Turner349c18f2017-06-05 21:40:33 +0000150
151struct YAMLCrossModuleExportsSubsection : public YAMLSubsectionBase {
152 YAMLCrossModuleExportsSubsection()
153 : YAMLSubsectionBase(DebugSubsectionKind::CrossScopeExports) {}
154
155 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000156 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000157 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000158 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner349c18f2017-06-05 21:40:33 +0000159 static Expected<std::shared_ptr<YAMLCrossModuleExportsSubsection>>
160 fromCodeViewSubsection(const DebugCrossModuleExportsSubsectionRef &Exports);
161
162 std::vector<CrossModuleExport> Exports;
163};
164
165struct YAMLCrossModuleImportsSubsection : public YAMLSubsectionBase {
166 YAMLCrossModuleImportsSubsection()
167 : YAMLSubsectionBase(DebugSubsectionKind::CrossScopeImports) {}
168
169 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000170 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000171 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000172 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner349c18f2017-06-05 21:40:33 +0000173 static Expected<std::shared_ptr<YAMLCrossModuleImportsSubsection>>
174 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
175 const DebugCrossModuleImportsSubsectionRef &Imports);
176
177 std::vector<YAMLCrossModuleImport> Imports;
178};
Zachary Turnerdeb39132017-06-09 00:28:08 +0000179
180struct YAMLSymbolsSubsection : public YAMLSubsectionBase {
181 YAMLSymbolsSubsection() : YAMLSubsectionBase(DebugSubsectionKind::Symbols) {}
182
183 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000184 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000185 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000186 const codeview::StringsAndChecksums &SC) const override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000187 static Expected<std::shared_ptr<YAMLSymbolsSubsection>>
188 fromCodeViewSubsection(const DebugSymbolsSubsectionRef &Symbols);
189
190 std::vector<CodeViewYAML::SymbolRecord> Symbols;
191};
192
193struct YAMLStringTableSubsection : public YAMLSubsectionBase {
194 YAMLStringTableSubsection()
195 : YAMLSubsectionBase(DebugSubsectionKind::StringTable) {}
196
197 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000198 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000199 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000200 const codeview::StringsAndChecksums &SC) const override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000201 static Expected<std::shared_ptr<YAMLStringTableSubsection>>
202 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings);
203
204 std::vector<StringRef> Strings;
205};
206
207struct YAMLFrameDataSubsection : public YAMLSubsectionBase {
208 YAMLFrameDataSubsection()
209 : YAMLSubsectionBase(DebugSubsectionKind::FrameData) {}
210
211 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000212 std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000213 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000214 const codeview::StringsAndChecksums &SC) const override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000215 static Expected<std::shared_ptr<YAMLFrameDataSubsection>>
216 fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings,
217 const DebugFrameDataSubsectionRef &Frames);
218
219 std::vector<YAMLFrameData> Frames;
220};
Zachary Turner3226fe92017-06-09 20:46:52 +0000221
222struct YAMLCoffSymbolRVASubsection : public YAMLSubsectionBase {
223 YAMLCoffSymbolRVASubsection()
224 : YAMLSubsectionBase(DebugSubsectionKind::CoffSymbolRVA) {}
225
226 void map(IO &IO) override;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000227 std::shared_ptr<DebugSubsection>
Zachary Turner3226fe92017-06-09 20:46:52 +0000228 toCodeViewSubsection(BumpPtrAllocator &Allocator,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000229 const codeview::StringsAndChecksums &SC) const override;
Zachary Turner3226fe92017-06-09 20:46:52 +0000230 static Expected<std::shared_ptr<YAMLCoffSymbolRVASubsection>>
231 fromCodeViewSubsection(const DebugSymbolRVASubsectionRef &RVAs);
232
233 std::vector<uint32_t> RVAs;
234};
Eugene Zelenko28082ab2017-07-01 01:35:55 +0000235
236} // end anonymous namespace
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000237
238void ScalarBitSetTraits<LineFlags>::bitset(IO &io, LineFlags &Flags) {
239 io.bitSetCase(Flags, "HasColumnInfo", LF_HaveColumns);
240 io.enumFallback<Hex16>(Flags);
241}
242
243void ScalarEnumerationTraits<FileChecksumKind>::enumeration(
244 IO &io, FileChecksumKind &Kind) {
245 io.enumCase(Kind, "None", FileChecksumKind::None);
246 io.enumCase(Kind, "MD5", FileChecksumKind::MD5);
247 io.enumCase(Kind, "SHA1", FileChecksumKind::SHA1);
248 io.enumCase(Kind, "SHA256", FileChecksumKind::SHA256);
249}
250
251void ScalarTraits<HexFormattedString>::output(const HexFormattedString &Value,
252 void *ctx, raw_ostream &Out) {
253 StringRef Bytes(reinterpret_cast<const char *>(Value.Bytes.data()),
254 Value.Bytes.size());
255 Out << toHex(Bytes);
256}
257
258StringRef ScalarTraits<HexFormattedString>::input(StringRef Scalar, void *ctxt,
259 HexFormattedString &Value) {
260 std::string H = fromHex(Scalar);
261 Value.Bytes.assign(H.begin(), H.end());
262 return StringRef();
263}
264
265void MappingTraits<SourceLineEntry>::mapping(IO &IO, SourceLineEntry &Obj) {
266 IO.mapRequired("Offset", Obj.Offset);
267 IO.mapRequired("LineStart", Obj.LineStart);
268 IO.mapRequired("IsStatement", Obj.IsStatement);
269 IO.mapRequired("EndDelta", Obj.EndDelta);
270}
271
272void MappingTraits<SourceColumnEntry>::mapping(IO &IO, SourceColumnEntry &Obj) {
273 IO.mapRequired("StartColumn", Obj.StartColumn);
274 IO.mapRequired("EndColumn", Obj.EndColumn);
275}
276
277void MappingTraits<SourceLineBlock>::mapping(IO &IO, SourceLineBlock &Obj) {
278 IO.mapRequired("FileName", Obj.FileName);
279 IO.mapRequired("Lines", Obj.Lines);
280 IO.mapRequired("Columns", Obj.Columns);
281}
282
Zachary Turner349c18f2017-06-05 21:40:33 +0000283void MappingTraits<CrossModuleExport>::mapping(IO &IO, CrossModuleExport &Obj) {
284 IO.mapRequired("LocalId", Obj.Local);
285 IO.mapRequired("GlobalId", Obj.Global);
286}
287
288void MappingTraits<YAMLCrossModuleImport>::mapping(IO &IO,
289 YAMLCrossModuleImport &Obj) {
290 IO.mapRequired("Module", Obj.ModuleName);
291 IO.mapRequired("Imports", Obj.ImportIds);
292}
293
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000294void MappingTraits<SourceFileChecksumEntry>::mapping(
295 IO &IO, SourceFileChecksumEntry &Obj) {
296 IO.mapRequired("FileName", Obj.FileName);
297 IO.mapRequired("Kind", Obj.Kind);
298 IO.mapRequired("Checksum", Obj.ChecksumBytes);
299}
300
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000301void MappingTraits<InlineeSite>::mapping(IO &IO, InlineeSite &Obj) {
302 IO.mapRequired("FileName", Obj.FileName);
303 IO.mapRequired("LineNum", Obj.SourceLineNum);
304 IO.mapRequired("Inlinee", Obj.Inlinee);
305 IO.mapOptional("ExtraFiles", Obj.ExtraFiles);
306}
307
Zachary Turnerdeb39132017-06-09 00:28:08 +0000308void MappingTraits<YAMLFrameData>::mapping(IO &IO, YAMLFrameData &Obj) {
309 IO.mapRequired("CodeSize", Obj.CodeSize);
310 IO.mapRequired("FrameFunc", Obj.FrameFunc);
311 IO.mapRequired("LocalSize", Obj.LocalSize);
312 IO.mapOptional("MaxStackSize", Obj.MaxStackSize);
313 IO.mapOptional("ParamsSize", Obj.ParamsSize);
314 IO.mapOptional("PrologSize", Obj.PrologSize);
315 IO.mapOptional("RvaStart", Obj.RvaStart);
316 IO.mapOptional("SavedRegsSize", Obj.SavedRegsSize);
317}
318
Zachary Turner92dcdda2017-06-02 19:49:14 +0000319void YAMLChecksumsSubsection::map(IO &IO) {
320 IO.mapTag("!FileChecksums", true);
321 IO.mapRequired("Checksums", Checksums);
322}
323
324void YAMLLinesSubsection::map(IO &IO) {
325 IO.mapTag("!Lines", true);
326 IO.mapRequired("CodeSize", Lines.CodeSize);
327
328 IO.mapRequired("Flags", Lines.Flags);
329 IO.mapRequired("RelocOffset", Lines.RelocOffset);
330 IO.mapRequired("RelocSegment", Lines.RelocSegment);
331 IO.mapRequired("Blocks", Lines.Blocks);
332}
333
334void YAMLInlineeLinesSubsection::map(IO &IO) {
335 IO.mapTag("!InlineeLines", true);
336 IO.mapRequired("HasExtraFiles", InlineeLines.HasExtraFiles);
337 IO.mapRequired("Sites", InlineeLines.Sites);
338}
339
Zachary Turner349c18f2017-06-05 21:40:33 +0000340void YAMLCrossModuleExportsSubsection::map(IO &IO) {
341 IO.mapTag("!CrossModuleExports", true);
342 IO.mapOptional("Exports", Exports);
343}
344
345void YAMLCrossModuleImportsSubsection::map(IO &IO) {
346 IO.mapTag("!CrossModuleImports", true);
347 IO.mapOptional("Imports", Imports);
348}
349
Zachary Turnerdeb39132017-06-09 00:28:08 +0000350void YAMLSymbolsSubsection::map(IO &IO) {
351 IO.mapTag("!Symbols", true);
352 IO.mapRequired("Records", Symbols);
353}
354
355void YAMLStringTableSubsection::map(IO &IO) {
356 IO.mapTag("!StringTable", true);
357 IO.mapRequired("Strings", Strings);
358}
359
360void YAMLFrameDataSubsection::map(IO &IO) {
361 IO.mapTag("!FrameData", true);
362 IO.mapRequired("Frames", Frames);
363}
364
Zachary Turner3226fe92017-06-09 20:46:52 +0000365void YAMLCoffSymbolRVASubsection::map(IO &IO) {
366 IO.mapTag("!COFFSymbolRVAs", true);
367 IO.mapRequired("RVAs", RVAs);
368}
369
Zachary Turner92dcdda2017-06-02 19:49:14 +0000370void MappingTraits<YAMLDebugSubsection>::mapping(
371 IO &IO, YAMLDebugSubsection &Subsection) {
372 if (!IO.outputting()) {
373 if (IO.mapTag("!FileChecksums")) {
374 auto SS = std::make_shared<YAMLChecksumsSubsection>();
375 Subsection.Subsection = SS;
376 } else if (IO.mapTag("!Lines")) {
377 Subsection.Subsection = std::make_shared<YAMLLinesSubsection>();
378 } else if (IO.mapTag("!InlineeLines")) {
379 Subsection.Subsection = std::make_shared<YAMLInlineeLinesSubsection>();
Zachary Turner349c18f2017-06-05 21:40:33 +0000380 } else if (IO.mapTag("!CrossModuleExports")) {
381 Subsection.Subsection =
382 std::make_shared<YAMLCrossModuleExportsSubsection>();
383 } else if (IO.mapTag("!CrossModuleImports")) {
384 Subsection.Subsection =
385 std::make_shared<YAMLCrossModuleImportsSubsection>();
Zachary Turnerdeb39132017-06-09 00:28:08 +0000386 } else if (IO.mapTag("!Symbols")) {
387 Subsection.Subsection = std::make_shared<YAMLSymbolsSubsection>();
388 } else if (IO.mapTag("!StringTable")) {
389 Subsection.Subsection = std::make_shared<YAMLStringTableSubsection>();
390 } else if (IO.mapTag("!FrameData")) {
391 Subsection.Subsection = std::make_shared<YAMLFrameDataSubsection>();
Zachary Turner3226fe92017-06-09 20:46:52 +0000392 } else if (IO.mapTag("!COFFSymbolRVAs")) {
393 Subsection.Subsection = std::make_shared<YAMLCoffSymbolRVASubsection>();
Zachary Turner92dcdda2017-06-02 19:49:14 +0000394 } else {
395 llvm_unreachable("Unexpected subsection tag!");
396 }
397 }
398 Subsection.Subsection->map(IO);
399}
400
Zachary Turnera8cfc292017-06-14 15:59:27 +0000401std::shared_ptr<DebugSubsection> YAMLChecksumsSubsection::toCodeViewSubsection(
402 BumpPtrAllocator &Allocator,
403 const codeview::StringsAndChecksums &SC) const {
404 assert(SC.hasStrings());
405 auto Result = std::make_shared<DebugChecksumsSubsection>(*SC.strings());
Zachary Turner92dcdda2017-06-02 19:49:14 +0000406 for (const auto &CS : Checksums) {
407 Result->addChecksum(CS.FileName, CS.Kind, CS.ChecksumBytes.Bytes);
408 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000409 return Result;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000410}
411
Zachary Turnera8cfc292017-06-14 15:59:27 +0000412std::shared_ptr<DebugSubsection> YAMLLinesSubsection::toCodeViewSubsection(
413 BumpPtrAllocator &Allocator,
414 const codeview::StringsAndChecksums &SC) const {
415 assert(SC.hasStrings() && SC.hasChecksums());
Zachary Turner92dcdda2017-06-02 19:49:14 +0000416 auto Result =
Zachary Turnera8cfc292017-06-14 15:59:27 +0000417 std::make_shared<DebugLinesSubsection>(*SC.checksums(), *SC.strings());
Zachary Turner92dcdda2017-06-02 19:49:14 +0000418 Result->setCodeSize(Lines.CodeSize);
419 Result->setRelocationAddress(Lines.RelocSegment, Lines.RelocOffset);
420 Result->setFlags(Lines.Flags);
421 for (const auto &LC : Lines.Blocks) {
422 Result->createBlock(LC.FileName);
423 if (Result->hasColumnInfo()) {
424 for (const auto &Item : zip(LC.Lines, LC.Columns)) {
425 auto &L = std::get<0>(Item);
426 auto &C = std::get<1>(Item);
427 uint32_t LE = L.LineStart + L.EndDelta;
428 Result->addLineAndColumnInfo(L.Offset,
429 LineInfo(L.LineStart, LE, L.IsStatement),
430 C.StartColumn, C.EndColumn);
431 }
432 } else {
433 for (const auto &L : LC.Lines) {
434 uint32_t LE = L.LineStart + L.EndDelta;
435 Result->addLineInfo(L.Offset, LineInfo(L.LineStart, LE, L.IsStatement));
436 }
437 }
438 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000439 return Result;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000440}
441
Zachary Turnera8cfc292017-06-14 15:59:27 +0000442std::shared_ptr<DebugSubsection>
Zachary Turner92dcdda2017-06-02 19:49:14 +0000443YAMLInlineeLinesSubsection::toCodeViewSubsection(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000444 BumpPtrAllocator &Allocator,
445 const codeview::StringsAndChecksums &SC) const {
446 assert(SC.hasChecksums());
447 auto Result = std::make_shared<DebugInlineeLinesSubsection>(
448 *SC.checksums(), InlineeLines.HasExtraFiles);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000449
450 for (const auto &Site : InlineeLines.Sites) {
451 Result->addInlineSite(TypeIndex(Site.Inlinee), Site.FileName,
452 Site.SourceLineNum);
453 if (!InlineeLines.HasExtraFiles)
454 continue;
455
456 for (auto EF : Site.ExtraFiles) {
457 Result->addExtraFile(EF);
458 }
459 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000460 return Result;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000461}
462
Zachary Turnera8cfc292017-06-14 15:59:27 +0000463std::shared_ptr<DebugSubsection>
Zachary Turner349c18f2017-06-05 21:40:33 +0000464YAMLCrossModuleExportsSubsection::toCodeViewSubsection(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000465 BumpPtrAllocator &Allocator,
466 const codeview::StringsAndChecksums &SC) const {
467 auto Result = std::make_shared<DebugCrossModuleExportsSubsection>();
Zachary Turner349c18f2017-06-05 21:40:33 +0000468 for (const auto &M : Exports)
469 Result->addMapping(M.Local, M.Global);
Zachary Turnera8cfc292017-06-14 15:59:27 +0000470 return Result;
Zachary Turner349c18f2017-06-05 21:40:33 +0000471}
472
Zachary Turnera8cfc292017-06-14 15:59:27 +0000473std::shared_ptr<DebugSubsection>
Zachary Turner349c18f2017-06-05 21:40:33 +0000474YAMLCrossModuleImportsSubsection::toCodeViewSubsection(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000475 BumpPtrAllocator &Allocator,
476 const codeview::StringsAndChecksums &SC) const {
477 assert(SC.hasStrings());
478
479 auto Result =
480 std::make_shared<DebugCrossModuleImportsSubsection>(*SC.strings());
Zachary Turner349c18f2017-06-05 21:40:33 +0000481 for (const auto &M : Imports) {
482 for (const auto Id : M.ImportIds)
483 Result->addImport(M.ModuleName, Id);
484 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000485 return Result;
Zachary Turner349c18f2017-06-05 21:40:33 +0000486}
487
Zachary Turnera8cfc292017-06-14 15:59:27 +0000488std::shared_ptr<DebugSubsection> YAMLSymbolsSubsection::toCodeViewSubsection(
489 BumpPtrAllocator &Allocator,
490 const codeview::StringsAndChecksums &SC) const {
491 auto Result = std::make_shared<DebugSymbolsSubsection>();
Zachary Turnerdeb39132017-06-09 00:28:08 +0000492 for (const auto &Sym : Symbols)
493 Result->addSymbol(
494 Sym.toCodeViewSymbol(Allocator, CodeViewContainer::ObjectFile));
Zachary Turnera8cfc292017-06-14 15:59:27 +0000495 return Result;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000496}
497
Zachary Turnera8cfc292017-06-14 15:59:27 +0000498std::shared_ptr<DebugSubsection>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000499YAMLStringTableSubsection::toCodeViewSubsection(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000500 BumpPtrAllocator &Allocator,
501 const codeview::StringsAndChecksums &SC) const {
502 auto Result = std::make_shared<DebugStringTableSubsection>();
Zachary Turnerdeb39132017-06-09 00:28:08 +0000503 for (const auto &Str : this->Strings)
504 Result->insert(Str);
Zachary Turnera8cfc292017-06-14 15:59:27 +0000505 return Result;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000506}
507
Zachary Turnera8cfc292017-06-14 15:59:27 +0000508std::shared_ptr<DebugSubsection> YAMLFrameDataSubsection::toCodeViewSubsection(
509 BumpPtrAllocator &Allocator,
510 const codeview::StringsAndChecksums &SC) const {
511 assert(SC.hasStrings());
512
Zachary Turner42e7cc1b2018-09-11 22:35:01 +0000513 auto Result = std::make_shared<DebugFrameDataSubsection>(true);
Zachary Turnerdeb39132017-06-09 00:28:08 +0000514 for (const auto &YF : Frames) {
515 codeview::FrameData F;
516 F.CodeSize = YF.CodeSize;
517 F.Flags = YF.Flags;
518 F.LocalSize = YF.LocalSize;
519 F.MaxStackSize = YF.MaxStackSize;
520 F.ParamsSize = YF.ParamsSize;
521 F.PrologSize = YF.PrologSize;
522 F.RvaStart = YF.RvaStart;
523 F.SavedRegsSize = YF.SavedRegsSize;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000524 F.FrameFunc = SC.strings()->insert(YF.FrameFunc);
Zachary Turnerdeb39132017-06-09 00:28:08 +0000525 Result->addFrameData(F);
526 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000527 return Result;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000528}
529
Zachary Turnera8cfc292017-06-14 15:59:27 +0000530std::shared_ptr<DebugSubsection>
Zachary Turner3226fe92017-06-09 20:46:52 +0000531YAMLCoffSymbolRVASubsection::toCodeViewSubsection(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000532 BumpPtrAllocator &Allocator,
533 const codeview::StringsAndChecksums &SC) const {
534 auto Result = std::make_shared<DebugSymbolRVASubsection>();
Zachary Turner3226fe92017-06-09 20:46:52 +0000535 for (const auto &RVA : RVAs)
536 Result->addRVA(RVA);
Zachary Turnera8cfc292017-06-14 15:59:27 +0000537 return Result;
Zachary Turner3226fe92017-06-09 20:46:52 +0000538}
539
Zachary Turner92dcdda2017-06-02 19:49:14 +0000540static Expected<SourceFileChecksumEntry>
541convertOneChecksum(const DebugStringTableSubsectionRef &Strings,
542 const FileChecksumEntry &CS) {
543 auto ExpectedString = Strings.getString(CS.FileNameOffset);
544 if (!ExpectedString)
545 return ExpectedString.takeError();
546
547 SourceFileChecksumEntry Result;
548 Result.ChecksumBytes.Bytes = CS.Checksum;
549 Result.Kind = CS.Kind;
550 Result.FileName = *ExpectedString;
551 return Result;
552}
553
554static Expected<StringRef>
555getFileName(const DebugStringTableSubsectionRef &Strings,
556 const DebugChecksumsSubsectionRef &Checksums, uint32_t FileID) {
557 auto Iter = Checksums.getArray().at(FileID);
558 if (Iter == Checksums.getArray().end())
559 return make_error<CodeViewError>(cv_error_code::no_records);
560 uint32_t Offset = Iter->FileNameOffset;
561 return Strings.getString(Offset);
562}
563
564Expected<std::shared_ptr<YAMLChecksumsSubsection>>
565YAMLChecksumsSubsection::fromCodeViewSubsection(
566 const DebugStringTableSubsectionRef &Strings,
567 const DebugChecksumsSubsectionRef &FC) {
568 auto Result = std::make_shared<YAMLChecksumsSubsection>();
569
570 for (const auto &CS : FC) {
571 auto ConvertedCS = convertOneChecksum(Strings, CS);
572 if (!ConvertedCS)
573 return ConvertedCS.takeError();
574 Result->Checksums.push_back(*ConvertedCS);
575 }
576 return Result;
577}
578
579Expected<std::shared_ptr<YAMLLinesSubsection>>
580YAMLLinesSubsection::fromCodeViewSubsection(
581 const DebugStringTableSubsectionRef &Strings,
582 const DebugChecksumsSubsectionRef &Checksums,
583 const DebugLinesSubsectionRef &Lines) {
584 auto Result = std::make_shared<YAMLLinesSubsection>();
585 Result->Lines.CodeSize = Lines.header()->CodeSize;
586 Result->Lines.RelocOffset = Lines.header()->RelocOffset;
587 Result->Lines.RelocSegment = Lines.header()->RelocSegment;
588 Result->Lines.Flags = static_cast<LineFlags>(uint16_t(Lines.header()->Flags));
589 for (const auto &L : Lines) {
590 SourceLineBlock Block;
591 auto EF = getFileName(Strings, Checksums, L.NameIndex);
592 if (!EF)
593 return EF.takeError();
594 Block.FileName = *EF;
595 if (Lines.hasColumnInfo()) {
596 for (const auto &C : L.Columns) {
597 SourceColumnEntry SCE;
598 SCE.EndColumn = C.EndColumn;
599 SCE.StartColumn = C.StartColumn;
600 Block.Columns.push_back(SCE);
601 }
602 }
603 for (const auto &LN : L.LineNumbers) {
604 SourceLineEntry SLE;
605 LineInfo LI(LN.Flags);
606 SLE.Offset = LN.Offset;
607 SLE.LineStart = LI.getStartLine();
608 SLE.EndDelta = LI.getLineDelta();
609 SLE.IsStatement = LI.isStatement();
610 Block.Lines.push_back(SLE);
611 }
612 Result->Lines.Blocks.push_back(Block);
613 }
614 return Result;
615}
616
617Expected<std::shared_ptr<YAMLInlineeLinesSubsection>>
618YAMLInlineeLinesSubsection::fromCodeViewSubsection(
619 const DebugStringTableSubsectionRef &Strings,
620 const DebugChecksumsSubsectionRef &Checksums,
621 const DebugInlineeLinesSubsectionRef &Lines) {
622 auto Result = std::make_shared<YAMLInlineeLinesSubsection>();
623
624 Result->InlineeLines.HasExtraFiles = Lines.hasExtraFiles();
625 for (const auto &IL : Lines) {
626 InlineeSite Site;
627 auto ExpF = getFileName(Strings, Checksums, IL.Header->FileID);
628 if (!ExpF)
629 return ExpF.takeError();
630 Site.FileName = *ExpF;
631 Site.Inlinee = IL.Header->Inlinee.getIndex();
632 Site.SourceLineNum = IL.Header->SourceLineNum;
633 if (Lines.hasExtraFiles()) {
634 for (const auto EF : IL.ExtraFiles) {
635 auto ExpF2 = getFileName(Strings, Checksums, EF);
636 if (!ExpF2)
637 return ExpF2.takeError();
638 Site.ExtraFiles.push_back(*ExpF2);
639 }
640 }
641 Result->InlineeLines.Sites.push_back(Site);
642 }
643 return Result;
644}
645
Zachary Turner349c18f2017-06-05 21:40:33 +0000646Expected<std::shared_ptr<YAMLCrossModuleExportsSubsection>>
647YAMLCrossModuleExportsSubsection::fromCodeViewSubsection(
648 const DebugCrossModuleExportsSubsectionRef &Exports) {
649 auto Result = std::make_shared<YAMLCrossModuleExportsSubsection>();
650 Result->Exports.assign(Exports.begin(), Exports.end());
651 return Result;
652}
653
654Expected<std::shared_ptr<YAMLCrossModuleImportsSubsection>>
655YAMLCrossModuleImportsSubsection::fromCodeViewSubsection(
656 const DebugStringTableSubsectionRef &Strings,
657 const DebugCrossModuleImportsSubsectionRef &Imports) {
658 auto Result = std::make_shared<YAMLCrossModuleImportsSubsection>();
659 for (const auto &CMI : Imports) {
660 YAMLCrossModuleImport YCMI;
661 auto ExpectedStr = Strings.getString(CMI.Header->ModuleNameOffset);
662 if (!ExpectedStr)
663 return ExpectedStr.takeError();
664 YCMI.ModuleName = *ExpectedStr;
665 YCMI.ImportIds.assign(CMI.Imports.begin(), CMI.Imports.end());
666 Result->Imports.push_back(YCMI);
667 }
668 return Result;
669}
670
Zachary Turnerdeb39132017-06-09 00:28:08 +0000671Expected<std::shared_ptr<YAMLSymbolsSubsection>>
672YAMLSymbolsSubsection::fromCodeViewSubsection(
673 const DebugSymbolsSubsectionRef &Symbols) {
674 auto Result = std::make_shared<YAMLSymbolsSubsection>();
675 for (const auto &Sym : Symbols) {
676 auto S = CodeViewYAML::SymbolRecord::fromCodeViewSymbol(Sym);
677 if (!S)
678 return joinErrors(make_error<CodeViewError>(
679 cv_error_code::corrupt_record,
680 "Invalid CodeView Symbol Record in SymbolRecord "
681 "subsection of .debug$S while converting to YAML!"),
682 S.takeError());
683
684 Result->Symbols.push_back(*S);
685 }
686 return Result;
687}
688
689Expected<std::shared_ptr<YAMLStringTableSubsection>>
690YAMLStringTableSubsection::fromCodeViewSubsection(
691 const DebugStringTableSubsectionRef &Strings) {
692 auto Result = std::make_shared<YAMLStringTableSubsection>();
693 BinaryStreamReader Reader(Strings.getBuffer());
694 StringRef S;
695 // First item is a single null string, skip it.
696 if (auto EC = Reader.readCString(S))
697 return std::move(EC);
698 assert(S.empty());
699 while (Reader.bytesRemaining() > 0) {
700 if (auto EC = Reader.readCString(S))
701 return std::move(EC);
702 Result->Strings.push_back(S);
703 }
704 return Result;
705}
706
707Expected<std::shared_ptr<YAMLFrameDataSubsection>>
708YAMLFrameDataSubsection::fromCodeViewSubsection(
709 const DebugStringTableSubsectionRef &Strings,
710 const DebugFrameDataSubsectionRef &Frames) {
711 auto Result = std::make_shared<YAMLFrameDataSubsection>();
712 for (const auto &F : Frames) {
713 YAMLFrameData YF;
714 YF.CodeSize = F.CodeSize;
715 YF.Flags = F.Flags;
716 YF.LocalSize = F.LocalSize;
717 YF.MaxStackSize = F.MaxStackSize;
718 YF.ParamsSize = F.ParamsSize;
719 YF.PrologSize = F.PrologSize;
720 YF.RvaStart = F.RvaStart;
721 YF.SavedRegsSize = F.SavedRegsSize;
722
723 auto ES = Strings.getString(F.FrameFunc);
724 if (!ES)
725 return joinErrors(
726 make_error<CodeViewError>(
727 cv_error_code::no_records,
728 "Could not find string for string id while mapping FrameData!"),
729 ES.takeError());
730 YF.FrameFunc = *ES;
731 Result->Frames.push_back(YF);
732 }
733 return Result;
734}
735
Zachary Turner3226fe92017-06-09 20:46:52 +0000736Expected<std::shared_ptr<YAMLCoffSymbolRVASubsection>>
737YAMLCoffSymbolRVASubsection::fromCodeViewSubsection(
738 const DebugSymbolRVASubsectionRef &Section) {
739 auto Result = std::make_shared<YAMLCoffSymbolRVASubsection>();
740 for (const auto &RVA : Section) {
741 Result->RVAs.push_back(RVA);
742 }
743 return Result;
744}
745
Zachary Turnera8cfc292017-06-14 15:59:27 +0000746Expected<std::vector<std::shared_ptr<DebugSubsection>>>
Zachary Turnerdeb39132017-06-09 00:28:08 +0000747llvm::CodeViewYAML::toCodeViewSubsectionList(
748 BumpPtrAllocator &Allocator, ArrayRef<YAMLDebugSubsection> Subsections,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000749 const codeview::StringsAndChecksums &SC) {
750 std::vector<std::shared_ptr<DebugSubsection>> Result;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000751 if (Subsections.empty())
Zachary Turner64726f22017-06-02 21:00:22 +0000752 return std::move(Result);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000753
Zachary Turner92dcdda2017-06-02 19:49:14 +0000754 for (const auto &SS : Subsections) {
Zachary Turnera8cfc292017-06-14 15:59:27 +0000755 std::shared_ptr<DebugSubsection> CVS;
756 CVS = SS.Subsection->toCodeViewSubsection(Allocator, SC);
Zachary Turner349c18f2017-06-05 21:40:33 +0000757 assert(CVS != nullptr);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000758 Result.push_back(std::move(CVS));
759 }
760 return std::move(Result);
761}
762
763namespace {
Eugene Zelenko28082ab2017-07-01 01:35:55 +0000764
Zachary Turner92dcdda2017-06-02 19:49:14 +0000765struct SubsectionConversionVisitor : public DebugSubsectionVisitor {
Eugene Zelenko28082ab2017-07-01 01:35:55 +0000766 SubsectionConversionVisitor() = default;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000767
768 Error visitUnknown(DebugUnknownSubsectionRef &Unknown) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000769 Error visitLines(DebugLinesSubsectionRef &Lines,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000770 const StringsAndChecksumsRef &State) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000771 Error visitFileChecksums(DebugChecksumsSubsectionRef &Checksums,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000772 const StringsAndChecksumsRef &State) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000773 Error visitInlineeLines(DebugInlineeLinesSubsectionRef &Inlinees,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000774 const StringsAndChecksumsRef &State) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000775 Error visitCrossModuleExports(DebugCrossModuleExportsSubsectionRef &Checksums,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000776 const StringsAndChecksumsRef &State) override;
Zachary Turner1bf77622017-06-08 23:49:01 +0000777 Error visitCrossModuleImports(DebugCrossModuleImportsSubsectionRef &Inlinees,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000778 const StringsAndChecksumsRef &State) override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000779 Error visitStringTable(DebugStringTableSubsectionRef &ST,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000780 const StringsAndChecksumsRef &State) override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000781 Error visitSymbols(DebugSymbolsSubsectionRef &Symbols,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000782 const StringsAndChecksumsRef &State) override;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000783 Error visitFrameData(DebugFrameDataSubsectionRef &Symbols,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000784 const StringsAndChecksumsRef &State) override;
Zachary Turner3226fe92017-06-09 20:46:52 +0000785 Error visitCOFFSymbolRVAs(DebugSymbolRVASubsectionRef &Symbols,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000786 const StringsAndChecksumsRef &State) override;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000787
788 YAMLDebugSubsection Subsection;
Zachary Turner92dcdda2017-06-02 19:49:14 +0000789};
790
Eugene Zelenko28082ab2017-07-01 01:35:55 +0000791} // end anonymous namespace
792
Zachary Turner92dcdda2017-06-02 19:49:14 +0000793Error SubsectionConversionVisitor::visitUnknown(
794 DebugUnknownSubsectionRef &Unknown) {
795 return make_error<CodeViewError>(cv_error_code::operation_unsupported);
796}
797
Zachary Turner1bf77622017-06-08 23:49:01 +0000798Error SubsectionConversionVisitor::visitLines(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000799 DebugLinesSubsectionRef &Lines, const StringsAndChecksumsRef &State) {
Zachary Turner1bf77622017-06-08 23:49:01 +0000800 auto Result = YAMLLinesSubsection::fromCodeViewSubsection(
801 State.strings(), State.checksums(), Lines);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000802 if (!Result)
803 return Result.takeError();
804 Subsection.Subsection = *Result;
805 return Error::success();
806}
807
808Error SubsectionConversionVisitor::visitFileChecksums(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000809 DebugChecksumsSubsectionRef &Checksums,
810 const StringsAndChecksumsRef &State) {
Zachary Turner1bf77622017-06-08 23:49:01 +0000811 auto Result = YAMLChecksumsSubsection::fromCodeViewSubsection(State.strings(),
812 Checksums);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000813 if (!Result)
814 return Result.takeError();
815 Subsection.Subsection = *Result;
816 return Error::success();
817}
818
819Error SubsectionConversionVisitor::visitInlineeLines(
Zachary Turner1bf77622017-06-08 23:49:01 +0000820 DebugInlineeLinesSubsectionRef &Inlinees,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000821 const StringsAndChecksumsRef &State) {
Zachary Turner92dcdda2017-06-02 19:49:14 +0000822 auto Result = YAMLInlineeLinesSubsection::fromCodeViewSubsection(
Zachary Turner1bf77622017-06-08 23:49:01 +0000823 State.strings(), State.checksums(), Inlinees);
Zachary Turner92dcdda2017-06-02 19:49:14 +0000824 if (!Result)
825 return Result.takeError();
826 Subsection.Subsection = *Result;
827 return Error::success();
828}
Zachary Turner349c18f2017-06-05 21:40:33 +0000829
830Error SubsectionConversionVisitor::visitCrossModuleExports(
Zachary Turner1bf77622017-06-08 23:49:01 +0000831 DebugCrossModuleExportsSubsectionRef &Exports,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000832 const StringsAndChecksumsRef &State) {
Zachary Turner349c18f2017-06-05 21:40:33 +0000833 auto Result =
834 YAMLCrossModuleExportsSubsection::fromCodeViewSubsection(Exports);
835 if (!Result)
836 return Result.takeError();
837 Subsection.Subsection = *Result;
838 return Error::success();
839}
840
841Error SubsectionConversionVisitor::visitCrossModuleImports(
Zachary Turner1bf77622017-06-08 23:49:01 +0000842 DebugCrossModuleImportsSubsectionRef &Imports,
Zachary Turnera8cfc292017-06-14 15:59:27 +0000843 const StringsAndChecksumsRef &State) {
Zachary Turner349c18f2017-06-05 21:40:33 +0000844 auto Result = YAMLCrossModuleImportsSubsection::fromCodeViewSubsection(
Zachary Turner1bf77622017-06-08 23:49:01 +0000845 State.strings(), Imports);
Zachary Turner349c18f2017-06-05 21:40:33 +0000846 if (!Result)
847 return Result.takeError();
848 Subsection.Subsection = *Result;
849 return Error::success();
850}
Zachary Turnerdeb39132017-06-09 00:28:08 +0000851
852Error SubsectionConversionVisitor::visitStringTable(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000853 DebugStringTableSubsectionRef &Strings,
854 const StringsAndChecksumsRef &State) {
Zachary Turnerdeb39132017-06-09 00:28:08 +0000855 auto Result = YAMLStringTableSubsection::fromCodeViewSubsection(Strings);
856 if (!Result)
857 return Result.takeError();
858 Subsection.Subsection = *Result;
859 return Error::success();
860}
861
862Error SubsectionConversionVisitor::visitSymbols(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000863 DebugSymbolsSubsectionRef &Symbols, const StringsAndChecksumsRef &State) {
Zachary Turnerdeb39132017-06-09 00:28:08 +0000864 auto Result = YAMLSymbolsSubsection::fromCodeViewSubsection(Symbols);
865 if (!Result)
866 return Result.takeError();
867 Subsection.Subsection = *Result;
868 return Error::success();
869}
870
871Error SubsectionConversionVisitor::visitFrameData(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000872 DebugFrameDataSubsectionRef &Frames, const StringsAndChecksumsRef &State) {
Zachary Turnerdeb39132017-06-09 00:28:08 +0000873 auto Result =
874 YAMLFrameDataSubsection::fromCodeViewSubsection(State.strings(), Frames);
875 if (!Result)
876 return Result.takeError();
877 Subsection.Subsection = *Result;
878 return Error::success();
879}
Zachary Turner3226fe92017-06-09 20:46:52 +0000880
881Error SubsectionConversionVisitor::visitCOFFSymbolRVAs(
Zachary Turnera8cfc292017-06-14 15:59:27 +0000882 DebugSymbolRVASubsectionRef &RVAs, const StringsAndChecksumsRef &State) {
Zachary Turner3226fe92017-06-09 20:46:52 +0000883 auto Result = YAMLCoffSymbolRVASubsection::fromCodeViewSubsection(RVAs);
884 if (!Result)
885 return Result.takeError();
886 Subsection.Subsection = *Result;
887 return Error::success();
888}
Zachary Turner92dcdda2017-06-02 19:49:14 +0000889
Zachary Turnera8cfc292017-06-14 15:59:27 +0000890Expected<YAMLDebugSubsection>
891YAMLDebugSubsection::fromCodeViewSubection(const StringsAndChecksumsRef &SC,
892 const DebugSubsectionRecord &SS) {
Zachary Turner1bf77622017-06-08 23:49:01 +0000893 SubsectionConversionVisitor V;
Zachary Turnera8cfc292017-06-14 15:59:27 +0000894 if (auto EC = visitDebugSubsection(SS, V, SC))
Zachary Turner92dcdda2017-06-02 19:49:14 +0000895 return std::move(EC);
896
897 return V.Subsection;
Zachary Turner1b88f4f2017-05-31 04:17:13 +0000898}
Zachary Turnerdeb39132017-06-09 00:28:08 +0000899
Zachary Turnera8cfc292017-06-14 15:59:27 +0000900std::vector<YAMLDebugSubsection>
901llvm::CodeViewYAML::fromDebugS(ArrayRef<uint8_t> Data,
902 const StringsAndChecksumsRef &SC) {
903 BinaryStreamReader Reader(Data, support::little);
904 uint32_t Magic;
Zachary Turnerdeb39132017-06-09 00:28:08 +0000905
Zachary Turnera8cfc292017-06-14 15:59:27 +0000906 ExitOnError Err("Invalid .debug$S section!");
907 Err(Reader.readInteger(Magic));
908 assert(Magic == COFF::DEBUG_SECTION_MAGIC && "Invalid .debug$S section!");
909
910 DebugSubsectionArray Subsections;
911 Err(Reader.readArray(Subsections, Reader.bytesRemaining()));
912
913 std::vector<YAMLDebugSubsection> Result;
914
915 for (const auto &SS : Subsections) {
916 auto YamlSS = Err(YAMLDebugSubsection::fromCodeViewSubection(SC, SS));
917 Result.push_back(YamlSS);
Zachary Turnerdeb39132017-06-09 00:28:08 +0000918 }
Zachary Turnera8cfc292017-06-14 15:59:27 +0000919 return Result;
920}
921
922void llvm::CodeViewYAML::initializeStringsAndChecksums(
923 ArrayRef<YAMLDebugSubsection> Sections, codeview::StringsAndChecksums &SC) {
924 // String Table and Checksums subsections don't use the allocator.
925 BumpPtrAllocator Allocator;
926
927 // It's possible for checksums and strings to even appear in different debug$S
928 // sections, so we have to make this a stateful function that can build up
929 // the strings and checksums field over multiple iterations.
930
931 // File Checksums require the string table, but may become before it, so we
932 // have to scan for strings first, then scan for checksums again from the
933 // beginning.
934 if (!SC.hasStrings()) {
935 for (const auto &SS : Sections) {
936 if (SS.Subsection->Kind != DebugSubsectionKind::StringTable)
937 continue;
938
939 auto Result = SS.Subsection->toCodeViewSubsection(Allocator, SC);
940 SC.setStrings(
941 std::static_pointer_cast<DebugStringTableSubsection>(Result));
942 break;
943 }
944 }
945
946 if (SC.hasStrings() && !SC.hasChecksums()) {
947 for (const auto &SS : Sections) {
948 if (SS.Subsection->Kind != DebugSubsectionKind::FileChecksums)
949 continue;
950
951 auto Result = SS.Subsection->toCodeViewSubsection(Allocator, SC);
952 SC.setChecksums(
953 std::static_pointer_cast<DebugChecksumsSubsection>(Result));
954 break;
955 }
956 }
Zachary Turnerdeb39132017-06-09 00:28:08 +0000957}