]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - contrib/llvm/lib/DebugInfo/PDB/DIA/DIASession.cpp
Merge llvm, clang, compiler-rt, libc++, libunwind, lld, lldb and openmp
[FreeBSD/FreeBSD.git] / contrib / llvm / lib / DebugInfo / PDB / DIA / DIASession.cpp
1 //===- 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 //===----------------------------------------------------------------------===//
9 #include "llvm/DebugInfo/PDB/DIA/DIASession.h"
10 #include "llvm/ADT/STLExtras.h"
11 #include "llvm/DebugInfo/PDB/DIA/DIAEnumDebugStreams.h"
12 #include "llvm/DebugInfo/PDB/DIA/DIAEnumFrameData.h"
13 #include "llvm/DebugInfo/PDB/DIA/DIAEnumInjectedSources.h"
14 #include "llvm/DebugInfo/PDB/DIA/DIAEnumLineNumbers.h"
15 #include "llvm/DebugInfo/PDB/DIA/DIAEnumSectionContribs.h"
16 #include "llvm/DebugInfo/PDB/DIA/DIAEnumSourceFiles.h"
17 #include "llvm/DebugInfo/PDB/DIA/DIAEnumTables.h"
18 #include "llvm/DebugInfo/PDB/DIA/DIAError.h"
19 #include "llvm/DebugInfo/PDB/DIA/DIARawSymbol.h"
20 #include "llvm/DebugInfo/PDB/DIA/DIASourceFile.h"
21 #include "llvm/DebugInfo/PDB/DIA/DIASupport.h"
22 #include "llvm/DebugInfo/PDB/GenericError.h"
23 #include "llvm/DebugInfo/PDB/PDB.h"
24 #include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h"
25 #include "llvm/DebugInfo/PDB/PDBSymbolExe.h"
26 #include "llvm/Support/ConvertUTF.h"
27 #include "llvm/Support/Format.h"
28 #include "llvm/Support/FormatVariadic.h"
29 #include "llvm/Support/raw_ostream.h"
30
31 using namespace llvm;
32 using namespace llvm::pdb;
33
34 template <typename... Ts>
35 static Error ErrorFromHResult(HRESULT Result, const char *Str, Ts &&... Args) {
36   SmallString<64> MessageStorage;
37   StringRef Context;
38   if (sizeof...(Args) > 0) {
39     MessageStorage = formatv(Str, std::forward<Ts>(Args)...).str();
40     Context = MessageStorage;
41   } else
42     Context = Str;
43
44   switch (Result) {
45   case E_PDB_NOT_FOUND:
46     return errorCodeToError(std::error_code(ENOENT, std::generic_category()));
47   case E_PDB_FORMAT:
48     return make_error<DIAError>(dia_error_code::invalid_file_format, Context);
49   case E_INVALIDARG:
50     return make_error<DIAError>(dia_error_code::invalid_parameter, Context);
51   case E_UNEXPECTED:
52     return make_error<DIAError>(dia_error_code::already_loaded, Context);
53   case E_PDB_INVALID_SIG:
54   case E_PDB_INVALID_AGE:
55     return make_error<DIAError>(dia_error_code::debug_info_mismatch, Context);
56   default: {
57     std::string S;
58     raw_string_ostream OS(S);
59     OS << "HRESULT: " << format_hex(static_cast<DWORD>(Result), 10, true)
60        << ": " << Context;
61     return make_error<DIAError>(dia_error_code::unspecified, OS.str());
62   }
63   }
64 }
65
66 static Error LoadDIA(CComPtr<IDiaDataSource> &DiaDataSource) {
67   if (SUCCEEDED(CoCreateInstance(CLSID_DiaSource, nullptr, CLSCTX_INPROC_SERVER,
68                                  IID_IDiaDataSource,
69                                  reinterpret_cast<LPVOID *>(&DiaDataSource))))
70     return Error::success();
71
72 // If the CoCreateInstance call above failed, msdia*.dll is not registered.
73 // Try loading the DLL corresponding to the #included DIA SDK.
74 #if !defined(_MSC_VER)
75   return llvm::make_error<PDBError>(pdb_error_code::dia_failed_loading);
76 #else
77   const wchar_t *msdia_dll = nullptr;
78 #if _MSC_VER >= 1900 && _MSC_VER < 2000
79   msdia_dll = L"msdia140.dll"; // VS2015
80 #elif _MSC_VER >= 1800
81   msdia_dll = L"msdia120.dll"; // VS2013
82 #else
83 #error "Unknown Visual Studio version."
84 #endif
85
86   HRESULT HR;
87   if (FAILED(HR = NoRegCoCreate(msdia_dll, CLSID_DiaSource, IID_IDiaDataSource,
88                                 reinterpret_cast<LPVOID *>(&DiaDataSource))))
89     return ErrorFromHResult(HR, "Calling NoRegCoCreate");
90   return Error::success();
91 #endif
92 }
93
94 DIASession::DIASession(CComPtr<IDiaSession> DiaSession) : Session(DiaSession) {}
95
96 Error DIASession::createFromPdb(StringRef Path,
97                                 std::unique_ptr<IPDBSession> &Session) {
98   CComPtr<IDiaDataSource> DiaDataSource;
99   CComPtr<IDiaSession> DiaSession;
100
101   // We assume that CoInitializeEx has already been called by the executable.
102   if (auto E = LoadDIA(DiaDataSource))
103     return E;
104
105   llvm::SmallVector<UTF16, 128> Path16;
106   if (!llvm::convertUTF8ToUTF16String(Path, Path16))
107     return make_error<PDBError>(pdb_error_code::invalid_utf8_path, Path);
108
109   const wchar_t *Path16Str = reinterpret_cast<const wchar_t *>(Path16.data());
110   HRESULT HR;
111   if (FAILED(HR = DiaDataSource->loadDataFromPdb(Path16Str))) {
112     return ErrorFromHResult(HR, "Calling loadDataFromPdb {0}", Path);
113   }
114
115   if (FAILED(HR = DiaDataSource->openSession(&DiaSession)))
116     return ErrorFromHResult(HR, "Calling openSession");
117
118   Session.reset(new DIASession(DiaSession));
119   return Error::success();
120 }
121
122 Error DIASession::createFromExe(StringRef Path,
123                                 std::unique_ptr<IPDBSession> &Session) {
124   CComPtr<IDiaDataSource> DiaDataSource;
125   CComPtr<IDiaSession> DiaSession;
126
127   // We assume that CoInitializeEx has already been called by the executable.
128   if (auto EC = LoadDIA(DiaDataSource))
129     return EC;
130
131   llvm::SmallVector<UTF16, 128> Path16;
132   if (!llvm::convertUTF8ToUTF16String(Path, Path16))
133     return make_error<PDBError>(pdb_error_code::invalid_utf8_path, Path);
134
135   const wchar_t *Path16Str = reinterpret_cast<const wchar_t *>(Path16.data());
136   HRESULT HR;
137   if (FAILED(HR = DiaDataSource->loadDataForExe(Path16Str, nullptr, nullptr)))
138     return ErrorFromHResult(HR, "Calling loadDataForExe");
139
140   if (FAILED(HR = DiaDataSource->openSession(&DiaSession)))
141     return ErrorFromHResult(HR, "Calling openSession");
142
143   Session.reset(new DIASession(DiaSession));
144   return Error::success();
145 }
146
147 uint64_t DIASession::getLoadAddress() const {
148   uint64_t LoadAddress;
149   bool success = (S_OK == Session->get_loadAddress(&LoadAddress));
150   return (success) ? LoadAddress : 0;
151 }
152
153 bool DIASession::setLoadAddress(uint64_t Address) {
154   return (S_OK == Session->put_loadAddress(Address));
155 }
156
157 std::unique_ptr<PDBSymbolExe> DIASession::getGlobalScope() {
158   CComPtr<IDiaSymbol> GlobalScope;
159   if (S_OK != Session->get_globalScope(&GlobalScope))
160     return nullptr;
161
162   auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, GlobalScope);
163   auto PdbSymbol(PDBSymbol::create(*this, std::move(RawSymbol)));
164   std::unique_ptr<PDBSymbolExe> ExeSymbol(
165       static_cast<PDBSymbolExe *>(PdbSymbol.release()));
166   return ExeSymbol;
167 }
168
169 bool DIASession::addressForVA(uint64_t VA, uint32_t &Section,
170                               uint32_t &Offset) const {
171   DWORD ArgSection, ArgOffset = 0;
172   if (S_OK == Session->addressForVA(VA, &ArgSection, &ArgOffset)) {
173     Section = static_cast<uint32_t>(ArgSection);
174     Offset = static_cast<uint32_t>(ArgOffset);
175     return true;
176   }
177   return false;
178 }
179
180 bool DIASession::addressForRVA(uint32_t RVA, uint32_t &Section,
181                                uint32_t &Offset) const {
182   DWORD ArgSection, ArgOffset = 0;
183   if (S_OK == Session->addressForRVA(RVA, &ArgSection, &ArgOffset)) {
184     Section = static_cast<uint32_t>(ArgSection);
185     Offset = static_cast<uint32_t>(ArgOffset);
186     return true;
187   }
188   return false;
189 }
190
191 std::unique_ptr<PDBSymbol>
192 DIASession::getSymbolById(SymIndexId SymbolId) const {
193   CComPtr<IDiaSymbol> LocatedSymbol;
194   if (S_OK != Session->symbolById(SymbolId, &LocatedSymbol))
195     return nullptr;
196
197   auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, LocatedSymbol);
198   return PDBSymbol::create(*this, std::move(RawSymbol));
199 }
200
201 std::unique_ptr<PDBSymbol>
202 DIASession::findSymbolByAddress(uint64_t Address, PDB_SymType Type) const {
203   enum SymTagEnum EnumVal = static_cast<enum SymTagEnum>(Type);
204
205   CComPtr<IDiaSymbol> Symbol;
206   if (S_OK != Session->findSymbolByVA(Address, EnumVal, &Symbol)) {
207     ULONGLONG LoadAddr = 0;
208     if (S_OK != Session->get_loadAddress(&LoadAddr))
209       return nullptr;
210     DWORD RVA = static_cast<DWORD>(Address - LoadAddr);
211     if (S_OK != Session->findSymbolByRVA(RVA, EnumVal, &Symbol))
212       return nullptr;
213   }
214   auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, Symbol);
215   return PDBSymbol::create(*this, std::move(RawSymbol));
216 }
217
218 std::unique_ptr<PDBSymbol> DIASession::findSymbolByRVA(uint32_t RVA,
219                                                        PDB_SymType Type) const {
220   enum SymTagEnum EnumVal = static_cast<enum SymTagEnum>(Type);
221
222   CComPtr<IDiaSymbol> Symbol;
223   if (S_OK != Session->findSymbolByRVA(RVA, EnumVal, &Symbol))
224     return nullptr;
225
226   auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, Symbol);
227   return PDBSymbol::create(*this, std::move(RawSymbol));
228 }
229
230 std::unique_ptr<PDBSymbol>
231 DIASession::findSymbolBySectOffset(uint32_t Sect, uint32_t Offset,
232                                    PDB_SymType Type) const {
233   enum SymTagEnum EnumVal = static_cast<enum SymTagEnum>(Type);
234
235   CComPtr<IDiaSymbol> Symbol;
236   if (S_OK != Session->findSymbolByAddr(Sect, Offset, EnumVal, &Symbol))
237     return nullptr;
238
239   auto RawSymbol = llvm::make_unique<DIARawSymbol>(*this, Symbol);
240   return PDBSymbol::create(*this, std::move(RawSymbol));
241 }
242
243 std::unique_ptr<IPDBEnumLineNumbers>
244 DIASession::findLineNumbers(const PDBSymbolCompiland &Compiland,
245                             const IPDBSourceFile &File) const {
246   const DIARawSymbol &RawCompiland =
247       static_cast<const DIARawSymbol &>(Compiland.getRawSymbol());
248   const DIASourceFile &RawFile = static_cast<const DIASourceFile &>(File);
249
250   CComPtr<IDiaEnumLineNumbers> LineNumbers;
251   if (S_OK != Session->findLines(RawCompiland.getDiaSymbol(),
252                                  RawFile.getDiaFile(), &LineNumbers))
253     return nullptr;
254
255   return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
256 }
257
258 std::unique_ptr<IPDBEnumLineNumbers>
259 DIASession::findLineNumbersByAddress(uint64_t Address, uint32_t Length) const {
260   CComPtr<IDiaEnumLineNumbers> LineNumbers;
261   if (S_OK != Session->findLinesByVA(Address, Length, &LineNumbers)) {
262     ULONGLONG LoadAddr = 0;
263     if (S_OK != Session->get_loadAddress(&LoadAddr))
264       return nullptr;
265     DWORD RVA = static_cast<DWORD>(Address - LoadAddr);
266     if (S_OK != Session->findLinesByRVA(RVA, Length, &LineNumbers))
267       return nullptr;
268   }
269   return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
270 }
271
272 std::unique_ptr<IPDBEnumLineNumbers>
273 DIASession::findLineNumbersByRVA(uint32_t RVA, uint32_t Length) const {
274   CComPtr<IDiaEnumLineNumbers> LineNumbers;
275   if (S_OK != Session->findLinesByRVA(RVA, Length, &LineNumbers))
276     return nullptr;
277
278   return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
279 }
280
281 std::unique_ptr<IPDBEnumLineNumbers>
282 DIASession::findLineNumbersBySectOffset(uint32_t Section, uint32_t Offset,
283                                         uint32_t Length) const {
284   CComPtr<IDiaEnumLineNumbers> LineNumbers;
285   if (S_OK != Session->findLinesByAddr(Section, Offset, Length, &LineNumbers))
286     return nullptr;
287
288   return llvm::make_unique<DIAEnumLineNumbers>(LineNumbers);
289 }
290
291 std::unique_ptr<IPDBEnumSourceFiles>
292 DIASession::findSourceFiles(const PDBSymbolCompiland *Compiland,
293                             llvm::StringRef Pattern,
294                             PDB_NameSearchFlags Flags) const {
295   IDiaSymbol *DiaCompiland = nullptr;
296   CComBSTR Utf16Pattern;
297   if (!Pattern.empty())
298     Utf16Pattern = CComBSTR(Pattern.data());
299
300   if (Compiland)
301     DiaCompiland = static_cast<const DIARawSymbol &>(Compiland->getRawSymbol())
302                        .getDiaSymbol();
303
304   Flags = static_cast<PDB_NameSearchFlags>(
305       Flags | PDB_NameSearchFlags::NS_FileNameExtMatch);
306   CComPtr<IDiaEnumSourceFiles> SourceFiles;
307   if (S_OK !=
308       Session->findFile(DiaCompiland, Utf16Pattern.m_str, Flags, &SourceFiles))
309     return nullptr;
310   return llvm::make_unique<DIAEnumSourceFiles>(*this, SourceFiles);
311 }
312
313 std::unique_ptr<IPDBSourceFile>
314 DIASession::findOneSourceFile(const PDBSymbolCompiland *Compiland,
315                               llvm::StringRef Pattern,
316                               PDB_NameSearchFlags Flags) const {
317   auto SourceFiles = findSourceFiles(Compiland, Pattern, Flags);
318   if (!SourceFiles || SourceFiles->getChildCount() == 0)
319     return nullptr;
320   return SourceFiles->getNext();
321 }
322
323 std::unique_ptr<IPDBEnumChildren<PDBSymbolCompiland>>
324 DIASession::findCompilandsForSourceFile(llvm::StringRef Pattern,
325                                         PDB_NameSearchFlags Flags) const {
326   auto File = findOneSourceFile(nullptr, Pattern, Flags);
327   if (!File)
328     return nullptr;
329   return File->getCompilands();
330 }
331
332 std::unique_ptr<PDBSymbolCompiland>
333 DIASession::findOneCompilandForSourceFile(llvm::StringRef Pattern,
334                                           PDB_NameSearchFlags Flags) const {
335   auto Compilands = findCompilandsForSourceFile(Pattern, Flags);
336   if (!Compilands || Compilands->getChildCount() == 0)
337     return nullptr;
338   return Compilands->getNext();
339 }
340
341 std::unique_ptr<IPDBEnumSourceFiles> DIASession::getAllSourceFiles() const {
342   CComPtr<IDiaEnumSourceFiles> Files;
343   if (S_OK != Session->findFile(nullptr, nullptr, nsNone, &Files))
344     return nullptr;
345
346   return llvm::make_unique<DIAEnumSourceFiles>(*this, Files);
347 }
348
349 std::unique_ptr<IPDBEnumSourceFiles> DIASession::getSourceFilesForCompiland(
350     const PDBSymbolCompiland &Compiland) const {
351   CComPtr<IDiaEnumSourceFiles> Files;
352
353   const DIARawSymbol &RawSymbol =
354       static_cast<const DIARawSymbol &>(Compiland.getRawSymbol());
355   if (S_OK !=
356       Session->findFile(RawSymbol.getDiaSymbol(), nullptr, nsNone, &Files))
357     return nullptr;
358
359   return llvm::make_unique<DIAEnumSourceFiles>(*this, Files);
360 }
361
362 std::unique_ptr<IPDBSourceFile>
363 DIASession::getSourceFileById(uint32_t FileId) const {
364   CComPtr<IDiaSourceFile> LocatedFile;
365   if (S_OK != Session->findFileById(FileId, &LocatedFile))
366     return nullptr;
367
368   return llvm::make_unique<DIASourceFile>(*this, LocatedFile);
369 }
370
371 std::unique_ptr<IPDBEnumDataStreams> DIASession::getDebugStreams() const {
372   CComPtr<IDiaEnumDebugStreams> DiaEnumerator;
373   if (S_OK != Session->getEnumDebugStreams(&DiaEnumerator))
374     return nullptr;
375
376   return llvm::make_unique<DIAEnumDebugStreams>(DiaEnumerator);
377 }
378
379 std::unique_ptr<IPDBEnumTables> DIASession::getEnumTables() const {
380   CComPtr<IDiaEnumTables> DiaEnumerator;
381   if (S_OK != Session->getEnumTables(&DiaEnumerator))
382     return nullptr;
383
384   return llvm::make_unique<DIAEnumTables>(DiaEnumerator);
385 }
386
387 template <class T> static CComPtr<T> getTableEnumerator(IDiaSession &Session) {
388   CComPtr<T> Enumerator;
389   CComPtr<IDiaEnumTables> ET;
390   CComPtr<IDiaTable> Table;
391   ULONG Count = 0;
392
393   if (Session.getEnumTables(&ET) != S_OK)
394     return nullptr;
395
396   while (ET->Next(1, &Table, &Count) == S_OK && Count == 1) {
397     // There is only one table that matches the given iid
398     if (S_OK == Table->QueryInterface(__uuidof(T), (void **)&Enumerator))
399       break;
400     Table.Release();
401   }
402   return Enumerator;
403 }
404 std::unique_ptr<IPDBEnumInjectedSources>
405 DIASession::getInjectedSources() const {
406   CComPtr<IDiaEnumInjectedSources> Files =
407       getTableEnumerator<IDiaEnumInjectedSources>(*Session);
408   if (!Files)
409     return nullptr;
410
411   return llvm::make_unique<DIAEnumInjectedSources>(Files);
412 }
413
414 std::unique_ptr<IPDBEnumSectionContribs>
415 DIASession::getSectionContribs() const {
416   CComPtr<IDiaEnumSectionContribs> Sections =
417       getTableEnumerator<IDiaEnumSectionContribs>(*Session);
418   if (!Sections)
419     return nullptr;
420
421   return llvm::make_unique<DIAEnumSectionContribs>(*this, Sections);
422 }
423
424 std::unique_ptr<IPDBEnumFrameData>
425 DIASession::getFrameData() const {
426   CComPtr<IDiaEnumFrameData> FD =
427       getTableEnumerator<IDiaEnumFrameData>(*Session);
428   if (!FD)
429     return nullptr;
430
431   return llvm::make_unique<DIAEnumFrameData>(FD);
432 }