1f37b6182SDimitry Andric //===- ModuleDebugStream.cpp - PDB Module Info Stream Access --------------===//
2f37b6182SDimitry Andric //
3f37b6182SDimitry Andric //                     The LLVM Compiler Infrastructure
4f37b6182SDimitry Andric //
5f37b6182SDimitry Andric // This file is distributed under the University of Illinois Open Source
6f37b6182SDimitry Andric // License. See LICENSE.TXT for details.
7f37b6182SDimitry Andric //
8f37b6182SDimitry Andric //===----------------------------------------------------------------------===//
9f37b6182SDimitry Andric 
10f37b6182SDimitry Andric #include "llvm/DebugInfo/PDB/Native/ModuleDebugStream.h"
11f37b6182SDimitry Andric #include "llvm/ADT/iterator_range.h"
12a580b014SDimitry Andric #include "llvm/DebugInfo/CodeView/CodeView.h"
13a580b014SDimitry Andric #include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h"
14*b5893f02SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
15f37b6182SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolRecord.h"
16*b5893f02SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolRecordHelpers.h"
17f37b6182SDimitry Andric #include "llvm/DebugInfo/PDB/Native/DbiModuleDescriptor.h"
18f37b6182SDimitry Andric #include "llvm/DebugInfo/PDB/Native/RawError.h"
19f37b6182SDimitry Andric #include "llvm/Support/BinaryStreamReader.h"
20f37b6182SDimitry Andric #include "llvm/Support/BinaryStreamRef.h"
21f37b6182SDimitry Andric #include "llvm/Support/Error.h"
22f37b6182SDimitry Andric #include <algorithm>
23f37b6182SDimitry Andric #include <cstdint>
24f37b6182SDimitry Andric 
25f37b6182SDimitry Andric using namespace llvm;
26f37b6182SDimitry Andric using namespace llvm::codeview;
27f37b6182SDimitry Andric using namespace llvm::msf;
28f37b6182SDimitry Andric using namespace llvm::pdb;
29f37b6182SDimitry Andric 
ModuleDebugStreamRef(const DbiModuleDescriptor & Module,std::unique_ptr<MappedBlockStream> Stream)30f37b6182SDimitry Andric ModuleDebugStreamRef::ModuleDebugStreamRef(
31f37b6182SDimitry Andric     const DbiModuleDescriptor &Module,
32f37b6182SDimitry Andric     std::unique_ptr<MappedBlockStream> Stream)
33f37b6182SDimitry Andric     : Mod(Module), Stream(std::move(Stream)) {}
34f37b6182SDimitry Andric 
35f37b6182SDimitry Andric ModuleDebugStreamRef::~ModuleDebugStreamRef() = default;
36f37b6182SDimitry Andric 
reload()37f37b6182SDimitry Andric Error ModuleDebugStreamRef::reload() {
38f37b6182SDimitry Andric   BinaryStreamReader Reader(*Stream);
39f37b6182SDimitry Andric 
40f37b6182SDimitry Andric   uint32_t SymbolSize = Mod.getSymbolDebugInfoByteSize();
41f37b6182SDimitry Andric   uint32_t C11Size = Mod.getC11LineInfoByteSize();
42f37b6182SDimitry Andric   uint32_t C13Size = Mod.getC13LineInfoByteSize();
43f37b6182SDimitry Andric 
44f37b6182SDimitry Andric   if (C11Size > 0 && C13Size > 0)
45f37b6182SDimitry Andric     return make_error<RawError>(raw_error_code::corrupt_file,
46f37b6182SDimitry Andric                                 "Module has both C11 and C13 line info");
47f37b6182SDimitry Andric 
48f37b6182SDimitry Andric   BinaryStreamRef S;
49f37b6182SDimitry Andric 
50f37b6182SDimitry Andric   if (auto EC = Reader.readInteger(Signature))
51f37b6182SDimitry Andric     return EC;
52*b5893f02SDimitry Andric   Reader.setOffset(0);
53*b5893f02SDimitry Andric   if (auto EC = Reader.readSubstream(SymbolsSubstream, SymbolSize))
54edd7eaddSDimitry Andric     return EC;
55edd7eaddSDimitry Andric   if (auto EC = Reader.readSubstream(C11LinesSubstream, C11Size))
56edd7eaddSDimitry Andric     return EC;
57edd7eaddSDimitry Andric   if (auto EC = Reader.readSubstream(C13LinesSubstream, C13Size))
58f37b6182SDimitry Andric     return EC;
59f37b6182SDimitry Andric 
60edd7eaddSDimitry Andric   BinaryStreamReader SymbolReader(SymbolsSubstream.StreamData);
61*b5893f02SDimitry Andric   if (auto EC = SymbolReader.readArray(
62*b5893f02SDimitry Andric           SymbolArray, SymbolReader.bytesRemaining(), sizeof(uint32_t)))
63f37b6182SDimitry Andric     return EC;
64f37b6182SDimitry Andric 
65edd7eaddSDimitry Andric   BinaryStreamReader SubsectionsReader(C13LinesSubstream.StreamData);
666d97bb29SDimitry Andric   if (auto EC = SubsectionsReader.readArray(Subsections,
676d97bb29SDimitry Andric                                             SubsectionsReader.bytesRemaining()))
68f37b6182SDimitry Andric     return EC;
69f37b6182SDimitry Andric 
70f37b6182SDimitry Andric   uint32_t GlobalRefsSize;
71f37b6182SDimitry Andric   if (auto EC = Reader.readInteger(GlobalRefsSize))
72f37b6182SDimitry Andric     return EC;
73edd7eaddSDimitry Andric   if (auto EC = Reader.readSubstream(GlobalRefsSubstream, GlobalRefsSize))
74f37b6182SDimitry Andric     return EC;
75f37b6182SDimitry Andric   if (Reader.bytesRemaining() > 0)
76f37b6182SDimitry Andric     return make_error<RawError>(raw_error_code::corrupt_file,
77f37b6182SDimitry Andric                                 "Unexpected bytes in module stream.");
78f37b6182SDimitry Andric 
79f37b6182SDimitry Andric   return Error::success();
80f37b6182SDimitry Andric }
81f37b6182SDimitry Andric 
82*b5893f02SDimitry Andric const codeview::CVSymbolArray
getSymbolArrayForScope(uint32_t ScopeBegin) const83*b5893f02SDimitry Andric ModuleDebugStreamRef::getSymbolArrayForScope(uint32_t ScopeBegin) const {
84*b5893f02SDimitry Andric   return limitSymbolArrayToScope(SymbolArray, ScopeBegin);
85*b5893f02SDimitry Andric }
86*b5893f02SDimitry Andric 
getSymbolsSubstream() const87edd7eaddSDimitry Andric BinarySubstreamRef ModuleDebugStreamRef::getSymbolsSubstream() const {
88edd7eaddSDimitry Andric   return SymbolsSubstream;
89edd7eaddSDimitry Andric }
90edd7eaddSDimitry Andric 
getC11LinesSubstream() const91edd7eaddSDimitry Andric BinarySubstreamRef ModuleDebugStreamRef::getC11LinesSubstream() const {
92edd7eaddSDimitry Andric   return C11LinesSubstream;
93edd7eaddSDimitry Andric }
94edd7eaddSDimitry Andric 
getC13LinesSubstream() const95edd7eaddSDimitry Andric BinarySubstreamRef ModuleDebugStreamRef::getC13LinesSubstream() const {
96edd7eaddSDimitry Andric   return C13LinesSubstream;
97edd7eaddSDimitry Andric }
98edd7eaddSDimitry Andric 
getGlobalRefsSubstream() const99edd7eaddSDimitry Andric BinarySubstreamRef ModuleDebugStreamRef::getGlobalRefsSubstream() const {
100edd7eaddSDimitry Andric   return GlobalRefsSubstream;
101edd7eaddSDimitry Andric }
102edd7eaddSDimitry Andric 
103f37b6182SDimitry Andric iterator_range<codeview::CVSymbolArray::Iterator>
symbols(bool * HadError) const104f37b6182SDimitry Andric ModuleDebugStreamRef::symbols(bool *HadError) const {
105edd7eaddSDimitry Andric   return make_range(SymbolArray.begin(HadError), SymbolArray.end());
106f37b6182SDimitry Andric }
107f37b6182SDimitry Andric 
readSymbolAtOffset(uint32_t Offset) const108*b5893f02SDimitry Andric CVSymbol ModuleDebugStreamRef::readSymbolAtOffset(uint32_t Offset) const {
109*b5893f02SDimitry Andric   auto Iter = SymbolArray.at(Offset);
110*b5893f02SDimitry Andric   assert(Iter != SymbolArray.end());
111*b5893f02SDimitry Andric   return *Iter;
112*b5893f02SDimitry Andric }
113*b5893f02SDimitry Andric 
114a580b014SDimitry Andric iterator_range<ModuleDebugStreamRef::DebugSubsectionIterator>
subsections() const1156d97bb29SDimitry Andric ModuleDebugStreamRef::subsections() const {
1166d97bb29SDimitry Andric   return make_range(Subsections.begin(), Subsections.end());
117f37b6182SDimitry Andric }
118f37b6182SDimitry Andric 
hasDebugSubsections() const1196d97bb29SDimitry Andric bool ModuleDebugStreamRef::hasDebugSubsections() const {
120edd7eaddSDimitry Andric   return !C13LinesSubstream.empty();
121f37b6182SDimitry Andric }
122f37b6182SDimitry Andric 
commit()123f37b6182SDimitry Andric Error ModuleDebugStreamRef::commit() { return Error::success(); }
1246d97bb29SDimitry Andric 
1256d97bb29SDimitry Andric Expected<codeview::DebugChecksumsSubsectionRef>
findChecksumsSubsection() const1266d97bb29SDimitry Andric ModuleDebugStreamRef::findChecksumsSubsection() const {
127db17bf38SDimitry Andric   codeview::DebugChecksumsSubsectionRef Result;
1286d97bb29SDimitry Andric   for (const auto &SS : subsections()) {
1296d97bb29SDimitry Andric     if (SS.kind() != DebugSubsectionKind::FileChecksums)
1306d97bb29SDimitry Andric       continue;
1316d97bb29SDimitry Andric 
1326d97bb29SDimitry Andric     if (auto EC = Result.initialize(SS.getRecordData()))
1336d97bb29SDimitry Andric       return std::move(EC);
1346d97bb29SDimitry Andric     return Result;
1356d97bb29SDimitry Andric   }
136db17bf38SDimitry Andric   return Result;
1376d97bb29SDimitry Andric }
138