167c56014SZachary Turner //===- ModuleDebugStream.cpp - PDB Module Info Stream Access --------------===// 267c56014SZachary Turner // 367c56014SZachary Turner // The LLVM Compiler Infrastructure 467c56014SZachary Turner // 567c56014SZachary Turner // This file is distributed under the University of Illinois Open Source 667c56014SZachary Turner // License. See LICENSE.TXT for details. 767c56014SZachary Turner // 867c56014SZachary Turner //===----------------------------------------------------------------------===// 967c56014SZachary Turner 1067c56014SZachary Turner #include "llvm/DebugInfo/PDB/Native/ModuleDebugStream.h" 1167c56014SZachary Turner #include "llvm/ADT/iterator_range.h" 124fcfc199SEugene Zelenko #include "llvm/DebugInfo/CodeView/CodeView.h" 134fcfc199SEugene Zelenko #include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h" 14*bb3d7e56SZachary Turner #include "llvm/DebugInfo/CodeView/SymbolDeserializer.h" 1567c56014SZachary Turner #include "llvm/DebugInfo/CodeView/SymbolRecord.h" 16*bb3d7e56SZachary Turner #include "llvm/DebugInfo/CodeView/SymbolRecordHelpers.h" 1767c56014SZachary Turner #include "llvm/DebugInfo/PDB/Native/DbiModuleDescriptor.h" 1867c56014SZachary Turner #include "llvm/DebugInfo/PDB/Native/RawError.h" 1967c56014SZachary Turner #include "llvm/Support/BinaryStreamReader.h" 2067c56014SZachary Turner #include "llvm/Support/BinaryStreamRef.h" 2167c56014SZachary Turner #include "llvm/Support/Error.h" 2267c56014SZachary Turner #include <algorithm> 2367c56014SZachary Turner #include <cstdint> 2467c56014SZachary Turner 2567c56014SZachary Turner using namespace llvm; 26c37cb0c6SZachary Turner using namespace llvm::codeview; 2767c56014SZachary Turner using namespace llvm::msf; 2867c56014SZachary Turner using namespace llvm::pdb; 2967c56014SZachary Turner 307cc13e55SZachary Turner ModuleDebugStreamRef::ModuleDebugStreamRef( 317cc13e55SZachary Turner const DbiModuleDescriptor &Module, 3267c56014SZachary Turner std::unique_ptr<MappedBlockStream> Stream) 3367c56014SZachary Turner : Mod(Module), Stream(std::move(Stream)) {} 3467c56014SZachary Turner 357cc13e55SZachary Turner ModuleDebugStreamRef::~ModuleDebugStreamRef() = default; 3667c56014SZachary Turner 377cc13e55SZachary Turner Error ModuleDebugStreamRef::reload() { 3867c56014SZachary Turner BinaryStreamReader Reader(*Stream); 3967c56014SZachary Turner 4067c56014SZachary Turner uint32_t SymbolSize = Mod.getSymbolDebugInfoByteSize(); 415b6e4e0aSZachary Turner uint32_t C11Size = Mod.getC11LineInfoByteSize(); 4267c56014SZachary Turner uint32_t C13Size = Mod.getC13LineInfoByteSize(); 4367c56014SZachary Turner 4467c56014SZachary Turner if (C11Size > 0 && C13Size > 0) 4567c56014SZachary Turner return make_error<RawError>(raw_error_code::corrupt_file, 4667c56014SZachary Turner "Module has both C11 and C13 line info"); 4767c56014SZachary Turner 4867c56014SZachary Turner BinaryStreamRef S; 4967c56014SZachary Turner 5067c56014SZachary Turner if (auto EC = Reader.readInteger(Signature)) 5167c56014SZachary Turner return EC; 52579264bdSZachary Turner Reader.setOffset(0); 53579264bdSZachary Turner if (auto EC = Reader.readSubstream(SymbolsSubstream, SymbolSize)) 54fa332827SZachary Turner return EC; 55fa332827SZachary Turner if (auto EC = Reader.readSubstream(C11LinesSubstream, C11Size)) 56fa332827SZachary Turner return EC; 57fa332827SZachary Turner if (auto EC = Reader.readSubstream(C13LinesSubstream, C13Size)) 5867c56014SZachary Turner return EC; 5967c56014SZachary Turner 60fa332827SZachary Turner BinaryStreamReader SymbolReader(SymbolsSubstream.StreamData); 61579264bdSZachary Turner if (auto EC = SymbolReader.readArray( 62579264bdSZachary Turner SymbolArray, SymbolReader.bytesRemaining(), sizeof(uint32_t))) 6367c56014SZachary Turner return EC; 6467c56014SZachary Turner 65fa332827SZachary Turner BinaryStreamReader SubsectionsReader(C13LinesSubstream.StreamData); 6692dcdda6SZachary Turner if (auto EC = SubsectionsReader.readArray(Subsections, 6792dcdda6SZachary Turner SubsectionsReader.bytesRemaining())) 6867c56014SZachary Turner return EC; 6967c56014SZachary Turner 7067c56014SZachary Turner uint32_t GlobalRefsSize; 7167c56014SZachary Turner if (auto EC = Reader.readInteger(GlobalRefsSize)) 7267c56014SZachary Turner return EC; 73fa332827SZachary Turner if (auto EC = Reader.readSubstream(GlobalRefsSubstream, GlobalRefsSize)) 7467c56014SZachary Turner return EC; 7567c56014SZachary Turner if (Reader.bytesRemaining() > 0) 7667c56014SZachary Turner return make_error<RawError>(raw_error_code::corrupt_file, 7767c56014SZachary Turner "Unexpected bytes in module stream."); 7867c56014SZachary Turner 7967c56014SZachary Turner return Error::success(); 8067c56014SZachary Turner } 8167c56014SZachary Turner 82*bb3d7e56SZachary Turner const codeview::CVSymbolArray 83*bb3d7e56SZachary Turner ModuleDebugStreamRef::getSymbolArrayForScope(uint32_t ScopeBegin) const { 84*bb3d7e56SZachary Turner return limitSymbolArrayToScope(SymbolArray, ScopeBegin); 85*bb3d7e56SZachary Turner } 86*bb3d7e56SZachary Turner 87fa332827SZachary Turner BinarySubstreamRef ModuleDebugStreamRef::getSymbolsSubstream() const { 88fa332827SZachary Turner return SymbolsSubstream; 89fa332827SZachary Turner } 90fa332827SZachary Turner 91fa332827SZachary Turner BinarySubstreamRef ModuleDebugStreamRef::getC11LinesSubstream() const { 92fa332827SZachary Turner return C11LinesSubstream; 93fa332827SZachary Turner } 94fa332827SZachary Turner 95fa332827SZachary Turner BinarySubstreamRef ModuleDebugStreamRef::getC13LinesSubstream() const { 96fa332827SZachary Turner return C13LinesSubstream; 97fa332827SZachary Turner } 98fa332827SZachary Turner 99fa332827SZachary Turner BinarySubstreamRef ModuleDebugStreamRef::getGlobalRefsSubstream() const { 100fa332827SZachary Turner return GlobalRefsSubstream; 101fa332827SZachary Turner } 102fa332827SZachary Turner 10367c56014SZachary Turner iterator_range<codeview::CVSymbolArray::Iterator> 1047cc13e55SZachary Turner ModuleDebugStreamRef::symbols(bool *HadError) const { 105fa332827SZachary Turner return make_range(SymbolArray.begin(HadError), SymbolArray.end()); 10667c56014SZachary Turner } 10767c56014SZachary Turner 10894926a6dSZachary Turner CVSymbol ModuleDebugStreamRef::readSymbolAtOffset(uint32_t Offset) const { 109579264bdSZachary Turner auto Iter = SymbolArray.at(Offset); 11094926a6dSZachary Turner assert(Iter != SymbolArray.end()); 11194926a6dSZachary Turner return *Iter; 11294926a6dSZachary Turner } 11394926a6dSZachary Turner 1144fcfc199SEugene Zelenko iterator_range<ModuleDebugStreamRef::DebugSubsectionIterator> 11592dcdda6SZachary Turner ModuleDebugStreamRef::subsections() const { 11692dcdda6SZachary Turner return make_range(Subsections.begin(), Subsections.end()); 11767c56014SZachary Turner } 11867c56014SZachary Turner 11992dcdda6SZachary Turner bool ModuleDebugStreamRef::hasDebugSubsections() const { 120fa332827SZachary Turner return !C13LinesSubstream.empty(); 12167c56014SZachary Turner } 12267c56014SZachary Turner 1237cc13e55SZachary Turner Error ModuleDebugStreamRef::commit() { return Error::success(); } 12492dcdda6SZachary Turner 12592dcdda6SZachary Turner Expected<codeview::DebugChecksumsSubsectionRef> 12692dcdda6SZachary Turner ModuleDebugStreamRef::findChecksumsSubsection() const { 127349c18f8SZachary Turner codeview::DebugChecksumsSubsectionRef Result; 12892dcdda6SZachary Turner for (const auto &SS : subsections()) { 12992dcdda6SZachary Turner if (SS.kind() != DebugSubsectionKind::FileChecksums) 13092dcdda6SZachary Turner continue; 13192dcdda6SZachary Turner 13292dcdda6SZachary Turner if (auto EC = Result.initialize(SS.getRecordData())) 13392dcdda6SZachary Turner return std::move(EC); 13492dcdda6SZachary Turner return Result; 13592dcdda6SZachary Turner } 136349c18f8SZachary Turner return Result; 13792dcdda6SZachary Turner } 138