1307f5ae8SZachary Turner //===-- SymbolFileNativePDB.cpp ---------------------------------*- C++ -*-===//
2307f5ae8SZachary Turner //
3307f5ae8SZachary Turner //                     The LLVM Compiler Infrastructure
4307f5ae8SZachary Turner //
5307f5ae8SZachary Turner // This file is distributed under the University of Illinois Open Source
6307f5ae8SZachary Turner // License. See LICENSE.TXT for details.
7307f5ae8SZachary Turner //
8307f5ae8SZachary Turner //===----------------------------------------------------------------------===//
9307f5ae8SZachary Turner 
10307f5ae8SZachary Turner #include "SymbolFileNativePDB.h"
11307f5ae8SZachary Turner 
122f7efbc9SZachary Turner #include "clang/AST/Attr.h"
132f7efbc9SZachary Turner #include "clang/AST/CharUnits.h"
142f7efbc9SZachary Turner #include "clang/AST/Decl.h"
152f7efbc9SZachary Turner #include "clang/AST/DeclCXX.h"
16056e4ab4SZachary Turner #include "clang/AST/Type.h"
172f7efbc9SZachary Turner 
18307f5ae8SZachary Turner #include "lldb/Core/Module.h"
19307f5ae8SZachary Turner #include "lldb/Core/PluginManager.h"
209f727950SZachary Turner #include "lldb/Core/StreamBuffer.h"
21056e4ab4SZachary Turner #include "lldb/Core/StreamFile.h"
222f7efbc9SZachary Turner #include "lldb/Symbol/ClangASTContext.h"
232f7efbc9SZachary Turner #include "lldb/Symbol/ClangASTImporter.h"
242f7efbc9SZachary Turner #include "lldb/Symbol/ClangExternalASTSourceCommon.h"
25056e4ab4SZachary Turner #include "lldb/Symbol/ClangUtil.h"
26307f5ae8SZachary Turner #include "lldb/Symbol/CompileUnit.h"
27307f5ae8SZachary Turner #include "lldb/Symbol/LineTable.h"
28307f5ae8SZachary Turner #include "lldb/Symbol/ObjectFile.h"
29307f5ae8SZachary Turner #include "lldb/Symbol/SymbolContext.h"
30307f5ae8SZachary Turner #include "lldb/Symbol/SymbolVendor.h"
319f727950SZachary Turner #include "lldb/Symbol/Variable.h"
329f727950SZachary Turner #include "lldb/Symbol/VariableList.h"
33307f5ae8SZachary Turner 
34307f5ae8SZachary Turner #include "llvm/DebugInfo/CodeView/CVRecord.h"
352f7efbc9SZachary Turner #include "llvm/DebugInfo/CodeView/CVTypeVisitor.h"
36307f5ae8SZachary Turner #include "llvm/DebugInfo/CodeView/DebugLinesSubsection.h"
372f7efbc9SZachary Turner #include "llvm/DebugInfo/CodeView/LazyRandomTypeCollection.h"
38307f5ae8SZachary Turner #include "llvm/DebugInfo/CodeView/RecordName.h"
39307f5ae8SZachary Turner #include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
402f7efbc9SZachary Turner #include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
41307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/DbiStream.h"
42307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/GlobalsStream.h"
43307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/InfoStream.h"
44307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/ModuleDebugStream.h"
45307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/PDBFile.h"
46307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/SymbolStream.h"
472f7efbc9SZachary Turner #include "llvm/DebugInfo/PDB/Native/TpiStream.h"
48307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/PDBTypes.h"
49056e4ab4SZachary Turner #include "llvm/Demangle/MicrosoftDemangle.h"
50307f5ae8SZachary Turner #include "llvm/Object/COFF.h"
51307f5ae8SZachary Turner #include "llvm/Support/Allocator.h"
52307f5ae8SZachary Turner #include "llvm/Support/BinaryStreamReader.h"
53056e4ab4SZachary Turner #include "llvm/Support/Error.h"
54307f5ae8SZachary Turner #include "llvm/Support/ErrorOr.h"
55307f5ae8SZachary Turner #include "llvm/Support/MemoryBuffer.h"
56307f5ae8SZachary Turner 
57*a93458b0SZachary Turner #include "DWARFLocationExpression.h"
58307f5ae8SZachary Turner #include "PdbSymUid.h"
59307f5ae8SZachary Turner #include "PdbUtil.h"
602f7efbc9SZachary Turner #include "UdtRecordCompleter.h"
61307f5ae8SZachary Turner 
62307f5ae8SZachary Turner using namespace lldb;
63307f5ae8SZachary Turner using namespace lldb_private;
642f7efbc9SZachary Turner using namespace npdb;
65307f5ae8SZachary Turner using namespace llvm::codeview;
66307f5ae8SZachary Turner using namespace llvm::pdb;
67307f5ae8SZachary Turner 
68307f5ae8SZachary Turner static lldb::LanguageType TranslateLanguage(PDB_Lang lang) {
69307f5ae8SZachary Turner   switch (lang) {
70307f5ae8SZachary Turner   case PDB_Lang::Cpp:
71307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeC_plus_plus;
72307f5ae8SZachary Turner   case PDB_Lang::C:
73307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeC;
74307f5ae8SZachary Turner   default:
75307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeUnknown;
76307f5ae8SZachary Turner   }
77307f5ae8SZachary Turner }
78307f5ae8SZachary Turner 
79307f5ae8SZachary Turner static std::unique_ptr<PDBFile> loadPDBFile(std::string PdbPath,
80307f5ae8SZachary Turner                                             llvm::BumpPtrAllocator &Allocator) {
81307f5ae8SZachary Turner   llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> ErrorOrBuffer =
82307f5ae8SZachary Turner       llvm::MemoryBuffer::getFile(PdbPath, /*FileSize=*/-1,
83307f5ae8SZachary Turner                                   /*RequiresNullTerminator=*/false);
84307f5ae8SZachary Turner   if (!ErrorOrBuffer)
85307f5ae8SZachary Turner     return nullptr;
86307f5ae8SZachary Turner   std::unique_ptr<llvm::MemoryBuffer> Buffer = std::move(*ErrorOrBuffer);
87307f5ae8SZachary Turner 
88307f5ae8SZachary Turner   llvm::StringRef Path = Buffer->getBufferIdentifier();
89307f5ae8SZachary Turner   auto Stream = llvm::make_unique<llvm::MemoryBufferByteStream>(
90307f5ae8SZachary Turner       std::move(Buffer), llvm::support::little);
91307f5ae8SZachary Turner 
92307f5ae8SZachary Turner   auto File = llvm::make_unique<PDBFile>(Path, std::move(Stream), Allocator);
938040eea9SZachary Turner   if (auto EC = File->parseFileHeaders()) {
948040eea9SZachary Turner     llvm::consumeError(std::move(EC));
95307f5ae8SZachary Turner     return nullptr;
968040eea9SZachary Turner   }
978040eea9SZachary Turner   if (auto EC = File->parseStreamData()) {
988040eea9SZachary Turner     llvm::consumeError(std::move(EC));
99307f5ae8SZachary Turner     return nullptr;
1008040eea9SZachary Turner   }
101307f5ae8SZachary Turner 
102307f5ae8SZachary Turner   return File;
103307f5ae8SZachary Turner }
104307f5ae8SZachary Turner 
105307f5ae8SZachary Turner static std::unique_ptr<PDBFile>
106307f5ae8SZachary Turner loadMatchingPDBFile(std::string exe_path, llvm::BumpPtrAllocator &allocator) {
107307f5ae8SZachary Turner   // Try to find a matching PDB for an EXE.
108307f5ae8SZachary Turner   using namespace llvm::object;
109307f5ae8SZachary Turner   auto expected_binary = createBinary(exe_path);
110307f5ae8SZachary Turner 
111307f5ae8SZachary Turner   // If the file isn't a PE/COFF executable, fail.
112307f5ae8SZachary Turner   if (!expected_binary) {
113307f5ae8SZachary Turner     llvm::consumeError(expected_binary.takeError());
114307f5ae8SZachary Turner     return nullptr;
115307f5ae8SZachary Turner   }
116307f5ae8SZachary Turner   OwningBinary<Binary> binary = std::move(*expected_binary);
117307f5ae8SZachary Turner 
118307f5ae8SZachary Turner   auto *obj = llvm::dyn_cast<llvm::object::COFFObjectFile>(binary.getBinary());
119307f5ae8SZachary Turner   if (!obj)
120307f5ae8SZachary Turner     return nullptr;
121307f5ae8SZachary Turner   const llvm::codeview::DebugInfo *pdb_info = nullptr;
122307f5ae8SZachary Turner 
123307f5ae8SZachary Turner   // If it doesn't have a debug directory, fail.
124307f5ae8SZachary Turner   llvm::StringRef pdb_file;
125307f5ae8SZachary Turner   auto ec = obj->getDebugPDBInfo(pdb_info, pdb_file);
126307f5ae8SZachary Turner   if (ec)
127307f5ae8SZachary Turner     return nullptr;
128307f5ae8SZachary Turner 
129307f5ae8SZachary Turner   // if the file doesn't exist, is not a pdb, or doesn't have a matching guid,
130307f5ae8SZachary Turner   // fail.
131307f5ae8SZachary Turner   llvm::file_magic magic;
132307f5ae8SZachary Turner   ec = llvm::identify_magic(pdb_file, magic);
133307f5ae8SZachary Turner   if (ec || magic != llvm::file_magic::pdb)
134307f5ae8SZachary Turner     return nullptr;
135307f5ae8SZachary Turner   std::unique_ptr<PDBFile> pdb = loadPDBFile(pdb_file, allocator);
1368040eea9SZachary Turner   if (!pdb)
1378040eea9SZachary Turner     return nullptr;
1388040eea9SZachary Turner 
139307f5ae8SZachary Turner   auto expected_info = pdb->getPDBInfoStream();
140307f5ae8SZachary Turner   if (!expected_info) {
141307f5ae8SZachary Turner     llvm::consumeError(expected_info.takeError());
142307f5ae8SZachary Turner     return nullptr;
143307f5ae8SZachary Turner   }
144307f5ae8SZachary Turner   llvm::codeview::GUID guid;
145307f5ae8SZachary Turner   memcpy(&guid, pdb_info->PDB70.Signature, 16);
146307f5ae8SZachary Turner 
147307f5ae8SZachary Turner   if (expected_info->getGuid() != guid)
148307f5ae8SZachary Turner     return nullptr;
149307f5ae8SZachary Turner   return pdb;
150307f5ae8SZachary Turner }
151307f5ae8SZachary Turner 
152307f5ae8SZachary Turner static bool IsFunctionPrologue(const CompilandIndexItem &cci,
153307f5ae8SZachary Turner                                lldb::addr_t addr) {
154307f5ae8SZachary Turner   // FIXME: Implement this.
155307f5ae8SZachary Turner   return false;
156307f5ae8SZachary Turner }
157307f5ae8SZachary Turner 
158307f5ae8SZachary Turner static bool IsFunctionEpilogue(const CompilandIndexItem &cci,
159307f5ae8SZachary Turner                                lldb::addr_t addr) {
160307f5ae8SZachary Turner   // FIXME: Implement this.
161307f5ae8SZachary Turner   return false;
162307f5ae8SZachary Turner }
163307f5ae8SZachary Turner 
1642f7efbc9SZachary Turner static clang::MSInheritanceAttr::Spelling
1652f7efbc9SZachary Turner GetMSInheritance(LazyRandomTypeCollection &tpi, const ClassRecord &record) {
1662f7efbc9SZachary Turner   if (record.DerivationList == TypeIndex::None())
1672f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance;
1682f7efbc9SZachary Turner 
1692f7efbc9SZachary Turner   CVType bases = tpi.getType(record.DerivationList);
1702f7efbc9SZachary Turner   ArgListRecord base_list;
1712f7efbc9SZachary Turner   cantFail(TypeDeserializer::deserializeAs<ArgListRecord>(bases, base_list));
1722f7efbc9SZachary Turner   if (base_list.ArgIndices.empty())
1732f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance;
1742f7efbc9SZachary Turner 
1752f7efbc9SZachary Turner   int base_count = 0;
1762f7efbc9SZachary Turner   for (TypeIndex ti : base_list.ArgIndices) {
1772f7efbc9SZachary Turner     CVType base = tpi.getType(ti);
1782f7efbc9SZachary Turner     if (base.kind() == LF_VBCLASS || base.kind() == LF_IVBCLASS)
1792f7efbc9SZachary Turner       return clang::MSInheritanceAttr::Spelling::Keyword_virtual_inheritance;
1802f7efbc9SZachary Turner     ++base_count;
1812f7efbc9SZachary Turner   }
1822f7efbc9SZachary Turner 
1832f7efbc9SZachary Turner   if (base_count > 1)
1842f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Keyword_multiple_inheritance;
1852f7efbc9SZachary Turner   return clang::MSInheritanceAttr::Keyword_single_inheritance;
1862f7efbc9SZachary Turner }
1872f7efbc9SZachary Turner 
1882f7efbc9SZachary Turner static llvm::StringRef GetSimpleTypeName(SimpleTypeKind kind) {
1892f7efbc9SZachary Turner   switch (kind) {
1902f7efbc9SZachary Turner   case SimpleTypeKind::Boolean128:
1912f7efbc9SZachary Turner   case SimpleTypeKind::Boolean16:
1922f7efbc9SZachary Turner   case SimpleTypeKind::Boolean32:
1932f7efbc9SZachary Turner   case SimpleTypeKind::Boolean64:
1942f7efbc9SZachary Turner   case SimpleTypeKind::Boolean8:
1952f7efbc9SZachary Turner     return "bool";
1962f7efbc9SZachary Turner   case SimpleTypeKind::Byte:
1972f7efbc9SZachary Turner   case SimpleTypeKind::UnsignedCharacter:
1982f7efbc9SZachary Turner     return "unsigned char";
1992f7efbc9SZachary Turner   case SimpleTypeKind::NarrowCharacter:
2002f7efbc9SZachary Turner     return "char";
2012f7efbc9SZachary Turner   case SimpleTypeKind::SignedCharacter:
2022f7efbc9SZachary Turner   case SimpleTypeKind::SByte:
20371ebb721SZachary Turner     return "signed char";
2042f7efbc9SZachary Turner   case SimpleTypeKind::Character16:
2052f7efbc9SZachary Turner     return "char16_t";
2062f7efbc9SZachary Turner   case SimpleTypeKind::Character32:
2072f7efbc9SZachary Turner     return "char32_t";
2082f7efbc9SZachary Turner   case SimpleTypeKind::Complex80:
2092f7efbc9SZachary Turner   case SimpleTypeKind::Complex64:
2102f7efbc9SZachary Turner   case SimpleTypeKind::Complex32:
2112f7efbc9SZachary Turner     return "complex";
2122f7efbc9SZachary Turner   case SimpleTypeKind::Float128:
2132f7efbc9SZachary Turner   case SimpleTypeKind::Float80:
2142f7efbc9SZachary Turner     return "long double";
2152f7efbc9SZachary Turner   case SimpleTypeKind::Float64:
2162f7efbc9SZachary Turner     return "double";
2172f7efbc9SZachary Turner   case SimpleTypeKind::Float32:
2182f7efbc9SZachary Turner     return "float";
2192f7efbc9SZachary Turner   case SimpleTypeKind::Float16:
2202f7efbc9SZachary Turner     return "single";
2212f7efbc9SZachary Turner   case SimpleTypeKind::Int128:
2222f7efbc9SZachary Turner     return "__int128";
2232f7efbc9SZachary Turner   case SimpleTypeKind::Int64:
2242f7efbc9SZachary Turner   case SimpleTypeKind::Int64Quad:
22571ebb721SZachary Turner     return "int64_t";
2262f7efbc9SZachary Turner   case SimpleTypeKind::Int32:
2272f7efbc9SZachary Turner     return "int";
2282f7efbc9SZachary Turner   case SimpleTypeKind::Int16:
2292f7efbc9SZachary Turner     return "short";
2302f7efbc9SZachary Turner   case SimpleTypeKind::UInt128:
2312f7efbc9SZachary Turner     return "unsigned __int128";
2322f7efbc9SZachary Turner   case SimpleTypeKind::UInt64:
2332f7efbc9SZachary Turner   case SimpleTypeKind::UInt64Quad:
23471ebb721SZachary Turner     return "uint64_t";
2352f7efbc9SZachary Turner   case SimpleTypeKind::HResult:
2362f7efbc9SZachary Turner     return "HRESULT";
2372f7efbc9SZachary Turner   case SimpleTypeKind::UInt32:
2382f7efbc9SZachary Turner     return "unsigned";
2392f7efbc9SZachary Turner   case SimpleTypeKind::UInt16:
2402f7efbc9SZachary Turner   case SimpleTypeKind::UInt16Short:
2412f7efbc9SZachary Turner     return "unsigned short";
2422f7efbc9SZachary Turner   case SimpleTypeKind::Int32Long:
2432f7efbc9SZachary Turner     return "long";
2442f7efbc9SZachary Turner   case SimpleTypeKind::UInt32Long:
2452f7efbc9SZachary Turner     return "unsigned long";
2462f7efbc9SZachary Turner   case SimpleTypeKind::Void:
2472f7efbc9SZachary Turner     return "void";
2482f7efbc9SZachary Turner   case SimpleTypeKind::WideCharacter:
2492f7efbc9SZachary Turner     return "wchar_t";
2502f7efbc9SZachary Turner   default:
2512f7efbc9SZachary Turner     return "";
2522f7efbc9SZachary Turner   }
2532f7efbc9SZachary Turner }
2542f7efbc9SZachary Turner 
2552f7efbc9SZachary Turner static bool IsClassRecord(TypeLeafKind kind) {
2562f7efbc9SZachary Turner   switch (kind) {
2572f7efbc9SZachary Turner   case LF_STRUCTURE:
2582f7efbc9SZachary Turner   case LF_CLASS:
2592f7efbc9SZachary Turner   case LF_INTERFACE:
2602f7efbc9SZachary Turner     return true;
2612f7efbc9SZachary Turner   default:
2622f7efbc9SZachary Turner     return false;
2632f7efbc9SZachary Turner   }
2642f7efbc9SZachary Turner }
2652f7efbc9SZachary Turner 
266544a66d8SZachary Turner static bool IsCVarArgsFunction(llvm::ArrayRef<TypeIndex> args) {
267544a66d8SZachary Turner   if (args.empty())
268544a66d8SZachary Turner     return false;
269544a66d8SZachary Turner   return args.back() == TypeIndex::None();
270544a66d8SZachary Turner }
271544a66d8SZachary Turner 
2722f7efbc9SZachary Turner static clang::TagTypeKind TranslateUdtKind(const TagRecord &cr) {
2732f7efbc9SZachary Turner   switch (cr.Kind) {
2742f7efbc9SZachary Turner   case TypeRecordKind::Class:
2752f7efbc9SZachary Turner     return clang::TTK_Class;
2762f7efbc9SZachary Turner   case TypeRecordKind::Struct:
2772f7efbc9SZachary Turner     return clang::TTK_Struct;
2782f7efbc9SZachary Turner   case TypeRecordKind::Union:
2792f7efbc9SZachary Turner     return clang::TTK_Union;
2802f7efbc9SZachary Turner   case TypeRecordKind::Interface:
2812f7efbc9SZachary Turner     return clang::TTK_Interface;
2822f7efbc9SZachary Turner   case TypeRecordKind::Enum:
2832f7efbc9SZachary Turner     return clang::TTK_Enum;
2842f7efbc9SZachary Turner   default:
2852f7efbc9SZachary Turner     lldbassert(false && "Invalid tag record kind!");
2862f7efbc9SZachary Turner     return clang::TTK_Struct;
2872f7efbc9SZachary Turner   }
2882f7efbc9SZachary Turner }
2892f7efbc9SZachary Turner 
290544a66d8SZachary Turner static llvm::Optional<clang::CallingConv>
291544a66d8SZachary Turner TranslateCallingConvention(llvm::codeview::CallingConvention conv) {
292544a66d8SZachary Turner   using CC = llvm::codeview::CallingConvention;
293544a66d8SZachary Turner   switch (conv) {
294544a66d8SZachary Turner 
295544a66d8SZachary Turner   case CC::NearC:
296544a66d8SZachary Turner   case CC::FarC:
297544a66d8SZachary Turner     return clang::CallingConv::CC_C;
298544a66d8SZachary Turner   case CC::NearPascal:
299544a66d8SZachary Turner   case CC::FarPascal:
300544a66d8SZachary Turner     return clang::CallingConv::CC_X86Pascal;
301544a66d8SZachary Turner   case CC::NearFast:
302544a66d8SZachary Turner   case CC::FarFast:
303544a66d8SZachary Turner     return clang::CallingConv::CC_X86FastCall;
304544a66d8SZachary Turner   case CC::NearStdCall:
305544a66d8SZachary Turner   case CC::FarStdCall:
306544a66d8SZachary Turner     return clang::CallingConv::CC_X86StdCall;
307544a66d8SZachary Turner   case CC::ThisCall:
308544a66d8SZachary Turner     return clang::CallingConv::CC_X86ThisCall;
309544a66d8SZachary Turner   case CC::NearVector:
310544a66d8SZachary Turner     return clang::CallingConv::CC_X86VectorCall;
311544a66d8SZachary Turner   default:
312544a66d8SZachary Turner     return llvm::None;
313544a66d8SZachary Turner   }
314544a66d8SZachary Turner }
315544a66d8SZachary Turner 
316307f5ae8SZachary Turner void SymbolFileNativePDB::Initialize() {
317307f5ae8SZachary Turner   PluginManager::RegisterPlugin(GetPluginNameStatic(),
318307f5ae8SZachary Turner                                 GetPluginDescriptionStatic(), CreateInstance,
319307f5ae8SZachary Turner                                 DebuggerInitialize);
320307f5ae8SZachary Turner }
321307f5ae8SZachary Turner 
322307f5ae8SZachary Turner void SymbolFileNativePDB::Terminate() {
323307f5ae8SZachary Turner   PluginManager::UnregisterPlugin(CreateInstance);
324307f5ae8SZachary Turner }
325307f5ae8SZachary Turner 
326b96181c2SZachary Turner void SymbolFileNativePDB::DebuggerInitialize(Debugger &debugger) {}
327307f5ae8SZachary Turner 
328b96181c2SZachary Turner ConstString SymbolFileNativePDB::GetPluginNameStatic() {
329307f5ae8SZachary Turner   static ConstString g_name("native-pdb");
330307f5ae8SZachary Turner   return g_name;
331307f5ae8SZachary Turner }
332307f5ae8SZachary Turner 
333307f5ae8SZachary Turner const char *SymbolFileNativePDB::GetPluginDescriptionStatic() {
334307f5ae8SZachary Turner   return "Microsoft PDB debug symbol cross-platform file reader.";
335307f5ae8SZachary Turner }
336307f5ae8SZachary Turner 
337b96181c2SZachary Turner SymbolFile *SymbolFileNativePDB::CreateInstance(ObjectFile *obj_file) {
338307f5ae8SZachary Turner   return new SymbolFileNativePDB(obj_file);
339307f5ae8SZachary Turner }
340307f5ae8SZachary Turner 
341b96181c2SZachary Turner SymbolFileNativePDB::SymbolFileNativePDB(ObjectFile *object_file)
342307f5ae8SZachary Turner     : SymbolFile(object_file) {}
343307f5ae8SZachary Turner 
344307f5ae8SZachary Turner SymbolFileNativePDB::~SymbolFileNativePDB() {}
345307f5ae8SZachary Turner 
346307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::CalculateAbilities() {
347307f5ae8SZachary Turner   uint32_t abilities = 0;
348307f5ae8SZachary Turner   if (!m_obj_file)
349307f5ae8SZachary Turner     return 0;
350307f5ae8SZachary Turner 
351307f5ae8SZachary Turner   if (!m_index) {
352307f5ae8SZachary Turner     // Lazily load and match the PDB file, but only do this once.
353307f5ae8SZachary Turner     std::unique_ptr<PDBFile> file_up =
354307f5ae8SZachary Turner         loadMatchingPDBFile(m_obj_file->GetFileSpec().GetPath(), m_allocator);
355307f5ae8SZachary Turner 
356307f5ae8SZachary Turner     if (!file_up) {
357307f5ae8SZachary Turner       auto module_sp = m_obj_file->GetModule();
358307f5ae8SZachary Turner       if (!module_sp)
359307f5ae8SZachary Turner         return 0;
360307f5ae8SZachary Turner       // See if any symbol file is specified through `--symfile` option.
361307f5ae8SZachary Turner       FileSpec symfile = module_sp->GetSymbolFileFileSpec();
362307f5ae8SZachary Turner       if (!symfile)
363307f5ae8SZachary Turner         return 0;
364307f5ae8SZachary Turner       file_up = loadPDBFile(symfile.GetPath(), m_allocator);
365307f5ae8SZachary Turner     }
366307f5ae8SZachary Turner 
367307f5ae8SZachary Turner     if (!file_up)
368307f5ae8SZachary Turner       return 0;
369307f5ae8SZachary Turner 
370307f5ae8SZachary Turner     auto expected_index = PdbIndex::create(std::move(file_up));
371307f5ae8SZachary Turner     if (!expected_index) {
372307f5ae8SZachary Turner       llvm::consumeError(expected_index.takeError());
373307f5ae8SZachary Turner       return 0;
374307f5ae8SZachary Turner     }
375307f5ae8SZachary Turner     m_index = std::move(*expected_index);
376307f5ae8SZachary Turner   }
377307f5ae8SZachary Turner   if (!m_index)
378307f5ae8SZachary Turner     return 0;
379307f5ae8SZachary Turner 
380307f5ae8SZachary Turner   // We don't especially have to be precise here.  We only distinguish between
381307f5ae8SZachary Turner   // stripped and not stripped.
382307f5ae8SZachary Turner   abilities = kAllAbilities;
383307f5ae8SZachary Turner 
384307f5ae8SZachary Turner   if (m_index->dbi().isStripped())
385307f5ae8SZachary Turner     abilities &= ~(Blocks | LocalVariables);
386307f5ae8SZachary Turner   return abilities;
387307f5ae8SZachary Turner }
388307f5ae8SZachary Turner 
389307f5ae8SZachary Turner void SymbolFileNativePDB::InitializeObject() {
390307f5ae8SZachary Turner   m_obj_load_address = m_obj_file->GetFileOffset();
391307f5ae8SZachary Turner   m_index->SetLoadAddress(m_obj_load_address);
392307f5ae8SZachary Turner   m_index->ParseSectionContribs();
3932f7efbc9SZachary Turner 
3942f7efbc9SZachary Turner   TypeSystem *ts = GetTypeSystemForLanguage(eLanguageTypeC_plus_plus);
3952f7efbc9SZachary Turner   m_clang = llvm::dyn_cast_or_null<ClangASTContext>(ts);
3962f7efbc9SZachary Turner   m_importer = llvm::make_unique<ClangASTImporter>();
397056e4ab4SZachary Turner 
398056e4ab4SZachary Turner   PreprocessTpiStream();
3992f7efbc9SZachary Turner   lldbassert(m_clang);
400307f5ae8SZachary Turner }
401307f5ae8SZachary Turner 
40203a24052SZachary Turner static llvm::Optional<CVTagRecord>
40303a24052SZachary Turner GetNestedTagRecord(const NestedTypeRecord &Record, const CVTagRecord &parent,
40403a24052SZachary Turner                    TpiStream &tpi) {
40503a24052SZachary Turner   // An LF_NESTTYPE is essentially a nested typedef / using declaration, but it
40603a24052SZachary Turner   // is also used to indicate the primary definition of a nested class.  That is
40703a24052SZachary Turner   // to say, if you have:
40803a24052SZachary Turner   // struct A {
40903a24052SZachary Turner   //   struct B {};
41003a24052SZachary Turner   //   using C = B;
41103a24052SZachary Turner   // };
41203a24052SZachary Turner   // Then in the debug info, this will appear as:
41303a24052SZachary Turner   // LF_STRUCTURE `A::B` [type index = N]
41403a24052SZachary Turner   // LF_STRUCTURE `A`
41503a24052SZachary Turner   //   LF_NESTTYPE [name = `B`, index = N]
41603a24052SZachary Turner   //   LF_NESTTYPE [name = `C`, index = N]
41703a24052SZachary Turner   // In order to accurately reconstruct the decl context hierarchy, we need to
41803a24052SZachary Turner   // know which ones are actual definitions and which ones are just aliases.
41903a24052SZachary Turner 
42003a24052SZachary Turner   // If it's a simple type, then this is something like `using foo = int`.
42103a24052SZachary Turner   if (Record.Type.isSimple())
42203a24052SZachary Turner     return llvm::None;
42303a24052SZachary Turner 
4242af34166SZachary Turner   CVType cvt = tpi.getType(Record.Type);
4252af34166SZachary Turner 
4262af34166SZachary Turner   if (!IsTagRecord(cvt))
4272af34166SZachary Turner     return llvm::None;
4282af34166SZachary Turner 
42903a24052SZachary Turner   // If it's an inner definition, then treat whatever name we have here as a
43003a24052SZachary Turner   // single component of a mangled name.  So we can inject it into the parent's
43103a24052SZachary Turner   // mangled name to see if it matches.
4322af34166SZachary Turner   CVTagRecord child = CVTagRecord::create(cvt);
43303a24052SZachary Turner   std::string qname = parent.asTag().getUniqueName();
43403a24052SZachary Turner   if (qname.size() < 4 || child.asTag().getUniqueName().size() < 4)
43503a24052SZachary Turner     return llvm::None;
43603a24052SZachary Turner 
43703a24052SZachary Turner   // qname[3] is the tag type identifier (struct, class, union, etc).  Since the
43803a24052SZachary Turner   // inner tag type is not necessarily the same as the outer tag type, re-write
43903a24052SZachary Turner   // it to match the inner tag type.
44003a24052SZachary Turner   qname[3] = child.asTag().getUniqueName()[3];
44103a24052SZachary Turner   std::string piece = Record.Name;
44203a24052SZachary Turner   piece.push_back('@');
44303a24052SZachary Turner   qname.insert(4, std::move(piece));
44403a24052SZachary Turner   if (qname != child.asTag().UniqueName)
44503a24052SZachary Turner     return llvm::None;
44603a24052SZachary Turner 
44703a24052SZachary Turner   return std::move(child);
44803a24052SZachary Turner }
44903a24052SZachary Turner 
450056e4ab4SZachary Turner void SymbolFileNativePDB::PreprocessTpiStream() {
451056e4ab4SZachary Turner   LazyRandomTypeCollection &types = m_index->tpi().typeCollection();
452056e4ab4SZachary Turner 
453056e4ab4SZachary Turner   for (auto ti = types.getFirst(); ti; ti = types.getNext(*ti)) {
454056e4ab4SZachary Turner     CVType type = types.getType(*ti);
455056e4ab4SZachary Turner     if (!IsTagRecord(type))
456056e4ab4SZachary Turner       continue;
457056e4ab4SZachary Turner 
458056e4ab4SZachary Turner     CVTagRecord tag = CVTagRecord::create(type);
459056e4ab4SZachary Turner     // We're looking for LF_NESTTYPE records in the field list, so ignore
460056e4ab4SZachary Turner     // forward references (no field list), and anything without a nested class
461056e4ab4SZachary Turner     // (since there won't be any LF_NESTTYPE records).
462056e4ab4SZachary Turner     if (tag.asTag().isForwardRef() || !tag.asTag().containsNestedClass())
463056e4ab4SZachary Turner       continue;
464056e4ab4SZachary Turner 
465056e4ab4SZachary Turner     struct ProcessTpiStream : public TypeVisitorCallbacks {
466056e4ab4SZachary Turner       ProcessTpiStream(PdbIndex &index, TypeIndex parent,
46703a24052SZachary Turner                        const CVTagRecord &parent_cvt,
468056e4ab4SZachary Turner                        llvm::DenseMap<TypeIndex, TypeIndex> &parents)
46903a24052SZachary Turner           : index(index), parents(parents), parent(parent),
47003a24052SZachary Turner             parent_cvt(parent_cvt) {}
471056e4ab4SZachary Turner 
472056e4ab4SZachary Turner       PdbIndex &index;
473056e4ab4SZachary Turner       llvm::DenseMap<TypeIndex, TypeIndex> &parents;
474056e4ab4SZachary Turner       TypeIndex parent;
47503a24052SZachary Turner       const CVTagRecord &parent_cvt;
476056e4ab4SZachary Turner 
477056e4ab4SZachary Turner       llvm::Error visitKnownMember(CVMemberRecord &CVR,
478056e4ab4SZachary Turner                                    NestedTypeRecord &Record) override {
47903a24052SZachary Turner         llvm::Optional<CVTagRecord> tag =
48003a24052SZachary Turner             GetNestedTagRecord(Record, parent_cvt, index.tpi());
48103a24052SZachary Turner         if (!tag)
482056e4ab4SZachary Turner           return llvm::ErrorSuccess();
48303a24052SZachary Turner 
48403a24052SZachary Turner         parents[Record.Type] = parent;
48503a24052SZachary Turner         if (!tag->asTag().isForwardRef())
48603a24052SZachary Turner           return llvm::ErrorSuccess();
48703a24052SZachary Turner 
488056e4ab4SZachary Turner         llvm::Expected<TypeIndex> full_decl =
489056e4ab4SZachary Turner             index.tpi().findFullDeclForForwardRef(Record.Type);
490056e4ab4SZachary Turner         if (!full_decl) {
491056e4ab4SZachary Turner           llvm::consumeError(full_decl.takeError());
492056e4ab4SZachary Turner           return llvm::ErrorSuccess();
493056e4ab4SZachary Turner         }
494056e4ab4SZachary Turner         parents[*full_decl] = parent;
495056e4ab4SZachary Turner         return llvm::ErrorSuccess();
496056e4ab4SZachary Turner       }
497056e4ab4SZachary Turner     };
498056e4ab4SZachary Turner 
499056e4ab4SZachary Turner     CVType field_list = m_index->tpi().getType(tag.asTag().FieldList);
50003a24052SZachary Turner     ProcessTpiStream process(*m_index, *ti, tag, m_parent_types);
501056e4ab4SZachary Turner     llvm::Error error = visitMemberRecordStream(field_list.data(), process);
502056e4ab4SZachary Turner     if (error)
503056e4ab4SZachary Turner       llvm::consumeError(std::move(error));
504056e4ab4SZachary Turner   }
505056e4ab4SZachary Turner }
506056e4ab4SZachary Turner 
507307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::GetNumCompileUnits() {
508307f5ae8SZachary Turner   const DbiModuleList &modules = m_index->dbi().modules();
509307f5ae8SZachary Turner   uint32_t count = modules.getModuleCount();
510307f5ae8SZachary Turner   if (count == 0)
511307f5ae8SZachary Turner     return count;
512307f5ae8SZachary Turner 
513307f5ae8SZachary Turner   // The linker can inject an additional "dummy" compilation unit into the
514307f5ae8SZachary Turner   // PDB. Ignore this special compile unit for our purposes, if it is there.
515307f5ae8SZachary Turner   // It is always the last one.
516307f5ae8SZachary Turner   DbiModuleDescriptor last = modules.getModuleDescriptor(count - 1);
517307f5ae8SZachary Turner   if (last.getModuleName() == "* Linker *")
518307f5ae8SZachary Turner     --count;
519307f5ae8SZachary Turner   return count;
520307f5ae8SZachary Turner }
521307f5ae8SZachary Turner 
5226284aee9SZachary Turner lldb::FunctionSP SymbolFileNativePDB::CreateFunction(PdbCompilandSymId func_id,
523307f5ae8SZachary Turner                                                      const SymbolContext &sc) {
5246284aee9SZachary Turner   const CompilandIndexItem *cci =
5256284aee9SZachary Turner       m_index->compilands().GetCompiland(func_id.modi);
526307f5ae8SZachary Turner   lldbassert(cci);
5276284aee9SZachary Turner   CVSymbol sym_record = cci->m_debug_stream.readSymbolAtOffset(func_id.offset);
528307f5ae8SZachary Turner 
529307f5ae8SZachary Turner   lldbassert(sym_record.kind() == S_LPROC32 || sym_record.kind() == S_GPROC32);
530307f5ae8SZachary Turner   SegmentOffsetLength sol = GetSegmentOffsetAndLength(sym_record);
531307f5ae8SZachary Turner 
532307f5ae8SZachary Turner   auto file_vm_addr = m_index->MakeVirtualAddress(sol.so);
533307f5ae8SZachary Turner   if (file_vm_addr == LLDB_INVALID_ADDRESS || file_vm_addr == 0)
534307f5ae8SZachary Turner     return nullptr;
535307f5ae8SZachary Turner 
536307f5ae8SZachary Turner   AddressRange func_range(file_vm_addr, sol.length,
537307f5ae8SZachary Turner                           sc.module_sp->GetSectionList());
538307f5ae8SZachary Turner   if (!func_range.GetBaseAddress().IsValid())
539307f5ae8SZachary Turner     return nullptr;
540307f5ae8SZachary Turner 
541b96181c2SZachary Turner   Type *func_type = nullptr;
542307f5ae8SZachary Turner 
543307f5ae8SZachary Turner   // FIXME: Resolve types and mangled names.
5449fbf9350SZachary Turner   PdbTypeSymId sig_id(TypeIndex::None(), false);
545307f5ae8SZachary Turner   Mangled mangled(getSymbolName(sym_record));
546307f5ae8SZachary Turner   FunctionSP func_sp = std::make_shared<Function>(
5476284aee9SZachary Turner       sc.comp_unit, toOpaqueUid(func_id), toOpaqueUid(sig_id), mangled,
548307f5ae8SZachary Turner       func_type, func_range);
549307f5ae8SZachary Turner 
550307f5ae8SZachary Turner   sc.comp_unit->AddFunction(func_sp);
551307f5ae8SZachary Turner   return func_sp;
552307f5ae8SZachary Turner }
553307f5ae8SZachary Turner 
554307f5ae8SZachary Turner CompUnitSP
555307f5ae8SZachary Turner SymbolFileNativePDB::CreateCompileUnit(const CompilandIndexItem &cci) {
556307f5ae8SZachary Turner   lldb::LanguageType lang =
557307f5ae8SZachary Turner       cci.m_compile_opts ? TranslateLanguage(cci.m_compile_opts->getLanguage())
558307f5ae8SZachary Turner                          : lldb::eLanguageTypeUnknown;
559307f5ae8SZachary Turner 
560307f5ae8SZachary Turner   LazyBool optimized = eLazyBoolNo;
561307f5ae8SZachary Turner   if (cci.m_compile_opts && cci.m_compile_opts->hasOptimizations())
562307f5ae8SZachary Turner     optimized = eLazyBoolYes;
563307f5ae8SZachary Turner 
564307f5ae8SZachary Turner   llvm::StringRef source_file_name =
565307f5ae8SZachary Turner       m_index->compilands().GetMainSourceFile(cci);
5668f3be7a3SJonas Devlieghere   FileSpec fs(source_file_name);
567307f5ae8SZachary Turner 
568307f5ae8SZachary Turner   CompUnitSP cu_sp =
569307f5ae8SZachary Turner       std::make_shared<CompileUnit>(m_obj_file->GetModule(), nullptr, fs,
5706284aee9SZachary Turner                                     toOpaqueUid(cci.m_id), lang, optimized);
571307f5ae8SZachary Turner 
5726284aee9SZachary Turner   m_obj_file->GetModule()->GetSymbolVendor()->SetCompileUnitAtIndex(
5736284aee9SZachary Turner       cci.m_id.modi, cu_sp);
574307f5ae8SZachary Turner   return cu_sp;
575307f5ae8SZachary Turner }
576307f5ae8SZachary Turner 
5776284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateModifierType(PdbTypeSymId type_id,
5782f7efbc9SZachary Turner                                                      const ModifierRecord &mr) {
5792f7efbc9SZachary Turner   TpiStream &stream = m_index->tpi();
5802f7efbc9SZachary Turner 
5812f7efbc9SZachary Turner   TypeSP t = GetOrCreateType(mr.ModifiedType);
5822f7efbc9SZachary Turner   CompilerType ct = t->GetForwardCompilerType();
5832f7efbc9SZachary Turner   if ((mr.Modifiers & ModifierOptions::Const) != ModifierOptions::None)
5842f7efbc9SZachary Turner     ct = ct.AddConstModifier();
5852f7efbc9SZachary Turner   if ((mr.Modifiers & ModifierOptions::Volatile) != ModifierOptions::None)
5862f7efbc9SZachary Turner     ct = ct.AddVolatileModifier();
5872f7efbc9SZachary Turner   std::string name;
5882f7efbc9SZachary Turner   if (mr.ModifiedType.isSimple())
5892f7efbc9SZachary Turner     name = GetSimpleTypeName(mr.ModifiedType.getSimpleKind());
5902f7efbc9SZachary Turner   else
5912f7efbc9SZachary Turner     name = computeTypeName(stream.typeCollection(), mr.ModifiedType);
5922f7efbc9SZachary Turner   Declaration decl;
5936284aee9SZachary Turner   return std::make_shared<Type>(toOpaqueUid(type_id), m_clang->GetSymbolFile(),
5942f7efbc9SZachary Turner                                 ConstString(name), t->GetByteSize(), nullptr,
5952f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
5962f7efbc9SZachary Turner                                 ct, Type::eResolveStateFull);
5972f7efbc9SZachary Turner }
5982f7efbc9SZachary Turner 
5992f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreatePointerType(
6006284aee9SZachary Turner     PdbTypeSymId type_id, const llvm::codeview::PointerRecord &pr) {
6012f7efbc9SZachary Turner   TypeSP pointee = GetOrCreateType(pr.ReferentType);
602544a66d8SZachary Turner   if (!pointee)
603544a66d8SZachary Turner     return nullptr;
6042f7efbc9SZachary Turner   CompilerType pointee_ct = pointee->GetForwardCompilerType();
6052f7efbc9SZachary Turner   lldbassert(pointee_ct);
6062f7efbc9SZachary Turner   Declaration decl;
6072f7efbc9SZachary Turner 
6082f7efbc9SZachary Turner   if (pr.isPointerToMember()) {
6092f7efbc9SZachary Turner     MemberPointerInfo mpi = pr.getMemberInfo();
6102f7efbc9SZachary Turner     TypeSP class_type = GetOrCreateType(mpi.ContainingType);
6112f7efbc9SZachary Turner 
6122f7efbc9SZachary Turner     CompilerType ct = ClangASTContext::CreateMemberPointerType(
6132f7efbc9SZachary Turner         class_type->GetLayoutCompilerType(), pointee_ct);
6142f7efbc9SZachary Turner 
6152f7efbc9SZachary Turner     return std::make_shared<Type>(
6166284aee9SZachary Turner         toOpaqueUid(type_id), m_clang->GetSymbolFile(), ConstString(),
6172f7efbc9SZachary Turner         pr.getSize(), nullptr, LLDB_INVALID_UID, Type::eEncodingIsUID, decl, ct,
6182f7efbc9SZachary Turner         Type::eResolveStateFull);
6192f7efbc9SZachary Turner   }
6202f7efbc9SZachary Turner 
6212f7efbc9SZachary Turner   CompilerType pointer_ct = pointee_ct;
6222f7efbc9SZachary Turner   if (pr.getMode() == PointerMode::LValueReference)
6232f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetLValueReferenceType();
6242f7efbc9SZachary Turner   else if (pr.getMode() == PointerMode::RValueReference)
6252f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetRValueReferenceType();
6262f7efbc9SZachary Turner   else
6272f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetPointerType();
6282f7efbc9SZachary Turner 
6292f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Const) != PointerOptions::None)
6302f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddConstModifier();
6312f7efbc9SZachary Turner 
6322f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Volatile) != PointerOptions::None)
6332f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddVolatileModifier();
6342f7efbc9SZachary Turner 
6352f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Restrict) != PointerOptions::None)
6362f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddRestrictModifier();
6372f7efbc9SZachary Turner 
6386284aee9SZachary Turner   return std::make_shared<Type>(toOpaqueUid(type_id), m_clang->GetSymbolFile(),
6392f7efbc9SZachary Turner                                 ConstString(), pr.getSize(), nullptr,
6402f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
6412f7efbc9SZachary Turner                                 pointer_ct, Type::eResolveStateFull);
6422f7efbc9SZachary Turner }
6432f7efbc9SZachary Turner 
6442f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateSimpleType(TypeIndex ti) {
6459fbf9350SZachary Turner   uint64_t uid = toOpaqueUid(PdbTypeSymId(ti, false));
646544a66d8SZachary Turner   if (ti == TypeIndex::NullptrT()) {
647544a66d8SZachary Turner     CompilerType ct = m_clang->GetBasicType(eBasicTypeNullPtr);
648544a66d8SZachary Turner     Declaration decl;
6496284aee9SZachary Turner     return std::make_shared<Type>(
6506284aee9SZachary Turner         uid, this, ConstString("std::nullptr_t"), 0, nullptr, LLDB_INVALID_UID,
6516284aee9SZachary Turner         Type::eEncodingIsUID, decl, ct, Type::eResolveStateFull);
652544a66d8SZachary Turner   }
653544a66d8SZachary Turner 
6542f7efbc9SZachary Turner   if (ti.getSimpleMode() != SimpleTypeMode::Direct) {
6552f7efbc9SZachary Turner     TypeSP direct_sp = GetOrCreateType(ti.makeDirect());
6562f7efbc9SZachary Turner     CompilerType ct = direct_sp->GetFullCompilerType();
6572f7efbc9SZachary Turner     ct = ct.GetPointerType();
65871ebb721SZachary Turner     uint32_t pointer_size = 0;
6592f7efbc9SZachary Turner     switch (ti.getSimpleMode()) {
6602f7efbc9SZachary Turner     case SimpleTypeMode::FarPointer32:
6612f7efbc9SZachary Turner     case SimpleTypeMode::NearPointer32:
6622f7efbc9SZachary Turner       pointer_size = 4;
6632f7efbc9SZachary Turner       break;
6642f7efbc9SZachary Turner     case SimpleTypeMode::NearPointer64:
6652f7efbc9SZachary Turner       pointer_size = 8;
6662f7efbc9SZachary Turner       break;
6672f7efbc9SZachary Turner     default:
6682f7efbc9SZachary Turner       // 128-bit and 16-bit pointers unsupported.
6692f7efbc9SZachary Turner       return nullptr;
6702f7efbc9SZachary Turner     }
6712f7efbc9SZachary Turner     Declaration decl;
6726284aee9SZachary Turner     return std::make_shared<Type>(uid, m_clang->GetSymbolFile(), ConstString(),
6736284aee9SZachary Turner                                   pointer_size, nullptr, LLDB_INVALID_UID,
6746284aee9SZachary Turner                                   Type::eEncodingIsUID, decl, ct,
6756284aee9SZachary Turner                                   Type::eResolveStateFull);
6762f7efbc9SZachary Turner   }
6772f7efbc9SZachary Turner 
6782f7efbc9SZachary Turner   if (ti.getSimpleKind() == SimpleTypeKind::NotTranslated)
6792f7efbc9SZachary Turner     return nullptr;
6802f7efbc9SZachary Turner 
6812f7efbc9SZachary Turner   lldb::BasicType bt = GetCompilerTypeForSimpleKind(ti.getSimpleKind());
682544a66d8SZachary Turner   if (bt == lldb::eBasicTypeInvalid)
683544a66d8SZachary Turner     return nullptr;
6842f7efbc9SZachary Turner   CompilerType ct = m_clang->GetBasicType(bt);
6852f7efbc9SZachary Turner   size_t size = GetTypeSizeForSimpleKind(ti.getSimpleKind());
6862f7efbc9SZachary Turner 
6872f7efbc9SZachary Turner   llvm::StringRef type_name = GetSimpleTypeName(ti.getSimpleKind());
6882f7efbc9SZachary Turner 
6892f7efbc9SZachary Turner   Declaration decl;
6906284aee9SZachary Turner   return std::make_shared<Type>(uid, m_clang->GetSymbolFile(),
6912f7efbc9SZachary Turner                                 ConstString(type_name), size, nullptr,
6922f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
6932f7efbc9SZachary Turner                                 ct, Type::eResolveStateFull);
6942f7efbc9SZachary Turner }
6952f7efbc9SZachary Turner 
696056e4ab4SZachary Turner static std::string RenderDemanglerNode(llvm::ms_demangle::Node *n) {
697056e4ab4SZachary Turner   OutputStream OS;
698056e4ab4SZachary Turner   initializeOutputStream(nullptr, nullptr, OS, 1024);
699056e4ab4SZachary Turner   n->output(OS, llvm::ms_demangle::OF_Default);
700056e4ab4SZachary Turner   OS << '\0';
701056e4ab4SZachary Turner   return {OS.getBuffer()};
702056e4ab4SZachary Turner }
703056e4ab4SZachary Turner 
70403a24052SZachary Turner static bool
70503a24052SZachary Turner AnyScopesHaveTemplateParams(llvm::ArrayRef<llvm::ms_demangle::Node *> scopes) {
70603a24052SZachary Turner   for (llvm::ms_demangle::Node *n : scopes) {
70703a24052SZachary Turner     auto *idn = static_cast<llvm::ms_demangle::IdentifierNode *>(n);
70803a24052SZachary Turner     if (idn->TemplateParams)
70903a24052SZachary Turner       return true;
71003a24052SZachary Turner   }
71103a24052SZachary Turner   return false;
71203a24052SZachary Turner }
71303a24052SZachary Turner 
714056e4ab4SZachary Turner std::pair<clang::DeclContext *, std::string>
715056e4ab4SZachary Turner SymbolFileNativePDB::CreateDeclInfoForType(const TagRecord &record,
716056e4ab4SZachary Turner                                            TypeIndex ti) {
717056e4ab4SZachary Turner   llvm::ms_demangle::Demangler demangler;
718056e4ab4SZachary Turner   StringView sv(record.UniqueName.begin(), record.UniqueName.size());
719056e4ab4SZachary Turner   llvm::ms_demangle::TagTypeNode *ttn = demangler.parseTagUniqueName(sv);
720056e4ab4SZachary Turner   llvm::ms_demangle::IdentifierNode *idn =
721056e4ab4SZachary Turner       ttn->QualifiedName->getUnqualifiedIdentifier();
722056e4ab4SZachary Turner   std::string uname = RenderDemanglerNode(idn);
723056e4ab4SZachary Turner 
724056e4ab4SZachary Turner   llvm::ms_demangle::NodeArrayNode *name_components =
725056e4ab4SZachary Turner       ttn->QualifiedName->Components;
726056e4ab4SZachary Turner   llvm::ArrayRef<llvm::ms_demangle::Node *> scopes(name_components->Nodes,
727056e4ab4SZachary Turner                                                    name_components->Count - 1);
728056e4ab4SZachary Turner 
729056e4ab4SZachary Turner   clang::DeclContext *context = m_clang->GetTranslationUnitDecl();
730056e4ab4SZachary Turner 
731056e4ab4SZachary Turner   // If this type doesn't have a parent type in the debug info, then the best we
732056e4ab4SZachary Turner   // can do is to say that it's either a series of namespaces (if the scope is
733056e4ab4SZachary Turner   // non-empty), or the translation unit (if the scope is empty).
734056e4ab4SZachary Turner   auto parent_iter = m_parent_types.find(ti);
735056e4ab4SZachary Turner   if (parent_iter == m_parent_types.end()) {
736056e4ab4SZachary Turner     if (scopes.empty())
737056e4ab4SZachary Turner       return {context, uname};
738056e4ab4SZachary Turner 
73903a24052SZachary Turner     // If there is no parent in the debug info, but some of the scopes have
74003a24052SZachary Turner     // template params, then this is a case of bad debug info.  See, for
74103a24052SZachary Turner     // example, llvm.org/pr39607.  We don't want to create an ambiguity between
74203a24052SZachary Turner     // a NamespaceDecl and a CXXRecordDecl, so instead we create a class at
74303a24052SZachary Turner     // global scope with the fully qualified name.
74403a24052SZachary Turner     if (AnyScopesHaveTemplateParams(scopes))
74503a24052SZachary Turner       return {context, record.Name};
74603a24052SZachary Turner 
747056e4ab4SZachary Turner     for (llvm::ms_demangle::Node *scope : scopes) {
748056e4ab4SZachary Turner       auto *nii = static_cast<llvm::ms_demangle::NamedIdentifierNode *>(scope);
749056e4ab4SZachary Turner       std::string str = RenderDemanglerNode(nii);
750056e4ab4SZachary Turner       context = m_clang->GetUniqueNamespaceDeclaration(str.c_str(), context);
751056e4ab4SZachary Turner     }
752056e4ab4SZachary Turner     return {context, uname};
753056e4ab4SZachary Turner   }
754056e4ab4SZachary Turner 
755056e4ab4SZachary Turner   // Otherwise, all we need to do is get the parent type of this type and
756056e4ab4SZachary Turner   // recurse into our lazy type creation / AST reconstruction logic to get an
757056e4ab4SZachary Turner   // LLDB TypeSP for the parent.  This will cause the AST to automatically get
758056e4ab4SZachary Turner   // the right DeclContext created for any parent.
759056e4ab4SZachary Turner   TypeSP parent = GetOrCreateType(parent_iter->second);
760056e4ab4SZachary Turner   if (!parent)
761056e4ab4SZachary Turner     return {context, uname};
762056e4ab4SZachary Turner   CompilerType parent_ct = parent->GetForwardCompilerType();
763056e4ab4SZachary Turner   clang::QualType qt = ClangUtil::GetCanonicalQualType(parent_ct);
764056e4ab4SZachary Turner   context = clang::TagDecl::castToDeclContext(qt->getAsTagDecl());
765056e4ab4SZachary Turner   return {context, uname};
766056e4ab4SZachary Turner }
767056e4ab4SZachary Turner 
7682f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateClassStructUnion(
7696284aee9SZachary Turner     PdbTypeSymId type_id, const llvm::codeview::TagRecord &record, size_t size,
7702f7efbc9SZachary Turner     clang::TagTypeKind ttk, clang::MSInheritanceAttr::Spelling inheritance) {
7712f7efbc9SZachary Turner 
772056e4ab4SZachary Turner   clang::DeclContext *decl_context = nullptr;
773056e4ab4SZachary Turner   std::string uname;
7746284aee9SZachary Turner   std::tie(decl_context, uname) = CreateDeclInfoForType(record, type_id.index);
7752f7efbc9SZachary Turner 
7762f7efbc9SZachary Turner   lldb::AccessType access =
7772f7efbc9SZachary Turner       (ttk == clang::TTK_Class) ? lldb::eAccessPrivate : lldb::eAccessPublic;
7782f7efbc9SZachary Turner 
7792f7efbc9SZachary Turner   ClangASTMetadata metadata;
7806284aee9SZachary Turner   metadata.SetUserID(toOpaqueUid(type_id));
7812f7efbc9SZachary Turner   metadata.SetIsDynamicCXXType(false);
7822f7efbc9SZachary Turner 
7832f7efbc9SZachary Turner   CompilerType ct =
784056e4ab4SZachary Turner       m_clang->CreateRecordType(decl_context, access, uname.c_str(), ttk,
7852f7efbc9SZachary Turner                                 lldb::eLanguageTypeC_plus_plus, &metadata);
786056e4ab4SZachary Turner 
7872f7efbc9SZachary Turner   lldbassert(ct.IsValid());
7882f7efbc9SZachary Turner 
7892f7efbc9SZachary Turner   clang::CXXRecordDecl *record_decl =
7902f7efbc9SZachary Turner       m_clang->GetAsCXXRecordDecl(ct.GetOpaqueQualType());
7912f7efbc9SZachary Turner   lldbassert(record_decl);
7922f7efbc9SZachary Turner 
7932f7efbc9SZachary Turner   clang::MSInheritanceAttr *attr = clang::MSInheritanceAttr::CreateImplicit(
7942f7efbc9SZachary Turner       *m_clang->getASTContext(), inheritance);
7952f7efbc9SZachary Turner   record_decl->addAttr(attr);
7962f7efbc9SZachary Turner 
7972f7efbc9SZachary Turner   ClangASTContext::StartTagDeclarationDefinition(ct);
7982f7efbc9SZachary Turner 
7992f7efbc9SZachary Turner   // Even if it's possible, don't complete it at this point. Just mark it
8002f7efbc9SZachary Turner   // forward resolved, and if/when LLDB needs the full definition, it can
8012f7efbc9SZachary Turner   // ask us.
8022f7efbc9SZachary Turner   ClangASTContext::SetHasExternalStorage(ct.GetOpaqueQualType(), true);
8032f7efbc9SZachary Turner 
8042f7efbc9SZachary Turner   // FIXME: Search IPI stream for LF_UDT_MOD_SRC_LINE.
8052f7efbc9SZachary Turner   Declaration decl;
8066284aee9SZachary Turner   return std::make_shared<Type>(toOpaqueUid(type_id), m_clang->GetSymbolFile(),
807056e4ab4SZachary Turner                                 ConstString(uname), size, nullptr,
8082f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
8092f7efbc9SZachary Turner                                 ct, Type::eResolveStateForward);
8102f7efbc9SZachary Turner }
8112f7efbc9SZachary Turner 
8126284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbTypeSymId type_id,
8132f7efbc9SZachary Turner                                                 const ClassRecord &cr) {
8142f7efbc9SZachary Turner   clang::TagTypeKind ttk = TranslateUdtKind(cr);
8152f7efbc9SZachary Turner 
8162f7efbc9SZachary Turner   clang::MSInheritanceAttr::Spelling inheritance =
8172f7efbc9SZachary Turner       GetMSInheritance(m_index->tpi().typeCollection(), cr);
8186284aee9SZachary Turner   return CreateClassStructUnion(type_id, cr, cr.getSize(), ttk, inheritance);
8192f7efbc9SZachary Turner }
8202f7efbc9SZachary Turner 
8216284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbTypeSymId type_id,
8222f7efbc9SZachary Turner                                                 const UnionRecord &ur) {
8232f7efbc9SZachary Turner   return CreateClassStructUnion(
8246284aee9SZachary Turner       type_id, ur, ur.getSize(), clang::TTK_Union,
8252f7efbc9SZachary Turner       clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance);
8262f7efbc9SZachary Turner }
8272f7efbc9SZachary Turner 
8286284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbTypeSymId type_id,
8292f7efbc9SZachary Turner                                                 const EnumRecord &er) {
8302af34166SZachary Turner   clang::DeclContext *decl_context = nullptr;
8312af34166SZachary Turner   std::string uname;
8326284aee9SZachary Turner   std::tie(decl_context, uname) = CreateDeclInfoForType(er, type_id.index);
8332f7efbc9SZachary Turner 
8342f7efbc9SZachary Turner   Declaration decl;
8352f7efbc9SZachary Turner   TypeSP underlying_type = GetOrCreateType(er.UnderlyingType);
8362f7efbc9SZachary Turner   CompilerType enum_ct = m_clang->CreateEnumerationType(
8372af34166SZachary Turner       uname.c_str(), decl_context, decl, underlying_type->GetFullCompilerType(),
8382af34166SZachary Turner       er.isScoped());
8392f7efbc9SZachary Turner 
8402f7efbc9SZachary Turner   ClangASTContext::StartTagDeclarationDefinition(enum_ct);
84112abab57SZachary Turner   ClangASTContext::SetHasExternalStorage(enum_ct.GetOpaqueQualType(), true);
8422f7efbc9SZachary Turner 
8432f7efbc9SZachary Turner   // We're just going to forward resolve this for now.  We'll complete
8442f7efbc9SZachary Turner   // it only if the user requests.
8452f7efbc9SZachary Turner   return std::make_shared<lldb_private::Type>(
8466284aee9SZachary Turner       toOpaqueUid(type_id), m_clang->GetSymbolFile(), ConstString(uname),
8472f7efbc9SZachary Turner       underlying_type->GetByteSize(), nullptr, LLDB_INVALID_UID,
8482f7efbc9SZachary Turner       lldb_private::Type::eEncodingIsUID, decl, enum_ct,
8492f7efbc9SZachary Turner       lldb_private::Type::eResolveStateForward);
8502f7efbc9SZachary Turner }
8512f7efbc9SZachary Turner 
8526284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateArrayType(PdbTypeSymId type_id,
853511bff21SZachary Turner                                             const ArrayRecord &ar) {
854511bff21SZachary Turner   TypeSP element_type = GetOrCreateType(ar.ElementType);
855511bff21SZachary Turner   uint64_t element_count = ar.Size / element_type->GetByteSize();
856511bff21SZachary Turner 
857511bff21SZachary Turner   CompilerType element_ct = element_type->GetFullCompilerType();
858511bff21SZachary Turner 
859511bff21SZachary Turner   CompilerType array_ct =
860511bff21SZachary Turner       m_clang->CreateArrayType(element_ct, element_count, false);
861511bff21SZachary Turner 
862511bff21SZachary Turner   Declaration decl;
863511bff21SZachary Turner   TypeSP array_sp = std::make_shared<lldb_private::Type>(
8646284aee9SZachary Turner       toOpaqueUid(type_id), m_clang->GetSymbolFile(), ConstString(), ar.Size,
865511bff21SZachary Turner       nullptr, LLDB_INVALID_UID, lldb_private::Type::eEncodingIsUID, decl,
866511bff21SZachary Turner       array_ct, lldb_private::Type::eResolveStateFull);
867511bff21SZachary Turner   array_sp->SetEncodingType(element_type.get());
868511bff21SZachary Turner   return array_sp;
869511bff21SZachary Turner }
870511bff21SZachary Turner 
8716284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateProcedureType(PdbTypeSymId type_id,
872544a66d8SZachary Turner                                                 const ProcedureRecord &pr) {
873544a66d8SZachary Turner   TpiStream &stream = m_index->tpi();
874544a66d8SZachary Turner   CVType args_cvt = stream.getType(pr.ArgumentList);
875544a66d8SZachary Turner   ArgListRecord args;
876544a66d8SZachary Turner   llvm::cantFail(
877544a66d8SZachary Turner       TypeDeserializer::deserializeAs<ArgListRecord>(args_cvt, args));
878544a66d8SZachary Turner 
879544a66d8SZachary Turner   llvm::ArrayRef<TypeIndex> arg_indices = llvm::makeArrayRef(args.ArgIndices);
880544a66d8SZachary Turner   bool is_variadic = IsCVarArgsFunction(arg_indices);
881544a66d8SZachary Turner   if (is_variadic)
882544a66d8SZachary Turner     arg_indices = arg_indices.drop_back();
883544a66d8SZachary Turner 
884544a66d8SZachary Turner   std::vector<CompilerType> arg_list;
885544a66d8SZachary Turner   arg_list.reserve(arg_list.size());
886544a66d8SZachary Turner 
887544a66d8SZachary Turner   for (TypeIndex arg_index : arg_indices) {
888544a66d8SZachary Turner     TypeSP arg_sp = GetOrCreateType(arg_index);
889544a66d8SZachary Turner     if (!arg_sp)
890544a66d8SZachary Turner       return nullptr;
891544a66d8SZachary Turner     arg_list.push_back(arg_sp->GetFullCompilerType());
892544a66d8SZachary Turner   }
893544a66d8SZachary Turner 
894544a66d8SZachary Turner   TypeSP return_type_sp = GetOrCreateType(pr.ReturnType);
895544a66d8SZachary Turner   if (!return_type_sp)
896544a66d8SZachary Turner     return nullptr;
897544a66d8SZachary Turner 
898544a66d8SZachary Turner   llvm::Optional<clang::CallingConv> cc =
899544a66d8SZachary Turner       TranslateCallingConvention(pr.CallConv);
900544a66d8SZachary Turner   if (!cc)
901544a66d8SZachary Turner     return nullptr;
902544a66d8SZachary Turner 
903544a66d8SZachary Turner   CompilerType return_ct = return_type_sp->GetFullCompilerType();
904544a66d8SZachary Turner   CompilerType func_sig_ast_type = m_clang->CreateFunctionType(
905544a66d8SZachary Turner       return_ct, arg_list.data(), arg_list.size(), is_variadic, 0, *cc);
906544a66d8SZachary Turner 
907544a66d8SZachary Turner   Declaration decl;
908544a66d8SZachary Turner   return std::make_shared<lldb_private::Type>(
9096284aee9SZachary Turner       toOpaqueUid(type_id), this, ConstString(), 0, nullptr, LLDB_INVALID_UID,
910544a66d8SZachary Turner       lldb_private::Type::eEncodingIsUID, decl, func_sig_ast_type,
911544a66d8SZachary Turner       lldb_private::Type::eResolveStateFull);
912544a66d8SZachary Turner }
913544a66d8SZachary Turner 
9146284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateType(PdbTypeSymId type_id) {
9156284aee9SZachary Turner   if (type_id.index.isSimple())
9166284aee9SZachary Turner     return CreateSimpleType(type_id.index);
9172f7efbc9SZachary Turner 
9186284aee9SZachary Turner   TpiStream &stream = type_id.is_ipi ? m_index->ipi() : m_index->tpi();
9196284aee9SZachary Turner   CVType cvt = stream.getType(type_id.index);
9202f7efbc9SZachary Turner 
9212f7efbc9SZachary Turner   if (cvt.kind() == LF_MODIFIER) {
9222f7efbc9SZachary Turner     ModifierRecord modifier;
9232f7efbc9SZachary Turner     llvm::cantFail(
9242f7efbc9SZachary Turner         TypeDeserializer::deserializeAs<ModifierRecord>(cvt, modifier));
9256284aee9SZachary Turner     return CreateModifierType(type_id, modifier);
9262f7efbc9SZachary Turner   }
9272f7efbc9SZachary Turner 
9282f7efbc9SZachary Turner   if (cvt.kind() == LF_POINTER) {
9292f7efbc9SZachary Turner     PointerRecord pointer;
9302f7efbc9SZachary Turner     llvm::cantFail(
9312f7efbc9SZachary Turner         TypeDeserializer::deserializeAs<PointerRecord>(cvt, pointer));
9326284aee9SZachary Turner     return CreatePointerType(type_id, pointer);
9332f7efbc9SZachary Turner   }
9342f7efbc9SZachary Turner 
9352f7efbc9SZachary Turner   if (IsClassRecord(cvt.kind())) {
9362f7efbc9SZachary Turner     ClassRecord cr;
9372f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ClassRecord>(cvt, cr));
9386284aee9SZachary Turner     return CreateTagType(type_id, cr);
9392f7efbc9SZachary Turner   }
9402f7efbc9SZachary Turner 
9412f7efbc9SZachary Turner   if (cvt.kind() == LF_ENUM) {
9422f7efbc9SZachary Turner     EnumRecord er;
9432f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<EnumRecord>(cvt, er));
9446284aee9SZachary Turner     return CreateTagType(type_id, er);
9452f7efbc9SZachary Turner   }
9462f7efbc9SZachary Turner 
9472f7efbc9SZachary Turner   if (cvt.kind() == LF_UNION) {
9482f7efbc9SZachary Turner     UnionRecord ur;
9492f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<UnionRecord>(cvt, ur));
9506284aee9SZachary Turner     return CreateTagType(type_id, ur);
9512f7efbc9SZachary Turner   }
9522f7efbc9SZachary Turner 
953511bff21SZachary Turner   if (cvt.kind() == LF_ARRAY) {
954511bff21SZachary Turner     ArrayRecord ar;
955511bff21SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ArrayRecord>(cvt, ar));
9566284aee9SZachary Turner     return CreateArrayType(type_id, ar);
957511bff21SZachary Turner   }
958511bff21SZachary Turner 
959544a66d8SZachary Turner   if (cvt.kind() == LF_PROCEDURE) {
960544a66d8SZachary Turner     ProcedureRecord pr;
961544a66d8SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ProcedureRecord>(cvt, pr));
9626284aee9SZachary Turner     return CreateProcedureType(type_id, pr);
963544a66d8SZachary Turner   }
964544a66d8SZachary Turner 
9652f7efbc9SZachary Turner   return nullptr;
9662f7efbc9SZachary Turner }
9672f7efbc9SZachary Turner 
9686284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateAndCacheType(PdbTypeSymId type_id) {
9692f7efbc9SZachary Turner   // If they search for a UDT which is a forward ref, try and resolve the full
9702f7efbc9SZachary Turner   // decl and just map the forward ref uid to the full decl record.
9716284aee9SZachary Turner   llvm::Optional<PdbTypeSymId> full_decl_uid;
9726284aee9SZachary Turner   if (IsForwardRefUdt(type_id, m_index->tpi())) {
9736284aee9SZachary Turner     auto expected_full_ti =
9746284aee9SZachary Turner         m_index->tpi().findFullDeclForForwardRef(type_id.index);
9752f7efbc9SZachary Turner     if (!expected_full_ti)
9762f7efbc9SZachary Turner       llvm::consumeError(expected_full_ti.takeError());
9776284aee9SZachary Turner     else if (*expected_full_ti != type_id.index) {
9789fbf9350SZachary Turner       full_decl_uid = PdbTypeSymId(*expected_full_ti, false);
9792f7efbc9SZachary Turner 
9802f7efbc9SZachary Turner       // It's possible that a lookup would occur for the full decl causing it
9812f7efbc9SZachary Turner       // to be cached, then a second lookup would occur for the forward decl.
9822f7efbc9SZachary Turner       // We don't want to create a second full decl, so make sure the full
9832f7efbc9SZachary Turner       // decl hasn't already been cached.
9846284aee9SZachary Turner       auto full_iter = m_types.find(toOpaqueUid(*full_decl_uid));
9852f7efbc9SZachary Turner       if (full_iter != m_types.end()) {
9862f7efbc9SZachary Turner         TypeSP result = full_iter->second;
9872f7efbc9SZachary Turner         // Map the forward decl to the TypeSP for the full decl so we can take
9882f7efbc9SZachary Turner         // the fast path next time.
9896284aee9SZachary Turner         m_types[toOpaqueUid(type_id)] = result;
9902f7efbc9SZachary Turner         return result;
9912f7efbc9SZachary Turner       }
9922f7efbc9SZachary Turner     }
9932f7efbc9SZachary Turner   }
9942f7efbc9SZachary Turner 
9956284aee9SZachary Turner   PdbTypeSymId best_decl_id = full_decl_uid ? *full_decl_uid : type_id;
9966284aee9SZachary Turner   TypeSP result = CreateType(best_decl_id);
997544a66d8SZachary Turner   if (!result)
998544a66d8SZachary Turner     return nullptr;
9996284aee9SZachary Turner 
10006284aee9SZachary Turner   uint64_t best_uid = toOpaqueUid(best_decl_id);
10016284aee9SZachary Turner   m_types[best_uid] = result;
10022f7efbc9SZachary Turner   // If we had both a forward decl and a full decl, make both point to the new
10032f7efbc9SZachary Turner   // type.
10042f7efbc9SZachary Turner   if (full_decl_uid)
10056284aee9SZachary Turner     m_types[toOpaqueUid(type_id)] = result;
10062f7efbc9SZachary Turner 
10076284aee9SZachary Turner   if (IsTagRecord(best_decl_id, m_index->tpi())) {
10082f7efbc9SZachary Turner     clang::TagDecl *record_decl =
10092f7efbc9SZachary Turner         m_clang->GetAsTagDecl(result->GetForwardCompilerType());
10102f7efbc9SZachary Turner     lldbassert(record_decl);
10112f7efbc9SZachary Turner 
10126284aee9SZachary Turner     m_uid_to_decl[best_uid] = record_decl;
10132f7efbc9SZachary Turner     m_decl_to_status[record_decl] =
10146284aee9SZachary Turner         DeclStatus(best_uid, Type::eResolveStateForward);
10152f7efbc9SZachary Turner   }
10162f7efbc9SZachary Turner   return result;
10172f7efbc9SZachary Turner }
10182f7efbc9SZachary Turner 
10196284aee9SZachary Turner TypeSP SymbolFileNativePDB::GetOrCreateType(PdbTypeSymId type_id) {
10202f7efbc9SZachary Turner   // We can't use try_emplace / overwrite here because the process of creating
10212f7efbc9SZachary Turner   // a type could create nested types, which could invalidate iterators.  So
10222f7efbc9SZachary Turner   // we have to do a 2-phase lookup / insert.
10236284aee9SZachary Turner   auto iter = m_types.find(toOpaqueUid(type_id));
10242f7efbc9SZachary Turner   if (iter != m_types.end())
10252f7efbc9SZachary Turner     return iter->second;
10262f7efbc9SZachary Turner 
10276284aee9SZachary Turner   return CreateAndCacheType(type_id);
10282f7efbc9SZachary Turner }
10292f7efbc9SZachary Turner 
10306284aee9SZachary Turner VariableSP SymbolFileNativePDB::CreateGlobalVariable(PdbGlobalSymId var_id) {
10316284aee9SZachary Turner   CVSymbol sym = m_index->symrecords().readRecord(var_id.offset);
10322af34166SZachary Turner   if (sym.kind() == S_CONSTANT)
10336284aee9SZachary Turner     return CreateConstantSymbol(var_id, sym);
10342af34166SZachary Turner 
10359f727950SZachary Turner   lldb::ValueType scope = eValueTypeInvalid;
10369f727950SZachary Turner   TypeIndex ti;
10379f727950SZachary Turner   llvm::StringRef name;
10389f727950SZachary Turner   lldb::addr_t addr = 0;
10399f727950SZachary Turner   uint16_t section = 0;
10409f727950SZachary Turner   uint32_t offset = 0;
10419f727950SZachary Turner   bool is_external = false;
10429f727950SZachary Turner   switch (sym.kind()) {
10439f727950SZachary Turner   case S_GDATA32:
10449f727950SZachary Turner     is_external = true;
10459f727950SZachary Turner     LLVM_FALLTHROUGH;
10469f727950SZachary Turner   case S_LDATA32: {
10479f727950SZachary Turner     DataSym ds(sym.kind());
10489f727950SZachary Turner     llvm::cantFail(SymbolDeserializer::deserializeAs<DataSym>(sym, ds));
10499f727950SZachary Turner     ti = ds.Type;
10509f727950SZachary Turner     scope = (sym.kind() == S_GDATA32) ? eValueTypeVariableGlobal
10519f727950SZachary Turner                                       : eValueTypeVariableStatic;
10529f727950SZachary Turner     name = ds.Name;
10539f727950SZachary Turner     section = ds.Segment;
10549f727950SZachary Turner     offset = ds.DataOffset;
10559f727950SZachary Turner     addr = m_index->MakeVirtualAddress(ds.Segment, ds.DataOffset);
10569f727950SZachary Turner     break;
10579f727950SZachary Turner   }
10589f727950SZachary Turner   case S_GTHREAD32:
10599f727950SZachary Turner     is_external = true;
10609f727950SZachary Turner     LLVM_FALLTHROUGH;
10619f727950SZachary Turner   case S_LTHREAD32: {
10629f727950SZachary Turner     ThreadLocalDataSym tlds(sym.kind());
10639f727950SZachary Turner     llvm::cantFail(
10649f727950SZachary Turner         SymbolDeserializer::deserializeAs<ThreadLocalDataSym>(sym, tlds));
10659f727950SZachary Turner     ti = tlds.Type;
10669f727950SZachary Turner     name = tlds.Name;
10679f727950SZachary Turner     section = tlds.Segment;
10689f727950SZachary Turner     offset = tlds.DataOffset;
10699f727950SZachary Turner     addr = m_index->MakeVirtualAddress(tlds.Segment, tlds.DataOffset);
10709f727950SZachary Turner     scope = eValueTypeVariableThreadLocal;
10719f727950SZachary Turner     break;
10729f727950SZachary Turner   }
10739f727950SZachary Turner   default:
10749f727950SZachary Turner     llvm_unreachable("unreachable!");
10759f727950SZachary Turner   }
10769f727950SZachary Turner 
10779f727950SZachary Turner   CompUnitSP comp_unit;
10789f727950SZachary Turner   llvm::Optional<uint16_t> modi = m_index->GetModuleIndexForVa(addr);
10799f727950SZachary Turner   if (modi) {
10806284aee9SZachary Turner     CompilandIndexItem &cci = m_index->compilands().GetOrCreateCompiland(*modi);
10819f727950SZachary Turner     comp_unit = GetOrCreateCompileUnit(cci);
10829f727950SZachary Turner   }
10839f727950SZachary Turner 
10849f727950SZachary Turner   Declaration decl;
10859fbf9350SZachary Turner   PdbTypeSymId tid(ti, false);
10869f727950SZachary Turner   SymbolFileTypeSP type_sp =
10876284aee9SZachary Turner       std::make_shared<SymbolFileType>(*this, toOpaqueUid(tid));
10889f727950SZachary Turner   Variable::RangeList ranges;
10899f727950SZachary Turner 
10909f727950SZachary Turner   DWARFExpression location = MakeGlobalLocationExpression(
10919f727950SZachary Turner       section, offset, GetObjectFile()->GetModule());
10929f727950SZachary Turner 
10939f727950SZachary Turner   std::string global_name("::");
10949f727950SZachary Turner   global_name += name;
10959f727950SZachary Turner   VariableSP var_sp = std::make_shared<Variable>(
10966284aee9SZachary Turner       toOpaqueUid(var_id), name.str().c_str(), global_name.c_str(), type_sp,
10979f727950SZachary Turner       scope, comp_unit.get(), ranges, &decl, location, is_external, false,
10989f727950SZachary Turner       false);
10999f727950SZachary Turner   var_sp->SetLocationIsConstantValueData(false);
11009f727950SZachary Turner 
11019f727950SZachary Turner   return var_sp;
11029f727950SZachary Turner }
11039f727950SZachary Turner 
11042af34166SZachary Turner lldb::VariableSP
11056284aee9SZachary Turner SymbolFileNativePDB::CreateConstantSymbol(PdbGlobalSymId var_id,
11062af34166SZachary Turner                                           const CVSymbol &cvs) {
11072af34166SZachary Turner   TpiStream &tpi = m_index->tpi();
11082af34166SZachary Turner   ConstantSym constant(cvs.kind());
11092af34166SZachary Turner 
11102af34166SZachary Turner   llvm::cantFail(SymbolDeserializer::deserializeAs<ConstantSym>(cvs, constant));
11112af34166SZachary Turner   std::string global_name("::");
11122af34166SZachary Turner   global_name += constant.Name;
11139fbf9350SZachary Turner   PdbTypeSymId tid(constant.Type, false);
11142af34166SZachary Turner   SymbolFileTypeSP type_sp =
11156284aee9SZachary Turner       std::make_shared<SymbolFileType>(*this, toOpaqueUid(tid));
11162af34166SZachary Turner 
11172af34166SZachary Turner   Declaration decl;
11182af34166SZachary Turner   Variable::RangeList ranges;
11192af34166SZachary Turner   ModuleSP module = GetObjectFile()->GetModule();
1120*a93458b0SZachary Turner   DWARFExpression location = MakeConstantLocationExpression(
1121*a93458b0SZachary Turner       constant.Type, tpi, constant.Value, module);
11222af34166SZachary Turner 
11232af34166SZachary Turner   VariableSP var_sp = std::make_shared<Variable>(
11246284aee9SZachary Turner       toOpaqueUid(var_id), constant.Name.str().c_str(), global_name.c_str(),
11252af34166SZachary Turner       type_sp, eValueTypeVariableGlobal, module.get(), ranges, &decl, location,
11262af34166SZachary Turner       false, false, false);
11272af34166SZachary Turner   var_sp->SetLocationIsConstantValueData(true);
11282af34166SZachary Turner   return var_sp;
11292af34166SZachary Turner }
11302af34166SZachary Turner 
11316284aee9SZachary Turner VariableSP
11326284aee9SZachary Turner SymbolFileNativePDB::GetOrCreateGlobalVariable(PdbGlobalSymId var_id) {
11336284aee9SZachary Turner   auto emplace_result = m_global_vars.try_emplace(toOpaqueUid(var_id), nullptr);
11349f727950SZachary Turner   if (emplace_result.second)
11356284aee9SZachary Turner     emplace_result.first->second = CreateGlobalVariable(var_id);
11369f727950SZachary Turner 
11379f727950SZachary Turner   return emplace_result.first->second;
11389f727950SZachary Turner }
11399f727950SZachary Turner 
11406284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::GetOrCreateType(TypeIndex ti) {
11419fbf9350SZachary Turner   return GetOrCreateType(PdbTypeSymId(ti, false));
11422f7efbc9SZachary Turner }
11432f7efbc9SZachary Turner 
11446284aee9SZachary Turner FunctionSP SymbolFileNativePDB::GetOrCreateFunction(PdbCompilandSymId func_id,
1145307f5ae8SZachary Turner                                                     const SymbolContext &sc) {
11466284aee9SZachary Turner   auto emplace_result = m_functions.try_emplace(toOpaqueUid(func_id), nullptr);
1147307f5ae8SZachary Turner   if (emplace_result.second)
11486284aee9SZachary Turner     emplace_result.first->second = CreateFunction(func_id, sc);
1149307f5ae8SZachary Turner 
1150307f5ae8SZachary Turner   lldbassert(emplace_result.first->second);
1151307f5ae8SZachary Turner   return emplace_result.first->second;
1152307f5ae8SZachary Turner }
1153307f5ae8SZachary Turner 
1154307f5ae8SZachary Turner CompUnitSP
1155307f5ae8SZachary Turner SymbolFileNativePDB::GetOrCreateCompileUnit(const CompilandIndexItem &cci) {
11566284aee9SZachary Turner 
1157307f5ae8SZachary Turner   auto emplace_result =
11586284aee9SZachary Turner       m_compilands.try_emplace(toOpaqueUid(cci.m_id), nullptr);
1159307f5ae8SZachary Turner   if (emplace_result.second)
1160307f5ae8SZachary Turner     emplace_result.first->second = CreateCompileUnit(cci);
1161307f5ae8SZachary Turner 
1162307f5ae8SZachary Turner   lldbassert(emplace_result.first->second);
1163307f5ae8SZachary Turner   return emplace_result.first->second;
1164307f5ae8SZachary Turner }
1165307f5ae8SZachary Turner 
1166307f5ae8SZachary Turner lldb::CompUnitSP SymbolFileNativePDB::ParseCompileUnitAtIndex(uint32_t index) {
1167307f5ae8SZachary Turner   if (index >= GetNumCompileUnits())
1168307f5ae8SZachary Turner     return CompUnitSP();
1169307f5ae8SZachary Turner   lldbassert(index < UINT16_MAX);
1170307f5ae8SZachary Turner   if (index >= UINT16_MAX)
1171307f5ae8SZachary Turner     return nullptr;
1172307f5ae8SZachary Turner 
1173307f5ae8SZachary Turner   CompilandIndexItem &item = m_index->compilands().GetOrCreateCompiland(index);
1174307f5ae8SZachary Turner 
1175307f5ae8SZachary Turner   return GetOrCreateCompileUnit(item);
1176307f5ae8SZachary Turner }
1177307f5ae8SZachary Turner 
1178b96181c2SZachary Turner lldb::LanguageType
1179b96181c2SZachary Turner SymbolFileNativePDB::ParseCompileUnitLanguage(const SymbolContext &sc) {
1180307f5ae8SZachary Turner   // What fields should I expect to be filled out on the SymbolContext?  Is it
1181307f5ae8SZachary Turner   // safe to assume that `sc.comp_unit` is valid?
1182307f5ae8SZachary Turner   if (!sc.comp_unit)
1183307f5ae8SZachary Turner     return lldb::eLanguageTypeUnknown;
11846284aee9SZachary Turner   PdbSymUid uid(sc.comp_unit->GetID());
11856284aee9SZachary Turner   lldbassert(uid.kind() == PdbSymUidKind::Compiland);
1186307f5ae8SZachary Turner 
11876284aee9SZachary Turner   CompilandIndexItem *item =
11886284aee9SZachary Turner       m_index->compilands().GetCompiland(uid.asCompiland().modi);
1189307f5ae8SZachary Turner   lldbassert(item);
1190307f5ae8SZachary Turner   if (!item->m_compile_opts)
1191307f5ae8SZachary Turner     return lldb::eLanguageTypeUnknown;
1192307f5ae8SZachary Turner 
1193307f5ae8SZachary Turner   return TranslateLanguage(item->m_compile_opts->getLanguage());
1194307f5ae8SZachary Turner }
1195307f5ae8SZachary Turner 
1196b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseCompileUnitFunctions(const SymbolContext &sc) {
1197307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
1198307f5ae8SZachary Turner   return false;
1199307f5ae8SZachary Turner }
1200307f5ae8SZachary Turner 
1201307f5ae8SZachary Turner static bool NeedsResolvedCompileUnit(uint32_t resolve_scope) {
1202307f5ae8SZachary Turner   // If any of these flags are set, we need to resolve the compile unit.
1203307f5ae8SZachary Turner   uint32_t flags = eSymbolContextCompUnit;
1204307f5ae8SZachary Turner   flags |= eSymbolContextVariable;
1205307f5ae8SZachary Turner   flags |= eSymbolContextFunction;
1206307f5ae8SZachary Turner   flags |= eSymbolContextBlock;
1207307f5ae8SZachary Turner   flags |= eSymbolContextLineEntry;
1208307f5ae8SZachary Turner   return (resolve_scope & flags) != 0;
1209307f5ae8SZachary Turner }
1210307f5ae8SZachary Turner 
1211991e4453SZachary Turner uint32_t SymbolFileNativePDB::ResolveSymbolContext(
1212991e4453SZachary Turner     const Address &addr, SymbolContextItem resolve_scope, SymbolContext &sc) {
1213307f5ae8SZachary Turner   uint32_t resolved_flags = 0;
1214307f5ae8SZachary Turner   lldb::addr_t file_addr = addr.GetFileAddress();
1215307f5ae8SZachary Turner 
1216307f5ae8SZachary Turner   if (NeedsResolvedCompileUnit(resolve_scope)) {
1217307f5ae8SZachary Turner     llvm::Optional<uint16_t> modi = m_index->GetModuleIndexForVa(file_addr);
1218307f5ae8SZachary Turner     if (!modi)
1219307f5ae8SZachary Turner       return 0;
12206284aee9SZachary Turner     CompilandIndexItem *cci = m_index->compilands().GetCompiland(*modi);
1221307f5ae8SZachary Turner     if (!cci)
1222307f5ae8SZachary Turner       return 0;
1223307f5ae8SZachary Turner 
1224307f5ae8SZachary Turner     sc.comp_unit = GetOrCreateCompileUnit(*cci).get();
1225307f5ae8SZachary Turner     resolved_flags |= eSymbolContextCompUnit;
1226307f5ae8SZachary Turner   }
1227307f5ae8SZachary Turner 
1228307f5ae8SZachary Turner   if (resolve_scope & eSymbolContextFunction) {
1229307f5ae8SZachary Turner     lldbassert(sc.comp_unit);
1230307f5ae8SZachary Turner     std::vector<SymbolAndUid> matches = m_index->FindSymbolsByVa(file_addr);
1231307f5ae8SZachary Turner     for (const auto &match : matches) {
12326284aee9SZachary Turner       if (match.uid.kind() != PdbSymUidKind::CompilandSym)
1233307f5ae8SZachary Turner         continue;
12346284aee9SZachary Turner       PdbCompilandSymId csid = match.uid.asCompilandSym();
12356284aee9SZachary Turner       CVSymbol cvs = m_index->ReadSymbolRecord(csid);
12366284aee9SZachary Turner       if (CVSymToPDBSym(cvs.kind()) != PDB_SymType::Function)
12376284aee9SZachary Turner         continue;
12386284aee9SZachary Turner       sc.function = GetOrCreateFunction(csid, sc).get();
1239307f5ae8SZachary Turner     }
1240307f5ae8SZachary Turner     resolved_flags |= eSymbolContextFunction;
1241307f5ae8SZachary Turner   }
1242307f5ae8SZachary Turner 
1243307f5ae8SZachary Turner   if (resolve_scope & eSymbolContextLineEntry) {
1244307f5ae8SZachary Turner     lldbassert(sc.comp_unit);
1245307f5ae8SZachary Turner     if (auto *line_table = sc.comp_unit->GetLineTable()) {
1246307f5ae8SZachary Turner       if (line_table->FindLineEntryByAddress(addr, sc.line_entry))
1247307f5ae8SZachary Turner         resolved_flags |= eSymbolContextLineEntry;
1248307f5ae8SZachary Turner     }
1249307f5ae8SZachary Turner   }
1250307f5ae8SZachary Turner 
1251307f5ae8SZachary Turner   return resolved_flags;
1252307f5ae8SZachary Turner }
1253307f5ae8SZachary Turner 
1254307f5ae8SZachary Turner static void AppendLineEntryToSequence(LineTable &table, LineSequence &sequence,
1255307f5ae8SZachary Turner                                       const CompilandIndexItem &cci,
1256307f5ae8SZachary Turner                                       lldb::addr_t base_addr,
1257307f5ae8SZachary Turner                                       uint32_t file_number,
1258307f5ae8SZachary Turner                                       const LineFragmentHeader &block,
1259307f5ae8SZachary Turner                                       const LineNumberEntry &cur) {
1260307f5ae8SZachary Turner   LineInfo cur_info(cur.Flags);
1261307f5ae8SZachary Turner 
1262307f5ae8SZachary Turner   if (cur_info.isAlwaysStepInto() || cur_info.isNeverStepInto())
1263307f5ae8SZachary Turner     return;
1264307f5ae8SZachary Turner 
1265307f5ae8SZachary Turner   uint64_t addr = base_addr + cur.Offset;
1266307f5ae8SZachary Turner 
1267307f5ae8SZachary Turner   bool is_statement = cur_info.isStatement();
1268307f5ae8SZachary Turner   bool is_prologue = IsFunctionPrologue(cci, addr);
1269307f5ae8SZachary Turner   bool is_epilogue = IsFunctionEpilogue(cci, addr);
1270307f5ae8SZachary Turner 
1271307f5ae8SZachary Turner   uint32_t lno = cur_info.getStartLine();
1272307f5ae8SZachary Turner 
1273307f5ae8SZachary Turner   table.AppendLineEntryToSequence(&sequence, addr, lno, 0, file_number,
1274307f5ae8SZachary Turner                                   is_statement, false, is_prologue, is_epilogue,
1275307f5ae8SZachary Turner                                   false);
1276307f5ae8SZachary Turner }
1277307f5ae8SZachary Turner 
1278307f5ae8SZachary Turner static void TerminateLineSequence(LineTable &table,
1279307f5ae8SZachary Turner                                   const LineFragmentHeader &block,
1280307f5ae8SZachary Turner                                   lldb::addr_t base_addr, uint32_t file_number,
1281307f5ae8SZachary Turner                                   uint32_t last_line,
1282307f5ae8SZachary Turner                                   std::unique_ptr<LineSequence> seq) {
1283307f5ae8SZachary Turner   // The end is always a terminal entry, so insert it regardless.
1284307f5ae8SZachary Turner   table.AppendLineEntryToSequence(seq.get(), base_addr + block.CodeSize,
1285307f5ae8SZachary Turner                                   last_line, 0, file_number, false, false,
1286307f5ae8SZachary Turner                                   false, false, true);
1287307f5ae8SZachary Turner   table.InsertSequence(seq.release());
1288307f5ae8SZachary Turner }
1289307f5ae8SZachary Turner 
1290b96181c2SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitLineTable(const SymbolContext &sc) {
1291307f5ae8SZachary Turner   // Unfortunately LLDB is set up to parse the entire compile unit line table
1292307f5ae8SZachary Turner   // all at once, even if all it really needs is line info for a specific
1293307f5ae8SZachary Turner   // function.  In the future it would be nice if it could set the sc.m_function
1294307f5ae8SZachary Turner   // member, and we could only get the line info for the function in question.
1295307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
12966284aee9SZachary Turner   PdbSymUid cu_id(sc.comp_unit->GetID());
12976284aee9SZachary Turner   lldbassert(cu_id.kind() == PdbSymUidKind::Compiland);
12986284aee9SZachary Turner   CompilandIndexItem *cci =
12996284aee9SZachary Turner       m_index->compilands().GetCompiland(cu_id.asCompiland().modi);
1300307f5ae8SZachary Turner   lldbassert(cci);
1301307f5ae8SZachary Turner   auto line_table = llvm::make_unique<LineTable>(sc.comp_unit);
1302307f5ae8SZachary Turner 
1303307f5ae8SZachary Turner   // This is basically a copy of the .debug$S subsections from all original COFF
1304307f5ae8SZachary Turner   // object files merged together with address relocations applied.  We are
1305307f5ae8SZachary Turner   // looking for all DEBUG_S_LINES subsections.
1306307f5ae8SZachary Turner   for (const DebugSubsectionRecord &dssr :
1307307f5ae8SZachary Turner        cci->m_debug_stream.getSubsectionsArray()) {
1308307f5ae8SZachary Turner     if (dssr.kind() != DebugSubsectionKind::Lines)
1309307f5ae8SZachary Turner       continue;
1310307f5ae8SZachary Turner 
1311307f5ae8SZachary Turner     DebugLinesSubsectionRef lines;
1312307f5ae8SZachary Turner     llvm::BinaryStreamReader reader(dssr.getRecordData());
1313307f5ae8SZachary Turner     if (auto EC = lines.initialize(reader)) {
1314307f5ae8SZachary Turner       llvm::consumeError(std::move(EC));
1315307f5ae8SZachary Turner       return false;
1316307f5ae8SZachary Turner     }
1317307f5ae8SZachary Turner 
1318307f5ae8SZachary Turner     const LineFragmentHeader *lfh = lines.header();
1319307f5ae8SZachary Turner     uint64_t virtual_addr =
1320307f5ae8SZachary Turner         m_index->MakeVirtualAddress(lfh->RelocSegment, lfh->RelocOffset);
1321307f5ae8SZachary Turner 
1322307f5ae8SZachary Turner     const auto &checksums = cci->m_strings.checksums().getArray();
1323307f5ae8SZachary Turner     const auto &strings = cci->m_strings.strings();
1324307f5ae8SZachary Turner     for (const LineColumnEntry &group : lines) {
1325307f5ae8SZachary Turner       // Indices in this structure are actually offsets of records in the
1326307f5ae8SZachary Turner       // DEBUG_S_FILECHECKSUMS subsection.  Those entries then have an index
1327307f5ae8SZachary Turner       // into the global PDB string table.
1328307f5ae8SZachary Turner       auto iter = checksums.at(group.NameIndex);
1329307f5ae8SZachary Turner       if (iter == checksums.end())
1330307f5ae8SZachary Turner         continue;
1331307f5ae8SZachary Turner 
1332307f5ae8SZachary Turner       llvm::Expected<llvm::StringRef> efn =
1333307f5ae8SZachary Turner           strings.getString(iter->FileNameOffset);
1334307f5ae8SZachary Turner       if (!efn) {
1335307f5ae8SZachary Turner         llvm::consumeError(efn.takeError());
1336307f5ae8SZachary Turner         continue;
1337307f5ae8SZachary Turner       }
1338307f5ae8SZachary Turner 
1339307f5ae8SZachary Turner       // LLDB wants the index of the file in the list of support files.
1340307f5ae8SZachary Turner       auto fn_iter = llvm::find(cci->m_file_list, *efn);
1341307f5ae8SZachary Turner       lldbassert(fn_iter != cci->m_file_list.end());
1342307f5ae8SZachary Turner       uint32_t file_index = std::distance(cci->m_file_list.begin(), fn_iter);
1343307f5ae8SZachary Turner 
1344307f5ae8SZachary Turner       std::unique_ptr<LineSequence> sequence(
1345307f5ae8SZachary Turner           line_table->CreateLineSequenceContainer());
1346307f5ae8SZachary Turner       lldbassert(!group.LineNumbers.empty());
1347307f5ae8SZachary Turner 
1348307f5ae8SZachary Turner       for (const LineNumberEntry &entry : group.LineNumbers) {
1349307f5ae8SZachary Turner         AppendLineEntryToSequence(*line_table, *sequence, *cci, virtual_addr,
1350307f5ae8SZachary Turner                                   file_index, *lfh, entry);
1351307f5ae8SZachary Turner       }
1352307f5ae8SZachary Turner       LineInfo last_line(group.LineNumbers.back().Flags);
1353307f5ae8SZachary Turner       TerminateLineSequence(*line_table, *lfh, virtual_addr, file_index,
1354307f5ae8SZachary Turner                             last_line.getEndLine(), std::move(sequence));
1355307f5ae8SZachary Turner     }
1356307f5ae8SZachary Turner   }
1357307f5ae8SZachary Turner 
1358307f5ae8SZachary Turner   if (line_table->GetSize() == 0)
1359307f5ae8SZachary Turner     return false;
1360307f5ae8SZachary Turner 
1361307f5ae8SZachary Turner   sc.comp_unit->SetLineTable(line_table.release());
1362307f5ae8SZachary Turner   return true;
1363307f5ae8SZachary Turner }
1364307f5ae8SZachary Turner 
1365b96181c2SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitDebugMacros(const SymbolContext &sc) {
1366307f5ae8SZachary Turner   // PDB doesn't contain information about macros
1367307f5ae8SZachary Turner   return false;
1368307f5ae8SZachary Turner }
1369307f5ae8SZachary Turner 
1370307f5ae8SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitSupportFiles(
1371b96181c2SZachary Turner     const SymbolContext &sc, FileSpecList &support_files) {
1372307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
1373307f5ae8SZachary Turner 
13746284aee9SZachary Turner   PdbSymUid cu_id(sc.comp_unit->GetID());
13756284aee9SZachary Turner   lldbassert(cu_id.kind() == PdbSymUidKind::Compiland);
13766284aee9SZachary Turner   CompilandIndexItem *cci =
13776284aee9SZachary Turner       m_index->compilands().GetCompiland(cu_id.asCompiland().modi);
1378307f5ae8SZachary Turner   lldbassert(cci);
1379307f5ae8SZachary Turner 
1380307f5ae8SZachary Turner   for (llvm::StringRef f : cci->m_file_list) {
1381307f5ae8SZachary Turner     FileSpec::Style style =
1382307f5ae8SZachary Turner         f.startswith("/") ? FileSpec::Style::posix : FileSpec::Style::windows;
13838f3be7a3SJonas Devlieghere     FileSpec spec(f, style);
1384307f5ae8SZachary Turner     support_files.Append(spec);
1385307f5ae8SZachary Turner   }
1386307f5ae8SZachary Turner 
1387307f5ae8SZachary Turner   return true;
1388307f5ae8SZachary Turner }
1389307f5ae8SZachary Turner 
1390307f5ae8SZachary Turner bool SymbolFileNativePDB::ParseImportedModules(
1391b96181c2SZachary Turner     const SymbolContext &sc, std::vector<ConstString> &imported_modules) {
1392307f5ae8SZachary Turner   // PDB does not yet support module debug info
1393307f5ae8SZachary Turner   return false;
1394307f5ae8SZachary Turner }
1395307f5ae8SZachary Turner 
1396b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseFunctionBlocks(const SymbolContext &sc) {
1397307f5ae8SZachary Turner   lldbassert(sc.comp_unit && sc.function);
1398307f5ae8SZachary Turner   return 0;
1399307f5ae8SZachary Turner }
1400307f5ae8SZachary Turner 
14014911023fSZachary Turner void SymbolFileNativePDB::DumpClangAST(Stream &s) {
14024911023fSZachary Turner   if (!m_clang)
14034911023fSZachary Turner     return;
14044911023fSZachary Turner   m_clang->Dump(s);
14054911023fSZachary Turner }
14064911023fSZachary Turner 
14079f727950SZachary Turner uint32_t SymbolFileNativePDB::FindGlobalVariables(
14089f727950SZachary Turner     const ConstString &name, const CompilerDeclContext *parent_decl_ctx,
14099f727950SZachary Turner     uint32_t max_matches, VariableList &variables) {
14109f727950SZachary Turner   using SymbolAndOffset = std::pair<uint32_t, llvm::codeview::CVSymbol>;
14119f727950SZachary Turner 
14129f727950SZachary Turner   std::vector<SymbolAndOffset> results = m_index->globals().findRecordsByName(
14139f727950SZachary Turner       name.GetStringRef(), m_index->symrecords());
14149f727950SZachary Turner   for (const SymbolAndOffset &result : results) {
14159f727950SZachary Turner     VariableSP var;
14169f727950SZachary Turner     switch (result.second.kind()) {
14179f727950SZachary Turner     case SymbolKind::S_GDATA32:
14189f727950SZachary Turner     case SymbolKind::S_LDATA32:
14199f727950SZachary Turner     case SymbolKind::S_GTHREAD32:
14202af34166SZachary Turner     case SymbolKind::S_LTHREAD32:
14212af34166SZachary Turner     case SymbolKind::S_CONSTANT: {
14229fbf9350SZachary Turner       PdbGlobalSymId global(result.first, false);
14236284aee9SZachary Turner       var = GetOrCreateGlobalVariable(global);
14249f727950SZachary Turner       variables.AddVariable(var);
14259f727950SZachary Turner       break;
14269f727950SZachary Turner     }
14279f727950SZachary Turner     default:
14289f727950SZachary Turner       continue;
14299f727950SZachary Turner     }
14309f727950SZachary Turner   }
14319f727950SZachary Turner   return variables.GetSize();
14329f727950SZachary Turner }
14339f727950SZachary Turner 
1434307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::FindFunctions(
1435b96181c2SZachary Turner     const ConstString &name, const CompilerDeclContext *parent_decl_ctx,
1436117b1fa1SZachary Turner     FunctionNameType name_type_mask, bool include_inlines, bool append,
1437b96181c2SZachary Turner     SymbolContextList &sc_list) {
1438307f5ae8SZachary Turner   // For now we only support lookup by method name.
1439307f5ae8SZachary Turner   if (!(name_type_mask & eFunctionNameTypeMethod))
1440307f5ae8SZachary Turner     return 0;
1441307f5ae8SZachary Turner 
1442307f5ae8SZachary Turner   using SymbolAndOffset = std::pair<uint32_t, llvm::codeview::CVSymbol>;
1443307f5ae8SZachary Turner 
1444307f5ae8SZachary Turner   std::vector<SymbolAndOffset> matches = m_index->globals().findRecordsByName(
1445307f5ae8SZachary Turner       name.GetStringRef(), m_index->symrecords());
1446307f5ae8SZachary Turner   for (const SymbolAndOffset &match : matches) {
1447307f5ae8SZachary Turner     if (match.second.kind() != S_PROCREF && match.second.kind() != S_LPROCREF)
1448307f5ae8SZachary Turner       continue;
1449307f5ae8SZachary Turner     ProcRefSym proc(match.second.kind());
1450307f5ae8SZachary Turner     cantFail(SymbolDeserializer::deserializeAs<ProcRefSym>(match.second, proc));
1451307f5ae8SZachary Turner 
1452307f5ae8SZachary Turner     if (!IsValidRecord(proc))
1453307f5ae8SZachary Turner       continue;
1454307f5ae8SZachary Turner 
14556284aee9SZachary Turner     CompilandIndexItem &cci =
14566284aee9SZachary Turner         m_index->compilands().GetOrCreateCompiland(proc.modi());
1457b96181c2SZachary Turner     SymbolContext sc;
1458307f5ae8SZachary Turner 
1459307f5ae8SZachary Turner     sc.comp_unit = GetOrCreateCompileUnit(cci).get();
1460307f5ae8SZachary Turner     sc.module_sp = sc.comp_unit->GetModule();
14619fbf9350SZachary Turner     PdbCompilandSymId func_id(proc.modi(), proc.SymOffset);
14626284aee9SZachary Turner     sc.function = GetOrCreateFunction(func_id, sc).get();
1463307f5ae8SZachary Turner 
1464307f5ae8SZachary Turner     sc_list.Append(sc);
1465307f5ae8SZachary Turner   }
1466307f5ae8SZachary Turner 
1467307f5ae8SZachary Turner   return sc_list.GetSize();
1468307f5ae8SZachary Turner }
1469307f5ae8SZachary Turner 
1470b96181c2SZachary Turner uint32_t SymbolFileNativePDB::FindFunctions(const RegularExpression &regex,
1471307f5ae8SZachary Turner                                             bool include_inlines, bool append,
1472b96181c2SZachary Turner                                             SymbolContextList &sc_list) {
1473307f5ae8SZachary Turner   return 0;
1474307f5ae8SZachary Turner }
1475307f5ae8SZachary Turner 
1476b96181c2SZachary Turner uint32_t SymbolFileNativePDB::FindTypes(
1477b96181c2SZachary Turner     const SymbolContext &sc, const ConstString &name,
1478b96181c2SZachary Turner     const CompilerDeclContext *parent_decl_ctx, bool append,
1479b96181c2SZachary Turner     uint32_t max_matches, llvm::DenseSet<SymbolFile *> &searched_symbol_files,
1480b96181c2SZachary Turner     TypeMap &types) {
14812f7efbc9SZachary Turner   if (!append)
14822f7efbc9SZachary Turner     types.Clear();
14832f7efbc9SZachary Turner   if (!name)
1484b96181c2SZachary Turner     return 0;
14852f7efbc9SZachary Turner 
14862f7efbc9SZachary Turner   searched_symbol_files.clear();
14872f7efbc9SZachary Turner   searched_symbol_files.insert(this);
14882f7efbc9SZachary Turner 
14892f7efbc9SZachary Turner   // There is an assumption 'name' is not a regex
14902f7efbc9SZachary Turner   size_t match_count = FindTypesByName(name.GetStringRef(), max_matches, types);
14912f7efbc9SZachary Turner 
14922f7efbc9SZachary Turner   return match_count;
1493b96181c2SZachary Turner }
1494b96181c2SZachary Turner 
1495b96181c2SZachary Turner size_t
1496b96181c2SZachary Turner SymbolFileNativePDB::FindTypes(const std::vector<CompilerContext> &context,
1497b96181c2SZachary Turner                                bool append, TypeMap &types) {
1498b96181c2SZachary Turner   return 0;
1499b96181c2SZachary Turner }
1500b96181c2SZachary Turner 
15012f7efbc9SZachary Turner size_t SymbolFileNativePDB::FindTypesByName(llvm::StringRef name,
15022f7efbc9SZachary Turner                                             uint32_t max_matches,
15032f7efbc9SZachary Turner                                             TypeMap &types) {
15042f7efbc9SZachary Turner 
15052f7efbc9SZachary Turner   size_t match_count = 0;
15062f7efbc9SZachary Turner   std::vector<TypeIndex> matches = m_index->tpi().findRecordsByName(name);
15072f7efbc9SZachary Turner   if (max_matches > 0 && max_matches < matches.size())
15082f7efbc9SZachary Turner     matches.resize(max_matches);
15092f7efbc9SZachary Turner 
15102f7efbc9SZachary Turner   for (TypeIndex ti : matches) {
15112f7efbc9SZachary Turner     TypeSP type = GetOrCreateType(ti);
15122f7efbc9SZachary Turner     if (!type)
15132f7efbc9SZachary Turner       continue;
15142f7efbc9SZachary Turner 
15152f7efbc9SZachary Turner     types.Insert(type);
15162f7efbc9SZachary Turner     ++match_count;
15172f7efbc9SZachary Turner   }
15182f7efbc9SZachary Turner   return match_count;
15192f7efbc9SZachary Turner }
15202f7efbc9SZachary Turner 
1521b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseTypes(const SymbolContext &sc) { return 0; }
1522b96181c2SZachary Turner 
1523b96181c2SZachary Turner Type *SymbolFileNativePDB::ResolveTypeUID(lldb::user_id_t type_uid) {
15242f7efbc9SZachary Turner   auto iter = m_types.find(type_uid);
15252f7efbc9SZachary Turner   // lldb should not be passing us non-sensical type uids.  the only way it
15262f7efbc9SZachary Turner   // could have a type uid in the first place is if we handed it out, in which
15279f727950SZachary Turner   // case we should know about the type.  However, that doesn't mean we've
15289f727950SZachary Turner   // instantiated it yet.  We can vend out a UID for a future type.  So if the
15299f727950SZachary Turner   // type doesn't exist, let's instantiate it now.
15309f727950SZachary Turner   if (iter != m_types.end())
15312f7efbc9SZachary Turner     return &*iter->second;
15329f727950SZachary Turner 
15336284aee9SZachary Turner   PdbSymUid uid(type_uid);
15346284aee9SZachary Turner   lldbassert(uid.kind() == PdbSymUidKind::Type);
15356284aee9SZachary Turner   PdbTypeSymId type_id = uid.asTypeSym();
15366284aee9SZachary Turner   if (type_id.index.isNoneType())
15379f727950SZachary Turner     return nullptr;
15389f727950SZachary Turner 
15396284aee9SZachary Turner   TypeSP type_sp = CreateAndCacheType(type_id);
15409f727950SZachary Turner   return &*type_sp;
1541b96181c2SZachary Turner }
1542b96181c2SZachary Turner 
1543eca07c59SAdrian Prantl llvm::Optional<SymbolFile::ArrayInfo>
1544eca07c59SAdrian Prantl SymbolFileNativePDB::GetDynamicArrayInfoForUID(
1545eca07c59SAdrian Prantl     lldb::user_id_t type_uid, const lldb_private::ExecutionContext *exe_ctx) {
1546eca07c59SAdrian Prantl   return llvm::None;
1547eca07c59SAdrian Prantl }
1548eca07c59SAdrian Prantl 
1549eca07c59SAdrian Prantl 
1550b96181c2SZachary Turner bool SymbolFileNativePDB::CompleteType(CompilerType &compiler_type) {
15512f7efbc9SZachary Turner   // If this is not in our map, it's an error.
15522f7efbc9SZachary Turner   clang::TagDecl *tag_decl = m_clang->GetAsTagDecl(compiler_type);
15532f7efbc9SZachary Turner   lldbassert(tag_decl);
15542f7efbc9SZachary Turner   auto status_iter = m_decl_to_status.find(tag_decl);
15552f7efbc9SZachary Turner   lldbassert(status_iter != m_decl_to_status.end());
15562f7efbc9SZachary Turner 
15572f7efbc9SZachary Turner   // If it's already complete, just return.
15582f7efbc9SZachary Turner   DeclStatus &status = status_iter->second;
15592f7efbc9SZachary Turner   if (status.status == Type::eResolveStateFull)
15602f7efbc9SZachary Turner     return true;
15612f7efbc9SZachary Turner 
15626284aee9SZachary Turner   PdbTypeSymId type_id = PdbSymUid(status.uid).asTypeSym();
15632f7efbc9SZachary Turner 
15646284aee9SZachary Turner   lldbassert(IsTagRecord(type_id, m_index->tpi()));
15652f7efbc9SZachary Turner 
15662f7efbc9SZachary Turner   ClangASTContext::SetHasExternalStorage(compiler_type.GetOpaqueQualType(),
15672f7efbc9SZachary Turner                                          false);
15682f7efbc9SZachary Turner 
15692f7efbc9SZachary Turner   // In CreateAndCacheType, we already go out of our way to resolve forward
15702f7efbc9SZachary Turner   // ref UDTs to full decls, and the uids we vend out always refer to full
15712f7efbc9SZachary Turner   // decls if a full decl exists in the debug info.  So if we don't have a full
15722f7efbc9SZachary Turner   // decl here, it means one doesn't exist in the debug info, and we can't
15732f7efbc9SZachary Turner   // complete the type.
15742f7efbc9SZachary Turner   CVType cvt = m_index->tpi().getType(TypeIndex(type_id.index));
15752f7efbc9SZachary Turner   if (IsForwardRefUdt(cvt))
15762f7efbc9SZachary Turner     return false;
15772f7efbc9SZachary Turner 
15786284aee9SZachary Turner   auto types_iter = m_types.find(status.uid);
15792f7efbc9SZachary Turner   lldbassert(types_iter != m_types.end());
15802f7efbc9SZachary Turner 
1581511bff21SZachary Turner   if (cvt.kind() == LF_MODIFIER) {
1582511bff21SZachary Turner     TypeIndex unmodified_type = LookThroughModifierRecord(cvt);
1583511bff21SZachary Turner     cvt = m_index->tpi().getType(unmodified_type);
1584511bff21SZachary Turner     // LF_MODIFIERS usually point to forward decls, so this is the one case
1585511bff21SZachary Turner     // where we won't have been able to resolve a forward decl to a full decl
1586511bff21SZachary Turner     // earlier on.  So we need to do that now.
1587511bff21SZachary Turner     if (IsForwardRefUdt(cvt)) {
1588511bff21SZachary Turner       llvm::Expected<TypeIndex> expected_full_ti =
1589511bff21SZachary Turner           m_index->tpi().findFullDeclForForwardRef(unmodified_type);
1590511bff21SZachary Turner       if (!expected_full_ti) {
1591511bff21SZachary Turner         llvm::consumeError(expected_full_ti.takeError());
1592511bff21SZachary Turner         return false;
1593511bff21SZachary Turner       }
1594511bff21SZachary Turner       cvt = m_index->tpi().getType(*expected_full_ti);
1595511bff21SZachary Turner       lldbassert(!IsForwardRefUdt(cvt));
1596511bff21SZachary Turner       unmodified_type = *expected_full_ti;
1597511bff21SZachary Turner     }
15989fbf9350SZachary Turner     type_id = PdbTypeSymId(unmodified_type, false);
1599511bff21SZachary Turner   }
16002f7efbc9SZachary Turner   TypeIndex field_list_ti = GetFieldListIndex(cvt);
16012f7efbc9SZachary Turner   CVType field_list_cvt = m_index->tpi().getType(field_list_ti);
16022f7efbc9SZachary Turner   if (field_list_cvt.kind() != LF_FIELDLIST)
16032f7efbc9SZachary Turner     return false;
16042f7efbc9SZachary Turner 
16052f7efbc9SZachary Turner   // Visit all members of this class, then perform any finalization necessary
16062f7efbc9SZachary Turner   // to complete the class.
16076284aee9SZachary Turner   UdtRecordCompleter completer(type_id, compiler_type, *tag_decl, *this);
16082f7efbc9SZachary Turner   auto error =
16092f7efbc9SZachary Turner       llvm::codeview::visitMemberRecordStream(field_list_cvt.data(), completer);
16102f7efbc9SZachary Turner   completer.complete();
16112f7efbc9SZachary Turner 
16122f7efbc9SZachary Turner   status.status = Type::eResolveStateFull;
16132f7efbc9SZachary Turner   if (!error)
16142f7efbc9SZachary Turner     return true;
16152f7efbc9SZachary Turner 
16162f7efbc9SZachary Turner   llvm::consumeError(std::move(error));
1617b96181c2SZachary Turner   return false;
1618b96181c2SZachary Turner }
1619b96181c2SZachary Turner 
1620b96181c2SZachary Turner size_t SymbolFileNativePDB::GetTypes(lldb_private::SymbolContextScope *sc_scope,
1621117b1fa1SZachary Turner                                      TypeClass type_mask,
1622b96181c2SZachary Turner                                      lldb_private::TypeList &type_list) {
1623b96181c2SZachary Turner   return 0;
1624b96181c2SZachary Turner }
1625b96181c2SZachary Turner 
1626b96181c2SZachary Turner CompilerDeclContext
1627b96181c2SZachary Turner SymbolFileNativePDB::FindNamespace(const SymbolContext &sc,
1628b96181c2SZachary Turner                                    const ConstString &name,
1629b96181c2SZachary Turner                                    const CompilerDeclContext *parent_decl_ctx) {
1630307f5ae8SZachary Turner   return {};
1631307f5ae8SZachary Turner }
1632307f5ae8SZachary Turner 
1633b96181c2SZachary Turner TypeSystem *
1634307f5ae8SZachary Turner SymbolFileNativePDB::GetTypeSystemForLanguage(lldb::LanguageType language) {
1635307f5ae8SZachary Turner   auto type_system =
1636307f5ae8SZachary Turner       m_obj_file->GetModule()->GetTypeSystemForLanguage(language);
1637307f5ae8SZachary Turner   if (type_system)
1638307f5ae8SZachary Turner     type_system->SetSymbolFile(this);
1639307f5ae8SZachary Turner   return type_system;
1640307f5ae8SZachary Turner }
1641307f5ae8SZachary Turner 
1642b96181c2SZachary Turner ConstString SymbolFileNativePDB::GetPluginName() {
1643307f5ae8SZachary Turner   static ConstString g_name("pdb");
1644307f5ae8SZachary Turner   return g_name;
1645307f5ae8SZachary Turner }
1646307f5ae8SZachary Turner 
1647307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::GetPluginVersion() { return 1; }
1648