| //===- CodeViewYAMLDebugSections.cpp - CodeView YAMLIO debug sections -----===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file defines classes for handling the YAML representation of CodeView |
| // Debug Info. |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "llvm/ObjectYAML/CodeViewYAMLDebugSections.h" |
| #include "llvm/ADT/STLExtras.h" |
| #include "llvm/ADT/StringExtras.h" |
| #include "llvm/ADT/StringRef.h" |
| #include "llvm/BinaryFormat/COFF.h" |
| #include "llvm/DebugInfo/CodeView/CodeView.h" |
| #include "llvm/DebugInfo/CodeView/CodeViewError.h" |
| #include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h" |
| #include "llvm/DebugInfo/CodeView/DebugCrossExSubsection.h" |
| #include "llvm/DebugInfo/CodeView/DebugCrossImpSubsection.h" |
| #include "llvm/DebugInfo/CodeView/DebugFrameDataSubsection.h" |
| #include "llvm/DebugInfo/CodeView/DebugInlineeLinesSubsection.h" |
| #include "llvm/DebugInfo/CodeView/DebugLinesSubsection.h" |
| #include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h" |
| #include "llvm/DebugInfo/CodeView/DebugSubsection.h" |
| #include "llvm/DebugInfo/CodeView/DebugSubsectionVisitor.h" |
| #include "llvm/DebugInfo/CodeView/DebugSymbolRVASubsection.h" |
| #include "llvm/DebugInfo/CodeView/DebugSymbolsSubsection.h" |
| #include "llvm/DebugInfo/CodeView/Line.h" |
| #include "llvm/DebugInfo/CodeView/StringsAndChecksums.h" |
| #include "llvm/DebugInfo/CodeView/TypeIndex.h" |
| #include "llvm/ObjectYAML/CodeViewYAMLSymbols.h" |
| #include "llvm/Support/Allocator.h" |
| #include "llvm/Support/BinaryStreamReader.h" |
| #include "llvm/Support/Endian.h" |
| #include "llvm/Support/Error.h" |
| #include "llvm/Support/ErrorHandling.h" |
| #include "llvm/Support/YAMLTraits.h" |
| #include "llvm/Support/raw_ostream.h" |
| #include <algorithm> |
| #include <cassert> |
| #include <cstdint> |
| #include <memory> |
| #include <string> |
| #include <tuple> |
| #include <vector> |
| |
| using namespace llvm; |
| using namespace llvm::codeview; |
| using namespace llvm::CodeViewYAML; |
| using namespace llvm::CodeViewYAML::detail; |
| using namespace llvm::yaml; |
| |
| LLVM_YAML_IS_SEQUENCE_VECTOR(SourceFileChecksumEntry) |
| LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineEntry) |
| LLVM_YAML_IS_SEQUENCE_VECTOR(SourceColumnEntry) |
| LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineBlock) |
| LLVM_YAML_IS_SEQUENCE_VECTOR(SourceLineInfo) |
| LLVM_YAML_IS_SEQUENCE_VECTOR(InlineeSite) |
| LLVM_YAML_IS_SEQUENCE_VECTOR(InlineeInfo) |
| LLVM_YAML_IS_SEQUENCE_VECTOR(CrossModuleExport) |
| LLVM_YAML_IS_SEQUENCE_VECTOR(YAMLCrossModuleImport) |
| LLVM_YAML_IS_SEQUENCE_VECTOR(YAMLFrameData) |
| |
| LLVM_YAML_DECLARE_SCALAR_TRAITS(HexFormattedString, QuotingType::None) |
| LLVM_YAML_DECLARE_ENUM_TRAITS(DebugSubsectionKind) |
| LLVM_YAML_DECLARE_ENUM_TRAITS(FileChecksumKind) |
| LLVM_YAML_DECLARE_BITSET_TRAITS(LineFlags) |
| |
| LLVM_YAML_DECLARE_MAPPING_TRAITS(CrossModuleExport) |
| LLVM_YAML_DECLARE_MAPPING_TRAITS(YAMLFrameData) |
| LLVM_YAML_DECLARE_MAPPING_TRAITS(YAMLCrossModuleImport) |
| LLVM_YAML_DECLARE_MAPPING_TRAITS(CrossModuleImportItem) |
| LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceLineEntry) |
| LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceColumnEntry) |
| LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceFileChecksumEntry) |
| LLVM_YAML_DECLARE_MAPPING_TRAITS(SourceLineBlock) |
| LLVM_YAML_DECLARE_MAPPING_TRAITS(InlineeSite) |
| |
| namespace llvm { |
| namespace CodeViewYAML { |
| namespace detail { |
| |
| struct YAMLSubsectionBase { |
| explicit YAMLSubsectionBase(DebugSubsectionKind Kind) : Kind(Kind) {} |
| virtual ~YAMLSubsectionBase() = default; |
| |
| virtual void map(IO &IO) = 0; |
| virtual std::shared_ptr<DebugSubsection> |
| toCodeViewSubsection(BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const = 0; |
| |
| DebugSubsectionKind Kind; |
| }; |
| |
| } // end namespace detail |
| } // end namespace CodeViewYAML |
| } // end namespace llvm |
| |
| namespace { |
| |
| struct YAMLChecksumsSubsection : public YAMLSubsectionBase { |
| YAMLChecksumsSubsection() |
| : YAMLSubsectionBase(DebugSubsectionKind::FileChecksums) {} |
| |
| void map(IO &IO) override; |
| std::shared_ptr<DebugSubsection> |
| toCodeViewSubsection(BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const override; |
| static Expected<std::shared_ptr<YAMLChecksumsSubsection>> |
| fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings, |
| const DebugChecksumsSubsectionRef &FC); |
| |
| std::vector<SourceFileChecksumEntry> Checksums; |
| }; |
| |
| struct YAMLLinesSubsection : public YAMLSubsectionBase { |
| YAMLLinesSubsection() : YAMLSubsectionBase(DebugSubsectionKind::Lines) {} |
| |
| void map(IO &IO) override; |
| std::shared_ptr<DebugSubsection> |
| toCodeViewSubsection(BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const override; |
| static Expected<std::shared_ptr<YAMLLinesSubsection>> |
| fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings, |
| const DebugChecksumsSubsectionRef &Checksums, |
| const DebugLinesSubsectionRef &Lines); |
| |
| SourceLineInfo Lines; |
| }; |
| |
| struct YAMLInlineeLinesSubsection : public YAMLSubsectionBase { |
| YAMLInlineeLinesSubsection() |
| : YAMLSubsectionBase(DebugSubsectionKind::InlineeLines) {} |
| |
| void map(IO &IO) override; |
| std::shared_ptr<DebugSubsection> |
| toCodeViewSubsection(BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const override; |
| static Expected<std::shared_ptr<YAMLInlineeLinesSubsection>> |
| fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings, |
| const DebugChecksumsSubsectionRef &Checksums, |
| const DebugInlineeLinesSubsectionRef &Lines); |
| |
| InlineeInfo InlineeLines; |
| }; |
| |
| struct YAMLCrossModuleExportsSubsection : public YAMLSubsectionBase { |
| YAMLCrossModuleExportsSubsection() |
| : YAMLSubsectionBase(DebugSubsectionKind::CrossScopeExports) {} |
| |
| void map(IO &IO) override; |
| std::shared_ptr<DebugSubsection> |
| toCodeViewSubsection(BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const override; |
| static Expected<std::shared_ptr<YAMLCrossModuleExportsSubsection>> |
| fromCodeViewSubsection(const DebugCrossModuleExportsSubsectionRef &Exports); |
| |
| std::vector<CrossModuleExport> Exports; |
| }; |
| |
| struct YAMLCrossModuleImportsSubsection : public YAMLSubsectionBase { |
| YAMLCrossModuleImportsSubsection() |
| : YAMLSubsectionBase(DebugSubsectionKind::CrossScopeImports) {} |
| |
| void map(IO &IO) override; |
| std::shared_ptr<DebugSubsection> |
| toCodeViewSubsection(BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const override; |
| static Expected<std::shared_ptr<YAMLCrossModuleImportsSubsection>> |
| fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings, |
| const DebugCrossModuleImportsSubsectionRef &Imports); |
| |
| std::vector<YAMLCrossModuleImport> Imports; |
| }; |
| |
| struct YAMLSymbolsSubsection : public YAMLSubsectionBase { |
| YAMLSymbolsSubsection() : YAMLSubsectionBase(DebugSubsectionKind::Symbols) {} |
| |
| void map(IO &IO) override; |
| std::shared_ptr<DebugSubsection> |
| toCodeViewSubsection(BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const override; |
| static Expected<std::shared_ptr<YAMLSymbolsSubsection>> |
| fromCodeViewSubsection(const DebugSymbolsSubsectionRef &Symbols); |
| |
| std::vector<CodeViewYAML::SymbolRecord> Symbols; |
| }; |
| |
| struct YAMLStringTableSubsection : public YAMLSubsectionBase { |
| YAMLStringTableSubsection() |
| : YAMLSubsectionBase(DebugSubsectionKind::StringTable) {} |
| |
| void map(IO &IO) override; |
| std::shared_ptr<DebugSubsection> |
| toCodeViewSubsection(BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const override; |
| static Expected<std::shared_ptr<YAMLStringTableSubsection>> |
| fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings); |
| |
| std::vector<StringRef> Strings; |
| }; |
| |
| struct YAMLFrameDataSubsection : public YAMLSubsectionBase { |
| YAMLFrameDataSubsection() |
| : YAMLSubsectionBase(DebugSubsectionKind::FrameData) {} |
| |
| void map(IO &IO) override; |
| std::shared_ptr<DebugSubsection> |
| toCodeViewSubsection(BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const override; |
| static Expected<std::shared_ptr<YAMLFrameDataSubsection>> |
| fromCodeViewSubsection(const DebugStringTableSubsectionRef &Strings, |
| const DebugFrameDataSubsectionRef &Frames); |
| |
| std::vector<YAMLFrameData> Frames; |
| }; |
| |
| struct YAMLCoffSymbolRVASubsection : public YAMLSubsectionBase { |
| YAMLCoffSymbolRVASubsection() |
| : YAMLSubsectionBase(DebugSubsectionKind::CoffSymbolRVA) {} |
| |
| void map(IO &IO) override; |
| std::shared_ptr<DebugSubsection> |
| toCodeViewSubsection(BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const override; |
| static Expected<std::shared_ptr<YAMLCoffSymbolRVASubsection>> |
| fromCodeViewSubsection(const DebugSymbolRVASubsectionRef &RVAs); |
| |
| std::vector<uint32_t> RVAs; |
| }; |
| |
| } // end anonymous namespace |
| |
| void ScalarBitSetTraits<LineFlags>::bitset(IO &io, LineFlags &Flags) { |
| io.bitSetCase(Flags, "HasColumnInfo", LF_HaveColumns); |
| io.enumFallback<Hex16>(Flags); |
| } |
| |
| void ScalarEnumerationTraits<FileChecksumKind>::enumeration( |
| IO &io, FileChecksumKind &Kind) { |
| io.enumCase(Kind, "None", FileChecksumKind::None); |
| io.enumCase(Kind, "MD5", FileChecksumKind::MD5); |
| io.enumCase(Kind, "SHA1", FileChecksumKind::SHA1); |
| io.enumCase(Kind, "SHA256", FileChecksumKind::SHA256); |
| } |
| |
| void ScalarTraits<HexFormattedString>::output(const HexFormattedString &Value, |
| void *ctx, raw_ostream &Out) { |
| StringRef Bytes(reinterpret_cast<const char *>(Value.Bytes.data()), |
| Value.Bytes.size()); |
| Out << toHex(Bytes); |
| } |
| |
| StringRef ScalarTraits<HexFormattedString>::input(StringRef Scalar, void *ctxt, |
| HexFormattedString &Value) { |
| std::string H = fromHex(Scalar); |
| Value.Bytes.assign(H.begin(), H.end()); |
| return StringRef(); |
| } |
| |
| void MappingTraits<SourceLineEntry>::mapping(IO &IO, SourceLineEntry &Obj) { |
| IO.mapRequired("Offset", Obj.Offset); |
| IO.mapRequired("LineStart", Obj.LineStart); |
| IO.mapRequired("IsStatement", Obj.IsStatement); |
| IO.mapRequired("EndDelta", Obj.EndDelta); |
| } |
| |
| void MappingTraits<SourceColumnEntry>::mapping(IO &IO, SourceColumnEntry &Obj) { |
| IO.mapRequired("StartColumn", Obj.StartColumn); |
| IO.mapRequired("EndColumn", Obj.EndColumn); |
| } |
| |
| void MappingTraits<SourceLineBlock>::mapping(IO &IO, SourceLineBlock &Obj) { |
| IO.mapRequired("FileName", Obj.FileName); |
| IO.mapRequired("Lines", Obj.Lines); |
| IO.mapRequired("Columns", Obj.Columns); |
| } |
| |
| void MappingTraits<CrossModuleExport>::mapping(IO &IO, CrossModuleExport &Obj) { |
| IO.mapRequired("LocalId", Obj.Local); |
| IO.mapRequired("GlobalId", Obj.Global); |
| } |
| |
| void MappingTraits<YAMLCrossModuleImport>::mapping(IO &IO, |
| YAMLCrossModuleImport &Obj) { |
| IO.mapRequired("Module", Obj.ModuleName); |
| IO.mapRequired("Imports", Obj.ImportIds); |
| } |
| |
| void MappingTraits<SourceFileChecksumEntry>::mapping( |
| IO &IO, SourceFileChecksumEntry &Obj) { |
| IO.mapRequired("FileName", Obj.FileName); |
| IO.mapRequired("Kind", Obj.Kind); |
| IO.mapRequired("Checksum", Obj.ChecksumBytes); |
| } |
| |
| void MappingTraits<InlineeSite>::mapping(IO &IO, InlineeSite &Obj) { |
| IO.mapRequired("FileName", Obj.FileName); |
| IO.mapRequired("LineNum", Obj.SourceLineNum); |
| IO.mapRequired("Inlinee", Obj.Inlinee); |
| IO.mapOptional("ExtraFiles", Obj.ExtraFiles); |
| } |
| |
| void MappingTraits<YAMLFrameData>::mapping(IO &IO, YAMLFrameData &Obj) { |
| IO.mapRequired("CodeSize", Obj.CodeSize); |
| IO.mapRequired("FrameFunc", Obj.FrameFunc); |
| IO.mapRequired("LocalSize", Obj.LocalSize); |
| IO.mapOptional("MaxStackSize", Obj.MaxStackSize); |
| IO.mapOptional("ParamsSize", Obj.ParamsSize); |
| IO.mapOptional("PrologSize", Obj.PrologSize); |
| IO.mapOptional("RvaStart", Obj.RvaStart); |
| IO.mapOptional("SavedRegsSize", Obj.SavedRegsSize); |
| } |
| |
| void YAMLChecksumsSubsection::map(IO &IO) { |
| IO.mapTag("!FileChecksums", true); |
| IO.mapRequired("Checksums", Checksums); |
| } |
| |
| void YAMLLinesSubsection::map(IO &IO) { |
| IO.mapTag("!Lines", true); |
| IO.mapRequired("CodeSize", Lines.CodeSize); |
| |
| IO.mapRequired("Flags", Lines.Flags); |
| IO.mapRequired("RelocOffset", Lines.RelocOffset); |
| IO.mapRequired("RelocSegment", Lines.RelocSegment); |
| IO.mapRequired("Blocks", Lines.Blocks); |
| } |
| |
| void YAMLInlineeLinesSubsection::map(IO &IO) { |
| IO.mapTag("!InlineeLines", true); |
| IO.mapRequired("HasExtraFiles", InlineeLines.HasExtraFiles); |
| IO.mapRequired("Sites", InlineeLines.Sites); |
| } |
| |
| void YAMLCrossModuleExportsSubsection::map(IO &IO) { |
| IO.mapTag("!CrossModuleExports", true); |
| IO.mapOptional("Exports", Exports); |
| } |
| |
| void YAMLCrossModuleImportsSubsection::map(IO &IO) { |
| IO.mapTag("!CrossModuleImports", true); |
| IO.mapOptional("Imports", Imports); |
| } |
| |
| void YAMLSymbolsSubsection::map(IO &IO) { |
| IO.mapTag("!Symbols", true); |
| IO.mapRequired("Records", Symbols); |
| } |
| |
| void YAMLStringTableSubsection::map(IO &IO) { |
| IO.mapTag("!StringTable", true); |
| IO.mapRequired("Strings", Strings); |
| } |
| |
| void YAMLFrameDataSubsection::map(IO &IO) { |
| IO.mapTag("!FrameData", true); |
| IO.mapRequired("Frames", Frames); |
| } |
| |
| void YAMLCoffSymbolRVASubsection::map(IO &IO) { |
| IO.mapTag("!COFFSymbolRVAs", true); |
| IO.mapRequired("RVAs", RVAs); |
| } |
| |
| void MappingTraits<YAMLDebugSubsection>::mapping( |
| IO &IO, YAMLDebugSubsection &Subsection) { |
| if (!IO.outputting()) { |
| if (IO.mapTag("!FileChecksums")) { |
| auto SS = std::make_shared<YAMLChecksumsSubsection>(); |
| Subsection.Subsection = SS; |
| } else if (IO.mapTag("!Lines")) { |
| Subsection.Subsection = std::make_shared<YAMLLinesSubsection>(); |
| } else if (IO.mapTag("!InlineeLines")) { |
| Subsection.Subsection = std::make_shared<YAMLInlineeLinesSubsection>(); |
| } else if (IO.mapTag("!CrossModuleExports")) { |
| Subsection.Subsection = |
| std::make_shared<YAMLCrossModuleExportsSubsection>(); |
| } else if (IO.mapTag("!CrossModuleImports")) { |
| Subsection.Subsection = |
| std::make_shared<YAMLCrossModuleImportsSubsection>(); |
| } else if (IO.mapTag("!Symbols")) { |
| Subsection.Subsection = std::make_shared<YAMLSymbolsSubsection>(); |
| } else if (IO.mapTag("!StringTable")) { |
| Subsection.Subsection = std::make_shared<YAMLStringTableSubsection>(); |
| } else if (IO.mapTag("!FrameData")) { |
| Subsection.Subsection = std::make_shared<YAMLFrameDataSubsection>(); |
| } else if (IO.mapTag("!COFFSymbolRVAs")) { |
| Subsection.Subsection = std::make_shared<YAMLCoffSymbolRVASubsection>(); |
| } else { |
| llvm_unreachable("Unexpected subsection tag!"); |
| } |
| } |
| Subsection.Subsection->map(IO); |
| } |
| |
| std::shared_ptr<DebugSubsection> YAMLChecksumsSubsection::toCodeViewSubsection( |
| BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const { |
| assert(SC.hasStrings()); |
| auto Result = std::make_shared<DebugChecksumsSubsection>(*SC.strings()); |
| for (const auto &CS : Checksums) { |
| Result->addChecksum(CS.FileName, CS.Kind, CS.ChecksumBytes.Bytes); |
| } |
| return Result; |
| } |
| |
| std::shared_ptr<DebugSubsection> YAMLLinesSubsection::toCodeViewSubsection( |
| BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const { |
| assert(SC.hasStrings() && SC.hasChecksums()); |
| auto Result = |
| std::make_shared<DebugLinesSubsection>(*SC.checksums(), *SC.strings()); |
| Result->setCodeSize(Lines.CodeSize); |
| Result->setRelocationAddress(Lines.RelocSegment, Lines.RelocOffset); |
| Result->setFlags(Lines.Flags); |
| for (const auto &LC : Lines.Blocks) { |
| Result->createBlock(LC.FileName); |
| if (Result->hasColumnInfo()) { |
| for (const auto &Item : zip(LC.Lines, LC.Columns)) { |
| auto &L = std::get<0>(Item); |
| auto &C = std::get<1>(Item); |
| uint32_t LE = L.LineStart + L.EndDelta; |
| Result->addLineAndColumnInfo(L.Offset, |
| LineInfo(L.LineStart, LE, L.IsStatement), |
| C.StartColumn, C.EndColumn); |
| } |
| } else { |
| for (const auto &L : LC.Lines) { |
| uint32_t LE = L.LineStart + L.EndDelta; |
| Result->addLineInfo(L.Offset, LineInfo(L.LineStart, LE, L.IsStatement)); |
| } |
| } |
| } |
| return Result; |
| } |
| |
| std::shared_ptr<DebugSubsection> |
| YAMLInlineeLinesSubsection::toCodeViewSubsection( |
| BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const { |
| assert(SC.hasChecksums()); |
| auto Result = std::make_shared<DebugInlineeLinesSubsection>( |
| *SC.checksums(), InlineeLines.HasExtraFiles); |
| |
| for (const auto &Site : InlineeLines.Sites) { |
| Result->addInlineSite(TypeIndex(Site.Inlinee), Site.FileName, |
| Site.SourceLineNum); |
| if (!InlineeLines.HasExtraFiles) |
| continue; |
| |
| for (auto EF : Site.ExtraFiles) { |
| Result->addExtraFile(EF); |
| } |
| } |
| return Result; |
| } |
| |
| std::shared_ptr<DebugSubsection> |
| YAMLCrossModuleExportsSubsection::toCodeViewSubsection( |
| BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const { |
| auto Result = std::make_shared<DebugCrossModuleExportsSubsection>(); |
| for (const auto &M : Exports) |
| Result->addMapping(M.Local, M.Global); |
| return Result; |
| } |
| |
| std::shared_ptr<DebugSubsection> |
| YAMLCrossModuleImportsSubsection::toCodeViewSubsection( |
| BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const { |
| assert(SC.hasStrings()); |
| |
| auto Result = |
| std::make_shared<DebugCrossModuleImportsSubsection>(*SC.strings()); |
| for (const auto &M : Imports) { |
| for (const auto Id : M.ImportIds) |
| Result->addImport(M.ModuleName, Id); |
| } |
| return Result; |
| } |
| |
| std::shared_ptr<DebugSubsection> YAMLSymbolsSubsection::toCodeViewSubsection( |
| BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const { |
| auto Result = std::make_shared<DebugSymbolsSubsection>(); |
| for (const auto &Sym : Symbols) |
| Result->addSymbol( |
| Sym.toCodeViewSymbol(Allocator, CodeViewContainer::ObjectFile)); |
| return Result; |
| } |
| |
| std::shared_ptr<DebugSubsection> |
| YAMLStringTableSubsection::toCodeViewSubsection( |
| BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const { |
| auto Result = std::make_shared<DebugStringTableSubsection>(); |
| for (const auto &Str : this->Strings) |
| Result->insert(Str); |
| return Result; |
| } |
| |
| std::shared_ptr<DebugSubsection> YAMLFrameDataSubsection::toCodeViewSubsection( |
| BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const { |
| assert(SC.hasStrings()); |
| |
| auto Result = std::make_shared<DebugFrameDataSubsection>(); |
| for (const auto &YF : Frames) { |
| codeview::FrameData F; |
| F.CodeSize = YF.CodeSize; |
| F.Flags = YF.Flags; |
| F.LocalSize = YF.LocalSize; |
| F.MaxStackSize = YF.MaxStackSize; |
| F.ParamsSize = YF.ParamsSize; |
| F.PrologSize = YF.PrologSize; |
| F.RvaStart = YF.RvaStart; |
| F.SavedRegsSize = YF.SavedRegsSize; |
| F.FrameFunc = SC.strings()->insert(YF.FrameFunc); |
| Result->addFrameData(F); |
| } |
| return Result; |
| } |
| |
| std::shared_ptr<DebugSubsection> |
| YAMLCoffSymbolRVASubsection::toCodeViewSubsection( |
| BumpPtrAllocator &Allocator, |
| const codeview::StringsAndChecksums &SC) const { |
| auto Result = std::make_shared<DebugSymbolRVASubsection>(); |
| for (const auto &RVA : RVAs) |
| Result->addRVA(RVA); |
| return Result; |
| } |
| |
| static Expected<SourceFileChecksumEntry> |
| convertOneChecksum(const DebugStringTableSubsectionRef &Strings, |
| const FileChecksumEntry &CS) { |
| auto ExpectedString = Strings.getString(CS.FileNameOffset); |
| if (!ExpectedString) |
| return ExpectedString.takeError(); |
| |
| SourceFileChecksumEntry Result; |
| Result.ChecksumBytes.Bytes = CS.Checksum; |
| Result.Kind = CS.Kind; |
| Result.FileName = *ExpectedString; |
| return Result; |
| } |
| |
| static Expected<StringRef> |
| getFileName(const DebugStringTableSubsectionRef &Strings, |
| const DebugChecksumsSubsectionRef &Checksums, uint32_t FileID) { |
| auto Iter = Checksums.getArray().at(FileID); |
| if (Iter == Checksums.getArray().end()) |
| return make_error<CodeViewError>(cv_error_code::no_records); |
| uint32_t Offset = Iter->FileNameOffset; |
| return Strings.getString(Offset); |
| } |
| |
| Expected<std::shared_ptr<YAMLChecksumsSubsection>> |
| YAMLChecksumsSubsection::fromCodeViewSubsection( |
| const DebugStringTableSubsectionRef &Strings, |
| const DebugChecksumsSubsectionRef &FC) { |
| auto Result = std::make_shared<YAMLChecksumsSubsection>(); |
| |
| for (const auto &CS : FC) { |
| auto ConvertedCS = convertOneChecksum(Strings, CS); |
| if (!ConvertedCS) |
| return ConvertedCS.takeError(); |
| Result->Checksums.push_back(*ConvertedCS); |
| } |
| return Result; |
| } |
| |
| Expected<std::shared_ptr<YAMLLinesSubsection>> |
| YAMLLinesSubsection::fromCodeViewSubsection( |
| const DebugStringTableSubsectionRef &Strings, |
| const DebugChecksumsSubsectionRef &Checksums, |
| const DebugLinesSubsectionRef &Lines) { |
| auto Result = std::make_shared<YAMLLinesSubsection>(); |
| Result->Lines.CodeSize = Lines.header()->CodeSize; |
| Result->Lines.RelocOffset = Lines.header()->RelocOffset; |
| Result->Lines.RelocSegment = Lines.header()->RelocSegment; |
| Result->Lines.Flags = static_cast<LineFlags>(uint16_t(Lines.header()->Flags)); |
| for (const auto &L : Lines) { |
| SourceLineBlock Block; |
| auto EF = getFileName(Strings, Checksums, L.NameIndex); |
| if (!EF) |
| return EF.takeError(); |
| Block.FileName = *EF; |
| if (Lines.hasColumnInfo()) { |
| for (const auto &C : L.Columns) { |
| SourceColumnEntry SCE; |
| SCE.EndColumn = C.EndColumn; |
| SCE.StartColumn = C.StartColumn; |
| Block.Columns.push_back(SCE); |
| } |
| } |
| for (const auto &LN : L.LineNumbers) { |
| SourceLineEntry SLE; |
| LineInfo LI(LN.Flags); |
| SLE.Offset = LN.Offset; |
| SLE.LineStart = LI.getStartLine(); |
| SLE.EndDelta = LI.getLineDelta(); |
| SLE.IsStatement = LI.isStatement(); |
| Block.Lines.push_back(SLE); |
| } |
| Result->Lines.Blocks.push_back(Block); |
| } |
| return Result; |
| } |
| |
| Expected<std::shared_ptr<YAMLInlineeLinesSubsection>> |
| YAMLInlineeLinesSubsection::fromCodeViewSubsection( |
| const DebugStringTableSubsectionRef &Strings, |
| const DebugChecksumsSubsectionRef &Checksums, |
| const DebugInlineeLinesSubsectionRef &Lines) { |
| auto Result = std::make_shared<YAMLInlineeLinesSubsection>(); |
| |
| Result->InlineeLines.HasExtraFiles = Lines.hasExtraFiles(); |
| for (const auto &IL : Lines) { |
| InlineeSite Site; |
| auto ExpF = getFileName(Strings, Checksums, IL.Header->FileID); |
| if (!ExpF) |
| return ExpF.takeError(); |
| Site.FileName = *ExpF; |
| Site.Inlinee = IL.Header->Inlinee.getIndex(); |
| Site.SourceLineNum = IL.Header->SourceLineNum; |
| if (Lines.hasExtraFiles()) { |
| for (const auto EF : IL.ExtraFiles) { |
| auto ExpF2 = getFileName(Strings, Checksums, EF); |
| if (!ExpF2) |
| return ExpF2.takeError(); |
| Site.ExtraFiles.push_back(*ExpF2); |
| } |
| } |
| Result->InlineeLines.Sites.push_back(Site); |
| } |
| return Result; |
| } |
| |
| Expected<std::shared_ptr<YAMLCrossModuleExportsSubsection>> |
| YAMLCrossModuleExportsSubsection::fromCodeViewSubsection( |
| const DebugCrossModuleExportsSubsectionRef &Exports) { |
| auto Result = std::make_shared<YAMLCrossModuleExportsSubsection>(); |
| Result->Exports.assign(Exports.begin(), Exports.end()); |
| return Result; |
| } |
| |
| Expected<std::shared_ptr<YAMLCrossModuleImportsSubsection>> |
| YAMLCrossModuleImportsSubsection::fromCodeViewSubsection( |
| const DebugStringTableSubsectionRef &Strings, |
| const DebugCrossModuleImportsSubsectionRef &Imports) { |
| auto Result = std::make_shared<YAMLCrossModuleImportsSubsection>(); |
| for (const auto &CMI : Imports) { |
| YAMLCrossModuleImport YCMI; |
| auto ExpectedStr = Strings.getString(CMI.Header->ModuleNameOffset); |
| if (!ExpectedStr) |
| return ExpectedStr.takeError(); |
| YCMI.ModuleName = *ExpectedStr; |
| YCMI.ImportIds.assign(CMI.Imports.begin(), CMI.Imports.end()); |
| Result->Imports.push_back(YCMI); |
| } |
| return Result; |
| } |
| |
| Expected<std::shared_ptr<YAMLSymbolsSubsection>> |
| YAMLSymbolsSubsection::fromCodeViewSubsection( |
| const DebugSymbolsSubsectionRef &Symbols) { |
| auto Result = std::make_shared<YAMLSymbolsSubsection>(); |
| for (const auto &Sym : Symbols) { |
| auto S = CodeViewYAML::SymbolRecord::fromCodeViewSymbol(Sym); |
| if (!S) |
| return joinErrors(make_error<CodeViewError>( |
| cv_error_code::corrupt_record, |
| "Invalid CodeView Symbol Record in SymbolRecord " |
| "subsection of .debug$S while converting to YAML!"), |
| S.takeError()); |
| |
| Result->Symbols.push_back(*S); |
| } |
| return Result; |
| } |
| |
| Expected<std::shared_ptr<YAMLStringTableSubsection>> |
| YAMLStringTableSubsection::fromCodeViewSubsection( |
| const DebugStringTableSubsectionRef &Strings) { |
| auto Result = std::make_shared<YAMLStringTableSubsection>(); |
| BinaryStreamReader Reader(Strings.getBuffer()); |
| StringRef S; |
| // First item is a single null string, skip it. |
| if (auto EC = Reader.readCString(S)) |
| return std::move(EC); |
| assert(S.empty()); |
| while (Reader.bytesRemaining() > 0) { |
| if (auto EC = Reader.readCString(S)) |
| return std::move(EC); |
| Result->Strings.push_back(S); |
| } |
| return Result; |
| } |
| |
| Expected<std::shared_ptr<YAMLFrameDataSubsection>> |
| YAMLFrameDataSubsection::fromCodeViewSubsection( |
| const DebugStringTableSubsectionRef &Strings, |
| const DebugFrameDataSubsectionRef &Frames) { |
| auto Result = std::make_shared<YAMLFrameDataSubsection>(); |
| for (const auto &F : Frames) { |
| YAMLFrameData YF; |
| YF.CodeSize = F.CodeSize; |
| YF.Flags = F.Flags; |
| YF.LocalSize = F.LocalSize; |
| YF.MaxStackSize = F.MaxStackSize; |
| YF.ParamsSize = F.ParamsSize; |
| YF.PrologSize = F.PrologSize; |
| YF.RvaStart = F.RvaStart; |
| YF.SavedRegsSize = F.SavedRegsSize; |
| |
| auto ES = Strings.getString(F.FrameFunc); |
| if (!ES) |
| return joinErrors( |
| make_error<CodeViewError>( |
| cv_error_code::no_records, |
| "Could not find string for string id while mapping FrameData!"), |
| ES.takeError()); |
| YF.FrameFunc = *ES; |
| Result->Frames.push_back(YF); |
| } |
| return Result; |
| } |
| |
| Expected<std::shared_ptr<YAMLCoffSymbolRVASubsection>> |
| YAMLCoffSymbolRVASubsection::fromCodeViewSubsection( |
| const DebugSymbolRVASubsectionRef &Section) { |
| auto Result = std::make_shared<YAMLCoffSymbolRVASubsection>(); |
| for (const auto &RVA : Section) { |
| Result->RVAs.push_back(RVA); |
| } |
| return Result; |
| } |
| |
| Expected<std::vector<std::shared_ptr<DebugSubsection>>> |
| llvm::CodeViewYAML::toCodeViewSubsectionList( |
| BumpPtrAllocator &Allocator, ArrayRef<YAMLDebugSubsection> Subsections, |
| const codeview::StringsAndChecksums &SC) { |
| std::vector<std::shared_ptr<DebugSubsection>> Result; |
| if (Subsections.empty()) |
| return std::move(Result); |
| |
| for (const auto &SS : Subsections) { |
| std::shared_ptr<DebugSubsection> CVS; |
| CVS = SS.Subsection->toCodeViewSubsection(Allocator, SC); |
| assert(CVS != nullptr); |
| Result.push_back(std::move(CVS)); |
| } |
| return std::move(Result); |
| } |
| |
| namespace { |
| |
| struct SubsectionConversionVisitor : public DebugSubsectionVisitor { |
| SubsectionConversionVisitor() = default; |
| |
| Error visitUnknown(DebugUnknownSubsectionRef &Unknown) override; |
| Error visitLines(DebugLinesSubsectionRef &Lines, |
| const StringsAndChecksumsRef &State) override; |
| Error visitFileChecksums(DebugChecksumsSubsectionRef &Checksums, |
| const StringsAndChecksumsRef &State) override; |
| Error visitInlineeLines(DebugInlineeLinesSubsectionRef &Inlinees, |
| const StringsAndChecksumsRef &State) override; |
| Error visitCrossModuleExports(DebugCrossModuleExportsSubsectionRef &Checksums, |
| const StringsAndChecksumsRef &State) override; |
| Error visitCrossModuleImports(DebugCrossModuleImportsSubsectionRef &Inlinees, |
| const StringsAndChecksumsRef &State) override; |
| Error visitStringTable(DebugStringTableSubsectionRef &ST, |
| const StringsAndChecksumsRef &State) override; |
| Error visitSymbols(DebugSymbolsSubsectionRef &Symbols, |
| const StringsAndChecksumsRef &State) override; |
| Error visitFrameData(DebugFrameDataSubsectionRef &Symbols, |
| const StringsAndChecksumsRef &State) override; |
| Error visitCOFFSymbolRVAs(DebugSymbolRVASubsectionRef &Symbols, |
| const StringsAndChecksumsRef &State) override; |
| |
| YAMLDebugSubsection Subsection; |
| }; |
| |
| } // end anonymous namespace |
| |
| Error SubsectionConversionVisitor::visitUnknown( |
| DebugUnknownSubsectionRef &Unknown) { |
| return make_error<CodeViewError>(cv_error_code::operation_unsupported); |
| } |
| |
| Error SubsectionConversionVisitor::visitLines( |
| DebugLinesSubsectionRef &Lines, const StringsAndChecksumsRef &State) { |
| auto Result = YAMLLinesSubsection::fromCodeViewSubsection( |
| State.strings(), State.checksums(), Lines); |
| if (!Result) |
| return Result.takeError(); |
| Subsection.Subsection = *Result; |
| return Error::success(); |
| } |
| |
| Error SubsectionConversionVisitor::visitFileChecksums( |
| DebugChecksumsSubsectionRef &Checksums, |
| const StringsAndChecksumsRef &State) { |
| auto Result = YAMLChecksumsSubsection::fromCodeViewSubsection(State.strings(), |
| Checksums); |
| if (!Result) |
| return Result.takeError(); |
| Subsection.Subsection = *Result; |
| return Error::success(); |
| } |
| |
| Error SubsectionConversionVisitor::visitInlineeLines( |
| DebugInlineeLinesSubsectionRef &Inlinees, |
| const StringsAndChecksumsRef &State) { |
| auto Result = YAMLInlineeLinesSubsection::fromCodeViewSubsection( |
| State.strings(), State.checksums(), Inlinees); |
| if (!Result) |
| return Result.takeError(); |
| Subsection.Subsection = *Result; |
| return Error::success(); |
| } |
| |
| Error SubsectionConversionVisitor::visitCrossModuleExports( |
| DebugCrossModuleExportsSubsectionRef &Exports, |
| const StringsAndChecksumsRef &State) { |
| auto Result = |
| YAMLCrossModuleExportsSubsection::fromCodeViewSubsection(Exports); |
| if (!Result) |
| return Result.takeError(); |
| Subsection.Subsection = *Result; |
| return Error::success(); |
| } |
| |
| Error SubsectionConversionVisitor::visitCrossModuleImports( |
| DebugCrossModuleImportsSubsectionRef &Imports, |
| const StringsAndChecksumsRef &State) { |
| auto Result = YAMLCrossModuleImportsSubsection::fromCodeViewSubsection( |
| State.strings(), Imports); |
| if (!Result) |
| return Result.takeError(); |
| Subsection.Subsection = *Result; |
| return Error::success(); |
| } |
| |
| Error SubsectionConversionVisitor::visitStringTable( |
| DebugStringTableSubsectionRef &Strings, |
| const StringsAndChecksumsRef &State) { |
| auto Result = YAMLStringTableSubsection::fromCodeViewSubsection(Strings); |
| if (!Result) |
| return Result.takeError(); |
| Subsection.Subsection = *Result; |
| return Error::success(); |
| } |
| |
| Error SubsectionConversionVisitor::visitSymbols( |
| DebugSymbolsSubsectionRef &Symbols, const StringsAndChecksumsRef &State) { |
| auto Result = YAMLSymbolsSubsection::fromCodeViewSubsection(Symbols); |
| if (!Result) |
| return Result.takeError(); |
| Subsection.Subsection = *Result; |
| return Error::success(); |
| } |
| |
| Error SubsectionConversionVisitor::visitFrameData( |
| DebugFrameDataSubsectionRef &Frames, const StringsAndChecksumsRef &State) { |
| auto Result = |
| YAMLFrameDataSubsection::fromCodeViewSubsection(State.strings(), Frames); |
| if (!Result) |
| return Result.takeError(); |
| Subsection.Subsection = *Result; |
| return Error::success(); |
| } |
| |
| Error SubsectionConversionVisitor::visitCOFFSymbolRVAs( |
| DebugSymbolRVASubsectionRef &RVAs, const StringsAndChecksumsRef &State) { |
| auto Result = YAMLCoffSymbolRVASubsection::fromCodeViewSubsection(RVAs); |
| if (!Result) |
| return Result.takeError(); |
| Subsection.Subsection = *Result; |
| return Error::success(); |
| } |
| |
| Expected<YAMLDebugSubsection> |
| YAMLDebugSubsection::fromCodeViewSubection(const StringsAndChecksumsRef &SC, |
| const DebugSubsectionRecord &SS) { |
| SubsectionConversionVisitor V; |
| if (auto EC = visitDebugSubsection(SS, V, SC)) |
| return std::move(EC); |
| |
| return V.Subsection; |
| } |
| |
| std::vector<YAMLDebugSubsection> |
| llvm::CodeViewYAML::fromDebugS(ArrayRef<uint8_t> Data, |
| const StringsAndChecksumsRef &SC) { |
| BinaryStreamReader Reader(Data, support::little); |
| uint32_t Magic; |
| |
| ExitOnError Err("Invalid .debug$S section!"); |
| Err(Reader.readInteger(Magic)); |
| assert(Magic == COFF::DEBUG_SECTION_MAGIC && "Invalid .debug$S section!"); |
| |
| DebugSubsectionArray Subsections; |
| Err(Reader.readArray(Subsections, Reader.bytesRemaining())); |
| |
| std::vector<YAMLDebugSubsection> Result; |
| |
| for (const auto &SS : Subsections) { |
| auto YamlSS = Err(YAMLDebugSubsection::fromCodeViewSubection(SC, SS)); |
| Result.push_back(YamlSS); |
| } |
| return Result; |
| } |
| |
| void llvm::CodeViewYAML::initializeStringsAndChecksums( |
| ArrayRef<YAMLDebugSubsection> Sections, codeview::StringsAndChecksums &SC) { |
| // String Table and Checksums subsections don't use the allocator. |
| BumpPtrAllocator Allocator; |
| |
| // It's possible for checksums and strings to even appear in different debug$S |
| // sections, so we have to make this a stateful function that can build up |
| // the strings and checksums field over multiple iterations. |
| |
| // File Checksums require the string table, but may become before it, so we |
| // have to scan for strings first, then scan for checksums again from the |
| // beginning. |
| if (!SC.hasStrings()) { |
| for (const auto &SS : Sections) { |
| if (SS.Subsection->Kind != DebugSubsectionKind::StringTable) |
| continue; |
| |
| auto Result = SS.Subsection->toCodeViewSubsection(Allocator, SC); |
| SC.setStrings( |
| std::static_pointer_cast<DebugStringTableSubsection>(Result)); |
| break; |
| } |
| } |
| |
| if (SC.hasStrings() && !SC.hasChecksums()) { |
| for (const auto &SS : Sections) { |
| if (SS.Subsection->Kind != DebugSubsectionKind::FileChecksums) |
| continue; |
| |
| auto Result = SS.Subsection->toCodeViewSubsection(Allocator, SC); |
| SC.setChecksums( |
| std::static_pointer_cast<DebugChecksumsSubsection>(Result)); |
| break; |
| } |
| } |
| } |