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 
57307f5ae8SZachary Turner #include "PdbSymUid.h"
58307f5ae8SZachary Turner #include "PdbUtil.h"
592f7efbc9SZachary Turner #include "UdtRecordCompleter.h"
60307f5ae8SZachary Turner 
61307f5ae8SZachary Turner using namespace lldb;
62307f5ae8SZachary Turner using namespace lldb_private;
632f7efbc9SZachary Turner using namespace npdb;
64307f5ae8SZachary Turner using namespace llvm::codeview;
65307f5ae8SZachary Turner using namespace llvm::pdb;
66307f5ae8SZachary Turner 
67307f5ae8SZachary Turner static lldb::LanguageType TranslateLanguage(PDB_Lang lang) {
68307f5ae8SZachary Turner   switch (lang) {
69307f5ae8SZachary Turner   case PDB_Lang::Cpp:
70307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeC_plus_plus;
71307f5ae8SZachary Turner   case PDB_Lang::C:
72307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeC;
73307f5ae8SZachary Turner   default:
74307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeUnknown;
75307f5ae8SZachary Turner   }
76307f5ae8SZachary Turner }
77307f5ae8SZachary Turner 
78307f5ae8SZachary Turner static std::unique_ptr<PDBFile> loadPDBFile(std::string PdbPath,
79307f5ae8SZachary Turner                                             llvm::BumpPtrAllocator &Allocator) {
80307f5ae8SZachary Turner   llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> ErrorOrBuffer =
81307f5ae8SZachary Turner       llvm::MemoryBuffer::getFile(PdbPath, /*FileSize=*/-1,
82307f5ae8SZachary Turner                                   /*RequiresNullTerminator=*/false);
83307f5ae8SZachary Turner   if (!ErrorOrBuffer)
84307f5ae8SZachary Turner     return nullptr;
85307f5ae8SZachary Turner   std::unique_ptr<llvm::MemoryBuffer> Buffer = std::move(*ErrorOrBuffer);
86307f5ae8SZachary Turner 
87307f5ae8SZachary Turner   llvm::StringRef Path = Buffer->getBufferIdentifier();
88307f5ae8SZachary Turner   auto Stream = llvm::make_unique<llvm::MemoryBufferByteStream>(
89307f5ae8SZachary Turner       std::move(Buffer), llvm::support::little);
90307f5ae8SZachary Turner 
91307f5ae8SZachary Turner   auto File = llvm::make_unique<PDBFile>(Path, std::move(Stream), Allocator);
928040eea9SZachary Turner   if (auto EC = File->parseFileHeaders()) {
938040eea9SZachary Turner     llvm::consumeError(std::move(EC));
94307f5ae8SZachary Turner     return nullptr;
958040eea9SZachary Turner   }
968040eea9SZachary Turner   if (auto EC = File->parseStreamData()) {
978040eea9SZachary Turner     llvm::consumeError(std::move(EC));
98307f5ae8SZachary Turner     return nullptr;
998040eea9SZachary Turner   }
100307f5ae8SZachary Turner 
101307f5ae8SZachary Turner   return File;
102307f5ae8SZachary Turner }
103307f5ae8SZachary Turner 
104307f5ae8SZachary Turner static std::unique_ptr<PDBFile>
105307f5ae8SZachary Turner loadMatchingPDBFile(std::string exe_path, llvm::BumpPtrAllocator &allocator) {
106307f5ae8SZachary Turner   // Try to find a matching PDB for an EXE.
107307f5ae8SZachary Turner   using namespace llvm::object;
108307f5ae8SZachary Turner   auto expected_binary = createBinary(exe_path);
109307f5ae8SZachary Turner 
110307f5ae8SZachary Turner   // If the file isn't a PE/COFF executable, fail.
111307f5ae8SZachary Turner   if (!expected_binary) {
112307f5ae8SZachary Turner     llvm::consumeError(expected_binary.takeError());
113307f5ae8SZachary Turner     return nullptr;
114307f5ae8SZachary Turner   }
115307f5ae8SZachary Turner   OwningBinary<Binary> binary = std::move(*expected_binary);
116307f5ae8SZachary Turner 
117307f5ae8SZachary Turner   auto *obj = llvm::dyn_cast<llvm::object::COFFObjectFile>(binary.getBinary());
118307f5ae8SZachary Turner   if (!obj)
119307f5ae8SZachary Turner     return nullptr;
120307f5ae8SZachary Turner   const llvm::codeview::DebugInfo *pdb_info = nullptr;
121307f5ae8SZachary Turner 
122307f5ae8SZachary Turner   // If it doesn't have a debug directory, fail.
123307f5ae8SZachary Turner   llvm::StringRef pdb_file;
124307f5ae8SZachary Turner   auto ec = obj->getDebugPDBInfo(pdb_info, pdb_file);
125307f5ae8SZachary Turner   if (ec)
126307f5ae8SZachary Turner     return nullptr;
127307f5ae8SZachary Turner 
128307f5ae8SZachary Turner   // if the file doesn't exist, is not a pdb, or doesn't have a matching guid,
129307f5ae8SZachary Turner   // fail.
130307f5ae8SZachary Turner   llvm::file_magic magic;
131307f5ae8SZachary Turner   ec = llvm::identify_magic(pdb_file, magic);
132307f5ae8SZachary Turner   if (ec || magic != llvm::file_magic::pdb)
133307f5ae8SZachary Turner     return nullptr;
134307f5ae8SZachary Turner   std::unique_ptr<PDBFile> pdb = loadPDBFile(pdb_file, allocator);
1358040eea9SZachary Turner   if (!pdb)
1368040eea9SZachary Turner     return nullptr;
1378040eea9SZachary Turner 
138307f5ae8SZachary Turner   auto expected_info = pdb->getPDBInfoStream();
139307f5ae8SZachary Turner   if (!expected_info) {
140307f5ae8SZachary Turner     llvm::consumeError(expected_info.takeError());
141307f5ae8SZachary Turner     return nullptr;
142307f5ae8SZachary Turner   }
143307f5ae8SZachary Turner   llvm::codeview::GUID guid;
144307f5ae8SZachary Turner   memcpy(&guid, pdb_info->PDB70.Signature, 16);
145307f5ae8SZachary Turner 
146307f5ae8SZachary Turner   if (expected_info->getGuid() != guid)
147307f5ae8SZachary Turner     return nullptr;
148307f5ae8SZachary Turner   return pdb;
149307f5ae8SZachary Turner }
150307f5ae8SZachary Turner 
151307f5ae8SZachary Turner static bool IsFunctionPrologue(const CompilandIndexItem &cci,
152307f5ae8SZachary Turner                                lldb::addr_t addr) {
153307f5ae8SZachary Turner   // FIXME: Implement this.
154307f5ae8SZachary Turner   return false;
155307f5ae8SZachary Turner }
156307f5ae8SZachary Turner 
157307f5ae8SZachary Turner static bool IsFunctionEpilogue(const CompilandIndexItem &cci,
158307f5ae8SZachary Turner                                lldb::addr_t addr) {
159307f5ae8SZachary Turner   // FIXME: Implement this.
160307f5ae8SZachary Turner   return false;
161307f5ae8SZachary Turner }
162307f5ae8SZachary Turner 
1632f7efbc9SZachary Turner static clang::MSInheritanceAttr::Spelling
1642f7efbc9SZachary Turner GetMSInheritance(LazyRandomTypeCollection &tpi, const ClassRecord &record) {
1652f7efbc9SZachary Turner   if (record.DerivationList == TypeIndex::None())
1662f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance;
1672f7efbc9SZachary Turner 
1682f7efbc9SZachary Turner   CVType bases = tpi.getType(record.DerivationList);
1692f7efbc9SZachary Turner   ArgListRecord base_list;
1702f7efbc9SZachary Turner   cantFail(TypeDeserializer::deserializeAs<ArgListRecord>(bases, base_list));
1712f7efbc9SZachary Turner   if (base_list.ArgIndices.empty())
1722f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance;
1732f7efbc9SZachary Turner 
1742f7efbc9SZachary Turner   int base_count = 0;
1752f7efbc9SZachary Turner   for (TypeIndex ti : base_list.ArgIndices) {
1762f7efbc9SZachary Turner     CVType base = tpi.getType(ti);
1772f7efbc9SZachary Turner     if (base.kind() == LF_VBCLASS || base.kind() == LF_IVBCLASS)
1782f7efbc9SZachary Turner       return clang::MSInheritanceAttr::Spelling::Keyword_virtual_inheritance;
1792f7efbc9SZachary Turner     ++base_count;
1802f7efbc9SZachary Turner   }
1812f7efbc9SZachary Turner 
1822f7efbc9SZachary Turner   if (base_count > 1)
1832f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Keyword_multiple_inheritance;
1842f7efbc9SZachary Turner   return clang::MSInheritanceAttr::Keyword_single_inheritance;
1852f7efbc9SZachary Turner }
1862f7efbc9SZachary Turner 
1872f7efbc9SZachary Turner static lldb::BasicType GetCompilerTypeForSimpleKind(SimpleTypeKind kind) {
1882f7efbc9SZachary Turner   switch (kind) {
1892f7efbc9SZachary Turner   case SimpleTypeKind::Boolean128:
1902f7efbc9SZachary Turner   case SimpleTypeKind::Boolean16:
1912f7efbc9SZachary Turner   case SimpleTypeKind::Boolean32:
1922f7efbc9SZachary Turner   case SimpleTypeKind::Boolean64:
1932f7efbc9SZachary Turner   case SimpleTypeKind::Boolean8:
1942f7efbc9SZachary Turner     return lldb::eBasicTypeBool;
1952f7efbc9SZachary Turner   case SimpleTypeKind::Byte:
1962f7efbc9SZachary Turner   case SimpleTypeKind::UnsignedCharacter:
1972f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedChar;
1982f7efbc9SZachary Turner   case SimpleTypeKind::NarrowCharacter:
1992f7efbc9SZachary Turner     return lldb::eBasicTypeChar;
2002f7efbc9SZachary Turner   case SimpleTypeKind::SignedCharacter:
2012f7efbc9SZachary Turner   case SimpleTypeKind::SByte:
2022f7efbc9SZachary Turner     return lldb::eBasicTypeSignedChar;
2032f7efbc9SZachary Turner   case SimpleTypeKind::Character16:
2042f7efbc9SZachary Turner     return lldb::eBasicTypeChar16;
2052f7efbc9SZachary Turner   case SimpleTypeKind::Character32:
2062f7efbc9SZachary Turner     return lldb::eBasicTypeChar32;
2072f7efbc9SZachary Turner   case SimpleTypeKind::Complex80:
2082f7efbc9SZachary Turner     return lldb::eBasicTypeLongDoubleComplex;
2092f7efbc9SZachary Turner   case SimpleTypeKind::Complex64:
2102f7efbc9SZachary Turner     return lldb::eBasicTypeDoubleComplex;
2112f7efbc9SZachary Turner   case SimpleTypeKind::Complex32:
2122f7efbc9SZachary Turner     return lldb::eBasicTypeFloatComplex;
2132f7efbc9SZachary Turner   case SimpleTypeKind::Float128:
2142f7efbc9SZachary Turner   case SimpleTypeKind::Float80:
2152f7efbc9SZachary Turner     return lldb::eBasicTypeLongDouble;
2162f7efbc9SZachary Turner   case SimpleTypeKind::Float64:
2172f7efbc9SZachary Turner     return lldb::eBasicTypeDouble;
2182f7efbc9SZachary Turner   case SimpleTypeKind::Float32:
2192f7efbc9SZachary Turner     return lldb::eBasicTypeFloat;
2202f7efbc9SZachary Turner   case SimpleTypeKind::Float16:
2212f7efbc9SZachary Turner     return lldb::eBasicTypeHalf;
2222f7efbc9SZachary Turner   case SimpleTypeKind::Int128:
2232f7efbc9SZachary Turner     return lldb::eBasicTypeInt128;
2242f7efbc9SZachary Turner   case SimpleTypeKind::Int64:
2252f7efbc9SZachary Turner   case SimpleTypeKind::Int64Quad:
2262f7efbc9SZachary Turner     return lldb::eBasicTypeLongLong;
2272f7efbc9SZachary Turner   case SimpleTypeKind::Int32:
2282f7efbc9SZachary Turner     return lldb::eBasicTypeInt;
2292f7efbc9SZachary Turner   case SimpleTypeKind::Int16:
2302f7efbc9SZachary Turner   case SimpleTypeKind::Int16Short:
2312f7efbc9SZachary Turner     return lldb::eBasicTypeShort;
2322f7efbc9SZachary Turner   case SimpleTypeKind::UInt128:
2332f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedInt128;
2342f7efbc9SZachary Turner   case SimpleTypeKind::UInt64:
2352f7efbc9SZachary Turner   case SimpleTypeKind::UInt64Quad:
2362f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedLongLong;
2372f7efbc9SZachary Turner   case SimpleTypeKind::HResult:
2382f7efbc9SZachary Turner   case SimpleTypeKind::UInt32:
2392f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedInt;
2402f7efbc9SZachary Turner   case SimpleTypeKind::UInt16:
2412f7efbc9SZachary Turner   case SimpleTypeKind::UInt16Short:
2422f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedShort;
2432f7efbc9SZachary Turner   case SimpleTypeKind::Int32Long:
2442f7efbc9SZachary Turner     return lldb::eBasicTypeLong;
2452f7efbc9SZachary Turner   case SimpleTypeKind::UInt32Long:
2462f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedLong;
2472f7efbc9SZachary Turner   case SimpleTypeKind::Void:
2482f7efbc9SZachary Turner     return lldb::eBasicTypeVoid;
2492f7efbc9SZachary Turner   case SimpleTypeKind::WideCharacter:
2502f7efbc9SZachary Turner     return lldb::eBasicTypeWChar;
2512f7efbc9SZachary Turner   default:
2522f7efbc9SZachary Turner     return lldb::eBasicTypeInvalid;
2532f7efbc9SZachary Turner   }
2542f7efbc9SZachary Turner }
2552f7efbc9SZachary Turner 
2562f7efbc9SZachary Turner static size_t GetTypeSizeForSimpleKind(SimpleTypeKind kind) {
2572f7efbc9SZachary Turner   switch (kind) {
2582f7efbc9SZachary Turner   case SimpleTypeKind::Boolean128:
2592f7efbc9SZachary Turner   case SimpleTypeKind::Int128:
2602f7efbc9SZachary Turner   case SimpleTypeKind::UInt128:
2612f7efbc9SZachary Turner   case SimpleTypeKind::Float128:
2622f7efbc9SZachary Turner     return 16;
2632f7efbc9SZachary Turner   case SimpleTypeKind::Complex80:
2642f7efbc9SZachary Turner   case SimpleTypeKind::Float80:
2652f7efbc9SZachary Turner     return 10;
2662f7efbc9SZachary Turner   case SimpleTypeKind::Boolean64:
2672f7efbc9SZachary Turner   case SimpleTypeKind::Complex64:
2682f7efbc9SZachary Turner   case SimpleTypeKind::UInt64:
2692f7efbc9SZachary Turner   case SimpleTypeKind::UInt64Quad:
2702f7efbc9SZachary Turner   case SimpleTypeKind::Float64:
2712f7efbc9SZachary Turner   case SimpleTypeKind::Int64:
2722f7efbc9SZachary Turner   case SimpleTypeKind::Int64Quad:
2732f7efbc9SZachary Turner     return 8;
2742f7efbc9SZachary Turner   case SimpleTypeKind::Boolean32:
2752f7efbc9SZachary Turner   case SimpleTypeKind::Character32:
2762f7efbc9SZachary Turner   case SimpleTypeKind::Complex32:
2772f7efbc9SZachary Turner   case SimpleTypeKind::Float32:
2782f7efbc9SZachary Turner   case SimpleTypeKind::Int32:
2792f7efbc9SZachary Turner   case SimpleTypeKind::Int32Long:
2802f7efbc9SZachary Turner   case SimpleTypeKind::UInt32Long:
2812f7efbc9SZachary Turner   case SimpleTypeKind::HResult:
2822f7efbc9SZachary Turner   case SimpleTypeKind::UInt32:
2832f7efbc9SZachary Turner     return 4;
2842f7efbc9SZachary Turner   case SimpleTypeKind::Boolean16:
2852f7efbc9SZachary Turner   case SimpleTypeKind::Character16:
2862f7efbc9SZachary Turner   case SimpleTypeKind::Float16:
2872f7efbc9SZachary Turner   case SimpleTypeKind::Int16:
2882f7efbc9SZachary Turner   case SimpleTypeKind::Int16Short:
2892f7efbc9SZachary Turner   case SimpleTypeKind::UInt16:
2902f7efbc9SZachary Turner   case SimpleTypeKind::UInt16Short:
2912f7efbc9SZachary Turner   case SimpleTypeKind::WideCharacter:
2922f7efbc9SZachary Turner     return 2;
2932f7efbc9SZachary Turner   case SimpleTypeKind::Boolean8:
2942f7efbc9SZachary Turner   case SimpleTypeKind::Byte:
2952f7efbc9SZachary Turner   case SimpleTypeKind::UnsignedCharacter:
2962f7efbc9SZachary Turner   case SimpleTypeKind::NarrowCharacter:
2972f7efbc9SZachary Turner   case SimpleTypeKind::SignedCharacter:
2982f7efbc9SZachary Turner   case SimpleTypeKind::SByte:
2992f7efbc9SZachary Turner     return 1;
3002f7efbc9SZachary Turner   case SimpleTypeKind::Void:
3012f7efbc9SZachary Turner   default:
3022f7efbc9SZachary Turner     return 0;
3032f7efbc9SZachary Turner   }
3042f7efbc9SZachary Turner }
3052f7efbc9SZachary Turner 
3062f7efbc9SZachary Turner static llvm::StringRef GetSimpleTypeName(SimpleTypeKind kind) {
3072f7efbc9SZachary Turner   switch (kind) {
3082f7efbc9SZachary Turner   case SimpleTypeKind::Boolean128:
3092f7efbc9SZachary Turner   case SimpleTypeKind::Boolean16:
3102f7efbc9SZachary Turner   case SimpleTypeKind::Boolean32:
3112f7efbc9SZachary Turner   case SimpleTypeKind::Boolean64:
3122f7efbc9SZachary Turner   case SimpleTypeKind::Boolean8:
3132f7efbc9SZachary Turner     return "bool";
3142f7efbc9SZachary Turner   case SimpleTypeKind::Byte:
3152f7efbc9SZachary Turner   case SimpleTypeKind::UnsignedCharacter:
3162f7efbc9SZachary Turner     return "unsigned char";
3172f7efbc9SZachary Turner   case SimpleTypeKind::NarrowCharacter:
3182f7efbc9SZachary Turner     return "char";
3192f7efbc9SZachary Turner   case SimpleTypeKind::SignedCharacter:
3202f7efbc9SZachary Turner   case SimpleTypeKind::SByte:
32171ebb721SZachary Turner     return "signed char";
3222f7efbc9SZachary Turner   case SimpleTypeKind::Character16:
3232f7efbc9SZachary Turner     return "char16_t";
3242f7efbc9SZachary Turner   case SimpleTypeKind::Character32:
3252f7efbc9SZachary Turner     return "char32_t";
3262f7efbc9SZachary Turner   case SimpleTypeKind::Complex80:
3272f7efbc9SZachary Turner   case SimpleTypeKind::Complex64:
3282f7efbc9SZachary Turner   case SimpleTypeKind::Complex32:
3292f7efbc9SZachary Turner     return "complex";
3302f7efbc9SZachary Turner   case SimpleTypeKind::Float128:
3312f7efbc9SZachary Turner   case SimpleTypeKind::Float80:
3322f7efbc9SZachary Turner     return "long double";
3332f7efbc9SZachary Turner   case SimpleTypeKind::Float64:
3342f7efbc9SZachary Turner     return "double";
3352f7efbc9SZachary Turner   case SimpleTypeKind::Float32:
3362f7efbc9SZachary Turner     return "float";
3372f7efbc9SZachary Turner   case SimpleTypeKind::Float16:
3382f7efbc9SZachary Turner     return "single";
3392f7efbc9SZachary Turner   case SimpleTypeKind::Int128:
3402f7efbc9SZachary Turner     return "__int128";
3412f7efbc9SZachary Turner   case SimpleTypeKind::Int64:
3422f7efbc9SZachary Turner   case SimpleTypeKind::Int64Quad:
34371ebb721SZachary Turner     return "int64_t";
3442f7efbc9SZachary Turner   case SimpleTypeKind::Int32:
3452f7efbc9SZachary Turner     return "int";
3462f7efbc9SZachary Turner   case SimpleTypeKind::Int16:
3472f7efbc9SZachary Turner     return "short";
3482f7efbc9SZachary Turner   case SimpleTypeKind::UInt128:
3492f7efbc9SZachary Turner     return "unsigned __int128";
3502f7efbc9SZachary Turner   case SimpleTypeKind::UInt64:
3512f7efbc9SZachary Turner   case SimpleTypeKind::UInt64Quad:
35271ebb721SZachary Turner     return "uint64_t";
3532f7efbc9SZachary Turner   case SimpleTypeKind::HResult:
3542f7efbc9SZachary Turner     return "HRESULT";
3552f7efbc9SZachary Turner   case SimpleTypeKind::UInt32:
3562f7efbc9SZachary Turner     return "unsigned";
3572f7efbc9SZachary Turner   case SimpleTypeKind::UInt16:
3582f7efbc9SZachary Turner   case SimpleTypeKind::UInt16Short:
3592f7efbc9SZachary Turner     return "unsigned short";
3602f7efbc9SZachary Turner   case SimpleTypeKind::Int32Long:
3612f7efbc9SZachary Turner     return "long";
3622f7efbc9SZachary Turner   case SimpleTypeKind::UInt32Long:
3632f7efbc9SZachary Turner     return "unsigned long";
3642f7efbc9SZachary Turner   case SimpleTypeKind::Void:
3652f7efbc9SZachary Turner     return "void";
3662f7efbc9SZachary Turner   case SimpleTypeKind::WideCharacter:
3672f7efbc9SZachary Turner     return "wchar_t";
3682f7efbc9SZachary Turner   default:
3692f7efbc9SZachary Turner     return "";
3702f7efbc9SZachary Turner   }
3712f7efbc9SZachary Turner }
3722f7efbc9SZachary Turner 
3732f7efbc9SZachary Turner static bool IsClassRecord(TypeLeafKind kind) {
3742f7efbc9SZachary Turner   switch (kind) {
3752f7efbc9SZachary Turner   case LF_STRUCTURE:
3762f7efbc9SZachary Turner   case LF_CLASS:
3772f7efbc9SZachary Turner   case LF_INTERFACE:
3782f7efbc9SZachary Turner     return true;
3792f7efbc9SZachary Turner   default:
3802f7efbc9SZachary Turner     return false;
3812f7efbc9SZachary Turner   }
3822f7efbc9SZachary Turner }
3832f7efbc9SZachary Turner 
3842f7efbc9SZachary Turner static PDB_SymType GetPdbSymType(TpiStream &tpi, TypeIndex ti) {
3852f7efbc9SZachary Turner   if (ti.isSimple()) {
3862f7efbc9SZachary Turner     if (ti.getSimpleMode() == SimpleTypeMode::Direct)
3872f7efbc9SZachary Turner       return PDB_SymType::BuiltinType;
3882f7efbc9SZachary Turner     return PDB_SymType::PointerType;
3892f7efbc9SZachary Turner   }
3902f7efbc9SZachary Turner 
3912f7efbc9SZachary Turner   CVType cvt = tpi.getType(ti);
3922f7efbc9SZachary Turner   TypeLeafKind kind = cvt.kind();
3932f7efbc9SZachary Turner   if (kind != LF_MODIFIER)
3942f7efbc9SZachary Turner     return CVTypeToPDBType(kind);
3952f7efbc9SZachary Turner 
3962f7efbc9SZachary Turner   // If this is an LF_MODIFIER, look through it to get the kind that it
3972f7efbc9SZachary Turner   // modifies.  Note that it's not possible to have an LF_MODIFIER that
3982f7efbc9SZachary Turner   // modifies another LF_MODIFIER, although this would handle that anyway.
399511bff21SZachary Turner   return GetPdbSymType(tpi, LookThroughModifierRecord(cvt));
4002f7efbc9SZachary Turner }
4012f7efbc9SZachary Turner 
402544a66d8SZachary Turner static bool IsCVarArgsFunction(llvm::ArrayRef<TypeIndex> args) {
403544a66d8SZachary Turner   if (args.empty())
404544a66d8SZachary Turner     return false;
405544a66d8SZachary Turner   return args.back() == TypeIndex::None();
406544a66d8SZachary Turner }
407544a66d8SZachary Turner 
4082f7efbc9SZachary Turner static clang::TagTypeKind TranslateUdtKind(const TagRecord &cr) {
4092f7efbc9SZachary Turner   switch (cr.Kind) {
4102f7efbc9SZachary Turner   case TypeRecordKind::Class:
4112f7efbc9SZachary Turner     return clang::TTK_Class;
4122f7efbc9SZachary Turner   case TypeRecordKind::Struct:
4132f7efbc9SZachary Turner     return clang::TTK_Struct;
4142f7efbc9SZachary Turner   case TypeRecordKind::Union:
4152f7efbc9SZachary Turner     return clang::TTK_Union;
4162f7efbc9SZachary Turner   case TypeRecordKind::Interface:
4172f7efbc9SZachary Turner     return clang::TTK_Interface;
4182f7efbc9SZachary Turner   case TypeRecordKind::Enum:
4192f7efbc9SZachary Turner     return clang::TTK_Enum;
4202f7efbc9SZachary Turner   default:
4212f7efbc9SZachary Turner     lldbassert(false && "Invalid tag record kind!");
4222f7efbc9SZachary Turner     return clang::TTK_Struct;
4232f7efbc9SZachary Turner   }
4242f7efbc9SZachary Turner }
4252f7efbc9SZachary Turner 
426544a66d8SZachary Turner static llvm::Optional<clang::CallingConv>
427544a66d8SZachary Turner TranslateCallingConvention(llvm::codeview::CallingConvention conv) {
428544a66d8SZachary Turner   using CC = llvm::codeview::CallingConvention;
429544a66d8SZachary Turner   switch (conv) {
430544a66d8SZachary Turner 
431544a66d8SZachary Turner   case CC::NearC:
432544a66d8SZachary Turner   case CC::FarC:
433544a66d8SZachary Turner     return clang::CallingConv::CC_C;
434544a66d8SZachary Turner   case CC::NearPascal:
435544a66d8SZachary Turner   case CC::FarPascal:
436544a66d8SZachary Turner     return clang::CallingConv::CC_X86Pascal;
437544a66d8SZachary Turner   case CC::NearFast:
438544a66d8SZachary Turner   case CC::FarFast:
439544a66d8SZachary Turner     return clang::CallingConv::CC_X86FastCall;
440544a66d8SZachary Turner   case CC::NearStdCall:
441544a66d8SZachary Turner   case CC::FarStdCall:
442544a66d8SZachary Turner     return clang::CallingConv::CC_X86StdCall;
443544a66d8SZachary Turner   case CC::ThisCall:
444544a66d8SZachary Turner     return clang::CallingConv::CC_X86ThisCall;
445544a66d8SZachary Turner   case CC::NearVector:
446544a66d8SZachary Turner     return clang::CallingConv::CC_X86VectorCall;
447544a66d8SZachary Turner   default:
448544a66d8SZachary Turner     return llvm::None;
449544a66d8SZachary Turner   }
450544a66d8SZachary Turner }
451544a66d8SZachary Turner 
452307f5ae8SZachary Turner void SymbolFileNativePDB::Initialize() {
453307f5ae8SZachary Turner   PluginManager::RegisterPlugin(GetPluginNameStatic(),
454307f5ae8SZachary Turner                                 GetPluginDescriptionStatic(), CreateInstance,
455307f5ae8SZachary Turner                                 DebuggerInitialize);
456307f5ae8SZachary Turner }
457307f5ae8SZachary Turner 
458307f5ae8SZachary Turner void SymbolFileNativePDB::Terminate() {
459307f5ae8SZachary Turner   PluginManager::UnregisterPlugin(CreateInstance);
460307f5ae8SZachary Turner }
461307f5ae8SZachary Turner 
462b96181c2SZachary Turner void SymbolFileNativePDB::DebuggerInitialize(Debugger &debugger) {}
463307f5ae8SZachary Turner 
464b96181c2SZachary Turner ConstString SymbolFileNativePDB::GetPluginNameStatic() {
465307f5ae8SZachary Turner   static ConstString g_name("native-pdb");
466307f5ae8SZachary Turner   return g_name;
467307f5ae8SZachary Turner }
468307f5ae8SZachary Turner 
469307f5ae8SZachary Turner const char *SymbolFileNativePDB::GetPluginDescriptionStatic() {
470307f5ae8SZachary Turner   return "Microsoft PDB debug symbol cross-platform file reader.";
471307f5ae8SZachary Turner }
472307f5ae8SZachary Turner 
473b96181c2SZachary Turner SymbolFile *SymbolFileNativePDB::CreateInstance(ObjectFile *obj_file) {
474307f5ae8SZachary Turner   return new SymbolFileNativePDB(obj_file);
475307f5ae8SZachary Turner }
476307f5ae8SZachary Turner 
477b96181c2SZachary Turner SymbolFileNativePDB::SymbolFileNativePDB(ObjectFile *object_file)
478307f5ae8SZachary Turner     : SymbolFile(object_file) {}
479307f5ae8SZachary Turner 
480307f5ae8SZachary Turner SymbolFileNativePDB::~SymbolFileNativePDB() {}
481307f5ae8SZachary Turner 
482307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::CalculateAbilities() {
483307f5ae8SZachary Turner   uint32_t abilities = 0;
484307f5ae8SZachary Turner   if (!m_obj_file)
485307f5ae8SZachary Turner     return 0;
486307f5ae8SZachary Turner 
487307f5ae8SZachary Turner   if (!m_index) {
488307f5ae8SZachary Turner     // Lazily load and match the PDB file, but only do this once.
489307f5ae8SZachary Turner     std::unique_ptr<PDBFile> file_up =
490307f5ae8SZachary Turner         loadMatchingPDBFile(m_obj_file->GetFileSpec().GetPath(), m_allocator);
491307f5ae8SZachary Turner 
492307f5ae8SZachary Turner     if (!file_up) {
493307f5ae8SZachary Turner       auto module_sp = m_obj_file->GetModule();
494307f5ae8SZachary Turner       if (!module_sp)
495307f5ae8SZachary Turner         return 0;
496307f5ae8SZachary Turner       // See if any symbol file is specified through `--symfile` option.
497307f5ae8SZachary Turner       FileSpec symfile = module_sp->GetSymbolFileFileSpec();
498307f5ae8SZachary Turner       if (!symfile)
499307f5ae8SZachary Turner         return 0;
500307f5ae8SZachary Turner       file_up = loadPDBFile(symfile.GetPath(), m_allocator);
501307f5ae8SZachary Turner     }
502307f5ae8SZachary Turner 
503307f5ae8SZachary Turner     if (!file_up)
504307f5ae8SZachary Turner       return 0;
505307f5ae8SZachary Turner 
506307f5ae8SZachary Turner     auto expected_index = PdbIndex::create(std::move(file_up));
507307f5ae8SZachary Turner     if (!expected_index) {
508307f5ae8SZachary Turner       llvm::consumeError(expected_index.takeError());
509307f5ae8SZachary Turner       return 0;
510307f5ae8SZachary Turner     }
511307f5ae8SZachary Turner     m_index = std::move(*expected_index);
512307f5ae8SZachary Turner   }
513307f5ae8SZachary Turner   if (!m_index)
514307f5ae8SZachary Turner     return 0;
515307f5ae8SZachary Turner 
516307f5ae8SZachary Turner   // We don't especially have to be precise here.  We only distinguish between
517307f5ae8SZachary Turner   // stripped and not stripped.
518307f5ae8SZachary Turner   abilities = kAllAbilities;
519307f5ae8SZachary Turner 
520307f5ae8SZachary Turner   if (m_index->dbi().isStripped())
521307f5ae8SZachary Turner     abilities &= ~(Blocks | LocalVariables);
522307f5ae8SZachary Turner   return abilities;
523307f5ae8SZachary Turner }
524307f5ae8SZachary Turner 
525307f5ae8SZachary Turner void SymbolFileNativePDB::InitializeObject() {
526307f5ae8SZachary Turner   m_obj_load_address = m_obj_file->GetFileOffset();
527307f5ae8SZachary Turner   m_index->SetLoadAddress(m_obj_load_address);
528307f5ae8SZachary Turner   m_index->ParseSectionContribs();
5292f7efbc9SZachary Turner 
5302f7efbc9SZachary Turner   TypeSystem *ts = GetTypeSystemForLanguage(eLanguageTypeC_plus_plus);
5312f7efbc9SZachary Turner   m_clang = llvm::dyn_cast_or_null<ClangASTContext>(ts);
5322f7efbc9SZachary Turner   m_importer = llvm::make_unique<ClangASTImporter>();
533056e4ab4SZachary Turner 
534056e4ab4SZachary Turner   PreprocessTpiStream();
5352f7efbc9SZachary Turner   lldbassert(m_clang);
536307f5ae8SZachary Turner }
537307f5ae8SZachary Turner 
538*03a24052SZachary Turner static llvm::Optional<CVTagRecord>
539*03a24052SZachary Turner GetNestedTagRecord(const NestedTypeRecord &Record, const CVTagRecord &parent,
540*03a24052SZachary Turner                    TpiStream &tpi) {
541*03a24052SZachary Turner   // An LF_NESTTYPE is essentially a nested typedef / using declaration, but it
542*03a24052SZachary Turner   // is also used to indicate the primary definition of a nested class.  That is
543*03a24052SZachary Turner   // to say, if you have:
544*03a24052SZachary Turner   // struct A {
545*03a24052SZachary Turner   //   struct B {};
546*03a24052SZachary Turner   //   using C = B;
547*03a24052SZachary Turner   // };
548*03a24052SZachary Turner   // Then in the debug info, this will appear as:
549*03a24052SZachary Turner   // LF_STRUCTURE `A::B` [type index = N]
550*03a24052SZachary Turner   // LF_STRUCTURE `A`
551*03a24052SZachary Turner   //   LF_NESTTYPE [name = `B`, index = N]
552*03a24052SZachary Turner   //   LF_NESTTYPE [name = `C`, index = N]
553*03a24052SZachary Turner   // In order to accurately reconstruct the decl context hierarchy, we need to
554*03a24052SZachary Turner   // know which ones are actual definitions and which ones are just aliases.
555*03a24052SZachary Turner 
556*03a24052SZachary Turner   // If it's a simple type, then this is something like `using foo = int`.
557*03a24052SZachary Turner   if (Record.Type.isSimple())
558*03a24052SZachary Turner     return llvm::None;
559*03a24052SZachary Turner 
560*03a24052SZachary Turner   // If it's an inner definition, then treat whatever name we have here as a
561*03a24052SZachary Turner   // single component of a mangled name.  So we can inject it into the parent's
562*03a24052SZachary Turner   // mangled name to see if it matches.
563*03a24052SZachary Turner   CVTagRecord child = CVTagRecord::create(tpi.getType(Record.Type));
564*03a24052SZachary Turner   std::string qname = parent.asTag().getUniqueName();
565*03a24052SZachary Turner   if (qname.size() < 4 || child.asTag().getUniqueName().size() < 4)
566*03a24052SZachary Turner     return llvm::None;
567*03a24052SZachary Turner 
568*03a24052SZachary Turner   // qname[3] is the tag type identifier (struct, class, union, etc).  Since the
569*03a24052SZachary Turner   // inner tag type is not necessarily the same as the outer tag type, re-write
570*03a24052SZachary Turner   // it to match the inner tag type.
571*03a24052SZachary Turner   qname[3] = child.asTag().getUniqueName()[3];
572*03a24052SZachary Turner   std::string piece = Record.Name;
573*03a24052SZachary Turner   piece.push_back('@');
574*03a24052SZachary Turner   qname.insert(4, std::move(piece));
575*03a24052SZachary Turner   if (qname != child.asTag().UniqueName)
576*03a24052SZachary Turner     return llvm::None;
577*03a24052SZachary Turner 
578*03a24052SZachary Turner   return std::move(child);
579*03a24052SZachary Turner }
580*03a24052SZachary Turner 
581056e4ab4SZachary Turner void SymbolFileNativePDB::PreprocessTpiStream() {
582056e4ab4SZachary Turner   LazyRandomTypeCollection &types = m_index->tpi().typeCollection();
583056e4ab4SZachary Turner 
584056e4ab4SZachary Turner   for (auto ti = types.getFirst(); ti; ti = types.getNext(*ti)) {
585056e4ab4SZachary Turner     CVType type = types.getType(*ti);
586056e4ab4SZachary Turner     if (!IsTagRecord(type))
587056e4ab4SZachary Turner       continue;
588056e4ab4SZachary Turner 
589056e4ab4SZachary Turner     CVTagRecord tag = CVTagRecord::create(type);
590056e4ab4SZachary Turner     // We're looking for LF_NESTTYPE records in the field list, so ignore
591056e4ab4SZachary Turner     // forward references (no field list), and anything without a nested class
592056e4ab4SZachary Turner     // (since there won't be any LF_NESTTYPE records).
593056e4ab4SZachary Turner     if (tag.asTag().isForwardRef() || !tag.asTag().containsNestedClass())
594056e4ab4SZachary Turner       continue;
595056e4ab4SZachary Turner 
596056e4ab4SZachary Turner     struct ProcessTpiStream : public TypeVisitorCallbacks {
597056e4ab4SZachary Turner       ProcessTpiStream(PdbIndex &index, TypeIndex parent,
598*03a24052SZachary Turner                        const CVTagRecord &parent_cvt,
599056e4ab4SZachary Turner                        llvm::DenseMap<TypeIndex, TypeIndex> &parents)
600*03a24052SZachary Turner           : index(index), parents(parents), parent(parent),
601*03a24052SZachary Turner             parent_cvt(parent_cvt) {}
602056e4ab4SZachary Turner 
603056e4ab4SZachary Turner       PdbIndex &index;
604056e4ab4SZachary Turner       llvm::DenseMap<TypeIndex, TypeIndex> &parents;
605056e4ab4SZachary Turner       TypeIndex parent;
606*03a24052SZachary Turner       const CVTagRecord &parent_cvt;
607056e4ab4SZachary Turner 
608056e4ab4SZachary Turner       llvm::Error visitKnownMember(CVMemberRecord &CVR,
609056e4ab4SZachary Turner                                    NestedTypeRecord &Record) override {
610*03a24052SZachary Turner         llvm::Optional<CVTagRecord> tag =
611*03a24052SZachary Turner             GetNestedTagRecord(Record, parent_cvt, index.tpi());
612*03a24052SZachary Turner         if (!tag)
613056e4ab4SZachary Turner           return llvm::ErrorSuccess();
614*03a24052SZachary Turner 
615*03a24052SZachary Turner         parents[Record.Type] = parent;
616*03a24052SZachary Turner         if (!tag->asTag().isForwardRef())
617*03a24052SZachary Turner           return llvm::ErrorSuccess();
618*03a24052SZachary Turner 
619056e4ab4SZachary Turner         llvm::Expected<TypeIndex> full_decl =
620056e4ab4SZachary Turner             index.tpi().findFullDeclForForwardRef(Record.Type);
621056e4ab4SZachary Turner         if (!full_decl) {
622056e4ab4SZachary Turner           llvm::consumeError(full_decl.takeError());
623056e4ab4SZachary Turner           return llvm::ErrorSuccess();
624056e4ab4SZachary Turner         }
625056e4ab4SZachary Turner         parents[*full_decl] = parent;
626056e4ab4SZachary Turner         return llvm::ErrorSuccess();
627056e4ab4SZachary Turner       }
628056e4ab4SZachary Turner     };
629056e4ab4SZachary Turner 
630056e4ab4SZachary Turner     CVType field_list = m_index->tpi().getType(tag.asTag().FieldList);
631*03a24052SZachary Turner     ProcessTpiStream process(*m_index, *ti, tag, m_parent_types);
632056e4ab4SZachary Turner     llvm::Error error = visitMemberRecordStream(field_list.data(), process);
633056e4ab4SZachary Turner     if (error)
634056e4ab4SZachary Turner       llvm::consumeError(std::move(error));
635056e4ab4SZachary Turner   }
636056e4ab4SZachary Turner }
637056e4ab4SZachary Turner 
638307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::GetNumCompileUnits() {
639307f5ae8SZachary Turner   const DbiModuleList &modules = m_index->dbi().modules();
640307f5ae8SZachary Turner   uint32_t count = modules.getModuleCount();
641307f5ae8SZachary Turner   if (count == 0)
642307f5ae8SZachary Turner     return count;
643307f5ae8SZachary Turner 
644307f5ae8SZachary Turner   // The linker can inject an additional "dummy" compilation unit into the
645307f5ae8SZachary Turner   // PDB. Ignore this special compile unit for our purposes, if it is there.
646307f5ae8SZachary Turner   // It is always the last one.
647307f5ae8SZachary Turner   DbiModuleDescriptor last = modules.getModuleDescriptor(count - 1);
648307f5ae8SZachary Turner   if (last.getModuleName() == "* Linker *")
649307f5ae8SZachary Turner     --count;
650307f5ae8SZachary Turner   return count;
651307f5ae8SZachary Turner }
652307f5ae8SZachary Turner 
653307f5ae8SZachary Turner lldb::FunctionSP SymbolFileNativePDB::CreateFunction(PdbSymUid func_uid,
654307f5ae8SZachary Turner                                                      const SymbolContext &sc) {
655307f5ae8SZachary Turner   lldbassert(func_uid.tag() == PDB_SymType::Function);
656307f5ae8SZachary Turner 
657307f5ae8SZachary Turner   PdbSymUid cuid = PdbSymUid::makeCompilandId(func_uid.asCuSym().modi);
658307f5ae8SZachary Turner 
659307f5ae8SZachary Turner   const CompilandIndexItem *cci = m_index->compilands().GetCompiland(cuid);
660307f5ae8SZachary Turner   lldbassert(cci);
661307f5ae8SZachary Turner   CVSymbol sym_record =
662307f5ae8SZachary Turner       cci->m_debug_stream.readSymbolAtOffset(func_uid.asCuSym().offset);
663307f5ae8SZachary Turner 
664307f5ae8SZachary Turner   lldbassert(sym_record.kind() == S_LPROC32 || sym_record.kind() == S_GPROC32);
665307f5ae8SZachary Turner   SegmentOffsetLength sol = GetSegmentOffsetAndLength(sym_record);
666307f5ae8SZachary Turner 
667307f5ae8SZachary Turner   auto file_vm_addr = m_index->MakeVirtualAddress(sol.so);
668307f5ae8SZachary Turner   if (file_vm_addr == LLDB_INVALID_ADDRESS || file_vm_addr == 0)
669307f5ae8SZachary Turner     return nullptr;
670307f5ae8SZachary Turner 
671307f5ae8SZachary Turner   AddressRange func_range(file_vm_addr, sol.length,
672307f5ae8SZachary Turner                           sc.module_sp->GetSectionList());
673307f5ae8SZachary Turner   if (!func_range.GetBaseAddress().IsValid())
674307f5ae8SZachary Turner     return nullptr;
675307f5ae8SZachary Turner 
676b96181c2SZachary Turner   Type *func_type = nullptr;
677307f5ae8SZachary Turner 
678307f5ae8SZachary Turner   // FIXME: Resolve types and mangled names.
679307f5ae8SZachary Turner   PdbSymUid sig_uid =
680307f5ae8SZachary Turner       PdbSymUid::makeTypeSymId(PDB_SymType::FunctionSig, TypeIndex{0}, false);
681307f5ae8SZachary Turner   Mangled mangled(getSymbolName(sym_record));
682307f5ae8SZachary Turner   FunctionSP func_sp = std::make_shared<Function>(
683307f5ae8SZachary Turner       sc.comp_unit, func_uid.toOpaqueId(), sig_uid.toOpaqueId(), mangled,
684307f5ae8SZachary Turner       func_type, func_range);
685307f5ae8SZachary Turner 
686307f5ae8SZachary Turner   sc.comp_unit->AddFunction(func_sp);
687307f5ae8SZachary Turner   return func_sp;
688307f5ae8SZachary Turner }
689307f5ae8SZachary Turner 
690307f5ae8SZachary Turner CompUnitSP
691307f5ae8SZachary Turner SymbolFileNativePDB::CreateCompileUnit(const CompilandIndexItem &cci) {
692307f5ae8SZachary Turner   lldb::LanguageType lang =
693307f5ae8SZachary Turner       cci.m_compile_opts ? TranslateLanguage(cci.m_compile_opts->getLanguage())
694307f5ae8SZachary Turner                          : lldb::eLanguageTypeUnknown;
695307f5ae8SZachary Turner 
696307f5ae8SZachary Turner   LazyBool optimized = eLazyBoolNo;
697307f5ae8SZachary Turner   if (cci.m_compile_opts && cci.m_compile_opts->hasOptimizations())
698307f5ae8SZachary Turner     optimized = eLazyBoolYes;
699307f5ae8SZachary Turner 
700307f5ae8SZachary Turner   llvm::StringRef source_file_name =
701307f5ae8SZachary Turner       m_index->compilands().GetMainSourceFile(cci);
7028f3be7a3SJonas Devlieghere   FileSpec fs(source_file_name);
703307f5ae8SZachary Turner 
704307f5ae8SZachary Turner   CompUnitSP cu_sp =
705307f5ae8SZachary Turner       std::make_shared<CompileUnit>(m_obj_file->GetModule(), nullptr, fs,
706307f5ae8SZachary Turner                                     cci.m_uid.toOpaqueId(), lang, optimized);
707307f5ae8SZachary Turner 
708307f5ae8SZachary Turner   const PdbCompilandId &cuid = cci.m_uid.asCompiland();
709307f5ae8SZachary Turner   m_obj_file->GetModule()->GetSymbolVendor()->SetCompileUnitAtIndex(cuid.modi,
710307f5ae8SZachary Turner                                                                     cu_sp);
711307f5ae8SZachary Turner   return cu_sp;
712307f5ae8SZachary Turner }
713307f5ae8SZachary Turner 
7142f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateModifierType(PdbSymUid type_uid,
7152f7efbc9SZachary Turner                                                      const ModifierRecord &mr) {
7162f7efbc9SZachary Turner   TpiStream &stream = m_index->tpi();
7172f7efbc9SZachary Turner 
7182f7efbc9SZachary Turner   TypeSP t = GetOrCreateType(mr.ModifiedType);
7192f7efbc9SZachary Turner   CompilerType ct = t->GetForwardCompilerType();
7202f7efbc9SZachary Turner   if ((mr.Modifiers & ModifierOptions::Const) != ModifierOptions::None)
7212f7efbc9SZachary Turner     ct = ct.AddConstModifier();
7222f7efbc9SZachary Turner   if ((mr.Modifiers & ModifierOptions::Volatile) != ModifierOptions::None)
7232f7efbc9SZachary Turner     ct = ct.AddVolatileModifier();
7242f7efbc9SZachary Turner   std::string name;
7252f7efbc9SZachary Turner   if (mr.ModifiedType.isSimple())
7262f7efbc9SZachary Turner     name = GetSimpleTypeName(mr.ModifiedType.getSimpleKind());
7272f7efbc9SZachary Turner   else
7282f7efbc9SZachary Turner     name = computeTypeName(stream.typeCollection(), mr.ModifiedType);
7292f7efbc9SZachary Turner   Declaration decl;
7302f7efbc9SZachary Turner   return std::make_shared<Type>(type_uid.toOpaqueId(), m_clang->GetSymbolFile(),
7312f7efbc9SZachary Turner                                 ConstString(name), t->GetByteSize(), nullptr,
7322f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
7332f7efbc9SZachary Turner                                 ct, Type::eResolveStateFull);
7342f7efbc9SZachary Turner }
7352f7efbc9SZachary Turner 
7362f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreatePointerType(
7372f7efbc9SZachary Turner     PdbSymUid type_uid, const llvm::codeview::PointerRecord &pr) {
7382f7efbc9SZachary Turner   TypeSP pointee = GetOrCreateType(pr.ReferentType);
739544a66d8SZachary Turner   if (!pointee)
740544a66d8SZachary Turner     return nullptr;
7412f7efbc9SZachary Turner   CompilerType pointee_ct = pointee->GetForwardCompilerType();
7422f7efbc9SZachary Turner   lldbassert(pointee_ct);
7432f7efbc9SZachary Turner   Declaration decl;
7442f7efbc9SZachary Turner 
7452f7efbc9SZachary Turner   if (pr.isPointerToMember()) {
7462f7efbc9SZachary Turner     MemberPointerInfo mpi = pr.getMemberInfo();
7472f7efbc9SZachary Turner     TypeSP class_type = GetOrCreateType(mpi.ContainingType);
7482f7efbc9SZachary Turner 
7492f7efbc9SZachary Turner     CompilerType ct = ClangASTContext::CreateMemberPointerType(
7502f7efbc9SZachary Turner         class_type->GetLayoutCompilerType(), pointee_ct);
7512f7efbc9SZachary Turner 
7522f7efbc9SZachary Turner     return std::make_shared<Type>(
7532f7efbc9SZachary Turner         type_uid.toOpaqueId(), m_clang->GetSymbolFile(), ConstString(),
7542f7efbc9SZachary Turner         pr.getSize(), nullptr, LLDB_INVALID_UID, Type::eEncodingIsUID, decl, ct,
7552f7efbc9SZachary Turner         Type::eResolveStateFull);
7562f7efbc9SZachary Turner   }
7572f7efbc9SZachary Turner 
7582f7efbc9SZachary Turner   CompilerType pointer_ct = pointee_ct;
7592f7efbc9SZachary Turner   if (pr.getMode() == PointerMode::LValueReference)
7602f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetLValueReferenceType();
7612f7efbc9SZachary Turner   else if (pr.getMode() == PointerMode::RValueReference)
7622f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetRValueReferenceType();
7632f7efbc9SZachary Turner   else
7642f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetPointerType();
7652f7efbc9SZachary Turner 
7662f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Const) != PointerOptions::None)
7672f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddConstModifier();
7682f7efbc9SZachary Turner 
7692f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Volatile) != PointerOptions::None)
7702f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddVolatileModifier();
7712f7efbc9SZachary Turner 
7722f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Restrict) != PointerOptions::None)
7732f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddRestrictModifier();
7742f7efbc9SZachary Turner 
7752f7efbc9SZachary Turner   return std::make_shared<Type>(type_uid.toOpaqueId(), m_clang->GetSymbolFile(),
7762f7efbc9SZachary Turner                                 ConstString(), pr.getSize(), nullptr,
7772f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
7782f7efbc9SZachary Turner                                 pointer_ct, Type::eResolveStateFull);
7792f7efbc9SZachary Turner }
7802f7efbc9SZachary Turner 
7812f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateSimpleType(TypeIndex ti) {
782544a66d8SZachary Turner   if (ti == TypeIndex::NullptrT()) {
783544a66d8SZachary Turner     PdbSymUid uid =
784544a66d8SZachary Turner         PdbSymUid::makeTypeSymId(PDB_SymType::BuiltinType, ti, false);
785544a66d8SZachary Turner     CompilerType ct = m_clang->GetBasicType(eBasicTypeNullPtr);
786544a66d8SZachary Turner     Declaration decl;
787544a66d8SZachary Turner     return std::make_shared<Type>(uid.toOpaqueId(), this,
788544a66d8SZachary Turner                                   ConstString("std::nullptr_t"), 0, nullptr,
789544a66d8SZachary Turner                                   LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
790544a66d8SZachary Turner                                   ct, Type::eResolveStateFull);
791544a66d8SZachary Turner   }
792544a66d8SZachary Turner 
7932f7efbc9SZachary Turner   if (ti.getSimpleMode() != SimpleTypeMode::Direct) {
7942f7efbc9SZachary Turner     PdbSymUid uid =
7952f7efbc9SZachary Turner         PdbSymUid::makeTypeSymId(PDB_SymType::PointerType, ti, false);
7962f7efbc9SZachary Turner     TypeSP direct_sp = GetOrCreateType(ti.makeDirect());
7972f7efbc9SZachary Turner     CompilerType ct = direct_sp->GetFullCompilerType();
7982f7efbc9SZachary Turner     ct = ct.GetPointerType();
79971ebb721SZachary Turner     uint32_t pointer_size = 0;
8002f7efbc9SZachary Turner     switch (ti.getSimpleMode()) {
8012f7efbc9SZachary Turner     case SimpleTypeMode::FarPointer32:
8022f7efbc9SZachary Turner     case SimpleTypeMode::NearPointer32:
8032f7efbc9SZachary Turner       pointer_size = 4;
8042f7efbc9SZachary Turner       break;
8052f7efbc9SZachary Turner     case SimpleTypeMode::NearPointer64:
8062f7efbc9SZachary Turner       pointer_size = 8;
8072f7efbc9SZachary Turner       break;
8082f7efbc9SZachary Turner     default:
8092f7efbc9SZachary Turner       // 128-bit and 16-bit pointers unsupported.
8102f7efbc9SZachary Turner       return nullptr;
8112f7efbc9SZachary Turner     }
8122f7efbc9SZachary Turner     Declaration decl;
8132f7efbc9SZachary Turner     return std::make_shared<Type>(uid.toOpaqueId(), m_clang->GetSymbolFile(),
8142f7efbc9SZachary Turner                                   ConstString(), pointer_size, nullptr,
8152f7efbc9SZachary Turner                                   LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
8162f7efbc9SZachary Turner                                   ct, Type::eResolveStateFull);
8172f7efbc9SZachary Turner   }
8182f7efbc9SZachary Turner 
8192f7efbc9SZachary Turner   PdbSymUid uid = PdbSymUid::makeTypeSymId(PDB_SymType::BuiltinType, ti, false);
8202f7efbc9SZachary Turner   if (ti.getSimpleKind() == SimpleTypeKind::NotTranslated)
8212f7efbc9SZachary Turner     return nullptr;
8222f7efbc9SZachary Turner 
8232f7efbc9SZachary Turner   lldb::BasicType bt = GetCompilerTypeForSimpleKind(ti.getSimpleKind());
824544a66d8SZachary Turner   if (bt == lldb::eBasicTypeInvalid)
825544a66d8SZachary Turner     return nullptr;
8262f7efbc9SZachary Turner   CompilerType ct = m_clang->GetBasicType(bt);
8272f7efbc9SZachary Turner   size_t size = GetTypeSizeForSimpleKind(ti.getSimpleKind());
8282f7efbc9SZachary Turner 
8292f7efbc9SZachary Turner   llvm::StringRef type_name = GetSimpleTypeName(ti.getSimpleKind());
8302f7efbc9SZachary Turner 
8312f7efbc9SZachary Turner   Declaration decl;
8322f7efbc9SZachary Turner   return std::make_shared<Type>(uid.toOpaqueId(), m_clang->GetSymbolFile(),
8332f7efbc9SZachary Turner                                 ConstString(type_name), size, nullptr,
8342f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
8352f7efbc9SZachary Turner                                 ct, Type::eResolveStateFull);
8362f7efbc9SZachary Turner }
8372f7efbc9SZachary Turner 
838056e4ab4SZachary Turner static std::string RenderDemanglerNode(llvm::ms_demangle::Node *n) {
839056e4ab4SZachary Turner   OutputStream OS;
840056e4ab4SZachary Turner   initializeOutputStream(nullptr, nullptr, OS, 1024);
841056e4ab4SZachary Turner   n->output(OS, llvm::ms_demangle::OF_Default);
842056e4ab4SZachary Turner   OS << '\0';
843056e4ab4SZachary Turner   return {OS.getBuffer()};
844056e4ab4SZachary Turner }
845056e4ab4SZachary Turner 
846*03a24052SZachary Turner static bool
847*03a24052SZachary Turner AnyScopesHaveTemplateParams(llvm::ArrayRef<llvm::ms_demangle::Node *> scopes) {
848*03a24052SZachary Turner   for (llvm::ms_demangle::Node *n : scopes) {
849*03a24052SZachary Turner     auto *idn = static_cast<llvm::ms_demangle::IdentifierNode *>(n);
850*03a24052SZachary Turner     if (idn->TemplateParams)
851*03a24052SZachary Turner       return true;
852*03a24052SZachary Turner   }
853*03a24052SZachary Turner   return false;
854*03a24052SZachary Turner }
855*03a24052SZachary Turner 
856056e4ab4SZachary Turner std::pair<clang::DeclContext *, std::string>
857056e4ab4SZachary Turner SymbolFileNativePDB::CreateDeclInfoForType(const TagRecord &record,
858056e4ab4SZachary Turner                                            TypeIndex ti) {
859056e4ab4SZachary Turner   llvm::ms_demangle::Demangler demangler;
860056e4ab4SZachary Turner   StringView sv(record.UniqueName.begin(), record.UniqueName.size());
861056e4ab4SZachary Turner   llvm::ms_demangle::TagTypeNode *ttn = demangler.parseTagUniqueName(sv);
862056e4ab4SZachary Turner   llvm::ms_demangle::IdentifierNode *idn =
863056e4ab4SZachary Turner       ttn->QualifiedName->getUnqualifiedIdentifier();
864056e4ab4SZachary Turner   std::string uname = RenderDemanglerNode(idn);
865056e4ab4SZachary Turner 
866056e4ab4SZachary Turner   llvm::ms_demangle::NodeArrayNode *name_components =
867056e4ab4SZachary Turner       ttn->QualifiedName->Components;
868056e4ab4SZachary Turner   llvm::ArrayRef<llvm::ms_demangle::Node *> scopes(name_components->Nodes,
869056e4ab4SZachary Turner                                                    name_components->Count - 1);
870056e4ab4SZachary Turner 
871056e4ab4SZachary Turner   clang::DeclContext *context = m_clang->GetTranslationUnitDecl();
872056e4ab4SZachary Turner 
873056e4ab4SZachary Turner   // If this type doesn't have a parent type in the debug info, then the best we
874056e4ab4SZachary Turner   // can do is to say that it's either a series of namespaces (if the scope is
875056e4ab4SZachary Turner   // non-empty), or the translation unit (if the scope is empty).
876056e4ab4SZachary Turner   auto parent_iter = m_parent_types.find(ti);
877056e4ab4SZachary Turner   if (parent_iter == m_parent_types.end()) {
878056e4ab4SZachary Turner     if (scopes.empty())
879056e4ab4SZachary Turner       return {context, uname};
880056e4ab4SZachary Turner 
881*03a24052SZachary Turner     // If there is no parent in the debug info, but some of the scopes have
882*03a24052SZachary Turner     // template params, then this is a case of bad debug info.  See, for
883*03a24052SZachary Turner     // example, llvm.org/pr39607.  We don't want to create an ambiguity between
884*03a24052SZachary Turner     // a NamespaceDecl and a CXXRecordDecl, so instead we create a class at
885*03a24052SZachary Turner     // global scope with the fully qualified name.
886*03a24052SZachary Turner     if (AnyScopesHaveTemplateParams(scopes))
887*03a24052SZachary Turner       return {context, record.Name};
888*03a24052SZachary Turner 
889056e4ab4SZachary Turner     for (llvm::ms_demangle::Node *scope : scopes) {
890056e4ab4SZachary Turner       auto *nii = static_cast<llvm::ms_demangle::NamedIdentifierNode *>(scope);
891056e4ab4SZachary Turner       std::string str = RenderDemanglerNode(nii);
892056e4ab4SZachary Turner       context = m_clang->GetUniqueNamespaceDeclaration(str.c_str(), context);
893056e4ab4SZachary Turner     }
894056e4ab4SZachary Turner     return {context, uname};
895056e4ab4SZachary Turner   }
896056e4ab4SZachary Turner 
897056e4ab4SZachary Turner   // Otherwise, all we need to do is get the parent type of this type and
898056e4ab4SZachary Turner   // recurse into our lazy type creation / AST reconstruction logic to get an
899056e4ab4SZachary Turner   // LLDB TypeSP for the parent.  This will cause the AST to automatically get
900056e4ab4SZachary Turner   // the right DeclContext created for any parent.
901056e4ab4SZachary Turner   TypeSP parent = GetOrCreateType(parent_iter->second);
902056e4ab4SZachary Turner   if (!parent)
903056e4ab4SZachary Turner     return {context, uname};
904056e4ab4SZachary Turner   CompilerType parent_ct = parent->GetForwardCompilerType();
905056e4ab4SZachary Turner   clang::QualType qt = ClangUtil::GetCanonicalQualType(parent_ct);
906056e4ab4SZachary Turner   context = clang::TagDecl::castToDeclContext(qt->getAsTagDecl());
907056e4ab4SZachary Turner   return {context, uname};
908056e4ab4SZachary Turner }
909056e4ab4SZachary Turner 
9102f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateClassStructUnion(
911056e4ab4SZachary Turner     PdbSymUid type_uid, const llvm::codeview::TagRecord &record, size_t size,
9122f7efbc9SZachary Turner     clang::TagTypeKind ttk, clang::MSInheritanceAttr::Spelling inheritance) {
9132f7efbc9SZachary Turner 
914056e4ab4SZachary Turner   const PdbTypeSymId &tid = type_uid.asTypeSym();
915056e4ab4SZachary Turner   TypeIndex ti(tid.index);
916056e4ab4SZachary Turner   clang::DeclContext *decl_context = nullptr;
917056e4ab4SZachary Turner   std::string uname;
918056e4ab4SZachary Turner   std::tie(decl_context, uname) = CreateDeclInfoForType(record, ti);
9192f7efbc9SZachary Turner 
9202f7efbc9SZachary Turner   lldb::AccessType access =
9212f7efbc9SZachary Turner       (ttk == clang::TTK_Class) ? lldb::eAccessPrivate : lldb::eAccessPublic;
9222f7efbc9SZachary Turner 
9232f7efbc9SZachary Turner   ClangASTMetadata metadata;
9242f7efbc9SZachary Turner   metadata.SetUserID(type_uid.toOpaqueId());
9252f7efbc9SZachary Turner   metadata.SetIsDynamicCXXType(false);
9262f7efbc9SZachary Turner 
9272f7efbc9SZachary Turner   CompilerType ct =
928056e4ab4SZachary Turner       m_clang->CreateRecordType(decl_context, access, uname.c_str(), ttk,
9292f7efbc9SZachary Turner                                 lldb::eLanguageTypeC_plus_plus, &metadata);
930056e4ab4SZachary Turner 
9312f7efbc9SZachary Turner   lldbassert(ct.IsValid());
9322f7efbc9SZachary Turner 
9332f7efbc9SZachary Turner   clang::CXXRecordDecl *record_decl =
9342f7efbc9SZachary Turner       m_clang->GetAsCXXRecordDecl(ct.GetOpaqueQualType());
9352f7efbc9SZachary Turner   lldbassert(record_decl);
9362f7efbc9SZachary Turner 
9372f7efbc9SZachary Turner   clang::MSInheritanceAttr *attr = clang::MSInheritanceAttr::CreateImplicit(
9382f7efbc9SZachary Turner       *m_clang->getASTContext(), inheritance);
9392f7efbc9SZachary Turner   record_decl->addAttr(attr);
9402f7efbc9SZachary Turner 
9412f7efbc9SZachary Turner   ClangASTContext::StartTagDeclarationDefinition(ct);
9422f7efbc9SZachary Turner 
9432f7efbc9SZachary Turner   // Even if it's possible, don't complete it at this point. Just mark it
9442f7efbc9SZachary Turner   // forward resolved, and if/when LLDB needs the full definition, it can
9452f7efbc9SZachary Turner   // ask us.
9462f7efbc9SZachary Turner   ClangASTContext::SetHasExternalStorage(ct.GetOpaqueQualType(), true);
9472f7efbc9SZachary Turner 
9482f7efbc9SZachary Turner   // FIXME: Search IPI stream for LF_UDT_MOD_SRC_LINE.
9492f7efbc9SZachary Turner   Declaration decl;
9502f7efbc9SZachary Turner   return std::make_shared<Type>(type_uid.toOpaqueId(), m_clang->GetSymbolFile(),
951056e4ab4SZachary Turner                                 ConstString(uname), size, nullptr,
9522f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
9532f7efbc9SZachary Turner                                 ct, Type::eResolveStateForward);
9542f7efbc9SZachary Turner }
9552f7efbc9SZachary Turner 
9562f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbSymUid type_uid,
9572f7efbc9SZachary Turner                                                 const ClassRecord &cr) {
9582f7efbc9SZachary Turner   clang::TagTypeKind ttk = TranslateUdtKind(cr);
9592f7efbc9SZachary Turner 
9602f7efbc9SZachary Turner   clang::MSInheritanceAttr::Spelling inheritance =
9612f7efbc9SZachary Turner       GetMSInheritance(m_index->tpi().typeCollection(), cr);
962056e4ab4SZachary Turner   return CreateClassStructUnion(type_uid, cr, cr.getSize(), ttk, inheritance);
9632f7efbc9SZachary Turner }
9642f7efbc9SZachary Turner 
9652f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbSymUid type_uid,
9662f7efbc9SZachary Turner                                                 const UnionRecord &ur) {
9672f7efbc9SZachary Turner   return CreateClassStructUnion(
968056e4ab4SZachary Turner       type_uid, ur, ur.getSize(), clang::TTK_Union,
9692f7efbc9SZachary Turner       clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance);
9702f7efbc9SZachary Turner }
9712f7efbc9SZachary Turner 
9722f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbSymUid type_uid,
9732f7efbc9SZachary Turner                                                 const EnumRecord &er) {
9742f7efbc9SZachary Turner   llvm::StringRef name = DropNameScope(er.getName());
9752f7efbc9SZachary Turner 
9762f7efbc9SZachary Turner   clang::DeclContext *decl_context = m_clang->GetTranslationUnitDecl();
9772f7efbc9SZachary Turner 
9782f7efbc9SZachary Turner   Declaration decl;
9792f7efbc9SZachary Turner   TypeSP underlying_type = GetOrCreateType(er.UnderlyingType);
9802f7efbc9SZachary Turner   CompilerType enum_ct = m_clang->CreateEnumerationType(
9812f7efbc9SZachary Turner       name.str().c_str(), decl_context, decl,
9822f7efbc9SZachary Turner       underlying_type->GetFullCompilerType(), er.isScoped());
9832f7efbc9SZachary Turner 
9842f7efbc9SZachary Turner   ClangASTContext::StartTagDeclarationDefinition(enum_ct);
98512abab57SZachary Turner   ClangASTContext::SetHasExternalStorage(enum_ct.GetOpaqueQualType(), true);
9862f7efbc9SZachary Turner 
9872f7efbc9SZachary Turner   // We're just going to forward resolve this for now.  We'll complete
9882f7efbc9SZachary Turner   // it only if the user requests.
9892f7efbc9SZachary Turner   return std::make_shared<lldb_private::Type>(
9902f7efbc9SZachary Turner       type_uid.toOpaqueId(), m_clang->GetSymbolFile(), ConstString(name),
9912f7efbc9SZachary Turner       underlying_type->GetByteSize(), nullptr, LLDB_INVALID_UID,
9922f7efbc9SZachary Turner       lldb_private::Type::eEncodingIsUID, decl, enum_ct,
9932f7efbc9SZachary Turner       lldb_private::Type::eResolveStateForward);
9942f7efbc9SZachary Turner }
9952f7efbc9SZachary Turner 
996511bff21SZachary Turner TypeSP SymbolFileNativePDB::CreateArrayType(PdbSymUid type_uid,
997511bff21SZachary Turner                                             const ArrayRecord &ar) {
998511bff21SZachary Turner   TypeSP element_type = GetOrCreateType(ar.ElementType);
999511bff21SZachary Turner   uint64_t element_count = ar.Size / element_type->GetByteSize();
1000511bff21SZachary Turner 
1001511bff21SZachary Turner   CompilerType element_ct = element_type->GetFullCompilerType();
1002511bff21SZachary Turner 
1003511bff21SZachary Turner   CompilerType array_ct =
1004511bff21SZachary Turner       m_clang->CreateArrayType(element_ct, element_count, false);
1005511bff21SZachary Turner 
1006511bff21SZachary Turner   Declaration decl;
1007511bff21SZachary Turner   TypeSP array_sp = std::make_shared<lldb_private::Type>(
1008511bff21SZachary Turner       type_uid.toOpaqueId(), m_clang->GetSymbolFile(), ConstString(), ar.Size,
1009511bff21SZachary Turner       nullptr, LLDB_INVALID_UID, lldb_private::Type::eEncodingIsUID, decl,
1010511bff21SZachary Turner       array_ct, lldb_private::Type::eResolveStateFull);
1011511bff21SZachary Turner   array_sp->SetEncodingType(element_type.get());
1012511bff21SZachary Turner   return array_sp;
1013511bff21SZachary Turner }
1014511bff21SZachary Turner 
1015544a66d8SZachary Turner TypeSP SymbolFileNativePDB::CreateProcedureType(PdbSymUid type_uid,
1016544a66d8SZachary Turner                                                 const ProcedureRecord &pr) {
1017544a66d8SZachary Turner   TpiStream &stream = m_index->tpi();
1018544a66d8SZachary Turner   CVType args_cvt = stream.getType(pr.ArgumentList);
1019544a66d8SZachary Turner   ArgListRecord args;
1020544a66d8SZachary Turner   llvm::cantFail(
1021544a66d8SZachary Turner       TypeDeserializer::deserializeAs<ArgListRecord>(args_cvt, args));
1022544a66d8SZachary Turner 
1023544a66d8SZachary Turner   llvm::ArrayRef<TypeIndex> arg_indices = llvm::makeArrayRef(args.ArgIndices);
1024544a66d8SZachary Turner   bool is_variadic = IsCVarArgsFunction(arg_indices);
1025544a66d8SZachary Turner   if (is_variadic)
1026544a66d8SZachary Turner     arg_indices = arg_indices.drop_back();
1027544a66d8SZachary Turner 
1028544a66d8SZachary Turner   std::vector<CompilerType> arg_list;
1029544a66d8SZachary Turner   arg_list.reserve(arg_list.size());
1030544a66d8SZachary Turner 
1031544a66d8SZachary Turner   for (TypeIndex arg_index : arg_indices) {
1032544a66d8SZachary Turner     TypeSP arg_sp = GetOrCreateType(arg_index);
1033544a66d8SZachary Turner     if (!arg_sp)
1034544a66d8SZachary Turner       return nullptr;
1035544a66d8SZachary Turner     arg_list.push_back(arg_sp->GetFullCompilerType());
1036544a66d8SZachary Turner   }
1037544a66d8SZachary Turner 
1038544a66d8SZachary Turner   TypeSP return_type_sp = GetOrCreateType(pr.ReturnType);
1039544a66d8SZachary Turner   if (!return_type_sp)
1040544a66d8SZachary Turner     return nullptr;
1041544a66d8SZachary Turner 
1042544a66d8SZachary Turner   llvm::Optional<clang::CallingConv> cc =
1043544a66d8SZachary Turner       TranslateCallingConvention(pr.CallConv);
1044544a66d8SZachary Turner   if (!cc)
1045544a66d8SZachary Turner     return nullptr;
1046544a66d8SZachary Turner 
1047544a66d8SZachary Turner   CompilerType return_ct = return_type_sp->GetFullCompilerType();
1048544a66d8SZachary Turner   CompilerType func_sig_ast_type = m_clang->CreateFunctionType(
1049544a66d8SZachary Turner       return_ct, arg_list.data(), arg_list.size(), is_variadic, 0, *cc);
1050544a66d8SZachary Turner 
1051544a66d8SZachary Turner   Declaration decl;
1052544a66d8SZachary Turner   return std::make_shared<lldb_private::Type>(
1053544a66d8SZachary Turner       type_uid.toOpaqueId(), this, ConstString(), 0, nullptr, LLDB_INVALID_UID,
1054544a66d8SZachary Turner       lldb_private::Type::eEncodingIsUID, decl, func_sig_ast_type,
1055544a66d8SZachary Turner       lldb_private::Type::eResolveStateFull);
1056544a66d8SZachary Turner }
1057544a66d8SZachary Turner 
10582f7efbc9SZachary Turner TypeSP SymbolFileNativePDB::CreateType(PdbSymUid type_uid) {
10592f7efbc9SZachary Turner   const PdbTypeSymId &tsid = type_uid.asTypeSym();
10602f7efbc9SZachary Turner   TypeIndex index(tsid.index);
10612f7efbc9SZachary Turner 
10622f7efbc9SZachary Turner   if (index.getIndex() < TypeIndex::FirstNonSimpleIndex)
10632f7efbc9SZachary Turner     return CreateSimpleType(index);
10642f7efbc9SZachary Turner 
10652f7efbc9SZachary Turner   TpiStream &stream = tsid.is_ipi ? m_index->ipi() : m_index->tpi();
10662f7efbc9SZachary Turner   CVType cvt = stream.getType(index);
10672f7efbc9SZachary Turner 
10682f7efbc9SZachary Turner   if (cvt.kind() == LF_MODIFIER) {
10692f7efbc9SZachary Turner     ModifierRecord modifier;
10702f7efbc9SZachary Turner     llvm::cantFail(
10712f7efbc9SZachary Turner         TypeDeserializer::deserializeAs<ModifierRecord>(cvt, modifier));
10722f7efbc9SZachary Turner     return CreateModifierType(type_uid, modifier);
10732f7efbc9SZachary Turner   }
10742f7efbc9SZachary Turner 
10752f7efbc9SZachary Turner   if (cvt.kind() == LF_POINTER) {
10762f7efbc9SZachary Turner     PointerRecord pointer;
10772f7efbc9SZachary Turner     llvm::cantFail(
10782f7efbc9SZachary Turner         TypeDeserializer::deserializeAs<PointerRecord>(cvt, pointer));
10792f7efbc9SZachary Turner     return CreatePointerType(type_uid, pointer);
10802f7efbc9SZachary Turner   }
10812f7efbc9SZachary Turner 
10822f7efbc9SZachary Turner   if (IsClassRecord(cvt.kind())) {
10832f7efbc9SZachary Turner     ClassRecord cr;
10842f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ClassRecord>(cvt, cr));
10852f7efbc9SZachary Turner     return CreateTagType(type_uid, cr);
10862f7efbc9SZachary Turner   }
10872f7efbc9SZachary Turner 
10882f7efbc9SZachary Turner   if (cvt.kind() == LF_ENUM) {
10892f7efbc9SZachary Turner     EnumRecord er;
10902f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<EnumRecord>(cvt, er));
10912f7efbc9SZachary Turner     return CreateTagType(type_uid, er);
10922f7efbc9SZachary Turner   }
10932f7efbc9SZachary Turner 
10942f7efbc9SZachary Turner   if (cvt.kind() == LF_UNION) {
10952f7efbc9SZachary Turner     UnionRecord ur;
10962f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<UnionRecord>(cvt, ur));
10972f7efbc9SZachary Turner     return CreateTagType(type_uid, ur);
10982f7efbc9SZachary Turner   }
10992f7efbc9SZachary Turner 
1100511bff21SZachary Turner   if (cvt.kind() == LF_ARRAY) {
1101511bff21SZachary Turner     ArrayRecord ar;
1102511bff21SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ArrayRecord>(cvt, ar));
1103511bff21SZachary Turner     return CreateArrayType(type_uid, ar);
1104511bff21SZachary Turner   }
1105511bff21SZachary Turner 
1106544a66d8SZachary Turner   if (cvt.kind() == LF_PROCEDURE) {
1107544a66d8SZachary Turner     ProcedureRecord pr;
1108544a66d8SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ProcedureRecord>(cvt, pr));
1109544a66d8SZachary Turner     return CreateProcedureType(type_uid, pr);
1110544a66d8SZachary Turner   }
1111544a66d8SZachary Turner 
11122f7efbc9SZachary Turner   return nullptr;
11132f7efbc9SZachary Turner }
11142f7efbc9SZachary Turner 
11152f7efbc9SZachary Turner TypeSP SymbolFileNativePDB::CreateAndCacheType(PdbSymUid type_uid) {
11162f7efbc9SZachary Turner   // If they search for a UDT which is a forward ref, try and resolve the full
11172f7efbc9SZachary Turner   // decl and just map the forward ref uid to the full decl record.
11182f7efbc9SZachary Turner   llvm::Optional<PdbSymUid> full_decl_uid;
11192f7efbc9SZachary Turner   if (type_uid.tag() == PDB_SymType::UDT ||
11202f7efbc9SZachary Turner       type_uid.tag() == PDB_SymType::Enum) {
11212f7efbc9SZachary Turner     const PdbTypeSymId &type_id = type_uid.asTypeSym();
11222f7efbc9SZachary Turner     TypeIndex ti(type_id.index);
11232f7efbc9SZachary Turner     lldbassert(!ti.isSimple());
11242f7efbc9SZachary Turner     CVType cvt = m_index->tpi().getType(ti);
11252f7efbc9SZachary Turner 
11262f7efbc9SZachary Turner     if (IsForwardRefUdt(cvt)) {
11272f7efbc9SZachary Turner       auto expected_full_ti = m_index->tpi().findFullDeclForForwardRef(ti);
11282f7efbc9SZachary Turner       if (!expected_full_ti)
11292f7efbc9SZachary Turner         llvm::consumeError(expected_full_ti.takeError());
1130544a66d8SZachary Turner       else if (*expected_full_ti != ti) {
11312f7efbc9SZachary Turner         full_decl_uid = PdbSymUid::makeTypeSymId(
11322f7efbc9SZachary Turner             type_uid.tag(), *expected_full_ti, type_id.is_ipi);
11332f7efbc9SZachary Turner 
11342f7efbc9SZachary Turner         // It's possible that a lookup would occur for the full decl causing it
11352f7efbc9SZachary Turner         // to be cached, then a second lookup would occur for the forward decl.
11362f7efbc9SZachary Turner         // We don't want to create a second full decl, so make sure the full
11372f7efbc9SZachary Turner         // decl hasn't already been cached.
11382f7efbc9SZachary Turner         auto full_iter = m_types.find(full_decl_uid->toOpaqueId());
11392f7efbc9SZachary Turner         if (full_iter != m_types.end()) {
11402f7efbc9SZachary Turner           TypeSP result = full_iter->second;
11412f7efbc9SZachary Turner           // Map the forward decl to the TypeSP for the full decl so we can take
11422f7efbc9SZachary Turner           // the fast path next time.
11432f7efbc9SZachary Turner           m_types[type_uid.toOpaqueId()] = result;
11442f7efbc9SZachary Turner           return result;
11452f7efbc9SZachary Turner         }
11462f7efbc9SZachary Turner       }
11472f7efbc9SZachary Turner     }
11482f7efbc9SZachary Turner   }
11492f7efbc9SZachary Turner 
11502f7efbc9SZachary Turner   PdbSymUid best_uid = full_decl_uid ? *full_decl_uid : type_uid;
11512f7efbc9SZachary Turner   TypeSP result = CreateType(best_uid);
1152544a66d8SZachary Turner   if (!result)
1153544a66d8SZachary Turner     return nullptr;
11542f7efbc9SZachary Turner   m_types[best_uid.toOpaqueId()] = result;
11552f7efbc9SZachary Turner   // If we had both a forward decl and a full decl, make both point to the new
11562f7efbc9SZachary Turner   // type.
11572f7efbc9SZachary Turner   if (full_decl_uid)
11582f7efbc9SZachary Turner     m_types[type_uid.toOpaqueId()] = result;
11592f7efbc9SZachary Turner 
11602f7efbc9SZachary Turner   const PdbTypeSymId &type_id = best_uid.asTypeSym();
11612f7efbc9SZachary Turner   if (best_uid.tag() == PDB_SymType::UDT ||
11622f7efbc9SZachary Turner       best_uid.tag() == PDB_SymType::Enum) {
11632f7efbc9SZachary Turner     clang::TagDecl *record_decl =
11642f7efbc9SZachary Turner         m_clang->GetAsTagDecl(result->GetForwardCompilerType());
11652f7efbc9SZachary Turner     lldbassert(record_decl);
11662f7efbc9SZachary Turner 
11672f7efbc9SZachary Turner     TypeIndex ti(type_id.index);
11682f7efbc9SZachary Turner     m_uid_to_decl[best_uid.toOpaqueId()] = record_decl;
11692f7efbc9SZachary Turner     m_decl_to_status[record_decl] =
11702f7efbc9SZachary Turner         DeclStatus(best_uid.toOpaqueId(), Type::eResolveStateForward);
11712f7efbc9SZachary Turner   }
11722f7efbc9SZachary Turner   return result;
11732f7efbc9SZachary Turner }
11742f7efbc9SZachary Turner 
11752f7efbc9SZachary Turner TypeSP SymbolFileNativePDB::GetOrCreateType(PdbSymUid type_uid) {
11762f7efbc9SZachary Turner   lldbassert(PdbSymUid::isTypeSym(type_uid.tag()));
11772f7efbc9SZachary Turner   // We can't use try_emplace / overwrite here because the process of creating
11782f7efbc9SZachary Turner   // a type could create nested types, which could invalidate iterators.  So
11792f7efbc9SZachary Turner   // we have to do a 2-phase lookup / insert.
11802f7efbc9SZachary Turner   auto iter = m_types.find(type_uid.toOpaqueId());
11812f7efbc9SZachary Turner   if (iter != m_types.end())
11822f7efbc9SZachary Turner     return iter->second;
11832f7efbc9SZachary Turner 
11842f7efbc9SZachary Turner   return CreateAndCacheType(type_uid);
11852f7efbc9SZachary Turner }
11862f7efbc9SZachary Turner 
11879f727950SZachary Turner static DWARFExpression MakeGlobalLocationExpression(uint16_t section,
11889f727950SZachary Turner                                                     uint32_t offset,
11899f727950SZachary Turner                                                     ModuleSP module) {
11909f727950SZachary Turner   assert(section > 0);
11919f727950SZachary Turner   assert(module);
11929f727950SZachary Turner 
11939f727950SZachary Turner   const ArchSpec &architecture = module->GetArchitecture();
11949f727950SZachary Turner   ByteOrder byte_order = architecture.GetByteOrder();
11959f727950SZachary Turner   uint32_t address_size = architecture.GetAddressByteSize();
11969f727950SZachary Turner   uint32_t byte_size = architecture.GetDataByteSize();
11979f727950SZachary Turner   assert(byte_order != eByteOrderInvalid && address_size != 0);
11989f727950SZachary Turner 
11999f727950SZachary Turner   RegisterKind register_kind = eRegisterKindDWARF;
12009f727950SZachary Turner   StreamBuffer<32> stream(Stream::eBinary, address_size, byte_order);
12019f727950SZachary Turner   stream.PutHex8(DW_OP_addr);
12029f727950SZachary Turner 
12039f727950SZachary Turner   SectionList *section_list = module->GetSectionList();
12049f727950SZachary Turner   assert(section_list);
12059f727950SZachary Turner 
12069f727950SZachary Turner   // Section indices in PDB are 1-based, but in DWARF they are 0-based, so we
12079f727950SZachary Turner   // need to subtract 1.
12089f727950SZachary Turner   uint32_t section_idx = section - 1;
12099f727950SZachary Turner   if (section_idx >= section_list->GetSize())
12109f727950SZachary Turner     return DWARFExpression(nullptr);
12119f727950SZachary Turner 
12129f727950SZachary Turner   auto section_ptr = section_list->GetSectionAtIndex(section_idx);
12139f727950SZachary Turner   if (!section_ptr)
12149f727950SZachary Turner     return DWARFExpression(nullptr);
12159f727950SZachary Turner 
12169f727950SZachary Turner   stream.PutMaxHex64(section_ptr->GetFileAddress() + offset, address_size,
12179f727950SZachary Turner                      byte_order);
12189f727950SZachary Turner   DataBufferSP buffer =
12199f727950SZachary Turner       std::make_shared<DataBufferHeap>(stream.GetData(), stream.GetSize());
12209f727950SZachary Turner   DataExtractor extractor(buffer, byte_order, address_size, byte_size);
12219f727950SZachary Turner   DWARFExpression result(module, extractor, nullptr, 0, buffer->GetByteSize());
12229f727950SZachary Turner   result.SetRegisterKind(register_kind);
12239f727950SZachary Turner   return result;
12249f727950SZachary Turner }
12259f727950SZachary Turner 
12269f727950SZachary Turner VariableSP SymbolFileNativePDB::CreateGlobalVariable(PdbSymUid var_uid) {
12279f727950SZachary Turner   const PdbCuSymId &cu_sym = var_uid.asCuSym();
12289f727950SZachary Turner   lldbassert(cu_sym.global);
12299f727950SZachary Turner   CVSymbol sym = m_index->symrecords().readRecord(cu_sym.offset);
12309f727950SZachary Turner   lldb::ValueType scope = eValueTypeInvalid;
12319f727950SZachary Turner   TypeIndex ti;
12329f727950SZachary Turner   llvm::StringRef name;
12339f727950SZachary Turner   lldb::addr_t addr = 0;
12349f727950SZachary Turner   uint16_t section = 0;
12359f727950SZachary Turner   uint32_t offset = 0;
12369f727950SZachary Turner   bool is_external = false;
12379f727950SZachary Turner   switch (sym.kind()) {
12389f727950SZachary Turner   case S_GDATA32:
12399f727950SZachary Turner     is_external = true;
12409f727950SZachary Turner     LLVM_FALLTHROUGH;
12419f727950SZachary Turner   case S_LDATA32: {
12429f727950SZachary Turner     DataSym ds(sym.kind());
12439f727950SZachary Turner     llvm::cantFail(SymbolDeserializer::deserializeAs<DataSym>(sym, ds));
12449f727950SZachary Turner     ti = ds.Type;
12459f727950SZachary Turner     scope = (sym.kind() == S_GDATA32) ? eValueTypeVariableGlobal
12469f727950SZachary Turner                                       : eValueTypeVariableStatic;
12479f727950SZachary Turner     name = ds.Name;
12489f727950SZachary Turner     section = ds.Segment;
12499f727950SZachary Turner     offset = ds.DataOffset;
12509f727950SZachary Turner     addr = m_index->MakeVirtualAddress(ds.Segment, ds.DataOffset);
12519f727950SZachary Turner     break;
12529f727950SZachary Turner   }
12539f727950SZachary Turner   case S_GTHREAD32:
12549f727950SZachary Turner     is_external = true;
12559f727950SZachary Turner     LLVM_FALLTHROUGH;
12569f727950SZachary Turner   case S_LTHREAD32: {
12579f727950SZachary Turner     ThreadLocalDataSym tlds(sym.kind());
12589f727950SZachary Turner     llvm::cantFail(
12599f727950SZachary Turner         SymbolDeserializer::deserializeAs<ThreadLocalDataSym>(sym, tlds));
12609f727950SZachary Turner     ti = tlds.Type;
12619f727950SZachary Turner     name = tlds.Name;
12629f727950SZachary Turner     section = tlds.Segment;
12639f727950SZachary Turner     offset = tlds.DataOffset;
12649f727950SZachary Turner     addr = m_index->MakeVirtualAddress(tlds.Segment, tlds.DataOffset);
12659f727950SZachary Turner     scope = eValueTypeVariableThreadLocal;
12669f727950SZachary Turner     break;
12679f727950SZachary Turner   }
12689f727950SZachary Turner   default:
12699f727950SZachary Turner     llvm_unreachable("unreachable!");
12709f727950SZachary Turner   }
12719f727950SZachary Turner 
12729f727950SZachary Turner   CompUnitSP comp_unit;
12739f727950SZachary Turner   llvm::Optional<uint16_t> modi = m_index->GetModuleIndexForVa(addr);
12749f727950SZachary Turner   if (modi) {
12759f727950SZachary Turner     PdbSymUid cuid = PdbSymUid::makeCompilandId(*modi);
12769f727950SZachary Turner     CompilandIndexItem &cci = m_index->compilands().GetOrCreateCompiland(cuid);
12779f727950SZachary Turner     comp_unit = GetOrCreateCompileUnit(cci);
12789f727950SZachary Turner   }
12799f727950SZachary Turner 
12809f727950SZachary Turner   Declaration decl;
12819f727950SZachary Turner   PDB_SymType pdbst = GetPdbSymType(m_index->tpi(), ti);
12829f727950SZachary Turner   PdbSymUid tuid = PdbSymUid::makeTypeSymId(pdbst, ti, false);
12839f727950SZachary Turner   SymbolFileTypeSP type_sp =
12849f727950SZachary Turner       std::make_shared<SymbolFileType>(*this, tuid.toOpaqueId());
12859f727950SZachary Turner   Variable::RangeList ranges;
12869f727950SZachary Turner 
12879f727950SZachary Turner   DWARFExpression location = MakeGlobalLocationExpression(
12889f727950SZachary Turner       section, offset, GetObjectFile()->GetModule());
12899f727950SZachary Turner 
12909f727950SZachary Turner   std::string global_name("::");
12919f727950SZachary Turner   global_name += name;
12929f727950SZachary Turner   VariableSP var_sp = std::make_shared<Variable>(
12939f727950SZachary Turner       var_uid.toOpaqueId(), name.str().c_str(), global_name.c_str(), type_sp,
12949f727950SZachary Turner       scope, comp_unit.get(), ranges, &decl, location, is_external, false,
12959f727950SZachary Turner       false);
12969f727950SZachary Turner   var_sp->SetLocationIsConstantValueData(false);
12979f727950SZachary Turner 
12989f727950SZachary Turner   return var_sp;
12999f727950SZachary Turner }
13009f727950SZachary Turner 
13019f727950SZachary Turner VariableSP SymbolFileNativePDB::GetOrCreateGlobalVariable(PdbSymUid var_uid) {
13029f727950SZachary Turner   lldbassert(var_uid.isGlobalVariable());
13039f727950SZachary Turner 
13049f727950SZachary Turner   auto emplace_result =
13059f727950SZachary Turner       m_global_vars.try_emplace(var_uid.toOpaqueId(), nullptr);
13069f727950SZachary Turner   if (emplace_result.second)
13079f727950SZachary Turner     emplace_result.first->second = CreateGlobalVariable(var_uid);
13089f727950SZachary Turner 
13099f727950SZachary Turner   return emplace_result.first->second;
13109f727950SZachary Turner }
13119f727950SZachary Turner 
13122f7efbc9SZachary Turner lldb::TypeSP
13132f7efbc9SZachary Turner SymbolFileNativePDB::GetOrCreateType(llvm::codeview::TypeIndex ti) {
13142f7efbc9SZachary Turner   PDB_SymType pdbst = GetPdbSymType(m_index->tpi(), ti);
13152f7efbc9SZachary Turner   PdbSymUid tuid = PdbSymUid::makeTypeSymId(pdbst, ti, false);
13162f7efbc9SZachary Turner   return GetOrCreateType(tuid);
13172f7efbc9SZachary Turner }
13182f7efbc9SZachary Turner 
1319307f5ae8SZachary Turner FunctionSP SymbolFileNativePDB::GetOrCreateFunction(PdbSymUid func_uid,
1320307f5ae8SZachary Turner                                                     const SymbolContext &sc) {
1321307f5ae8SZachary Turner   lldbassert(func_uid.tag() == PDB_SymType::Function);
1322307f5ae8SZachary Turner   auto emplace_result = m_functions.try_emplace(func_uid.toOpaqueId(), nullptr);
1323307f5ae8SZachary Turner   if (emplace_result.second)
1324307f5ae8SZachary Turner     emplace_result.first->second = CreateFunction(func_uid, sc);
1325307f5ae8SZachary Turner 
1326307f5ae8SZachary Turner   lldbassert(emplace_result.first->second);
1327307f5ae8SZachary Turner   return emplace_result.first->second;
1328307f5ae8SZachary Turner }
1329307f5ae8SZachary Turner 
1330307f5ae8SZachary Turner CompUnitSP
1331307f5ae8SZachary Turner SymbolFileNativePDB::GetOrCreateCompileUnit(const CompilandIndexItem &cci) {
1332307f5ae8SZachary Turner   auto emplace_result =
1333307f5ae8SZachary Turner       m_compilands.try_emplace(cci.m_uid.toOpaqueId(), nullptr);
1334307f5ae8SZachary Turner   if (emplace_result.second)
1335307f5ae8SZachary Turner     emplace_result.first->second = CreateCompileUnit(cci);
1336307f5ae8SZachary Turner 
1337307f5ae8SZachary Turner   lldbassert(emplace_result.first->second);
1338307f5ae8SZachary Turner   return emplace_result.first->second;
1339307f5ae8SZachary Turner }
1340307f5ae8SZachary Turner 
1341307f5ae8SZachary Turner lldb::CompUnitSP SymbolFileNativePDB::ParseCompileUnitAtIndex(uint32_t index) {
1342307f5ae8SZachary Turner   if (index >= GetNumCompileUnits())
1343307f5ae8SZachary Turner     return CompUnitSP();
1344307f5ae8SZachary Turner   lldbassert(index < UINT16_MAX);
1345307f5ae8SZachary Turner   if (index >= UINT16_MAX)
1346307f5ae8SZachary Turner     return nullptr;
1347307f5ae8SZachary Turner 
1348307f5ae8SZachary Turner   CompilandIndexItem &item = m_index->compilands().GetOrCreateCompiland(index);
1349307f5ae8SZachary Turner 
1350307f5ae8SZachary Turner   return GetOrCreateCompileUnit(item);
1351307f5ae8SZachary Turner }
1352307f5ae8SZachary Turner 
1353b96181c2SZachary Turner lldb::LanguageType
1354b96181c2SZachary Turner SymbolFileNativePDB::ParseCompileUnitLanguage(const SymbolContext &sc) {
1355307f5ae8SZachary Turner   // What fields should I expect to be filled out on the SymbolContext?  Is it
1356307f5ae8SZachary Turner   // safe to assume that `sc.comp_unit` is valid?
1357307f5ae8SZachary Turner   if (!sc.comp_unit)
1358307f5ae8SZachary Turner     return lldb::eLanguageTypeUnknown;
1359307f5ae8SZachary Turner   PdbSymUid uid = PdbSymUid::fromOpaqueId(sc.comp_unit->GetID());
1360307f5ae8SZachary Turner   lldbassert(uid.tag() == PDB_SymType::Compiland);
1361307f5ae8SZachary Turner 
1362307f5ae8SZachary Turner   CompilandIndexItem *item = m_index->compilands().GetCompiland(uid);
1363307f5ae8SZachary Turner   lldbassert(item);
1364307f5ae8SZachary Turner   if (!item->m_compile_opts)
1365307f5ae8SZachary Turner     return lldb::eLanguageTypeUnknown;
1366307f5ae8SZachary Turner 
1367307f5ae8SZachary Turner   return TranslateLanguage(item->m_compile_opts->getLanguage());
1368307f5ae8SZachary Turner }
1369307f5ae8SZachary Turner 
1370b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseCompileUnitFunctions(const SymbolContext &sc) {
1371307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
1372307f5ae8SZachary Turner   return false;
1373307f5ae8SZachary Turner }
1374307f5ae8SZachary Turner 
1375307f5ae8SZachary Turner static bool NeedsResolvedCompileUnit(uint32_t resolve_scope) {
1376307f5ae8SZachary Turner   // If any of these flags are set, we need to resolve the compile unit.
1377307f5ae8SZachary Turner   uint32_t flags = eSymbolContextCompUnit;
1378307f5ae8SZachary Turner   flags |= eSymbolContextVariable;
1379307f5ae8SZachary Turner   flags |= eSymbolContextFunction;
1380307f5ae8SZachary Turner   flags |= eSymbolContextBlock;
1381307f5ae8SZachary Turner   flags |= eSymbolContextLineEntry;
1382307f5ae8SZachary Turner   return (resolve_scope & flags) != 0;
1383307f5ae8SZachary Turner }
1384307f5ae8SZachary Turner 
1385991e4453SZachary Turner uint32_t SymbolFileNativePDB::ResolveSymbolContext(
1386991e4453SZachary Turner     const Address &addr, SymbolContextItem resolve_scope, SymbolContext &sc) {
1387307f5ae8SZachary Turner   uint32_t resolved_flags = 0;
1388307f5ae8SZachary Turner   lldb::addr_t file_addr = addr.GetFileAddress();
1389307f5ae8SZachary Turner 
1390307f5ae8SZachary Turner   if (NeedsResolvedCompileUnit(resolve_scope)) {
1391307f5ae8SZachary Turner     llvm::Optional<uint16_t> modi = m_index->GetModuleIndexForVa(file_addr);
1392307f5ae8SZachary Turner     if (!modi)
1393307f5ae8SZachary Turner       return 0;
1394307f5ae8SZachary Turner     PdbSymUid cuid = PdbSymUid::makeCompilandId(*modi);
1395307f5ae8SZachary Turner     CompilandIndexItem *cci = m_index->compilands().GetCompiland(cuid);
1396307f5ae8SZachary Turner     if (!cci)
1397307f5ae8SZachary Turner       return 0;
1398307f5ae8SZachary Turner 
1399307f5ae8SZachary Turner     sc.comp_unit = GetOrCreateCompileUnit(*cci).get();
1400307f5ae8SZachary Turner     resolved_flags |= eSymbolContextCompUnit;
1401307f5ae8SZachary Turner   }
1402307f5ae8SZachary Turner 
1403307f5ae8SZachary Turner   if (resolve_scope & eSymbolContextFunction) {
1404307f5ae8SZachary Turner     lldbassert(sc.comp_unit);
1405307f5ae8SZachary Turner     std::vector<SymbolAndUid> matches = m_index->FindSymbolsByVa(file_addr);
1406307f5ae8SZachary Turner     for (const auto &match : matches) {
1407307f5ae8SZachary Turner       if (match.uid.tag() != PDB_SymType::Function)
1408307f5ae8SZachary Turner         continue;
1409307f5ae8SZachary Turner       sc.function = GetOrCreateFunction(match.uid, sc).get();
1410307f5ae8SZachary Turner     }
1411307f5ae8SZachary Turner     resolved_flags |= eSymbolContextFunction;
1412307f5ae8SZachary Turner   }
1413307f5ae8SZachary Turner 
1414307f5ae8SZachary Turner   if (resolve_scope & eSymbolContextLineEntry) {
1415307f5ae8SZachary Turner     lldbassert(sc.comp_unit);
1416307f5ae8SZachary Turner     if (auto *line_table = sc.comp_unit->GetLineTable()) {
1417307f5ae8SZachary Turner       if (line_table->FindLineEntryByAddress(addr, sc.line_entry))
1418307f5ae8SZachary Turner         resolved_flags |= eSymbolContextLineEntry;
1419307f5ae8SZachary Turner     }
1420307f5ae8SZachary Turner   }
1421307f5ae8SZachary Turner 
1422307f5ae8SZachary Turner   return resolved_flags;
1423307f5ae8SZachary Turner }
1424307f5ae8SZachary Turner 
1425307f5ae8SZachary Turner static void AppendLineEntryToSequence(LineTable &table, LineSequence &sequence,
1426307f5ae8SZachary Turner                                       const CompilandIndexItem &cci,
1427307f5ae8SZachary Turner                                       lldb::addr_t base_addr,
1428307f5ae8SZachary Turner                                       uint32_t file_number,
1429307f5ae8SZachary Turner                                       const LineFragmentHeader &block,
1430307f5ae8SZachary Turner                                       const LineNumberEntry &cur) {
1431307f5ae8SZachary Turner   LineInfo cur_info(cur.Flags);
1432307f5ae8SZachary Turner 
1433307f5ae8SZachary Turner   if (cur_info.isAlwaysStepInto() || cur_info.isNeverStepInto())
1434307f5ae8SZachary Turner     return;
1435307f5ae8SZachary Turner 
1436307f5ae8SZachary Turner   uint64_t addr = base_addr + cur.Offset;
1437307f5ae8SZachary Turner 
1438307f5ae8SZachary Turner   bool is_statement = cur_info.isStatement();
1439307f5ae8SZachary Turner   bool is_prologue = IsFunctionPrologue(cci, addr);
1440307f5ae8SZachary Turner   bool is_epilogue = IsFunctionEpilogue(cci, addr);
1441307f5ae8SZachary Turner 
1442307f5ae8SZachary Turner   uint32_t lno = cur_info.getStartLine();
1443307f5ae8SZachary Turner 
1444307f5ae8SZachary Turner   table.AppendLineEntryToSequence(&sequence, addr, lno, 0, file_number,
1445307f5ae8SZachary Turner                                   is_statement, false, is_prologue, is_epilogue,
1446307f5ae8SZachary Turner                                   false);
1447307f5ae8SZachary Turner }
1448307f5ae8SZachary Turner 
1449307f5ae8SZachary Turner static void TerminateLineSequence(LineTable &table,
1450307f5ae8SZachary Turner                                   const LineFragmentHeader &block,
1451307f5ae8SZachary Turner                                   lldb::addr_t base_addr, uint32_t file_number,
1452307f5ae8SZachary Turner                                   uint32_t last_line,
1453307f5ae8SZachary Turner                                   std::unique_ptr<LineSequence> seq) {
1454307f5ae8SZachary Turner   // The end is always a terminal entry, so insert it regardless.
1455307f5ae8SZachary Turner   table.AppendLineEntryToSequence(seq.get(), base_addr + block.CodeSize,
1456307f5ae8SZachary Turner                                   last_line, 0, file_number, false, false,
1457307f5ae8SZachary Turner                                   false, false, true);
1458307f5ae8SZachary Turner   table.InsertSequence(seq.release());
1459307f5ae8SZachary Turner }
1460307f5ae8SZachary Turner 
1461b96181c2SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitLineTable(const SymbolContext &sc) {
1462307f5ae8SZachary Turner   // Unfortunately LLDB is set up to parse the entire compile unit line table
1463307f5ae8SZachary Turner   // all at once, even if all it really needs is line info for a specific
1464307f5ae8SZachary Turner   // function.  In the future it would be nice if it could set the sc.m_function
1465307f5ae8SZachary Turner   // member, and we could only get the line info for the function in question.
1466307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
1467307f5ae8SZachary Turner   PdbSymUid cu_id = PdbSymUid::fromOpaqueId(sc.comp_unit->GetID());
1468307f5ae8SZachary Turner   lldbassert(cu_id.isCompiland());
1469307f5ae8SZachary Turner   CompilandIndexItem *cci = m_index->compilands().GetCompiland(cu_id);
1470307f5ae8SZachary Turner   lldbassert(cci);
1471307f5ae8SZachary Turner   auto line_table = llvm::make_unique<LineTable>(sc.comp_unit);
1472307f5ae8SZachary Turner 
1473307f5ae8SZachary Turner   // This is basically a copy of the .debug$S subsections from all original COFF
1474307f5ae8SZachary Turner   // object files merged together with address relocations applied.  We are
1475307f5ae8SZachary Turner   // looking for all DEBUG_S_LINES subsections.
1476307f5ae8SZachary Turner   for (const DebugSubsectionRecord &dssr :
1477307f5ae8SZachary Turner        cci->m_debug_stream.getSubsectionsArray()) {
1478307f5ae8SZachary Turner     if (dssr.kind() != DebugSubsectionKind::Lines)
1479307f5ae8SZachary Turner       continue;
1480307f5ae8SZachary Turner 
1481307f5ae8SZachary Turner     DebugLinesSubsectionRef lines;
1482307f5ae8SZachary Turner     llvm::BinaryStreamReader reader(dssr.getRecordData());
1483307f5ae8SZachary Turner     if (auto EC = lines.initialize(reader)) {
1484307f5ae8SZachary Turner       llvm::consumeError(std::move(EC));
1485307f5ae8SZachary Turner       return false;
1486307f5ae8SZachary Turner     }
1487307f5ae8SZachary Turner 
1488307f5ae8SZachary Turner     const LineFragmentHeader *lfh = lines.header();
1489307f5ae8SZachary Turner     uint64_t virtual_addr =
1490307f5ae8SZachary Turner         m_index->MakeVirtualAddress(lfh->RelocSegment, lfh->RelocOffset);
1491307f5ae8SZachary Turner 
1492307f5ae8SZachary Turner     const auto &checksums = cci->m_strings.checksums().getArray();
1493307f5ae8SZachary Turner     const auto &strings = cci->m_strings.strings();
1494307f5ae8SZachary Turner     for (const LineColumnEntry &group : lines) {
1495307f5ae8SZachary Turner       // Indices in this structure are actually offsets of records in the
1496307f5ae8SZachary Turner       // DEBUG_S_FILECHECKSUMS subsection.  Those entries then have an index
1497307f5ae8SZachary Turner       // into the global PDB string table.
1498307f5ae8SZachary Turner       auto iter = checksums.at(group.NameIndex);
1499307f5ae8SZachary Turner       if (iter == checksums.end())
1500307f5ae8SZachary Turner         continue;
1501307f5ae8SZachary Turner 
1502307f5ae8SZachary Turner       llvm::Expected<llvm::StringRef> efn =
1503307f5ae8SZachary Turner           strings.getString(iter->FileNameOffset);
1504307f5ae8SZachary Turner       if (!efn) {
1505307f5ae8SZachary Turner         llvm::consumeError(efn.takeError());
1506307f5ae8SZachary Turner         continue;
1507307f5ae8SZachary Turner       }
1508307f5ae8SZachary Turner 
1509307f5ae8SZachary Turner       // LLDB wants the index of the file in the list of support files.
1510307f5ae8SZachary Turner       auto fn_iter = llvm::find(cci->m_file_list, *efn);
1511307f5ae8SZachary Turner       lldbassert(fn_iter != cci->m_file_list.end());
1512307f5ae8SZachary Turner       uint32_t file_index = std::distance(cci->m_file_list.begin(), fn_iter);
1513307f5ae8SZachary Turner 
1514307f5ae8SZachary Turner       std::unique_ptr<LineSequence> sequence(
1515307f5ae8SZachary Turner           line_table->CreateLineSequenceContainer());
1516307f5ae8SZachary Turner       lldbassert(!group.LineNumbers.empty());
1517307f5ae8SZachary Turner 
1518307f5ae8SZachary Turner       for (const LineNumberEntry &entry : group.LineNumbers) {
1519307f5ae8SZachary Turner         AppendLineEntryToSequence(*line_table, *sequence, *cci, virtual_addr,
1520307f5ae8SZachary Turner                                   file_index, *lfh, entry);
1521307f5ae8SZachary Turner       }
1522307f5ae8SZachary Turner       LineInfo last_line(group.LineNumbers.back().Flags);
1523307f5ae8SZachary Turner       TerminateLineSequence(*line_table, *lfh, virtual_addr, file_index,
1524307f5ae8SZachary Turner                             last_line.getEndLine(), std::move(sequence));
1525307f5ae8SZachary Turner     }
1526307f5ae8SZachary Turner   }
1527307f5ae8SZachary Turner 
1528307f5ae8SZachary Turner   if (line_table->GetSize() == 0)
1529307f5ae8SZachary Turner     return false;
1530307f5ae8SZachary Turner 
1531307f5ae8SZachary Turner   sc.comp_unit->SetLineTable(line_table.release());
1532307f5ae8SZachary Turner   return true;
1533307f5ae8SZachary Turner }
1534307f5ae8SZachary Turner 
1535b96181c2SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitDebugMacros(const SymbolContext &sc) {
1536307f5ae8SZachary Turner   // PDB doesn't contain information about macros
1537307f5ae8SZachary Turner   return false;
1538307f5ae8SZachary Turner }
1539307f5ae8SZachary Turner 
1540307f5ae8SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitSupportFiles(
1541b96181c2SZachary Turner     const SymbolContext &sc, FileSpecList &support_files) {
1542307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
1543307f5ae8SZachary Turner 
1544307f5ae8SZachary Turner   PdbSymUid comp_uid = PdbSymUid::fromOpaqueId(sc.comp_unit->GetID());
1545307f5ae8SZachary Turner   lldbassert(comp_uid.tag() == PDB_SymType::Compiland);
1546307f5ae8SZachary Turner 
1547307f5ae8SZachary Turner   const CompilandIndexItem *cci = m_index->compilands().GetCompiland(comp_uid);
1548307f5ae8SZachary Turner   lldbassert(cci);
1549307f5ae8SZachary Turner 
1550307f5ae8SZachary Turner   for (llvm::StringRef f : cci->m_file_list) {
1551307f5ae8SZachary Turner     FileSpec::Style style =
1552307f5ae8SZachary Turner         f.startswith("/") ? FileSpec::Style::posix : FileSpec::Style::windows;
15538f3be7a3SJonas Devlieghere     FileSpec spec(f, style);
1554307f5ae8SZachary Turner     support_files.Append(spec);
1555307f5ae8SZachary Turner   }
1556307f5ae8SZachary Turner 
1557307f5ae8SZachary Turner   return true;
1558307f5ae8SZachary Turner }
1559307f5ae8SZachary Turner 
1560307f5ae8SZachary Turner bool SymbolFileNativePDB::ParseImportedModules(
1561b96181c2SZachary Turner     const SymbolContext &sc, std::vector<ConstString> &imported_modules) {
1562307f5ae8SZachary Turner   // PDB does not yet support module debug info
1563307f5ae8SZachary Turner   return false;
1564307f5ae8SZachary Turner }
1565307f5ae8SZachary Turner 
1566b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseFunctionBlocks(const SymbolContext &sc) {
1567307f5ae8SZachary Turner   lldbassert(sc.comp_unit && sc.function);
1568307f5ae8SZachary Turner   return 0;
1569307f5ae8SZachary Turner }
1570307f5ae8SZachary Turner 
15714911023fSZachary Turner void SymbolFileNativePDB::DumpClangAST(Stream &s) {
15724911023fSZachary Turner   if (!m_clang)
15734911023fSZachary Turner     return;
15744911023fSZachary Turner   m_clang->Dump(s);
15754911023fSZachary Turner }
15764911023fSZachary Turner 
15779f727950SZachary Turner uint32_t SymbolFileNativePDB::FindGlobalVariables(
15789f727950SZachary Turner     const ConstString &name, const CompilerDeclContext *parent_decl_ctx,
15799f727950SZachary Turner     uint32_t max_matches, VariableList &variables) {
15809f727950SZachary Turner   using SymbolAndOffset = std::pair<uint32_t, llvm::codeview::CVSymbol>;
15819f727950SZachary Turner 
15829f727950SZachary Turner   std::vector<SymbolAndOffset> results = m_index->globals().findRecordsByName(
15839f727950SZachary Turner       name.GetStringRef(), m_index->symrecords());
15849f727950SZachary Turner   for (const SymbolAndOffset &result : results) {
15859f727950SZachary Turner     VariableSP var;
15869f727950SZachary Turner     switch (result.second.kind()) {
15879f727950SZachary Turner     case SymbolKind::S_GDATA32:
15889f727950SZachary Turner     case SymbolKind::S_LDATA32:
15899f727950SZachary Turner     case SymbolKind::S_GTHREAD32:
15909f727950SZachary Turner     case SymbolKind::S_LTHREAD32: {
15919f727950SZachary Turner       PdbSymUid uid = PdbSymUid::makeGlobalVariableUid(result.first);
15929f727950SZachary Turner       var = GetOrCreateGlobalVariable(uid);
15939f727950SZachary Turner       variables.AddVariable(var);
15949f727950SZachary Turner       break;
15959f727950SZachary Turner     }
15969f727950SZachary Turner     default:
15979f727950SZachary Turner       continue;
15989f727950SZachary Turner     }
15999f727950SZachary Turner   }
16009f727950SZachary Turner   return variables.GetSize();
16019f727950SZachary Turner }
16029f727950SZachary Turner 
1603307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::FindFunctions(
1604b96181c2SZachary Turner     const ConstString &name, const CompilerDeclContext *parent_decl_ctx,
1605117b1fa1SZachary Turner     FunctionNameType name_type_mask, bool include_inlines, bool append,
1606b96181c2SZachary Turner     SymbolContextList &sc_list) {
1607307f5ae8SZachary Turner   // For now we only support lookup by method name.
1608307f5ae8SZachary Turner   if (!(name_type_mask & eFunctionNameTypeMethod))
1609307f5ae8SZachary Turner     return 0;
1610307f5ae8SZachary Turner 
1611307f5ae8SZachary Turner   using SymbolAndOffset = std::pair<uint32_t, llvm::codeview::CVSymbol>;
1612307f5ae8SZachary Turner 
1613307f5ae8SZachary Turner   std::vector<SymbolAndOffset> matches = m_index->globals().findRecordsByName(
1614307f5ae8SZachary Turner       name.GetStringRef(), m_index->symrecords());
1615307f5ae8SZachary Turner   for (const SymbolAndOffset &match : matches) {
1616307f5ae8SZachary Turner     if (match.second.kind() != S_PROCREF && match.second.kind() != S_LPROCREF)
1617307f5ae8SZachary Turner       continue;
1618307f5ae8SZachary Turner     ProcRefSym proc(match.second.kind());
1619307f5ae8SZachary Turner     cantFail(SymbolDeserializer::deserializeAs<ProcRefSym>(match.second, proc));
1620307f5ae8SZachary Turner 
1621307f5ae8SZachary Turner     if (!IsValidRecord(proc))
1622307f5ae8SZachary Turner       continue;
1623307f5ae8SZachary Turner 
1624307f5ae8SZachary Turner     PdbSymUid cuid = PdbSymUid::makeCompilandId(proc);
1625307f5ae8SZachary Turner     CompilandIndexItem &cci = m_index->compilands().GetOrCreateCompiland(cuid);
1626b96181c2SZachary Turner     SymbolContext sc;
1627307f5ae8SZachary Turner 
1628307f5ae8SZachary Turner     sc.comp_unit = GetOrCreateCompileUnit(cci).get();
1629307f5ae8SZachary Turner     sc.module_sp = sc.comp_unit->GetModule();
1630307f5ae8SZachary Turner     PdbSymUid func_uid = PdbSymUid::makeCuSymId(proc);
1631307f5ae8SZachary Turner     sc.function = GetOrCreateFunction(func_uid, sc).get();
1632307f5ae8SZachary Turner 
1633307f5ae8SZachary Turner     sc_list.Append(sc);
1634307f5ae8SZachary Turner   }
1635307f5ae8SZachary Turner 
1636307f5ae8SZachary Turner   return sc_list.GetSize();
1637307f5ae8SZachary Turner }
1638307f5ae8SZachary Turner 
1639b96181c2SZachary Turner uint32_t SymbolFileNativePDB::FindFunctions(const RegularExpression &regex,
1640307f5ae8SZachary Turner                                             bool include_inlines, bool append,
1641b96181c2SZachary Turner                                             SymbolContextList &sc_list) {
1642307f5ae8SZachary Turner   return 0;
1643307f5ae8SZachary Turner }
1644307f5ae8SZachary Turner 
1645b96181c2SZachary Turner uint32_t SymbolFileNativePDB::FindTypes(
1646b96181c2SZachary Turner     const SymbolContext &sc, const ConstString &name,
1647b96181c2SZachary Turner     const CompilerDeclContext *parent_decl_ctx, bool append,
1648b96181c2SZachary Turner     uint32_t max_matches, llvm::DenseSet<SymbolFile *> &searched_symbol_files,
1649b96181c2SZachary Turner     TypeMap &types) {
16502f7efbc9SZachary Turner   if (!append)
16512f7efbc9SZachary Turner     types.Clear();
16522f7efbc9SZachary Turner   if (!name)
1653b96181c2SZachary Turner     return 0;
16542f7efbc9SZachary Turner 
16552f7efbc9SZachary Turner   searched_symbol_files.clear();
16562f7efbc9SZachary Turner   searched_symbol_files.insert(this);
16572f7efbc9SZachary Turner 
16582f7efbc9SZachary Turner   // There is an assumption 'name' is not a regex
16592f7efbc9SZachary Turner   size_t match_count = FindTypesByName(name.GetStringRef(), max_matches, types);
16602f7efbc9SZachary Turner 
16612f7efbc9SZachary Turner   return match_count;
1662b96181c2SZachary Turner }
1663b96181c2SZachary Turner 
1664b96181c2SZachary Turner size_t
1665b96181c2SZachary Turner SymbolFileNativePDB::FindTypes(const std::vector<CompilerContext> &context,
1666b96181c2SZachary Turner                                bool append, TypeMap &types) {
1667b96181c2SZachary Turner   return 0;
1668b96181c2SZachary Turner }
1669b96181c2SZachary Turner 
16702f7efbc9SZachary Turner size_t SymbolFileNativePDB::FindTypesByName(llvm::StringRef name,
16712f7efbc9SZachary Turner                                             uint32_t max_matches,
16722f7efbc9SZachary Turner                                             TypeMap &types) {
16732f7efbc9SZachary Turner 
16742f7efbc9SZachary Turner   size_t match_count = 0;
16752f7efbc9SZachary Turner   std::vector<TypeIndex> matches = m_index->tpi().findRecordsByName(name);
16762f7efbc9SZachary Turner   if (max_matches > 0 && max_matches < matches.size())
16772f7efbc9SZachary Turner     matches.resize(max_matches);
16782f7efbc9SZachary Turner 
16792f7efbc9SZachary Turner   for (TypeIndex ti : matches) {
16802f7efbc9SZachary Turner     TypeSP type = GetOrCreateType(ti);
16812f7efbc9SZachary Turner     if (!type)
16822f7efbc9SZachary Turner       continue;
16832f7efbc9SZachary Turner 
16842f7efbc9SZachary Turner     types.Insert(type);
16852f7efbc9SZachary Turner     ++match_count;
16862f7efbc9SZachary Turner   }
16872f7efbc9SZachary Turner   return match_count;
16882f7efbc9SZachary Turner }
16892f7efbc9SZachary Turner 
1690b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseTypes(const SymbolContext &sc) { return 0; }
1691b96181c2SZachary Turner 
1692b96181c2SZachary Turner Type *SymbolFileNativePDB::ResolveTypeUID(lldb::user_id_t type_uid) {
16932f7efbc9SZachary Turner   auto iter = m_types.find(type_uid);
16942f7efbc9SZachary Turner   // lldb should not be passing us non-sensical type uids.  the only way it
16952f7efbc9SZachary Turner   // could have a type uid in the first place is if we handed it out, in which
16969f727950SZachary Turner   // case we should know about the type.  However, that doesn't mean we've
16979f727950SZachary Turner   // instantiated it yet.  We can vend out a UID for a future type.  So if the
16989f727950SZachary Turner   // type doesn't exist, let's instantiate it now.
16999f727950SZachary Turner   if (iter != m_types.end())
17002f7efbc9SZachary Turner     return &*iter->second;
17019f727950SZachary Turner 
17029f727950SZachary Turner   PdbSymUid uid = PdbSymUid::fromOpaqueId(type_uid);
17039f727950SZachary Turner   lldbassert(uid.isTypeSym(uid.tag()));
17049f727950SZachary Turner   const PdbTypeSymId &type_id = uid.asTypeSym();
17059f727950SZachary Turner   TypeIndex ti(type_id.index);
17069f727950SZachary Turner   if (ti.isNoneType())
17079f727950SZachary Turner     return nullptr;
17089f727950SZachary Turner 
17099f727950SZachary Turner   TypeSP type_sp = CreateAndCacheType(uid);
17109f727950SZachary Turner   return &*type_sp;
1711b96181c2SZachary Turner }
1712b96181c2SZachary Turner 
1713eca07c59SAdrian Prantl llvm::Optional<SymbolFile::ArrayInfo>
1714eca07c59SAdrian Prantl SymbolFileNativePDB::GetDynamicArrayInfoForUID(
1715eca07c59SAdrian Prantl     lldb::user_id_t type_uid, const lldb_private::ExecutionContext *exe_ctx) {
1716eca07c59SAdrian Prantl   return llvm::None;
1717eca07c59SAdrian Prantl }
1718eca07c59SAdrian Prantl 
1719eca07c59SAdrian Prantl 
1720b96181c2SZachary Turner bool SymbolFileNativePDB::CompleteType(CompilerType &compiler_type) {
17212f7efbc9SZachary Turner   // If this is not in our map, it's an error.
17222f7efbc9SZachary Turner   clang::TagDecl *tag_decl = m_clang->GetAsTagDecl(compiler_type);
17232f7efbc9SZachary Turner   lldbassert(tag_decl);
17242f7efbc9SZachary Turner   auto status_iter = m_decl_to_status.find(tag_decl);
17252f7efbc9SZachary Turner   lldbassert(status_iter != m_decl_to_status.end());
17262f7efbc9SZachary Turner 
17272f7efbc9SZachary Turner   // If it's already complete, just return.
17282f7efbc9SZachary Turner   DeclStatus &status = status_iter->second;
17292f7efbc9SZachary Turner   if (status.status == Type::eResolveStateFull)
17302f7efbc9SZachary Turner     return true;
17312f7efbc9SZachary Turner 
17322f7efbc9SZachary Turner   PdbSymUid uid = PdbSymUid::fromOpaqueId(status.uid);
17332f7efbc9SZachary Turner   lldbassert(uid.tag() == PDB_SymType::UDT || uid.tag() == PDB_SymType::Enum);
17342f7efbc9SZachary Turner 
17352f7efbc9SZachary Turner   const PdbTypeSymId &type_id = uid.asTypeSym();
17362f7efbc9SZachary Turner 
17372f7efbc9SZachary Turner   ClangASTContext::SetHasExternalStorage(compiler_type.GetOpaqueQualType(),
17382f7efbc9SZachary Turner                                          false);
17392f7efbc9SZachary Turner 
17402f7efbc9SZachary Turner   // In CreateAndCacheType, we already go out of our way to resolve forward
17412f7efbc9SZachary Turner   // ref UDTs to full decls, and the uids we vend out always refer to full
17422f7efbc9SZachary Turner   // decls if a full decl exists in the debug info.  So if we don't have a full
17432f7efbc9SZachary Turner   // decl here, it means one doesn't exist in the debug info, and we can't
17442f7efbc9SZachary Turner   // complete the type.
17452f7efbc9SZachary Turner   CVType cvt = m_index->tpi().getType(TypeIndex(type_id.index));
17462f7efbc9SZachary Turner   if (IsForwardRefUdt(cvt))
17472f7efbc9SZachary Turner     return false;
17482f7efbc9SZachary Turner 
17492f7efbc9SZachary Turner   auto types_iter = m_types.find(uid.toOpaqueId());
17502f7efbc9SZachary Turner   lldbassert(types_iter != m_types.end());
17512f7efbc9SZachary Turner 
1752511bff21SZachary Turner   if (cvt.kind() == LF_MODIFIER) {
1753511bff21SZachary Turner     TypeIndex unmodified_type = LookThroughModifierRecord(cvt);
1754511bff21SZachary Turner     cvt = m_index->tpi().getType(unmodified_type);
1755511bff21SZachary Turner     // LF_MODIFIERS usually point to forward decls, so this is the one case
1756511bff21SZachary Turner     // where we won't have been able to resolve a forward decl to a full decl
1757511bff21SZachary Turner     // earlier on.  So we need to do that now.
1758511bff21SZachary Turner     if (IsForwardRefUdt(cvt)) {
1759511bff21SZachary Turner       llvm::Expected<TypeIndex> expected_full_ti =
1760511bff21SZachary Turner           m_index->tpi().findFullDeclForForwardRef(unmodified_type);
1761511bff21SZachary Turner       if (!expected_full_ti) {
1762511bff21SZachary Turner         llvm::consumeError(expected_full_ti.takeError());
1763511bff21SZachary Turner         return false;
1764511bff21SZachary Turner       }
1765511bff21SZachary Turner       cvt = m_index->tpi().getType(*expected_full_ti);
1766511bff21SZachary Turner       lldbassert(!IsForwardRefUdt(cvt));
1767511bff21SZachary Turner       unmodified_type = *expected_full_ti;
1768511bff21SZachary Turner     }
1769511bff21SZachary Turner     uid = PdbSymUid::makeTypeSymId(uid.tag(), unmodified_type, false);
1770511bff21SZachary Turner   }
17712f7efbc9SZachary Turner   TypeIndex field_list_ti = GetFieldListIndex(cvt);
17722f7efbc9SZachary Turner   CVType field_list_cvt = m_index->tpi().getType(field_list_ti);
17732f7efbc9SZachary Turner   if (field_list_cvt.kind() != LF_FIELDLIST)
17742f7efbc9SZachary Turner     return false;
17752f7efbc9SZachary Turner 
17762f7efbc9SZachary Turner   // Visit all members of this class, then perform any finalization necessary
17772f7efbc9SZachary Turner   // to complete the class.
17782f7efbc9SZachary Turner   UdtRecordCompleter completer(uid, compiler_type, *tag_decl, *this);
17792f7efbc9SZachary Turner   auto error =
17802f7efbc9SZachary Turner       llvm::codeview::visitMemberRecordStream(field_list_cvt.data(), completer);
17812f7efbc9SZachary Turner   completer.complete();
17822f7efbc9SZachary Turner 
17832f7efbc9SZachary Turner   status.status = Type::eResolveStateFull;
17842f7efbc9SZachary Turner   if (!error)
17852f7efbc9SZachary Turner     return true;
17862f7efbc9SZachary Turner 
17872f7efbc9SZachary Turner   llvm::consumeError(std::move(error));
1788b96181c2SZachary Turner   return false;
1789b96181c2SZachary Turner }
1790b96181c2SZachary Turner 
1791b96181c2SZachary Turner size_t SymbolFileNativePDB::GetTypes(lldb_private::SymbolContextScope *sc_scope,
1792117b1fa1SZachary Turner                                      TypeClass type_mask,
1793b96181c2SZachary Turner                                      lldb_private::TypeList &type_list) {
1794b96181c2SZachary Turner   return 0;
1795b96181c2SZachary Turner }
1796b96181c2SZachary Turner 
1797b96181c2SZachary Turner CompilerDeclContext
1798b96181c2SZachary Turner SymbolFileNativePDB::FindNamespace(const SymbolContext &sc,
1799b96181c2SZachary Turner                                    const ConstString &name,
1800b96181c2SZachary Turner                                    const CompilerDeclContext *parent_decl_ctx) {
1801307f5ae8SZachary Turner   return {};
1802307f5ae8SZachary Turner }
1803307f5ae8SZachary Turner 
1804b96181c2SZachary Turner TypeSystem *
1805307f5ae8SZachary Turner SymbolFileNativePDB::GetTypeSystemForLanguage(lldb::LanguageType language) {
1806307f5ae8SZachary Turner   auto type_system =
1807307f5ae8SZachary Turner       m_obj_file->GetModule()->GetTypeSystemForLanguage(language);
1808307f5ae8SZachary Turner   if (type_system)
1809307f5ae8SZachary Turner     type_system->SetSymbolFile(this);
1810307f5ae8SZachary Turner   return type_system;
1811307f5ae8SZachary Turner }
1812307f5ae8SZachary Turner 
1813b96181c2SZachary Turner ConstString SymbolFileNativePDB::GetPluginName() {
1814307f5ae8SZachary Turner   static ConstString g_name("pdb");
1815307f5ae8SZachary Turner   return g_name;
1816307f5ae8SZachary Turner }
1817307f5ae8SZachary Turner 
1818307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::GetPluginVersion() { return 1; }
1819