blob: 861b7bb4c26da492dfd83ea2c37d88ff703cc5e8 [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 Turner4b083542015-04-17 22:40:36 +000012#include "llvm/DebugInfo/PDB/DIA/DIAEnumLineNumbers.h"
Zachary Turnera5549172015-02-10 22:43:25 +000013#include "llvm/DebugInfo/PDB/DIA/DIAEnumSourceFiles.h"
Zachary Turner819e77d2016-05-06 20:51:57 +000014#include "llvm/DebugInfo/PDB/DIA/DIAError.h"
Zachary Turnercffff262015-02-10 21:17:52 +000015#include "llvm/DebugInfo/PDB/DIA/DIARawSymbol.h"
Zachary Turnercffff262015-02-10 21:17:52 +000016#include "llvm/DebugInfo/PDB/DIA/DIASourceFile.h"
Zachary Turner819e77d2016-05-06 20:51:57 +000017#include "llvm/DebugInfo/PDB/DIA/DIASupport.h"
18#include "llvm/DebugInfo/PDB/GenericError.h"
19#include "llvm/DebugInfo/PDB/PDB.h"
Chandler Carruth71f308a2015-02-13 09:09:03 +000020#include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h"
21#include "llvm/DebugInfo/PDB/PDBSymbolExe.h"
Zachary Turnercffff262015-02-10 21:17:52 +000022#include "llvm/Support/ConvertUTF.h"
23
24using namespace llvm;
Zachary Turnerec28fc32016-05-04 20:32:13 +000025using namespace llvm::pdb;
Zachary Turnercffff262015-02-10 21:17:52 +000026
Reid Klecknerfb58be82016-10-12 21:51:14 +000027static Error ErrorFromHResult(HRESULT Result) {
Zachary Turner819e77d2016-05-06 20:51:57 +000028 switch (Result) {
29 case E_PDB_NOT_FOUND:
30 return make_error<GenericError>(generic_error_code::invalid_path);
31 case E_PDB_FORMAT:
32 return make_error<DIAError>(dia_error_code::invalid_file_format);
33 case E_INVALIDARG:
34 return make_error<DIAError>(dia_error_code::invalid_parameter);
35 case E_UNEXPECTED:
36 return make_error<DIAError>(dia_error_code::already_loaded);
37 case E_PDB_INVALID_SIG:
38 case E_PDB_INVALID_AGE:
39 return make_error<DIAError>(dia_error_code::debug_info_mismatch);
40 default:
41 return make_error<DIAError>(dia_error_code::unspecified);
42 }
43}
44
Reid Klecknerfb58be82016-10-12 21:51:14 +000045static Error LoadDIA(CComPtr<IDiaDataSource> &DiaDataSource) {
Nico Weber73853ab2016-04-01 22:21:51 +000046 if (SUCCEEDED(CoCreateInstance(CLSID_DiaSource, nullptr, CLSCTX_INPROC_SERVER,
47 IID_IDiaDataSource,
48 reinterpret_cast<LPVOID *>(&DiaDataSource))))
Zachary Turner819e77d2016-05-06 20:51:57 +000049 return Error::success();
Nico Weber73853ab2016-04-01 22:21:51 +000050
Zachary Turner819e77d2016-05-06 20:51:57 +000051// If the CoCreateInstance call above failed, msdia*.dll is not registered.
52// Try loading the DLL corresponding to the #included DIA SDK.
Nico Weber73853ab2016-04-01 22:21:51 +000053#if !defined(_MSC_VER)
Zachary Turner819e77d2016-05-06 20:51:57 +000054 return llvm::make_error<GenericError>(
55 "DIA is only supported when using MSVC.");
Reid Klecknerfb58be82016-10-12 21:51:14 +000056#else
Nico Weber73853ab2016-04-01 22:21:51 +000057 const wchar_t *msdia_dll = nullptr;
Reid Klecknerfb58be82016-10-12 21:51:14 +000058#if _MSC_VER >= 1900 && _MSC_VER < 2000
Nico Weber73853ab2016-04-01 22:21:51 +000059 msdia_dll = L"msdia140.dll"; // VS2015
Reid Klecknerfb58be82016-10-12 21:51:14 +000060#elif _MSC_VER >= 1800
Nico Weber73853ab2016-04-01 22:21:51 +000061 msdia_dll = L"msdia120.dll"; // VS2013
62#else
63#error "Unknown Visual Studio version."
64#endif
Zachary Turner23ee87b2016-04-19 17:36:58 +000065
Zachary Turner819e77d2016-05-06 20:51:57 +000066 HRESULT HR;
67 if (FAILED(HR = NoRegCoCreate(msdia_dll, CLSID_DiaSource, IID_IDiaDataSource,
68 reinterpret_cast<LPVOID *>(&DiaDataSource))))
69 return ErrorFromHResult(HR);
70 return Error::success();
Reid Klecknerfb58be82016-10-12 21:51:14 +000071#endif
Nico Weber73853ab2016-04-01 22:21:51 +000072}
Zachary Turnercffff262015-02-10 21:17:52 +000073
74DIASession::DIASession(CComPtr<IDiaSession> DiaSession) : Session(DiaSession) {}
75
Zachary Turner819e77d2016-05-06 20:51:57 +000076Error DIASession::createFromPdb(StringRef Path,
77 std::unique_ptr<IPDBSession> &Session) {
Zachary Turnerccf04152015-02-28 20:23:18 +000078 CComPtr<IDiaDataSource> DiaDataSource;
79 CComPtr<IDiaSession> DiaSession;
Zachary Turnercffff262015-02-10 21:17:52 +000080
81 // We assume that CoInitializeEx has already been called by the executable.
Zachary Turner819e77d2016-05-06 20:51:57 +000082 if (auto E = LoadDIA(DiaDataSource))
83 return E;
Zachary Turnercffff262015-02-10 21:17:52 +000084
85 llvm::SmallVector<UTF16, 128> Path16;
86 if (!llvm::convertUTF8ToUTF16String(Path, Path16))
Zachary Turner819e77d2016-05-06 20:51:57 +000087 return make_error<GenericError>(generic_error_code::invalid_path);
Zachary Turnercffff262015-02-10 21:17:52 +000088
89 const wchar_t *Path16Str = reinterpret_cast<const wchar_t*>(Path16.data());
Zachary Turner819e77d2016-05-06 20:51:57 +000090 HRESULT HR;
91 if (FAILED(HR = DiaDataSource->loadDataFromPdb(Path16Str)))
92 return ErrorFromHResult(HR);
Zachary Turnercffff262015-02-10 21:17:52 +000093
Zachary Turner819e77d2016-05-06 20:51:57 +000094 if (FAILED(HR = DiaDataSource->openSession(&DiaSession)))
95 return ErrorFromHResult(HR);
Zachary Turnerccf04152015-02-28 20:23:18 +000096
97 Session.reset(new DIASession(DiaSession));
Zachary Turner819e77d2016-05-06 20:51:57 +000098 return Error::success();
Zachary Turnercffff262015-02-10 21:17:52 +000099}
100
Zachary Turner819e77d2016-05-06 20:51:57 +0000101Error DIASession::createFromExe(StringRef Path,
102 std::unique_ptr<IPDBSession> &Session) {
Zachary Turner4b083542015-04-17 22:40:36 +0000103 CComPtr<IDiaDataSource> DiaDataSource;
104 CComPtr<IDiaSession> DiaSession;
105
106 // We assume that CoInitializeEx has already been called by the executable.
Zachary Turner819e77d2016-05-06 20:51:57 +0000107 if (auto EC = LoadDIA(DiaDataSource))
108 return EC;
Zachary Turner4b083542015-04-17 22:40:36 +0000109
110 llvm::SmallVector<UTF16, 128> Path16;
111 if (!llvm::convertUTF8ToUTF16String(Path, Path16))
Zachary Turner819e77d2016-05-06 20:51:57 +0000112 return make_error<GenericError>(generic_error_code::invalid_path, Path);
Zachary Turner4b083542015-04-17 22:40:36 +0000113
114 const wchar_t *Path16Str = reinterpret_cast<const wchar_t *>(Path16.data());
Zachary Turner819e77d2016-05-06 20:51:57 +0000115 HRESULT HR;
116 if (FAILED(HR = DiaDataSource->loadDataForExe(Path16Str, nullptr, nullptr)))
117 return ErrorFromHResult(HR);
Zachary Turner4b083542015-04-17 22:40:36 +0000118
Zachary Turner819e77d2016-05-06 20:51:57 +0000119 if (FAILED(HR = DiaDataSource->openSession(&DiaSession)))
120 return ErrorFromHResult(HR);
Zachary Turner4b083542015-04-17 22:40:36 +0000121
122 Session.reset(new DIASession(DiaSession));
Zachary Turner819e77d2016-05-06 20:51:57 +0000123 return Error::success();
Zachary Turner4b083542015-04-17 22:40:36 +0000124}
125
Zachary Turnercffff262015-02-10 21:17:52 +0000126uint64_t DIASession::getLoadAddress() const {
127 uint64_t LoadAddress;
128 bool success = (S_OK == Session->get_loadAddress(&LoadAddress));
129 return (success) ? LoadAddress : 0;
130}
131
132void DIASession::setLoadAddress(uint64_t Address) {
133 Session->put_loadAddress(Address);
134}
135
136std::unique_ptr<PDBSymbolExe> DIASession::getGlobalScope() const {
137 CComPtr<IDiaSymbol> GlobalScope;
138 if (S_OK != Session->get_globalScope(&GlobalScope))
139 return nullptr;
140
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000141 auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, GlobalScope);
Zachary Turnercffff262015-02-10 21:17:52 +0000142 auto PdbSymbol(PDBSymbol::create(*this, std::move(RawSymbol)));
143 std::unique_ptr<PDBSymbolExe> ExeSymbol(
144 static_cast<PDBSymbolExe *>(PdbSymbol.release()));
145 return ExeSymbol;
146}
147
148std::unique_ptr<PDBSymbol> DIASession::getSymbolById(uint32_t SymbolId) const {
149 CComPtr<IDiaSymbol> LocatedSymbol;
150 if (S_OK != Session->symbolById(SymbolId, &LocatedSymbol))
151 return nullptr;
152
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000153 auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, LocatedSymbol);
Zachary Turnercffff262015-02-10 21:17:52 +0000154 return PDBSymbol::create(*this, std::move(RawSymbol));
155}
156
Zachary Turner4b083542015-04-17 22:40:36 +0000157std::unique_ptr<PDBSymbol>
Zachary Turnere5cb2692015-05-01 20:24:26 +0000158DIASession::findSymbolByAddress(uint64_t Address, PDB_SymType Type) const {
159 enum SymTagEnum EnumVal = static_cast<enum SymTagEnum>(Type);
160
Zachary Turner4b083542015-04-17 22:40:36 +0000161 CComPtr<IDiaSymbol> Symbol;
Zachary Turnere5cb2692015-05-01 20:24:26 +0000162 if (S_OK != Session->findSymbolByVA(Address, EnumVal, &Symbol)) {
163 ULONGLONG LoadAddr = 0;
164 if (S_OK != Session->get_loadAddress(&LoadAddr))
165 return nullptr;
166 DWORD RVA = static_cast<DWORD>(Address - LoadAddr);
167 if (S_OK != Session->findSymbolByRVA(RVA, EnumVal, &Symbol))
168 return nullptr;
169 }
Zachary Turner4b083542015-04-17 22:40:36 +0000170 auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, Symbol);
171 return PDBSymbol::create(*this, std::move(RawSymbol));
172}
173
174std::unique_ptr<IPDBEnumLineNumbers>
Zachary Turner43ec3af2016-02-18 18:47:29 +0000175DIASession::findLineNumbers(const PDBSymbolCompiland &Compiland,
176 const IPDBSourceFile &File) const {
177 const DIARawSymbol &RawCompiland =
178 static_cast<const DIARawSymbol &>(Compiland.getRawSymbol());
179 const DIASourceFile &RawFile = static_cast<const DIASourceFile &>(File);
180
181 CComPtr<IDiaEnumLineNumbers> LineNumbers;
182 if (S_OK !=
183 Session->findLines(RawCompiland.getDiaSymbol(), RawFile.getDiaFile(),
184 &LineNumbers))
185 return nullptr;
186
187 return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
188}
189
190std::unique_ptr<IPDBEnumLineNumbers>
Zachary Turner4b083542015-04-17 22:40:36 +0000191DIASession::findLineNumbersByAddress(uint64_t Address, uint32_t Length) const {
192 CComPtr<IDiaEnumLineNumbers> LineNumbers;
193 if (S_OK != Session->findLinesByVA(Address, Length, &LineNumbers))
194 return nullptr;
195
196 return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
197}
198
Zachary Turner43ec3af2016-02-18 18:47:29 +0000199std::unique_ptr<IPDBEnumSourceFiles>
200DIASession::findSourceFiles(const PDBSymbolCompiland *Compiland,
201 llvm::StringRef Pattern,
202 PDB_NameSearchFlags Flags) const {
203 IDiaSymbol *DiaCompiland = nullptr;
204 CComBSTR Utf16Pattern;
205 if (!Pattern.empty())
206 Utf16Pattern = CComBSTR(Pattern.data());
207
208 if (Compiland)
209 DiaCompiland = static_cast<const DIARawSymbol &>(Compiland->getRawSymbol())
210 .getDiaSymbol();
211
212 Flags = static_cast<PDB_NameSearchFlags>(
213 Flags | PDB_NameSearchFlags::NS_FileNameExtMatch);
214 CComPtr<IDiaEnumSourceFiles> SourceFiles;
215 if (S_OK !=
216 Session->findFile(DiaCompiland, Utf16Pattern.m_str, Flags, &SourceFiles))
217 return nullptr;
218 return llvm::make_unique<DIAEnumSourceFiles>(*this, SourceFiles);
219}
220
221std::unique_ptr<IPDBSourceFile>
222DIASession::findOneSourceFile(const PDBSymbolCompiland *Compiland,
223 llvm::StringRef Pattern,
224 PDB_NameSearchFlags Flags) const {
225 auto SourceFiles = findSourceFiles(Compiland, Pattern, Flags);
226 if (!SourceFiles || SourceFiles->getChildCount() == 0)
227 return nullptr;
228 return SourceFiles->getNext();
229}
230
231std::unique_ptr<IPDBEnumChildren<PDBSymbolCompiland>>
232DIASession::findCompilandsForSourceFile(llvm::StringRef Pattern,
233 PDB_NameSearchFlags Flags) const {
234 auto File = findOneSourceFile(nullptr, Pattern, Flags);
235 if (!File)
236 return nullptr;
237 return File->getCompilands();
238}
239
240std::unique_ptr<PDBSymbolCompiland>
241DIASession::findOneCompilandForSourceFile(llvm::StringRef Pattern,
242 PDB_NameSearchFlags Flags) const {
243 auto Compilands = findCompilandsForSourceFile(Pattern, Flags);
244 if (!Compilands || Compilands->getChildCount() == 0)
245 return nullptr;
246 return Compilands->getNext();
247}
248
Zachary Turnera5549172015-02-10 22:43:25 +0000249std::unique_ptr<IPDBEnumSourceFiles> DIASession::getAllSourceFiles() const {
250 CComPtr<IDiaEnumSourceFiles> Files;
251 if (S_OK != Session->findFile(nullptr, nullptr, nsNone, &Files))
252 return nullptr;
253
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000254 return llvm::make_unique<DIAEnumSourceFiles>(*this, Files);
Zachary Turnera5549172015-02-10 22:43:25 +0000255}
256
257std::unique_ptr<IPDBEnumSourceFiles> DIASession::getSourceFilesForCompiland(
258 const PDBSymbolCompiland &Compiland) const {
259 CComPtr<IDiaEnumSourceFiles> Files;
260
261 const DIARawSymbol &RawSymbol =
262 static_cast<const DIARawSymbol &>(Compiland.getRawSymbol());
263 if (S_OK !=
264 Session->findFile(RawSymbol.getDiaSymbol(), nullptr, nsNone, &Files))
265 return nullptr;
266
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000267 return llvm::make_unique<DIAEnumSourceFiles>(*this, Files);
Zachary Turnera5549172015-02-10 22:43:25 +0000268}
269
Zachary Turnercffff262015-02-10 21:17:52 +0000270std::unique_ptr<IPDBSourceFile>
271DIASession::getSourceFileById(uint32_t FileId) const {
272 CComPtr<IDiaSourceFile> LocatedFile;
273 if (S_OK != Session->findFileById(FileId, &LocatedFile))
274 return nullptr;
275
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000276 return llvm::make_unique<DIASourceFile>(*this, LocatedFile);
Zachary Turnercffff262015-02-10 21:17:52 +0000277}
278
279std::unique_ptr<IPDBEnumDataStreams> DIASession::getDebugStreams() const {
280 CComPtr<IDiaEnumDebugStreams> DiaEnumerator;
281 if (S_OK != Session->getEnumDebugStreams(&DiaEnumerator))
282 return nullptr;
283
Zachary Turnerbe6d1e42015-02-10 23:46:48 +0000284 return llvm::make_unique<DIAEnumDebugStreams>(DiaEnumerator);
Zachary Turnercffff262015-02-10 21:17:52 +0000285}