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"
40*a42bbe39SZachary Turner #include "llvm/DebugInfo/CodeView/SymbolRecordHelpers.h"
412f7efbc9SZachary Turner #include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
42307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/DbiStream.h"
43307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/GlobalsStream.h"
44307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/InfoStream.h"
45307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/ModuleDebugStream.h"
46307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/PDBFile.h"
47307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/SymbolStream.h"
482f7efbc9SZachary Turner #include "llvm/DebugInfo/PDB/Native/TpiStream.h"
49307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/PDBTypes.h"
50056e4ab4SZachary Turner #include "llvm/Demangle/MicrosoftDemangle.h"
51307f5ae8SZachary Turner #include "llvm/Object/COFF.h"
52307f5ae8SZachary Turner #include "llvm/Support/Allocator.h"
53307f5ae8SZachary Turner #include "llvm/Support/BinaryStreamReader.h"
54056e4ab4SZachary Turner #include "llvm/Support/Error.h"
55307f5ae8SZachary Turner #include "llvm/Support/ErrorOr.h"
56307f5ae8SZachary Turner #include "llvm/Support/MemoryBuffer.h"
57307f5ae8SZachary Turner 
58a93458b0SZachary Turner #include "DWARFLocationExpression.h"
59307f5ae8SZachary Turner #include "PdbSymUid.h"
60307f5ae8SZachary Turner #include "PdbUtil.h"
612f7efbc9SZachary Turner #include "UdtRecordCompleter.h"
62307f5ae8SZachary Turner 
63307f5ae8SZachary Turner using namespace lldb;
64307f5ae8SZachary Turner using namespace lldb_private;
652f7efbc9SZachary Turner using namespace npdb;
66307f5ae8SZachary Turner using namespace llvm::codeview;
67307f5ae8SZachary Turner using namespace llvm::pdb;
68307f5ae8SZachary Turner 
69307f5ae8SZachary Turner static lldb::LanguageType TranslateLanguage(PDB_Lang lang) {
70307f5ae8SZachary Turner   switch (lang) {
71307f5ae8SZachary Turner   case PDB_Lang::Cpp:
72307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeC_plus_plus;
73307f5ae8SZachary Turner   case PDB_Lang::C:
74307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeC;
75307f5ae8SZachary Turner   default:
76307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeUnknown;
77307f5ae8SZachary Turner   }
78307f5ae8SZachary Turner }
79307f5ae8SZachary Turner 
80307f5ae8SZachary Turner static std::unique_ptr<PDBFile> loadPDBFile(std::string PdbPath,
81307f5ae8SZachary Turner                                             llvm::BumpPtrAllocator &Allocator) {
82307f5ae8SZachary Turner   llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> ErrorOrBuffer =
83307f5ae8SZachary Turner       llvm::MemoryBuffer::getFile(PdbPath, /*FileSize=*/-1,
84307f5ae8SZachary Turner                                   /*RequiresNullTerminator=*/false);
85307f5ae8SZachary Turner   if (!ErrorOrBuffer)
86307f5ae8SZachary Turner     return nullptr;
87307f5ae8SZachary Turner   std::unique_ptr<llvm::MemoryBuffer> Buffer = std::move(*ErrorOrBuffer);
88307f5ae8SZachary Turner 
89307f5ae8SZachary Turner   llvm::StringRef Path = Buffer->getBufferIdentifier();
90307f5ae8SZachary Turner   auto Stream = llvm::make_unique<llvm::MemoryBufferByteStream>(
91307f5ae8SZachary Turner       std::move(Buffer), llvm::support::little);
92307f5ae8SZachary Turner 
93307f5ae8SZachary Turner   auto File = llvm::make_unique<PDBFile>(Path, std::move(Stream), Allocator);
948040eea9SZachary Turner   if (auto EC = File->parseFileHeaders()) {
958040eea9SZachary Turner     llvm::consumeError(std::move(EC));
96307f5ae8SZachary Turner     return nullptr;
978040eea9SZachary Turner   }
988040eea9SZachary Turner   if (auto EC = File->parseStreamData()) {
998040eea9SZachary Turner     llvm::consumeError(std::move(EC));
100307f5ae8SZachary Turner     return nullptr;
1018040eea9SZachary Turner   }
102307f5ae8SZachary Turner 
103307f5ae8SZachary Turner   return File;
104307f5ae8SZachary Turner }
105307f5ae8SZachary Turner 
106307f5ae8SZachary Turner static std::unique_ptr<PDBFile>
107307f5ae8SZachary Turner loadMatchingPDBFile(std::string exe_path, llvm::BumpPtrAllocator &allocator) {
108307f5ae8SZachary Turner   // Try to find a matching PDB for an EXE.
109307f5ae8SZachary Turner   using namespace llvm::object;
110307f5ae8SZachary Turner   auto expected_binary = createBinary(exe_path);
111307f5ae8SZachary Turner 
112307f5ae8SZachary Turner   // If the file isn't a PE/COFF executable, fail.
113307f5ae8SZachary Turner   if (!expected_binary) {
114307f5ae8SZachary Turner     llvm::consumeError(expected_binary.takeError());
115307f5ae8SZachary Turner     return nullptr;
116307f5ae8SZachary Turner   }
117307f5ae8SZachary Turner   OwningBinary<Binary> binary = std::move(*expected_binary);
118307f5ae8SZachary Turner 
119307f5ae8SZachary Turner   auto *obj = llvm::dyn_cast<llvm::object::COFFObjectFile>(binary.getBinary());
120307f5ae8SZachary Turner   if (!obj)
121307f5ae8SZachary Turner     return nullptr;
122307f5ae8SZachary Turner   const llvm::codeview::DebugInfo *pdb_info = nullptr;
123307f5ae8SZachary Turner 
124307f5ae8SZachary Turner   // If it doesn't have a debug directory, fail.
125307f5ae8SZachary Turner   llvm::StringRef pdb_file;
126307f5ae8SZachary Turner   auto ec = obj->getDebugPDBInfo(pdb_info, pdb_file);
127307f5ae8SZachary Turner   if (ec)
128307f5ae8SZachary Turner     return nullptr;
129307f5ae8SZachary Turner 
130307f5ae8SZachary Turner   // if the file doesn't exist, is not a pdb, or doesn't have a matching guid,
131307f5ae8SZachary Turner   // fail.
132307f5ae8SZachary Turner   llvm::file_magic magic;
133307f5ae8SZachary Turner   ec = llvm::identify_magic(pdb_file, magic);
134307f5ae8SZachary Turner   if (ec || magic != llvm::file_magic::pdb)
135307f5ae8SZachary Turner     return nullptr;
136307f5ae8SZachary Turner   std::unique_ptr<PDBFile> pdb = loadPDBFile(pdb_file, allocator);
1378040eea9SZachary Turner   if (!pdb)
1388040eea9SZachary Turner     return nullptr;
1398040eea9SZachary Turner 
140307f5ae8SZachary Turner   auto expected_info = pdb->getPDBInfoStream();
141307f5ae8SZachary Turner   if (!expected_info) {
142307f5ae8SZachary Turner     llvm::consumeError(expected_info.takeError());
143307f5ae8SZachary Turner     return nullptr;
144307f5ae8SZachary Turner   }
145307f5ae8SZachary Turner   llvm::codeview::GUID guid;
146307f5ae8SZachary Turner   memcpy(&guid, pdb_info->PDB70.Signature, 16);
147307f5ae8SZachary Turner 
148307f5ae8SZachary Turner   if (expected_info->getGuid() != guid)
149307f5ae8SZachary Turner     return nullptr;
150307f5ae8SZachary Turner   return pdb;
151307f5ae8SZachary Turner }
152307f5ae8SZachary Turner 
153307f5ae8SZachary Turner static bool IsFunctionPrologue(const CompilandIndexItem &cci,
154307f5ae8SZachary Turner                                lldb::addr_t addr) {
155307f5ae8SZachary Turner   // FIXME: Implement this.
156307f5ae8SZachary Turner   return false;
157307f5ae8SZachary Turner }
158307f5ae8SZachary Turner 
159307f5ae8SZachary Turner static bool IsFunctionEpilogue(const CompilandIndexItem &cci,
160307f5ae8SZachary Turner                                lldb::addr_t addr) {
161307f5ae8SZachary Turner   // FIXME: Implement this.
162307f5ae8SZachary Turner   return false;
163307f5ae8SZachary Turner }
164307f5ae8SZachary Turner 
1652f7efbc9SZachary Turner static clang::MSInheritanceAttr::Spelling
1662f7efbc9SZachary Turner GetMSInheritance(LazyRandomTypeCollection &tpi, const ClassRecord &record) {
1672f7efbc9SZachary Turner   if (record.DerivationList == TypeIndex::None())
1682f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance;
1692f7efbc9SZachary Turner 
1702f7efbc9SZachary Turner   CVType bases = tpi.getType(record.DerivationList);
1712f7efbc9SZachary Turner   ArgListRecord base_list;
1722f7efbc9SZachary Turner   cantFail(TypeDeserializer::deserializeAs<ArgListRecord>(bases, base_list));
1732f7efbc9SZachary Turner   if (base_list.ArgIndices.empty())
1742f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance;
1752f7efbc9SZachary Turner 
1762f7efbc9SZachary Turner   int base_count = 0;
1772f7efbc9SZachary Turner   for (TypeIndex ti : base_list.ArgIndices) {
1782f7efbc9SZachary Turner     CVType base = tpi.getType(ti);
1792f7efbc9SZachary Turner     if (base.kind() == LF_VBCLASS || base.kind() == LF_IVBCLASS)
1802f7efbc9SZachary Turner       return clang::MSInheritanceAttr::Spelling::Keyword_virtual_inheritance;
1812f7efbc9SZachary Turner     ++base_count;
1822f7efbc9SZachary Turner   }
1832f7efbc9SZachary Turner 
1842f7efbc9SZachary Turner   if (base_count > 1)
1852f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Keyword_multiple_inheritance;
1862f7efbc9SZachary Turner   return clang::MSInheritanceAttr::Keyword_single_inheritance;
1872f7efbc9SZachary Turner }
1882f7efbc9SZachary Turner 
1892f7efbc9SZachary Turner static llvm::StringRef GetSimpleTypeName(SimpleTypeKind kind) {
1902f7efbc9SZachary Turner   switch (kind) {
1912f7efbc9SZachary Turner   case SimpleTypeKind::Boolean128:
1922f7efbc9SZachary Turner   case SimpleTypeKind::Boolean16:
1932f7efbc9SZachary Turner   case SimpleTypeKind::Boolean32:
1942f7efbc9SZachary Turner   case SimpleTypeKind::Boolean64:
1952f7efbc9SZachary Turner   case SimpleTypeKind::Boolean8:
1962f7efbc9SZachary Turner     return "bool";
1972f7efbc9SZachary Turner   case SimpleTypeKind::Byte:
1982f7efbc9SZachary Turner   case SimpleTypeKind::UnsignedCharacter:
1992f7efbc9SZachary Turner     return "unsigned char";
2002f7efbc9SZachary Turner   case SimpleTypeKind::NarrowCharacter:
2012f7efbc9SZachary Turner     return "char";
2022f7efbc9SZachary Turner   case SimpleTypeKind::SignedCharacter:
2032f7efbc9SZachary Turner   case SimpleTypeKind::SByte:
20471ebb721SZachary Turner     return "signed char";
2052f7efbc9SZachary Turner   case SimpleTypeKind::Character16:
2062f7efbc9SZachary Turner     return "char16_t";
2072f7efbc9SZachary Turner   case SimpleTypeKind::Character32:
2082f7efbc9SZachary Turner     return "char32_t";
2092f7efbc9SZachary Turner   case SimpleTypeKind::Complex80:
2102f7efbc9SZachary Turner   case SimpleTypeKind::Complex64:
2112f7efbc9SZachary Turner   case SimpleTypeKind::Complex32:
2122f7efbc9SZachary Turner     return "complex";
2132f7efbc9SZachary Turner   case SimpleTypeKind::Float128:
2142f7efbc9SZachary Turner   case SimpleTypeKind::Float80:
2152f7efbc9SZachary Turner     return "long double";
2162f7efbc9SZachary Turner   case SimpleTypeKind::Float64:
2172f7efbc9SZachary Turner     return "double";
2182f7efbc9SZachary Turner   case SimpleTypeKind::Float32:
2192f7efbc9SZachary Turner     return "float";
2202f7efbc9SZachary Turner   case SimpleTypeKind::Float16:
2212f7efbc9SZachary Turner     return "single";
2222f7efbc9SZachary Turner   case SimpleTypeKind::Int128:
2232f7efbc9SZachary Turner     return "__int128";
2242f7efbc9SZachary Turner   case SimpleTypeKind::Int64:
2252f7efbc9SZachary Turner   case SimpleTypeKind::Int64Quad:
22671ebb721SZachary Turner     return "int64_t";
2272f7efbc9SZachary Turner   case SimpleTypeKind::Int32:
2282f7efbc9SZachary Turner     return "int";
2292f7efbc9SZachary Turner   case SimpleTypeKind::Int16:
2302f7efbc9SZachary Turner     return "short";
2312f7efbc9SZachary Turner   case SimpleTypeKind::UInt128:
2322f7efbc9SZachary Turner     return "unsigned __int128";
2332f7efbc9SZachary Turner   case SimpleTypeKind::UInt64:
2342f7efbc9SZachary Turner   case SimpleTypeKind::UInt64Quad:
23571ebb721SZachary Turner     return "uint64_t";
2362f7efbc9SZachary Turner   case SimpleTypeKind::HResult:
2372f7efbc9SZachary Turner     return "HRESULT";
2382f7efbc9SZachary Turner   case SimpleTypeKind::UInt32:
2392f7efbc9SZachary Turner     return "unsigned";
2402f7efbc9SZachary Turner   case SimpleTypeKind::UInt16:
2412f7efbc9SZachary Turner   case SimpleTypeKind::UInt16Short:
2422f7efbc9SZachary Turner     return "unsigned short";
2432f7efbc9SZachary Turner   case SimpleTypeKind::Int32Long:
2442f7efbc9SZachary Turner     return "long";
2452f7efbc9SZachary Turner   case SimpleTypeKind::UInt32Long:
2462f7efbc9SZachary Turner     return "unsigned long";
2472f7efbc9SZachary Turner   case SimpleTypeKind::Void:
2482f7efbc9SZachary Turner     return "void";
2492f7efbc9SZachary Turner   case SimpleTypeKind::WideCharacter:
2502f7efbc9SZachary Turner     return "wchar_t";
2512f7efbc9SZachary Turner   default:
2522f7efbc9SZachary Turner     return "";
2532f7efbc9SZachary Turner   }
2542f7efbc9SZachary Turner }
2552f7efbc9SZachary Turner 
2562f7efbc9SZachary Turner static bool IsClassRecord(TypeLeafKind kind) {
2572f7efbc9SZachary Turner   switch (kind) {
2582f7efbc9SZachary Turner   case LF_STRUCTURE:
2592f7efbc9SZachary Turner   case LF_CLASS:
2602f7efbc9SZachary Turner   case LF_INTERFACE:
2612f7efbc9SZachary Turner     return true;
2622f7efbc9SZachary Turner   default:
2632f7efbc9SZachary Turner     return false;
2642f7efbc9SZachary Turner   }
2652f7efbc9SZachary Turner }
2662f7efbc9SZachary Turner 
267544a66d8SZachary Turner static bool IsCVarArgsFunction(llvm::ArrayRef<TypeIndex> args) {
268544a66d8SZachary Turner   if (args.empty())
269544a66d8SZachary Turner     return false;
270544a66d8SZachary Turner   return args.back() == TypeIndex::None();
271544a66d8SZachary Turner }
272544a66d8SZachary Turner 
2732f7efbc9SZachary Turner static clang::TagTypeKind TranslateUdtKind(const TagRecord &cr) {
2742f7efbc9SZachary Turner   switch (cr.Kind) {
2752f7efbc9SZachary Turner   case TypeRecordKind::Class:
2762f7efbc9SZachary Turner     return clang::TTK_Class;
2772f7efbc9SZachary Turner   case TypeRecordKind::Struct:
2782f7efbc9SZachary Turner     return clang::TTK_Struct;
2792f7efbc9SZachary Turner   case TypeRecordKind::Union:
2802f7efbc9SZachary Turner     return clang::TTK_Union;
2812f7efbc9SZachary Turner   case TypeRecordKind::Interface:
2822f7efbc9SZachary Turner     return clang::TTK_Interface;
2832f7efbc9SZachary Turner   case TypeRecordKind::Enum:
2842f7efbc9SZachary Turner     return clang::TTK_Enum;
2852f7efbc9SZachary Turner   default:
2862f7efbc9SZachary Turner     lldbassert(false && "Invalid tag record kind!");
2872f7efbc9SZachary Turner     return clang::TTK_Struct;
2882f7efbc9SZachary Turner   }
2892f7efbc9SZachary Turner }
2902f7efbc9SZachary Turner 
291544a66d8SZachary Turner static llvm::Optional<clang::CallingConv>
292544a66d8SZachary Turner TranslateCallingConvention(llvm::codeview::CallingConvention conv) {
293544a66d8SZachary Turner   using CC = llvm::codeview::CallingConvention;
294544a66d8SZachary Turner   switch (conv) {
295544a66d8SZachary Turner 
296544a66d8SZachary Turner   case CC::NearC:
297544a66d8SZachary Turner   case CC::FarC:
298544a66d8SZachary Turner     return clang::CallingConv::CC_C;
299544a66d8SZachary Turner   case CC::NearPascal:
300544a66d8SZachary Turner   case CC::FarPascal:
301544a66d8SZachary Turner     return clang::CallingConv::CC_X86Pascal;
302544a66d8SZachary Turner   case CC::NearFast:
303544a66d8SZachary Turner   case CC::FarFast:
304544a66d8SZachary Turner     return clang::CallingConv::CC_X86FastCall;
305544a66d8SZachary Turner   case CC::NearStdCall:
306544a66d8SZachary Turner   case CC::FarStdCall:
307544a66d8SZachary Turner     return clang::CallingConv::CC_X86StdCall;
308544a66d8SZachary Turner   case CC::ThisCall:
309544a66d8SZachary Turner     return clang::CallingConv::CC_X86ThisCall;
310544a66d8SZachary Turner   case CC::NearVector:
311544a66d8SZachary Turner     return clang::CallingConv::CC_X86VectorCall;
312544a66d8SZachary Turner   default:
313544a66d8SZachary Turner     return llvm::None;
314544a66d8SZachary Turner   }
315544a66d8SZachary Turner }
316544a66d8SZachary Turner 
317307f5ae8SZachary Turner void SymbolFileNativePDB::Initialize() {
318307f5ae8SZachary Turner   PluginManager::RegisterPlugin(GetPluginNameStatic(),
319307f5ae8SZachary Turner                                 GetPluginDescriptionStatic(), CreateInstance,
320307f5ae8SZachary Turner                                 DebuggerInitialize);
321307f5ae8SZachary Turner }
322307f5ae8SZachary Turner 
323307f5ae8SZachary Turner void SymbolFileNativePDB::Terminate() {
324307f5ae8SZachary Turner   PluginManager::UnregisterPlugin(CreateInstance);
325307f5ae8SZachary Turner }
326307f5ae8SZachary Turner 
327b96181c2SZachary Turner void SymbolFileNativePDB::DebuggerInitialize(Debugger &debugger) {}
328307f5ae8SZachary Turner 
329b96181c2SZachary Turner ConstString SymbolFileNativePDB::GetPluginNameStatic() {
330307f5ae8SZachary Turner   static ConstString g_name("native-pdb");
331307f5ae8SZachary Turner   return g_name;
332307f5ae8SZachary Turner }
333307f5ae8SZachary Turner 
334307f5ae8SZachary Turner const char *SymbolFileNativePDB::GetPluginDescriptionStatic() {
335307f5ae8SZachary Turner   return "Microsoft PDB debug symbol cross-platform file reader.";
336307f5ae8SZachary Turner }
337307f5ae8SZachary Turner 
338b96181c2SZachary Turner SymbolFile *SymbolFileNativePDB::CreateInstance(ObjectFile *obj_file) {
339307f5ae8SZachary Turner   return new SymbolFileNativePDB(obj_file);
340307f5ae8SZachary Turner }
341307f5ae8SZachary Turner 
342b96181c2SZachary Turner SymbolFileNativePDB::SymbolFileNativePDB(ObjectFile *object_file)
343307f5ae8SZachary Turner     : SymbolFile(object_file) {}
344307f5ae8SZachary Turner 
345307f5ae8SZachary Turner SymbolFileNativePDB::~SymbolFileNativePDB() {}
346307f5ae8SZachary Turner 
347307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::CalculateAbilities() {
348307f5ae8SZachary Turner   uint32_t abilities = 0;
349307f5ae8SZachary Turner   if (!m_obj_file)
350307f5ae8SZachary Turner     return 0;
351307f5ae8SZachary Turner 
352307f5ae8SZachary Turner   if (!m_index) {
353307f5ae8SZachary Turner     // Lazily load and match the PDB file, but only do this once.
354307f5ae8SZachary Turner     std::unique_ptr<PDBFile> file_up =
355307f5ae8SZachary Turner         loadMatchingPDBFile(m_obj_file->GetFileSpec().GetPath(), m_allocator);
356307f5ae8SZachary Turner 
357307f5ae8SZachary Turner     if (!file_up) {
358307f5ae8SZachary Turner       auto module_sp = m_obj_file->GetModule();
359307f5ae8SZachary Turner       if (!module_sp)
360307f5ae8SZachary Turner         return 0;
361307f5ae8SZachary Turner       // See if any symbol file is specified through `--symfile` option.
362307f5ae8SZachary Turner       FileSpec symfile = module_sp->GetSymbolFileFileSpec();
363307f5ae8SZachary Turner       if (!symfile)
364307f5ae8SZachary Turner         return 0;
365307f5ae8SZachary Turner       file_up = loadPDBFile(symfile.GetPath(), m_allocator);
366307f5ae8SZachary Turner     }
367307f5ae8SZachary Turner 
368307f5ae8SZachary Turner     if (!file_up)
369307f5ae8SZachary Turner       return 0;
370307f5ae8SZachary Turner 
371307f5ae8SZachary Turner     auto expected_index = PdbIndex::create(std::move(file_up));
372307f5ae8SZachary Turner     if (!expected_index) {
373307f5ae8SZachary Turner       llvm::consumeError(expected_index.takeError());
374307f5ae8SZachary Turner       return 0;
375307f5ae8SZachary Turner     }
376307f5ae8SZachary Turner     m_index = std::move(*expected_index);
377307f5ae8SZachary Turner   }
378307f5ae8SZachary Turner   if (!m_index)
379307f5ae8SZachary Turner     return 0;
380307f5ae8SZachary Turner 
381307f5ae8SZachary Turner   // We don't especially have to be precise here.  We only distinguish between
382307f5ae8SZachary Turner   // stripped and not stripped.
383307f5ae8SZachary Turner   abilities = kAllAbilities;
384307f5ae8SZachary Turner 
385307f5ae8SZachary Turner   if (m_index->dbi().isStripped())
386307f5ae8SZachary Turner     abilities &= ~(Blocks | LocalVariables);
387307f5ae8SZachary Turner   return abilities;
388307f5ae8SZachary Turner }
389307f5ae8SZachary Turner 
390307f5ae8SZachary Turner void SymbolFileNativePDB::InitializeObject() {
391307f5ae8SZachary Turner   m_obj_load_address = m_obj_file->GetFileOffset();
392307f5ae8SZachary Turner   m_index->SetLoadAddress(m_obj_load_address);
393307f5ae8SZachary Turner   m_index->ParseSectionContribs();
3942f7efbc9SZachary Turner 
3952f7efbc9SZachary Turner   TypeSystem *ts = GetTypeSystemForLanguage(eLanguageTypeC_plus_plus);
3962f7efbc9SZachary Turner   m_clang = llvm::dyn_cast_or_null<ClangASTContext>(ts);
3972f7efbc9SZachary Turner   m_importer = llvm::make_unique<ClangASTImporter>();
398056e4ab4SZachary Turner 
399056e4ab4SZachary Turner   PreprocessTpiStream();
4002f7efbc9SZachary Turner   lldbassert(m_clang);
401307f5ae8SZachary Turner }
402307f5ae8SZachary Turner 
40303a24052SZachary Turner static llvm::Optional<CVTagRecord>
40403a24052SZachary Turner GetNestedTagRecord(const NestedTypeRecord &Record, const CVTagRecord &parent,
40503a24052SZachary Turner                    TpiStream &tpi) {
40603a24052SZachary Turner   // An LF_NESTTYPE is essentially a nested typedef / using declaration, but it
40703a24052SZachary Turner   // is also used to indicate the primary definition of a nested class.  That is
40803a24052SZachary Turner   // to say, if you have:
40903a24052SZachary Turner   // struct A {
41003a24052SZachary Turner   //   struct B {};
41103a24052SZachary Turner   //   using C = B;
41203a24052SZachary Turner   // };
41303a24052SZachary Turner   // Then in the debug info, this will appear as:
41403a24052SZachary Turner   // LF_STRUCTURE `A::B` [type index = N]
41503a24052SZachary Turner   // LF_STRUCTURE `A`
41603a24052SZachary Turner   //   LF_NESTTYPE [name = `B`, index = N]
41703a24052SZachary Turner   //   LF_NESTTYPE [name = `C`, index = N]
41803a24052SZachary Turner   // In order to accurately reconstruct the decl context hierarchy, we need to
41903a24052SZachary Turner   // know which ones are actual definitions and which ones are just aliases.
42003a24052SZachary Turner 
42103a24052SZachary Turner   // If it's a simple type, then this is something like `using foo = int`.
42203a24052SZachary Turner   if (Record.Type.isSimple())
42303a24052SZachary Turner     return llvm::None;
42403a24052SZachary Turner 
4252af34166SZachary Turner   CVType cvt = tpi.getType(Record.Type);
4262af34166SZachary Turner 
4272af34166SZachary Turner   if (!IsTagRecord(cvt))
4282af34166SZachary Turner     return llvm::None;
4292af34166SZachary Turner 
43003a24052SZachary Turner   // If it's an inner definition, then treat whatever name we have here as a
43103a24052SZachary Turner   // single component of a mangled name.  So we can inject it into the parent's
43203a24052SZachary Turner   // mangled name to see if it matches.
4332af34166SZachary Turner   CVTagRecord child = CVTagRecord::create(cvt);
43403a24052SZachary Turner   std::string qname = parent.asTag().getUniqueName();
43503a24052SZachary Turner   if (qname.size() < 4 || child.asTag().getUniqueName().size() < 4)
43603a24052SZachary Turner     return llvm::None;
43703a24052SZachary Turner 
43803a24052SZachary Turner   // qname[3] is the tag type identifier (struct, class, union, etc).  Since the
43903a24052SZachary Turner   // inner tag type is not necessarily the same as the outer tag type, re-write
44003a24052SZachary Turner   // it to match the inner tag type.
44103a24052SZachary Turner   qname[3] = child.asTag().getUniqueName()[3];
44203a24052SZachary Turner   std::string piece = Record.Name;
44303a24052SZachary Turner   piece.push_back('@');
44403a24052SZachary Turner   qname.insert(4, std::move(piece));
44503a24052SZachary Turner   if (qname != child.asTag().UniqueName)
44603a24052SZachary Turner     return llvm::None;
44703a24052SZachary Turner 
44803a24052SZachary Turner   return std::move(child);
44903a24052SZachary Turner }
45003a24052SZachary Turner 
451056e4ab4SZachary Turner void SymbolFileNativePDB::PreprocessTpiStream() {
452056e4ab4SZachary Turner   LazyRandomTypeCollection &types = m_index->tpi().typeCollection();
453056e4ab4SZachary Turner 
454056e4ab4SZachary Turner   for (auto ti = types.getFirst(); ti; ti = types.getNext(*ti)) {
455056e4ab4SZachary Turner     CVType type = types.getType(*ti);
456056e4ab4SZachary Turner     if (!IsTagRecord(type))
457056e4ab4SZachary Turner       continue;
458056e4ab4SZachary Turner 
459056e4ab4SZachary Turner     CVTagRecord tag = CVTagRecord::create(type);
460056e4ab4SZachary Turner     // We're looking for LF_NESTTYPE records in the field list, so ignore
461056e4ab4SZachary Turner     // forward references (no field list), and anything without a nested class
462056e4ab4SZachary Turner     // (since there won't be any LF_NESTTYPE records).
463056e4ab4SZachary Turner     if (tag.asTag().isForwardRef() || !tag.asTag().containsNestedClass())
464056e4ab4SZachary Turner       continue;
465056e4ab4SZachary Turner 
466056e4ab4SZachary Turner     struct ProcessTpiStream : public TypeVisitorCallbacks {
467056e4ab4SZachary Turner       ProcessTpiStream(PdbIndex &index, TypeIndex parent,
46803a24052SZachary Turner                        const CVTagRecord &parent_cvt,
469056e4ab4SZachary Turner                        llvm::DenseMap<TypeIndex, TypeIndex> &parents)
47003a24052SZachary Turner           : index(index), parents(parents), parent(parent),
47103a24052SZachary Turner             parent_cvt(parent_cvt) {}
472056e4ab4SZachary Turner 
473056e4ab4SZachary Turner       PdbIndex &index;
474056e4ab4SZachary Turner       llvm::DenseMap<TypeIndex, TypeIndex> &parents;
475056e4ab4SZachary Turner       TypeIndex parent;
47603a24052SZachary Turner       const CVTagRecord &parent_cvt;
477056e4ab4SZachary Turner 
478056e4ab4SZachary Turner       llvm::Error visitKnownMember(CVMemberRecord &CVR,
479056e4ab4SZachary Turner                                    NestedTypeRecord &Record) override {
48003a24052SZachary Turner         llvm::Optional<CVTagRecord> tag =
48103a24052SZachary Turner             GetNestedTagRecord(Record, parent_cvt, index.tpi());
48203a24052SZachary Turner         if (!tag)
483056e4ab4SZachary Turner           return llvm::ErrorSuccess();
48403a24052SZachary Turner 
48503a24052SZachary Turner         parents[Record.Type] = parent;
48603a24052SZachary Turner         if (!tag->asTag().isForwardRef())
48703a24052SZachary Turner           return llvm::ErrorSuccess();
48803a24052SZachary Turner 
489056e4ab4SZachary Turner         llvm::Expected<TypeIndex> full_decl =
490056e4ab4SZachary Turner             index.tpi().findFullDeclForForwardRef(Record.Type);
491056e4ab4SZachary Turner         if (!full_decl) {
492056e4ab4SZachary Turner           llvm::consumeError(full_decl.takeError());
493056e4ab4SZachary Turner           return llvm::ErrorSuccess();
494056e4ab4SZachary Turner         }
495056e4ab4SZachary Turner         parents[*full_decl] = parent;
496056e4ab4SZachary Turner         return llvm::ErrorSuccess();
497056e4ab4SZachary Turner       }
498056e4ab4SZachary Turner     };
499056e4ab4SZachary Turner 
500056e4ab4SZachary Turner     CVType field_list = m_index->tpi().getType(tag.asTag().FieldList);
50103a24052SZachary Turner     ProcessTpiStream process(*m_index, *ti, tag, m_parent_types);
502056e4ab4SZachary Turner     llvm::Error error = visitMemberRecordStream(field_list.data(), process);
503056e4ab4SZachary Turner     if (error)
504056e4ab4SZachary Turner       llvm::consumeError(std::move(error));
505056e4ab4SZachary Turner   }
506056e4ab4SZachary Turner }
507056e4ab4SZachary Turner 
508307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::GetNumCompileUnits() {
509307f5ae8SZachary Turner   const DbiModuleList &modules = m_index->dbi().modules();
510307f5ae8SZachary Turner   uint32_t count = modules.getModuleCount();
511307f5ae8SZachary Turner   if (count == 0)
512307f5ae8SZachary Turner     return count;
513307f5ae8SZachary Turner 
514307f5ae8SZachary Turner   // The linker can inject an additional "dummy" compilation unit into the
515307f5ae8SZachary Turner   // PDB. Ignore this special compile unit for our purposes, if it is there.
516307f5ae8SZachary Turner   // It is always the last one.
517307f5ae8SZachary Turner   DbiModuleDescriptor last = modules.getModuleDescriptor(count - 1);
518307f5ae8SZachary Turner   if (last.getModuleName() == "* Linker *")
519307f5ae8SZachary Turner     --count;
520307f5ae8SZachary Turner   return count;
521307f5ae8SZachary Turner }
522307f5ae8SZachary Turner 
5236284aee9SZachary Turner lldb::FunctionSP SymbolFileNativePDB::CreateFunction(PdbCompilandSymId func_id,
524307f5ae8SZachary Turner                                                      const SymbolContext &sc) {
5256284aee9SZachary Turner   const CompilandIndexItem *cci =
5266284aee9SZachary Turner       m_index->compilands().GetCompiland(func_id.modi);
527307f5ae8SZachary Turner   lldbassert(cci);
5286284aee9SZachary Turner   CVSymbol sym_record = cci->m_debug_stream.readSymbolAtOffset(func_id.offset);
529307f5ae8SZachary Turner 
530307f5ae8SZachary Turner   lldbassert(sym_record.kind() == S_LPROC32 || sym_record.kind() == S_GPROC32);
531307f5ae8SZachary Turner   SegmentOffsetLength sol = GetSegmentOffsetAndLength(sym_record);
532307f5ae8SZachary Turner 
533307f5ae8SZachary Turner   auto file_vm_addr = m_index->MakeVirtualAddress(sol.so);
534307f5ae8SZachary Turner   if (file_vm_addr == LLDB_INVALID_ADDRESS || file_vm_addr == 0)
535307f5ae8SZachary Turner     return nullptr;
536307f5ae8SZachary Turner 
537307f5ae8SZachary Turner   AddressRange func_range(file_vm_addr, sol.length,
538307f5ae8SZachary Turner                           sc.module_sp->GetSectionList());
539307f5ae8SZachary Turner   if (!func_range.GetBaseAddress().IsValid())
540307f5ae8SZachary Turner     return nullptr;
541307f5ae8SZachary Turner 
542*a42bbe39SZachary Turner   ProcSym proc(static_cast<SymbolRecordKind>(sym_record.kind()));
543*a42bbe39SZachary Turner   cantFail(SymbolDeserializer::deserializeAs<ProcSym>(sym_record, proc));
544*a42bbe39SZachary Turner   TypeSP func_type = GetOrCreateType(proc.FunctionType);
545307f5ae8SZachary Turner 
546*a42bbe39SZachary Turner   PdbTypeSymId sig_id(proc.FunctionType, false);
547*a42bbe39SZachary Turner   Mangled mangled(proc.Name);
548307f5ae8SZachary Turner   FunctionSP func_sp = std::make_shared<Function>(
5496284aee9SZachary Turner       sc.comp_unit, toOpaqueUid(func_id), toOpaqueUid(sig_id), mangled,
550*a42bbe39SZachary Turner       func_type.get(), func_range);
551307f5ae8SZachary Turner 
552307f5ae8SZachary Turner   sc.comp_unit->AddFunction(func_sp);
553*a42bbe39SZachary Turner 
554*a42bbe39SZachary Turner   clang::StorageClass storage = clang::SC_None;
555*a42bbe39SZachary Turner   if (sym_record.kind() == S_LPROC32)
556*a42bbe39SZachary Turner     storage = clang::SC_Static;
557*a42bbe39SZachary Turner 
558*a42bbe39SZachary Turner   // There are two ways we could retrieve the parameter list.  The first is by
559*a42bbe39SZachary Turner   // iterating the arguments on the function signature type, however that would
560*a42bbe39SZachary Turner   // only tell us the types of the arguments and not the names.  The second is
561*a42bbe39SZachary Turner   // to iterate the CVSymbol records that follow the S_GPROC32 / S_LPROC32 until
562*a42bbe39SZachary Turner   // we have the correct number of arguments as stated by the function
563*a42bbe39SZachary Turner   // signature. The latter has more potential to go wrong in the face of
564*a42bbe39SZachary Turner   // improper debug info simply because we're assuming more about the layout of
565*a42bbe39SZachary Turner   // the records, but it is the only way to get argument names.
566*a42bbe39SZachary Turner   CVType sig_cvt;
567*a42bbe39SZachary Turner   CVType arg_list_cvt;
568*a42bbe39SZachary Turner   ProcedureRecord sig_record;
569*a42bbe39SZachary Turner   ArgListRecord arg_list_record;
570*a42bbe39SZachary Turner 
571*a42bbe39SZachary Turner   sig_cvt = m_index->tpi().getType(proc.FunctionType);
572*a42bbe39SZachary Turner   if (sig_cvt.kind() != LF_PROCEDURE)
573*a42bbe39SZachary Turner     return func_sp;
574*a42bbe39SZachary Turner   cantFail(
575*a42bbe39SZachary Turner       TypeDeserializer::deserializeAs<ProcedureRecord>(sig_cvt, sig_record));
576*a42bbe39SZachary Turner 
577*a42bbe39SZachary Turner   CompilerDeclContext context =
578*a42bbe39SZachary Turner       GetDeclContextContainingUID(toOpaqueUid(func_id));
579*a42bbe39SZachary Turner 
580*a42bbe39SZachary Turner   clang::DeclContext *decl_context =
581*a42bbe39SZachary Turner       static_cast<clang::DeclContext *>(context.GetOpaqueDeclContext());
582*a42bbe39SZachary Turner   clang::FunctionDecl *function_decl = m_clang->CreateFunctionDeclaration(
583*a42bbe39SZachary Turner       decl_context, proc.Name.str().c_str(),
584*a42bbe39SZachary Turner       func_type->GetForwardCompilerType(), storage, false);
585*a42bbe39SZachary Turner 
586*a42bbe39SZachary Turner   lldbassert(m_uid_to_decl.count(toOpaqueUid(func_id)) == 0);
587*a42bbe39SZachary Turner   m_uid_to_decl[toOpaqueUid(func_id)] = function_decl;
588*a42bbe39SZachary Turner   CVSymbolArray scope = limitSymbolArrayToScope(
589*a42bbe39SZachary Turner       cci->m_debug_stream.getSymbolArray(), func_id.offset);
590*a42bbe39SZachary Turner 
591*a42bbe39SZachary Turner   uint32_t params_remaining = sig_record.getParameterCount();
592*a42bbe39SZachary Turner   auto begin = scope.begin();
593*a42bbe39SZachary Turner   auto end = scope.end();
594*a42bbe39SZachary Turner   std::vector<clang::ParmVarDecl *> params;
595*a42bbe39SZachary Turner   while (begin != end && params_remaining > 0) {
596*a42bbe39SZachary Turner     uint32_t record_offset = begin.offset();
597*a42bbe39SZachary Turner     CVSymbol sym = *begin++;
598*a42bbe39SZachary Turner 
599*a42bbe39SZachary Turner     TypeIndex param_type;
600*a42bbe39SZachary Turner     llvm::StringRef param_name;
601*a42bbe39SZachary Turner     switch (sym.kind()) {
602*a42bbe39SZachary Turner     case S_REGREL32: {
603*a42bbe39SZachary Turner       RegRelativeSym reg(SymbolRecordKind::RegRelativeSym);
604*a42bbe39SZachary Turner       cantFail(SymbolDeserializer::deserializeAs<RegRelativeSym>(sym, reg));
605*a42bbe39SZachary Turner       param_type = reg.Type;
606*a42bbe39SZachary Turner       param_name = reg.Name;
607*a42bbe39SZachary Turner       break;
608*a42bbe39SZachary Turner     }
609*a42bbe39SZachary Turner     case S_REGISTER: {
610*a42bbe39SZachary Turner       RegisterSym reg(SymbolRecordKind::RegisterSym);
611*a42bbe39SZachary Turner       cantFail(SymbolDeserializer::deserializeAs<RegisterSym>(sym, reg));
612*a42bbe39SZachary Turner       param_type = reg.Index;
613*a42bbe39SZachary Turner       param_name = reg.Name;
614*a42bbe39SZachary Turner       break;
615*a42bbe39SZachary Turner     }
616*a42bbe39SZachary Turner     case S_LOCAL: {
617*a42bbe39SZachary Turner       LocalSym local(SymbolRecordKind::LocalSym);
618*a42bbe39SZachary Turner       cantFail(SymbolDeserializer::deserializeAs<LocalSym>(sym, local));
619*a42bbe39SZachary Turner       if ((local.Flags & LocalSymFlags::IsParameter) == LocalSymFlags::None)
620*a42bbe39SZachary Turner         continue;
621*a42bbe39SZachary Turner       param_type = local.Type;
622*a42bbe39SZachary Turner       param_name = local.Name;
623*a42bbe39SZachary Turner       break;
624*a42bbe39SZachary Turner     }
625*a42bbe39SZachary Turner     case S_BLOCK32:
626*a42bbe39SZachary Turner       // All parameters should come before the first block.  If that isn't the
627*a42bbe39SZachary Turner       // case, then perhaps this is bad debug info that doesn't contain
628*a42bbe39SZachary Turner       // information about all parameters.
629*a42bbe39SZachary Turner       params_remaining = 0;
630*a42bbe39SZachary Turner       continue;
631*a42bbe39SZachary Turner     default:
632*a42bbe39SZachary Turner       continue;
633*a42bbe39SZachary Turner     }
634*a42bbe39SZachary Turner 
635*a42bbe39SZachary Turner     PdbCompilandSymId param_uid(func_id.modi, record_offset);
636*a42bbe39SZachary Turner     TypeSP type_sp = GetOrCreateType(param_type);
637*a42bbe39SZachary Turner     clang::ParmVarDecl *param = m_clang->CreateParameterDeclaration(
638*a42bbe39SZachary Turner         param_name.str().c_str(), type_sp->GetForwardCompilerType(),
639*a42bbe39SZachary Turner         clang::SC_None);
640*a42bbe39SZachary Turner     lldbassert(m_uid_to_decl.count(toOpaqueUid(param_uid)) == 0);
641*a42bbe39SZachary Turner 
642*a42bbe39SZachary Turner     m_uid_to_decl[toOpaqueUid(param_uid)] = param;
643*a42bbe39SZachary Turner     params.push_back(param);
644*a42bbe39SZachary Turner     --params_remaining;
645*a42bbe39SZachary Turner   }
646*a42bbe39SZachary Turner 
647*a42bbe39SZachary Turner   if (!params.empty())
648*a42bbe39SZachary Turner     m_clang->SetFunctionParameters(function_decl, params.data(), params.size());
649*a42bbe39SZachary Turner 
650307f5ae8SZachary Turner   return func_sp;
651307f5ae8SZachary Turner }
652307f5ae8SZachary Turner 
653307f5ae8SZachary Turner CompUnitSP
654307f5ae8SZachary Turner SymbolFileNativePDB::CreateCompileUnit(const CompilandIndexItem &cci) {
655307f5ae8SZachary Turner   lldb::LanguageType lang =
656307f5ae8SZachary Turner       cci.m_compile_opts ? TranslateLanguage(cci.m_compile_opts->getLanguage())
657307f5ae8SZachary Turner                          : lldb::eLanguageTypeUnknown;
658307f5ae8SZachary Turner 
659307f5ae8SZachary Turner   LazyBool optimized = eLazyBoolNo;
660307f5ae8SZachary Turner   if (cci.m_compile_opts && cci.m_compile_opts->hasOptimizations())
661307f5ae8SZachary Turner     optimized = eLazyBoolYes;
662307f5ae8SZachary Turner 
663307f5ae8SZachary Turner   llvm::StringRef source_file_name =
664307f5ae8SZachary Turner       m_index->compilands().GetMainSourceFile(cci);
6658f3be7a3SJonas Devlieghere   FileSpec fs(source_file_name);
666307f5ae8SZachary Turner 
667307f5ae8SZachary Turner   CompUnitSP cu_sp =
668307f5ae8SZachary Turner       std::make_shared<CompileUnit>(m_obj_file->GetModule(), nullptr, fs,
6696284aee9SZachary Turner                                     toOpaqueUid(cci.m_id), lang, optimized);
670307f5ae8SZachary Turner 
6716284aee9SZachary Turner   m_obj_file->GetModule()->GetSymbolVendor()->SetCompileUnitAtIndex(
6726284aee9SZachary Turner       cci.m_id.modi, cu_sp);
673307f5ae8SZachary Turner   return cu_sp;
674307f5ae8SZachary Turner }
675307f5ae8SZachary Turner 
6766284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateModifierType(PdbTypeSymId type_id,
6772f7efbc9SZachary Turner                                                      const ModifierRecord &mr) {
6782f7efbc9SZachary Turner   TpiStream &stream = m_index->tpi();
6792f7efbc9SZachary Turner 
6802f7efbc9SZachary Turner   TypeSP t = GetOrCreateType(mr.ModifiedType);
6812f7efbc9SZachary Turner   CompilerType ct = t->GetForwardCompilerType();
6822f7efbc9SZachary Turner   if ((mr.Modifiers & ModifierOptions::Const) != ModifierOptions::None)
6832f7efbc9SZachary Turner     ct = ct.AddConstModifier();
6842f7efbc9SZachary Turner   if ((mr.Modifiers & ModifierOptions::Volatile) != ModifierOptions::None)
6852f7efbc9SZachary Turner     ct = ct.AddVolatileModifier();
6862f7efbc9SZachary Turner   std::string name;
6872f7efbc9SZachary Turner   if (mr.ModifiedType.isSimple())
6882f7efbc9SZachary Turner     name = GetSimpleTypeName(mr.ModifiedType.getSimpleKind());
6892f7efbc9SZachary Turner   else
6902f7efbc9SZachary Turner     name = computeTypeName(stream.typeCollection(), mr.ModifiedType);
6912f7efbc9SZachary Turner   Declaration decl;
6926284aee9SZachary Turner   return std::make_shared<Type>(toOpaqueUid(type_id), m_clang->GetSymbolFile(),
6932f7efbc9SZachary Turner                                 ConstString(name), t->GetByteSize(), nullptr,
6942f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
6952f7efbc9SZachary Turner                                 ct, Type::eResolveStateFull);
6962f7efbc9SZachary Turner }
6972f7efbc9SZachary Turner 
6982f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreatePointerType(
6996284aee9SZachary Turner     PdbTypeSymId type_id, const llvm::codeview::PointerRecord &pr) {
7002f7efbc9SZachary Turner   TypeSP pointee = GetOrCreateType(pr.ReferentType);
701544a66d8SZachary Turner   if (!pointee)
702544a66d8SZachary Turner     return nullptr;
7032f7efbc9SZachary Turner   CompilerType pointee_ct = pointee->GetForwardCompilerType();
7042f7efbc9SZachary Turner   lldbassert(pointee_ct);
7052f7efbc9SZachary Turner   Declaration decl;
7062f7efbc9SZachary Turner 
7072f7efbc9SZachary Turner   if (pr.isPointerToMember()) {
7082f7efbc9SZachary Turner     MemberPointerInfo mpi = pr.getMemberInfo();
7092f7efbc9SZachary Turner     TypeSP class_type = GetOrCreateType(mpi.ContainingType);
7102f7efbc9SZachary Turner 
7112f7efbc9SZachary Turner     CompilerType ct = ClangASTContext::CreateMemberPointerType(
7122f7efbc9SZachary Turner         class_type->GetLayoutCompilerType(), pointee_ct);
7132f7efbc9SZachary Turner 
7142f7efbc9SZachary Turner     return std::make_shared<Type>(
7156284aee9SZachary Turner         toOpaqueUid(type_id), m_clang->GetSymbolFile(), ConstString(),
7162f7efbc9SZachary Turner         pr.getSize(), nullptr, LLDB_INVALID_UID, Type::eEncodingIsUID, decl, ct,
7172f7efbc9SZachary Turner         Type::eResolveStateFull);
7182f7efbc9SZachary Turner   }
7192f7efbc9SZachary Turner 
7202f7efbc9SZachary Turner   CompilerType pointer_ct = pointee_ct;
7212f7efbc9SZachary Turner   if (pr.getMode() == PointerMode::LValueReference)
7222f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetLValueReferenceType();
7232f7efbc9SZachary Turner   else if (pr.getMode() == PointerMode::RValueReference)
7242f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetRValueReferenceType();
7252f7efbc9SZachary Turner   else
7262f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetPointerType();
7272f7efbc9SZachary Turner 
7282f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Const) != PointerOptions::None)
7292f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddConstModifier();
7302f7efbc9SZachary Turner 
7312f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Volatile) != PointerOptions::None)
7322f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddVolatileModifier();
7332f7efbc9SZachary Turner 
7342f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Restrict) != PointerOptions::None)
7352f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddRestrictModifier();
7362f7efbc9SZachary Turner 
7376284aee9SZachary Turner   return std::make_shared<Type>(toOpaqueUid(type_id), m_clang->GetSymbolFile(),
7382f7efbc9SZachary Turner                                 ConstString(), pr.getSize(), nullptr,
7392f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
7402f7efbc9SZachary Turner                                 pointer_ct, Type::eResolveStateFull);
7412f7efbc9SZachary Turner }
7422f7efbc9SZachary Turner 
7432f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateSimpleType(TypeIndex ti) {
7449fbf9350SZachary Turner   uint64_t uid = toOpaqueUid(PdbTypeSymId(ti, false));
745544a66d8SZachary Turner   if (ti == TypeIndex::NullptrT()) {
746544a66d8SZachary Turner     CompilerType ct = m_clang->GetBasicType(eBasicTypeNullPtr);
747544a66d8SZachary Turner     Declaration decl;
7486284aee9SZachary Turner     return std::make_shared<Type>(
7496284aee9SZachary Turner         uid, this, ConstString("std::nullptr_t"), 0, nullptr, LLDB_INVALID_UID,
7506284aee9SZachary Turner         Type::eEncodingIsUID, decl, ct, Type::eResolveStateFull);
751544a66d8SZachary Turner   }
752544a66d8SZachary Turner 
7532f7efbc9SZachary Turner   if (ti.getSimpleMode() != SimpleTypeMode::Direct) {
7542f7efbc9SZachary Turner     TypeSP direct_sp = GetOrCreateType(ti.makeDirect());
7552f7efbc9SZachary Turner     CompilerType ct = direct_sp->GetFullCompilerType();
7562f7efbc9SZachary Turner     ct = ct.GetPointerType();
75771ebb721SZachary Turner     uint32_t pointer_size = 0;
7582f7efbc9SZachary Turner     switch (ti.getSimpleMode()) {
7592f7efbc9SZachary Turner     case SimpleTypeMode::FarPointer32:
7602f7efbc9SZachary Turner     case SimpleTypeMode::NearPointer32:
7612f7efbc9SZachary Turner       pointer_size = 4;
7622f7efbc9SZachary Turner       break;
7632f7efbc9SZachary Turner     case SimpleTypeMode::NearPointer64:
7642f7efbc9SZachary Turner       pointer_size = 8;
7652f7efbc9SZachary Turner       break;
7662f7efbc9SZachary Turner     default:
7672f7efbc9SZachary Turner       // 128-bit and 16-bit pointers unsupported.
7682f7efbc9SZachary Turner       return nullptr;
7692f7efbc9SZachary Turner     }
7702f7efbc9SZachary Turner     Declaration decl;
7716284aee9SZachary Turner     return std::make_shared<Type>(uid, m_clang->GetSymbolFile(), ConstString(),
7726284aee9SZachary Turner                                   pointer_size, nullptr, LLDB_INVALID_UID,
7736284aee9SZachary Turner                                   Type::eEncodingIsUID, decl, ct,
7746284aee9SZachary Turner                                   Type::eResolveStateFull);
7752f7efbc9SZachary Turner   }
7762f7efbc9SZachary Turner 
7772f7efbc9SZachary Turner   if (ti.getSimpleKind() == SimpleTypeKind::NotTranslated)
7782f7efbc9SZachary Turner     return nullptr;
7792f7efbc9SZachary Turner 
7802f7efbc9SZachary Turner   lldb::BasicType bt = GetCompilerTypeForSimpleKind(ti.getSimpleKind());
781544a66d8SZachary Turner   if (bt == lldb::eBasicTypeInvalid)
782544a66d8SZachary Turner     return nullptr;
7832f7efbc9SZachary Turner   CompilerType ct = m_clang->GetBasicType(bt);
7842f7efbc9SZachary Turner   size_t size = GetTypeSizeForSimpleKind(ti.getSimpleKind());
7852f7efbc9SZachary Turner 
7862f7efbc9SZachary Turner   llvm::StringRef type_name = GetSimpleTypeName(ti.getSimpleKind());
7872f7efbc9SZachary Turner 
7882f7efbc9SZachary Turner   Declaration decl;
7896284aee9SZachary Turner   return std::make_shared<Type>(uid, m_clang->GetSymbolFile(),
7902f7efbc9SZachary Turner                                 ConstString(type_name), size, nullptr,
7912f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
7922f7efbc9SZachary Turner                                 ct, Type::eResolveStateFull);
7932f7efbc9SZachary Turner }
7942f7efbc9SZachary Turner 
795056e4ab4SZachary Turner static std::string RenderDemanglerNode(llvm::ms_demangle::Node *n) {
796056e4ab4SZachary Turner   OutputStream OS;
797056e4ab4SZachary Turner   initializeOutputStream(nullptr, nullptr, OS, 1024);
798056e4ab4SZachary Turner   n->output(OS, llvm::ms_demangle::OF_Default);
799056e4ab4SZachary Turner   OS << '\0';
800056e4ab4SZachary Turner   return {OS.getBuffer()};
801056e4ab4SZachary Turner }
802056e4ab4SZachary Turner 
80303a24052SZachary Turner static bool
80403a24052SZachary Turner AnyScopesHaveTemplateParams(llvm::ArrayRef<llvm::ms_demangle::Node *> scopes) {
80503a24052SZachary Turner   for (llvm::ms_demangle::Node *n : scopes) {
80603a24052SZachary Turner     auto *idn = static_cast<llvm::ms_demangle::IdentifierNode *>(n);
80703a24052SZachary Turner     if (idn->TemplateParams)
80803a24052SZachary Turner       return true;
80903a24052SZachary Turner   }
81003a24052SZachary Turner   return false;
81103a24052SZachary Turner }
81203a24052SZachary Turner 
813056e4ab4SZachary Turner std::pair<clang::DeclContext *, std::string>
814056e4ab4SZachary Turner SymbolFileNativePDB::CreateDeclInfoForType(const TagRecord &record,
815056e4ab4SZachary Turner                                            TypeIndex ti) {
816*a42bbe39SZachary Turner   // FIXME: Move this to GetDeclContextContainingUID.
817*a42bbe39SZachary Turner 
818056e4ab4SZachary Turner   llvm::ms_demangle::Demangler demangler;
819056e4ab4SZachary Turner   StringView sv(record.UniqueName.begin(), record.UniqueName.size());
820056e4ab4SZachary Turner   llvm::ms_demangle::TagTypeNode *ttn = demangler.parseTagUniqueName(sv);
821056e4ab4SZachary Turner   llvm::ms_demangle::IdentifierNode *idn =
822056e4ab4SZachary Turner       ttn->QualifiedName->getUnqualifiedIdentifier();
823056e4ab4SZachary Turner   std::string uname = RenderDemanglerNode(idn);
824056e4ab4SZachary Turner 
825056e4ab4SZachary Turner   llvm::ms_demangle::NodeArrayNode *name_components =
826056e4ab4SZachary Turner       ttn->QualifiedName->Components;
827056e4ab4SZachary Turner   llvm::ArrayRef<llvm::ms_demangle::Node *> scopes(name_components->Nodes,
828056e4ab4SZachary Turner                                                    name_components->Count - 1);
829056e4ab4SZachary Turner 
830056e4ab4SZachary Turner   clang::DeclContext *context = m_clang->GetTranslationUnitDecl();
831056e4ab4SZachary Turner 
832056e4ab4SZachary Turner   // If this type doesn't have a parent type in the debug info, then the best we
833056e4ab4SZachary Turner   // can do is to say that it's either a series of namespaces (if the scope is
834056e4ab4SZachary Turner   // non-empty), or the translation unit (if the scope is empty).
835056e4ab4SZachary Turner   auto parent_iter = m_parent_types.find(ti);
836056e4ab4SZachary Turner   if (parent_iter == m_parent_types.end()) {
837056e4ab4SZachary Turner     if (scopes.empty())
838056e4ab4SZachary Turner       return {context, uname};
839056e4ab4SZachary Turner 
84003a24052SZachary Turner     // If there is no parent in the debug info, but some of the scopes have
84103a24052SZachary Turner     // template params, then this is a case of bad debug info.  See, for
84203a24052SZachary Turner     // example, llvm.org/pr39607.  We don't want to create an ambiguity between
84303a24052SZachary Turner     // a NamespaceDecl and a CXXRecordDecl, so instead we create a class at
84403a24052SZachary Turner     // global scope with the fully qualified name.
84503a24052SZachary Turner     if (AnyScopesHaveTemplateParams(scopes))
84603a24052SZachary Turner       return {context, record.Name};
84703a24052SZachary Turner 
848056e4ab4SZachary Turner     for (llvm::ms_demangle::Node *scope : scopes) {
849056e4ab4SZachary Turner       auto *nii = static_cast<llvm::ms_demangle::NamedIdentifierNode *>(scope);
850056e4ab4SZachary Turner       std::string str = RenderDemanglerNode(nii);
851056e4ab4SZachary Turner       context = m_clang->GetUniqueNamespaceDeclaration(str.c_str(), context);
852056e4ab4SZachary Turner     }
853056e4ab4SZachary Turner     return {context, uname};
854056e4ab4SZachary Turner   }
855056e4ab4SZachary Turner 
856056e4ab4SZachary Turner   // Otherwise, all we need to do is get the parent type of this type and
857056e4ab4SZachary Turner   // recurse into our lazy type creation / AST reconstruction logic to get an
858056e4ab4SZachary Turner   // LLDB TypeSP for the parent.  This will cause the AST to automatically get
859056e4ab4SZachary Turner   // the right DeclContext created for any parent.
860056e4ab4SZachary Turner   TypeSP parent = GetOrCreateType(parent_iter->second);
861056e4ab4SZachary Turner   if (!parent)
862056e4ab4SZachary Turner     return {context, uname};
863056e4ab4SZachary Turner   CompilerType parent_ct = parent->GetForwardCompilerType();
864056e4ab4SZachary Turner   clang::QualType qt = ClangUtil::GetCanonicalQualType(parent_ct);
865056e4ab4SZachary Turner   context = clang::TagDecl::castToDeclContext(qt->getAsTagDecl());
866056e4ab4SZachary Turner   return {context, uname};
867056e4ab4SZachary Turner }
868056e4ab4SZachary Turner 
8692f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateClassStructUnion(
8706284aee9SZachary Turner     PdbTypeSymId type_id, const llvm::codeview::TagRecord &record, size_t size,
8712f7efbc9SZachary Turner     clang::TagTypeKind ttk, clang::MSInheritanceAttr::Spelling inheritance) {
8722f7efbc9SZachary Turner 
873056e4ab4SZachary Turner   clang::DeclContext *decl_context = nullptr;
874056e4ab4SZachary Turner   std::string uname;
8756284aee9SZachary Turner   std::tie(decl_context, uname) = CreateDeclInfoForType(record, type_id.index);
8762f7efbc9SZachary Turner 
8772f7efbc9SZachary Turner   lldb::AccessType access =
8782f7efbc9SZachary Turner       (ttk == clang::TTK_Class) ? lldb::eAccessPrivate : lldb::eAccessPublic;
8792f7efbc9SZachary Turner 
8802f7efbc9SZachary Turner   ClangASTMetadata metadata;
8816284aee9SZachary Turner   metadata.SetUserID(toOpaqueUid(type_id));
8822f7efbc9SZachary Turner   metadata.SetIsDynamicCXXType(false);
8832f7efbc9SZachary Turner 
8842f7efbc9SZachary Turner   CompilerType ct =
885056e4ab4SZachary Turner       m_clang->CreateRecordType(decl_context, access, uname.c_str(), ttk,
8862f7efbc9SZachary Turner                                 lldb::eLanguageTypeC_plus_plus, &metadata);
887056e4ab4SZachary Turner 
8882f7efbc9SZachary Turner   lldbassert(ct.IsValid());
8892f7efbc9SZachary Turner 
8902f7efbc9SZachary Turner   clang::CXXRecordDecl *record_decl =
8912f7efbc9SZachary Turner       m_clang->GetAsCXXRecordDecl(ct.GetOpaqueQualType());
8922f7efbc9SZachary Turner   lldbassert(record_decl);
8932f7efbc9SZachary Turner 
8942f7efbc9SZachary Turner   clang::MSInheritanceAttr *attr = clang::MSInheritanceAttr::CreateImplicit(
8952f7efbc9SZachary Turner       *m_clang->getASTContext(), inheritance);
8962f7efbc9SZachary Turner   record_decl->addAttr(attr);
8972f7efbc9SZachary Turner 
8982f7efbc9SZachary Turner   ClangASTContext::StartTagDeclarationDefinition(ct);
8992f7efbc9SZachary Turner 
9002f7efbc9SZachary Turner   // Even if it's possible, don't complete it at this point. Just mark it
9012f7efbc9SZachary Turner   // forward resolved, and if/when LLDB needs the full definition, it can
9022f7efbc9SZachary Turner   // ask us.
9032f7efbc9SZachary Turner   ClangASTContext::SetHasExternalStorage(ct.GetOpaqueQualType(), true);
9042f7efbc9SZachary Turner 
9052f7efbc9SZachary Turner   // FIXME: Search IPI stream for LF_UDT_MOD_SRC_LINE.
9062f7efbc9SZachary Turner   Declaration decl;
9076284aee9SZachary Turner   return std::make_shared<Type>(toOpaqueUid(type_id), m_clang->GetSymbolFile(),
908056e4ab4SZachary Turner                                 ConstString(uname), size, nullptr,
9092f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
9102f7efbc9SZachary Turner                                 ct, Type::eResolveStateForward);
9112f7efbc9SZachary Turner }
9122f7efbc9SZachary Turner 
9136284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbTypeSymId type_id,
9142f7efbc9SZachary Turner                                                 const ClassRecord &cr) {
9152f7efbc9SZachary Turner   clang::TagTypeKind ttk = TranslateUdtKind(cr);
9162f7efbc9SZachary Turner 
9172f7efbc9SZachary Turner   clang::MSInheritanceAttr::Spelling inheritance =
9182f7efbc9SZachary Turner       GetMSInheritance(m_index->tpi().typeCollection(), cr);
9196284aee9SZachary Turner   return CreateClassStructUnion(type_id, cr, cr.getSize(), ttk, inheritance);
9202f7efbc9SZachary Turner }
9212f7efbc9SZachary Turner 
9226284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbTypeSymId type_id,
9232f7efbc9SZachary Turner                                                 const UnionRecord &ur) {
9242f7efbc9SZachary Turner   return CreateClassStructUnion(
9256284aee9SZachary Turner       type_id, ur, ur.getSize(), clang::TTK_Union,
9262f7efbc9SZachary Turner       clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance);
9272f7efbc9SZachary Turner }
9282f7efbc9SZachary Turner 
9296284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbTypeSymId type_id,
9302f7efbc9SZachary Turner                                                 const EnumRecord &er) {
9312af34166SZachary Turner   clang::DeclContext *decl_context = nullptr;
9322af34166SZachary Turner   std::string uname;
9336284aee9SZachary Turner   std::tie(decl_context, uname) = CreateDeclInfoForType(er, type_id.index);
9342f7efbc9SZachary Turner 
9352f7efbc9SZachary Turner   Declaration decl;
9362f7efbc9SZachary Turner   TypeSP underlying_type = GetOrCreateType(er.UnderlyingType);
9372f7efbc9SZachary Turner   CompilerType enum_ct = m_clang->CreateEnumerationType(
9382af34166SZachary Turner       uname.c_str(), decl_context, decl, underlying_type->GetFullCompilerType(),
9392af34166SZachary Turner       er.isScoped());
9402f7efbc9SZachary Turner 
9412f7efbc9SZachary Turner   ClangASTContext::StartTagDeclarationDefinition(enum_ct);
94212abab57SZachary Turner   ClangASTContext::SetHasExternalStorage(enum_ct.GetOpaqueQualType(), true);
9432f7efbc9SZachary Turner 
9442f7efbc9SZachary Turner   // We're just going to forward resolve this for now.  We'll complete
9452f7efbc9SZachary Turner   // it only if the user requests.
9462f7efbc9SZachary Turner   return std::make_shared<lldb_private::Type>(
9476284aee9SZachary Turner       toOpaqueUid(type_id), m_clang->GetSymbolFile(), ConstString(uname),
9482f7efbc9SZachary Turner       underlying_type->GetByteSize(), nullptr, LLDB_INVALID_UID,
9492f7efbc9SZachary Turner       lldb_private::Type::eEncodingIsUID, decl, enum_ct,
9502f7efbc9SZachary Turner       lldb_private::Type::eResolveStateForward);
9512f7efbc9SZachary Turner }
9522f7efbc9SZachary Turner 
9536284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateArrayType(PdbTypeSymId type_id,
954511bff21SZachary Turner                                             const ArrayRecord &ar) {
955511bff21SZachary Turner   TypeSP element_type = GetOrCreateType(ar.ElementType);
956511bff21SZachary Turner   uint64_t element_count = ar.Size / element_type->GetByteSize();
957511bff21SZachary Turner 
958511bff21SZachary Turner   CompilerType element_ct = element_type->GetFullCompilerType();
959511bff21SZachary Turner 
960511bff21SZachary Turner   CompilerType array_ct =
961511bff21SZachary Turner       m_clang->CreateArrayType(element_ct, element_count, false);
962511bff21SZachary Turner 
963511bff21SZachary Turner   Declaration decl;
964511bff21SZachary Turner   TypeSP array_sp = std::make_shared<lldb_private::Type>(
9656284aee9SZachary Turner       toOpaqueUid(type_id), m_clang->GetSymbolFile(), ConstString(), ar.Size,
966511bff21SZachary Turner       nullptr, LLDB_INVALID_UID, lldb_private::Type::eEncodingIsUID, decl,
967511bff21SZachary Turner       array_ct, lldb_private::Type::eResolveStateFull);
968511bff21SZachary Turner   array_sp->SetEncodingType(element_type.get());
969511bff21SZachary Turner   return array_sp;
970511bff21SZachary Turner }
971511bff21SZachary Turner 
9726284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateProcedureType(PdbTypeSymId type_id,
973544a66d8SZachary Turner                                                 const ProcedureRecord &pr) {
974544a66d8SZachary Turner   TpiStream &stream = m_index->tpi();
975544a66d8SZachary Turner   CVType args_cvt = stream.getType(pr.ArgumentList);
976544a66d8SZachary Turner   ArgListRecord args;
977544a66d8SZachary Turner   llvm::cantFail(
978544a66d8SZachary Turner       TypeDeserializer::deserializeAs<ArgListRecord>(args_cvt, args));
979544a66d8SZachary Turner 
980544a66d8SZachary Turner   llvm::ArrayRef<TypeIndex> arg_indices = llvm::makeArrayRef(args.ArgIndices);
981544a66d8SZachary Turner   bool is_variadic = IsCVarArgsFunction(arg_indices);
982544a66d8SZachary Turner   if (is_variadic)
983544a66d8SZachary Turner     arg_indices = arg_indices.drop_back();
984544a66d8SZachary Turner 
985544a66d8SZachary Turner   std::vector<CompilerType> arg_list;
986544a66d8SZachary Turner   arg_list.reserve(arg_list.size());
987544a66d8SZachary Turner 
988544a66d8SZachary Turner   for (TypeIndex arg_index : arg_indices) {
989544a66d8SZachary Turner     TypeSP arg_sp = GetOrCreateType(arg_index);
990544a66d8SZachary Turner     if (!arg_sp)
991544a66d8SZachary Turner       return nullptr;
992544a66d8SZachary Turner     arg_list.push_back(arg_sp->GetFullCompilerType());
993544a66d8SZachary Turner   }
994544a66d8SZachary Turner 
995544a66d8SZachary Turner   TypeSP return_type_sp = GetOrCreateType(pr.ReturnType);
996544a66d8SZachary Turner   if (!return_type_sp)
997544a66d8SZachary Turner     return nullptr;
998544a66d8SZachary Turner 
999544a66d8SZachary Turner   llvm::Optional<clang::CallingConv> cc =
1000544a66d8SZachary Turner       TranslateCallingConvention(pr.CallConv);
1001544a66d8SZachary Turner   if (!cc)
1002544a66d8SZachary Turner     return nullptr;
1003544a66d8SZachary Turner 
1004544a66d8SZachary Turner   CompilerType return_ct = return_type_sp->GetFullCompilerType();
1005544a66d8SZachary Turner   CompilerType func_sig_ast_type = m_clang->CreateFunctionType(
1006544a66d8SZachary Turner       return_ct, arg_list.data(), arg_list.size(), is_variadic, 0, *cc);
1007544a66d8SZachary Turner 
1008544a66d8SZachary Turner   Declaration decl;
1009544a66d8SZachary Turner   return std::make_shared<lldb_private::Type>(
10106284aee9SZachary Turner       toOpaqueUid(type_id), this, ConstString(), 0, nullptr, LLDB_INVALID_UID,
1011544a66d8SZachary Turner       lldb_private::Type::eEncodingIsUID, decl, func_sig_ast_type,
1012544a66d8SZachary Turner       lldb_private::Type::eResolveStateFull);
1013544a66d8SZachary Turner }
1014544a66d8SZachary Turner 
10156284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateType(PdbTypeSymId type_id) {
10166284aee9SZachary Turner   if (type_id.index.isSimple())
10176284aee9SZachary Turner     return CreateSimpleType(type_id.index);
10182f7efbc9SZachary Turner 
10196284aee9SZachary Turner   TpiStream &stream = type_id.is_ipi ? m_index->ipi() : m_index->tpi();
10206284aee9SZachary Turner   CVType cvt = stream.getType(type_id.index);
10212f7efbc9SZachary Turner 
10222f7efbc9SZachary Turner   if (cvt.kind() == LF_MODIFIER) {
10232f7efbc9SZachary Turner     ModifierRecord modifier;
10242f7efbc9SZachary Turner     llvm::cantFail(
10252f7efbc9SZachary Turner         TypeDeserializer::deserializeAs<ModifierRecord>(cvt, modifier));
10266284aee9SZachary Turner     return CreateModifierType(type_id, modifier);
10272f7efbc9SZachary Turner   }
10282f7efbc9SZachary Turner 
10292f7efbc9SZachary Turner   if (cvt.kind() == LF_POINTER) {
10302f7efbc9SZachary Turner     PointerRecord pointer;
10312f7efbc9SZachary Turner     llvm::cantFail(
10322f7efbc9SZachary Turner         TypeDeserializer::deserializeAs<PointerRecord>(cvt, pointer));
10336284aee9SZachary Turner     return CreatePointerType(type_id, pointer);
10342f7efbc9SZachary Turner   }
10352f7efbc9SZachary Turner 
10362f7efbc9SZachary Turner   if (IsClassRecord(cvt.kind())) {
10372f7efbc9SZachary Turner     ClassRecord cr;
10382f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ClassRecord>(cvt, cr));
10396284aee9SZachary Turner     return CreateTagType(type_id, cr);
10402f7efbc9SZachary Turner   }
10412f7efbc9SZachary Turner 
10422f7efbc9SZachary Turner   if (cvt.kind() == LF_ENUM) {
10432f7efbc9SZachary Turner     EnumRecord er;
10442f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<EnumRecord>(cvt, er));
10456284aee9SZachary Turner     return CreateTagType(type_id, er);
10462f7efbc9SZachary Turner   }
10472f7efbc9SZachary Turner 
10482f7efbc9SZachary Turner   if (cvt.kind() == LF_UNION) {
10492f7efbc9SZachary Turner     UnionRecord ur;
10502f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<UnionRecord>(cvt, ur));
10516284aee9SZachary Turner     return CreateTagType(type_id, ur);
10522f7efbc9SZachary Turner   }
10532f7efbc9SZachary Turner 
1054511bff21SZachary Turner   if (cvt.kind() == LF_ARRAY) {
1055511bff21SZachary Turner     ArrayRecord ar;
1056511bff21SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ArrayRecord>(cvt, ar));
10576284aee9SZachary Turner     return CreateArrayType(type_id, ar);
1058511bff21SZachary Turner   }
1059511bff21SZachary Turner 
1060544a66d8SZachary Turner   if (cvt.kind() == LF_PROCEDURE) {
1061544a66d8SZachary Turner     ProcedureRecord pr;
1062544a66d8SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ProcedureRecord>(cvt, pr));
10636284aee9SZachary Turner     return CreateProcedureType(type_id, pr);
1064544a66d8SZachary Turner   }
1065544a66d8SZachary Turner 
10662f7efbc9SZachary Turner   return nullptr;
10672f7efbc9SZachary Turner }
10682f7efbc9SZachary Turner 
10696284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateAndCacheType(PdbTypeSymId type_id) {
10702f7efbc9SZachary Turner   // If they search for a UDT which is a forward ref, try and resolve the full
10712f7efbc9SZachary Turner   // decl and just map the forward ref uid to the full decl record.
10726284aee9SZachary Turner   llvm::Optional<PdbTypeSymId> full_decl_uid;
10736284aee9SZachary Turner   if (IsForwardRefUdt(type_id, m_index->tpi())) {
10746284aee9SZachary Turner     auto expected_full_ti =
10756284aee9SZachary Turner         m_index->tpi().findFullDeclForForwardRef(type_id.index);
10762f7efbc9SZachary Turner     if (!expected_full_ti)
10772f7efbc9SZachary Turner       llvm::consumeError(expected_full_ti.takeError());
10786284aee9SZachary Turner     else if (*expected_full_ti != type_id.index) {
10799fbf9350SZachary Turner       full_decl_uid = PdbTypeSymId(*expected_full_ti, false);
10802f7efbc9SZachary Turner 
10812f7efbc9SZachary Turner       // It's possible that a lookup would occur for the full decl causing it
10822f7efbc9SZachary Turner       // to be cached, then a second lookup would occur for the forward decl.
10832f7efbc9SZachary Turner       // We don't want to create a second full decl, so make sure the full
10842f7efbc9SZachary Turner       // decl hasn't already been cached.
10856284aee9SZachary Turner       auto full_iter = m_types.find(toOpaqueUid(*full_decl_uid));
10862f7efbc9SZachary Turner       if (full_iter != m_types.end()) {
10872f7efbc9SZachary Turner         TypeSP result = full_iter->second;
10882f7efbc9SZachary Turner         // Map the forward decl to the TypeSP for the full decl so we can take
10892f7efbc9SZachary Turner         // the fast path next time.
10906284aee9SZachary Turner         m_types[toOpaqueUid(type_id)] = result;
10912f7efbc9SZachary Turner         return result;
10922f7efbc9SZachary Turner       }
10932f7efbc9SZachary Turner     }
10942f7efbc9SZachary Turner   }
10952f7efbc9SZachary Turner 
10966284aee9SZachary Turner   PdbTypeSymId best_decl_id = full_decl_uid ? *full_decl_uid : type_id;
10976284aee9SZachary Turner   TypeSP result = CreateType(best_decl_id);
1098544a66d8SZachary Turner   if (!result)
1099544a66d8SZachary Turner     return nullptr;
11006284aee9SZachary Turner 
11016284aee9SZachary Turner   uint64_t best_uid = toOpaqueUid(best_decl_id);
11026284aee9SZachary Turner   m_types[best_uid] = result;
11032f7efbc9SZachary Turner   // If we had both a forward decl and a full decl, make both point to the new
11042f7efbc9SZachary Turner   // type.
11052f7efbc9SZachary Turner   if (full_decl_uid)
11066284aee9SZachary Turner     m_types[toOpaqueUid(type_id)] = result;
11072f7efbc9SZachary Turner 
11086284aee9SZachary Turner   if (IsTagRecord(best_decl_id, m_index->tpi())) {
11092f7efbc9SZachary Turner     clang::TagDecl *record_decl =
11102f7efbc9SZachary Turner         m_clang->GetAsTagDecl(result->GetForwardCompilerType());
11112f7efbc9SZachary Turner     lldbassert(record_decl);
11122f7efbc9SZachary Turner 
11136284aee9SZachary Turner     m_uid_to_decl[best_uid] = record_decl;
11142f7efbc9SZachary Turner     m_decl_to_status[record_decl] =
11156284aee9SZachary Turner         DeclStatus(best_uid, Type::eResolveStateForward);
11162f7efbc9SZachary Turner   }
11172f7efbc9SZachary Turner   return result;
11182f7efbc9SZachary Turner }
11192f7efbc9SZachary Turner 
11206284aee9SZachary Turner TypeSP SymbolFileNativePDB::GetOrCreateType(PdbTypeSymId type_id) {
11212f7efbc9SZachary Turner   // We can't use try_emplace / overwrite here because the process of creating
11222f7efbc9SZachary Turner   // a type could create nested types, which could invalidate iterators.  So
11232f7efbc9SZachary Turner   // we have to do a 2-phase lookup / insert.
11246284aee9SZachary Turner   auto iter = m_types.find(toOpaqueUid(type_id));
11252f7efbc9SZachary Turner   if (iter != m_types.end())
11262f7efbc9SZachary Turner     return iter->second;
11272f7efbc9SZachary Turner 
11286284aee9SZachary Turner   return CreateAndCacheType(type_id);
11292f7efbc9SZachary Turner }
11302f7efbc9SZachary Turner 
11316284aee9SZachary Turner VariableSP SymbolFileNativePDB::CreateGlobalVariable(PdbGlobalSymId var_id) {
11326284aee9SZachary Turner   CVSymbol sym = m_index->symrecords().readRecord(var_id.offset);
11332af34166SZachary Turner   if (sym.kind() == S_CONSTANT)
11346284aee9SZachary Turner     return CreateConstantSymbol(var_id, sym);
11352af34166SZachary Turner 
11369f727950SZachary Turner   lldb::ValueType scope = eValueTypeInvalid;
11379f727950SZachary Turner   TypeIndex ti;
11389f727950SZachary Turner   llvm::StringRef name;
11399f727950SZachary Turner   lldb::addr_t addr = 0;
11409f727950SZachary Turner   uint16_t section = 0;
11419f727950SZachary Turner   uint32_t offset = 0;
11429f727950SZachary Turner   bool is_external = false;
11439f727950SZachary Turner   switch (sym.kind()) {
11449f727950SZachary Turner   case S_GDATA32:
11459f727950SZachary Turner     is_external = true;
11469f727950SZachary Turner     LLVM_FALLTHROUGH;
11479f727950SZachary Turner   case S_LDATA32: {
11489f727950SZachary Turner     DataSym ds(sym.kind());
11499f727950SZachary Turner     llvm::cantFail(SymbolDeserializer::deserializeAs<DataSym>(sym, ds));
11509f727950SZachary Turner     ti = ds.Type;
11519f727950SZachary Turner     scope = (sym.kind() == S_GDATA32) ? eValueTypeVariableGlobal
11529f727950SZachary Turner                                       : eValueTypeVariableStatic;
11539f727950SZachary Turner     name = ds.Name;
11549f727950SZachary Turner     section = ds.Segment;
11559f727950SZachary Turner     offset = ds.DataOffset;
11569f727950SZachary Turner     addr = m_index->MakeVirtualAddress(ds.Segment, ds.DataOffset);
11579f727950SZachary Turner     break;
11589f727950SZachary Turner   }
11599f727950SZachary Turner   case S_GTHREAD32:
11609f727950SZachary Turner     is_external = true;
11619f727950SZachary Turner     LLVM_FALLTHROUGH;
11629f727950SZachary Turner   case S_LTHREAD32: {
11639f727950SZachary Turner     ThreadLocalDataSym tlds(sym.kind());
11649f727950SZachary Turner     llvm::cantFail(
11659f727950SZachary Turner         SymbolDeserializer::deserializeAs<ThreadLocalDataSym>(sym, tlds));
11669f727950SZachary Turner     ti = tlds.Type;
11679f727950SZachary Turner     name = tlds.Name;
11689f727950SZachary Turner     section = tlds.Segment;
11699f727950SZachary Turner     offset = tlds.DataOffset;
11709f727950SZachary Turner     addr = m_index->MakeVirtualAddress(tlds.Segment, tlds.DataOffset);
11719f727950SZachary Turner     scope = eValueTypeVariableThreadLocal;
11729f727950SZachary Turner     break;
11739f727950SZachary Turner   }
11749f727950SZachary Turner   default:
11759f727950SZachary Turner     llvm_unreachable("unreachable!");
11769f727950SZachary Turner   }
11779f727950SZachary Turner 
11789f727950SZachary Turner   CompUnitSP comp_unit;
11799f727950SZachary Turner   llvm::Optional<uint16_t> modi = m_index->GetModuleIndexForVa(addr);
11809f727950SZachary Turner   if (modi) {
11816284aee9SZachary Turner     CompilandIndexItem &cci = m_index->compilands().GetOrCreateCompiland(*modi);
11829f727950SZachary Turner     comp_unit = GetOrCreateCompileUnit(cci);
11839f727950SZachary Turner   }
11849f727950SZachary Turner 
11859f727950SZachary Turner   Declaration decl;
11869fbf9350SZachary Turner   PdbTypeSymId tid(ti, false);
11879f727950SZachary Turner   SymbolFileTypeSP type_sp =
11886284aee9SZachary Turner       std::make_shared<SymbolFileType>(*this, toOpaqueUid(tid));
11899f727950SZachary Turner   Variable::RangeList ranges;
11909f727950SZachary Turner 
11919f727950SZachary Turner   DWARFExpression location = MakeGlobalLocationExpression(
11929f727950SZachary Turner       section, offset, GetObjectFile()->GetModule());
11939f727950SZachary Turner 
11949f727950SZachary Turner   std::string global_name("::");
11959f727950SZachary Turner   global_name += name;
11969f727950SZachary Turner   VariableSP var_sp = std::make_shared<Variable>(
11976284aee9SZachary Turner       toOpaqueUid(var_id), name.str().c_str(), global_name.c_str(), type_sp,
11989f727950SZachary Turner       scope, comp_unit.get(), ranges, &decl, location, is_external, false,
11999f727950SZachary Turner       false);
12009f727950SZachary Turner   var_sp->SetLocationIsConstantValueData(false);
12019f727950SZachary Turner 
12029f727950SZachary Turner   return var_sp;
12039f727950SZachary Turner }
12049f727950SZachary Turner 
12052af34166SZachary Turner lldb::VariableSP
12066284aee9SZachary Turner SymbolFileNativePDB::CreateConstantSymbol(PdbGlobalSymId var_id,
12072af34166SZachary Turner                                           const CVSymbol &cvs) {
12082af34166SZachary Turner   TpiStream &tpi = m_index->tpi();
12092af34166SZachary Turner   ConstantSym constant(cvs.kind());
12102af34166SZachary Turner 
12112af34166SZachary Turner   llvm::cantFail(SymbolDeserializer::deserializeAs<ConstantSym>(cvs, constant));
12122af34166SZachary Turner   std::string global_name("::");
12132af34166SZachary Turner   global_name += constant.Name;
12149fbf9350SZachary Turner   PdbTypeSymId tid(constant.Type, false);
12152af34166SZachary Turner   SymbolFileTypeSP type_sp =
12166284aee9SZachary Turner       std::make_shared<SymbolFileType>(*this, toOpaqueUid(tid));
12172af34166SZachary Turner 
12182af34166SZachary Turner   Declaration decl;
12192af34166SZachary Turner   Variable::RangeList ranges;
12202af34166SZachary Turner   ModuleSP module = GetObjectFile()->GetModule();
1221a93458b0SZachary Turner   DWARFExpression location = MakeConstantLocationExpression(
1222a93458b0SZachary Turner       constant.Type, tpi, constant.Value, module);
12232af34166SZachary Turner 
12242af34166SZachary Turner   VariableSP var_sp = std::make_shared<Variable>(
12256284aee9SZachary Turner       toOpaqueUid(var_id), constant.Name.str().c_str(), global_name.c_str(),
12262af34166SZachary Turner       type_sp, eValueTypeVariableGlobal, module.get(), ranges, &decl, location,
12272af34166SZachary Turner       false, false, false);
12282af34166SZachary Turner   var_sp->SetLocationIsConstantValueData(true);
12292af34166SZachary Turner   return var_sp;
12302af34166SZachary Turner }
12312af34166SZachary Turner 
12326284aee9SZachary Turner VariableSP
12336284aee9SZachary Turner SymbolFileNativePDB::GetOrCreateGlobalVariable(PdbGlobalSymId var_id) {
12346284aee9SZachary Turner   auto emplace_result = m_global_vars.try_emplace(toOpaqueUid(var_id), nullptr);
12359f727950SZachary Turner   if (emplace_result.second)
12366284aee9SZachary Turner     emplace_result.first->second = CreateGlobalVariable(var_id);
12379f727950SZachary Turner 
12389f727950SZachary Turner   return emplace_result.first->second;
12399f727950SZachary Turner }
12409f727950SZachary Turner 
12416284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::GetOrCreateType(TypeIndex ti) {
12429fbf9350SZachary Turner   return GetOrCreateType(PdbTypeSymId(ti, false));
12432f7efbc9SZachary Turner }
12442f7efbc9SZachary Turner 
12456284aee9SZachary Turner FunctionSP SymbolFileNativePDB::GetOrCreateFunction(PdbCompilandSymId func_id,
1246307f5ae8SZachary Turner                                                     const SymbolContext &sc) {
12476284aee9SZachary Turner   auto emplace_result = m_functions.try_emplace(toOpaqueUid(func_id), nullptr);
1248307f5ae8SZachary Turner   if (emplace_result.second)
12496284aee9SZachary Turner     emplace_result.first->second = CreateFunction(func_id, sc);
1250307f5ae8SZachary Turner 
1251307f5ae8SZachary Turner   lldbassert(emplace_result.first->second);
1252307f5ae8SZachary Turner   return emplace_result.first->second;
1253307f5ae8SZachary Turner }
1254307f5ae8SZachary Turner 
1255307f5ae8SZachary Turner CompUnitSP
1256307f5ae8SZachary Turner SymbolFileNativePDB::GetOrCreateCompileUnit(const CompilandIndexItem &cci) {
12576284aee9SZachary Turner 
1258307f5ae8SZachary Turner   auto emplace_result =
12596284aee9SZachary Turner       m_compilands.try_emplace(toOpaqueUid(cci.m_id), nullptr);
1260307f5ae8SZachary Turner   if (emplace_result.second)
1261307f5ae8SZachary Turner     emplace_result.first->second = CreateCompileUnit(cci);
1262307f5ae8SZachary Turner 
1263307f5ae8SZachary Turner   lldbassert(emplace_result.first->second);
1264307f5ae8SZachary Turner   return emplace_result.first->second;
1265307f5ae8SZachary Turner }
1266307f5ae8SZachary Turner 
1267307f5ae8SZachary Turner lldb::CompUnitSP SymbolFileNativePDB::ParseCompileUnitAtIndex(uint32_t index) {
1268307f5ae8SZachary Turner   if (index >= GetNumCompileUnits())
1269307f5ae8SZachary Turner     return CompUnitSP();
1270307f5ae8SZachary Turner   lldbassert(index < UINT16_MAX);
1271307f5ae8SZachary Turner   if (index >= UINT16_MAX)
1272307f5ae8SZachary Turner     return nullptr;
1273307f5ae8SZachary Turner 
1274307f5ae8SZachary Turner   CompilandIndexItem &item = m_index->compilands().GetOrCreateCompiland(index);
1275307f5ae8SZachary Turner 
1276307f5ae8SZachary Turner   return GetOrCreateCompileUnit(item);
1277307f5ae8SZachary Turner }
1278307f5ae8SZachary Turner 
1279b96181c2SZachary Turner lldb::LanguageType
1280b96181c2SZachary Turner SymbolFileNativePDB::ParseCompileUnitLanguage(const SymbolContext &sc) {
1281307f5ae8SZachary Turner   // What fields should I expect to be filled out on the SymbolContext?  Is it
1282307f5ae8SZachary Turner   // safe to assume that `sc.comp_unit` is valid?
1283307f5ae8SZachary Turner   if (!sc.comp_unit)
1284307f5ae8SZachary Turner     return lldb::eLanguageTypeUnknown;
12856284aee9SZachary Turner   PdbSymUid uid(sc.comp_unit->GetID());
12866284aee9SZachary Turner   lldbassert(uid.kind() == PdbSymUidKind::Compiland);
1287307f5ae8SZachary Turner 
12886284aee9SZachary Turner   CompilandIndexItem *item =
12896284aee9SZachary Turner       m_index->compilands().GetCompiland(uid.asCompiland().modi);
1290307f5ae8SZachary Turner   lldbassert(item);
1291307f5ae8SZachary Turner   if (!item->m_compile_opts)
1292307f5ae8SZachary Turner     return lldb::eLanguageTypeUnknown;
1293307f5ae8SZachary Turner 
1294307f5ae8SZachary Turner   return TranslateLanguage(item->m_compile_opts->getLanguage());
1295307f5ae8SZachary Turner }
1296307f5ae8SZachary Turner 
1297b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseCompileUnitFunctions(const SymbolContext &sc) {
1298307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
1299307f5ae8SZachary Turner   return false;
1300307f5ae8SZachary Turner }
1301307f5ae8SZachary Turner 
1302307f5ae8SZachary Turner static bool NeedsResolvedCompileUnit(uint32_t resolve_scope) {
1303307f5ae8SZachary Turner   // If any of these flags are set, we need to resolve the compile unit.
1304307f5ae8SZachary Turner   uint32_t flags = eSymbolContextCompUnit;
1305307f5ae8SZachary Turner   flags |= eSymbolContextVariable;
1306307f5ae8SZachary Turner   flags |= eSymbolContextFunction;
1307307f5ae8SZachary Turner   flags |= eSymbolContextBlock;
1308307f5ae8SZachary Turner   flags |= eSymbolContextLineEntry;
1309307f5ae8SZachary Turner   return (resolve_scope & flags) != 0;
1310307f5ae8SZachary Turner }
1311307f5ae8SZachary Turner 
1312991e4453SZachary Turner uint32_t SymbolFileNativePDB::ResolveSymbolContext(
1313991e4453SZachary Turner     const Address &addr, SymbolContextItem resolve_scope, SymbolContext &sc) {
1314307f5ae8SZachary Turner   uint32_t resolved_flags = 0;
1315307f5ae8SZachary Turner   lldb::addr_t file_addr = addr.GetFileAddress();
1316307f5ae8SZachary Turner 
1317307f5ae8SZachary Turner   if (NeedsResolvedCompileUnit(resolve_scope)) {
1318307f5ae8SZachary Turner     llvm::Optional<uint16_t> modi = m_index->GetModuleIndexForVa(file_addr);
1319307f5ae8SZachary Turner     if (!modi)
1320307f5ae8SZachary Turner       return 0;
13216284aee9SZachary Turner     CompilandIndexItem *cci = m_index->compilands().GetCompiland(*modi);
1322307f5ae8SZachary Turner     if (!cci)
1323307f5ae8SZachary Turner       return 0;
1324307f5ae8SZachary Turner 
1325307f5ae8SZachary Turner     sc.comp_unit = GetOrCreateCompileUnit(*cci).get();
1326307f5ae8SZachary Turner     resolved_flags |= eSymbolContextCompUnit;
1327307f5ae8SZachary Turner   }
1328307f5ae8SZachary Turner 
1329307f5ae8SZachary Turner   if (resolve_scope & eSymbolContextFunction) {
1330307f5ae8SZachary Turner     lldbassert(sc.comp_unit);
1331307f5ae8SZachary Turner     std::vector<SymbolAndUid> matches = m_index->FindSymbolsByVa(file_addr);
1332307f5ae8SZachary Turner     for (const auto &match : matches) {
13336284aee9SZachary Turner       if (match.uid.kind() != PdbSymUidKind::CompilandSym)
1334307f5ae8SZachary Turner         continue;
13356284aee9SZachary Turner       PdbCompilandSymId csid = match.uid.asCompilandSym();
13366284aee9SZachary Turner       CVSymbol cvs = m_index->ReadSymbolRecord(csid);
13376284aee9SZachary Turner       if (CVSymToPDBSym(cvs.kind()) != PDB_SymType::Function)
13386284aee9SZachary Turner         continue;
13396284aee9SZachary Turner       sc.function = GetOrCreateFunction(csid, sc).get();
1340307f5ae8SZachary Turner     }
1341307f5ae8SZachary Turner     resolved_flags |= eSymbolContextFunction;
1342307f5ae8SZachary Turner   }
1343307f5ae8SZachary Turner 
1344307f5ae8SZachary Turner   if (resolve_scope & eSymbolContextLineEntry) {
1345307f5ae8SZachary Turner     lldbassert(sc.comp_unit);
1346307f5ae8SZachary Turner     if (auto *line_table = sc.comp_unit->GetLineTable()) {
1347307f5ae8SZachary Turner       if (line_table->FindLineEntryByAddress(addr, sc.line_entry))
1348307f5ae8SZachary Turner         resolved_flags |= eSymbolContextLineEntry;
1349307f5ae8SZachary Turner     }
1350307f5ae8SZachary Turner   }
1351307f5ae8SZachary Turner 
1352307f5ae8SZachary Turner   return resolved_flags;
1353307f5ae8SZachary Turner }
1354307f5ae8SZachary Turner 
1355307f5ae8SZachary Turner static void AppendLineEntryToSequence(LineTable &table, LineSequence &sequence,
1356307f5ae8SZachary Turner                                       const CompilandIndexItem &cci,
1357307f5ae8SZachary Turner                                       lldb::addr_t base_addr,
1358307f5ae8SZachary Turner                                       uint32_t file_number,
1359307f5ae8SZachary Turner                                       const LineFragmentHeader &block,
1360307f5ae8SZachary Turner                                       const LineNumberEntry &cur) {
1361307f5ae8SZachary Turner   LineInfo cur_info(cur.Flags);
1362307f5ae8SZachary Turner 
1363307f5ae8SZachary Turner   if (cur_info.isAlwaysStepInto() || cur_info.isNeverStepInto())
1364307f5ae8SZachary Turner     return;
1365307f5ae8SZachary Turner 
1366307f5ae8SZachary Turner   uint64_t addr = base_addr + cur.Offset;
1367307f5ae8SZachary Turner 
1368307f5ae8SZachary Turner   bool is_statement = cur_info.isStatement();
1369307f5ae8SZachary Turner   bool is_prologue = IsFunctionPrologue(cci, addr);
1370307f5ae8SZachary Turner   bool is_epilogue = IsFunctionEpilogue(cci, addr);
1371307f5ae8SZachary Turner 
1372307f5ae8SZachary Turner   uint32_t lno = cur_info.getStartLine();
1373307f5ae8SZachary Turner 
1374307f5ae8SZachary Turner   table.AppendLineEntryToSequence(&sequence, addr, lno, 0, file_number,
1375307f5ae8SZachary Turner                                   is_statement, false, is_prologue, is_epilogue,
1376307f5ae8SZachary Turner                                   false);
1377307f5ae8SZachary Turner }
1378307f5ae8SZachary Turner 
1379307f5ae8SZachary Turner static void TerminateLineSequence(LineTable &table,
1380307f5ae8SZachary Turner                                   const LineFragmentHeader &block,
1381307f5ae8SZachary Turner                                   lldb::addr_t base_addr, uint32_t file_number,
1382307f5ae8SZachary Turner                                   uint32_t last_line,
1383307f5ae8SZachary Turner                                   std::unique_ptr<LineSequence> seq) {
1384307f5ae8SZachary Turner   // The end is always a terminal entry, so insert it regardless.
1385307f5ae8SZachary Turner   table.AppendLineEntryToSequence(seq.get(), base_addr + block.CodeSize,
1386307f5ae8SZachary Turner                                   last_line, 0, file_number, false, false,
1387307f5ae8SZachary Turner                                   false, false, true);
1388307f5ae8SZachary Turner   table.InsertSequence(seq.release());
1389307f5ae8SZachary Turner }
1390307f5ae8SZachary Turner 
1391b96181c2SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitLineTable(const SymbolContext &sc) {
1392307f5ae8SZachary Turner   // Unfortunately LLDB is set up to parse the entire compile unit line table
1393307f5ae8SZachary Turner   // all at once, even if all it really needs is line info for a specific
1394307f5ae8SZachary Turner   // function.  In the future it would be nice if it could set the sc.m_function
1395307f5ae8SZachary Turner   // member, and we could only get the line info for the function in question.
1396307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
13976284aee9SZachary Turner   PdbSymUid cu_id(sc.comp_unit->GetID());
13986284aee9SZachary Turner   lldbassert(cu_id.kind() == PdbSymUidKind::Compiland);
13996284aee9SZachary Turner   CompilandIndexItem *cci =
14006284aee9SZachary Turner       m_index->compilands().GetCompiland(cu_id.asCompiland().modi);
1401307f5ae8SZachary Turner   lldbassert(cci);
1402307f5ae8SZachary Turner   auto line_table = llvm::make_unique<LineTable>(sc.comp_unit);
1403307f5ae8SZachary Turner 
1404307f5ae8SZachary Turner   // This is basically a copy of the .debug$S subsections from all original COFF
1405307f5ae8SZachary Turner   // object files merged together with address relocations applied.  We are
1406307f5ae8SZachary Turner   // looking for all DEBUG_S_LINES subsections.
1407307f5ae8SZachary Turner   for (const DebugSubsectionRecord &dssr :
1408307f5ae8SZachary Turner        cci->m_debug_stream.getSubsectionsArray()) {
1409307f5ae8SZachary Turner     if (dssr.kind() != DebugSubsectionKind::Lines)
1410307f5ae8SZachary Turner       continue;
1411307f5ae8SZachary Turner 
1412307f5ae8SZachary Turner     DebugLinesSubsectionRef lines;
1413307f5ae8SZachary Turner     llvm::BinaryStreamReader reader(dssr.getRecordData());
1414307f5ae8SZachary Turner     if (auto EC = lines.initialize(reader)) {
1415307f5ae8SZachary Turner       llvm::consumeError(std::move(EC));
1416307f5ae8SZachary Turner       return false;
1417307f5ae8SZachary Turner     }
1418307f5ae8SZachary Turner 
1419307f5ae8SZachary Turner     const LineFragmentHeader *lfh = lines.header();
1420307f5ae8SZachary Turner     uint64_t virtual_addr =
1421307f5ae8SZachary Turner         m_index->MakeVirtualAddress(lfh->RelocSegment, lfh->RelocOffset);
1422307f5ae8SZachary Turner 
1423307f5ae8SZachary Turner     const auto &checksums = cci->m_strings.checksums().getArray();
1424307f5ae8SZachary Turner     const auto &strings = cci->m_strings.strings();
1425307f5ae8SZachary Turner     for (const LineColumnEntry &group : lines) {
1426307f5ae8SZachary Turner       // Indices in this structure are actually offsets of records in the
1427307f5ae8SZachary Turner       // DEBUG_S_FILECHECKSUMS subsection.  Those entries then have an index
1428307f5ae8SZachary Turner       // into the global PDB string table.
1429307f5ae8SZachary Turner       auto iter = checksums.at(group.NameIndex);
1430307f5ae8SZachary Turner       if (iter == checksums.end())
1431307f5ae8SZachary Turner         continue;
1432307f5ae8SZachary Turner 
1433307f5ae8SZachary Turner       llvm::Expected<llvm::StringRef> efn =
1434307f5ae8SZachary Turner           strings.getString(iter->FileNameOffset);
1435307f5ae8SZachary Turner       if (!efn) {
1436307f5ae8SZachary Turner         llvm::consumeError(efn.takeError());
1437307f5ae8SZachary Turner         continue;
1438307f5ae8SZachary Turner       }
1439307f5ae8SZachary Turner 
1440307f5ae8SZachary Turner       // LLDB wants the index of the file in the list of support files.
1441307f5ae8SZachary Turner       auto fn_iter = llvm::find(cci->m_file_list, *efn);
1442307f5ae8SZachary Turner       lldbassert(fn_iter != cci->m_file_list.end());
1443307f5ae8SZachary Turner       uint32_t file_index = std::distance(cci->m_file_list.begin(), fn_iter);
1444307f5ae8SZachary Turner 
1445307f5ae8SZachary Turner       std::unique_ptr<LineSequence> sequence(
1446307f5ae8SZachary Turner           line_table->CreateLineSequenceContainer());
1447307f5ae8SZachary Turner       lldbassert(!group.LineNumbers.empty());
1448307f5ae8SZachary Turner 
1449307f5ae8SZachary Turner       for (const LineNumberEntry &entry : group.LineNumbers) {
1450307f5ae8SZachary Turner         AppendLineEntryToSequence(*line_table, *sequence, *cci, virtual_addr,
1451307f5ae8SZachary Turner                                   file_index, *lfh, entry);
1452307f5ae8SZachary Turner       }
1453307f5ae8SZachary Turner       LineInfo last_line(group.LineNumbers.back().Flags);
1454307f5ae8SZachary Turner       TerminateLineSequence(*line_table, *lfh, virtual_addr, file_index,
1455307f5ae8SZachary Turner                             last_line.getEndLine(), std::move(sequence));
1456307f5ae8SZachary Turner     }
1457307f5ae8SZachary Turner   }
1458307f5ae8SZachary Turner 
1459307f5ae8SZachary Turner   if (line_table->GetSize() == 0)
1460307f5ae8SZachary Turner     return false;
1461307f5ae8SZachary Turner 
1462307f5ae8SZachary Turner   sc.comp_unit->SetLineTable(line_table.release());
1463307f5ae8SZachary Turner   return true;
1464307f5ae8SZachary Turner }
1465307f5ae8SZachary Turner 
1466b96181c2SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitDebugMacros(const SymbolContext &sc) {
1467307f5ae8SZachary Turner   // PDB doesn't contain information about macros
1468307f5ae8SZachary Turner   return false;
1469307f5ae8SZachary Turner }
1470307f5ae8SZachary Turner 
1471307f5ae8SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitSupportFiles(
1472b96181c2SZachary Turner     const SymbolContext &sc, FileSpecList &support_files) {
1473307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
1474307f5ae8SZachary Turner 
14756284aee9SZachary Turner   PdbSymUid cu_id(sc.comp_unit->GetID());
14766284aee9SZachary Turner   lldbassert(cu_id.kind() == PdbSymUidKind::Compiland);
14776284aee9SZachary Turner   CompilandIndexItem *cci =
14786284aee9SZachary Turner       m_index->compilands().GetCompiland(cu_id.asCompiland().modi);
1479307f5ae8SZachary Turner   lldbassert(cci);
1480307f5ae8SZachary Turner 
1481307f5ae8SZachary Turner   for (llvm::StringRef f : cci->m_file_list) {
1482307f5ae8SZachary Turner     FileSpec::Style style =
1483307f5ae8SZachary Turner         f.startswith("/") ? FileSpec::Style::posix : FileSpec::Style::windows;
14848f3be7a3SJonas Devlieghere     FileSpec spec(f, style);
1485307f5ae8SZachary Turner     support_files.Append(spec);
1486307f5ae8SZachary Turner   }
1487307f5ae8SZachary Turner 
1488307f5ae8SZachary Turner   return true;
1489307f5ae8SZachary Turner }
1490307f5ae8SZachary Turner 
1491307f5ae8SZachary Turner bool SymbolFileNativePDB::ParseImportedModules(
1492b96181c2SZachary Turner     const SymbolContext &sc, std::vector<ConstString> &imported_modules) {
1493307f5ae8SZachary Turner   // PDB does not yet support module debug info
1494307f5ae8SZachary Turner   return false;
1495307f5ae8SZachary Turner }
1496307f5ae8SZachary Turner 
1497b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseFunctionBlocks(const SymbolContext &sc) {
1498307f5ae8SZachary Turner   lldbassert(sc.comp_unit && sc.function);
1499307f5ae8SZachary Turner   return 0;
1500307f5ae8SZachary Turner }
1501307f5ae8SZachary Turner 
15024911023fSZachary Turner void SymbolFileNativePDB::DumpClangAST(Stream &s) {
15034911023fSZachary Turner   if (!m_clang)
15044911023fSZachary Turner     return;
15054911023fSZachary Turner   m_clang->Dump(s);
15064911023fSZachary Turner }
15074911023fSZachary Turner 
15089f727950SZachary Turner uint32_t SymbolFileNativePDB::FindGlobalVariables(
15099f727950SZachary Turner     const ConstString &name, const CompilerDeclContext *parent_decl_ctx,
15109f727950SZachary Turner     uint32_t max_matches, VariableList &variables) {
15119f727950SZachary Turner   using SymbolAndOffset = std::pair<uint32_t, llvm::codeview::CVSymbol>;
15129f727950SZachary Turner 
15139f727950SZachary Turner   std::vector<SymbolAndOffset> results = m_index->globals().findRecordsByName(
15149f727950SZachary Turner       name.GetStringRef(), m_index->symrecords());
15159f727950SZachary Turner   for (const SymbolAndOffset &result : results) {
15169f727950SZachary Turner     VariableSP var;
15179f727950SZachary Turner     switch (result.second.kind()) {
15189f727950SZachary Turner     case SymbolKind::S_GDATA32:
15199f727950SZachary Turner     case SymbolKind::S_LDATA32:
15209f727950SZachary Turner     case SymbolKind::S_GTHREAD32:
15212af34166SZachary Turner     case SymbolKind::S_LTHREAD32:
15222af34166SZachary Turner     case SymbolKind::S_CONSTANT: {
15239fbf9350SZachary Turner       PdbGlobalSymId global(result.first, false);
15246284aee9SZachary Turner       var = GetOrCreateGlobalVariable(global);
15259f727950SZachary Turner       variables.AddVariable(var);
15269f727950SZachary Turner       break;
15279f727950SZachary Turner     }
15289f727950SZachary Turner     default:
15299f727950SZachary Turner       continue;
15309f727950SZachary Turner     }
15319f727950SZachary Turner   }
15329f727950SZachary Turner   return variables.GetSize();
15339f727950SZachary Turner }
15349f727950SZachary Turner 
1535307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::FindFunctions(
1536b96181c2SZachary Turner     const ConstString &name, const CompilerDeclContext *parent_decl_ctx,
1537117b1fa1SZachary Turner     FunctionNameType name_type_mask, bool include_inlines, bool append,
1538b96181c2SZachary Turner     SymbolContextList &sc_list) {
1539307f5ae8SZachary Turner   // For now we only support lookup by method name.
1540307f5ae8SZachary Turner   if (!(name_type_mask & eFunctionNameTypeMethod))
1541307f5ae8SZachary Turner     return 0;
1542307f5ae8SZachary Turner 
1543307f5ae8SZachary Turner   using SymbolAndOffset = std::pair<uint32_t, llvm::codeview::CVSymbol>;
1544307f5ae8SZachary Turner 
1545307f5ae8SZachary Turner   std::vector<SymbolAndOffset> matches = m_index->globals().findRecordsByName(
1546307f5ae8SZachary Turner       name.GetStringRef(), m_index->symrecords());
1547307f5ae8SZachary Turner   for (const SymbolAndOffset &match : matches) {
1548307f5ae8SZachary Turner     if (match.second.kind() != S_PROCREF && match.second.kind() != S_LPROCREF)
1549307f5ae8SZachary Turner       continue;
1550307f5ae8SZachary Turner     ProcRefSym proc(match.second.kind());
1551307f5ae8SZachary Turner     cantFail(SymbolDeserializer::deserializeAs<ProcRefSym>(match.second, proc));
1552307f5ae8SZachary Turner 
1553307f5ae8SZachary Turner     if (!IsValidRecord(proc))
1554307f5ae8SZachary Turner       continue;
1555307f5ae8SZachary Turner 
15566284aee9SZachary Turner     CompilandIndexItem &cci =
15576284aee9SZachary Turner         m_index->compilands().GetOrCreateCompiland(proc.modi());
1558b96181c2SZachary Turner     SymbolContext sc;
1559307f5ae8SZachary Turner 
1560307f5ae8SZachary Turner     sc.comp_unit = GetOrCreateCompileUnit(cci).get();
1561307f5ae8SZachary Turner     sc.module_sp = sc.comp_unit->GetModule();
15629fbf9350SZachary Turner     PdbCompilandSymId func_id(proc.modi(), proc.SymOffset);
15636284aee9SZachary Turner     sc.function = GetOrCreateFunction(func_id, sc).get();
1564307f5ae8SZachary Turner 
1565307f5ae8SZachary Turner     sc_list.Append(sc);
1566307f5ae8SZachary Turner   }
1567307f5ae8SZachary Turner 
1568307f5ae8SZachary Turner   return sc_list.GetSize();
1569307f5ae8SZachary Turner }
1570307f5ae8SZachary Turner 
1571b96181c2SZachary Turner uint32_t SymbolFileNativePDB::FindFunctions(const RegularExpression &regex,
1572307f5ae8SZachary Turner                                             bool include_inlines, bool append,
1573b96181c2SZachary Turner                                             SymbolContextList &sc_list) {
1574307f5ae8SZachary Turner   return 0;
1575307f5ae8SZachary Turner }
1576307f5ae8SZachary Turner 
1577b96181c2SZachary Turner uint32_t SymbolFileNativePDB::FindTypes(
1578b96181c2SZachary Turner     const SymbolContext &sc, const ConstString &name,
1579b96181c2SZachary Turner     const CompilerDeclContext *parent_decl_ctx, bool append,
1580b96181c2SZachary Turner     uint32_t max_matches, llvm::DenseSet<SymbolFile *> &searched_symbol_files,
1581b96181c2SZachary Turner     TypeMap &types) {
15822f7efbc9SZachary Turner   if (!append)
15832f7efbc9SZachary Turner     types.Clear();
15842f7efbc9SZachary Turner   if (!name)
1585b96181c2SZachary Turner     return 0;
15862f7efbc9SZachary Turner 
15872f7efbc9SZachary Turner   searched_symbol_files.clear();
15882f7efbc9SZachary Turner   searched_symbol_files.insert(this);
15892f7efbc9SZachary Turner 
15902f7efbc9SZachary Turner   // There is an assumption 'name' is not a regex
15912f7efbc9SZachary Turner   size_t match_count = FindTypesByName(name.GetStringRef(), max_matches, types);
15922f7efbc9SZachary Turner 
15932f7efbc9SZachary Turner   return match_count;
1594b96181c2SZachary Turner }
1595b96181c2SZachary Turner 
1596b96181c2SZachary Turner size_t
1597b96181c2SZachary Turner SymbolFileNativePDB::FindTypes(const std::vector<CompilerContext> &context,
1598b96181c2SZachary Turner                                bool append, TypeMap &types) {
1599b96181c2SZachary Turner   return 0;
1600b96181c2SZachary Turner }
1601b96181c2SZachary Turner 
16022f7efbc9SZachary Turner size_t SymbolFileNativePDB::FindTypesByName(llvm::StringRef name,
16032f7efbc9SZachary Turner                                             uint32_t max_matches,
16042f7efbc9SZachary Turner                                             TypeMap &types) {
16052f7efbc9SZachary Turner 
16062f7efbc9SZachary Turner   size_t match_count = 0;
16072f7efbc9SZachary Turner   std::vector<TypeIndex> matches = m_index->tpi().findRecordsByName(name);
16082f7efbc9SZachary Turner   if (max_matches > 0 && max_matches < matches.size())
16092f7efbc9SZachary Turner     matches.resize(max_matches);
16102f7efbc9SZachary Turner 
16112f7efbc9SZachary Turner   for (TypeIndex ti : matches) {
16122f7efbc9SZachary Turner     TypeSP type = GetOrCreateType(ti);
16132f7efbc9SZachary Turner     if (!type)
16142f7efbc9SZachary Turner       continue;
16152f7efbc9SZachary Turner 
16162f7efbc9SZachary Turner     types.Insert(type);
16172f7efbc9SZachary Turner     ++match_count;
16182f7efbc9SZachary Turner   }
16192f7efbc9SZachary Turner   return match_count;
16202f7efbc9SZachary Turner }
16212f7efbc9SZachary Turner 
1622b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseTypes(const SymbolContext &sc) { return 0; }
1623b96181c2SZachary Turner 
1624*a42bbe39SZachary Turner CompilerDeclContext
1625*a42bbe39SZachary Turner SymbolFileNativePDB::GetDeclContextContainingUID(lldb::user_id_t uid) {
1626*a42bbe39SZachary Turner   // FIXME: This should look up the uid, decide if it's a symbol or a type, and
1627*a42bbe39SZachary Turner   // depending which it is, find the appropriate DeclContext.  Possibilities:
1628*a42bbe39SZachary Turner   // For classes and typedefs:
1629*a42bbe39SZachary Turner   //   * Function
1630*a42bbe39SZachary Turner   //   * Namespace
1631*a42bbe39SZachary Turner   //   * Global
1632*a42bbe39SZachary Turner   //   * Block
1633*a42bbe39SZachary Turner   //   * Class
1634*a42bbe39SZachary Turner   // For field list members:
1635*a42bbe39SZachary Turner   //   * Class
1636*a42bbe39SZachary Turner   // For variables:
1637*a42bbe39SZachary Turner   //   * Function
1638*a42bbe39SZachary Turner   //   * Namespace
1639*a42bbe39SZachary Turner   //   * Global
1640*a42bbe39SZachary Turner   //   * Block
1641*a42bbe39SZachary Turner   // For functions:
1642*a42bbe39SZachary Turner   //   * Namespace
1643*a42bbe39SZachary Turner   //   * Global
1644*a42bbe39SZachary Turner   //   * Class
1645*a42bbe39SZachary Turner   //
1646*a42bbe39SZachary Turner   // It is an error to call this function with a uid for any other symbol type.
1647*a42bbe39SZachary Turner   return {m_clang, m_clang->GetTranslationUnitDecl()};
1648*a42bbe39SZachary Turner }
1649*a42bbe39SZachary Turner 
1650b96181c2SZachary Turner Type *SymbolFileNativePDB::ResolveTypeUID(lldb::user_id_t type_uid) {
16512f7efbc9SZachary Turner   auto iter = m_types.find(type_uid);
16522f7efbc9SZachary Turner   // lldb should not be passing us non-sensical type uids.  the only way it
16532f7efbc9SZachary Turner   // could have a type uid in the first place is if we handed it out, in which
16549f727950SZachary Turner   // case we should know about the type.  However, that doesn't mean we've
16559f727950SZachary Turner   // instantiated it yet.  We can vend out a UID for a future type.  So if the
16569f727950SZachary Turner   // type doesn't exist, let's instantiate it now.
16579f727950SZachary Turner   if (iter != m_types.end())
16582f7efbc9SZachary Turner     return &*iter->second;
16599f727950SZachary Turner 
16606284aee9SZachary Turner   PdbSymUid uid(type_uid);
16616284aee9SZachary Turner   lldbassert(uid.kind() == PdbSymUidKind::Type);
16626284aee9SZachary Turner   PdbTypeSymId type_id = uid.asTypeSym();
16636284aee9SZachary Turner   if (type_id.index.isNoneType())
16649f727950SZachary Turner     return nullptr;
16659f727950SZachary Turner 
16666284aee9SZachary Turner   TypeSP type_sp = CreateAndCacheType(type_id);
16679f727950SZachary Turner   return &*type_sp;
1668b96181c2SZachary Turner }
1669b96181c2SZachary Turner 
1670eca07c59SAdrian Prantl llvm::Optional<SymbolFile::ArrayInfo>
1671eca07c59SAdrian Prantl SymbolFileNativePDB::GetDynamicArrayInfoForUID(
1672eca07c59SAdrian Prantl     lldb::user_id_t type_uid, const lldb_private::ExecutionContext *exe_ctx) {
1673eca07c59SAdrian Prantl   return llvm::None;
1674eca07c59SAdrian Prantl }
1675eca07c59SAdrian Prantl 
1676eca07c59SAdrian Prantl 
1677b96181c2SZachary Turner bool SymbolFileNativePDB::CompleteType(CompilerType &compiler_type) {
16782f7efbc9SZachary Turner   // If this is not in our map, it's an error.
16792f7efbc9SZachary Turner   clang::TagDecl *tag_decl = m_clang->GetAsTagDecl(compiler_type);
16802f7efbc9SZachary Turner   lldbassert(tag_decl);
16812f7efbc9SZachary Turner   auto status_iter = m_decl_to_status.find(tag_decl);
16822f7efbc9SZachary Turner   lldbassert(status_iter != m_decl_to_status.end());
16832f7efbc9SZachary Turner 
16842f7efbc9SZachary Turner   // If it's already complete, just return.
16852f7efbc9SZachary Turner   DeclStatus &status = status_iter->second;
16862f7efbc9SZachary Turner   if (status.status == Type::eResolveStateFull)
16872f7efbc9SZachary Turner     return true;
16882f7efbc9SZachary Turner 
16896284aee9SZachary Turner   PdbTypeSymId type_id = PdbSymUid(status.uid).asTypeSym();
16902f7efbc9SZachary Turner 
16916284aee9SZachary Turner   lldbassert(IsTagRecord(type_id, m_index->tpi()));
16922f7efbc9SZachary Turner 
16932f7efbc9SZachary Turner   ClangASTContext::SetHasExternalStorage(compiler_type.GetOpaqueQualType(),
16942f7efbc9SZachary Turner                                          false);
16952f7efbc9SZachary Turner 
16962f7efbc9SZachary Turner   // In CreateAndCacheType, we already go out of our way to resolve forward
16972f7efbc9SZachary Turner   // ref UDTs to full decls, and the uids we vend out always refer to full
16982f7efbc9SZachary Turner   // decls if a full decl exists in the debug info.  So if we don't have a full
16992f7efbc9SZachary Turner   // decl here, it means one doesn't exist in the debug info, and we can't
17002f7efbc9SZachary Turner   // complete the type.
17012f7efbc9SZachary Turner   CVType cvt = m_index->tpi().getType(TypeIndex(type_id.index));
17022f7efbc9SZachary Turner   if (IsForwardRefUdt(cvt))
17032f7efbc9SZachary Turner     return false;
17042f7efbc9SZachary Turner 
17056284aee9SZachary Turner   auto types_iter = m_types.find(status.uid);
17062f7efbc9SZachary Turner   lldbassert(types_iter != m_types.end());
17072f7efbc9SZachary Turner 
1708511bff21SZachary Turner   if (cvt.kind() == LF_MODIFIER) {
1709511bff21SZachary Turner     TypeIndex unmodified_type = LookThroughModifierRecord(cvt);
1710511bff21SZachary Turner     cvt = m_index->tpi().getType(unmodified_type);
1711511bff21SZachary Turner     // LF_MODIFIERS usually point to forward decls, so this is the one case
1712511bff21SZachary Turner     // where we won't have been able to resolve a forward decl to a full decl
1713511bff21SZachary Turner     // earlier on.  So we need to do that now.
1714511bff21SZachary Turner     if (IsForwardRefUdt(cvt)) {
1715511bff21SZachary Turner       llvm::Expected<TypeIndex> expected_full_ti =
1716511bff21SZachary Turner           m_index->tpi().findFullDeclForForwardRef(unmodified_type);
1717511bff21SZachary Turner       if (!expected_full_ti) {
1718511bff21SZachary Turner         llvm::consumeError(expected_full_ti.takeError());
1719511bff21SZachary Turner         return false;
1720511bff21SZachary Turner       }
1721511bff21SZachary Turner       cvt = m_index->tpi().getType(*expected_full_ti);
1722511bff21SZachary Turner       lldbassert(!IsForwardRefUdt(cvt));
1723511bff21SZachary Turner       unmodified_type = *expected_full_ti;
1724511bff21SZachary Turner     }
17259fbf9350SZachary Turner     type_id = PdbTypeSymId(unmodified_type, false);
1726511bff21SZachary Turner   }
17272f7efbc9SZachary Turner   TypeIndex field_list_ti = GetFieldListIndex(cvt);
17282f7efbc9SZachary Turner   CVType field_list_cvt = m_index->tpi().getType(field_list_ti);
17292f7efbc9SZachary Turner   if (field_list_cvt.kind() != LF_FIELDLIST)
17302f7efbc9SZachary Turner     return false;
17312f7efbc9SZachary Turner 
17322f7efbc9SZachary Turner   // Visit all members of this class, then perform any finalization necessary
17332f7efbc9SZachary Turner   // to complete the class.
17346284aee9SZachary Turner   UdtRecordCompleter completer(type_id, compiler_type, *tag_decl, *this);
17352f7efbc9SZachary Turner   auto error =
17362f7efbc9SZachary Turner       llvm::codeview::visitMemberRecordStream(field_list_cvt.data(), completer);
17372f7efbc9SZachary Turner   completer.complete();
17382f7efbc9SZachary Turner 
17392f7efbc9SZachary Turner   status.status = Type::eResolveStateFull;
17402f7efbc9SZachary Turner   if (!error)
17412f7efbc9SZachary Turner     return true;
17422f7efbc9SZachary Turner 
17432f7efbc9SZachary Turner   llvm::consumeError(std::move(error));
1744b96181c2SZachary Turner   return false;
1745b96181c2SZachary Turner }
1746b96181c2SZachary Turner 
1747b96181c2SZachary Turner size_t SymbolFileNativePDB::GetTypes(lldb_private::SymbolContextScope *sc_scope,
1748117b1fa1SZachary Turner                                      TypeClass type_mask,
1749b96181c2SZachary Turner                                      lldb_private::TypeList &type_list) {
1750b96181c2SZachary Turner   return 0;
1751b96181c2SZachary Turner }
1752b96181c2SZachary Turner 
1753b96181c2SZachary Turner CompilerDeclContext
1754b96181c2SZachary Turner SymbolFileNativePDB::FindNamespace(const SymbolContext &sc,
1755b96181c2SZachary Turner                                    const ConstString &name,
1756b96181c2SZachary Turner                                    const CompilerDeclContext *parent_decl_ctx) {
1757307f5ae8SZachary Turner   return {};
1758307f5ae8SZachary Turner }
1759307f5ae8SZachary Turner 
1760b96181c2SZachary Turner TypeSystem *
1761307f5ae8SZachary Turner SymbolFileNativePDB::GetTypeSystemForLanguage(lldb::LanguageType language) {
1762307f5ae8SZachary Turner   auto type_system =
1763307f5ae8SZachary Turner       m_obj_file->GetModule()->GetTypeSystemForLanguage(language);
1764307f5ae8SZachary Turner   if (type_system)
1765307f5ae8SZachary Turner     type_system->SetSymbolFile(this);
1766307f5ae8SZachary Turner   return type_system;
1767307f5ae8SZachary Turner }
1768307f5ae8SZachary Turner 
1769b96181c2SZachary Turner ConstString SymbolFileNativePDB::GetPluginName() {
1770307f5ae8SZachary Turner   static ConstString g_name("pdb");
1771307f5ae8SZachary Turner   return g_name;
1772307f5ae8SZachary Turner }
1773307f5ae8SZachary Turner 
1774307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::GetPluginVersion() { return 1; }
1775