blob: a6bbf6f9e4fc18511ea0f16a9a64f9903bcbd648 [file] [log] [blame]
Zachary Turnercffff262015-02-10 21:17:52 +00001//===- DIASession.cpp - DIA 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//===----------------------------------------------------------------------===//
Zachary Turner819e77d2016-05-06 20:51:57 +00009#include "llvm/DebugInfo/PDB/DIA/DIASession.h"
Zachary Turnerbe6d1e42015-02-10 23:46:48 +000010#include "llvm/ADT/STLExtras.h"
Zachary Turnercffff262015-02-10 21:17:52 +000011#include "llvm/DebugInfo/PDB/DIA/DIAEnumDebugStreams.h"
Zachary Turner679aead2018-03-13 17:46:06 +000012#include "llvm/DebugInfo/PDB/DIA/DIAEnumInjectedSources.h"
Zachary Turner4b083542015-04-17 22:40:36 +000013#include "llvm/DebugInfo/PDB/DIA/DIAEnumLineNumbers.h"
Zachary Turnera5549172015-02-10 22:43:25 +000014#include "llvm/DebugInfo/PDB/DIA/DIAEnumSourceFiles.h"
Aaron Smith89bca9e2017-11-16 14:33:09 +000015#include "llvm/DebugInfo/PDB/DIA/DIAEnumTables.h"
Zachary Turner819e77d2016-05-06 20:51:57 +000016#include "llvm/DebugInfo/PDB/DIA/DIAError.h"
Zachary Turnercffff262015-02-10 21:17:52 +000017#include "llvm/DebugInfo/PDB/DIA/DIARawSymbol.h"
Zachary Turnercffff262015-02-10 21:17:52 +000018#include "llvm/DebugInfo/PDB/DIA/DIASourceFile.h"
Zachary Turner819e77d2016-05-06 20:51:57 +000019#include "llvm/DebugInfo/PDB/DIA/DIASupport.h"
20#include "llvm/DebugInfo/PDB/GenericError.h"
21#include "llvm/DebugInfo/PDB/PDB.h"
Chandler Carruth71f308a2015-02-13 09:09:03 +000022#include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h"
23#include "llvm/DebugInfo/PDB/PDBSymbolExe.h"
Zachary Turnercffff262015-02-10 21:17:52 +000024#include "llvm/Support/ConvertUTF.h"
Zachary Turner38380322016-10-19 16:42:20 +000025#include "llvm/Support/Format.h"
Zachary Turner16901642017-04-24 17:47:24 +000026#include "llvm/Support/FormatVariadic.h"
Zachary Turner38380322016-10-19 16:42:20 +000027#include "llvm/Support/raw_ostream.h"
Zachary Turnercffff262015-02-10 21:17:52 +000028
29using namespace llvm;
Zachary Turnerec28fc32016-05-04 20:32:13 +000030using namespace llvm::pdb;
Zachary Turnercffff262015-02-10 21:17:52 +000031
Zachary Turner16901642017-04-24 17:47:24 +000032template <typename... Ts>
33static Error ErrorFromHResult(HRESULT Result, const char *Str, Ts &&... Args) {
34 SmallString<64> MessageStorage;
35 StringRef Context;
36 if (sizeof...(Args) > 0) {
37 MessageStorage = formatv(Str, std::forward<Ts>(Args)...).str();
38 Context = MessageStorage;
39 } else
40 Context = Str;
41
Zachary Turner819e77d2016-05-06 20:51:57 +000042 switch (Result) {
43 case E_PDB_NOT_FOUND:
Zachary Turner38380322016-10-19 16:42:20 +000044 return make_error<GenericError>(generic_error_code::invalid_path, Context);
Zachary Turner819e77d2016-05-06 20:51:57 +000045 case E_PDB_FORMAT:
Zachary Turner38380322016-10-19 16:42:20 +000046 return make_error<DIAError>(dia_error_code::invalid_file_format, Context);
Zachary Turner819e77d2016-05-06 20:51:57 +000047 case E_INVALIDARG:
Zachary Turner38380322016-10-19 16:42:20 +000048 return make_error<DIAError>(dia_error_code::invalid_parameter, Context);
Zachary Turner819e77d2016-05-06 20:51:57 +000049 case E_UNEXPECTED:
Zachary Turner38380322016-10-19 16:42:20 +000050 return make_error<DIAError>(dia_error_code::already_loaded, Context);
Zachary Turner819e77d2016-05-06 20:51:57 +000051 case E_PDB_INVALID_SIG:
52 case E_PDB_INVALID_AGE:
Zachary Turner38380322016-10-19 16:42:20 +000053 return make_error<DIAError>(dia_error_code::debug_info_mismatch, Context);
54 default: {
55 std::string S;
56 raw_string_ostream OS(S);
57 OS << "HRESULT: " << format_hex(static_cast<DWORD>(Result), 10, true)
58 << ": " << Context;
59 return make_error<DIAError>(dia_error_code::unspecified, OS.str());
60 }
Zachary Turner819e77d2016-05-06 20:51:57 +000061 }
62}
63
Reid Klecknerfb58be82016-10-12 21:51:14 +000064static Error LoadDIA(CComPtr<IDiaDataSource> &DiaDataSource) {
Nico Weber73853ab2016-04-01 22:21:51 +000065 if (SUCCEEDED(CoCreateInstance(CLSID_DiaSource, nullptr, CLSCTX_INPROC_SERVER,
66 IID_IDiaDataSource,
67 reinterpret_cast<LPVOID *>(&DiaDataSource))))
Zachary Turner819e77d2016-05-06 20:51:57 +000068 return Error::success();
Nico Weber73853ab2016-04-01 22:21:51 +000069
Zachary Turner819e77d2016-05-06 20:51:57 +000070// If the CoCreateInstance call above failed, msdia*.dll is not registered.
71// Try loading the DLL corresponding to the #included DIA SDK.
Nico Weber73853ab2016-04-01 22:21:51 +000072#if !defined(_MSC_VER)
Zachary Turner819e77d2016-05-06 20:51:57 +000073 return llvm::make_error<GenericError>(
74 "DIA is only supported when using MSVC.");
Reid Klecknerfb58be82016-10-12 21:51:14 +000075#else
Nico Weber73853ab2016-04-01 22:21:51 +000076 const wchar_t *msdia_dll = nullptr;
Reid Klecknerfb58be82016-10-12 21:51:14 +000077#if _MSC_VER >= 1900 && _MSC_VER < 2000
Nico Weber73853ab2016-04-01 22:21:51 +000078 msdia_dll = L"msdia140.dll"; // VS2015
Reid Klecknerfb58be82016-10-12 21:51:14 +000079#elif _MSC_VER >= 1800
Nico Weber73853ab2016-04-01 22:21:51 +000080 msdia_dll = L"msdia120.dll"; // VS2013
81#else
82#error "Unknown Visual Studio version."
83#endif
Zachary Turner23ee87b2016-04-19 17:36:58 +000084
Zachary Turner819e77d2016-05-06 20:51:57 +000085 HRESULT HR;
86 if (FAILED(HR = NoRegCoCreate(msdia_dll, CLSID_DiaSource, IID_IDiaDataSource,
87 reinterpret_cast<LPVOID *>(&DiaDataSource))))
Zachary Turner38380322016-10-19 16:42:20 +000088 return ErrorFromHResult(HR, "Calling NoRegCoCreate");
Zachary Turner819e77d2016-05-06 20:51:57 +000089 return Error::success();
Reid Klecknerfb58be82016-10-12 21:51:14 +000090#endif
Nico Weber73853ab2016-04-01 22:21:51 +000091}
Zachary Turnercffff262015-02-10 21:17:52 +000092
93DIASession::DIASession(CComPtr<IDiaSession> DiaSession) : Session(DiaSession) {}
94
Zachary Turner819e77d2016-05-06 20:51:57 +000095Error DIASession::createFromPdb(StringRef Path,
96 std::unique_ptr<IPDBSession> &Session) {
Zachary Turnerccf04152015-02-28 20:23:18 +000097 CComPtr<IDiaDataSource> DiaDataSource;
98 CComPtr<IDiaSession> DiaSession;
Zachary Turnercffff262015-02-10 21:17:52 +000099
100 // We assume that CoInitializeEx has already been called by the executable.
Zachary Turner819e77d2016-05-06 20:51:57 +0000101 if (auto E = LoadDIA(DiaDataSource))
102 return E;
Zachary Turnercffff262015-02-10 21:17:52 +0000103
104 llvm::SmallVector<UTF16, 128> Path16;
105 if (!llvm::convertUTF8ToUTF16String(Path, Path16))
Zachary Turner819e77d2016-05-06 20:51:57 +0000106 return make_error<GenericError>(generic_error_code::invalid_path);
Zachary Turnercffff262015-02-10 21:17:52 +0000107
108 const wchar_t *Path16Str = reinterpret_cast<const wchar_t*>(Path16.data());
Zachary Turner819e77d2016-05-06 20:51:57 +0000109 HRESULT HR;
Zachary Turner16901642017-04-24 17:47:24 +0000110 if (FAILED(HR = DiaDataSource->loadDataFromPdb(Path16Str))) {
111 return ErrorFromHResult(HR, "Calling loadDataFromPdb {0}", Path);
112 }
Zachary Turnercffff262015-02-10 21:17:52 +0000113
Zachary Turner819e77d2016-05-06 20:51:57 +0000114 if (FAILED(HR = DiaDataSource->openSession(&DiaSession)))
Zachary Turner38380322016-10-19 16:42:20 +0000115 return ErrorFromHResult(HR, "Calling openSession");
Zachary Turnerccf04152015-02-28 20:23:18 +0000116
117 Session.reset(new DIASession(DiaSession));
Zachary Turner819e77d2016-05-06 20:51:57 +0000118 return Error::success();
Zachary Turnercffff262015-02-10 21:17:52 +0000119}
120
Zachary Turner819e77d2016-05-06 20:51:57 +0000121Error DIASession::createFromExe(StringRef Path,
122 std::unique_ptr<IPDBSession> &Session) {
Zachary Turner4b083542015-04-17 22:40:36 +0000123 CComPtr<IDiaDataSource> DiaDataSource;
124 CComPtr<IDiaSession> DiaSession;
125
126 // We assume that CoInitializeEx has already been called by the executable.
Zachary Turner819e77d2016-05-06 20:51:57 +0000127 if (auto EC = LoadDIA(DiaDataSource))
128 return EC;
Zachary Turner4b083542015-04-17 22:40:36 +0000129
130 llvm::SmallVector<UTF16, 128> Path16;
131 if (!llvm::convertUTF8ToUTF16String(Path, Path16))
Zachary Turner819e77d2016-05-06 20:51:57 +0000132 return make_error<GenericError>(generic_error_code::invalid_path, Path);
Zachary Turner4b083542015-04-17 22:40:36 +0000133
134 const wchar_t *Path16Str = reinterpret_cast<const wchar_t *>(Path16.data());
Zachary Turner819e77d2016-05-06 20:51:57 +0000135 HRESULT HR;
136 if (FAILED(HR = DiaDataSource->loadDataForExe(Path16Str, nullptr, nullptr)))
Zachary Turner38380322016-10-19 16:42:20 +0000137 return ErrorFromHResult(HR, "Calling loadDataForExe");
Zachary Turner4b083542015-04-17 22:40:36 +0000138
Zachary Turner819e77d2016-05-06 20:51:57 +0000139 if (FAILED(HR = DiaDataSource->openSession(&DiaSession)))
Zachary Turner38380322016-10-19 16:42:20 +0000140 return ErrorFromHResult(HR, "Calling openSession");
Zachary Turner4b083542015-04-17 22:40:36 +0000141
142 Session.reset(new DIASession(DiaSession));
Zachary Turner819e77d2016-05-06 20:51:57 +0000143 return Error::success();
Zachary Turner4b083542015-04-17 22:40:36 +0000144}
145
Zachary Turnercffff262015-02-10 21:17:52 +0000146uint64_t DIASession::getLoadAddress() const {
147 uint64_t LoadAddress;
148 bool success = (S_OK == Session->get_loadAddress(&LoadAddress));
149 return (success) ? LoadAddress : 0;
150}
151
Aaron Smith89a19ac2018-02-23 00:02:27 +0000152bool DIASession::setLoadAddress(uint64_t Address) {
153 return (S_OK == Session->put_loadAddress(Address));
Zachary Turnercffff262015-02-10 21:17:52 +0000154}
155
Adrian McCarthy6a4b0802017-06-22 18:42:23 +0000156std::unique_ptr<PDBSymbolExe> DIASession::getGlobalScope() {
Zachary Turnercffff262015-02-10 21:17:52 +0000157 CComPtr<IDiaSymbol> GlobalScope;
158 if (S_OK != Session->get_globalScope(&GlobalScope))
159 return nullptr;
160
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000161 auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, GlobalScope);
Zachary Turnercffff262015-02-10 21:17:52 +0000162 auto PdbSymbol(PDBSymbol::create(*this, std::move(RawSymbol)));
163 std::unique_ptr<PDBSymbolExe> ExeSymbol(
164 static_cast<PDBSymbolExe *>(PdbSymbol.release()));
165 return ExeSymbol;
166}
167
168std::unique_ptr<PDBSymbol> DIASession::getSymbolById(uint32_t SymbolId) const {
169 CComPtr<IDiaSymbol> LocatedSymbol;
170 if (S_OK != Session->symbolById(SymbolId, &LocatedSymbol))
171 return nullptr;
172
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000173 auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, LocatedSymbol);
Zachary Turnercffff262015-02-10 21:17:52 +0000174 return PDBSymbol::create(*this, std::move(RawSymbol));
175}
176
Zachary Turner4b083542015-04-17 22:40:36 +0000177std::unique_ptr<PDBSymbol>
Zachary Turnere5cb2692015-05-01 20:24:26 +0000178DIASession::findSymbolByAddress(uint64_t Address, PDB_SymType Type) const {
179 enum SymTagEnum EnumVal = static_cast<enum SymTagEnum>(Type);
180
Zachary Turner4b083542015-04-17 22:40:36 +0000181 CComPtr<IDiaSymbol> Symbol;
Zachary Turnere5cb2692015-05-01 20:24:26 +0000182 if (S_OK != Session->findSymbolByVA(Address, EnumVal, &Symbol)) {
183 ULONGLONG LoadAddr = 0;
184 if (S_OK != Session->get_loadAddress(&LoadAddr))
185 return nullptr;
186 DWORD RVA = static_cast<DWORD>(Address - LoadAddr);
187 if (S_OK != Session->findSymbolByRVA(RVA, EnumVal, &Symbol))
188 return nullptr;
189 }
Zachary Turner4b083542015-04-17 22:40:36 +0000190 auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, Symbol);
191 return PDBSymbol::create(*this, std::move(RawSymbol));
192}
193
194std::unique_ptr<IPDBEnumLineNumbers>
Zachary Turner43ec3af2016-02-18 18:47:29 +0000195DIASession::findLineNumbers(const PDBSymbolCompiland &Compiland,
196 const IPDBSourceFile &File) const {
197 const DIARawSymbol &RawCompiland =
198 static_cast<const DIARawSymbol &>(Compiland.getRawSymbol());
199 const DIASourceFile &RawFile = static_cast<const DIASourceFile &>(File);
200
201 CComPtr<IDiaEnumLineNumbers> LineNumbers;
202 if (S_OK !=
203 Session->findLines(RawCompiland.getDiaSymbol(), RawFile.getDiaFile(),
204 &LineNumbers))
205 return nullptr;
206
207 return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
208}
209
210std::unique_ptr<IPDBEnumLineNumbers>
Zachary Turner4b083542015-04-17 22:40:36 +0000211DIASession::findLineNumbersByAddress(uint64_t Address, uint32_t Length) const {
212 CComPtr<IDiaEnumLineNumbers> LineNumbers;
213 if (S_OK != Session->findLinesByVA(Address, Length, &LineNumbers))
214 return nullptr;
215
216 return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
217}
218
Zachary Turner43ec3af2016-02-18 18:47:29 +0000219std::unique_ptr<IPDBEnumSourceFiles>
220DIASession::findSourceFiles(const PDBSymbolCompiland *Compiland,
221 llvm::StringRef Pattern,
222 PDB_NameSearchFlags Flags) const {
223 IDiaSymbol *DiaCompiland = nullptr;
224 CComBSTR Utf16Pattern;
225 if (!Pattern.empty())
226 Utf16Pattern = CComBSTR(Pattern.data());
227
228 if (Compiland)
229 DiaCompiland = static_cast<const DIARawSymbol &>(Compiland->getRawSymbol())
230 .getDiaSymbol();
231
232 Flags = static_cast<PDB_NameSearchFlags>(
233 Flags | PDB_NameSearchFlags::NS_FileNameExtMatch);
234 CComPtr<IDiaEnumSourceFiles> SourceFiles;
235 if (S_OK !=
236 Session->findFile(DiaCompiland, Utf16Pattern.m_str, Flags, &SourceFiles))
237 return nullptr;
238 return llvm::make_unique<DIAEnumSourceFiles>(*this, SourceFiles);
239}
240
241std::unique_ptr<IPDBSourceFile>
242DIASession::findOneSourceFile(const PDBSymbolCompiland *Compiland,
243 llvm::StringRef Pattern,
244 PDB_NameSearchFlags Flags) const {
245 auto SourceFiles = findSourceFiles(Compiland, Pattern, Flags);
246 if (!SourceFiles || SourceFiles->getChildCount() == 0)
247 return nullptr;
248 return SourceFiles->getNext();
249}
250
251std::unique_ptr<IPDBEnumChildren<PDBSymbolCompiland>>
252DIASession::findCompilandsForSourceFile(llvm::StringRef Pattern,
253 PDB_NameSearchFlags Flags) const {
254 auto File = findOneSourceFile(nullptr, Pattern, Flags);
255 if (!File)
256 return nullptr;
257 return File->getCompilands();
258}
259
260std::unique_ptr<PDBSymbolCompiland>
261DIASession::findOneCompilandForSourceFile(llvm::StringRef Pattern,
262 PDB_NameSearchFlags Flags) const {
263 auto Compilands = findCompilandsForSourceFile(Pattern, Flags);
264 if (!Compilands || Compilands->getChildCount() == 0)
265 return nullptr;
266 return Compilands->getNext();
267}
268
Zachary Turnera5549172015-02-10 22:43:25 +0000269std::unique_ptr<IPDBEnumSourceFiles> DIASession::getAllSourceFiles() const {
270 CComPtr<IDiaEnumSourceFiles> Files;
271 if (S_OK != Session->findFile(nullptr, nullptr, nsNone, &Files))
272 return nullptr;
273
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000274 return llvm::make_unique<DIAEnumSourceFiles>(*this, Files);
Zachary Turnera5549172015-02-10 22:43:25 +0000275}
276
277std::unique_ptr<IPDBEnumSourceFiles> DIASession::getSourceFilesForCompiland(
278 const PDBSymbolCompiland &Compiland) const {
279 CComPtr<IDiaEnumSourceFiles> Files;
280
281 const DIARawSymbol &RawSymbol =
282 static_cast<const DIARawSymbol &>(Compiland.getRawSymbol());
283 if (S_OK !=
284 Session->findFile(RawSymbol.getDiaSymbol(), nullptr, nsNone, &Files))
285 return nullptr;
286
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000287 return llvm::make_unique<DIAEnumSourceFiles>(*this, Files);
Zachary Turnera5549172015-02-10 22:43:25 +0000288}
289
Zachary Turnercffff262015-02-10 21:17:52 +0000290std::unique_ptr<IPDBSourceFile>
291DIASession::getSourceFileById(uint32_t FileId) const {
292 CComPtr<IDiaSourceFile> LocatedFile;
293 if (S_OK != Session->findFileById(FileId, &LocatedFile))
294 return nullptr;
295
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000296 return llvm::make_unique<DIASourceFile>(*this, LocatedFile);
Zachary Turnercffff262015-02-10 21:17:52 +0000297}
298
299std::unique_ptr<IPDBEnumDataStreams> DIASession::getDebugStreams() const {
300 CComPtr<IDiaEnumDebugStreams> DiaEnumerator;
301 if (S_OK != Session->getEnumDebugStreams(&DiaEnumerator))
302 return nullptr;
303
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000304 return llvm::make_unique<DIAEnumDebugStreams>(DiaEnumerator);
Zachary Turnercffff262015-02-10 21:17:52 +0000305}
Aaron Smith89bca9e2017-11-16 14:33:09 +0000306
307std::unique_ptr<IPDBEnumTables> DIASession::getEnumTables() const {
308 CComPtr<IDiaEnumTables> DiaEnumerator;
309 if (S_OK != Session->getEnumTables(&DiaEnumerator))
310 return nullptr;
311
312 return llvm::make_unique<DIAEnumTables>(DiaEnumerator);
313}
Zachary Turner679aead2018-03-13 17:46:06 +0000314
315static CComPtr<IDiaEnumInjectedSources>
316getEnumInjectedSources(IDiaSession &Session) {
317 CComPtr<IDiaEnumInjectedSources> EIS;
318 CComPtr<IDiaEnumTables> ET;
319 CComPtr<IDiaTable> Table;
320 ULONG Count = 0;
321
322 if (Session.getEnumTables(&ET) != S_OK)
323 return nullptr;
324
325 while (ET->Next(1, &Table, &Count) == S_OK && Count == 1) {
326 // There is only one table that matches the given iid
327 if (S_OK ==
328 Table->QueryInterface(__uuidof(IDiaEnumInjectedSources), (void **)&EIS))
329 break;
330 Table.Release();
331 }
332 return EIS;
333}
334std::unique_ptr<IPDBEnumInjectedSources>
335DIASession::getInjectedSources() const {
336 CComPtr<IDiaEnumInjectedSources> Files = getEnumInjectedSources(*Session);
337 if (!Files)
338 return nullptr;
339
340 return llvm::make_unique<DIAEnumInjectedSources>(*this, Files);
341}