blob: f3de40811b170288a90dfe4e73289e0270ad83ba [file] [log] [blame]
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +00001//===- NativeSession.cpp - Native implementation of IPDBSession -*- C++ -*-===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9
10#include "llvm/DebugInfo/PDB/Native/NativeSession.h"
11
12#include "llvm/ADT/STLExtras.h"
Adrian McCarthy8d090fc2017-07-12 19:38:11 +000013#include "llvm/DebugInfo/CodeView/TypeIndex.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000014#include "llvm/DebugInfo/PDB/IPDBEnumChildren.h"
15#include "llvm/DebugInfo/PDB/IPDBSourceFile.h"
Adrian McCarthy8d090fc2017-07-12 19:38:11 +000016#include "llvm/DebugInfo/PDB/Native/NativeBuiltinSymbol.h"
Adrian McCarthybf0afc32017-06-28 22:47:40 +000017#include "llvm/DebugInfo/PDB/Native/NativeCompilandSymbol.h"
Adrian McCarthyb41f03e2017-08-04 22:37:58 +000018#include "llvm/DebugInfo/PDB/Native/NativeEnumSymbol.h"
19#include "llvm/DebugInfo/PDB/Native/NativeEnumTypes.h"
Adrian McCarthy4d93d662017-03-29 19:27:08 +000020#include "llvm/DebugInfo/PDB/Native/NativeExeSymbol.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000021#include "llvm/DebugInfo/PDB/Native/PDBFile.h"
22#include "llvm/DebugInfo/PDB/Native/RawError.h"
Adrian McCarthyb41f03e2017-08-04 22:37:58 +000023#include "llvm/DebugInfo/PDB/Native/TpiStream.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000024#include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h"
25#include "llvm/DebugInfo/PDB/PDBSymbolExe.h"
Adrian McCarthyb41f03e2017-08-04 22:37:58 +000026#include "llvm/DebugInfo/PDB/PDBSymbolTypeEnum.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000027#include "llvm/Support/Allocator.h"
Zachary Turnerd9dc2822017-03-02 20:52:51 +000028#include "llvm/Support/BinaryByteStream.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000029#include "llvm/Support/Error.h"
30#include "llvm/Support/ErrorOr.h"
31#include "llvm/Support/MemoryBuffer.h"
Adrian McCarthybf0afc32017-06-28 22:47:40 +000032
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000033#include <algorithm>
Adrian McCarthyb41f03e2017-08-04 22:37:58 +000034#include <cassert>
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000035#include <memory>
Adrian McCarthybf0afc32017-06-28 22:47:40 +000036#include <utility>
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000037
38using namespace llvm;
39using namespace llvm::msf;
40using namespace llvm::pdb;
41
Adrian McCarthy8d090fc2017-07-12 19:38:11 +000042namespace {
43// Maps codeview::SimpleTypeKind of a built-in type to the parameters necessary
44// to instantiate a NativeBuiltinSymbol for that type.
Reid Kleckner0962cb22017-07-12 19:46:35 +000045static const struct BuiltinTypeEntry {
Adrian McCarthy8d090fc2017-07-12 19:38:11 +000046 codeview::SimpleTypeKind Kind;
47 PDB_BuiltinType Type;
48 uint32_t Size;
49} BuiltinTypes[] = {
50 {codeview::SimpleTypeKind::Int32, PDB_BuiltinType::Int, 4},
51 {codeview::SimpleTypeKind::UInt32, PDB_BuiltinType::UInt, 4},
52 {codeview::SimpleTypeKind::UInt32Long, PDB_BuiltinType::UInt, 4},
53 {codeview::SimpleTypeKind::UInt64Quad, PDB_BuiltinType::UInt, 8},
54 {codeview::SimpleTypeKind::NarrowCharacter, PDB_BuiltinType::Char, 1},
55 {codeview::SimpleTypeKind::SignedCharacter, PDB_BuiltinType::Char, 1},
56 {codeview::SimpleTypeKind::UnsignedCharacter, PDB_BuiltinType::UInt, 1},
57 {codeview::SimpleTypeKind::UInt16Short, PDB_BuiltinType::UInt, 2},
58 {codeview::SimpleTypeKind::Boolean8, PDB_BuiltinType::Bool, 1}
59 // This table can be grown as necessary, but these are the only types we've
60 // needed so far.
61};
62} // namespace
63
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000064NativeSession::NativeSession(std::unique_ptr<PDBFile> PdbFile,
65 std::unique_ptr<BumpPtrAllocator> Allocator)
Zachary Turner7999b4f2018-09-05 23:30:38 +000066 : Pdb(std::move(PdbFile)), Allocator(std::move(Allocator)) {
67 // Id 0 is reserved for the invalid symbol.
68 SymbolCache.push_back(nullptr);
69}
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000070
71NativeSession::~NativeSession() = default;
72
Peter Collingbourne75257bc2017-10-20 19:48:26 +000073Error NativeSession::createFromPdb(std::unique_ptr<MemoryBuffer> Buffer,
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000074 std::unique_ptr<IPDBSession> &Session) {
Peter Collingbourne75257bc2017-10-20 19:48:26 +000075 StringRef Path = Buffer->getBufferIdentifier();
Zachary Turner695ed562017-02-28 00:04:07 +000076 auto Stream = llvm::make_unique<MemoryBufferByteStream>(
77 std::move(Buffer), llvm::support::little);
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000078
79 auto Allocator = llvm::make_unique<BumpPtrAllocator>();
Zachary Turner7b327d02017-02-16 23:35:45 +000080 auto File = llvm::make_unique<PDBFile>(Path, std::move(Stream), *Allocator);
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000081 if (auto EC = File->parseFileHeaders())
82 return EC;
83 if (auto EC = File->parseStreamData())
84 return EC;
85
86 Session =
87 llvm::make_unique<NativeSession>(std::move(File), std::move(Allocator));
88
89 return Error::success();
90}
91
92Error NativeSession::createFromExe(StringRef Path,
93 std::unique_ptr<IPDBSession> &Session) {
94 return make_error<RawError>(raw_error_code::feature_unsupported);
95}
96
Adrian McCarthyb41f03e2017-08-04 22:37:58 +000097std::unique_ptr<PDBSymbolTypeEnum>
98NativeSession::createEnumSymbol(codeview::TypeIndex Index) {
99 const auto Id = findSymbolByTypeIndex(Index);
Zachary Turner7999b4f2018-09-05 23:30:38 +0000100 return PDBSymbol::createAs<PDBSymbolTypeEnum>(*this, *SymbolCache[Id]);
Adrian McCarthyb41f03e2017-08-04 22:37:58 +0000101}
102
103std::unique_ptr<IPDBEnumSymbols>
104NativeSession::createTypeEnumerator(codeview::TypeLeafKind Kind) {
105 auto Tpi = Pdb->getPDBTpiStream();
106 if (!Tpi) {
107 consumeError(Tpi.takeError());
108 return nullptr;
109 }
110 auto &Types = Tpi->typeCollection();
111 return std::unique_ptr<IPDBEnumSymbols>(
112 new NativeEnumTypes(*this, Types, codeview::LF_ENUM));
113}
114
Adrian McCarthy8d090fc2017-07-12 19:38:11 +0000115SymIndexId NativeSession::findSymbolByTypeIndex(codeview::TypeIndex Index) {
116 // First see if it's already in our cache.
117 const auto Entry = TypeIndexToSymbolId.find(Index);
118 if (Entry != TypeIndexToSymbolId.end())
119 return Entry->second;
120
121 // Symbols for built-in types are created on the fly.
122 if (Index.isSimple()) {
123 // FIXME: We will eventually need to handle pointers to other simple types,
124 // which are still simple types in the world of CodeView TypeIndexes.
125 if (Index.getSimpleMode() != codeview::SimpleTypeMode::Direct)
126 return 0;
127 const auto Kind = Index.getSimpleKind();
Reid Kleckner0962cb22017-07-12 19:46:35 +0000128 const auto It =
129 std::find_if(std::begin(BuiltinTypes), std::end(BuiltinTypes),
130 [Kind](const BuiltinTypeEntry &Builtin) {
131 return Builtin.Kind == Kind;
132 });
Adrian McCarthy8d090fc2017-07-12 19:38:11 +0000133 if (It == std::end(BuiltinTypes))
134 return 0;
135 SymIndexId Id = SymbolCache.size();
136 SymbolCache.emplace_back(
Reid Kleckner0962cb22017-07-12 19:46:35 +0000137 llvm::make_unique<NativeBuiltinSymbol>(*this, Id, It->Type, It->Size));
Adrian McCarthy8d090fc2017-07-12 19:38:11 +0000138 TypeIndexToSymbolId[Index] = Id;
139 return Id;
140 }
141
Adrian McCarthyb41f03e2017-08-04 22:37:58 +0000142 // We need to instantiate and cache the desired type symbol.
143 auto Tpi = Pdb->getPDBTpiStream();
144 if (!Tpi) {
145 consumeError(Tpi.takeError());
146 return 0;
147 }
148 auto &Types = Tpi->typeCollection();
149 const auto &I = Types.getType(Index);
150 const auto Id = static_cast<SymIndexId>(SymbolCache.size());
151 // TODO(amccarth): Make this handle all types, not just LF_ENUMs.
152 assert(I.kind() == codeview::LF_ENUM);
153 SymbolCache.emplace_back(llvm::make_unique<NativeEnumSymbol>(*this, Id, I));
154 TypeIndexToSymbolId[Index] = Id;
155 return Id;
Adrian McCarthy8d090fc2017-07-12 19:38:11 +0000156}
157
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000158uint64_t NativeSession::getLoadAddress() const { return 0; }
159
Aaron Smith89a19ac2018-02-23 00:02:27 +0000160bool NativeSession::setLoadAddress(uint64_t Address) { return false; }
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000161
Adrian McCarthy6a4b0802017-06-22 18:42:23 +0000162std::unique_ptr<PDBSymbolExe> NativeSession::getGlobalScope() {
Zachary Turner7999b4f2018-09-05 23:30:38 +0000163 return PDBSymbol::createAs<PDBSymbolExe>(*this, getNativeGlobalScope());
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000164}
165
166std::unique_ptr<PDBSymbol>
167NativeSession::getSymbolById(uint32_t SymbolId) const {
Adrian McCarthybf0afc32017-06-28 22:47:40 +0000168 // If the caller has a SymbolId, it'd better be in our SymbolCache.
169 return SymbolId < SymbolCache.size()
Zachary Turner7999b4f2018-09-05 23:30:38 +0000170 ? PDBSymbol::create(*this, *SymbolCache[SymbolId])
Adrian McCarthybf0afc32017-06-28 22:47:40 +0000171 : nullptr;
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000172}
173
Aaron Smith53708a52018-03-26 22:10:02 +0000174bool NativeSession::addressForVA(uint64_t VA, uint32_t &Section,
175 uint32_t &Offset) const {
176 return false;
177}
178
179bool NativeSession::addressForRVA(uint32_t VA, uint32_t &Section,
180 uint32_t &Offset) const {
181 return false;
182}
183
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000184std::unique_ptr<PDBSymbol>
185NativeSession::findSymbolByAddress(uint64_t Address, PDB_SymType Type) const {
186 return nullptr;
187}
188
Aaron Smith3dca0be2018-04-10 17:33:18 +0000189std::unique_ptr<PDBSymbol>
190NativeSession::findSymbolByRVA(uint32_t RVA, PDB_SymType Type) const {
191 return nullptr;
192}
193
194std::unique_ptr<PDBSymbol>
195NativeSession::findSymbolBySectOffset(uint32_t Sect, uint32_t Offset,
196 PDB_SymType Type) const {
197 return nullptr;
198}
199
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000200std::unique_ptr<IPDBEnumLineNumbers>
201NativeSession::findLineNumbers(const PDBSymbolCompiland &Compiland,
202 const IPDBSourceFile &File) const {
203 return nullptr;
204}
205
206std::unique_ptr<IPDBEnumLineNumbers>
207NativeSession::findLineNumbersByAddress(uint64_t Address,
208 uint32_t Length) const {
209 return nullptr;
210}
211
Aaron Smith40198f52018-03-15 06:04:51 +0000212std::unique_ptr<IPDBEnumLineNumbers>
Aaron Smithed81a9d2018-03-26 22:13:22 +0000213NativeSession::findLineNumbersByRVA(uint32_t RVA, uint32_t Length) const {
214 return nullptr;
215}
216
217std::unique_ptr<IPDBEnumLineNumbers>
Aaron Smith40198f52018-03-15 06:04:51 +0000218NativeSession::findLineNumbersBySectOffset(uint32_t Section, uint32_t Offset,
219 uint32_t Length) const {
220 return nullptr;
221}
222
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000223std::unique_ptr<IPDBEnumSourceFiles>
224NativeSession::findSourceFiles(const PDBSymbolCompiland *Compiland,
225 StringRef Pattern,
226 PDB_NameSearchFlags Flags) const {
227 return nullptr;
228}
229
230std::unique_ptr<IPDBSourceFile>
231NativeSession::findOneSourceFile(const PDBSymbolCompiland *Compiland,
232 StringRef Pattern,
233 PDB_NameSearchFlags Flags) const {
234 return nullptr;
235}
236
237std::unique_ptr<IPDBEnumChildren<PDBSymbolCompiland>>
238NativeSession::findCompilandsForSourceFile(StringRef Pattern,
239 PDB_NameSearchFlags Flags) const {
240 return nullptr;
241}
242
243std::unique_ptr<PDBSymbolCompiland>
244NativeSession::findOneCompilandForSourceFile(StringRef Pattern,
245 PDB_NameSearchFlags Flags) const {
246 return nullptr;
247}
248
249std::unique_ptr<IPDBEnumSourceFiles> NativeSession::getAllSourceFiles() const {
250 return nullptr;
251}
252
253std::unique_ptr<IPDBEnumSourceFiles> NativeSession::getSourceFilesForCompiland(
254 const PDBSymbolCompiland &Compiland) const {
255 return nullptr;
256}
257
258std::unique_ptr<IPDBSourceFile>
259NativeSession::getSourceFileById(uint32_t FileId) const {
260 return nullptr;
261}
262
263std::unique_ptr<IPDBEnumDataStreams> NativeSession::getDebugStreams() const {
264 return nullptr;
265}
Aaron Smith89bca9e2017-11-16 14:33:09 +0000266
267std::unique_ptr<IPDBEnumTables> NativeSession::getEnumTables() const {
268 return nullptr;
269}
Zachary Turner679aead2018-03-13 17:46:06 +0000270
271std::unique_ptr<IPDBEnumInjectedSources>
272NativeSession::getInjectedSources() const {
273 return nullptr;
274}
Aaron Smith523de052018-03-22 04:08:15 +0000275
276std::unique_ptr<IPDBEnumSectionContribs>
277NativeSession::getSectionContribs() const {
278 return nullptr;
279}
Zachary Turner7999b4f2018-09-05 23:30:38 +0000280
281NativeExeSymbol &NativeSession::getNativeGlobalScope() {
282 if (ExeSymbol == 0) {
283 ExeSymbol = static_cast<SymIndexId>(SymbolCache.size());
284 SymbolCache.push_back(llvm::make_unique<NativeExeSymbol>(*this, ExeSymbol));
285 }
286 return static_cast<NativeExeSymbol &>(*SymbolCache[ExeSymbol]);
287}