blob: 1eaecc4976128fcade328c4930c9bfe5cf8b2be0 [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/GenericError.h"
15#include "llvm/DebugInfo/PDB/IPDBEnumChildren.h"
16#include "llvm/DebugInfo/PDB/IPDBSourceFile.h"
Adrian McCarthy8d090fc2017-07-12 19:38:11 +000017#include "llvm/DebugInfo/PDB/Native/NativeBuiltinSymbol.h"
Adrian McCarthybf0afc32017-06-28 22:47:40 +000018#include "llvm/DebugInfo/PDB/Native/NativeCompilandSymbol.h"
Adrian McCarthyb41f03e2017-08-04 22:37:58 +000019#include "llvm/DebugInfo/PDB/Native/NativeEnumSymbol.h"
20#include "llvm/DebugInfo/PDB/Native/NativeEnumTypes.h"
Adrian McCarthy4d93d662017-03-29 19:27:08 +000021#include "llvm/DebugInfo/PDB/Native/NativeExeSymbol.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000022#include "llvm/DebugInfo/PDB/Native/PDBFile.h"
23#include "llvm/DebugInfo/PDB/Native/RawError.h"
Adrian McCarthyb41f03e2017-08-04 22:37:58 +000024#include "llvm/DebugInfo/PDB/Native/TpiStream.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000025#include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h"
26#include "llvm/DebugInfo/PDB/PDBSymbolExe.h"
Adrian McCarthyb41f03e2017-08-04 22:37:58 +000027#include "llvm/DebugInfo/PDB/PDBSymbolTypeEnum.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000028#include "llvm/Support/Allocator.h"
Zachary Turnerd9dc2822017-03-02 20:52:51 +000029#include "llvm/Support/BinaryByteStream.h"
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000030#include "llvm/Support/Error.h"
31#include "llvm/Support/ErrorOr.h"
32#include "llvm/Support/MemoryBuffer.h"
Adrian McCarthybf0afc32017-06-28 22:47:40 +000033
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000034#include <algorithm>
Adrian McCarthyb41f03e2017-08-04 22:37:58 +000035#include <cassert>
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000036#include <memory>
Adrian McCarthybf0afc32017-06-28 22:47:40 +000037#include <utility>
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000038
39using namespace llvm;
40using namespace llvm::msf;
41using namespace llvm::pdb;
42
Adrian McCarthy8d090fc2017-07-12 19:38:11 +000043namespace {
44// Maps codeview::SimpleTypeKind of a built-in type to the parameters necessary
45// to instantiate a NativeBuiltinSymbol for that type.
Reid Kleckner0962cb22017-07-12 19:46:35 +000046static const struct BuiltinTypeEntry {
Adrian McCarthy8d090fc2017-07-12 19:38:11 +000047 codeview::SimpleTypeKind Kind;
48 PDB_BuiltinType Type;
49 uint32_t Size;
50} BuiltinTypes[] = {
51 {codeview::SimpleTypeKind::Int32, PDB_BuiltinType::Int, 4},
52 {codeview::SimpleTypeKind::UInt32, PDB_BuiltinType::UInt, 4},
53 {codeview::SimpleTypeKind::UInt32Long, PDB_BuiltinType::UInt, 4},
54 {codeview::SimpleTypeKind::UInt64Quad, PDB_BuiltinType::UInt, 8},
55 {codeview::SimpleTypeKind::NarrowCharacter, PDB_BuiltinType::Char, 1},
56 {codeview::SimpleTypeKind::SignedCharacter, PDB_BuiltinType::Char, 1},
57 {codeview::SimpleTypeKind::UnsignedCharacter, PDB_BuiltinType::UInt, 1},
58 {codeview::SimpleTypeKind::UInt16Short, PDB_BuiltinType::UInt, 2},
59 {codeview::SimpleTypeKind::Boolean8, PDB_BuiltinType::Bool, 1}
60 // This table can be grown as necessary, but these are the only types we've
61 // needed so far.
62};
63} // namespace
64
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000065NativeSession::NativeSession(std::unique_ptr<PDBFile> PdbFile,
66 std::unique_ptr<BumpPtrAllocator> Allocator)
67 : Pdb(std::move(PdbFile)), Allocator(std::move(Allocator)) {}
68
69NativeSession::~NativeSession() = default;
70
Peter Collingbourne75257bc2017-10-20 19:48:26 +000071Error NativeSession::createFromPdb(std::unique_ptr<MemoryBuffer> Buffer,
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000072 std::unique_ptr<IPDBSession> &Session) {
Peter Collingbourne75257bc2017-10-20 19:48:26 +000073 StringRef Path = Buffer->getBufferIdentifier();
Zachary Turner695ed562017-02-28 00:04:07 +000074 auto Stream = llvm::make_unique<MemoryBufferByteStream>(
75 std::move(Buffer), llvm::support::little);
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000076
77 auto Allocator = llvm::make_unique<BumpPtrAllocator>();
Zachary Turner7b327d02017-02-16 23:35:45 +000078 auto File = llvm::make_unique<PDBFile>(Path, std::move(Stream), *Allocator);
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +000079 if (auto EC = File->parseFileHeaders())
80 return EC;
81 if (auto EC = File->parseStreamData())
82 return EC;
83
84 Session =
85 llvm::make_unique<NativeSession>(std::move(File), std::move(Allocator));
86
87 return Error::success();
88}
89
90Error NativeSession::createFromExe(StringRef Path,
91 std::unique_ptr<IPDBSession> &Session) {
92 return make_error<RawError>(raw_error_code::feature_unsupported);
93}
94
Adrian McCarthybf0afc32017-06-28 22:47:40 +000095std::unique_ptr<PDBSymbolCompiland>
96NativeSession::createCompilandSymbol(DbiModuleDescriptor MI) {
Adrian McCarthy8d090fc2017-07-12 19:38:11 +000097 const auto Id = static_cast<SymIndexId>(SymbolCache.size());
Adrian McCarthybf0afc32017-06-28 22:47:40 +000098 SymbolCache.push_back(
99 llvm::make_unique<NativeCompilandSymbol>(*this, Id, MI));
100 return llvm::make_unique<PDBSymbolCompiland>(
101 *this, std::unique_ptr<IPDBRawSymbol>(SymbolCache[Id]->clone()));
102}
103
Adrian McCarthyb41f03e2017-08-04 22:37:58 +0000104std::unique_ptr<PDBSymbolTypeEnum>
105NativeSession::createEnumSymbol(codeview::TypeIndex Index) {
106 const auto Id = findSymbolByTypeIndex(Index);
107 return llvm::make_unique<PDBSymbolTypeEnum>(
108 *this, std::unique_ptr<IPDBRawSymbol>(SymbolCache[Id]->clone()));
109}
110
111std::unique_ptr<IPDBEnumSymbols>
112NativeSession::createTypeEnumerator(codeview::TypeLeafKind Kind) {
113 auto Tpi = Pdb->getPDBTpiStream();
114 if (!Tpi) {
115 consumeError(Tpi.takeError());
116 return nullptr;
117 }
118 auto &Types = Tpi->typeCollection();
119 return std::unique_ptr<IPDBEnumSymbols>(
120 new NativeEnumTypes(*this, Types, codeview::LF_ENUM));
121}
122
Adrian McCarthy8d090fc2017-07-12 19:38:11 +0000123SymIndexId NativeSession::findSymbolByTypeIndex(codeview::TypeIndex Index) {
124 // First see if it's already in our cache.
125 const auto Entry = TypeIndexToSymbolId.find(Index);
126 if (Entry != TypeIndexToSymbolId.end())
127 return Entry->second;
128
129 // Symbols for built-in types are created on the fly.
130 if (Index.isSimple()) {
131 // FIXME: We will eventually need to handle pointers to other simple types,
132 // which are still simple types in the world of CodeView TypeIndexes.
133 if (Index.getSimpleMode() != codeview::SimpleTypeMode::Direct)
134 return 0;
135 const auto Kind = Index.getSimpleKind();
Reid Kleckner0962cb22017-07-12 19:46:35 +0000136 const auto It =
137 std::find_if(std::begin(BuiltinTypes), std::end(BuiltinTypes),
138 [Kind](const BuiltinTypeEntry &Builtin) {
139 return Builtin.Kind == Kind;
140 });
Adrian McCarthy8d090fc2017-07-12 19:38:11 +0000141 if (It == std::end(BuiltinTypes))
142 return 0;
143 SymIndexId Id = SymbolCache.size();
144 SymbolCache.emplace_back(
Reid Kleckner0962cb22017-07-12 19:46:35 +0000145 llvm::make_unique<NativeBuiltinSymbol>(*this, Id, It->Type, It->Size));
Adrian McCarthy8d090fc2017-07-12 19:38:11 +0000146 TypeIndexToSymbolId[Index] = Id;
147 return Id;
148 }
149
Adrian McCarthyb41f03e2017-08-04 22:37:58 +0000150 // We need to instantiate and cache the desired type symbol.
151 auto Tpi = Pdb->getPDBTpiStream();
152 if (!Tpi) {
153 consumeError(Tpi.takeError());
154 return 0;
155 }
156 auto &Types = Tpi->typeCollection();
157 const auto &I = Types.getType(Index);
158 const auto Id = static_cast<SymIndexId>(SymbolCache.size());
159 // TODO(amccarth): Make this handle all types, not just LF_ENUMs.
160 assert(I.kind() == codeview::LF_ENUM);
161 SymbolCache.emplace_back(llvm::make_unique<NativeEnumSymbol>(*this, Id, I));
162 TypeIndexToSymbolId[Index] = Id;
163 return Id;
Adrian McCarthy8d090fc2017-07-12 19:38:11 +0000164}
165
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000166uint64_t NativeSession::getLoadAddress() const { return 0; }
167
Aaron Smith89a19ac2018-02-23 00:02:27 +0000168bool NativeSession::setLoadAddress(uint64_t Address) { return false; }
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000169
Adrian McCarthy6a4b0802017-06-22 18:42:23 +0000170std::unique_ptr<PDBSymbolExe> NativeSession::getGlobalScope() {
Adrian McCarthy8d090fc2017-07-12 19:38:11 +0000171 const auto Id = static_cast<SymIndexId>(SymbolCache.size());
Adrian McCarthybf0afc32017-06-28 22:47:40 +0000172 SymbolCache.push_back(llvm::make_unique<NativeExeSymbol>(*this, Id));
173 auto RawSymbol = SymbolCache[Id]->clone();
Adrian McCarthy649b8e02017-02-24 00:10:47 +0000174 auto PdbSymbol(PDBSymbol::create(*this, std::move(RawSymbol)));
175 std::unique_ptr<PDBSymbolExe> ExeSymbol(
Adrian McCarthy6a4b0802017-06-22 18:42:23 +0000176 static_cast<PDBSymbolExe *>(PdbSymbol.release()));
Adrian McCarthy649b8e02017-02-24 00:10:47 +0000177 return ExeSymbol;
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000178}
179
180std::unique_ptr<PDBSymbol>
181NativeSession::getSymbolById(uint32_t SymbolId) const {
Adrian McCarthybf0afc32017-06-28 22:47:40 +0000182 // If the caller has a SymbolId, it'd better be in our SymbolCache.
183 return SymbolId < SymbolCache.size()
184 ? PDBSymbol::create(*this, SymbolCache[SymbolId]->clone())
185 : nullptr;
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000186}
187
Aaron Smith53708a52018-03-26 22:10:02 +0000188bool NativeSession::addressForVA(uint64_t VA, uint32_t &Section,
189 uint32_t &Offset) const {
190 return false;
191}
192
193bool NativeSession::addressForRVA(uint32_t VA, uint32_t &Section,
194 uint32_t &Offset) const {
195 return false;
196}
197
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000198std::unique_ptr<PDBSymbol>
199NativeSession::findSymbolByAddress(uint64_t Address, PDB_SymType Type) const {
200 return nullptr;
201}
202
203std::unique_ptr<IPDBEnumLineNumbers>
204NativeSession::findLineNumbers(const PDBSymbolCompiland &Compiland,
205 const IPDBSourceFile &File) const {
206 return nullptr;
207}
208
209std::unique_ptr<IPDBEnumLineNumbers>
210NativeSession::findLineNumbersByAddress(uint64_t Address,
211 uint32_t Length) const {
212 return nullptr;
213}
214
Aaron Smith40198f52018-03-15 06:04:51 +0000215std::unique_ptr<IPDBEnumLineNumbers>
216NativeSession::findLineNumbersBySectOffset(uint32_t Section, uint32_t Offset,
217 uint32_t Length) const {
218 return nullptr;
219}
220
Adrian McCarthy6b6b8c42017-01-25 22:38:55 +0000221std::unique_ptr<IPDBEnumSourceFiles>
222NativeSession::findSourceFiles(const PDBSymbolCompiland *Compiland,
223 StringRef Pattern,
224 PDB_NameSearchFlags Flags) const {
225 return nullptr;
226}
227
228std::unique_ptr<IPDBSourceFile>
229NativeSession::findOneSourceFile(const PDBSymbolCompiland *Compiland,
230 StringRef Pattern,
231 PDB_NameSearchFlags Flags) const {
232 return nullptr;
233}
234
235std::unique_ptr<IPDBEnumChildren<PDBSymbolCompiland>>
236NativeSession::findCompilandsForSourceFile(StringRef Pattern,
237 PDB_NameSearchFlags Flags) const {
238 return nullptr;
239}
240
241std::unique_ptr<PDBSymbolCompiland>
242NativeSession::findOneCompilandForSourceFile(StringRef Pattern,
243 PDB_NameSearchFlags Flags) const {
244 return nullptr;
245}
246
247std::unique_ptr<IPDBEnumSourceFiles> NativeSession::getAllSourceFiles() const {
248 return nullptr;
249}
250
251std::unique_ptr<IPDBEnumSourceFiles> NativeSession::getSourceFilesForCompiland(
252 const PDBSymbolCompiland &Compiland) const {
253 return nullptr;
254}
255
256std::unique_ptr<IPDBSourceFile>
257NativeSession::getSourceFileById(uint32_t FileId) const {
258 return nullptr;
259}
260
261std::unique_ptr<IPDBEnumDataStreams> NativeSession::getDebugStreams() const {
262 return nullptr;
263}
Aaron Smith89bca9e2017-11-16 14:33:09 +0000264
265std::unique_ptr<IPDBEnumTables> NativeSession::getEnumTables() const {
266 return nullptr;
267}
Zachary Turner679aead2018-03-13 17:46:06 +0000268
269std::unique_ptr<IPDBEnumInjectedSources>
270NativeSession::getInjectedSources() const {
271 return nullptr;
272}
Aaron Smith523de052018-03-22 04:08:15 +0000273
274std::unique_ptr<IPDBEnumSectionContribs>
275NativeSession::getSectionContribs() const {
276 return nullptr;
277}