1307f5ae8SZachary Turner //===-- SymbolFileNativePDB.cpp ---------------------------------*- C++ -*-===//
2307f5ae8SZachary Turner //
3307f5ae8SZachary Turner //                     The LLVM Compiler Infrastructure
4307f5ae8SZachary Turner //
5307f5ae8SZachary Turner // This file is distributed under the University of Illinois Open Source
6307f5ae8SZachary Turner // License. See LICENSE.TXT for details.
7307f5ae8SZachary Turner //
8307f5ae8SZachary Turner //===----------------------------------------------------------------------===//
9307f5ae8SZachary Turner 
10307f5ae8SZachary Turner #include "SymbolFileNativePDB.h"
11307f5ae8SZachary Turner 
122f7efbc9SZachary Turner #include "clang/AST/Attr.h"
132f7efbc9SZachary Turner #include "clang/AST/CharUnits.h"
142f7efbc9SZachary Turner #include "clang/AST/Decl.h"
152f7efbc9SZachary Turner #include "clang/AST/DeclCXX.h"
16056e4ab4SZachary Turner #include "clang/AST/Type.h"
172f7efbc9SZachary Turner 
18307f5ae8SZachary Turner #include "lldb/Core/Module.h"
19307f5ae8SZachary Turner #include "lldb/Core/PluginManager.h"
209f727950SZachary Turner #include "lldb/Core/StreamBuffer.h"
21056e4ab4SZachary Turner #include "lldb/Core/StreamFile.h"
222f7efbc9SZachary Turner #include "lldb/Symbol/ClangASTContext.h"
232f7efbc9SZachary Turner #include "lldb/Symbol/ClangASTImporter.h"
242f7efbc9SZachary Turner #include "lldb/Symbol/ClangExternalASTSourceCommon.h"
25056e4ab4SZachary Turner #include "lldb/Symbol/ClangUtil.h"
26307f5ae8SZachary Turner #include "lldb/Symbol/CompileUnit.h"
27307f5ae8SZachary Turner #include "lldb/Symbol/LineTable.h"
28307f5ae8SZachary Turner #include "lldb/Symbol/ObjectFile.h"
29307f5ae8SZachary Turner #include "lldb/Symbol/SymbolContext.h"
30307f5ae8SZachary Turner #include "lldb/Symbol/SymbolVendor.h"
319f727950SZachary Turner #include "lldb/Symbol/Variable.h"
329f727950SZachary Turner #include "lldb/Symbol/VariableList.h"
33307f5ae8SZachary Turner 
34307f5ae8SZachary Turner #include "llvm/DebugInfo/CodeView/CVRecord.h"
352f7efbc9SZachary Turner #include "llvm/DebugInfo/CodeView/CVTypeVisitor.h"
36307f5ae8SZachary Turner #include "llvm/DebugInfo/CodeView/DebugLinesSubsection.h"
372f7efbc9SZachary Turner #include "llvm/DebugInfo/CodeView/LazyRandomTypeCollection.h"
38307f5ae8SZachary Turner #include "llvm/DebugInfo/CodeView/RecordName.h"
39307f5ae8SZachary Turner #include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
402f7efbc9SZachary Turner #include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
41307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/DbiStream.h"
42307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/GlobalsStream.h"
43307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/InfoStream.h"
44307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/ModuleDebugStream.h"
45307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/PDBFile.h"
46307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/Native/SymbolStream.h"
472f7efbc9SZachary Turner #include "llvm/DebugInfo/PDB/Native/TpiStream.h"
48307f5ae8SZachary Turner #include "llvm/DebugInfo/PDB/PDBTypes.h"
49056e4ab4SZachary Turner #include "llvm/Demangle/MicrosoftDemangle.h"
50307f5ae8SZachary Turner #include "llvm/Object/COFF.h"
51307f5ae8SZachary Turner #include "llvm/Support/Allocator.h"
52307f5ae8SZachary Turner #include "llvm/Support/BinaryStreamReader.h"
53056e4ab4SZachary Turner #include "llvm/Support/Error.h"
54307f5ae8SZachary Turner #include "llvm/Support/ErrorOr.h"
55307f5ae8SZachary Turner #include "llvm/Support/MemoryBuffer.h"
56307f5ae8SZachary Turner 
57307f5ae8SZachary Turner #include "PdbSymUid.h"
58307f5ae8SZachary Turner #include "PdbUtil.h"
592f7efbc9SZachary Turner #include "UdtRecordCompleter.h"
60307f5ae8SZachary Turner 
61307f5ae8SZachary Turner using namespace lldb;
62307f5ae8SZachary Turner using namespace lldb_private;
632f7efbc9SZachary Turner using namespace npdb;
64307f5ae8SZachary Turner using namespace llvm::codeview;
65307f5ae8SZachary Turner using namespace llvm::pdb;
66307f5ae8SZachary Turner 
67307f5ae8SZachary Turner static lldb::LanguageType TranslateLanguage(PDB_Lang lang) {
68307f5ae8SZachary Turner   switch (lang) {
69307f5ae8SZachary Turner   case PDB_Lang::Cpp:
70307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeC_plus_plus;
71307f5ae8SZachary Turner   case PDB_Lang::C:
72307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeC;
73307f5ae8SZachary Turner   default:
74307f5ae8SZachary Turner     return lldb::LanguageType::eLanguageTypeUnknown;
75307f5ae8SZachary Turner   }
76307f5ae8SZachary Turner }
77307f5ae8SZachary Turner 
78307f5ae8SZachary Turner static std::unique_ptr<PDBFile> loadPDBFile(std::string PdbPath,
79307f5ae8SZachary Turner                                             llvm::BumpPtrAllocator &Allocator) {
80307f5ae8SZachary Turner   llvm::ErrorOr<std::unique_ptr<llvm::MemoryBuffer>> ErrorOrBuffer =
81307f5ae8SZachary Turner       llvm::MemoryBuffer::getFile(PdbPath, /*FileSize=*/-1,
82307f5ae8SZachary Turner                                   /*RequiresNullTerminator=*/false);
83307f5ae8SZachary Turner   if (!ErrorOrBuffer)
84307f5ae8SZachary Turner     return nullptr;
85307f5ae8SZachary Turner   std::unique_ptr<llvm::MemoryBuffer> Buffer = std::move(*ErrorOrBuffer);
86307f5ae8SZachary Turner 
87307f5ae8SZachary Turner   llvm::StringRef Path = Buffer->getBufferIdentifier();
88307f5ae8SZachary Turner   auto Stream = llvm::make_unique<llvm::MemoryBufferByteStream>(
89307f5ae8SZachary Turner       std::move(Buffer), llvm::support::little);
90307f5ae8SZachary Turner 
91307f5ae8SZachary Turner   auto File = llvm::make_unique<PDBFile>(Path, std::move(Stream), Allocator);
928040eea9SZachary Turner   if (auto EC = File->parseFileHeaders()) {
938040eea9SZachary Turner     llvm::consumeError(std::move(EC));
94307f5ae8SZachary Turner     return nullptr;
958040eea9SZachary Turner   }
968040eea9SZachary Turner   if (auto EC = File->parseStreamData()) {
978040eea9SZachary Turner     llvm::consumeError(std::move(EC));
98307f5ae8SZachary Turner     return nullptr;
998040eea9SZachary Turner   }
100307f5ae8SZachary Turner 
101307f5ae8SZachary Turner   return File;
102307f5ae8SZachary Turner }
103307f5ae8SZachary Turner 
104307f5ae8SZachary Turner static std::unique_ptr<PDBFile>
105307f5ae8SZachary Turner loadMatchingPDBFile(std::string exe_path, llvm::BumpPtrAllocator &allocator) {
106307f5ae8SZachary Turner   // Try to find a matching PDB for an EXE.
107307f5ae8SZachary Turner   using namespace llvm::object;
108307f5ae8SZachary Turner   auto expected_binary = createBinary(exe_path);
109307f5ae8SZachary Turner 
110307f5ae8SZachary Turner   // If the file isn't a PE/COFF executable, fail.
111307f5ae8SZachary Turner   if (!expected_binary) {
112307f5ae8SZachary Turner     llvm::consumeError(expected_binary.takeError());
113307f5ae8SZachary Turner     return nullptr;
114307f5ae8SZachary Turner   }
115307f5ae8SZachary Turner   OwningBinary<Binary> binary = std::move(*expected_binary);
116307f5ae8SZachary Turner 
117307f5ae8SZachary Turner   auto *obj = llvm::dyn_cast<llvm::object::COFFObjectFile>(binary.getBinary());
118307f5ae8SZachary Turner   if (!obj)
119307f5ae8SZachary Turner     return nullptr;
120307f5ae8SZachary Turner   const llvm::codeview::DebugInfo *pdb_info = nullptr;
121307f5ae8SZachary Turner 
122307f5ae8SZachary Turner   // If it doesn't have a debug directory, fail.
123307f5ae8SZachary Turner   llvm::StringRef pdb_file;
124307f5ae8SZachary Turner   auto ec = obj->getDebugPDBInfo(pdb_info, pdb_file);
125307f5ae8SZachary Turner   if (ec)
126307f5ae8SZachary Turner     return nullptr;
127307f5ae8SZachary Turner 
128307f5ae8SZachary Turner   // if the file doesn't exist, is not a pdb, or doesn't have a matching guid,
129307f5ae8SZachary Turner   // fail.
130307f5ae8SZachary Turner   llvm::file_magic magic;
131307f5ae8SZachary Turner   ec = llvm::identify_magic(pdb_file, magic);
132307f5ae8SZachary Turner   if (ec || magic != llvm::file_magic::pdb)
133307f5ae8SZachary Turner     return nullptr;
134307f5ae8SZachary Turner   std::unique_ptr<PDBFile> pdb = loadPDBFile(pdb_file, allocator);
1358040eea9SZachary Turner   if (!pdb)
1368040eea9SZachary Turner     return nullptr;
1378040eea9SZachary Turner 
138307f5ae8SZachary Turner   auto expected_info = pdb->getPDBInfoStream();
139307f5ae8SZachary Turner   if (!expected_info) {
140307f5ae8SZachary Turner     llvm::consumeError(expected_info.takeError());
141307f5ae8SZachary Turner     return nullptr;
142307f5ae8SZachary Turner   }
143307f5ae8SZachary Turner   llvm::codeview::GUID guid;
144307f5ae8SZachary Turner   memcpy(&guid, pdb_info->PDB70.Signature, 16);
145307f5ae8SZachary Turner 
146307f5ae8SZachary Turner   if (expected_info->getGuid() != guid)
147307f5ae8SZachary Turner     return nullptr;
148307f5ae8SZachary Turner   return pdb;
149307f5ae8SZachary Turner }
150307f5ae8SZachary Turner 
151307f5ae8SZachary Turner static bool IsFunctionPrologue(const CompilandIndexItem &cci,
152307f5ae8SZachary Turner                                lldb::addr_t addr) {
153307f5ae8SZachary Turner   // FIXME: Implement this.
154307f5ae8SZachary Turner   return false;
155307f5ae8SZachary Turner }
156307f5ae8SZachary Turner 
157307f5ae8SZachary Turner static bool IsFunctionEpilogue(const CompilandIndexItem &cci,
158307f5ae8SZachary Turner                                lldb::addr_t addr) {
159307f5ae8SZachary Turner   // FIXME: Implement this.
160307f5ae8SZachary Turner   return false;
161307f5ae8SZachary Turner }
162307f5ae8SZachary Turner 
1632f7efbc9SZachary Turner static clang::MSInheritanceAttr::Spelling
1642f7efbc9SZachary Turner GetMSInheritance(LazyRandomTypeCollection &tpi, const ClassRecord &record) {
1652f7efbc9SZachary Turner   if (record.DerivationList == TypeIndex::None())
1662f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance;
1672f7efbc9SZachary Turner 
1682f7efbc9SZachary Turner   CVType bases = tpi.getType(record.DerivationList);
1692f7efbc9SZachary Turner   ArgListRecord base_list;
1702f7efbc9SZachary Turner   cantFail(TypeDeserializer::deserializeAs<ArgListRecord>(bases, base_list));
1712f7efbc9SZachary Turner   if (base_list.ArgIndices.empty())
1722f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance;
1732f7efbc9SZachary Turner 
1742f7efbc9SZachary Turner   int base_count = 0;
1752f7efbc9SZachary Turner   for (TypeIndex ti : base_list.ArgIndices) {
1762f7efbc9SZachary Turner     CVType base = tpi.getType(ti);
1772f7efbc9SZachary Turner     if (base.kind() == LF_VBCLASS || base.kind() == LF_IVBCLASS)
1782f7efbc9SZachary Turner       return clang::MSInheritanceAttr::Spelling::Keyword_virtual_inheritance;
1792f7efbc9SZachary Turner     ++base_count;
1802f7efbc9SZachary Turner   }
1812f7efbc9SZachary Turner 
1822f7efbc9SZachary Turner   if (base_count > 1)
1832f7efbc9SZachary Turner     return clang::MSInheritanceAttr::Keyword_multiple_inheritance;
1842f7efbc9SZachary Turner   return clang::MSInheritanceAttr::Keyword_single_inheritance;
1852f7efbc9SZachary Turner }
1862f7efbc9SZachary Turner 
1872f7efbc9SZachary Turner static lldb::BasicType GetCompilerTypeForSimpleKind(SimpleTypeKind kind) {
1882f7efbc9SZachary Turner   switch (kind) {
1892f7efbc9SZachary Turner   case SimpleTypeKind::Boolean128:
1902f7efbc9SZachary Turner   case SimpleTypeKind::Boolean16:
1912f7efbc9SZachary Turner   case SimpleTypeKind::Boolean32:
1922f7efbc9SZachary Turner   case SimpleTypeKind::Boolean64:
1932f7efbc9SZachary Turner   case SimpleTypeKind::Boolean8:
1942f7efbc9SZachary Turner     return lldb::eBasicTypeBool;
1952f7efbc9SZachary Turner   case SimpleTypeKind::Byte:
1962f7efbc9SZachary Turner   case SimpleTypeKind::UnsignedCharacter:
1972f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedChar;
1982f7efbc9SZachary Turner   case SimpleTypeKind::NarrowCharacter:
1992f7efbc9SZachary Turner     return lldb::eBasicTypeChar;
2002f7efbc9SZachary Turner   case SimpleTypeKind::SignedCharacter:
2012f7efbc9SZachary Turner   case SimpleTypeKind::SByte:
2022f7efbc9SZachary Turner     return lldb::eBasicTypeSignedChar;
2032f7efbc9SZachary Turner   case SimpleTypeKind::Character16:
2042f7efbc9SZachary Turner     return lldb::eBasicTypeChar16;
2052f7efbc9SZachary Turner   case SimpleTypeKind::Character32:
2062f7efbc9SZachary Turner     return lldb::eBasicTypeChar32;
2072f7efbc9SZachary Turner   case SimpleTypeKind::Complex80:
2082f7efbc9SZachary Turner     return lldb::eBasicTypeLongDoubleComplex;
2092f7efbc9SZachary Turner   case SimpleTypeKind::Complex64:
2102f7efbc9SZachary Turner     return lldb::eBasicTypeDoubleComplex;
2112f7efbc9SZachary Turner   case SimpleTypeKind::Complex32:
2122f7efbc9SZachary Turner     return lldb::eBasicTypeFloatComplex;
2132f7efbc9SZachary Turner   case SimpleTypeKind::Float128:
2142f7efbc9SZachary Turner   case SimpleTypeKind::Float80:
2152f7efbc9SZachary Turner     return lldb::eBasicTypeLongDouble;
2162f7efbc9SZachary Turner   case SimpleTypeKind::Float64:
2172f7efbc9SZachary Turner     return lldb::eBasicTypeDouble;
2182f7efbc9SZachary Turner   case SimpleTypeKind::Float32:
2192f7efbc9SZachary Turner     return lldb::eBasicTypeFloat;
2202f7efbc9SZachary Turner   case SimpleTypeKind::Float16:
2212f7efbc9SZachary Turner     return lldb::eBasicTypeHalf;
2222f7efbc9SZachary Turner   case SimpleTypeKind::Int128:
2232f7efbc9SZachary Turner     return lldb::eBasicTypeInt128;
2242f7efbc9SZachary Turner   case SimpleTypeKind::Int64:
2252f7efbc9SZachary Turner   case SimpleTypeKind::Int64Quad:
2262f7efbc9SZachary Turner     return lldb::eBasicTypeLongLong;
2272f7efbc9SZachary Turner   case SimpleTypeKind::Int32:
2282f7efbc9SZachary Turner     return lldb::eBasicTypeInt;
2292f7efbc9SZachary Turner   case SimpleTypeKind::Int16:
2302f7efbc9SZachary Turner   case SimpleTypeKind::Int16Short:
2312f7efbc9SZachary Turner     return lldb::eBasicTypeShort;
2322f7efbc9SZachary Turner   case SimpleTypeKind::UInt128:
2332f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedInt128;
2342f7efbc9SZachary Turner   case SimpleTypeKind::UInt64:
2352f7efbc9SZachary Turner   case SimpleTypeKind::UInt64Quad:
2362f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedLongLong;
2372f7efbc9SZachary Turner   case SimpleTypeKind::HResult:
2382f7efbc9SZachary Turner   case SimpleTypeKind::UInt32:
2392f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedInt;
2402f7efbc9SZachary Turner   case SimpleTypeKind::UInt16:
2412f7efbc9SZachary Turner   case SimpleTypeKind::UInt16Short:
2422f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedShort;
2432f7efbc9SZachary Turner   case SimpleTypeKind::Int32Long:
2442f7efbc9SZachary Turner     return lldb::eBasicTypeLong;
2452f7efbc9SZachary Turner   case SimpleTypeKind::UInt32Long:
2462f7efbc9SZachary Turner     return lldb::eBasicTypeUnsignedLong;
2472f7efbc9SZachary Turner   case SimpleTypeKind::Void:
2482f7efbc9SZachary Turner     return lldb::eBasicTypeVoid;
2492f7efbc9SZachary Turner   case SimpleTypeKind::WideCharacter:
2502f7efbc9SZachary Turner     return lldb::eBasicTypeWChar;
2512f7efbc9SZachary Turner   default:
2522f7efbc9SZachary Turner     return lldb::eBasicTypeInvalid;
2532f7efbc9SZachary Turner   }
2542f7efbc9SZachary Turner }
2552f7efbc9SZachary Turner 
2562af34166SZachary Turner static bool IsSimpleTypeSignedInteger(SimpleTypeKind kind) {
2572af34166SZachary Turner   switch (kind) {
2582af34166SZachary Turner   case SimpleTypeKind::Int128:
2592af34166SZachary Turner   case SimpleTypeKind::Int64:
2602af34166SZachary Turner   case SimpleTypeKind::Int64Quad:
2612af34166SZachary Turner   case SimpleTypeKind::Int32:
2622af34166SZachary Turner   case SimpleTypeKind::Int32Long:
2632af34166SZachary Turner   case SimpleTypeKind::Int16:
2642af34166SZachary Turner   case SimpleTypeKind::Int16Short:
2652af34166SZachary Turner   case SimpleTypeKind::Float128:
2662af34166SZachary Turner   case SimpleTypeKind::Float80:
2672af34166SZachary Turner   case SimpleTypeKind::Float64:
2682af34166SZachary Turner   case SimpleTypeKind::Float32:
2692af34166SZachary Turner   case SimpleTypeKind::Float16:
2702af34166SZachary Turner   case SimpleTypeKind::NarrowCharacter:
2712af34166SZachary Turner   case SimpleTypeKind::SignedCharacter:
2722af34166SZachary Turner   case SimpleTypeKind::SByte:
2732af34166SZachary Turner     return true;
2742af34166SZachary Turner   default:
2752af34166SZachary Turner     return false;
2762af34166SZachary Turner   }
2772af34166SZachary Turner }
2782af34166SZachary Turner 
2792f7efbc9SZachary Turner static size_t GetTypeSizeForSimpleKind(SimpleTypeKind kind) {
2802f7efbc9SZachary Turner   switch (kind) {
2812f7efbc9SZachary Turner   case SimpleTypeKind::Boolean128:
2822f7efbc9SZachary Turner   case SimpleTypeKind::Int128:
2832f7efbc9SZachary Turner   case SimpleTypeKind::UInt128:
2842f7efbc9SZachary Turner   case SimpleTypeKind::Float128:
2852f7efbc9SZachary Turner     return 16;
2862f7efbc9SZachary Turner   case SimpleTypeKind::Complex80:
2872f7efbc9SZachary Turner   case SimpleTypeKind::Float80:
2882f7efbc9SZachary Turner     return 10;
2892f7efbc9SZachary Turner   case SimpleTypeKind::Boolean64:
2902f7efbc9SZachary Turner   case SimpleTypeKind::Complex64:
2912f7efbc9SZachary Turner   case SimpleTypeKind::UInt64:
2922f7efbc9SZachary Turner   case SimpleTypeKind::UInt64Quad:
2932f7efbc9SZachary Turner   case SimpleTypeKind::Float64:
2942f7efbc9SZachary Turner   case SimpleTypeKind::Int64:
2952f7efbc9SZachary Turner   case SimpleTypeKind::Int64Quad:
2962f7efbc9SZachary Turner     return 8;
2972f7efbc9SZachary Turner   case SimpleTypeKind::Boolean32:
2982f7efbc9SZachary Turner   case SimpleTypeKind::Character32:
2992f7efbc9SZachary Turner   case SimpleTypeKind::Complex32:
3002f7efbc9SZachary Turner   case SimpleTypeKind::Float32:
3012f7efbc9SZachary Turner   case SimpleTypeKind::Int32:
3022f7efbc9SZachary Turner   case SimpleTypeKind::Int32Long:
3032f7efbc9SZachary Turner   case SimpleTypeKind::UInt32Long:
3042f7efbc9SZachary Turner   case SimpleTypeKind::HResult:
3052f7efbc9SZachary Turner   case SimpleTypeKind::UInt32:
3062f7efbc9SZachary Turner     return 4;
3072f7efbc9SZachary Turner   case SimpleTypeKind::Boolean16:
3082f7efbc9SZachary Turner   case SimpleTypeKind::Character16:
3092f7efbc9SZachary Turner   case SimpleTypeKind::Float16:
3102f7efbc9SZachary Turner   case SimpleTypeKind::Int16:
3112f7efbc9SZachary Turner   case SimpleTypeKind::Int16Short:
3122f7efbc9SZachary Turner   case SimpleTypeKind::UInt16:
3132f7efbc9SZachary Turner   case SimpleTypeKind::UInt16Short:
3142f7efbc9SZachary Turner   case SimpleTypeKind::WideCharacter:
3152f7efbc9SZachary Turner     return 2;
3162f7efbc9SZachary Turner   case SimpleTypeKind::Boolean8:
3172f7efbc9SZachary Turner   case SimpleTypeKind::Byte:
3182f7efbc9SZachary Turner   case SimpleTypeKind::UnsignedCharacter:
3192f7efbc9SZachary Turner   case SimpleTypeKind::NarrowCharacter:
3202f7efbc9SZachary Turner   case SimpleTypeKind::SignedCharacter:
3212f7efbc9SZachary Turner   case SimpleTypeKind::SByte:
3222f7efbc9SZachary Turner     return 1;
3232f7efbc9SZachary Turner   case SimpleTypeKind::Void:
3242f7efbc9SZachary Turner   default:
3252f7efbc9SZachary Turner     return 0;
3262f7efbc9SZachary Turner   }
3272f7efbc9SZachary Turner }
3282f7efbc9SZachary Turner 
3292af34166SZachary Turner std::pair<size_t, bool> GetIntegralTypeInfo(TypeIndex ti, TpiStream &tpi) {
3302af34166SZachary Turner   if (ti.isSimple()) {
3312af34166SZachary Turner     SimpleTypeKind stk = ti.getSimpleKind();
3322af34166SZachary Turner     return {GetTypeSizeForSimpleKind(stk), IsSimpleTypeSignedInteger(stk)};
3332af34166SZachary Turner   }
3342af34166SZachary Turner 
3352af34166SZachary Turner   CVType cvt = tpi.getType(ti);
3362af34166SZachary Turner   switch (cvt.kind()) {
3372af34166SZachary Turner   case LF_MODIFIER: {
3382af34166SZachary Turner     ModifierRecord mfr;
3392af34166SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ModifierRecord>(cvt, mfr));
3402af34166SZachary Turner     return GetIntegralTypeInfo(mfr.ModifiedType, tpi);
3412af34166SZachary Turner   }
3422af34166SZachary Turner   case LF_POINTER: {
3432af34166SZachary Turner     PointerRecord pr;
3442af34166SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<PointerRecord>(cvt, pr));
3452af34166SZachary Turner     return GetIntegralTypeInfo(pr.ReferentType, tpi);
3462af34166SZachary Turner   }
3472af34166SZachary Turner   case LF_ENUM: {
3482af34166SZachary Turner     EnumRecord er;
3492af34166SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<EnumRecord>(cvt, er));
3502af34166SZachary Turner     return GetIntegralTypeInfo(er.UnderlyingType, tpi);
3512af34166SZachary Turner   }
3522af34166SZachary Turner   default:
3532af34166SZachary Turner     assert(false && "Type is not integral!");
3542af34166SZachary Turner     return {0, false};
3552af34166SZachary Turner   }
3562af34166SZachary Turner }
3572af34166SZachary Turner 
3582f7efbc9SZachary Turner static llvm::StringRef GetSimpleTypeName(SimpleTypeKind kind) {
3592f7efbc9SZachary Turner   switch (kind) {
3602f7efbc9SZachary Turner   case SimpleTypeKind::Boolean128:
3612f7efbc9SZachary Turner   case SimpleTypeKind::Boolean16:
3622f7efbc9SZachary Turner   case SimpleTypeKind::Boolean32:
3632f7efbc9SZachary Turner   case SimpleTypeKind::Boolean64:
3642f7efbc9SZachary Turner   case SimpleTypeKind::Boolean8:
3652f7efbc9SZachary Turner     return "bool";
3662f7efbc9SZachary Turner   case SimpleTypeKind::Byte:
3672f7efbc9SZachary Turner   case SimpleTypeKind::UnsignedCharacter:
3682f7efbc9SZachary Turner     return "unsigned char";
3692f7efbc9SZachary Turner   case SimpleTypeKind::NarrowCharacter:
3702f7efbc9SZachary Turner     return "char";
3712f7efbc9SZachary Turner   case SimpleTypeKind::SignedCharacter:
3722f7efbc9SZachary Turner   case SimpleTypeKind::SByte:
37371ebb721SZachary Turner     return "signed char";
3742f7efbc9SZachary Turner   case SimpleTypeKind::Character16:
3752f7efbc9SZachary Turner     return "char16_t";
3762f7efbc9SZachary Turner   case SimpleTypeKind::Character32:
3772f7efbc9SZachary Turner     return "char32_t";
3782f7efbc9SZachary Turner   case SimpleTypeKind::Complex80:
3792f7efbc9SZachary Turner   case SimpleTypeKind::Complex64:
3802f7efbc9SZachary Turner   case SimpleTypeKind::Complex32:
3812f7efbc9SZachary Turner     return "complex";
3822f7efbc9SZachary Turner   case SimpleTypeKind::Float128:
3832f7efbc9SZachary Turner   case SimpleTypeKind::Float80:
3842f7efbc9SZachary Turner     return "long double";
3852f7efbc9SZachary Turner   case SimpleTypeKind::Float64:
3862f7efbc9SZachary Turner     return "double";
3872f7efbc9SZachary Turner   case SimpleTypeKind::Float32:
3882f7efbc9SZachary Turner     return "float";
3892f7efbc9SZachary Turner   case SimpleTypeKind::Float16:
3902f7efbc9SZachary Turner     return "single";
3912f7efbc9SZachary Turner   case SimpleTypeKind::Int128:
3922f7efbc9SZachary Turner     return "__int128";
3932f7efbc9SZachary Turner   case SimpleTypeKind::Int64:
3942f7efbc9SZachary Turner   case SimpleTypeKind::Int64Quad:
39571ebb721SZachary Turner     return "int64_t";
3962f7efbc9SZachary Turner   case SimpleTypeKind::Int32:
3972f7efbc9SZachary Turner     return "int";
3982f7efbc9SZachary Turner   case SimpleTypeKind::Int16:
3992f7efbc9SZachary Turner     return "short";
4002f7efbc9SZachary Turner   case SimpleTypeKind::UInt128:
4012f7efbc9SZachary Turner     return "unsigned __int128";
4022f7efbc9SZachary Turner   case SimpleTypeKind::UInt64:
4032f7efbc9SZachary Turner   case SimpleTypeKind::UInt64Quad:
40471ebb721SZachary Turner     return "uint64_t";
4052f7efbc9SZachary Turner   case SimpleTypeKind::HResult:
4062f7efbc9SZachary Turner     return "HRESULT";
4072f7efbc9SZachary Turner   case SimpleTypeKind::UInt32:
4082f7efbc9SZachary Turner     return "unsigned";
4092f7efbc9SZachary Turner   case SimpleTypeKind::UInt16:
4102f7efbc9SZachary Turner   case SimpleTypeKind::UInt16Short:
4112f7efbc9SZachary Turner     return "unsigned short";
4122f7efbc9SZachary Turner   case SimpleTypeKind::Int32Long:
4132f7efbc9SZachary Turner     return "long";
4142f7efbc9SZachary Turner   case SimpleTypeKind::UInt32Long:
4152f7efbc9SZachary Turner     return "unsigned long";
4162f7efbc9SZachary Turner   case SimpleTypeKind::Void:
4172f7efbc9SZachary Turner     return "void";
4182f7efbc9SZachary Turner   case SimpleTypeKind::WideCharacter:
4192f7efbc9SZachary Turner     return "wchar_t";
4202f7efbc9SZachary Turner   default:
4212f7efbc9SZachary Turner     return "";
4222f7efbc9SZachary Turner   }
4232f7efbc9SZachary Turner }
4242f7efbc9SZachary Turner 
4252f7efbc9SZachary Turner static bool IsClassRecord(TypeLeafKind kind) {
4262f7efbc9SZachary Turner   switch (kind) {
4272f7efbc9SZachary Turner   case LF_STRUCTURE:
4282f7efbc9SZachary Turner   case LF_CLASS:
4292f7efbc9SZachary Turner   case LF_INTERFACE:
4302f7efbc9SZachary Turner     return true;
4312f7efbc9SZachary Turner   default:
4322f7efbc9SZachary Turner     return false;
4332f7efbc9SZachary Turner   }
4342f7efbc9SZachary Turner }
4352f7efbc9SZachary Turner 
4362f7efbc9SZachary Turner static PDB_SymType GetPdbSymType(TpiStream &tpi, TypeIndex ti) {
4372f7efbc9SZachary Turner   if (ti.isSimple()) {
4382f7efbc9SZachary Turner     if (ti.getSimpleMode() == SimpleTypeMode::Direct)
4392f7efbc9SZachary Turner       return PDB_SymType::BuiltinType;
4402f7efbc9SZachary Turner     return PDB_SymType::PointerType;
4412f7efbc9SZachary Turner   }
4422f7efbc9SZachary Turner 
4432f7efbc9SZachary Turner   CVType cvt = tpi.getType(ti);
4442f7efbc9SZachary Turner   TypeLeafKind kind = cvt.kind();
4452f7efbc9SZachary Turner   if (kind != LF_MODIFIER)
4462f7efbc9SZachary Turner     return CVTypeToPDBType(kind);
4472f7efbc9SZachary Turner 
4482f7efbc9SZachary Turner   // If this is an LF_MODIFIER, look through it to get the kind that it
4492f7efbc9SZachary Turner   // modifies.  Note that it's not possible to have an LF_MODIFIER that
4502f7efbc9SZachary Turner   // modifies another LF_MODIFIER, although this would handle that anyway.
451511bff21SZachary Turner   return GetPdbSymType(tpi, LookThroughModifierRecord(cvt));
4522f7efbc9SZachary Turner }
4532f7efbc9SZachary Turner 
454544a66d8SZachary Turner static bool IsCVarArgsFunction(llvm::ArrayRef<TypeIndex> args) {
455544a66d8SZachary Turner   if (args.empty())
456544a66d8SZachary Turner     return false;
457544a66d8SZachary Turner   return args.back() == TypeIndex::None();
458544a66d8SZachary Turner }
459544a66d8SZachary Turner 
4602f7efbc9SZachary Turner static clang::TagTypeKind TranslateUdtKind(const TagRecord &cr) {
4612f7efbc9SZachary Turner   switch (cr.Kind) {
4622f7efbc9SZachary Turner   case TypeRecordKind::Class:
4632f7efbc9SZachary Turner     return clang::TTK_Class;
4642f7efbc9SZachary Turner   case TypeRecordKind::Struct:
4652f7efbc9SZachary Turner     return clang::TTK_Struct;
4662f7efbc9SZachary Turner   case TypeRecordKind::Union:
4672f7efbc9SZachary Turner     return clang::TTK_Union;
4682f7efbc9SZachary Turner   case TypeRecordKind::Interface:
4692f7efbc9SZachary Turner     return clang::TTK_Interface;
4702f7efbc9SZachary Turner   case TypeRecordKind::Enum:
4712f7efbc9SZachary Turner     return clang::TTK_Enum;
4722f7efbc9SZachary Turner   default:
4732f7efbc9SZachary Turner     lldbassert(false && "Invalid tag record kind!");
4742f7efbc9SZachary Turner     return clang::TTK_Struct;
4752f7efbc9SZachary Turner   }
4762f7efbc9SZachary Turner }
4772f7efbc9SZachary Turner 
478544a66d8SZachary Turner static llvm::Optional<clang::CallingConv>
479544a66d8SZachary Turner TranslateCallingConvention(llvm::codeview::CallingConvention conv) {
480544a66d8SZachary Turner   using CC = llvm::codeview::CallingConvention;
481544a66d8SZachary Turner   switch (conv) {
482544a66d8SZachary Turner 
483544a66d8SZachary Turner   case CC::NearC:
484544a66d8SZachary Turner   case CC::FarC:
485544a66d8SZachary Turner     return clang::CallingConv::CC_C;
486544a66d8SZachary Turner   case CC::NearPascal:
487544a66d8SZachary Turner   case CC::FarPascal:
488544a66d8SZachary Turner     return clang::CallingConv::CC_X86Pascal;
489544a66d8SZachary Turner   case CC::NearFast:
490544a66d8SZachary Turner   case CC::FarFast:
491544a66d8SZachary Turner     return clang::CallingConv::CC_X86FastCall;
492544a66d8SZachary Turner   case CC::NearStdCall:
493544a66d8SZachary Turner   case CC::FarStdCall:
494544a66d8SZachary Turner     return clang::CallingConv::CC_X86StdCall;
495544a66d8SZachary Turner   case CC::ThisCall:
496544a66d8SZachary Turner     return clang::CallingConv::CC_X86ThisCall;
497544a66d8SZachary Turner   case CC::NearVector:
498544a66d8SZachary Turner     return clang::CallingConv::CC_X86VectorCall;
499544a66d8SZachary Turner   default:
500544a66d8SZachary Turner     return llvm::None;
501544a66d8SZachary Turner   }
502544a66d8SZachary Turner }
503544a66d8SZachary Turner 
504307f5ae8SZachary Turner void SymbolFileNativePDB::Initialize() {
505307f5ae8SZachary Turner   PluginManager::RegisterPlugin(GetPluginNameStatic(),
506307f5ae8SZachary Turner                                 GetPluginDescriptionStatic(), CreateInstance,
507307f5ae8SZachary Turner                                 DebuggerInitialize);
508307f5ae8SZachary Turner }
509307f5ae8SZachary Turner 
510307f5ae8SZachary Turner void SymbolFileNativePDB::Terminate() {
511307f5ae8SZachary Turner   PluginManager::UnregisterPlugin(CreateInstance);
512307f5ae8SZachary Turner }
513307f5ae8SZachary Turner 
514b96181c2SZachary Turner void SymbolFileNativePDB::DebuggerInitialize(Debugger &debugger) {}
515307f5ae8SZachary Turner 
516b96181c2SZachary Turner ConstString SymbolFileNativePDB::GetPluginNameStatic() {
517307f5ae8SZachary Turner   static ConstString g_name("native-pdb");
518307f5ae8SZachary Turner   return g_name;
519307f5ae8SZachary Turner }
520307f5ae8SZachary Turner 
521307f5ae8SZachary Turner const char *SymbolFileNativePDB::GetPluginDescriptionStatic() {
522307f5ae8SZachary Turner   return "Microsoft PDB debug symbol cross-platform file reader.";
523307f5ae8SZachary Turner }
524307f5ae8SZachary Turner 
525b96181c2SZachary Turner SymbolFile *SymbolFileNativePDB::CreateInstance(ObjectFile *obj_file) {
526307f5ae8SZachary Turner   return new SymbolFileNativePDB(obj_file);
527307f5ae8SZachary Turner }
528307f5ae8SZachary Turner 
529b96181c2SZachary Turner SymbolFileNativePDB::SymbolFileNativePDB(ObjectFile *object_file)
530307f5ae8SZachary Turner     : SymbolFile(object_file) {}
531307f5ae8SZachary Turner 
532307f5ae8SZachary Turner SymbolFileNativePDB::~SymbolFileNativePDB() {}
533307f5ae8SZachary Turner 
534307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::CalculateAbilities() {
535307f5ae8SZachary Turner   uint32_t abilities = 0;
536307f5ae8SZachary Turner   if (!m_obj_file)
537307f5ae8SZachary Turner     return 0;
538307f5ae8SZachary Turner 
539307f5ae8SZachary Turner   if (!m_index) {
540307f5ae8SZachary Turner     // Lazily load and match the PDB file, but only do this once.
541307f5ae8SZachary Turner     std::unique_ptr<PDBFile> file_up =
542307f5ae8SZachary Turner         loadMatchingPDBFile(m_obj_file->GetFileSpec().GetPath(), m_allocator);
543307f5ae8SZachary Turner 
544307f5ae8SZachary Turner     if (!file_up) {
545307f5ae8SZachary Turner       auto module_sp = m_obj_file->GetModule();
546307f5ae8SZachary Turner       if (!module_sp)
547307f5ae8SZachary Turner         return 0;
548307f5ae8SZachary Turner       // See if any symbol file is specified through `--symfile` option.
549307f5ae8SZachary Turner       FileSpec symfile = module_sp->GetSymbolFileFileSpec();
550307f5ae8SZachary Turner       if (!symfile)
551307f5ae8SZachary Turner         return 0;
552307f5ae8SZachary Turner       file_up = loadPDBFile(symfile.GetPath(), m_allocator);
553307f5ae8SZachary Turner     }
554307f5ae8SZachary Turner 
555307f5ae8SZachary Turner     if (!file_up)
556307f5ae8SZachary Turner       return 0;
557307f5ae8SZachary Turner 
558307f5ae8SZachary Turner     auto expected_index = PdbIndex::create(std::move(file_up));
559307f5ae8SZachary Turner     if (!expected_index) {
560307f5ae8SZachary Turner       llvm::consumeError(expected_index.takeError());
561307f5ae8SZachary Turner       return 0;
562307f5ae8SZachary Turner     }
563307f5ae8SZachary Turner     m_index = std::move(*expected_index);
564307f5ae8SZachary Turner   }
565307f5ae8SZachary Turner   if (!m_index)
566307f5ae8SZachary Turner     return 0;
567307f5ae8SZachary Turner 
568307f5ae8SZachary Turner   // We don't especially have to be precise here.  We only distinguish between
569307f5ae8SZachary Turner   // stripped and not stripped.
570307f5ae8SZachary Turner   abilities = kAllAbilities;
571307f5ae8SZachary Turner 
572307f5ae8SZachary Turner   if (m_index->dbi().isStripped())
573307f5ae8SZachary Turner     abilities &= ~(Blocks | LocalVariables);
574307f5ae8SZachary Turner   return abilities;
575307f5ae8SZachary Turner }
576307f5ae8SZachary Turner 
577307f5ae8SZachary Turner void SymbolFileNativePDB::InitializeObject() {
578307f5ae8SZachary Turner   m_obj_load_address = m_obj_file->GetFileOffset();
579307f5ae8SZachary Turner   m_index->SetLoadAddress(m_obj_load_address);
580307f5ae8SZachary Turner   m_index->ParseSectionContribs();
5812f7efbc9SZachary Turner 
5822f7efbc9SZachary Turner   TypeSystem *ts = GetTypeSystemForLanguage(eLanguageTypeC_plus_plus);
5832f7efbc9SZachary Turner   m_clang = llvm::dyn_cast_or_null<ClangASTContext>(ts);
5842f7efbc9SZachary Turner   m_importer = llvm::make_unique<ClangASTImporter>();
585056e4ab4SZachary Turner 
586056e4ab4SZachary Turner   PreprocessTpiStream();
5872f7efbc9SZachary Turner   lldbassert(m_clang);
588307f5ae8SZachary Turner }
589307f5ae8SZachary Turner 
59003a24052SZachary Turner static llvm::Optional<CVTagRecord>
59103a24052SZachary Turner GetNestedTagRecord(const NestedTypeRecord &Record, const CVTagRecord &parent,
59203a24052SZachary Turner                    TpiStream &tpi) {
59303a24052SZachary Turner   // An LF_NESTTYPE is essentially a nested typedef / using declaration, but it
59403a24052SZachary Turner   // is also used to indicate the primary definition of a nested class.  That is
59503a24052SZachary Turner   // to say, if you have:
59603a24052SZachary Turner   // struct A {
59703a24052SZachary Turner   //   struct B {};
59803a24052SZachary Turner   //   using C = B;
59903a24052SZachary Turner   // };
60003a24052SZachary Turner   // Then in the debug info, this will appear as:
60103a24052SZachary Turner   // LF_STRUCTURE `A::B` [type index = N]
60203a24052SZachary Turner   // LF_STRUCTURE `A`
60303a24052SZachary Turner   //   LF_NESTTYPE [name = `B`, index = N]
60403a24052SZachary Turner   //   LF_NESTTYPE [name = `C`, index = N]
60503a24052SZachary Turner   // In order to accurately reconstruct the decl context hierarchy, we need to
60603a24052SZachary Turner   // know which ones are actual definitions and which ones are just aliases.
60703a24052SZachary Turner 
60803a24052SZachary Turner   // If it's a simple type, then this is something like `using foo = int`.
60903a24052SZachary Turner   if (Record.Type.isSimple())
61003a24052SZachary Turner     return llvm::None;
61103a24052SZachary Turner 
6122af34166SZachary Turner   CVType cvt = tpi.getType(Record.Type);
6132af34166SZachary Turner 
6142af34166SZachary Turner   if (!IsTagRecord(cvt))
6152af34166SZachary Turner     return llvm::None;
6162af34166SZachary Turner 
61703a24052SZachary Turner   // If it's an inner definition, then treat whatever name we have here as a
61803a24052SZachary Turner   // single component of a mangled name.  So we can inject it into the parent's
61903a24052SZachary Turner   // mangled name to see if it matches.
6202af34166SZachary Turner   CVTagRecord child = CVTagRecord::create(cvt);
62103a24052SZachary Turner   std::string qname = parent.asTag().getUniqueName();
62203a24052SZachary Turner   if (qname.size() < 4 || child.asTag().getUniqueName().size() < 4)
62303a24052SZachary Turner     return llvm::None;
62403a24052SZachary Turner 
62503a24052SZachary Turner   // qname[3] is the tag type identifier (struct, class, union, etc).  Since the
62603a24052SZachary Turner   // inner tag type is not necessarily the same as the outer tag type, re-write
62703a24052SZachary Turner   // it to match the inner tag type.
62803a24052SZachary Turner   qname[3] = child.asTag().getUniqueName()[3];
62903a24052SZachary Turner   std::string piece = Record.Name;
63003a24052SZachary Turner   piece.push_back('@');
63103a24052SZachary Turner   qname.insert(4, std::move(piece));
63203a24052SZachary Turner   if (qname != child.asTag().UniqueName)
63303a24052SZachary Turner     return llvm::None;
63403a24052SZachary Turner 
63503a24052SZachary Turner   return std::move(child);
63603a24052SZachary Turner }
63703a24052SZachary Turner 
638056e4ab4SZachary Turner void SymbolFileNativePDB::PreprocessTpiStream() {
639056e4ab4SZachary Turner   LazyRandomTypeCollection &types = m_index->tpi().typeCollection();
640056e4ab4SZachary Turner 
641056e4ab4SZachary Turner   for (auto ti = types.getFirst(); ti; ti = types.getNext(*ti)) {
642056e4ab4SZachary Turner     CVType type = types.getType(*ti);
643056e4ab4SZachary Turner     if (!IsTagRecord(type))
644056e4ab4SZachary Turner       continue;
645056e4ab4SZachary Turner 
646056e4ab4SZachary Turner     CVTagRecord tag = CVTagRecord::create(type);
647056e4ab4SZachary Turner     // We're looking for LF_NESTTYPE records in the field list, so ignore
648056e4ab4SZachary Turner     // forward references (no field list), and anything without a nested class
649056e4ab4SZachary Turner     // (since there won't be any LF_NESTTYPE records).
650056e4ab4SZachary Turner     if (tag.asTag().isForwardRef() || !tag.asTag().containsNestedClass())
651056e4ab4SZachary Turner       continue;
652056e4ab4SZachary Turner 
653056e4ab4SZachary Turner     struct ProcessTpiStream : public TypeVisitorCallbacks {
654056e4ab4SZachary Turner       ProcessTpiStream(PdbIndex &index, TypeIndex parent,
65503a24052SZachary Turner                        const CVTagRecord &parent_cvt,
656056e4ab4SZachary Turner                        llvm::DenseMap<TypeIndex, TypeIndex> &parents)
65703a24052SZachary Turner           : index(index), parents(parents), parent(parent),
65803a24052SZachary Turner             parent_cvt(parent_cvt) {}
659056e4ab4SZachary Turner 
660056e4ab4SZachary Turner       PdbIndex &index;
661056e4ab4SZachary Turner       llvm::DenseMap<TypeIndex, TypeIndex> &parents;
662056e4ab4SZachary Turner       TypeIndex parent;
66303a24052SZachary Turner       const CVTagRecord &parent_cvt;
664056e4ab4SZachary Turner 
665056e4ab4SZachary Turner       llvm::Error visitKnownMember(CVMemberRecord &CVR,
666056e4ab4SZachary Turner                                    NestedTypeRecord &Record) override {
66703a24052SZachary Turner         llvm::Optional<CVTagRecord> tag =
66803a24052SZachary Turner             GetNestedTagRecord(Record, parent_cvt, index.tpi());
66903a24052SZachary Turner         if (!tag)
670056e4ab4SZachary Turner           return llvm::ErrorSuccess();
67103a24052SZachary Turner 
67203a24052SZachary Turner         parents[Record.Type] = parent;
67303a24052SZachary Turner         if (!tag->asTag().isForwardRef())
67403a24052SZachary Turner           return llvm::ErrorSuccess();
67503a24052SZachary Turner 
676056e4ab4SZachary Turner         llvm::Expected<TypeIndex> full_decl =
677056e4ab4SZachary Turner             index.tpi().findFullDeclForForwardRef(Record.Type);
678056e4ab4SZachary Turner         if (!full_decl) {
679056e4ab4SZachary Turner           llvm::consumeError(full_decl.takeError());
680056e4ab4SZachary Turner           return llvm::ErrorSuccess();
681056e4ab4SZachary Turner         }
682056e4ab4SZachary Turner         parents[*full_decl] = parent;
683056e4ab4SZachary Turner         return llvm::ErrorSuccess();
684056e4ab4SZachary Turner       }
685056e4ab4SZachary Turner     };
686056e4ab4SZachary Turner 
687056e4ab4SZachary Turner     CVType field_list = m_index->tpi().getType(tag.asTag().FieldList);
68803a24052SZachary Turner     ProcessTpiStream process(*m_index, *ti, tag, m_parent_types);
689056e4ab4SZachary Turner     llvm::Error error = visitMemberRecordStream(field_list.data(), process);
690056e4ab4SZachary Turner     if (error)
691056e4ab4SZachary Turner       llvm::consumeError(std::move(error));
692056e4ab4SZachary Turner   }
693056e4ab4SZachary Turner }
694056e4ab4SZachary Turner 
695307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::GetNumCompileUnits() {
696307f5ae8SZachary Turner   const DbiModuleList &modules = m_index->dbi().modules();
697307f5ae8SZachary Turner   uint32_t count = modules.getModuleCount();
698307f5ae8SZachary Turner   if (count == 0)
699307f5ae8SZachary Turner     return count;
700307f5ae8SZachary Turner 
701307f5ae8SZachary Turner   // The linker can inject an additional "dummy" compilation unit into the
702307f5ae8SZachary Turner   // PDB. Ignore this special compile unit for our purposes, if it is there.
703307f5ae8SZachary Turner   // It is always the last one.
704307f5ae8SZachary Turner   DbiModuleDescriptor last = modules.getModuleDescriptor(count - 1);
705307f5ae8SZachary Turner   if (last.getModuleName() == "* Linker *")
706307f5ae8SZachary Turner     --count;
707307f5ae8SZachary Turner   return count;
708307f5ae8SZachary Turner }
709307f5ae8SZachary Turner 
710*6284aee9SZachary Turner lldb::FunctionSP SymbolFileNativePDB::CreateFunction(PdbCompilandSymId func_id,
711307f5ae8SZachary Turner                                                      const SymbolContext &sc) {
712*6284aee9SZachary Turner   const CompilandIndexItem *cci =
713*6284aee9SZachary Turner       m_index->compilands().GetCompiland(func_id.modi);
714307f5ae8SZachary Turner   lldbassert(cci);
715*6284aee9SZachary Turner   CVSymbol sym_record = cci->m_debug_stream.readSymbolAtOffset(func_id.offset);
716307f5ae8SZachary Turner 
717307f5ae8SZachary Turner   lldbassert(sym_record.kind() == S_LPROC32 || sym_record.kind() == S_GPROC32);
718307f5ae8SZachary Turner   SegmentOffsetLength sol = GetSegmentOffsetAndLength(sym_record);
719307f5ae8SZachary Turner 
720307f5ae8SZachary Turner   auto file_vm_addr = m_index->MakeVirtualAddress(sol.so);
721307f5ae8SZachary Turner   if (file_vm_addr == LLDB_INVALID_ADDRESS || file_vm_addr == 0)
722307f5ae8SZachary Turner     return nullptr;
723307f5ae8SZachary Turner 
724307f5ae8SZachary Turner   AddressRange func_range(file_vm_addr, sol.length,
725307f5ae8SZachary Turner                           sc.module_sp->GetSectionList());
726307f5ae8SZachary Turner   if (!func_range.GetBaseAddress().IsValid())
727307f5ae8SZachary Turner     return nullptr;
728307f5ae8SZachary Turner 
729b96181c2SZachary Turner   Type *func_type = nullptr;
730307f5ae8SZachary Turner 
731307f5ae8SZachary Turner   // FIXME: Resolve types and mangled names.
732*6284aee9SZachary Turner   PdbTypeSymId sig_id{TypeIndex::None(), false};
733307f5ae8SZachary Turner   Mangled mangled(getSymbolName(sym_record));
734307f5ae8SZachary Turner   FunctionSP func_sp = std::make_shared<Function>(
735*6284aee9SZachary Turner       sc.comp_unit, toOpaqueUid(func_id), toOpaqueUid(sig_id), mangled,
736307f5ae8SZachary Turner       func_type, func_range);
737307f5ae8SZachary Turner 
738307f5ae8SZachary Turner   sc.comp_unit->AddFunction(func_sp);
739307f5ae8SZachary Turner   return func_sp;
740307f5ae8SZachary Turner }
741307f5ae8SZachary Turner 
742307f5ae8SZachary Turner CompUnitSP
743307f5ae8SZachary Turner SymbolFileNativePDB::CreateCompileUnit(const CompilandIndexItem &cci) {
744307f5ae8SZachary Turner   lldb::LanguageType lang =
745307f5ae8SZachary Turner       cci.m_compile_opts ? TranslateLanguage(cci.m_compile_opts->getLanguage())
746307f5ae8SZachary Turner                          : lldb::eLanguageTypeUnknown;
747307f5ae8SZachary Turner 
748307f5ae8SZachary Turner   LazyBool optimized = eLazyBoolNo;
749307f5ae8SZachary Turner   if (cci.m_compile_opts && cci.m_compile_opts->hasOptimizations())
750307f5ae8SZachary Turner     optimized = eLazyBoolYes;
751307f5ae8SZachary Turner 
752307f5ae8SZachary Turner   llvm::StringRef source_file_name =
753307f5ae8SZachary Turner       m_index->compilands().GetMainSourceFile(cci);
7548f3be7a3SJonas Devlieghere   FileSpec fs(source_file_name);
755307f5ae8SZachary Turner 
756307f5ae8SZachary Turner   CompUnitSP cu_sp =
757307f5ae8SZachary Turner       std::make_shared<CompileUnit>(m_obj_file->GetModule(), nullptr, fs,
758*6284aee9SZachary Turner                                     toOpaqueUid(cci.m_id), lang, optimized);
759307f5ae8SZachary Turner 
760*6284aee9SZachary Turner   m_obj_file->GetModule()->GetSymbolVendor()->SetCompileUnitAtIndex(
761*6284aee9SZachary Turner       cci.m_id.modi, cu_sp);
762307f5ae8SZachary Turner   return cu_sp;
763307f5ae8SZachary Turner }
764307f5ae8SZachary Turner 
765*6284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateModifierType(PdbTypeSymId type_id,
7662f7efbc9SZachary Turner                                                      const ModifierRecord &mr) {
7672f7efbc9SZachary Turner   TpiStream &stream = m_index->tpi();
7682f7efbc9SZachary Turner 
7692f7efbc9SZachary Turner   TypeSP t = GetOrCreateType(mr.ModifiedType);
7702f7efbc9SZachary Turner   CompilerType ct = t->GetForwardCompilerType();
7712f7efbc9SZachary Turner   if ((mr.Modifiers & ModifierOptions::Const) != ModifierOptions::None)
7722f7efbc9SZachary Turner     ct = ct.AddConstModifier();
7732f7efbc9SZachary Turner   if ((mr.Modifiers & ModifierOptions::Volatile) != ModifierOptions::None)
7742f7efbc9SZachary Turner     ct = ct.AddVolatileModifier();
7752f7efbc9SZachary Turner   std::string name;
7762f7efbc9SZachary Turner   if (mr.ModifiedType.isSimple())
7772f7efbc9SZachary Turner     name = GetSimpleTypeName(mr.ModifiedType.getSimpleKind());
7782f7efbc9SZachary Turner   else
7792f7efbc9SZachary Turner     name = computeTypeName(stream.typeCollection(), mr.ModifiedType);
7802f7efbc9SZachary Turner   Declaration decl;
781*6284aee9SZachary Turner   return std::make_shared<Type>(toOpaqueUid(type_id), m_clang->GetSymbolFile(),
7822f7efbc9SZachary Turner                                 ConstString(name), t->GetByteSize(), nullptr,
7832f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
7842f7efbc9SZachary Turner                                 ct, Type::eResolveStateFull);
7852f7efbc9SZachary Turner }
7862f7efbc9SZachary Turner 
7872f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreatePointerType(
788*6284aee9SZachary Turner     PdbTypeSymId type_id, const llvm::codeview::PointerRecord &pr) {
7892f7efbc9SZachary Turner   TypeSP pointee = GetOrCreateType(pr.ReferentType);
790544a66d8SZachary Turner   if (!pointee)
791544a66d8SZachary Turner     return nullptr;
7922f7efbc9SZachary Turner   CompilerType pointee_ct = pointee->GetForwardCompilerType();
7932f7efbc9SZachary Turner   lldbassert(pointee_ct);
7942f7efbc9SZachary Turner   Declaration decl;
7952f7efbc9SZachary Turner 
7962f7efbc9SZachary Turner   if (pr.isPointerToMember()) {
7972f7efbc9SZachary Turner     MemberPointerInfo mpi = pr.getMemberInfo();
7982f7efbc9SZachary Turner     TypeSP class_type = GetOrCreateType(mpi.ContainingType);
7992f7efbc9SZachary Turner 
8002f7efbc9SZachary Turner     CompilerType ct = ClangASTContext::CreateMemberPointerType(
8012f7efbc9SZachary Turner         class_type->GetLayoutCompilerType(), pointee_ct);
8022f7efbc9SZachary Turner 
8032f7efbc9SZachary Turner     return std::make_shared<Type>(
804*6284aee9SZachary Turner         toOpaqueUid(type_id), m_clang->GetSymbolFile(), ConstString(),
8052f7efbc9SZachary Turner         pr.getSize(), nullptr, LLDB_INVALID_UID, Type::eEncodingIsUID, decl, ct,
8062f7efbc9SZachary Turner         Type::eResolveStateFull);
8072f7efbc9SZachary Turner   }
8082f7efbc9SZachary Turner 
8092f7efbc9SZachary Turner   CompilerType pointer_ct = pointee_ct;
8102f7efbc9SZachary Turner   if (pr.getMode() == PointerMode::LValueReference)
8112f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetLValueReferenceType();
8122f7efbc9SZachary Turner   else if (pr.getMode() == PointerMode::RValueReference)
8132f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetRValueReferenceType();
8142f7efbc9SZachary Turner   else
8152f7efbc9SZachary Turner     pointer_ct = pointer_ct.GetPointerType();
8162f7efbc9SZachary Turner 
8172f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Const) != PointerOptions::None)
8182f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddConstModifier();
8192f7efbc9SZachary Turner 
8202f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Volatile) != PointerOptions::None)
8212f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddVolatileModifier();
8222f7efbc9SZachary Turner 
8232f7efbc9SZachary Turner   if ((pr.getOptions() & PointerOptions::Restrict) != PointerOptions::None)
8242f7efbc9SZachary Turner     pointer_ct = pointer_ct.AddRestrictModifier();
8252f7efbc9SZachary Turner 
826*6284aee9SZachary Turner   return std::make_shared<Type>(toOpaqueUid(type_id), m_clang->GetSymbolFile(),
8272f7efbc9SZachary Turner                                 ConstString(), pr.getSize(), nullptr,
8282f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
8292f7efbc9SZachary Turner                                 pointer_ct, Type::eResolveStateFull);
8302f7efbc9SZachary Turner }
8312f7efbc9SZachary Turner 
8322f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateSimpleType(TypeIndex ti) {
833*6284aee9SZachary Turner   uint64_t uid = toOpaqueUid(PdbTypeSymId{ti, false});
834544a66d8SZachary Turner   if (ti == TypeIndex::NullptrT()) {
835544a66d8SZachary Turner     CompilerType ct = m_clang->GetBasicType(eBasicTypeNullPtr);
836544a66d8SZachary Turner     Declaration decl;
837*6284aee9SZachary Turner     return std::make_shared<Type>(
838*6284aee9SZachary Turner         uid, this, ConstString("std::nullptr_t"), 0, nullptr, LLDB_INVALID_UID,
839*6284aee9SZachary Turner         Type::eEncodingIsUID, decl, ct, Type::eResolveStateFull);
840544a66d8SZachary Turner   }
841544a66d8SZachary Turner 
8422f7efbc9SZachary Turner   if (ti.getSimpleMode() != SimpleTypeMode::Direct) {
8432f7efbc9SZachary Turner     TypeSP direct_sp = GetOrCreateType(ti.makeDirect());
8442f7efbc9SZachary Turner     CompilerType ct = direct_sp->GetFullCompilerType();
8452f7efbc9SZachary Turner     ct = ct.GetPointerType();
84671ebb721SZachary Turner     uint32_t pointer_size = 0;
8472f7efbc9SZachary Turner     switch (ti.getSimpleMode()) {
8482f7efbc9SZachary Turner     case SimpleTypeMode::FarPointer32:
8492f7efbc9SZachary Turner     case SimpleTypeMode::NearPointer32:
8502f7efbc9SZachary Turner       pointer_size = 4;
8512f7efbc9SZachary Turner       break;
8522f7efbc9SZachary Turner     case SimpleTypeMode::NearPointer64:
8532f7efbc9SZachary Turner       pointer_size = 8;
8542f7efbc9SZachary Turner       break;
8552f7efbc9SZachary Turner     default:
8562f7efbc9SZachary Turner       // 128-bit and 16-bit pointers unsupported.
8572f7efbc9SZachary Turner       return nullptr;
8582f7efbc9SZachary Turner     }
8592f7efbc9SZachary Turner     Declaration decl;
860*6284aee9SZachary Turner     return std::make_shared<Type>(uid, m_clang->GetSymbolFile(), ConstString(),
861*6284aee9SZachary Turner                                   pointer_size, nullptr, LLDB_INVALID_UID,
862*6284aee9SZachary Turner                                   Type::eEncodingIsUID, decl, ct,
863*6284aee9SZachary Turner                                   Type::eResolveStateFull);
8642f7efbc9SZachary Turner   }
8652f7efbc9SZachary Turner 
8662f7efbc9SZachary Turner   if (ti.getSimpleKind() == SimpleTypeKind::NotTranslated)
8672f7efbc9SZachary Turner     return nullptr;
8682f7efbc9SZachary Turner 
8692f7efbc9SZachary Turner   lldb::BasicType bt = GetCompilerTypeForSimpleKind(ti.getSimpleKind());
870544a66d8SZachary Turner   if (bt == lldb::eBasicTypeInvalid)
871544a66d8SZachary Turner     return nullptr;
8722f7efbc9SZachary Turner   CompilerType ct = m_clang->GetBasicType(bt);
8732f7efbc9SZachary Turner   size_t size = GetTypeSizeForSimpleKind(ti.getSimpleKind());
8742f7efbc9SZachary Turner 
8752f7efbc9SZachary Turner   llvm::StringRef type_name = GetSimpleTypeName(ti.getSimpleKind());
8762f7efbc9SZachary Turner 
8772f7efbc9SZachary Turner   Declaration decl;
878*6284aee9SZachary Turner   return std::make_shared<Type>(uid, m_clang->GetSymbolFile(),
8792f7efbc9SZachary Turner                                 ConstString(type_name), size, nullptr,
8802f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
8812f7efbc9SZachary Turner                                 ct, Type::eResolveStateFull);
8822f7efbc9SZachary Turner }
8832f7efbc9SZachary Turner 
884056e4ab4SZachary Turner static std::string RenderDemanglerNode(llvm::ms_demangle::Node *n) {
885056e4ab4SZachary Turner   OutputStream OS;
886056e4ab4SZachary Turner   initializeOutputStream(nullptr, nullptr, OS, 1024);
887056e4ab4SZachary Turner   n->output(OS, llvm::ms_demangle::OF_Default);
888056e4ab4SZachary Turner   OS << '\0';
889056e4ab4SZachary Turner   return {OS.getBuffer()};
890056e4ab4SZachary Turner }
891056e4ab4SZachary Turner 
89203a24052SZachary Turner static bool
89303a24052SZachary Turner AnyScopesHaveTemplateParams(llvm::ArrayRef<llvm::ms_demangle::Node *> scopes) {
89403a24052SZachary Turner   for (llvm::ms_demangle::Node *n : scopes) {
89503a24052SZachary Turner     auto *idn = static_cast<llvm::ms_demangle::IdentifierNode *>(n);
89603a24052SZachary Turner     if (idn->TemplateParams)
89703a24052SZachary Turner       return true;
89803a24052SZachary Turner   }
89903a24052SZachary Turner   return false;
90003a24052SZachary Turner }
90103a24052SZachary Turner 
902056e4ab4SZachary Turner std::pair<clang::DeclContext *, std::string>
903056e4ab4SZachary Turner SymbolFileNativePDB::CreateDeclInfoForType(const TagRecord &record,
904056e4ab4SZachary Turner                                            TypeIndex ti) {
905056e4ab4SZachary Turner   llvm::ms_demangle::Demangler demangler;
906056e4ab4SZachary Turner   StringView sv(record.UniqueName.begin(), record.UniqueName.size());
907056e4ab4SZachary Turner   llvm::ms_demangle::TagTypeNode *ttn = demangler.parseTagUniqueName(sv);
908056e4ab4SZachary Turner   llvm::ms_demangle::IdentifierNode *idn =
909056e4ab4SZachary Turner       ttn->QualifiedName->getUnqualifiedIdentifier();
910056e4ab4SZachary Turner   std::string uname = RenderDemanglerNode(idn);
911056e4ab4SZachary Turner 
912056e4ab4SZachary Turner   llvm::ms_demangle::NodeArrayNode *name_components =
913056e4ab4SZachary Turner       ttn->QualifiedName->Components;
914056e4ab4SZachary Turner   llvm::ArrayRef<llvm::ms_demangle::Node *> scopes(name_components->Nodes,
915056e4ab4SZachary Turner                                                    name_components->Count - 1);
916056e4ab4SZachary Turner 
917056e4ab4SZachary Turner   clang::DeclContext *context = m_clang->GetTranslationUnitDecl();
918056e4ab4SZachary Turner 
919056e4ab4SZachary Turner   // If this type doesn't have a parent type in the debug info, then the best we
920056e4ab4SZachary Turner   // can do is to say that it's either a series of namespaces (if the scope is
921056e4ab4SZachary Turner   // non-empty), or the translation unit (if the scope is empty).
922056e4ab4SZachary Turner   auto parent_iter = m_parent_types.find(ti);
923056e4ab4SZachary Turner   if (parent_iter == m_parent_types.end()) {
924056e4ab4SZachary Turner     if (scopes.empty())
925056e4ab4SZachary Turner       return {context, uname};
926056e4ab4SZachary Turner 
92703a24052SZachary Turner     // If there is no parent in the debug info, but some of the scopes have
92803a24052SZachary Turner     // template params, then this is a case of bad debug info.  See, for
92903a24052SZachary Turner     // example, llvm.org/pr39607.  We don't want to create an ambiguity between
93003a24052SZachary Turner     // a NamespaceDecl and a CXXRecordDecl, so instead we create a class at
93103a24052SZachary Turner     // global scope with the fully qualified name.
93203a24052SZachary Turner     if (AnyScopesHaveTemplateParams(scopes))
93303a24052SZachary Turner       return {context, record.Name};
93403a24052SZachary Turner 
935056e4ab4SZachary Turner     for (llvm::ms_demangle::Node *scope : scopes) {
936056e4ab4SZachary Turner       auto *nii = static_cast<llvm::ms_demangle::NamedIdentifierNode *>(scope);
937056e4ab4SZachary Turner       std::string str = RenderDemanglerNode(nii);
938056e4ab4SZachary Turner       context = m_clang->GetUniqueNamespaceDeclaration(str.c_str(), context);
939056e4ab4SZachary Turner     }
940056e4ab4SZachary Turner     return {context, uname};
941056e4ab4SZachary Turner   }
942056e4ab4SZachary Turner 
943056e4ab4SZachary Turner   // Otherwise, all we need to do is get the parent type of this type and
944056e4ab4SZachary Turner   // recurse into our lazy type creation / AST reconstruction logic to get an
945056e4ab4SZachary Turner   // LLDB TypeSP for the parent.  This will cause the AST to automatically get
946056e4ab4SZachary Turner   // the right DeclContext created for any parent.
947056e4ab4SZachary Turner   TypeSP parent = GetOrCreateType(parent_iter->second);
948056e4ab4SZachary Turner   if (!parent)
949056e4ab4SZachary Turner     return {context, uname};
950056e4ab4SZachary Turner   CompilerType parent_ct = parent->GetForwardCompilerType();
951056e4ab4SZachary Turner   clang::QualType qt = ClangUtil::GetCanonicalQualType(parent_ct);
952056e4ab4SZachary Turner   context = clang::TagDecl::castToDeclContext(qt->getAsTagDecl());
953056e4ab4SZachary Turner   return {context, uname};
954056e4ab4SZachary Turner }
955056e4ab4SZachary Turner 
9562f7efbc9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateClassStructUnion(
957*6284aee9SZachary Turner     PdbTypeSymId type_id, const llvm::codeview::TagRecord &record, size_t size,
9582f7efbc9SZachary Turner     clang::TagTypeKind ttk, clang::MSInheritanceAttr::Spelling inheritance) {
9592f7efbc9SZachary Turner 
960056e4ab4SZachary Turner   clang::DeclContext *decl_context = nullptr;
961056e4ab4SZachary Turner   std::string uname;
962*6284aee9SZachary Turner   std::tie(decl_context, uname) = CreateDeclInfoForType(record, type_id.index);
9632f7efbc9SZachary Turner 
9642f7efbc9SZachary Turner   lldb::AccessType access =
9652f7efbc9SZachary Turner       (ttk == clang::TTK_Class) ? lldb::eAccessPrivate : lldb::eAccessPublic;
9662f7efbc9SZachary Turner 
9672f7efbc9SZachary Turner   ClangASTMetadata metadata;
968*6284aee9SZachary Turner   metadata.SetUserID(toOpaqueUid(type_id));
9692f7efbc9SZachary Turner   metadata.SetIsDynamicCXXType(false);
9702f7efbc9SZachary Turner 
9712f7efbc9SZachary Turner   CompilerType ct =
972056e4ab4SZachary Turner       m_clang->CreateRecordType(decl_context, access, uname.c_str(), ttk,
9732f7efbc9SZachary Turner                                 lldb::eLanguageTypeC_plus_plus, &metadata);
974056e4ab4SZachary Turner 
9752f7efbc9SZachary Turner   lldbassert(ct.IsValid());
9762f7efbc9SZachary Turner 
9772f7efbc9SZachary Turner   clang::CXXRecordDecl *record_decl =
9782f7efbc9SZachary Turner       m_clang->GetAsCXXRecordDecl(ct.GetOpaqueQualType());
9792f7efbc9SZachary Turner   lldbassert(record_decl);
9802f7efbc9SZachary Turner 
9812f7efbc9SZachary Turner   clang::MSInheritanceAttr *attr = clang::MSInheritanceAttr::CreateImplicit(
9822f7efbc9SZachary Turner       *m_clang->getASTContext(), inheritance);
9832f7efbc9SZachary Turner   record_decl->addAttr(attr);
9842f7efbc9SZachary Turner 
9852f7efbc9SZachary Turner   ClangASTContext::StartTagDeclarationDefinition(ct);
9862f7efbc9SZachary Turner 
9872f7efbc9SZachary Turner   // Even if it's possible, don't complete it at this point. Just mark it
9882f7efbc9SZachary Turner   // forward resolved, and if/when LLDB needs the full definition, it can
9892f7efbc9SZachary Turner   // ask us.
9902f7efbc9SZachary Turner   ClangASTContext::SetHasExternalStorage(ct.GetOpaqueQualType(), true);
9912f7efbc9SZachary Turner 
9922f7efbc9SZachary Turner   // FIXME: Search IPI stream for LF_UDT_MOD_SRC_LINE.
9932f7efbc9SZachary Turner   Declaration decl;
994*6284aee9SZachary Turner   return std::make_shared<Type>(toOpaqueUid(type_id), m_clang->GetSymbolFile(),
995056e4ab4SZachary Turner                                 ConstString(uname), size, nullptr,
9962f7efbc9SZachary Turner                                 LLDB_INVALID_UID, Type::eEncodingIsUID, decl,
9972f7efbc9SZachary Turner                                 ct, Type::eResolveStateForward);
9982f7efbc9SZachary Turner }
9992f7efbc9SZachary Turner 
1000*6284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbTypeSymId type_id,
10012f7efbc9SZachary Turner                                                 const ClassRecord &cr) {
10022f7efbc9SZachary Turner   clang::TagTypeKind ttk = TranslateUdtKind(cr);
10032f7efbc9SZachary Turner 
10042f7efbc9SZachary Turner   clang::MSInheritanceAttr::Spelling inheritance =
10052f7efbc9SZachary Turner       GetMSInheritance(m_index->tpi().typeCollection(), cr);
1006*6284aee9SZachary Turner   return CreateClassStructUnion(type_id, cr, cr.getSize(), ttk, inheritance);
10072f7efbc9SZachary Turner }
10082f7efbc9SZachary Turner 
1009*6284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbTypeSymId type_id,
10102f7efbc9SZachary Turner                                                 const UnionRecord &ur) {
10112f7efbc9SZachary Turner   return CreateClassStructUnion(
1012*6284aee9SZachary Turner       type_id, ur, ur.getSize(), clang::TTK_Union,
10132f7efbc9SZachary Turner       clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance);
10142f7efbc9SZachary Turner }
10152f7efbc9SZachary Turner 
1016*6284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::CreateTagType(PdbTypeSymId type_id,
10172f7efbc9SZachary Turner                                                 const EnumRecord &er) {
10182af34166SZachary Turner   clang::DeclContext *decl_context = nullptr;
10192af34166SZachary Turner   std::string uname;
1020*6284aee9SZachary Turner   std::tie(decl_context, uname) = CreateDeclInfoForType(er, type_id.index);
10212f7efbc9SZachary Turner 
10222f7efbc9SZachary Turner   Declaration decl;
10232f7efbc9SZachary Turner   TypeSP underlying_type = GetOrCreateType(er.UnderlyingType);
10242f7efbc9SZachary Turner   CompilerType enum_ct = m_clang->CreateEnumerationType(
10252af34166SZachary Turner       uname.c_str(), decl_context, decl, underlying_type->GetFullCompilerType(),
10262af34166SZachary Turner       er.isScoped());
10272f7efbc9SZachary Turner 
10282f7efbc9SZachary Turner   ClangASTContext::StartTagDeclarationDefinition(enum_ct);
102912abab57SZachary Turner   ClangASTContext::SetHasExternalStorage(enum_ct.GetOpaqueQualType(), true);
10302f7efbc9SZachary Turner 
10312f7efbc9SZachary Turner   // We're just going to forward resolve this for now.  We'll complete
10322f7efbc9SZachary Turner   // it only if the user requests.
10332f7efbc9SZachary Turner   return std::make_shared<lldb_private::Type>(
1034*6284aee9SZachary Turner       toOpaqueUid(type_id), m_clang->GetSymbolFile(), ConstString(uname),
10352f7efbc9SZachary Turner       underlying_type->GetByteSize(), nullptr, LLDB_INVALID_UID,
10362f7efbc9SZachary Turner       lldb_private::Type::eEncodingIsUID, decl, enum_ct,
10372f7efbc9SZachary Turner       lldb_private::Type::eResolveStateForward);
10382f7efbc9SZachary Turner }
10392f7efbc9SZachary Turner 
1040*6284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateArrayType(PdbTypeSymId type_id,
1041511bff21SZachary Turner                                             const ArrayRecord &ar) {
1042511bff21SZachary Turner   TypeSP element_type = GetOrCreateType(ar.ElementType);
1043511bff21SZachary Turner   uint64_t element_count = ar.Size / element_type->GetByteSize();
1044511bff21SZachary Turner 
1045511bff21SZachary Turner   CompilerType element_ct = element_type->GetFullCompilerType();
1046511bff21SZachary Turner 
1047511bff21SZachary Turner   CompilerType array_ct =
1048511bff21SZachary Turner       m_clang->CreateArrayType(element_ct, element_count, false);
1049511bff21SZachary Turner 
1050511bff21SZachary Turner   Declaration decl;
1051511bff21SZachary Turner   TypeSP array_sp = std::make_shared<lldb_private::Type>(
1052*6284aee9SZachary Turner       toOpaqueUid(type_id), m_clang->GetSymbolFile(), ConstString(), ar.Size,
1053511bff21SZachary Turner       nullptr, LLDB_INVALID_UID, lldb_private::Type::eEncodingIsUID, decl,
1054511bff21SZachary Turner       array_ct, lldb_private::Type::eResolveStateFull);
1055511bff21SZachary Turner   array_sp->SetEncodingType(element_type.get());
1056511bff21SZachary Turner   return array_sp;
1057511bff21SZachary Turner }
1058511bff21SZachary Turner 
1059*6284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateProcedureType(PdbTypeSymId type_id,
1060544a66d8SZachary Turner                                                 const ProcedureRecord &pr) {
1061544a66d8SZachary Turner   TpiStream &stream = m_index->tpi();
1062544a66d8SZachary Turner   CVType args_cvt = stream.getType(pr.ArgumentList);
1063544a66d8SZachary Turner   ArgListRecord args;
1064544a66d8SZachary Turner   llvm::cantFail(
1065544a66d8SZachary Turner       TypeDeserializer::deserializeAs<ArgListRecord>(args_cvt, args));
1066544a66d8SZachary Turner 
1067544a66d8SZachary Turner   llvm::ArrayRef<TypeIndex> arg_indices = llvm::makeArrayRef(args.ArgIndices);
1068544a66d8SZachary Turner   bool is_variadic = IsCVarArgsFunction(arg_indices);
1069544a66d8SZachary Turner   if (is_variadic)
1070544a66d8SZachary Turner     arg_indices = arg_indices.drop_back();
1071544a66d8SZachary Turner 
1072544a66d8SZachary Turner   std::vector<CompilerType> arg_list;
1073544a66d8SZachary Turner   arg_list.reserve(arg_list.size());
1074544a66d8SZachary Turner 
1075544a66d8SZachary Turner   for (TypeIndex arg_index : arg_indices) {
1076544a66d8SZachary Turner     TypeSP arg_sp = GetOrCreateType(arg_index);
1077544a66d8SZachary Turner     if (!arg_sp)
1078544a66d8SZachary Turner       return nullptr;
1079544a66d8SZachary Turner     arg_list.push_back(arg_sp->GetFullCompilerType());
1080544a66d8SZachary Turner   }
1081544a66d8SZachary Turner 
1082544a66d8SZachary Turner   TypeSP return_type_sp = GetOrCreateType(pr.ReturnType);
1083544a66d8SZachary Turner   if (!return_type_sp)
1084544a66d8SZachary Turner     return nullptr;
1085544a66d8SZachary Turner 
1086544a66d8SZachary Turner   llvm::Optional<clang::CallingConv> cc =
1087544a66d8SZachary Turner       TranslateCallingConvention(pr.CallConv);
1088544a66d8SZachary Turner   if (!cc)
1089544a66d8SZachary Turner     return nullptr;
1090544a66d8SZachary Turner 
1091544a66d8SZachary Turner   CompilerType return_ct = return_type_sp->GetFullCompilerType();
1092544a66d8SZachary Turner   CompilerType func_sig_ast_type = m_clang->CreateFunctionType(
1093544a66d8SZachary Turner       return_ct, arg_list.data(), arg_list.size(), is_variadic, 0, *cc);
1094544a66d8SZachary Turner 
1095544a66d8SZachary Turner   Declaration decl;
1096544a66d8SZachary Turner   return std::make_shared<lldb_private::Type>(
1097*6284aee9SZachary Turner       toOpaqueUid(type_id), this, ConstString(), 0, nullptr, LLDB_INVALID_UID,
1098544a66d8SZachary Turner       lldb_private::Type::eEncodingIsUID, decl, func_sig_ast_type,
1099544a66d8SZachary Turner       lldb_private::Type::eResolveStateFull);
1100544a66d8SZachary Turner }
1101544a66d8SZachary Turner 
1102*6284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateType(PdbTypeSymId type_id) {
1103*6284aee9SZachary Turner   if (type_id.index.isSimple())
1104*6284aee9SZachary Turner     return CreateSimpleType(type_id.index);
11052f7efbc9SZachary Turner 
1106*6284aee9SZachary Turner   TpiStream &stream = type_id.is_ipi ? m_index->ipi() : m_index->tpi();
1107*6284aee9SZachary Turner   CVType cvt = stream.getType(type_id.index);
11082f7efbc9SZachary Turner 
11092f7efbc9SZachary Turner   if (cvt.kind() == LF_MODIFIER) {
11102f7efbc9SZachary Turner     ModifierRecord modifier;
11112f7efbc9SZachary Turner     llvm::cantFail(
11122f7efbc9SZachary Turner         TypeDeserializer::deserializeAs<ModifierRecord>(cvt, modifier));
1113*6284aee9SZachary Turner     return CreateModifierType(type_id, modifier);
11142f7efbc9SZachary Turner   }
11152f7efbc9SZachary Turner 
11162f7efbc9SZachary Turner   if (cvt.kind() == LF_POINTER) {
11172f7efbc9SZachary Turner     PointerRecord pointer;
11182f7efbc9SZachary Turner     llvm::cantFail(
11192f7efbc9SZachary Turner         TypeDeserializer::deserializeAs<PointerRecord>(cvt, pointer));
1120*6284aee9SZachary Turner     return CreatePointerType(type_id, pointer);
11212f7efbc9SZachary Turner   }
11222f7efbc9SZachary Turner 
11232f7efbc9SZachary Turner   if (IsClassRecord(cvt.kind())) {
11242f7efbc9SZachary Turner     ClassRecord cr;
11252f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ClassRecord>(cvt, cr));
1126*6284aee9SZachary Turner     return CreateTagType(type_id, cr);
11272f7efbc9SZachary Turner   }
11282f7efbc9SZachary Turner 
11292f7efbc9SZachary Turner   if (cvt.kind() == LF_ENUM) {
11302f7efbc9SZachary Turner     EnumRecord er;
11312f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<EnumRecord>(cvt, er));
1132*6284aee9SZachary Turner     return CreateTagType(type_id, er);
11332f7efbc9SZachary Turner   }
11342f7efbc9SZachary Turner 
11352f7efbc9SZachary Turner   if (cvt.kind() == LF_UNION) {
11362f7efbc9SZachary Turner     UnionRecord ur;
11372f7efbc9SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<UnionRecord>(cvt, ur));
1138*6284aee9SZachary Turner     return CreateTagType(type_id, ur);
11392f7efbc9SZachary Turner   }
11402f7efbc9SZachary Turner 
1141511bff21SZachary Turner   if (cvt.kind() == LF_ARRAY) {
1142511bff21SZachary Turner     ArrayRecord ar;
1143511bff21SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ArrayRecord>(cvt, ar));
1144*6284aee9SZachary Turner     return CreateArrayType(type_id, ar);
1145511bff21SZachary Turner   }
1146511bff21SZachary Turner 
1147544a66d8SZachary Turner   if (cvt.kind() == LF_PROCEDURE) {
1148544a66d8SZachary Turner     ProcedureRecord pr;
1149544a66d8SZachary Turner     llvm::cantFail(TypeDeserializer::deserializeAs<ProcedureRecord>(cvt, pr));
1150*6284aee9SZachary Turner     return CreateProcedureType(type_id, pr);
1151544a66d8SZachary Turner   }
1152544a66d8SZachary Turner 
11532f7efbc9SZachary Turner   return nullptr;
11542f7efbc9SZachary Turner }
11552f7efbc9SZachary Turner 
1156*6284aee9SZachary Turner TypeSP SymbolFileNativePDB::CreateAndCacheType(PdbTypeSymId type_id) {
11572f7efbc9SZachary Turner   // If they search for a UDT which is a forward ref, try and resolve the full
11582f7efbc9SZachary Turner   // decl and just map the forward ref uid to the full decl record.
1159*6284aee9SZachary Turner   llvm::Optional<PdbTypeSymId> full_decl_uid;
1160*6284aee9SZachary Turner   if (IsForwardRefUdt(type_id, m_index->tpi())) {
1161*6284aee9SZachary Turner     auto expected_full_ti =
1162*6284aee9SZachary Turner         m_index->tpi().findFullDeclForForwardRef(type_id.index);
11632f7efbc9SZachary Turner     if (!expected_full_ti)
11642f7efbc9SZachary Turner       llvm::consumeError(expected_full_ti.takeError());
1165*6284aee9SZachary Turner     else if (*expected_full_ti != type_id.index) {
1166*6284aee9SZachary Turner       full_decl_uid = PdbTypeSymId{*expected_full_ti, false};
11672f7efbc9SZachary Turner 
11682f7efbc9SZachary Turner       // It's possible that a lookup would occur for the full decl causing it
11692f7efbc9SZachary Turner       // to be cached, then a second lookup would occur for the forward decl.
11702f7efbc9SZachary Turner       // We don't want to create a second full decl, so make sure the full
11712f7efbc9SZachary Turner       // decl hasn't already been cached.
1172*6284aee9SZachary Turner       auto full_iter = m_types.find(toOpaqueUid(*full_decl_uid));
11732f7efbc9SZachary Turner       if (full_iter != m_types.end()) {
11742f7efbc9SZachary Turner         TypeSP result = full_iter->second;
11752f7efbc9SZachary Turner         // Map the forward decl to the TypeSP for the full decl so we can take
11762f7efbc9SZachary Turner         // the fast path next time.
1177*6284aee9SZachary Turner         m_types[toOpaqueUid(type_id)] = result;
11782f7efbc9SZachary Turner         return result;
11792f7efbc9SZachary Turner       }
11802f7efbc9SZachary Turner     }
11812f7efbc9SZachary Turner   }
11822f7efbc9SZachary Turner 
1183*6284aee9SZachary Turner   PdbTypeSymId best_decl_id = full_decl_uid ? *full_decl_uid : type_id;
1184*6284aee9SZachary Turner   TypeSP result = CreateType(best_decl_id);
1185544a66d8SZachary Turner   if (!result)
1186544a66d8SZachary Turner     return nullptr;
1187*6284aee9SZachary Turner 
1188*6284aee9SZachary Turner   uint64_t best_uid = toOpaqueUid(best_decl_id);
1189*6284aee9SZachary Turner   m_types[best_uid] = result;
11902f7efbc9SZachary Turner   // If we had both a forward decl and a full decl, make both point to the new
11912f7efbc9SZachary Turner   // type.
11922f7efbc9SZachary Turner   if (full_decl_uid)
1193*6284aee9SZachary Turner     m_types[toOpaqueUid(type_id)] = result;
11942f7efbc9SZachary Turner 
1195*6284aee9SZachary Turner   if (IsTagRecord(best_decl_id, m_index->tpi())) {
11962f7efbc9SZachary Turner     clang::TagDecl *record_decl =
11972f7efbc9SZachary Turner         m_clang->GetAsTagDecl(result->GetForwardCompilerType());
11982f7efbc9SZachary Turner     lldbassert(record_decl);
11992f7efbc9SZachary Turner 
12002f7efbc9SZachary Turner     TypeIndex ti(type_id.index);
1201*6284aee9SZachary Turner     m_uid_to_decl[best_uid] = record_decl;
12022f7efbc9SZachary Turner     m_decl_to_status[record_decl] =
1203*6284aee9SZachary Turner         DeclStatus(best_uid, Type::eResolveStateForward);
12042f7efbc9SZachary Turner   }
12052f7efbc9SZachary Turner   return result;
12062f7efbc9SZachary Turner }
12072f7efbc9SZachary Turner 
1208*6284aee9SZachary Turner TypeSP SymbolFileNativePDB::GetOrCreateType(PdbTypeSymId type_id) {
12092f7efbc9SZachary Turner   // We can't use try_emplace / overwrite here because the process of creating
12102f7efbc9SZachary Turner   // a type could create nested types, which could invalidate iterators.  So
12112f7efbc9SZachary Turner   // we have to do a 2-phase lookup / insert.
1212*6284aee9SZachary Turner   auto iter = m_types.find(toOpaqueUid(type_id));
12132f7efbc9SZachary Turner   if (iter != m_types.end())
12142f7efbc9SZachary Turner     return iter->second;
12152f7efbc9SZachary Turner 
1216*6284aee9SZachary Turner   return CreateAndCacheType(type_id);
12172f7efbc9SZachary Turner }
12182f7efbc9SZachary Turner 
12192af34166SZachary Turner static DWARFExpression
12202af34166SZachary Turner MakeConstantLocationExpression(TypeIndex underlying_ti, TpiStream &tpi,
12212af34166SZachary Turner                                const ConstantSym &constant, ModuleSP module) {
12222af34166SZachary Turner   const ArchSpec &architecture = module->GetArchitecture();
12232af34166SZachary Turner   uint32_t address_size = architecture.GetAddressByteSize();
12242af34166SZachary Turner 
12252af34166SZachary Turner   size_t size = 0;
12262af34166SZachary Turner   bool is_signed = false;
12272af34166SZachary Turner   std::tie(size, is_signed) = GetIntegralTypeInfo(underlying_ti, tpi);
12282af34166SZachary Turner 
12292af34166SZachary Turner   union {
12302af34166SZachary Turner     llvm::support::little64_t I;
12312af34166SZachary Turner     llvm::support::ulittle64_t U;
12322af34166SZachary Turner   } Value;
12332af34166SZachary Turner 
12342af34166SZachary Turner   std::shared_ptr<DataBufferHeap> buffer = std::make_shared<DataBufferHeap>();
12352af34166SZachary Turner   buffer->SetByteSize(size);
12362af34166SZachary Turner 
12372af34166SZachary Turner   llvm::ArrayRef<uint8_t> bytes;
12382af34166SZachary Turner   if (is_signed) {
12392af34166SZachary Turner     Value.I = constant.Value.getSExtValue();
12402af34166SZachary Turner   } else {
12412af34166SZachary Turner     Value.U = constant.Value.getZExtValue();
12422af34166SZachary Turner   }
12432af34166SZachary Turner 
12442af34166SZachary Turner   bytes = llvm::makeArrayRef(reinterpret_cast<const uint8_t *>(&Value), 8)
12452af34166SZachary Turner               .take_front(size);
12462af34166SZachary Turner   buffer->CopyData(bytes.data(), size);
12472af34166SZachary Turner   DataExtractor extractor(buffer, lldb::eByteOrderLittle, address_size);
12482af34166SZachary Turner   DWARFExpression result(nullptr, extractor, nullptr, 0, size);
12492af34166SZachary Turner   return result;
12502af34166SZachary Turner }
12512af34166SZachary Turner 
12529f727950SZachary Turner static DWARFExpression MakeGlobalLocationExpression(uint16_t section,
12539f727950SZachary Turner                                                     uint32_t offset,
12549f727950SZachary Turner                                                     ModuleSP module) {
12559f727950SZachary Turner   assert(section > 0);
12569f727950SZachary Turner   assert(module);
12579f727950SZachary Turner 
12589f727950SZachary Turner   const ArchSpec &architecture = module->GetArchitecture();
12599f727950SZachary Turner   ByteOrder byte_order = architecture.GetByteOrder();
12609f727950SZachary Turner   uint32_t address_size = architecture.GetAddressByteSize();
12619f727950SZachary Turner   uint32_t byte_size = architecture.GetDataByteSize();
12629f727950SZachary Turner   assert(byte_order != eByteOrderInvalid && address_size != 0);
12639f727950SZachary Turner 
12649f727950SZachary Turner   RegisterKind register_kind = eRegisterKindDWARF;
12659f727950SZachary Turner   StreamBuffer<32> stream(Stream::eBinary, address_size, byte_order);
12669f727950SZachary Turner   stream.PutHex8(DW_OP_addr);
12679f727950SZachary Turner 
12689f727950SZachary Turner   SectionList *section_list = module->GetSectionList();
12699f727950SZachary Turner   assert(section_list);
12709f727950SZachary Turner 
12719f727950SZachary Turner   // Section indices in PDB are 1-based, but in DWARF they are 0-based, so we
12729f727950SZachary Turner   // need to subtract 1.
12739f727950SZachary Turner   uint32_t section_idx = section - 1;
12749f727950SZachary Turner   if (section_idx >= section_list->GetSize())
12759f727950SZachary Turner     return DWARFExpression(nullptr);
12769f727950SZachary Turner 
12779f727950SZachary Turner   auto section_ptr = section_list->GetSectionAtIndex(section_idx);
12789f727950SZachary Turner   if (!section_ptr)
12799f727950SZachary Turner     return DWARFExpression(nullptr);
12809f727950SZachary Turner 
12819f727950SZachary Turner   stream.PutMaxHex64(section_ptr->GetFileAddress() + offset, address_size,
12829f727950SZachary Turner                      byte_order);
12839f727950SZachary Turner   DataBufferSP buffer =
12849f727950SZachary Turner       std::make_shared<DataBufferHeap>(stream.GetData(), stream.GetSize());
12859f727950SZachary Turner   DataExtractor extractor(buffer, byte_order, address_size, byte_size);
12869f727950SZachary Turner   DWARFExpression result(module, extractor, nullptr, 0, buffer->GetByteSize());
12879f727950SZachary Turner   result.SetRegisterKind(register_kind);
12889f727950SZachary Turner   return result;
12899f727950SZachary Turner }
12909f727950SZachary Turner 
1291*6284aee9SZachary Turner VariableSP SymbolFileNativePDB::CreateGlobalVariable(PdbGlobalSymId var_id) {
1292*6284aee9SZachary Turner   CVSymbol sym = m_index->symrecords().readRecord(var_id.offset);
12932af34166SZachary Turner   if (sym.kind() == S_CONSTANT)
1294*6284aee9SZachary Turner     return CreateConstantSymbol(var_id, sym);
12952af34166SZachary Turner 
12969f727950SZachary Turner   lldb::ValueType scope = eValueTypeInvalid;
12979f727950SZachary Turner   TypeIndex ti;
12989f727950SZachary Turner   llvm::StringRef name;
12999f727950SZachary Turner   lldb::addr_t addr = 0;
13009f727950SZachary Turner   uint16_t section = 0;
13019f727950SZachary Turner   uint32_t offset = 0;
13029f727950SZachary Turner   bool is_external = false;
13039f727950SZachary Turner   switch (sym.kind()) {
13049f727950SZachary Turner   case S_GDATA32:
13059f727950SZachary Turner     is_external = true;
13069f727950SZachary Turner     LLVM_FALLTHROUGH;
13079f727950SZachary Turner   case S_LDATA32: {
13089f727950SZachary Turner     DataSym ds(sym.kind());
13099f727950SZachary Turner     llvm::cantFail(SymbolDeserializer::deserializeAs<DataSym>(sym, ds));
13109f727950SZachary Turner     ti = ds.Type;
13119f727950SZachary Turner     scope = (sym.kind() == S_GDATA32) ? eValueTypeVariableGlobal
13129f727950SZachary Turner                                       : eValueTypeVariableStatic;
13139f727950SZachary Turner     name = ds.Name;
13149f727950SZachary Turner     section = ds.Segment;
13159f727950SZachary Turner     offset = ds.DataOffset;
13169f727950SZachary Turner     addr = m_index->MakeVirtualAddress(ds.Segment, ds.DataOffset);
13179f727950SZachary Turner     break;
13189f727950SZachary Turner   }
13199f727950SZachary Turner   case S_GTHREAD32:
13209f727950SZachary Turner     is_external = true;
13219f727950SZachary Turner     LLVM_FALLTHROUGH;
13229f727950SZachary Turner   case S_LTHREAD32: {
13239f727950SZachary Turner     ThreadLocalDataSym tlds(sym.kind());
13249f727950SZachary Turner     llvm::cantFail(
13259f727950SZachary Turner         SymbolDeserializer::deserializeAs<ThreadLocalDataSym>(sym, tlds));
13269f727950SZachary Turner     ti = tlds.Type;
13279f727950SZachary Turner     name = tlds.Name;
13289f727950SZachary Turner     section = tlds.Segment;
13299f727950SZachary Turner     offset = tlds.DataOffset;
13309f727950SZachary Turner     addr = m_index->MakeVirtualAddress(tlds.Segment, tlds.DataOffset);
13319f727950SZachary Turner     scope = eValueTypeVariableThreadLocal;
13329f727950SZachary Turner     break;
13339f727950SZachary Turner   }
13349f727950SZachary Turner   default:
13359f727950SZachary Turner     llvm_unreachable("unreachable!");
13369f727950SZachary Turner   }
13379f727950SZachary Turner 
13389f727950SZachary Turner   CompUnitSP comp_unit;
13399f727950SZachary Turner   llvm::Optional<uint16_t> modi = m_index->GetModuleIndexForVa(addr);
13409f727950SZachary Turner   if (modi) {
1341*6284aee9SZachary Turner     CompilandIndexItem &cci = m_index->compilands().GetOrCreateCompiland(*modi);
13429f727950SZachary Turner     comp_unit = GetOrCreateCompileUnit(cci);
13439f727950SZachary Turner   }
13449f727950SZachary Turner 
13459f727950SZachary Turner   Declaration decl;
1346*6284aee9SZachary Turner   PdbTypeSymId tid{ti, false};
13479f727950SZachary Turner   SymbolFileTypeSP type_sp =
1348*6284aee9SZachary Turner       std::make_shared<SymbolFileType>(*this, toOpaqueUid(tid));
13499f727950SZachary Turner   Variable::RangeList ranges;
13509f727950SZachary Turner 
13519f727950SZachary Turner   DWARFExpression location = MakeGlobalLocationExpression(
13529f727950SZachary Turner       section, offset, GetObjectFile()->GetModule());
13539f727950SZachary Turner 
13549f727950SZachary Turner   std::string global_name("::");
13559f727950SZachary Turner   global_name += name;
13569f727950SZachary Turner   VariableSP var_sp = std::make_shared<Variable>(
1357*6284aee9SZachary Turner       toOpaqueUid(var_id), name.str().c_str(), global_name.c_str(), type_sp,
13589f727950SZachary Turner       scope, comp_unit.get(), ranges, &decl, location, is_external, false,
13599f727950SZachary Turner       false);
13609f727950SZachary Turner   var_sp->SetLocationIsConstantValueData(false);
13619f727950SZachary Turner 
13629f727950SZachary Turner   return var_sp;
13639f727950SZachary Turner }
13649f727950SZachary Turner 
13652af34166SZachary Turner lldb::VariableSP
1366*6284aee9SZachary Turner SymbolFileNativePDB::CreateConstantSymbol(PdbGlobalSymId var_id,
13672af34166SZachary Turner                                           const CVSymbol &cvs) {
13682af34166SZachary Turner   TpiStream &tpi = m_index->tpi();
13692af34166SZachary Turner   ConstantSym constant(cvs.kind());
13702af34166SZachary Turner 
13712af34166SZachary Turner   llvm::cantFail(SymbolDeserializer::deserializeAs<ConstantSym>(cvs, constant));
13722af34166SZachary Turner   std::string global_name("::");
13732af34166SZachary Turner   global_name += constant.Name;
1374*6284aee9SZachary Turner   PdbTypeSymId tid{constant.Type, false};
13752af34166SZachary Turner   SymbolFileTypeSP type_sp =
1376*6284aee9SZachary Turner       std::make_shared<SymbolFileType>(*this, toOpaqueUid(tid));
13772af34166SZachary Turner 
13782af34166SZachary Turner   Declaration decl;
13792af34166SZachary Turner   Variable::RangeList ranges;
13802af34166SZachary Turner   ModuleSP module = GetObjectFile()->GetModule();
13812af34166SZachary Turner   DWARFExpression location =
13822af34166SZachary Turner       MakeConstantLocationExpression(constant.Type, tpi, constant, module);
13832af34166SZachary Turner 
13842af34166SZachary Turner   VariableSP var_sp = std::make_shared<Variable>(
1385*6284aee9SZachary Turner       toOpaqueUid(var_id), constant.Name.str().c_str(), global_name.c_str(),
13862af34166SZachary Turner       type_sp, eValueTypeVariableGlobal, module.get(), ranges, &decl, location,
13872af34166SZachary Turner       false, false, false);
13882af34166SZachary Turner   var_sp->SetLocationIsConstantValueData(true);
13892af34166SZachary Turner   return var_sp;
13902af34166SZachary Turner }
13912af34166SZachary Turner 
1392*6284aee9SZachary Turner VariableSP
1393*6284aee9SZachary Turner SymbolFileNativePDB::GetOrCreateGlobalVariable(PdbGlobalSymId var_id) {
1394*6284aee9SZachary Turner   auto emplace_result = m_global_vars.try_emplace(toOpaqueUid(var_id), nullptr);
13959f727950SZachary Turner   if (emplace_result.second)
1396*6284aee9SZachary Turner     emplace_result.first->second = CreateGlobalVariable(var_id);
13979f727950SZachary Turner 
13989f727950SZachary Turner   return emplace_result.first->second;
13999f727950SZachary Turner }
14009f727950SZachary Turner 
1401*6284aee9SZachary Turner lldb::TypeSP SymbolFileNativePDB::GetOrCreateType(TypeIndex ti) {
1402*6284aee9SZachary Turner   return GetOrCreateType(PdbTypeSymId{ti, false});
14032f7efbc9SZachary Turner }
14042f7efbc9SZachary Turner 
1405*6284aee9SZachary Turner FunctionSP SymbolFileNativePDB::GetOrCreateFunction(PdbCompilandSymId func_id,
1406307f5ae8SZachary Turner                                                     const SymbolContext &sc) {
1407*6284aee9SZachary Turner   auto emplace_result = m_functions.try_emplace(toOpaqueUid(func_id), nullptr);
1408307f5ae8SZachary Turner   if (emplace_result.second)
1409*6284aee9SZachary Turner     emplace_result.first->second = CreateFunction(func_id, sc);
1410307f5ae8SZachary Turner 
1411307f5ae8SZachary Turner   lldbassert(emplace_result.first->second);
1412307f5ae8SZachary Turner   return emplace_result.first->second;
1413307f5ae8SZachary Turner }
1414307f5ae8SZachary Turner 
1415307f5ae8SZachary Turner CompUnitSP
1416307f5ae8SZachary Turner SymbolFileNativePDB::GetOrCreateCompileUnit(const CompilandIndexItem &cci) {
1417*6284aee9SZachary Turner 
1418307f5ae8SZachary Turner   auto emplace_result =
1419*6284aee9SZachary Turner       m_compilands.try_emplace(toOpaqueUid(cci.m_id), nullptr);
1420307f5ae8SZachary Turner   if (emplace_result.second)
1421307f5ae8SZachary Turner     emplace_result.first->second = CreateCompileUnit(cci);
1422307f5ae8SZachary Turner 
1423307f5ae8SZachary Turner   lldbassert(emplace_result.first->second);
1424307f5ae8SZachary Turner   return emplace_result.first->second;
1425307f5ae8SZachary Turner }
1426307f5ae8SZachary Turner 
1427307f5ae8SZachary Turner lldb::CompUnitSP SymbolFileNativePDB::ParseCompileUnitAtIndex(uint32_t index) {
1428307f5ae8SZachary Turner   if (index >= GetNumCompileUnits())
1429307f5ae8SZachary Turner     return CompUnitSP();
1430307f5ae8SZachary Turner   lldbassert(index < UINT16_MAX);
1431307f5ae8SZachary Turner   if (index >= UINT16_MAX)
1432307f5ae8SZachary Turner     return nullptr;
1433307f5ae8SZachary Turner 
1434307f5ae8SZachary Turner   CompilandIndexItem &item = m_index->compilands().GetOrCreateCompiland(index);
1435307f5ae8SZachary Turner 
1436307f5ae8SZachary Turner   return GetOrCreateCompileUnit(item);
1437307f5ae8SZachary Turner }
1438307f5ae8SZachary Turner 
1439b96181c2SZachary Turner lldb::LanguageType
1440b96181c2SZachary Turner SymbolFileNativePDB::ParseCompileUnitLanguage(const SymbolContext &sc) {
1441307f5ae8SZachary Turner   // What fields should I expect to be filled out on the SymbolContext?  Is it
1442307f5ae8SZachary Turner   // safe to assume that `sc.comp_unit` is valid?
1443307f5ae8SZachary Turner   if (!sc.comp_unit)
1444307f5ae8SZachary Turner     return lldb::eLanguageTypeUnknown;
1445*6284aee9SZachary Turner   PdbSymUid uid(sc.comp_unit->GetID());
1446*6284aee9SZachary Turner   lldbassert(uid.kind() == PdbSymUidKind::Compiland);
1447307f5ae8SZachary Turner 
1448*6284aee9SZachary Turner   CompilandIndexItem *item =
1449*6284aee9SZachary Turner       m_index->compilands().GetCompiland(uid.asCompiland().modi);
1450307f5ae8SZachary Turner   lldbassert(item);
1451307f5ae8SZachary Turner   if (!item->m_compile_opts)
1452307f5ae8SZachary Turner     return lldb::eLanguageTypeUnknown;
1453307f5ae8SZachary Turner 
1454307f5ae8SZachary Turner   return TranslateLanguage(item->m_compile_opts->getLanguage());
1455307f5ae8SZachary Turner }
1456307f5ae8SZachary Turner 
1457b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseCompileUnitFunctions(const SymbolContext &sc) {
1458307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
1459307f5ae8SZachary Turner   return false;
1460307f5ae8SZachary Turner }
1461307f5ae8SZachary Turner 
1462307f5ae8SZachary Turner static bool NeedsResolvedCompileUnit(uint32_t resolve_scope) {
1463307f5ae8SZachary Turner   // If any of these flags are set, we need to resolve the compile unit.
1464307f5ae8SZachary Turner   uint32_t flags = eSymbolContextCompUnit;
1465307f5ae8SZachary Turner   flags |= eSymbolContextVariable;
1466307f5ae8SZachary Turner   flags |= eSymbolContextFunction;
1467307f5ae8SZachary Turner   flags |= eSymbolContextBlock;
1468307f5ae8SZachary Turner   flags |= eSymbolContextLineEntry;
1469307f5ae8SZachary Turner   return (resolve_scope & flags) != 0;
1470307f5ae8SZachary Turner }
1471307f5ae8SZachary Turner 
1472991e4453SZachary Turner uint32_t SymbolFileNativePDB::ResolveSymbolContext(
1473991e4453SZachary Turner     const Address &addr, SymbolContextItem resolve_scope, SymbolContext &sc) {
1474307f5ae8SZachary Turner   uint32_t resolved_flags = 0;
1475307f5ae8SZachary Turner   lldb::addr_t file_addr = addr.GetFileAddress();
1476307f5ae8SZachary Turner 
1477307f5ae8SZachary Turner   if (NeedsResolvedCompileUnit(resolve_scope)) {
1478307f5ae8SZachary Turner     llvm::Optional<uint16_t> modi = m_index->GetModuleIndexForVa(file_addr);
1479307f5ae8SZachary Turner     if (!modi)
1480307f5ae8SZachary Turner       return 0;
1481*6284aee9SZachary Turner     CompilandIndexItem *cci = m_index->compilands().GetCompiland(*modi);
1482307f5ae8SZachary Turner     if (!cci)
1483307f5ae8SZachary Turner       return 0;
1484307f5ae8SZachary Turner 
1485307f5ae8SZachary Turner     sc.comp_unit = GetOrCreateCompileUnit(*cci).get();
1486307f5ae8SZachary Turner     resolved_flags |= eSymbolContextCompUnit;
1487307f5ae8SZachary Turner   }
1488307f5ae8SZachary Turner 
1489307f5ae8SZachary Turner   if (resolve_scope & eSymbolContextFunction) {
1490307f5ae8SZachary Turner     lldbassert(sc.comp_unit);
1491307f5ae8SZachary Turner     std::vector<SymbolAndUid> matches = m_index->FindSymbolsByVa(file_addr);
1492307f5ae8SZachary Turner     for (const auto &match : matches) {
1493*6284aee9SZachary Turner       if (match.uid.kind() != PdbSymUidKind::CompilandSym)
1494307f5ae8SZachary Turner         continue;
1495*6284aee9SZachary Turner       PdbCompilandSymId csid = match.uid.asCompilandSym();
1496*6284aee9SZachary Turner       CVSymbol cvs = m_index->ReadSymbolRecord(csid);
1497*6284aee9SZachary Turner       if (CVSymToPDBSym(cvs.kind()) != PDB_SymType::Function)
1498*6284aee9SZachary Turner         continue;
1499*6284aee9SZachary Turner       sc.function = GetOrCreateFunction(csid, sc).get();
1500307f5ae8SZachary Turner     }
1501307f5ae8SZachary Turner     resolved_flags |= eSymbolContextFunction;
1502307f5ae8SZachary Turner   }
1503307f5ae8SZachary Turner 
1504307f5ae8SZachary Turner   if (resolve_scope & eSymbolContextLineEntry) {
1505307f5ae8SZachary Turner     lldbassert(sc.comp_unit);
1506307f5ae8SZachary Turner     if (auto *line_table = sc.comp_unit->GetLineTable()) {
1507307f5ae8SZachary Turner       if (line_table->FindLineEntryByAddress(addr, sc.line_entry))
1508307f5ae8SZachary Turner         resolved_flags |= eSymbolContextLineEntry;
1509307f5ae8SZachary Turner     }
1510307f5ae8SZachary Turner   }
1511307f5ae8SZachary Turner 
1512307f5ae8SZachary Turner   return resolved_flags;
1513307f5ae8SZachary Turner }
1514307f5ae8SZachary Turner 
1515307f5ae8SZachary Turner static void AppendLineEntryToSequence(LineTable &table, LineSequence &sequence,
1516307f5ae8SZachary Turner                                       const CompilandIndexItem &cci,
1517307f5ae8SZachary Turner                                       lldb::addr_t base_addr,
1518307f5ae8SZachary Turner                                       uint32_t file_number,
1519307f5ae8SZachary Turner                                       const LineFragmentHeader &block,
1520307f5ae8SZachary Turner                                       const LineNumberEntry &cur) {
1521307f5ae8SZachary Turner   LineInfo cur_info(cur.Flags);
1522307f5ae8SZachary Turner 
1523307f5ae8SZachary Turner   if (cur_info.isAlwaysStepInto() || cur_info.isNeverStepInto())
1524307f5ae8SZachary Turner     return;
1525307f5ae8SZachary Turner 
1526307f5ae8SZachary Turner   uint64_t addr = base_addr + cur.Offset;
1527307f5ae8SZachary Turner 
1528307f5ae8SZachary Turner   bool is_statement = cur_info.isStatement();
1529307f5ae8SZachary Turner   bool is_prologue = IsFunctionPrologue(cci, addr);
1530307f5ae8SZachary Turner   bool is_epilogue = IsFunctionEpilogue(cci, addr);
1531307f5ae8SZachary Turner 
1532307f5ae8SZachary Turner   uint32_t lno = cur_info.getStartLine();
1533307f5ae8SZachary Turner 
1534307f5ae8SZachary Turner   table.AppendLineEntryToSequence(&sequence, addr, lno, 0, file_number,
1535307f5ae8SZachary Turner                                   is_statement, false, is_prologue, is_epilogue,
1536307f5ae8SZachary Turner                                   false);
1537307f5ae8SZachary Turner }
1538307f5ae8SZachary Turner 
1539307f5ae8SZachary Turner static void TerminateLineSequence(LineTable &table,
1540307f5ae8SZachary Turner                                   const LineFragmentHeader &block,
1541307f5ae8SZachary Turner                                   lldb::addr_t base_addr, uint32_t file_number,
1542307f5ae8SZachary Turner                                   uint32_t last_line,
1543307f5ae8SZachary Turner                                   std::unique_ptr<LineSequence> seq) {
1544307f5ae8SZachary Turner   // The end is always a terminal entry, so insert it regardless.
1545307f5ae8SZachary Turner   table.AppendLineEntryToSequence(seq.get(), base_addr + block.CodeSize,
1546307f5ae8SZachary Turner                                   last_line, 0, file_number, false, false,
1547307f5ae8SZachary Turner                                   false, false, true);
1548307f5ae8SZachary Turner   table.InsertSequence(seq.release());
1549307f5ae8SZachary Turner }
1550307f5ae8SZachary Turner 
1551b96181c2SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitLineTable(const SymbolContext &sc) {
1552307f5ae8SZachary Turner   // Unfortunately LLDB is set up to parse the entire compile unit line table
1553307f5ae8SZachary Turner   // all at once, even if all it really needs is line info for a specific
1554307f5ae8SZachary Turner   // function.  In the future it would be nice if it could set the sc.m_function
1555307f5ae8SZachary Turner   // member, and we could only get the line info for the function in question.
1556307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
1557*6284aee9SZachary Turner   PdbSymUid cu_id(sc.comp_unit->GetID());
1558*6284aee9SZachary Turner   lldbassert(cu_id.kind() == PdbSymUidKind::Compiland);
1559*6284aee9SZachary Turner   CompilandIndexItem *cci =
1560*6284aee9SZachary Turner       m_index->compilands().GetCompiland(cu_id.asCompiland().modi);
1561307f5ae8SZachary Turner   lldbassert(cci);
1562307f5ae8SZachary Turner   auto line_table = llvm::make_unique<LineTable>(sc.comp_unit);
1563307f5ae8SZachary Turner 
1564307f5ae8SZachary Turner   // This is basically a copy of the .debug$S subsections from all original COFF
1565307f5ae8SZachary Turner   // object files merged together with address relocations applied.  We are
1566307f5ae8SZachary Turner   // looking for all DEBUG_S_LINES subsections.
1567307f5ae8SZachary Turner   for (const DebugSubsectionRecord &dssr :
1568307f5ae8SZachary Turner        cci->m_debug_stream.getSubsectionsArray()) {
1569307f5ae8SZachary Turner     if (dssr.kind() != DebugSubsectionKind::Lines)
1570307f5ae8SZachary Turner       continue;
1571307f5ae8SZachary Turner 
1572307f5ae8SZachary Turner     DebugLinesSubsectionRef lines;
1573307f5ae8SZachary Turner     llvm::BinaryStreamReader reader(dssr.getRecordData());
1574307f5ae8SZachary Turner     if (auto EC = lines.initialize(reader)) {
1575307f5ae8SZachary Turner       llvm::consumeError(std::move(EC));
1576307f5ae8SZachary Turner       return false;
1577307f5ae8SZachary Turner     }
1578307f5ae8SZachary Turner 
1579307f5ae8SZachary Turner     const LineFragmentHeader *lfh = lines.header();
1580307f5ae8SZachary Turner     uint64_t virtual_addr =
1581307f5ae8SZachary Turner         m_index->MakeVirtualAddress(lfh->RelocSegment, lfh->RelocOffset);
1582307f5ae8SZachary Turner 
1583307f5ae8SZachary Turner     const auto &checksums = cci->m_strings.checksums().getArray();
1584307f5ae8SZachary Turner     const auto &strings = cci->m_strings.strings();
1585307f5ae8SZachary Turner     for (const LineColumnEntry &group : lines) {
1586307f5ae8SZachary Turner       // Indices in this structure are actually offsets of records in the
1587307f5ae8SZachary Turner       // DEBUG_S_FILECHECKSUMS subsection.  Those entries then have an index
1588307f5ae8SZachary Turner       // into the global PDB string table.
1589307f5ae8SZachary Turner       auto iter = checksums.at(group.NameIndex);
1590307f5ae8SZachary Turner       if (iter == checksums.end())
1591307f5ae8SZachary Turner         continue;
1592307f5ae8SZachary Turner 
1593307f5ae8SZachary Turner       llvm::Expected<llvm::StringRef> efn =
1594307f5ae8SZachary Turner           strings.getString(iter->FileNameOffset);
1595307f5ae8SZachary Turner       if (!efn) {
1596307f5ae8SZachary Turner         llvm::consumeError(efn.takeError());
1597307f5ae8SZachary Turner         continue;
1598307f5ae8SZachary Turner       }
1599307f5ae8SZachary Turner 
1600307f5ae8SZachary Turner       // LLDB wants the index of the file in the list of support files.
1601307f5ae8SZachary Turner       auto fn_iter = llvm::find(cci->m_file_list, *efn);
1602307f5ae8SZachary Turner       lldbassert(fn_iter != cci->m_file_list.end());
1603307f5ae8SZachary Turner       uint32_t file_index = std::distance(cci->m_file_list.begin(), fn_iter);
1604307f5ae8SZachary Turner 
1605307f5ae8SZachary Turner       std::unique_ptr<LineSequence> sequence(
1606307f5ae8SZachary Turner           line_table->CreateLineSequenceContainer());
1607307f5ae8SZachary Turner       lldbassert(!group.LineNumbers.empty());
1608307f5ae8SZachary Turner 
1609307f5ae8SZachary Turner       for (const LineNumberEntry &entry : group.LineNumbers) {
1610307f5ae8SZachary Turner         AppendLineEntryToSequence(*line_table, *sequence, *cci, virtual_addr,
1611307f5ae8SZachary Turner                                   file_index, *lfh, entry);
1612307f5ae8SZachary Turner       }
1613307f5ae8SZachary Turner       LineInfo last_line(group.LineNumbers.back().Flags);
1614307f5ae8SZachary Turner       TerminateLineSequence(*line_table, *lfh, virtual_addr, file_index,
1615307f5ae8SZachary Turner                             last_line.getEndLine(), std::move(sequence));
1616307f5ae8SZachary Turner     }
1617307f5ae8SZachary Turner   }
1618307f5ae8SZachary Turner 
1619307f5ae8SZachary Turner   if (line_table->GetSize() == 0)
1620307f5ae8SZachary Turner     return false;
1621307f5ae8SZachary Turner 
1622307f5ae8SZachary Turner   sc.comp_unit->SetLineTable(line_table.release());
1623307f5ae8SZachary Turner   return true;
1624307f5ae8SZachary Turner }
1625307f5ae8SZachary Turner 
1626b96181c2SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitDebugMacros(const SymbolContext &sc) {
1627307f5ae8SZachary Turner   // PDB doesn't contain information about macros
1628307f5ae8SZachary Turner   return false;
1629307f5ae8SZachary Turner }
1630307f5ae8SZachary Turner 
1631307f5ae8SZachary Turner bool SymbolFileNativePDB::ParseCompileUnitSupportFiles(
1632b96181c2SZachary Turner     const SymbolContext &sc, FileSpecList &support_files) {
1633307f5ae8SZachary Turner   lldbassert(sc.comp_unit);
1634307f5ae8SZachary Turner 
1635*6284aee9SZachary Turner   PdbSymUid cu_id(sc.comp_unit->GetID());
1636*6284aee9SZachary Turner   lldbassert(cu_id.kind() == PdbSymUidKind::Compiland);
1637*6284aee9SZachary Turner   CompilandIndexItem *cci =
1638*6284aee9SZachary Turner       m_index->compilands().GetCompiland(cu_id.asCompiland().modi);
1639307f5ae8SZachary Turner   lldbassert(cci);
1640307f5ae8SZachary Turner 
1641307f5ae8SZachary Turner   for (llvm::StringRef f : cci->m_file_list) {
1642307f5ae8SZachary Turner     FileSpec::Style style =
1643307f5ae8SZachary Turner         f.startswith("/") ? FileSpec::Style::posix : FileSpec::Style::windows;
16448f3be7a3SJonas Devlieghere     FileSpec spec(f, style);
1645307f5ae8SZachary Turner     support_files.Append(spec);
1646307f5ae8SZachary Turner   }
1647307f5ae8SZachary Turner 
1648307f5ae8SZachary Turner   return true;
1649307f5ae8SZachary Turner }
1650307f5ae8SZachary Turner 
1651307f5ae8SZachary Turner bool SymbolFileNativePDB::ParseImportedModules(
1652b96181c2SZachary Turner     const SymbolContext &sc, std::vector<ConstString> &imported_modules) {
1653307f5ae8SZachary Turner   // PDB does not yet support module debug info
1654307f5ae8SZachary Turner   return false;
1655307f5ae8SZachary Turner }
1656307f5ae8SZachary Turner 
1657b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseFunctionBlocks(const SymbolContext &sc) {
1658307f5ae8SZachary Turner   lldbassert(sc.comp_unit && sc.function);
1659307f5ae8SZachary Turner   return 0;
1660307f5ae8SZachary Turner }
1661307f5ae8SZachary Turner 
16624911023fSZachary Turner void SymbolFileNativePDB::DumpClangAST(Stream &s) {
16634911023fSZachary Turner   if (!m_clang)
16644911023fSZachary Turner     return;
16654911023fSZachary Turner   m_clang->Dump(s);
16664911023fSZachary Turner }
16674911023fSZachary Turner 
16689f727950SZachary Turner uint32_t SymbolFileNativePDB::FindGlobalVariables(
16699f727950SZachary Turner     const ConstString &name, const CompilerDeclContext *parent_decl_ctx,
16709f727950SZachary Turner     uint32_t max_matches, VariableList &variables) {
16719f727950SZachary Turner   using SymbolAndOffset = std::pair<uint32_t, llvm::codeview::CVSymbol>;
16729f727950SZachary Turner 
16739f727950SZachary Turner   std::vector<SymbolAndOffset> results = m_index->globals().findRecordsByName(
16749f727950SZachary Turner       name.GetStringRef(), m_index->symrecords());
16759f727950SZachary Turner   for (const SymbolAndOffset &result : results) {
16769f727950SZachary Turner     VariableSP var;
16779f727950SZachary Turner     switch (result.second.kind()) {
16789f727950SZachary Turner     case SymbolKind::S_GDATA32:
16799f727950SZachary Turner     case SymbolKind::S_LDATA32:
16809f727950SZachary Turner     case SymbolKind::S_GTHREAD32:
16812af34166SZachary Turner     case SymbolKind::S_LTHREAD32:
16822af34166SZachary Turner     case SymbolKind::S_CONSTANT: {
1683*6284aee9SZachary Turner       PdbGlobalSymId global{result.first, false};
1684*6284aee9SZachary Turner       var = GetOrCreateGlobalVariable(global);
16859f727950SZachary Turner       variables.AddVariable(var);
16869f727950SZachary Turner       break;
16879f727950SZachary Turner     }
16889f727950SZachary Turner     default:
16899f727950SZachary Turner       continue;
16909f727950SZachary Turner     }
16919f727950SZachary Turner   }
16929f727950SZachary Turner   return variables.GetSize();
16939f727950SZachary Turner }
16949f727950SZachary Turner 
1695307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::FindFunctions(
1696b96181c2SZachary Turner     const ConstString &name, const CompilerDeclContext *parent_decl_ctx,
1697117b1fa1SZachary Turner     FunctionNameType name_type_mask, bool include_inlines, bool append,
1698b96181c2SZachary Turner     SymbolContextList &sc_list) {
1699307f5ae8SZachary Turner   // For now we only support lookup by method name.
1700307f5ae8SZachary Turner   if (!(name_type_mask & eFunctionNameTypeMethod))
1701307f5ae8SZachary Turner     return 0;
1702307f5ae8SZachary Turner 
1703307f5ae8SZachary Turner   using SymbolAndOffset = std::pair<uint32_t, llvm::codeview::CVSymbol>;
1704307f5ae8SZachary Turner 
1705307f5ae8SZachary Turner   std::vector<SymbolAndOffset> matches = m_index->globals().findRecordsByName(
1706307f5ae8SZachary Turner       name.GetStringRef(), m_index->symrecords());
1707307f5ae8SZachary Turner   for (const SymbolAndOffset &match : matches) {
1708307f5ae8SZachary Turner     if (match.second.kind() != S_PROCREF && match.second.kind() != S_LPROCREF)
1709307f5ae8SZachary Turner       continue;
1710307f5ae8SZachary Turner     ProcRefSym proc(match.second.kind());
1711307f5ae8SZachary Turner     cantFail(SymbolDeserializer::deserializeAs<ProcRefSym>(match.second, proc));
1712307f5ae8SZachary Turner 
1713307f5ae8SZachary Turner     if (!IsValidRecord(proc))
1714307f5ae8SZachary Turner       continue;
1715307f5ae8SZachary Turner 
1716*6284aee9SZachary Turner     CompilandIndexItem &cci =
1717*6284aee9SZachary Turner         m_index->compilands().GetOrCreateCompiland(proc.modi());
1718b96181c2SZachary Turner     SymbolContext sc;
1719307f5ae8SZachary Turner 
1720307f5ae8SZachary Turner     sc.comp_unit = GetOrCreateCompileUnit(cci).get();
1721307f5ae8SZachary Turner     sc.module_sp = sc.comp_unit->GetModule();
1722*6284aee9SZachary Turner     PdbCompilandSymId func_id{proc.modi(), proc.SymOffset};
1723*6284aee9SZachary Turner     sc.function = GetOrCreateFunction(func_id, sc).get();
1724307f5ae8SZachary Turner 
1725307f5ae8SZachary Turner     sc_list.Append(sc);
1726307f5ae8SZachary Turner   }
1727307f5ae8SZachary Turner 
1728307f5ae8SZachary Turner   return sc_list.GetSize();
1729307f5ae8SZachary Turner }
1730307f5ae8SZachary Turner 
1731b96181c2SZachary Turner uint32_t SymbolFileNativePDB::FindFunctions(const RegularExpression &regex,
1732307f5ae8SZachary Turner                                             bool include_inlines, bool append,
1733b96181c2SZachary Turner                                             SymbolContextList &sc_list) {
1734307f5ae8SZachary Turner   return 0;
1735307f5ae8SZachary Turner }
1736307f5ae8SZachary Turner 
1737b96181c2SZachary Turner uint32_t SymbolFileNativePDB::FindTypes(
1738b96181c2SZachary Turner     const SymbolContext &sc, const ConstString &name,
1739b96181c2SZachary Turner     const CompilerDeclContext *parent_decl_ctx, bool append,
1740b96181c2SZachary Turner     uint32_t max_matches, llvm::DenseSet<SymbolFile *> &searched_symbol_files,
1741b96181c2SZachary Turner     TypeMap &types) {
17422f7efbc9SZachary Turner   if (!append)
17432f7efbc9SZachary Turner     types.Clear();
17442f7efbc9SZachary Turner   if (!name)
1745b96181c2SZachary Turner     return 0;
17462f7efbc9SZachary Turner 
17472f7efbc9SZachary Turner   searched_symbol_files.clear();
17482f7efbc9SZachary Turner   searched_symbol_files.insert(this);
17492f7efbc9SZachary Turner 
17502f7efbc9SZachary Turner   // There is an assumption 'name' is not a regex
17512f7efbc9SZachary Turner   size_t match_count = FindTypesByName(name.GetStringRef(), max_matches, types);
17522f7efbc9SZachary Turner 
17532f7efbc9SZachary Turner   return match_count;
1754b96181c2SZachary Turner }
1755b96181c2SZachary Turner 
1756b96181c2SZachary Turner size_t
1757b96181c2SZachary Turner SymbolFileNativePDB::FindTypes(const std::vector<CompilerContext> &context,
1758b96181c2SZachary Turner                                bool append, TypeMap &types) {
1759b96181c2SZachary Turner   return 0;
1760b96181c2SZachary Turner }
1761b96181c2SZachary Turner 
17622f7efbc9SZachary Turner size_t SymbolFileNativePDB::FindTypesByName(llvm::StringRef name,
17632f7efbc9SZachary Turner                                             uint32_t max_matches,
17642f7efbc9SZachary Turner                                             TypeMap &types) {
17652f7efbc9SZachary Turner 
17662f7efbc9SZachary Turner   size_t match_count = 0;
17672f7efbc9SZachary Turner   std::vector<TypeIndex> matches = m_index->tpi().findRecordsByName(name);
17682f7efbc9SZachary Turner   if (max_matches > 0 && max_matches < matches.size())
17692f7efbc9SZachary Turner     matches.resize(max_matches);
17702f7efbc9SZachary Turner 
17712f7efbc9SZachary Turner   for (TypeIndex ti : matches) {
17722f7efbc9SZachary Turner     TypeSP type = GetOrCreateType(ti);
17732f7efbc9SZachary Turner     if (!type)
17742f7efbc9SZachary Turner       continue;
17752f7efbc9SZachary Turner 
17762f7efbc9SZachary Turner     types.Insert(type);
17772f7efbc9SZachary Turner     ++match_count;
17782f7efbc9SZachary Turner   }
17792f7efbc9SZachary Turner   return match_count;
17802f7efbc9SZachary Turner }
17812f7efbc9SZachary Turner 
1782b96181c2SZachary Turner size_t SymbolFileNativePDB::ParseTypes(const SymbolContext &sc) { return 0; }
1783b96181c2SZachary Turner 
1784b96181c2SZachary Turner Type *SymbolFileNativePDB::ResolveTypeUID(lldb::user_id_t type_uid) {
17852f7efbc9SZachary Turner   auto iter = m_types.find(type_uid);
17862f7efbc9SZachary Turner   // lldb should not be passing us non-sensical type uids.  the only way it
17872f7efbc9SZachary Turner   // could have a type uid in the first place is if we handed it out, in which
17889f727950SZachary Turner   // case we should know about the type.  However, that doesn't mean we've
17899f727950SZachary Turner   // instantiated it yet.  We can vend out a UID for a future type.  So if the
17909f727950SZachary Turner   // type doesn't exist, let's instantiate it now.
17919f727950SZachary Turner   if (iter != m_types.end())
17922f7efbc9SZachary Turner     return &*iter->second;
17939f727950SZachary Turner 
1794*6284aee9SZachary Turner   PdbSymUid uid(type_uid);
1795*6284aee9SZachary Turner   lldbassert(uid.kind() == PdbSymUidKind::Type);
1796*6284aee9SZachary Turner   PdbTypeSymId type_id = uid.asTypeSym();
1797*6284aee9SZachary Turner   if (type_id.index.isNoneType())
17989f727950SZachary Turner     return nullptr;
17999f727950SZachary Turner 
1800*6284aee9SZachary Turner   TypeSP type_sp = CreateAndCacheType(type_id);
18019f727950SZachary Turner   return &*type_sp;
1802b96181c2SZachary Turner }
1803b96181c2SZachary Turner 
1804eca07c59SAdrian Prantl llvm::Optional<SymbolFile::ArrayInfo>
1805eca07c59SAdrian Prantl SymbolFileNativePDB::GetDynamicArrayInfoForUID(
1806eca07c59SAdrian Prantl     lldb::user_id_t type_uid, const lldb_private::ExecutionContext *exe_ctx) {
1807eca07c59SAdrian Prantl   return llvm::None;
1808eca07c59SAdrian Prantl }
1809eca07c59SAdrian Prantl 
1810eca07c59SAdrian Prantl 
1811b96181c2SZachary Turner bool SymbolFileNativePDB::CompleteType(CompilerType &compiler_type) {
18122f7efbc9SZachary Turner   // If this is not in our map, it's an error.
18132f7efbc9SZachary Turner   clang::TagDecl *tag_decl = m_clang->GetAsTagDecl(compiler_type);
18142f7efbc9SZachary Turner   lldbassert(tag_decl);
18152f7efbc9SZachary Turner   auto status_iter = m_decl_to_status.find(tag_decl);
18162f7efbc9SZachary Turner   lldbassert(status_iter != m_decl_to_status.end());
18172f7efbc9SZachary Turner 
18182f7efbc9SZachary Turner   // If it's already complete, just return.
18192f7efbc9SZachary Turner   DeclStatus &status = status_iter->second;
18202f7efbc9SZachary Turner   if (status.status == Type::eResolveStateFull)
18212f7efbc9SZachary Turner     return true;
18222f7efbc9SZachary Turner 
1823*6284aee9SZachary Turner   PdbTypeSymId type_id = PdbSymUid(status.uid).asTypeSym();
18242f7efbc9SZachary Turner 
1825*6284aee9SZachary Turner   lldbassert(IsTagRecord(type_id, m_index->tpi()));
18262f7efbc9SZachary Turner 
18272f7efbc9SZachary Turner   ClangASTContext::SetHasExternalStorage(compiler_type.GetOpaqueQualType(),
18282f7efbc9SZachary Turner                                          false);
18292f7efbc9SZachary Turner 
18302f7efbc9SZachary Turner   // In CreateAndCacheType, we already go out of our way to resolve forward
18312f7efbc9SZachary Turner   // ref UDTs to full decls, and the uids we vend out always refer to full
18322f7efbc9SZachary Turner   // decls if a full decl exists in the debug info.  So if we don't have a full
18332f7efbc9SZachary Turner   // decl here, it means one doesn't exist in the debug info, and we can't
18342f7efbc9SZachary Turner   // complete the type.
18352f7efbc9SZachary Turner   CVType cvt = m_index->tpi().getType(TypeIndex(type_id.index));
18362f7efbc9SZachary Turner   if (IsForwardRefUdt(cvt))
18372f7efbc9SZachary Turner     return false;
18382f7efbc9SZachary Turner 
1839*6284aee9SZachary Turner   auto types_iter = m_types.find(status.uid);
18402f7efbc9SZachary Turner   lldbassert(types_iter != m_types.end());
18412f7efbc9SZachary Turner 
1842511bff21SZachary Turner   if (cvt.kind() == LF_MODIFIER) {
1843511bff21SZachary Turner     TypeIndex unmodified_type = LookThroughModifierRecord(cvt);
1844511bff21SZachary Turner     cvt = m_index->tpi().getType(unmodified_type);
1845511bff21SZachary Turner     // LF_MODIFIERS usually point to forward decls, so this is the one case
1846511bff21SZachary Turner     // where we won't have been able to resolve a forward decl to a full decl
1847511bff21SZachary Turner     // earlier on.  So we need to do that now.
1848511bff21SZachary Turner     if (IsForwardRefUdt(cvt)) {
1849511bff21SZachary Turner       llvm::Expected<TypeIndex> expected_full_ti =
1850511bff21SZachary Turner           m_index->tpi().findFullDeclForForwardRef(unmodified_type);
1851511bff21SZachary Turner       if (!expected_full_ti) {
1852511bff21SZachary Turner         llvm::consumeError(expected_full_ti.takeError());
1853511bff21SZachary Turner         return false;
1854511bff21SZachary Turner       }
1855511bff21SZachary Turner       cvt = m_index->tpi().getType(*expected_full_ti);
1856511bff21SZachary Turner       lldbassert(!IsForwardRefUdt(cvt));
1857511bff21SZachary Turner       unmodified_type = *expected_full_ti;
1858511bff21SZachary Turner     }
1859*6284aee9SZachary Turner     type_id = PdbTypeSymId{unmodified_type, false};
1860511bff21SZachary Turner   }
18612f7efbc9SZachary Turner   TypeIndex field_list_ti = GetFieldListIndex(cvt);
18622f7efbc9SZachary Turner   CVType field_list_cvt = m_index->tpi().getType(field_list_ti);
18632f7efbc9SZachary Turner   if (field_list_cvt.kind() != LF_FIELDLIST)
18642f7efbc9SZachary Turner     return false;
18652f7efbc9SZachary Turner 
18662f7efbc9SZachary Turner   // Visit all members of this class, then perform any finalization necessary
18672f7efbc9SZachary Turner   // to complete the class.
1868*6284aee9SZachary Turner   UdtRecordCompleter completer(type_id, compiler_type, *tag_decl, *this);
18692f7efbc9SZachary Turner   auto error =
18702f7efbc9SZachary Turner       llvm::codeview::visitMemberRecordStream(field_list_cvt.data(), completer);
18712f7efbc9SZachary Turner   completer.complete();
18722f7efbc9SZachary Turner 
18732f7efbc9SZachary Turner   status.status = Type::eResolveStateFull;
18742f7efbc9SZachary Turner   if (!error)
18752f7efbc9SZachary Turner     return true;
18762f7efbc9SZachary Turner 
18772f7efbc9SZachary Turner   llvm::consumeError(std::move(error));
1878b96181c2SZachary Turner   return false;
1879b96181c2SZachary Turner }
1880b96181c2SZachary Turner 
1881b96181c2SZachary Turner size_t SymbolFileNativePDB::GetTypes(lldb_private::SymbolContextScope *sc_scope,
1882117b1fa1SZachary Turner                                      TypeClass type_mask,
1883b96181c2SZachary Turner                                      lldb_private::TypeList &type_list) {
1884b96181c2SZachary Turner   return 0;
1885b96181c2SZachary Turner }
1886b96181c2SZachary Turner 
1887b96181c2SZachary Turner CompilerDeclContext
1888b96181c2SZachary Turner SymbolFileNativePDB::FindNamespace(const SymbolContext &sc,
1889b96181c2SZachary Turner                                    const ConstString &name,
1890b96181c2SZachary Turner                                    const CompilerDeclContext *parent_decl_ctx) {
1891307f5ae8SZachary Turner   return {};
1892307f5ae8SZachary Turner }
1893307f5ae8SZachary Turner 
1894b96181c2SZachary Turner TypeSystem *
1895307f5ae8SZachary Turner SymbolFileNativePDB::GetTypeSystemForLanguage(lldb::LanguageType language) {
1896307f5ae8SZachary Turner   auto type_system =
1897307f5ae8SZachary Turner       m_obj_file->GetModule()->GetTypeSystemForLanguage(language);
1898307f5ae8SZachary Turner   if (type_system)
1899307f5ae8SZachary Turner     type_system->SetSymbolFile(this);
1900307f5ae8SZachary Turner   return type_system;
1901307f5ae8SZachary Turner }
1902307f5ae8SZachary Turner 
1903b96181c2SZachary Turner ConstString SymbolFileNativePDB::GetPluginName() {
1904307f5ae8SZachary Turner   static ConstString g_name("pdb");
1905307f5ae8SZachary Turner   return g_name;
1906307f5ae8SZachary Turner }
1907307f5ae8SZachary Turner 
1908307f5ae8SZachary Turner uint32_t SymbolFileNativePDB::GetPluginVersion() { return 1; }
1909