LLDB mainline
SymbolFileNativePDB.cpp
Go to the documentation of this file.
1//===-- SymbolFileNativePDB.cpp -------------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
10
16#include "lldb/Core/Module.h"
26#include "lldb/Utility/Log.h"
27
28#include "llvm/DebugInfo/CodeView/CVRecord.h"
29#include "llvm/DebugInfo/CodeView/CVTypeVisitor.h"
30#include "llvm/DebugInfo/CodeView/DebugLinesSubsection.h"
31#include "llvm/DebugInfo/CodeView/LazyRandomTypeCollection.h"
32#include "llvm/DebugInfo/CodeView/RecordName.h"
33#include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
34#include "llvm/DebugInfo/CodeView/SymbolRecordHelpers.h"
35#include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
36#include "llvm/DebugInfo/PDB/Native/DbiStream.h"
37#include "llvm/DebugInfo/PDB/Native/GlobalsStream.h"
38#include "llvm/DebugInfo/PDB/Native/InfoStream.h"
39#include "llvm/DebugInfo/PDB/Native/ModuleDebugStream.h"
40#include "llvm/DebugInfo/PDB/Native/NativeSession.h"
41#include "llvm/DebugInfo/PDB/Native/PDBFile.h"
42#include "llvm/DebugInfo/PDB/Native/PublicsStream.h"
43#include "llvm/DebugInfo/PDB/Native/SymbolStream.h"
44#include "llvm/DebugInfo/PDB/Native/TpiStream.h"
45#include "llvm/DebugInfo/PDB/PDB.h"
46#include "llvm/DebugInfo/PDB/PDBTypes.h"
47#include "llvm/Demangle/MicrosoftDemangle.h"
48#include "llvm/Object/COFF.h"
49#include "llvm/Support/Allocator.h"
50#include "llvm/Support/BinaryStreamReader.h"
51#include "llvm/Support/Error.h"
52#include "llvm/Support/ErrorOr.h"
53#include "llvm/Support/MemoryBuffer.h"
54
56#include "PdbSymUid.h"
57#include "PdbUtil.h"
58#include "UdtRecordCompleter.h"
59#include <optional>
60#include <string_view>
61
62using namespace lldb;
63using namespace lldb_private;
64using namespace npdb;
65using namespace llvm::codeview;
66using namespace llvm::pdb;
67
69
71 switch (lang) {
72 case PDB_Lang::Cpp:
74 case PDB_Lang::C:
76 case PDB_Lang::Swift:
78 case PDB_Lang::Rust:
80 case PDB_Lang::ObjC:
82 case PDB_Lang::ObjCpp:
84 default:
86 }
87}
88
89static std::unique_ptr<PDBFile>
90loadMatchingPDBFile(std::string exe_path, llvm::BumpPtrAllocator &allocator) {
91 // Try to find a matching PDB for an EXE.
92 using namespace llvm::object;
93 auto expected_binary = createBinary(exe_path);
94
95 // If the file isn't a PE/COFF executable, fail.
96 if (!expected_binary) {
97 llvm::consumeError(expected_binary.takeError());
98 return nullptr;
99 }
100 OwningBinary<Binary> binary = std::move(*expected_binary);
101
102 // TODO: Avoid opening the PE/COFF binary twice by reading this information
103 // directly from the lldb_private::ObjectFile.
104 auto *obj = llvm::dyn_cast<llvm::object::COFFObjectFile>(binary.getBinary());
105 if (!obj)
106 return nullptr;
107 const llvm::codeview::DebugInfo *pdb_info = nullptr;
108
109 // If it doesn't have a debug directory, fail.
110 llvm::StringRef pdb_file;
111 if (llvm::Error e = obj->getDebugPDBInfo(pdb_info, pdb_file)) {
112 consumeError(std::move(e));
113 return nullptr;
114 }
115
116 // If the file doesn't exist, perhaps the path specified at build time
117 // doesn't match the PDB's current location, so check the location of the
118 // executable.
119 if (!FileSystem::Instance().Exists(pdb_file)) {
120 const auto exe_dir = FileSpec(exe_path).CopyByRemovingLastPathComponent();
121 const auto pdb_name = FileSpec(pdb_file).GetFilename().GetCString();
122 pdb_file = exe_dir.CopyByAppendingPathComponent(pdb_name).GetPathAsConstString().GetStringRef();
123 }
124
125 // If the file is not a PDB or if it doesn't have a matching GUID, fail.
126 auto pdb = ObjectFilePDB::loadPDBFile(std::string(pdb_file), allocator);
127 if (!pdb)
128 return nullptr;
129
130 auto expected_info = pdb->getPDBInfoStream();
131 if (!expected_info) {
132 llvm::consumeError(expected_info.takeError());
133 return nullptr;
134 }
135 llvm::codeview::GUID guid;
136 memcpy(&guid, pdb_info->PDB70.Signature, 16);
137
138 if (expected_info->getGuid() != guid)
139 return nullptr;
140 return pdb;
141}
142
144 lldb::addr_t addr) {
145 // FIXME: Implement this.
146 return false;
147}
148
150 lldb::addr_t addr) {
151 // FIXME: Implement this.
152 return false;
153}
154
155// See llvm::codeview::TypeIndex::simpleTypeName as well as strForPrimitiveTi
156// from the original pdbdump:
157// https://github.com/microsoft/microsoft-pdb/blob/805655a28bd8198004be2ac27e6e0290121a5e89/pdbdump/pdbdump.cpp#L1896-L1974
158//
159// For 64bit integers we use "long long" like DIA instead of "__int64".
160static llvm::StringRef GetSimpleTypeName(SimpleTypeKind kind) {
161 switch (kind) {
162 case SimpleTypeKind::Boolean128:
163 return "__bool128";
164 case SimpleTypeKind::Boolean64:
165 return "__bool64";
166 case SimpleTypeKind::Boolean32:
167 return "__bool32";
168 case SimpleTypeKind::Boolean16:
169 return "__bool16";
170 case SimpleTypeKind::Boolean8:
171 return "bool";
172
173 case SimpleTypeKind::Byte:
174 case SimpleTypeKind::UnsignedCharacter:
175 return "unsigned char";
176 case SimpleTypeKind::NarrowCharacter:
177 return "char";
178 case SimpleTypeKind::SignedCharacter:
179 case SimpleTypeKind::SByte:
180 return "signed char";
181 case SimpleTypeKind::Character32:
182 return "char32_t";
183 case SimpleTypeKind::Character16:
184 return "char16_t";
185 case SimpleTypeKind::Character8:
186 return "char8_t";
187
188 case SimpleTypeKind::Complex128:
189 return "_Complex __float128";
190 case SimpleTypeKind::Complex80:
191 return "_Complex long double";
192 case SimpleTypeKind::Complex64:
193 return "_Complex double";
194 case SimpleTypeKind::Complex48:
195 return "_Complex __float48";
196 case SimpleTypeKind::Complex32:
197 case SimpleTypeKind::Complex32PartialPrecision:
198 return "_Complex float";
199 case SimpleTypeKind::Complex16:
200 return "_Complex _Float16";
201
202 case SimpleTypeKind::Float128:
203 return "__float128";
204 case SimpleTypeKind::Float80:
205 return "long double";
206 case SimpleTypeKind::Float64:
207 return "double";
208 case SimpleTypeKind::Float48:
209 return "__float48";
210 case SimpleTypeKind::Float32:
211 case SimpleTypeKind::Float32PartialPrecision:
212 return "float";
213 case SimpleTypeKind::Float16:
214 return "_Float16";
215
216 case SimpleTypeKind::Int128Oct:
217 case SimpleTypeKind::Int128:
218 return "__int128";
219 case SimpleTypeKind::Int64:
220 case SimpleTypeKind::Int64Quad:
221 return "long long";
222 case SimpleTypeKind::Int32Long:
223 return "long";
224 case SimpleTypeKind::Int32:
225 return "int";
226 case SimpleTypeKind::Int16:
227 case SimpleTypeKind::Int16Short:
228 return "short";
229
230 case SimpleTypeKind::UInt128Oct:
231 case SimpleTypeKind::UInt128:
232 return "unsigned __int128";
233 case SimpleTypeKind::UInt64:
234 case SimpleTypeKind::UInt64Quad:
235 return "unsigned long long";
236 case SimpleTypeKind::UInt32:
237 return "unsigned";
238 case SimpleTypeKind::UInt16:
239 case SimpleTypeKind::UInt16Short:
240 return "unsigned short";
241 case SimpleTypeKind::UInt32Long:
242 return "unsigned long";
243
244 case SimpleTypeKind::HResult:
245 return "HRESULT";
246 case SimpleTypeKind::Void:
247 return "void";
248 case SimpleTypeKind::WideCharacter:
249 return "wchar_t";
250
251 case SimpleTypeKind::None:
252 case SimpleTypeKind::NotTranslated:
253 return "";
254 }
255 return "";
256}
257
258static bool IsClassRecord(TypeLeafKind kind) {
259 switch (kind) {
260 case LF_STRUCTURE:
261 case LF_CLASS:
262 case LF_INTERFACE:
263 return true;
264 default:
265 return false;
266 }
267}
268
269static std::optional<CVTagRecord>
270GetNestedTagDefinition(const NestedTypeRecord &Record,
271 const CVTagRecord &parent, TpiStream &tpi) {
272 // An LF_NESTTYPE is essentially a nested typedef / using declaration, but it
273 // is also used to indicate the primary definition of a nested class. That is
274 // to say, if you have:
275 // struct A {
276 // struct B {};
277 // using C = B;
278 // };
279 // Then in the debug info, this will appear as:
280 // LF_STRUCTURE `A::B` [type index = N]
281 // LF_STRUCTURE `A`
282 // LF_NESTTYPE [name = `B`, index = N]
283 // LF_NESTTYPE [name = `C`, index = N]
284 // In order to accurately reconstruct the decl context hierarchy, we need to
285 // know which ones are actual definitions and which ones are just aliases.
286
287 // If it's a simple type, then this is something like `using foo = int`.
288 if (Record.Type.isSimple())
289 return std::nullopt;
290
291 CVType cvt = tpi.getType(Record.Type);
292
293 if (!IsTagRecord(cvt))
294 return std::nullopt;
295
296 // If it's an inner definition, then treat whatever name we have here as a
297 // single component of a mangled name. So we can inject it into the parent's
298 // mangled name to see if it matches.
299 CVTagRecord child = CVTagRecord::create(cvt);
300 std::string qname = std::string(parent.asTag().getUniqueName());
301 if (qname.size() < 4 || child.asTag().getUniqueName().size() < 4)
302 return std::nullopt;
303
304 // qname[3] is the tag type identifier (struct, class, union, etc). Since the
305 // inner tag type is not necessarily the same as the outer tag type, re-write
306 // it to match the inner tag type.
307 qname[3] = child.asTag().getUniqueName()[3];
308 std::string piece;
309 if (qname[3] == 'W')
310 piece = "4";
311 piece += Record.Name;
312 piece.push_back('@');
313 qname.insert(4, std::move(piece));
314 if (qname != child.asTag().UniqueName)
315 return std::nullopt;
316
317 return std::move(child);
318}
319
325
329
331
333 return "Microsoft PDB debug symbol cross-platform file reader.";
334}
335
338 return nullptr;
339
340 return new SymbolFileNativePDB(std::move(objfile_sp));
341}
342
345
347
349 uint32_t abilities = 0;
350 if (!m_objfile_sp)
351 return 0;
352
353 if (!m_index) {
354 // Lazily load and match the PDB file, but only do this once.
355 PDBFile *pdb_file;
356 if (auto *pdb = llvm::dyn_cast<ObjectFilePDB>(m_objfile_sp.get())) {
357 pdb_file = &pdb->GetPDBFile();
358 } else {
359 m_file_up = loadMatchingPDBFile(m_objfile_sp->GetFileSpec().GetPath(),
361 pdb_file = m_file_up.get();
362 }
363
364 if (!pdb_file)
365 return 0;
366
367 auto expected_index = PdbIndex::create(pdb_file);
368 if (!expected_index) {
369 llvm::consumeError(expected_index.takeError());
370 return 0;
371 }
372 m_index = std::move(*expected_index);
373 }
374 if (!m_index)
375 return 0;
376
377 // We don't especially have to be precise here. We only distinguish between
378 // stripped and not stripped.
379 abilities = kAllAbilities;
380
381 if (m_index->dbi().isStripped())
382 abilities &= ~(Blocks | LocalVariables);
383 return abilities;
384}
385
387 m_obj_load_address = m_objfile_sp->GetModule()
388 ->GetObjectFile()
389 ->GetBaseAddress()
390 .GetFileAddress();
391 m_index->SetLoadAddress(m_obj_load_address);
392 m_index->ParseSectionContribs();
393
394 auto ts_or_err = m_objfile_sp->GetModule()->GetTypeSystemForLanguage(
396 if (auto err = ts_or_err.takeError()) {
397 LLDB_LOG_ERROR(GetLog(LLDBLog::Symbols), std::move(err),
398 "Failed to initialize: {0}");
399 } else {
400 if (auto ts = *ts_or_err)
401 ts->SetSymbolFile(this);
403 }
404}
405
407 const DbiModuleList &modules = m_index->dbi().modules();
408 uint32_t count = modules.getModuleCount();
409 if (count == 0)
410 return count;
411
412 // The linker can inject an additional "dummy" compilation unit into the
413 // PDB. Ignore this special compile unit for our purposes, if it is there.
414 // It is always the last one.
415 DbiModuleDescriptor last = modules.getModuleDescriptor(count - 1);
416 if (last.getModuleName() == "* Linker *")
417 --count;
418 return count;
419}
420
422 CompilandIndexItem *cii = m_index->compilands().GetCompiland(block_id.modi);
423 CVSymbol sym = cii->m_debug_stream.readSymbolAtOffset(block_id.offset);
424 CompUnitSP comp_unit = GetOrCreateCompileUnit(*cii);
425 lldb::user_id_t opaque_block_uid = toOpaqueUid(block_id);
426 auto ts_or_err = GetTypeSystemForLanguage(comp_unit->GetLanguage());
427 if (auto err = ts_or_err.takeError())
428 return nullptr;
429 auto ts = *ts_or_err;
430 if (!ts)
431 return nullptr;
432 PdbAstBuilder* ast_builder = ts->GetNativePDBParser();
433
434 switch (sym.kind()) {
435 case S_GPROC32:
436 case S_LPROC32:
437 // This is a function. It must be global. Creating the Function entry
438 // for it automatically creates a block for it.
439 if (FunctionSP func = GetOrCreateFunction(block_id, *comp_unit))
440 return &func->GetBlock(false);
441 break;
442 case S_BLOCK32: {
443 // This is a block. Its parent is either a function or another block. In
444 // either case, its parent can be viewed as a block (e.g. a function
445 // contains 1 big block. So just get the parent block and add this block
446 // to it.
447 BlockSym block(static_cast<SymbolRecordKind>(sym.kind()));
448 cantFail(SymbolDeserializer::deserializeAs<BlockSym>(sym, block));
449 lldbassert(block.Parent != 0);
450 PdbCompilandSymId parent_id(block_id.modi, block.Parent);
451 Block *parent_block = GetOrCreateBlock(parent_id);
452 if (!parent_block)
453 return nullptr;
454 Function *func = parent_block->CalculateSymbolContextFunction();
455 lldbassert(func);
456 lldb::addr_t block_base =
457 m_index->MakeVirtualAddress(block.Segment, block.CodeOffset);
458 lldb::addr_t func_base = func->GetAddress().GetFileAddress();
459 BlockSP child_block = parent_block->CreateChild(opaque_block_uid);
460 if (block_base >= func_base)
461 child_block->AddRange(Block::Range(block_base - func_base, block.CodeSize));
462 else {
463 GetObjectFile()->GetModule()->ReportError(
464 "S_BLOCK32 at modi: {0:d} offset: {1:d}: adding range "
465 "[{2:x16}-{3:x16}) which has a base that is less than the "
466 "function's "
467 "low PC 0x%" PRIx64 ". Please file a bug and attach the file at the "
468 "start of this error message",
469 block_id.modi, block_id.offset, block_base,
470 block_base + block.CodeSize, func_base);
471 }
472 ast_builder->GetOrCreateBlockDecl(block_id);
473 m_blocks.insert({opaque_block_uid, child_block});
474 break;
475 }
476 case S_INLINESITE: {
477 // This ensures line table is parsed first so we have inline sites info.
478 comp_unit->GetLineTable();
479
480 std::shared_ptr<InlineSite> inline_site = m_inline_sites[opaque_block_uid];
481 Block *parent_block = GetOrCreateBlock(inline_site->parent_id);
482 if (!parent_block)
483 return nullptr;
484 BlockSP child_block = parent_block->CreateChild(opaque_block_uid);
485 ast_builder->GetOrCreateInlinedFunctionDecl(block_id);
486 // Copy ranges from InlineSite to Block.
487 for (size_t i = 0; i < inline_site->ranges.GetSize(); ++i) {
488 auto *entry = inline_site->ranges.GetEntryAtIndex(i);
489 child_block->AddRange(
490 Block::Range(entry->GetRangeBase(), entry->GetByteSize()));
491 }
492 child_block->FinalizeRanges();
493
494 // Get the inlined function callsite info.
495 Declaration &decl = inline_site->inline_function_info->GetDeclaration();
496 Declaration &callsite = inline_site->inline_function_info->GetCallSite();
497 child_block->SetInlinedFunctionInfo(
498 inline_site->inline_function_info->GetName().GetCString(), nullptr,
499 &decl, &callsite);
500 m_blocks.insert({opaque_block_uid, child_block});
501 break;
502 }
503 default:
504 lldbassert(false && "Symbol is not a block!");
505 }
506
507 return nullptr;
508}
509
511 CompileUnit &comp_unit) {
512 const CompilandIndexItem *cci =
513 m_index->compilands().GetCompiland(func_id.modi);
514 lldbassert(cci);
515 CVSymbol sym_record = cci->m_debug_stream.readSymbolAtOffset(func_id.offset);
516
517 lldbassert(sym_record.kind() == S_LPROC32 || sym_record.kind() == S_GPROC32);
519
520 auto file_vm_addr =
521 m_index->MakeVirtualAddress(sol.so.segment, sol.so.offset);
522 if (file_vm_addr == LLDB_INVALID_ADDRESS || file_vm_addr == 0)
523 return nullptr;
524
525 Address func_addr(file_vm_addr, comp_unit.GetModule()->GetSectionList());
526 if (!func_addr.IsValid())
527 return nullptr;
528
529 ProcSym proc(static_cast<SymbolRecordKind>(sym_record.kind()));
530 cantFail(SymbolDeserializer::deserializeAs<ProcSym>(sym_record, proc));
531 if (proc.FunctionType == TypeIndex::None())
532 return nullptr;
533 TypeSP func_type = GetOrCreateType(proc.FunctionType);
534 if (!func_type)
535 return nullptr;
536
537 PdbTypeSymId sig_id(proc.FunctionType, false);
538
539 std::optional<llvm::StringRef> mangled_opt = FindMangledSymbol(
540 SegmentOffset(proc.Segment, proc.CodeOffset), proc.FunctionType);
541 Mangled mangled(mangled_opt.value_or(proc.Name));
542
543 FunctionSP func_sp = std::make_shared<Function>(
544 &comp_unit, toOpaqueUid(func_id), toOpaqueUid(sig_id), mangled,
545 func_type.get(), func_addr,
546 AddressRanges{AddressRange(func_addr, sol.length)});
547
548 comp_unit.AddFunction(func_sp);
549
550 auto ts_or_err = GetTypeSystemForLanguage(comp_unit.GetLanguage());
551 if (auto err = ts_or_err.takeError())
552 return func_sp;
553 auto ts = *ts_or_err;
554 if (!ts)
555 return func_sp;
556 ts->GetNativePDBParser()->GetOrCreateFunctionDecl(func_id);
557
558 return func_sp;
559}
560
563 lldb::LanguageType lang =
564 cci.m_compile_opts ? TranslateLanguage(cci.m_compile_opts->getLanguage())
566
567 LazyBool optimized = eLazyBoolNo;
568 if (cci.m_compile_opts && cci.m_compile_opts->hasOptimizations())
569 optimized = eLazyBoolYes;
570
571 llvm::SmallString<64> source_file_name =
572 m_index->compilands().GetMainSourceFile(cci);
573 FileSpec fs(llvm::sys::path::convert_to_slash(
574 source_file_name, llvm::sys::path::Style::windows_backslash));
575
576 CompUnitSP cu_sp = std::make_shared<CompileUnit>(
577 m_objfile_sp->GetModule(), nullptr, std::make_shared<SupportFile>(fs),
578 toOpaqueUid(cci.m_id), lang, optimized);
579
580 SetCompileUnitAtIndex(cci.m_id.modi, cu_sp);
581 return cu_sp;
582}
583
585 const ModifierRecord &mr,
586 CompilerType ct) {
587 TpiStream &stream = m_index->tpi();
588
589 std::string name;
590
591 if ((mr.Modifiers & ModifierOptions::Const) != ModifierOptions::None)
592 name += "const ";
593 if ((mr.Modifiers & ModifierOptions::Volatile) != ModifierOptions::None)
594 name += "volatile ";
595 if ((mr.Modifiers & ModifierOptions::Unaligned) != ModifierOptions::None)
596 name += "__unaligned ";
597
598 if (mr.ModifiedType.isSimple())
599 name += GetSimpleTypeName(mr.ModifiedType.getSimpleKind());
600 else
601 name += computeTypeName(stream.typeCollection(), mr.ModifiedType);
602 Declaration decl;
603 lldb::TypeSP modified_type = GetOrCreateType(mr.ModifiedType);
604
605 return MakeType(toOpaqueUid(type_id), ConstString(name),
606 llvm::expectedToOptional(modified_type->GetByteSize(nullptr)),
607 nullptr, LLDB_INVALID_UID, Type::eEncodingIsUID, decl, ct,
609}
610
613 const llvm::codeview::PointerRecord &pr,
614 CompilerType ct) {
615 TypeSP pointee = GetOrCreateType(pr.ReferentType);
616 if (!pointee)
617 return nullptr;
618
619 if (pr.isPointerToMember()) {
620 MemberPointerInfo mpi = pr.getMemberInfo();
621 GetOrCreateType(mpi.ContainingType);
622 }
623
624 Declaration decl;
625 return MakeType(toOpaqueUid(type_id), ConstString(), pr.getSize(), nullptr,
628}
629
631 CompilerType ct) {
632 uint64_t uid = toOpaqueUid(PdbTypeSymId(ti, false));
633 if (ti == TypeIndex::NullptrT()) {
634 Declaration decl;
635 return MakeType(uid, ConstString("decltype(nullptr)"), std::nullopt,
636 nullptr, LLDB_INVALID_UID, Type::eEncodingIsUID, decl, ct,
638 }
639
640 if (ti.getSimpleMode() != SimpleTypeMode::Direct) {
641 TypeSP direct_sp = GetOrCreateType(ti.makeDirect());
642 uint32_t pointer_size = 0;
643 switch (ti.getSimpleMode()) {
644 case SimpleTypeMode::FarPointer32:
645 case SimpleTypeMode::NearPointer32:
646 pointer_size = 4;
647 break;
648 case SimpleTypeMode::NearPointer64:
649 pointer_size = 8;
650 break;
651 default:
652 // 128-bit and 16-bit pointers unsupported.
653 return nullptr;
654 }
655 Declaration decl;
656 return MakeType(uid, ConstString(), pointer_size, nullptr, LLDB_INVALID_UID,
658 }
659
660 if (ti.getSimpleKind() == SimpleTypeKind::NotTranslated)
661 return nullptr;
662
663 size_t size = GetTypeSizeForSimpleKind(ti.getSimpleKind());
664 llvm::StringRef type_name = GetSimpleTypeName(ti.getSimpleKind());
665
666 Declaration decl;
667 return MakeType(uid, ConstString(type_name), size, nullptr, LLDB_INVALID_UID,
669}
670
671static std::string GetUnqualifiedTypeName(const TagRecord &record) {
672 if (!record.hasUniqueName())
673 return std::string(MSVCUndecoratedNameParser::DropScope(record.Name));
674
675 llvm::ms_demangle::Demangler demangler;
676 std::string_view sv(record.UniqueName.begin(), record.UniqueName.size());
677 llvm::ms_demangle::TagTypeNode *ttn = demangler.parseTagUniqueName(sv);
678 if (demangler.Error)
679 return std::string(MSVCUndecoratedNameParser::DropScope(record.Name));
680
681 llvm::ms_demangle::IdentifierNode *idn =
682 ttn->QualifiedName->getUnqualifiedIdentifier();
683 return idn->toString();
684}
685
688 const TagRecord &record,
689 size_t size, CompilerType ct) {
690
691 std::string uname = GetUnqualifiedTypeName(record);
692
693 llvm::Expected<Declaration> maybeDecl = ResolveUdtDeclaration(type_id);
694 Declaration decl;
695 if (maybeDecl)
696 decl = std::move(*maybeDecl);
697 else
698 LLDB_LOG_ERROR(GetLog(LLDBLog::Symbols), maybeDecl.takeError(),
699 "Failed to resolve declaration for '{1}': {0}", uname);
700
701 return MakeType(toOpaqueUid(type_id), ConstString(uname), size, nullptr,
704}
705
707 const ClassRecord &cr,
708 CompilerType ct) {
709 return CreateClassStructUnion(type_id, cr, cr.getSize(), ct);
710}
711
713 const UnionRecord &ur,
714 CompilerType ct) {
715 return CreateClassStructUnion(type_id, ur, ur.getSize(), ct);
716}
717
719 const EnumRecord &er,
720 CompilerType ct) {
721 std::string uname = GetUnqualifiedTypeName(er);
722
723 llvm::Expected<Declaration> maybeDecl = ResolveUdtDeclaration(type_id);
724 Declaration decl;
725 if (maybeDecl)
726 decl = std::move(*maybeDecl);
727 else
728 LLDB_LOG_ERROR(GetLog(LLDBLog::Symbols), maybeDecl.takeError(),
729 "Failed to resolve declaration for '{1}': {0}", uname);
730
731 TypeSP underlying_type = GetOrCreateType(er.UnderlyingType);
732
733 return MakeType(
734 toOpaqueUid(type_id), ConstString(uname),
735 llvm::expectedToOptional(underlying_type->GetByteSize(nullptr)), nullptr,
738}
739
741 const ArrayRecord &ar,
742 CompilerType ct) {
743 TypeSP element_type = GetOrCreateType(ar.ElementType);
744
745 Declaration decl;
746 TypeSP array_sp =
747 MakeType(toOpaqueUid(type_id), ConstString(), ar.Size, nullptr,
750 array_sp->SetEncodingType(element_type.get());
751 return array_sp;
752}
753
755 const MemberFunctionRecord &mfr,
756 CompilerType ct) {
757 Declaration decl;
758 return MakeType(toOpaqueUid(type_id), ConstString(), 0, nullptr,
761}
762
771
773 if (type_id.index.isSimple())
774 return CreateSimpleType(type_id.index, ct);
775
776 TpiStream &stream = type_id.is_ipi ? m_index->ipi() : m_index->tpi();
777 CVType cvt = stream.getType(type_id.index);
778
779 if (cvt.kind() == LF_MODIFIER) {
780 ModifierRecord modifier;
781 llvm::cantFail(
782 TypeDeserializer::deserializeAs<ModifierRecord>(cvt, modifier));
783 return CreateModifierType(type_id, modifier, ct);
784 }
785
786 if (cvt.kind() == LF_POINTER) {
787 PointerRecord pointer;
788 llvm::cantFail(
789 TypeDeserializer::deserializeAs<PointerRecord>(cvt, pointer));
790 return CreatePointerType(type_id, pointer, ct);
791 }
792
793 if (IsClassRecord(cvt.kind())) {
794 ClassRecord cr;
795 llvm::cantFail(TypeDeserializer::deserializeAs<ClassRecord>(cvt, cr));
796 return CreateTagType(type_id, cr, ct);
797 }
798
799 if (cvt.kind() == LF_ENUM) {
800 EnumRecord er;
801 llvm::cantFail(TypeDeserializer::deserializeAs<EnumRecord>(cvt, er));
802 return CreateTagType(type_id, er, ct);
803 }
804
805 if (cvt.kind() == LF_UNION) {
806 UnionRecord ur;
807 llvm::cantFail(TypeDeserializer::deserializeAs<UnionRecord>(cvt, ur));
808 return CreateTagType(type_id, ur, ct);
809 }
810
811 if (cvt.kind() == LF_ARRAY) {
812 ArrayRecord ar;
813 llvm::cantFail(TypeDeserializer::deserializeAs<ArrayRecord>(cvt, ar));
814 return CreateArrayType(type_id, ar, ct);
815 }
816
817 if (cvt.kind() == LF_PROCEDURE) {
818 ProcedureRecord pr;
819 llvm::cantFail(TypeDeserializer::deserializeAs<ProcedureRecord>(cvt, pr));
820 return CreateProcedureType(type_id, pr, ct);
821 }
822 if (cvt.kind() == LF_MFUNCTION) {
823 MemberFunctionRecord mfr;
824 llvm::cantFail(TypeDeserializer::deserializeAs<MemberFunctionRecord>(cvt, mfr));
825 return CreateFunctionType(type_id, mfr, ct);
826 }
827
828 return nullptr;
829}
830
832 // If they search for a UDT which is a forward ref, try and resolve the full
833 // decl and just map the forward ref uid to the full decl record.
834 std::optional<PdbTypeSymId> full_decl_uid;
835 if (IsForwardRefUdt(type_id, m_index->tpi())) {
836 auto expected_full_ti =
837 m_index->tpi().findFullDeclForForwardRef(type_id.index);
838 if (!expected_full_ti)
839 llvm::consumeError(expected_full_ti.takeError());
840 else if (*expected_full_ti != type_id.index) {
841 full_decl_uid = PdbTypeSymId(*expected_full_ti, false);
842
843 // It's possible that a lookup would occur for the full decl causing it
844 // to be cached, then a second lookup would occur for the forward decl.
845 // We don't want to create a second full decl, so make sure the full
846 // decl hasn't already been cached.
847 auto full_iter = m_types.find(toOpaqueUid(*full_decl_uid));
848 if (full_iter != m_types.end()) {
849 TypeSP result = full_iter->second;
850 // Map the forward decl to the TypeSP for the full decl so we can take
851 // the fast path next time.
852 m_types[toOpaqueUid(type_id)] = result;
853 return result;
854 }
855 }
856 }
857
858 PdbTypeSymId best_decl_id = full_decl_uid ? *full_decl_uid : type_id;
860 if (auto err = ts_or_err.takeError())
861 return nullptr;
862 auto ts = *ts_or_err;
863 if (!ts)
864 return nullptr;
865
866 PdbAstBuilder* ast_builder = ts->GetNativePDBParser();
867 clang::QualType qt = ast_builder->GetOrCreateType(best_decl_id);
868 if (qt.isNull())
869 return nullptr;
870
871 TypeSP result = CreateType(best_decl_id, ast_builder->ToCompilerType(qt));
872 if (!result)
873 return nullptr;
874
875 uint64_t best_uid = toOpaqueUid(best_decl_id);
876 m_types[best_uid] = result;
877 // If we had both a forward decl and a full decl, make both point to the new
878 // type.
879 if (full_decl_uid)
880 m_types[toOpaqueUid(type_id)] = result;
881
882 return result;
883}
884
886 // We can't use try_emplace / overwrite here because the process of creating
887 // a type could create nested types, which could invalidate iterators. So
888 // we have to do a 2-phase lookup / insert.
889 auto iter = m_types.find(toOpaqueUid(type_id));
890 if (iter != m_types.end())
891 return iter->second;
892
893 TypeSP type = CreateAndCacheType(type_id);
894 if (type)
895 GetTypeList().Insert(type);
896 return type;
897}
898
900 CVSymbol sym = m_index->symrecords().readRecord(var_id.offset);
901 if (sym.kind() == S_CONSTANT)
902 return CreateConstantSymbol(var_id, sym);
903
905 TypeIndex ti;
906 llvm::StringRef name;
907 lldb::addr_t addr = 0;
908 uint16_t section = 0;
909 uint32_t offset = 0;
910 bool is_external = false;
911 switch (sym.kind()) {
912 case S_GDATA32:
913 is_external = true;
914 [[fallthrough]];
915 case S_LDATA32: {
916 DataSym ds(sym.kind());
917 llvm::cantFail(SymbolDeserializer::deserializeAs<DataSym>(sym, ds));
918 ti = ds.Type;
919 scope = (sym.kind() == S_GDATA32) ? eValueTypeVariableGlobal
921 name = ds.Name;
922 section = ds.Segment;
923 offset = ds.DataOffset;
924 addr = m_index->MakeVirtualAddress(ds.Segment, ds.DataOffset);
925 break;
926 }
927 case S_GTHREAD32:
928 is_external = true;
929 [[fallthrough]];
930 case S_LTHREAD32: {
931 ThreadLocalDataSym tlds(sym.kind());
932 llvm::cantFail(
933 SymbolDeserializer::deserializeAs<ThreadLocalDataSym>(sym, tlds));
934 ti = tlds.Type;
935 name = tlds.Name;
936 section = tlds.Segment;
937 offset = tlds.DataOffset;
938 addr = m_index->MakeVirtualAddress(tlds.Segment, tlds.DataOffset);
940 break;
941 }
942 default:
943 llvm_unreachable("unreachable!");
944 }
945
946 CompUnitSP comp_unit;
947 std::optional<uint16_t> modi = m_index->GetModuleIndexForVa(addr);
948 // Some globals has modi points to the linker module, ignore them.
949 if (!modi || modi >= GetNumCompileUnits())
950 return nullptr;
951
952 CompilandIndexItem &cci = m_index->compilands().GetOrCreateCompiland(*modi);
953 comp_unit = GetOrCreateCompileUnit(cci);
954
955 Declaration decl;
956 PdbTypeSymId tid(ti, false);
957 SymbolFileTypeSP type_sp =
958 std::make_shared<SymbolFileType>(*this, toOpaqueUid(tid));
959 Variable::RangeList ranges;
960 auto ts_or_err = GetTypeSystemForLanguage(comp_unit->GetLanguage());
961 if (auto err = ts_or_err.takeError())
962 return nullptr;
963 auto ts = *ts_or_err;
964 if (!ts)
965 return nullptr;
966
967 ts->GetNativePDBParser()->GetOrCreateVariableDecl(var_id);
968
969 ModuleSP module_sp = GetObjectFile()->GetModule();
970 DWARFExpressionList location(
971 module_sp, MakeGlobalLocationExpression(section, offset, module_sp),
972 nullptr);
973
974 std::string global_name("::");
975 global_name += name;
976 bool artificial = false;
977 bool location_is_constant_data = false;
978 bool static_member = false;
979 VariableSP var_sp = std::make_shared<Variable>(
980 toOpaqueUid(var_id), name.str().c_str(), global_name.c_str(), type_sp,
981 scope, comp_unit.get(), ranges, &decl, location, is_external, artificial,
982 location_is_constant_data, static_member);
983
984 return var_sp;
985}
986
989 const CVSymbol &cvs) {
990 TpiStream &tpi = m_index->tpi();
991 ConstantSym constant(cvs.kind());
992
993 llvm::cantFail(SymbolDeserializer::deserializeAs<ConstantSym>(cvs, constant));
994 std::string global_name("::");
995 global_name += constant.Name;
996 PdbTypeSymId tid(constant.Type, false);
997 SymbolFileTypeSP type_sp =
998 std::make_shared<SymbolFileType>(*this, toOpaqueUid(tid));
999
1000 Declaration decl;
1001 Variable::RangeList ranges;
1002 ModuleSP module = GetObjectFile()->GetModule();
1003 DWARFExpressionList location(module,
1005 constant.Type, tpi, constant.Value, module),
1006 nullptr);
1007
1008 bool external = false;
1009 bool artificial = false;
1010 bool location_is_constant_data = true;
1011 bool static_member = false;
1012 VariableSP var_sp = std::make_shared<Variable>(
1013 toOpaqueUid(var_id), constant.Name.str().c_str(), global_name.c_str(),
1014 type_sp, eValueTypeVariableGlobal, module.get(), ranges, &decl, location,
1015 external, artificial, location_is_constant_data, static_member);
1016 return var_sp;
1017}
1018
1021 auto emplace_result = m_global_vars.try_emplace(toOpaqueUid(var_id), nullptr);
1022 if (emplace_result.second) {
1023 if (VariableSP var_sp = CreateGlobalVariable(var_id))
1024 emplace_result.first->second = var_sp;
1025 else
1026 return nullptr;
1027 }
1028
1029 return emplace_result.first->second;
1030}
1031
1033 return GetOrCreateType(PdbTypeSymId(ti, false));
1034}
1035
1037 CompileUnit &comp_unit) {
1038 auto emplace_result = m_functions.try_emplace(toOpaqueUid(func_id), nullptr);
1039 if (emplace_result.second)
1040 emplace_result.first->second = CreateFunction(func_id, comp_unit);
1041
1042 return emplace_result.first->second;
1043}
1044
1047
1048 auto emplace_result =
1049 m_compilands.try_emplace(toOpaqueUid(cci.m_id), nullptr);
1050 if (emplace_result.second)
1051 emplace_result.first->second = CreateCompileUnit(cci);
1052
1053 lldbassert(emplace_result.first->second);
1054 return emplace_result.first->second;
1055}
1056
1058 auto iter = m_blocks.find(toOpaqueUid(block_id));
1059 if (iter != m_blocks.end())
1060 return iter->second.get();
1061
1062 return CreateBlock(block_id);
1063}
1064
1067 TypeSystem* ts_or_err = decl_ctx.GetTypeSystem();
1068 if (!ts_or_err)
1069 return;
1070 PdbAstBuilder* ast_builder = ts_or_err->GetNativePDBParser();
1071 clang::DeclContext *context = ast_builder->FromCompilerDeclContext(decl_ctx);
1072 if (!context)
1073 return;
1074 ast_builder->ParseDeclsForContext(*context);
1075}
1076
1078 if (index >= GetNumCompileUnits())
1079 return CompUnitSP();
1080 lldbassert(index < UINT16_MAX);
1081 if (index >= UINT16_MAX)
1082 return nullptr;
1083
1084 CompilandIndexItem &item = m_index->compilands().GetOrCreateCompiland(index);
1085
1086 return GetOrCreateCompileUnit(item);
1087}
1088
1090 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
1091 PdbSymUid uid(comp_unit.GetID());
1093
1094 CompilandIndexItem *item =
1095 m_index->compilands().GetCompiland(uid.asCompiland().modi);
1096 lldbassert(item);
1097 if (!item->m_compile_opts)
1099
1100 return TranslateLanguage(item->m_compile_opts->getLanguage());
1101}
1102
1104 auto *section_list = m_objfile_sp->GetSectionList();
1105 if (!section_list)
1106 return;
1107
1108 for (auto pid : m_index->publics().getPublicsTable()) {
1109 PdbGlobalSymId global{pid, true};
1110 CVSymbol sym = m_index->ReadSymbolRecord(global);
1111 auto kind = sym.kind();
1112 if (kind != S_PUB32)
1113 continue;
1114 PublicSym32 pub =
1115 llvm::cantFail(SymbolDeserializer::deserializeAs<PublicSym32>(sym));
1116
1117 auto section_sp = section_list->FindSectionByID(pub.Segment);
1118 if (!section_sp)
1119 continue;
1120
1122 if ((pub.Flags & PublicSymFlags::Function) != PublicSymFlags::None ||
1123 (pub.Flags & PublicSymFlags::Code) != PublicSymFlags::None)
1124 type = eSymbolTypeCode;
1125
1126 symtab.AddSymbol(Symbol(/*symID=*/pid,
1127 /*name=*/pub.Name,
1128 /*type=*/type,
1129 /*external=*/true,
1130 /*is_debug=*/true,
1131 /*is_trampoline=*/false,
1132 /*is_artificial=*/false,
1133 /*section_sp=*/section_sp,
1134 /*value=*/pub.Offset,
1135 /*size=*/0,
1136 /*size_is_valid=*/false,
1137 /*contains_linker_annotations=*/false,
1138 /*flags=*/0));
1139 }
1140}
1141
1143 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
1144 PdbSymUid uid{comp_unit.GetID()};
1146 uint16_t modi = uid.asCompiland().modi;
1147 CompilandIndexItem &cii = m_index->compilands().GetOrCreateCompiland(modi);
1148
1149 size_t count = comp_unit.GetNumFunctions();
1150 const CVSymbolArray &syms = cii.m_debug_stream.getSymbolArray();
1151 for (auto iter = syms.begin(); iter != syms.end(); ++iter) {
1152 if (iter->kind() != S_LPROC32 && iter->kind() != S_GPROC32)
1153 continue;
1154
1155 PdbCompilandSymId sym_id{modi, iter.offset()};
1156
1157 FunctionSP func = GetOrCreateFunction(sym_id, comp_unit);
1158 }
1159
1160 size_t new_count = comp_unit.GetNumFunctions();
1161 lldbassert(new_count >= count);
1162 return new_count - count;
1163}
1164
1165static bool NeedsResolvedCompileUnit(uint32_t resolve_scope) {
1166 // If any of these flags are set, we need to resolve the compile unit.
1167 uint32_t flags = eSymbolContextCompUnit;
1168 flags |= eSymbolContextVariable;
1169 flags |= eSymbolContextFunction;
1170 flags |= eSymbolContextBlock;
1171 flags |= eSymbolContextLineEntry;
1172 return (resolve_scope & flags) != 0;
1173}
1174
1176 const Address &addr, SymbolContextItem resolve_scope, SymbolContext &sc) {
1177 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
1178 uint32_t resolved_flags = 0;
1179 lldb::addr_t file_addr = addr.GetFileAddress();
1180
1181 if (NeedsResolvedCompileUnit(resolve_scope)) {
1182 std::optional<uint16_t> modi = m_index->GetModuleIndexForVa(file_addr);
1183 if (!modi)
1184 return 0;
1185 CompUnitSP cu_sp = GetCompileUnitAtIndex(*modi);
1186 if (!cu_sp)
1187 return 0;
1188
1189 sc.comp_unit = cu_sp.get();
1190 resolved_flags |= eSymbolContextCompUnit;
1191 }
1192
1193 if (resolve_scope & eSymbolContextFunction ||
1194 resolve_scope & eSymbolContextBlock) {
1196 std::vector<SymbolAndUid> matches = m_index->FindSymbolsByVa(file_addr);
1197 // Search the matches in reverse. This way if there are multiple matches
1198 // (for example we are 3 levels deep in a nested scope) it will find the
1199 // innermost one first.
1200 for (const auto &match : llvm::reverse(matches)) {
1201 if (match.uid.kind() != PdbSymUidKind::CompilandSym)
1202 continue;
1203
1204 PdbCompilandSymId csid = match.uid.asCompilandSym();
1205 CVSymbol cvs = m_index->ReadSymbolRecord(csid);
1206 PDB_SymType type = CVSymToPDBSym(cvs.kind());
1207 if (type != PDB_SymType::Function && type != PDB_SymType::Block)
1208 continue;
1209 if (type == PDB_SymType::Function) {
1210 sc.function = GetOrCreateFunction(csid, *sc.comp_unit).get();
1211 if (sc.function) {
1212 Block &block = sc.function->GetBlock(true);
1213 addr_t func_base = sc.function->GetAddress().GetFileAddress();
1214 addr_t offset = file_addr - func_base;
1215 sc.block = block.FindInnermostBlockByOffset(offset);
1216 }
1217 }
1218
1219 if (type == PDB_SymType::Block) {
1220 Block *block = GetOrCreateBlock(csid);
1221 if (!block)
1222 continue;
1224 if (sc.function) {
1225 sc.function->GetBlock(true);
1226 addr_t func_base = sc.function->GetAddress().GetFileAddress();
1227 addr_t offset = file_addr - func_base;
1228 sc.block = block->FindInnermostBlockByOffset(offset);
1229 }
1230 }
1231 if (sc.function)
1232 resolved_flags |= eSymbolContextFunction;
1233 if (sc.block)
1234 resolved_flags |= eSymbolContextBlock;
1235 break;
1236 }
1237 }
1238
1239 if (resolve_scope & eSymbolContextLineEntry) {
1241 if (auto *line_table = sc.comp_unit->GetLineTable()) {
1242 if (line_table->FindLineEntryByAddress(addr, sc.line_entry))
1243 resolved_flags |= eSymbolContextLineEntry;
1244 }
1245 }
1246
1247 return resolved_flags;
1248}
1249
1251 const SourceLocationSpec &src_location_spec,
1252 lldb::SymbolContextItem resolve_scope, SymbolContextList &sc_list) {
1253 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
1254 const uint32_t prev_size = sc_list.GetSize();
1255 if (resolve_scope & eSymbolContextCompUnit) {
1256 for (uint32_t cu_idx = 0, num_cus = GetNumCompileUnits(); cu_idx < num_cus;
1257 ++cu_idx) {
1258 CompileUnit *cu = ParseCompileUnitAtIndex(cu_idx).get();
1259 if (!cu)
1260 continue;
1261
1262 bool file_spec_matches_cu_file_spec = FileSpec::Match(
1263 src_location_spec.GetFileSpec(), cu->GetPrimaryFile());
1264 if (file_spec_matches_cu_file_spec) {
1265 cu->ResolveSymbolContext(src_location_spec, resolve_scope, sc_list);
1266 break;
1267 }
1268 }
1269 }
1270 return sc_list.GetSize() - prev_size;
1271}
1272
1274 // Unfortunately LLDB is set up to parse the entire compile unit line table
1275 // all at once, even if all it really needs is line info for a specific
1276 // function. In the future it would be nice if it could set the sc.m_function
1277 // member, and we could only get the line info for the function in question.
1278 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
1279 PdbSymUid cu_id(comp_unit.GetID());
1281 uint16_t modi = cu_id.asCompiland().modi;
1282 CompilandIndexItem *cii = m_index->compilands().GetCompiland(modi);
1283 lldbassert(cii);
1284
1285 // Parse DEBUG_S_LINES subsections first, then parse all S_INLINESITE records
1286 // in this CU. Add line entries into the set first so that if there are line
1287 // entries with same addres, the later is always more accurate than the
1288 // former.
1289 std::set<LineTable::Entry, LineTableEntryComparator> line_set;
1290
1291 // This is basically a copy of the .debug$S subsections from all original COFF
1292 // object files merged together with address relocations applied. We are
1293 // looking for all DEBUG_S_LINES subsections.
1294 for (const DebugSubsectionRecord &dssr :
1295 cii->m_debug_stream.getSubsectionsArray()) {
1296 if (dssr.kind() != DebugSubsectionKind::Lines)
1297 continue;
1298
1299 DebugLinesSubsectionRef lines;
1300 llvm::BinaryStreamReader reader(dssr.getRecordData());
1301 if (auto EC = lines.initialize(reader)) {
1302 llvm::consumeError(std::move(EC));
1303 return false;
1304 }
1305
1306 const LineFragmentHeader *lfh = lines.header();
1307 uint64_t virtual_addr =
1308 m_index->MakeVirtualAddress(lfh->RelocSegment, lfh->RelocOffset);
1309 if (virtual_addr == LLDB_INVALID_ADDRESS)
1310 continue;
1311
1312 for (const LineColumnEntry &group : lines) {
1313 llvm::Expected<uint32_t> file_index_or_err =
1314 GetFileIndex(*cii, group.NameIndex);
1315 if (!file_index_or_err)
1316 continue;
1317 uint32_t file_index = file_index_or_err.get();
1318 lldbassert(!group.LineNumbers.empty());
1321 for (const LineNumberEntry &entry : group.LineNumbers) {
1322 LineInfo cur_info(entry.Flags);
1323
1324 if (cur_info.isAlwaysStepInto() || cur_info.isNeverStepInto())
1325 continue;
1326
1327 uint64_t addr = virtual_addr + entry.Offset;
1328
1329 bool is_statement = cur_info.isStatement();
1330 bool is_prologue = IsFunctionPrologue(*cii, addr);
1331 bool is_epilogue = IsFunctionEpilogue(*cii, addr);
1332
1333 uint32_t lno = cur_info.getStartLine();
1334
1335 LineTable::Entry new_entry(addr, lno, 0, file_index, is_statement, false,
1336 is_prologue, is_epilogue, false);
1337 // Terminal entry has lower precedence than new entry.
1338 auto iter = line_set.find(new_entry);
1339 if (iter != line_set.end() && iter->is_terminal_entry)
1340 line_set.erase(iter);
1341 line_set.insert(new_entry);
1342
1343 if (line_entry.GetRangeBase() != LLDB_INVALID_ADDRESS) {
1344 line_entry.SetRangeEnd(addr);
1345 cii->m_global_line_table.Append(line_entry);
1346 }
1347 line_entry.SetRangeBase(addr);
1348 line_entry.data = {file_index, lno};
1349 }
1350 LineInfo last_line(group.LineNumbers.back().Flags);
1351 line_set.emplace(virtual_addr + lfh->CodeSize, last_line.getEndLine(), 0,
1352 file_index, false, false, false, false, true);
1353
1354 if (line_entry.GetRangeBase() != LLDB_INVALID_ADDRESS) {
1355 line_entry.SetRangeEnd(virtual_addr + lfh->CodeSize);
1356 cii->m_global_line_table.Append(line_entry);
1357 }
1358 }
1359 }
1360
1362
1363 // Parse all S_INLINESITE in this CU.
1364 const CVSymbolArray &syms = cii->m_debug_stream.getSymbolArray();
1365 for (auto iter = syms.begin(); iter != syms.end();) {
1366 if (iter->kind() != S_LPROC32 && iter->kind() != S_GPROC32) {
1367 ++iter;
1368 continue;
1369 }
1370
1371 uint32_t record_offset = iter.offset();
1372 CVSymbol func_record =
1373 cii->m_debug_stream.readSymbolAtOffset(record_offset);
1375 addr_t file_vm_addr =
1376 m_index->MakeVirtualAddress(sol.so.segment, sol.so.offset);
1377 if (file_vm_addr == LLDB_INVALID_ADDRESS)
1378 continue;
1379
1380 Address func_base(file_vm_addr, comp_unit.GetModule()->GetSectionList());
1381 PdbCompilandSymId func_id{modi, record_offset};
1382
1383 // Iterate all S_INLINESITEs in the function.
1384 auto parse_inline_sites = [&](SymbolKind kind, PdbCompilandSymId id) {
1385 if (kind != S_INLINESITE)
1386 return false;
1387
1388 ParseInlineSite(id, func_base);
1389
1390 for (const auto &line_entry :
1391 m_inline_sites[toOpaqueUid(id)]->line_entries) {
1392 // If line_entry is not terminal entry, remove previous line entry at
1393 // the same address and insert new one. Terminal entry inside an inline
1394 // site might not be terminal entry for its parent.
1395 if (!line_entry.is_terminal_entry)
1396 line_set.erase(line_entry);
1397 line_set.insert(line_entry);
1398 }
1399 // No longer useful after adding to line_set.
1400 m_inline_sites[toOpaqueUid(id)]->line_entries.clear();
1401 return true;
1402 };
1403 ParseSymbolArrayInScope(func_id, parse_inline_sites);
1404 // Jump to the end of the function record.
1405 iter = syms.at(getScopeEndOffset(func_record));
1406 }
1407
1409
1410 // Add line entries in line_set to line_table.
1411 std::vector<LineTable::Sequence> sequence(1);
1412 for (const auto &line_entry : line_set) {
1414 sequence.back(), line_entry.file_addr, line_entry.line,
1415 line_entry.column, line_entry.file_idx,
1416 line_entry.is_start_of_statement, line_entry.is_start_of_basic_block,
1417 line_entry.is_prologue_end, line_entry.is_epilogue_begin,
1418 line_entry.is_terminal_entry);
1419 }
1420 auto line_table =
1421 std::make_unique<LineTable>(&comp_unit, std::move(sequence));
1422
1423 if (line_table->GetSize() == 0)
1424 return false;
1425
1426 comp_unit.SetLineTable(line_table.release());
1427 return true;
1428}
1429
1431 // PDB doesn't contain information about macros
1432 return false;
1433}
1434
1435llvm::Expected<uint32_t>
1437 uint32_t file_id) {
1438 if (!cii.m_strings.hasChecksums() || !cii.m_strings.hasStrings())
1439 return llvm::make_error<RawError>(raw_error_code::no_entry);
1440
1441 const auto &checksums = cii.m_strings.checksums().getArray();
1442 const auto &strings = cii.m_strings.strings();
1443 // Indices in this structure are actually offsets of records in the
1444 // DEBUG_S_FILECHECKSUMS subsection. Those entries then have an index
1445 // into the global PDB string table.
1446 auto iter = checksums.at(file_id);
1447 if (iter == checksums.end())
1448 return llvm::make_error<RawError>(raw_error_code::no_entry);
1449
1450 llvm::Expected<llvm::StringRef> efn = strings.getString(iter->FileNameOffset);
1451 if (!efn) {
1452 return efn.takeError();
1453 }
1454
1455 // LLDB wants the index of the file in the list of support files.
1456 auto fn_iter = llvm::find(cii.m_file_list, *efn);
1457 if (fn_iter != cii.m_file_list.end())
1458 return std::distance(cii.m_file_list.begin(), fn_iter);
1459 return llvm::make_error<RawError>(raw_error_code::no_entry);
1460}
1461
1463 SupportFileList &support_files) {
1464 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
1465 PdbSymUid cu_id(comp_unit.GetID());
1467 CompilandIndexItem *cci =
1468 m_index->compilands().GetCompiland(cu_id.asCompiland().modi);
1469 lldbassert(cci);
1470
1471 for (llvm::StringRef f : cci->m_file_list) {
1472 FileSpec::Style style =
1473 f.starts_with("/") ? FileSpec::Style::posix : FileSpec::Style::windows;
1474 FileSpec spec(f, style);
1475 support_files.Append(spec);
1476 }
1477 return true;
1478}
1479
1481 const SymbolContext &sc, std::vector<SourceModule> &imported_modules) {
1482 // PDB does not yet support module debug info
1483 return false;
1484}
1485
1487 Address func_addr) {
1488 lldb::user_id_t opaque_uid = toOpaqueUid(id);
1489 if (m_inline_sites.contains(opaque_uid))
1490 return;
1491
1492 addr_t func_base = func_addr.GetFileAddress();
1493 CompilandIndexItem *cii = m_index->compilands().GetCompiland(id.modi);
1494 CVSymbol sym = cii->m_debug_stream.readSymbolAtOffset(id.offset);
1495 CompUnitSP comp_unit = GetOrCreateCompileUnit(*cii);
1496
1497 InlineSiteSym inline_site(static_cast<SymbolRecordKind>(sym.kind()));
1498 cantFail(SymbolDeserializer::deserializeAs<InlineSiteSym>(sym, inline_site));
1499 PdbCompilandSymId parent_id(id.modi, inline_site.Parent);
1500
1501 std::shared_ptr<InlineSite> inline_site_sp =
1502 std::make_shared<InlineSite>(parent_id);
1503
1504 // Get the inlined function declaration info.
1505 auto iter = cii->m_inline_map.find(inline_site.Inlinee);
1506 if (iter == cii->m_inline_map.end())
1507 return;
1508 InlineeSourceLine inlinee_line = iter->second;
1509
1510 const SupportFileList &files = comp_unit->GetSupportFiles();
1511 FileSpec decl_file;
1512 llvm::Expected<uint32_t> file_index_or_err =
1513 GetFileIndex(*cii, inlinee_line.Header->FileID);
1514 if (!file_index_or_err)
1515 return;
1516 uint32_t file_offset = file_index_or_err.get();
1517 decl_file = files.GetFileSpecAtIndex(file_offset);
1518 uint32_t decl_line = inlinee_line.Header->SourceLineNum;
1519 std::unique_ptr<Declaration> decl_up =
1520 std::make_unique<Declaration>(decl_file, decl_line);
1521
1522 // Parse range and line info.
1523 uint32_t code_offset = 0;
1524 int32_t line_offset = 0;
1525 std::optional<uint32_t> code_offset_base;
1526 std::optional<uint32_t> code_offset_end;
1527 std::optional<int32_t> cur_line_offset;
1528 std::optional<int32_t> next_line_offset;
1529 std::optional<uint32_t> next_file_offset;
1530
1531 bool is_terminal_entry = false;
1532 bool is_start_of_statement = true;
1533 // The first instruction is the prologue end.
1534 bool is_prologue_end = true;
1535
1536 auto update_code_offset = [&](uint32_t code_delta) {
1537 if (!code_offset_base)
1538 code_offset_base = code_offset;
1539 else if (!code_offset_end)
1540 code_offset_end = *code_offset_base + code_delta;
1541 };
1542 auto update_line_offset = [&](int32_t line_delta) {
1543 line_offset += line_delta;
1544 if (!code_offset_base || !cur_line_offset)
1545 cur_line_offset = line_offset;
1546 else
1547 next_line_offset = line_offset;
1548 ;
1549 };
1550 auto update_file_offset = [&](uint32_t offset) {
1551 if (!code_offset_base)
1552 file_offset = offset;
1553 else
1554 next_file_offset = offset;
1555 };
1556
1557 for (auto &annot : inline_site.annotations()) {
1558 switch (annot.OpCode) {
1559 case BinaryAnnotationsOpCode::CodeOffset:
1560 case BinaryAnnotationsOpCode::ChangeCodeOffset:
1561 case BinaryAnnotationsOpCode::ChangeCodeOffsetBase:
1562 code_offset += annot.U1;
1563 update_code_offset(annot.U1);
1564 break;
1565 case BinaryAnnotationsOpCode::ChangeLineOffset:
1566 update_line_offset(annot.S1);
1567 break;
1568 case BinaryAnnotationsOpCode::ChangeCodeLength:
1569 update_code_offset(annot.U1);
1570 code_offset += annot.U1;
1571 is_terminal_entry = true;
1572 break;
1573 case BinaryAnnotationsOpCode::ChangeCodeOffsetAndLineOffset:
1574 code_offset += annot.U1;
1575 update_code_offset(annot.U1);
1576 update_line_offset(annot.S1);
1577 break;
1578 case BinaryAnnotationsOpCode::ChangeCodeLengthAndCodeOffset:
1579 code_offset += annot.U2;
1580 update_code_offset(annot.U2);
1581 update_code_offset(annot.U1);
1582 code_offset += annot.U1;
1583 is_terminal_entry = true;
1584 break;
1585 case BinaryAnnotationsOpCode::ChangeFile:
1586 update_file_offset(annot.U1);
1587 break;
1588 default:
1589 break;
1590 }
1591
1592 // Add range if current range is finished.
1593 if (code_offset_base && code_offset_end && cur_line_offset) {
1594 inline_site_sp->ranges.Append(RangeSourceLineVector::Entry(
1595 *code_offset_base, *code_offset_end - *code_offset_base,
1596 decl_line + *cur_line_offset));
1597 // Set base, end, file offset and line offset for next range.
1598 if (next_file_offset)
1599 file_offset = *next_file_offset;
1600 if (next_line_offset) {
1601 cur_line_offset = next_line_offset;
1602 next_line_offset = std::nullopt;
1603 }
1604 code_offset_base = is_terminal_entry ? std::nullopt : code_offset_end;
1605 code_offset_end = next_file_offset = std::nullopt;
1606 }
1607 if (code_offset_base && cur_line_offset) {
1608 if (is_terminal_entry) {
1609 LineTable::Entry line_entry(
1610 func_base + *code_offset_base, decl_line + *cur_line_offset, 0,
1611 file_offset, false, false, false, false, true);
1612 inline_site_sp->line_entries.push_back(line_entry);
1613 } else {
1614 LineTable::Entry line_entry(func_base + *code_offset_base,
1615 decl_line + *cur_line_offset, 0,
1616 file_offset, is_start_of_statement, false,
1617 is_prologue_end, false, false);
1618 inline_site_sp->line_entries.push_back(line_entry);
1619 is_prologue_end = false;
1620 is_start_of_statement = false;
1621 }
1622 }
1623 if (is_terminal_entry)
1624 is_start_of_statement = true;
1625 is_terminal_entry = false;
1626 }
1627
1628 inline_site_sp->ranges.Sort();
1629
1630 // Get the inlined function callsite info.
1631 std::unique_ptr<Declaration> callsite_up;
1632 if (!inline_site_sp->ranges.IsEmpty()) {
1633 auto *entry = inline_site_sp->ranges.GetEntryAtIndex(0);
1634 addr_t base_offset = entry->GetRangeBase();
1635 if (cii->m_debug_stream.readSymbolAtOffset(parent_id.offset).kind() ==
1636 S_INLINESITE) {
1637 // Its parent is another inline site, lookup parent site's range vector
1638 // for callsite line.
1639 ParseInlineSite(parent_id, func_base);
1640 std::shared_ptr<InlineSite> parent_site =
1641 m_inline_sites[toOpaqueUid(parent_id)];
1642 FileSpec &parent_decl_file =
1643 parent_site->inline_function_info->GetDeclaration().GetFile();
1644 if (auto *parent_entry =
1645 parent_site->ranges.FindEntryThatContains(base_offset)) {
1646 callsite_up =
1647 std::make_unique<Declaration>(parent_decl_file, parent_entry->data);
1648 }
1649 } else {
1650 // Its parent is a function, lookup global line table for callsite.
1651 if (auto *entry = cii->m_global_line_table.FindEntryThatContains(
1652 func_base + base_offset)) {
1653 const FileSpec &callsite_file =
1654 files.GetFileSpecAtIndex(entry->data.first);
1655 callsite_up =
1656 std::make_unique<Declaration>(callsite_file, entry->data.second);
1657 }
1658 }
1659 }
1660
1661 // Get the inlined function name.
1662 CVType inlinee_cvt = m_index->ipi().getType(inline_site.Inlinee);
1663 std::string inlinee_name;
1664 if (inlinee_cvt.kind() == LF_MFUNC_ID) {
1665 MemberFuncIdRecord mfr;
1666 cantFail(
1667 TypeDeserializer::deserializeAs<MemberFuncIdRecord>(inlinee_cvt, mfr));
1668 LazyRandomTypeCollection &types = m_index->tpi().typeCollection();
1669 inlinee_name.append(std::string(types.getTypeName(mfr.ClassType)));
1670 inlinee_name.append("::");
1671 inlinee_name.append(mfr.getName().str());
1672 } else if (inlinee_cvt.kind() == LF_FUNC_ID) {
1673 FuncIdRecord fir;
1674 cantFail(TypeDeserializer::deserializeAs<FuncIdRecord>(inlinee_cvt, fir));
1675 TypeIndex parent_idx = fir.getParentScope();
1676 if (!parent_idx.isNoneType()) {
1677 LazyRandomTypeCollection &ids = m_index->ipi().typeCollection();
1678 inlinee_name.append(std::string(ids.getTypeName(parent_idx)));
1679 inlinee_name.append("::");
1680 }
1681 inlinee_name.append(fir.getName().str());
1682 }
1683 inline_site_sp->inline_function_info = std::make_shared<InlineFunctionInfo>(
1684 inlinee_name.c_str(), llvm::StringRef(), decl_up.get(),
1685 callsite_up.get());
1686
1687 m_inline_sites[opaque_uid] = inline_site_sp;
1688}
1689
1691 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
1692 PdbCompilandSymId func_id = PdbSymUid(func.GetID()).asCompilandSym();
1693 // After we iterate through inline sites inside the function, we already get
1694 // all the info needed, removing from the map to save memory.
1695 std::set<uint64_t> remove_uids;
1696 auto parse_blocks = [&](SymbolKind kind, PdbCompilandSymId id) {
1697 if (kind == S_GPROC32 || kind == S_LPROC32 || kind == S_BLOCK32 ||
1698 kind == S_INLINESITE) {
1699 GetOrCreateBlock(id);
1700 if (kind == S_INLINESITE)
1701 remove_uids.insert(toOpaqueUid(id));
1702 return true;
1703 }
1704 return false;
1705 };
1706 size_t count = ParseSymbolArrayInScope(func_id, parse_blocks);
1707 for (uint64_t uid : remove_uids) {
1708 m_inline_sites.erase(uid);
1709 }
1710
1711 func.GetBlock(false).SetBlockInfoHasBeenParsed(true, true);
1712 return count;
1713}
1714
1716 PdbCompilandSymId parent_id,
1717 llvm::function_ref<bool(SymbolKind, PdbCompilandSymId)> fn) {
1718 CompilandIndexItem *cii = m_index->compilands().GetCompiland(parent_id.modi);
1719 CVSymbolArray syms =
1720 cii->m_debug_stream.getSymbolArrayForScope(parent_id.offset);
1721
1722 size_t count = 1;
1723 for (auto iter = syms.begin(); iter != syms.end(); ++iter) {
1724 PdbCompilandSymId child_id(parent_id.modi, iter.offset());
1725 if (fn(iter->kind(), child_id))
1726 ++count;
1727 }
1728
1729 return count;
1730}
1731
1732void SymbolFileNativePDB::DumpClangAST(Stream &s, llvm::StringRef filter,
1733 bool show_color) {
1735 if (!ts_or_err)
1736 return;
1737 auto ts = *ts_or_err;
1738 TypeSystemClang *clang = llvm::dyn_cast_or_null<TypeSystemClang>(ts.get());
1739 if (!clang)
1740 return;
1741 clang->GetNativePDBParser()->Dump(s, filter, show_color);
1742}
1743
1745 if (!m_func_full_names.IsEmpty() || !m_global_variable_base_names.IsEmpty())
1746 return;
1747
1748 // (segment, code offset) -> gid
1749 std::map<std::pair<uint16_t, uint32_t>, uint32_t> func_addr_ids;
1750
1751 // First, look through all items in the globals table.
1752 for (const uint32_t gid : m_index->globals().getGlobalsTable()) {
1753 CVSymbol sym = m_index->symrecords().readRecord(gid);
1754 auto kind = sym.kind();
1755
1756 // If this is a global variable, we only need to look at the name
1757 llvm::StringRef name;
1758 switch (kind) {
1759 case SymbolKind::S_GDATA32:
1760 case SymbolKind::S_LDATA32: {
1761 DataSym data =
1762 llvm::cantFail(SymbolDeserializer::deserializeAs<DataSym>(sym));
1763 name = data.Name;
1764 break;
1765 }
1766 case SymbolKind::S_GTHREAD32:
1767 case SymbolKind::S_LTHREAD32: {
1768 ThreadLocalDataSym data = llvm::cantFail(
1769 SymbolDeserializer::deserializeAs<ThreadLocalDataSym>(sym));
1770 name = data.Name;
1771 break;
1772 }
1773 case SymbolKind::S_CONSTANT: {
1774 ConstantSym data =
1775 llvm::cantFail(SymbolDeserializer::deserializeAs<ConstantSym>(sym));
1776 name = data.Name;
1777 break;
1778 }
1779 default:
1780 break;
1781 }
1782
1783 if (!name.empty()) {
1784 llvm::StringRef base = MSVCUndecoratedNameParser::DropScope(name);
1785 if (base.empty())
1786 base = name;
1787
1788 m_global_variable_base_names.Append(ConstString(base), gid);
1789 continue;
1790 }
1791
1792 if (kind != S_PROCREF && kind != S_LPROCREF)
1793 continue;
1794
1795 // For functions, we need to follow the reference to the procedure and look
1796 // at the type
1797
1798 ProcRefSym ref =
1799 llvm::cantFail(SymbolDeserializer::deserializeAs<ProcRefSym>(sym));
1800 if (ref.Name.empty())
1801 continue;
1802
1803 // Find the function this is referencing.
1804 CompilandIndexItem &cci =
1805 m_index->compilands().GetOrCreateCompiland(ref.modi());
1806 auto iter = cci.m_debug_stream.getSymbolArray().at(ref.SymOffset);
1807 if (iter == cci.m_debug_stream.getSymbolArray().end())
1808 continue;
1809 kind = iter->kind();
1810 if (kind != S_GPROC32 && kind != S_LPROC32)
1811 continue;
1812
1813 ProcSym proc =
1814 llvm::cantFail(SymbolDeserializer::deserializeAs<ProcSym>(*iter));
1815 if ((proc.Flags & ProcSymFlags::IsUnreachable) != ProcSymFlags::None)
1816 continue;
1817 if (proc.Name.empty() || proc.FunctionType.isSimple())
1818 continue;
1819
1820 // The function/procedure symbol only contains the demangled name.
1821 // The mangled names are in the publics table. Save the address of this
1822 // function to lookup the mangled name later.
1823 func_addr_ids.emplace(std::make_pair(proc.Segment, proc.CodeOffset), gid);
1824
1825 llvm::StringRef basename = MSVCUndecoratedNameParser::DropScope(proc.Name);
1826 if (basename.empty())
1827 basename = proc.Name;
1828
1829 m_func_base_names.Append(ConstString(basename), gid);
1830 m_func_full_names.Append(ConstString(proc.Name), gid);
1831
1832 // To see if this is a member function, check the type.
1833 auto type = m_index->tpi().getType(proc.FunctionType);
1834 if (type.kind() == LF_MFUNCTION) {
1835 MemberFunctionRecord mfr;
1836 llvm::cantFail(
1837 TypeDeserializer::deserializeAs<MemberFunctionRecord>(type, mfr));
1838 if (!mfr.getThisType().isNoneType())
1839 m_func_method_names.Append(ConstString(basename), gid);
1840 }
1841 }
1842
1843 // The publics stream contains all mangled function names and their address.
1844 for (auto pid : m_index->publics().getPublicsTable()) {
1845 PdbGlobalSymId global{pid, true};
1846 CVSymbol sym = m_index->ReadSymbolRecord(global);
1847 auto kind = sym.kind();
1848 if (kind != S_PUB32)
1849 continue;
1850 PublicSym32 pub =
1851 llvm::cantFail(SymbolDeserializer::deserializeAs<PublicSym32>(sym));
1852 // We only care about mangled names - if the name isn't mangled, it's
1853 // already in the full name map.
1854 if (!Mangled::IsMangledName(pub.Name))
1855 continue;
1856
1857 // Check if this symbol is for one of our functions.
1858 auto it = func_addr_ids.find({pub.Segment, pub.Offset});
1859 if (it != func_addr_ids.end())
1860 m_func_full_names.Append(ConstString(pub.Name), it->second);
1861 }
1862
1863 // Sort them before value searching is working properly.
1864 m_func_full_names.Sort(std::less<uint32_t>());
1865 m_func_full_names.SizeToFit();
1866 m_func_method_names.Sort(std::less<uint32_t>());
1867 m_func_method_names.SizeToFit();
1868 m_func_base_names.Sort(std::less<uint32_t>());
1869 m_func_base_names.SizeToFit();
1870 m_global_variable_base_names.Sort(std::less<uint32_t>());
1871 m_global_variable_base_names.SizeToFit();
1872}
1873
1875 ConstString name, const CompilerDeclContext &parent_decl_ctx,
1876 uint32_t max_matches, VariableList &variables) {
1877 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
1878
1880
1881 std::vector<uint32_t> results;
1882 m_global_variable_base_names.GetValues(name, results);
1883
1884 size_t n_matches = 0;
1885 for (uint32_t gid : results) {
1886 PdbGlobalSymId global(gid, false);
1887
1888 if (parent_decl_ctx.IsValid() &&
1889 GetDeclContextContainingUID(toOpaqueUid(global)) != parent_decl_ctx)
1890 continue;
1891
1893 if (!var)
1894 continue;
1895 variables.AddVariable(var);
1896
1897 if (++n_matches >= max_matches)
1898 break;
1899 }
1900}
1901
1903 const Module::LookupInfo &lookup_info,
1904 const CompilerDeclContext &parent_decl_ctx, bool include_inlines,
1905 SymbolContextList &sc_list) {
1906 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
1907 ConstString name = lookup_info.GetLookupName();
1908 FunctionNameType name_type_mask = lookup_info.GetNameTypeMask();
1909 if (name_type_mask & eFunctionNameTypeFull)
1910 name = lookup_info.GetName();
1911
1912 if (!(name_type_mask & eFunctionNameTypeFull ||
1913 name_type_mask & eFunctionNameTypeBase ||
1914 name_type_mask & eFunctionNameTypeMethod))
1915 return;
1917
1918 std::set<uint32_t> resolved_ids; // avoid duplicate lookups
1919 auto resolve_from = [&](UniqueCStringMap<uint32_t> &Names) {
1920 std::vector<uint32_t> ids;
1921 if (!Names.GetValues(name, ids))
1922 return;
1923
1924 for (uint32_t id : ids) {
1925 if (!resolved_ids.insert(id).second)
1926 continue;
1927
1928 PdbGlobalSymId global{id, false};
1929 if (parent_decl_ctx.IsValid() &&
1930 GetDeclContextContainingUID(toOpaqueUid(global)) != parent_decl_ctx)
1931 continue;
1932
1933 CVSymbol sym = m_index->ReadSymbolRecord(global);
1934 auto kind = sym.kind();
1935 lldbassert(kind == S_PROCREF || kind == S_LPROCREF);
1936
1937 ProcRefSym proc =
1938 cantFail(SymbolDeserializer::deserializeAs<ProcRefSym>(sym));
1939
1940 if (!IsValidRecord(proc))
1941 continue;
1942
1943 CompilandIndexItem &cci =
1944 m_index->compilands().GetOrCreateCompiland(proc.modi());
1945 SymbolContext sc;
1946
1947 sc.comp_unit = GetOrCreateCompileUnit(cci).get();
1948 if (!sc.comp_unit)
1949 continue;
1950
1951 PdbCompilandSymId func_id(proc.modi(), proc.SymOffset);
1952 sc.function = GetOrCreateFunction(func_id, *sc.comp_unit).get();
1953 if (!sc.function)
1954 continue;
1955
1956 sc_list.Append(sc);
1957 }
1958 };
1959
1960 if (name_type_mask & eFunctionNameTypeFull)
1961 resolve_from(m_func_full_names);
1962 if (name_type_mask & eFunctionNameTypeBase)
1963 resolve_from(m_func_base_names);
1964 if (name_type_mask & eFunctionNameTypeMethod)
1965 resolve_from(m_func_method_names);
1966}
1967
1969 bool include_inlines,
1970 SymbolContextList &sc_list) {}
1971
1973 lldb_private::TypeResults &results) {
1974
1975 // Make sure we haven't already searched this SymbolFile before.
1976 if (results.AlreadySearched(this))
1977 return;
1978
1979 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
1980
1981 // We can't query for the full name because the type might reside
1982 // in an anonymous namespace. Search for the basename in our map and check the
1983 // matching types afterwards.
1984 std::vector<uint32_t> matches;
1985 m_type_base_names.GetValues(query.GetTypeBasename(), matches);
1986
1987 for (uint32_t match_idx : matches) {
1988 std::vector context = GetContextForType(TypeIndex(match_idx));
1989 if (context.empty())
1990 continue;
1991
1992 if (query.ContextMatches(context)) {
1993 TypeSP type_sp = GetOrCreateType(TypeIndex(match_idx));
1994 if (!type_sp)
1995 continue;
1996
1997 results.InsertUnique(type_sp);
1998 if (results.Done(query))
1999 return;
2000 }
2001 }
2002}
2003
2005 uint32_t max_matches,
2006 TypeMap &types) {
2007
2008 std::vector<TypeIndex> matches = m_index->tpi().findRecordsByName(name);
2009 if (max_matches > 0 && max_matches < matches.size())
2010 matches.resize(max_matches);
2011
2012 for (TypeIndex ti : matches) {
2013 TypeSP type = GetOrCreateType(ti);
2014 if (!type)
2015 continue;
2016
2017 types.Insert(type);
2018 }
2019}
2020
2022 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
2023 // Only do the full type scan the first time.
2025 return 0;
2026
2027 const size_t old_count = GetTypeList().GetSize();
2028 LazyRandomTypeCollection &types = m_index->tpi().typeCollection();
2029
2030 // First process the entire TPI stream.
2031 for (auto ti = types.getFirst(); ti; ti = types.getNext(*ti)) {
2032 TypeSP type = GetOrCreateType(*ti);
2033 if (type)
2034 (void)type->GetFullCompilerType();
2035 }
2036
2037 // Next look for S_UDT records in the globals stream.
2038 for (const uint32_t gid : m_index->globals().getGlobalsTable()) {
2039 PdbGlobalSymId global{gid, false};
2040 CVSymbol sym = m_index->ReadSymbolRecord(global);
2041 if (sym.kind() != S_UDT)
2042 continue;
2043
2044 UDTSym udt = llvm::cantFail(SymbolDeserializer::deserializeAs<UDTSym>(sym));
2045 bool is_typedef = true;
2046 if (IsTagRecord(PdbTypeSymId{udt.Type, false}, m_index->tpi())) {
2047 CVType cvt = m_index->tpi().getType(udt.Type);
2048 llvm::StringRef name = CVTagRecord::create(cvt).name();
2049 if (name == udt.Name)
2050 is_typedef = false;
2051 }
2052
2053 if (is_typedef)
2054 GetOrCreateTypedef(global);
2055 }
2056
2057 const size_t new_count = GetTypeList().GetSize();
2058
2059 m_done_full_type_scan = true;
2060
2061 return new_count - old_count;
2062}
2063
2064size_t
2066 VariableList &variables) {
2067 PdbSymUid sym_uid(comp_unit.GetID());
2069 for (const uint32_t gid : m_index->globals().getGlobalsTable()) {
2070 PdbGlobalSymId global{gid, false};
2071 CVSymbol sym = m_index->ReadSymbolRecord(global);
2072 // TODO: S_CONSTANT is not handled here to prevent a possible crash in
2073 // lldb_private::npdb::MakeConstantLocationExpression when it's a record
2074 // type (e.g. std::strong_ordering::equal). That function needs to be
2075 // updated to handle this case when we add S_CONSTANT case here.
2076 switch (sym.kind()) {
2077 case SymbolKind::S_GDATA32:
2078 case SymbolKind::S_LDATA32:
2079 case SymbolKind::S_GTHREAD32:
2080 case SymbolKind::S_LTHREAD32: {
2081 if (VariableSP var = GetOrCreateGlobalVariable(global))
2082 variables.AddVariable(var);
2083 break;
2084 }
2085 default:
2086 break;
2087 }
2088 }
2089 return variables.GetSize();
2090}
2091
2093 PdbCompilandSymId var_id,
2094 bool is_param) {
2095 ModuleSP module = GetObjectFile()->GetModule();
2096 Block *block = GetOrCreateBlock(scope_id);
2097 if (!block)
2098 return nullptr;
2099
2100 // Get function block.
2101 Block *func_block = block;
2102 while (func_block->GetParent()) {
2103 func_block = func_block->GetParent();
2104 }
2105
2106 Address addr;
2107 func_block->GetStartAddress(addr);
2108 VariableInfo var_info =
2109 GetVariableLocationInfo(*m_index, var_id, *func_block, module);
2110 Function *func = func_block->CalculateSymbolContextFunction();
2111 if (!func)
2112 return nullptr;
2113 // Use empty dwarf expr if optimized away so that it won't be filtered out
2114 // when lookuping local variables in this scope.
2115 if (!var_info.location.IsValid())
2116 var_info.location = DWARFExpressionList(module, DWARFExpression(), nullptr);
2118 CompilandIndexItem *cii = m_index->compilands().GetCompiland(var_id.modi);
2119 CompUnitSP comp_unit_sp = GetOrCreateCompileUnit(*cii);
2120 TypeSP type_sp = GetOrCreateType(var_info.type);
2121 if (!type_sp)
2122 return nullptr;
2123 std::string name = var_info.name.str();
2124 Declaration decl;
2125 SymbolFileTypeSP sftype =
2126 std::make_shared<SymbolFileType>(*this, type_sp->GetID());
2127
2128 is_param |= var_info.is_param;
2129 ValueType var_scope =
2131 bool external = false;
2132 bool artificial = false;
2133 bool location_is_constant_data = false;
2134 bool static_member = false;
2135 Variable::RangeList scope_ranges;
2136 VariableSP var_sp = std::make_shared<Variable>(
2137 toOpaqueUid(var_id), name.c_str(), name.c_str(), sftype, var_scope, block,
2138 scope_ranges, &decl, var_info.location, external, artificial,
2139 location_is_constant_data, static_member);
2140 if (!is_param) {
2141 auto ts_or_err = GetTypeSystemForLanguage(comp_unit_sp->GetLanguage());
2142 if (auto err = ts_or_err.takeError())
2143 return nullptr;
2144 auto ts = *ts_or_err;
2145 if (!ts)
2146 return nullptr;
2147
2148 ts->GetNativePDBParser()->GetOrCreateVariableDecl(scope_id, var_id);
2149 }
2150 m_local_variables[toOpaqueUid(var_id)] = var_sp;
2151 return var_sp;
2152}
2153
2155 PdbCompilandSymId scope_id, PdbCompilandSymId var_id, bool is_param) {
2156 auto iter = m_local_variables.find(toOpaqueUid(var_id));
2157 if (iter != m_local_variables.end())
2158 return iter->second;
2159
2160 return CreateLocalVariable(scope_id, var_id, is_param);
2161}
2162
2164 CVSymbol sym = m_index->ReadSymbolRecord(id);
2165 lldbassert(sym.kind() == SymbolKind::S_UDT);
2166
2167 UDTSym udt = llvm::cantFail(SymbolDeserializer::deserializeAs<UDTSym>(sym));
2168
2169 TypeSP target_type = GetOrCreateType(udt.Type);
2170
2172 if (auto err = ts_or_err.takeError())
2173 return nullptr;
2174 auto ts = *ts_or_err;
2175 if (!ts)
2176 return nullptr;
2177
2178 auto *typedef_decl = ts->GetNativePDBParser()->GetOrCreateTypedefDecl(id);
2179
2180 CompilerType ct = target_type->GetForwardCompilerType();
2181 if (auto *clang = llvm::dyn_cast_or_null<TypeSystemClang>(ts.get()))
2182 ct = clang->GetType(clang->getASTContext().getTypeDeclType(typedef_decl));
2183
2184 Declaration decl;
2185 return MakeType(toOpaqueUid(id), ConstString(udt.Name),
2186 llvm::expectedToOptional(target_type->GetByteSize(nullptr)),
2187 nullptr, target_type->GetID(),
2190}
2191
2193 auto iter = m_types.find(toOpaqueUid(id));
2194 if (iter != m_types.end())
2195 return iter->second;
2196
2197 return CreateTypedef(id);
2198}
2199
2201 Block *block = GetOrCreateBlock(block_id);
2202 if (!block)
2203 return 0;
2204
2205 size_t count = 0;
2206
2207 CompilandIndexItem *cii = m_index->compilands().GetCompiland(block_id.modi);
2208 CVSymbol sym = cii->m_debug_stream.readSymbolAtOffset(block_id.offset);
2209 uint32_t params_remaining = 0;
2210 switch (sym.kind()) {
2211 case S_GPROC32:
2212 case S_LPROC32: {
2213 ProcSym proc(static_cast<SymbolRecordKind>(sym.kind()));
2214 cantFail(SymbolDeserializer::deserializeAs<ProcSym>(sym, proc));
2215 CVType signature = m_index->tpi().getType(proc.FunctionType);
2216 if (signature.kind() == LF_PROCEDURE) {
2217 ProcedureRecord sig;
2218 if (llvm::Error e = TypeDeserializer::deserializeAs<ProcedureRecord>(
2219 signature, sig)) {
2220 llvm::consumeError(std::move(e));
2221 return 0;
2222 }
2223 params_remaining = sig.getParameterCount();
2224 } else if (signature.kind() == LF_MFUNCTION) {
2225 MemberFunctionRecord sig;
2226 if (llvm::Error e = TypeDeserializer::deserializeAs<MemberFunctionRecord>(
2227 signature, sig)) {
2228 llvm::consumeError(std::move(e));
2229 return 0;
2230 }
2231 params_remaining = sig.getParameterCount();
2232 } else
2233 return 0;
2234 break;
2235 }
2236 case S_BLOCK32:
2237 break;
2238 case S_INLINESITE:
2239 break;
2240 default:
2241 lldbassert(false && "Symbol is not a block!");
2242 return 0;
2243 }
2244
2245 VariableListSP variables = block->GetBlockVariableList(false);
2246 if (!variables) {
2247 variables = std::make_shared<VariableList>();
2248 block->SetVariableList(variables);
2249 }
2250
2251 CVSymbolArray syms = limitSymbolArrayToScope(
2252 cii->m_debug_stream.getSymbolArray(), block_id.offset);
2253
2254 // Skip the first record since it's a PROC32 or BLOCK32, and there's
2255 // no point examining it since we know it's not a local variable.
2256 syms.drop_front();
2257 auto iter = syms.begin();
2258 auto end = syms.end();
2259
2260 while (iter != end) {
2261 uint32_t record_offset = iter.offset();
2262 CVSymbol variable_cvs = *iter;
2263 PdbCompilandSymId child_sym_id(block_id.modi, record_offset);
2264 ++iter;
2265
2266 // If this is a block or inline site, recurse into its children and then
2267 // skip it.
2268 if (variable_cvs.kind() == S_BLOCK32 ||
2269 variable_cvs.kind() == S_INLINESITE) {
2270 uint32_t block_end = getScopeEndOffset(variable_cvs);
2271 count += ParseVariablesForBlock(child_sym_id);
2272 iter = syms.at(block_end);
2273 continue;
2274 }
2275
2276 bool is_param = params_remaining > 0;
2277 VariableSP variable;
2278 switch (variable_cvs.kind()) {
2279 case S_REGREL32:
2280 case S_REGISTER:
2281 case S_LOCAL:
2282 variable = GetOrCreateLocalVariable(block_id, child_sym_id, is_param);
2283 if (is_param)
2284 --params_remaining;
2285 if (variable)
2286 variables->AddVariableIfUnique(variable);
2287 break;
2288 default:
2289 break;
2290 }
2291 }
2292
2293 // Pass false for set_children, since we call this recursively so that the
2294 // children will call this for themselves.
2295 block->SetDidParseVariables(true, false);
2296
2297 return count;
2298}
2299
2301 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
2302 lldbassert(sc.function || sc.comp_unit);
2303
2304 VariableListSP variables;
2305 if (sc.block) {
2306 PdbSymUid block_id(sc.block->GetID());
2307
2308 size_t count = ParseVariablesForBlock(block_id.asCompilandSym());
2309 return count;
2310 }
2311
2312 if (sc.function) {
2313 PdbSymUid block_id(sc.function->GetID());
2314
2315 size_t count = ParseVariablesForBlock(block_id.asCompilandSym());
2316 return count;
2317 }
2318
2319 if (sc.comp_unit) {
2320 variables = sc.comp_unit->GetVariableList(false);
2321 if (!variables) {
2322 variables = std::make_shared<VariableList>();
2323 sc.comp_unit->SetVariableList(variables);
2324 }
2325 return ParseVariablesForCompileUnit(*sc.comp_unit, *variables);
2326 }
2327
2328 llvm_unreachable("Unreachable!");
2329}
2330
2333 if (auto err = ts_or_err.takeError())
2334 return CompilerDecl();
2335 auto ts = *ts_or_err;
2336 if (!ts)
2337 return {};
2338
2339 if (auto decl = ts->GetNativePDBParser()->GetOrCreateDeclForUid(uid))
2340 return *decl;
2341 return CompilerDecl();
2342}
2343
2347 if (auto err = ts_or_err.takeError())
2348 return {};
2349 auto ts = *ts_or_err;
2350 if (!ts)
2351 return {};
2352
2353 PdbAstBuilder *ast_builder = ts->GetNativePDBParser();
2354 clang::DeclContext *context =
2355 ast_builder->GetOrCreateDeclContextForUid(PdbSymUid(uid));
2356 if (!context)
2357 return {};
2358
2359 return ast_builder->ToCompilerDeclContext(*context);
2360}
2361
2365 if (auto err = ts_or_err.takeError())
2366 return CompilerDeclContext();
2367 auto ts = *ts_or_err;
2368 if (!ts)
2369 return {};
2370
2371 PdbAstBuilder *ast_builder = ts->GetNativePDBParser();
2372 clang::DeclContext *context = ast_builder->GetParentDeclContext(PdbSymUid(uid));
2373 if (!context)
2374 return CompilerDeclContext();
2375 return ast_builder->ToCompilerDeclContext(*context);
2376}
2377
2379 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
2380 auto iter = m_types.find(type_uid);
2381 // lldb should not be passing us non-sensical type uids. the only way it
2382 // could have a type uid in the first place is if we handed it out, in which
2383 // case we should know about the type. However, that doesn't mean we've
2384 // instantiated it yet. We can vend out a UID for a future type. So if the
2385 // type doesn't exist, let's instantiate it now.
2386 if (iter != m_types.end())
2387 return &*iter->second;
2388
2389 PdbSymUid uid(type_uid);
2391 PdbTypeSymId type_id = uid.asTypeSym();
2392 if (type_id.index.isNoneType())
2393 return nullptr;
2394
2395 TypeSP type_sp = CreateAndCacheType(type_id);
2396 if (!type_sp)
2397 return nullptr;
2398 return &*type_sp;
2399}
2400
2401std::optional<SymbolFile::ArrayInfo>
2403 lldb::user_id_t type_uid, const lldb_private::ExecutionContext *exe_ctx) {
2404 return std::nullopt;
2405}
2406
2408 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
2409 auto clang_type_system = compiler_type.GetTypeSystem<TypeSystemClang>();
2410 if (!clang_type_system)
2411 return false;
2412
2413 PdbAstBuilder *ast_builder =
2414 static_cast<PdbAstBuilder *>(clang_type_system->GetNativePDBParser());
2415 if (ast_builder &&
2416 ast_builder->GetClangASTImporter().CanImport(compiler_type))
2417 return ast_builder->GetClangASTImporter().CompleteType(compiler_type);
2418 clang::QualType qt =
2419 clang::QualType::getFromOpaquePtr(compiler_type.GetOpaqueQualType());
2420
2421 return ast_builder->CompleteType(qt);
2422}
2423
2425 TypeClass type_mask,
2426 lldb_private::TypeList &type_list) {}
2427
2430 const CompilerDeclContext &parent_decl_ctx,
2431 bool /* only_root_namespaces */) {
2432 std::lock_guard<std::recursive_mutex> guard(GetModuleMutex());
2434 if (auto err = ts_or_err.takeError())
2435 return {};
2436 auto ts = *ts_or_err;
2437 if (!ts)
2438 return {};
2439 auto *clang = llvm::dyn_cast_or_null<TypeSystemClang>(ts.get());
2440 if (!clang)
2441 return {};
2442
2443 PdbAstBuilder *ast_builder = clang->GetNativePDBParser();
2444 if (!ast_builder)
2445 return {};
2446
2447 clang::DeclContext *decl_context = nullptr;
2448 if (parent_decl_ctx)
2449 decl_context = static_cast<clang::DeclContext *>(
2450 parent_decl_ctx.GetOpaqueDeclContext());
2451
2452 auto *namespace_decl =
2453 ast_builder->FindNamespaceDecl(decl_context, name.GetStringRef());
2454 if (!namespace_decl)
2455 return CompilerDeclContext();
2456
2457 return clang->CreateDeclContext(namespace_decl);
2458}
2459
2460llvm::Expected<lldb::TypeSystemSP>
2462 auto type_system_or_err =
2463 m_objfile_sp->GetModule()->GetTypeSystemForLanguage(language);
2464 if (type_system_or_err)
2465 if (auto ts = *type_system_or_err)
2466 ts->SetSymbolFile(this);
2467 return type_system_or_err;
2468}
2469
2470uint64_t SymbolFileNativePDB::GetDebugInfoSize(bool load_all_debug_info) {
2471 // PDB files are a separate file that contains all debug info.
2472 return m_index->pdb().getFileSize();
2473}
2474
2476 LazyRandomTypeCollection &types = m_index->tpi().typeCollection();
2477
2478 llvm::DenseMap<TypeIndex, TypeIndex> forward_to_full;
2479 llvm::DenseMap<TypeIndex, TypeIndex> full_to_forward;
2480
2481 struct RecordIndices {
2482 TypeIndex forward;
2483 TypeIndex full;
2484 };
2485
2486 llvm::StringMap<RecordIndices> record_indices;
2487
2488 for (auto ti = types.getFirst(); ti; ti = types.getNext(*ti)) {
2489 CVType type = types.getType(*ti);
2490 if (!IsTagRecord(type))
2491 continue;
2492
2493 CVTagRecord tag = CVTagRecord::create(type);
2494
2495 RecordIndices &indices = record_indices[tag.asTag().getUniqueName()];
2496 if (tag.asTag().isForwardRef()) {
2497 indices.forward = *ti;
2498 } else {
2499 indices.full = *ti;
2500
2501 auto base_name = MSVCUndecoratedNameParser::DropScope(tag.name());
2502 m_type_base_names.Append(ConstString(base_name), ti->getIndex());
2503 }
2504
2505 if (indices.full != TypeIndex::None() &&
2506 indices.forward != TypeIndex::None()) {
2507 forward_to_full[indices.forward] = indices.full;
2508 full_to_forward[indices.full] = indices.forward;
2509 }
2510
2511 // We're looking for LF_NESTTYPE records in the field list, so ignore
2512 // forward references (no field list), and anything without a nested class
2513 // (since there won't be any LF_NESTTYPE records).
2514 if (tag.asTag().isForwardRef() || !tag.asTag().containsNestedClass())
2515 continue;
2516
2517 struct ProcessTpiStream : public TypeVisitorCallbacks {
2518 ProcessTpiStream(PdbIndex &index, TypeIndex parent,
2519 const CVTagRecord &parent_cvt,
2520 llvm::DenseMap<TypeIndex, TypeIndex> &parents)
2521 : index(index), parents(parents), parent(parent),
2522 parent_cvt(parent_cvt) {}
2523
2524 PdbIndex &index;
2525 llvm::DenseMap<TypeIndex, TypeIndex> &parents;
2526
2527 unsigned unnamed_type_index = 1;
2528 TypeIndex parent;
2529 const CVTagRecord &parent_cvt;
2530
2531 llvm::Error visitKnownMember(CVMemberRecord &CVR,
2532 NestedTypeRecord &Record) override {
2533 std::string unnamed_type_name;
2534 if (Record.Name.empty()) {
2535 unnamed_type_name =
2536 llvm::formatv("<unnamed-type-$S{0}>", unnamed_type_index).str();
2537 Record.Name = unnamed_type_name;
2538 ++unnamed_type_index;
2539 }
2540 std::optional<CVTagRecord> tag =
2541 GetNestedTagDefinition(Record, parent_cvt, index.tpi());
2542 if (!tag)
2543 return llvm::ErrorSuccess();
2544
2545 parents[Record.Type] = parent;
2546 return llvm::ErrorSuccess();
2547 }
2548 };
2549
2550 CVType field_list_cvt = m_index->tpi().getType(tag.asTag().FieldList);
2551 ProcessTpiStream process(*m_index, *ti, tag, m_parent_types);
2552 FieldListRecord field_list;
2553 if (llvm::Error error = TypeDeserializer::deserializeAs<FieldListRecord>(
2554 field_list_cvt, field_list))
2555 llvm::consumeError(std::move(error));
2556 if (llvm::Error error = visitMemberRecordStream(field_list.Data, process))
2557 llvm::consumeError(std::move(error));
2558 }
2559
2560 // After calling Append(), the type-name map needs to be sorted again to be
2561 // able to look up a type by its name.
2562 m_type_base_names.Sort(std::less<uint32_t>());
2563
2564 // Now that we know the forward -> full mapping of all type indices, we can
2565 // re-write all the indices. At the end of this process, we want a mapping
2566 // consisting of fwd -> full and full -> full for all child -> parent indices.
2567 // We can re-write the values in place, but for the keys, we must save them
2568 // off so that we don't modify the map in place while also iterating it.
2569 std::vector<TypeIndex> full_keys;
2570 std::vector<TypeIndex> fwd_keys;
2571 for (auto &entry : m_parent_types) {
2572 TypeIndex key = entry.first;
2573 TypeIndex value = entry.second;
2574
2575 auto iter = forward_to_full.find(value);
2576 if (iter != forward_to_full.end())
2577 entry.second = iter->second;
2578
2579 iter = forward_to_full.find(key);
2580 if (iter != forward_to_full.end())
2581 fwd_keys.push_back(key);
2582 else
2583 full_keys.push_back(key);
2584 }
2585 for (TypeIndex fwd : fwd_keys) {
2586 TypeIndex full = forward_to_full[fwd];
2587 TypeIndex parent_idx = m_parent_types[fwd];
2588 m_parent_types[full] = parent_idx;
2589 }
2590 for (TypeIndex full : full_keys) {
2591 TypeIndex fwd = full_to_forward[full];
2592 m_parent_types[fwd] = m_parent_types[full];
2593 }
2594}
2595
2596std::optional<PdbCompilandSymId>
2598 CVSymbol sym = m_index->ReadSymbolRecord(id);
2599 if (symbolOpensScope(sym.kind())) {
2600 // If this exact symbol opens a scope, we can just directly access its
2601 // parent.
2602 id.offset = getScopeParentOffset(sym);
2603 // Global symbols have parent offset of 0. Return std::nullopt to indicate
2604 // this.
2605 if (id.offset == 0)
2606 return std::nullopt;
2607 return id;
2608 }
2609
2610 // Otherwise we need to start at the beginning and iterate forward until we
2611 // reach (or pass) this particular symbol
2612 CompilandIndexItem &cii = m_index->compilands().GetOrCreateCompiland(id.modi);
2613 const CVSymbolArray &syms = cii.m_debug_stream.getSymbolArray();
2614
2615 auto begin = syms.begin();
2616 auto end = syms.at(id.offset);
2617 std::vector<PdbCompilandSymId> scope_stack;
2618
2619 while (begin != end) {
2620 if (begin.offset() > id.offset) {
2621 // We passed it. We couldn't even find this symbol record.
2622 lldbassert(false && "Invalid compiland symbol id!");
2623 return std::nullopt;
2624 }
2625
2626 // We haven't found the symbol yet. Check if we need to open or close the
2627 // scope stack.
2628 if (symbolOpensScope(begin->kind())) {
2629 // We can use the end offset of the scope to determine whether or not
2630 // we can just outright skip this entire scope.
2631 uint32_t scope_end = getScopeEndOffset(*begin);
2632 if (scope_end < id.offset) {
2633 begin = syms.at(scope_end);
2634 } else {
2635 // The symbol we're looking for is somewhere in this scope.
2636 scope_stack.emplace_back(id.modi, begin.offset());
2637 }
2638 } else if (symbolEndsScope(begin->kind())) {
2639 scope_stack.pop_back();
2640 }
2641 ++begin;
2642 }
2643 if (scope_stack.empty())
2644 return std::nullopt;
2645 // We have a match! Return the top of the stack
2646 return scope_stack.back();
2647}
2648
2649std::optional<llvm::codeview::TypeIndex>
2650SymbolFileNativePDB::GetParentType(llvm::codeview::TypeIndex ti) {
2651 auto parent_iter = m_parent_types.find(ti);
2652 if (parent_iter == m_parent_types.end())
2653 return std::nullopt;
2654 return parent_iter->second;
2655}
2656
2657std::vector<CompilerContext>
2659 CVType type = m_index->tpi().getType(ti);
2660 if (!IsTagRecord(type))
2661 return {};
2662
2663 CVTagRecord tag = CVTagRecord::create(type);
2664
2665 std::optional<Type::ParsedName> parsed_name =
2667 if (!parsed_name)
2668 return {{tag.contextKind(), ConstString(tag.name())}};
2669
2670 std::vector<CompilerContext> ctx;
2671 // assume everything is a namespace at first
2672 for (llvm::StringRef scope : parsed_name->scope) {
2673 ctx.emplace_back(CompilerContextKind::Namespace, ConstString(scope));
2674 }
2675 // we know the kind of our own type
2676 ctx.emplace_back(tag.contextKind(), ConstString(parsed_name->basename));
2677
2678 // try to find the kind of parents
2679 for (auto &el : llvm::reverse(llvm::drop_end(ctx))) {
2680 std::optional<TypeIndex> parent = GetParentType(ti);
2681 if (!parent)
2682 break;
2683
2684 ti = *parent;
2685 type = m_index->tpi().getType(ti);
2686 switch (type.kind()) {
2687 case LF_CLASS:
2688 case LF_STRUCTURE:
2689 case LF_INTERFACE:
2691 continue;
2692 case LF_UNION:
2694 continue;
2695 case LF_ENUM:
2696 el.kind = CompilerContextKind::Enum;
2697 continue;
2698 default:
2699 break;
2700 }
2701 break;
2702 }
2703 return ctx;
2704}
2705
2706std::optional<llvm::StringRef>
2708 const CompilandIndexItem *cci =
2709 m_index->compilands().GetCompiland(func_id.modi);
2710 if (!cci)
2711 return std::nullopt;
2712
2713 CVSymbol sym_record = cci->m_debug_stream.readSymbolAtOffset(func_id.offset);
2714 if (sym_record.kind() != S_LPROC32 && sym_record.kind() != S_GPROC32)
2715 return std::nullopt;
2716
2717 ProcSym proc(static_cast<SymbolRecordKind>(sym_record.kind()));
2718 cantFail(SymbolDeserializer::deserializeAs<ProcSym>(sym_record, proc));
2719
2720 return FindMangledSymbol(SegmentOffset(proc.Segment, proc.CodeOffset),
2721 proc.FunctionType);
2722}
2723
2724std::optional<llvm::StringRef>
2726 TypeIndex function_type) {
2727 auto symbol = m_index->publics().findByAddress(m_index->symrecords(),
2728 so.segment, so.offset);
2729 if (!symbol)
2730 return std::nullopt;
2731
2732 llvm::StringRef name = symbol->first.Name;
2733 // For functions, we might need to strip the mangled name. See
2734 // StripMangledFunctionName for more info.
2735 if (!function_type.isNoneType() &&
2736 (symbol->first.Flags & PublicSymFlags::Function) != PublicSymFlags::None)
2737 name = StripMangledFunctionName(name, function_type);
2738
2739 return name;
2740}
2741
2742llvm::StringRef
2744 PdbTypeSymId func_ty) {
2745 // "In non-64 bit environments" (on x86 in pactice), __cdecl functions get
2746 // prefixed with an underscore. For compilers using LLVM, this happens in LLVM
2747 // (as opposed to the compiler frontend). Because of this, DWARF doesn't
2748 // contain the "full" mangled name in DW_AT_linkage_name for these functions.
2749 // We strip the mangling here for compatibility with DWARF. See
2750 // llvm.org/pr161676 and
2751 // https://learn.microsoft.com/en-us/cpp/build/reference/decorated-names#FormatC
2752
2753 if (!mangled.starts_with('_') ||
2754 m_index->dbi().getMachineType() != PDB_Machine::x86)
2755 return mangled;
2756
2757 CVType cvt = m_index->tpi().getType(func_ty.index);
2758 PDB_CallingConv cc = PDB_CallingConv::NearC;
2759 if (cvt.kind() == LF_PROCEDURE) {
2760 ProcedureRecord proc;
2761 if (llvm::Error error =
2762 TypeDeserializer::deserializeAs<ProcedureRecord>(cvt, proc))
2763 llvm::consumeError(std::move(error));
2764 cc = proc.CallConv;
2765 } else if (cvt.kind() == LF_MFUNCTION) {
2766 MemberFunctionRecord mfunc;
2767 if (llvm::Error error =
2768 TypeDeserializer::deserializeAs<MemberFunctionRecord>(cvt, mfunc))
2769 llvm::consumeError(std::move(error));
2770 cc = mfunc.CallConv;
2771 } else {
2772 LLDB_LOG(GetLog(LLDBLog::Symbols), "Unexpected function type, got {0}",
2773 cvt.kind());
2774 return mangled;
2775 }
2776
2777 if (cc == PDB_CallingConv::NearC || cc == PDB_CallingConv::FarC)
2778 return mangled.drop_front();
2779
2780 return mangled;
2781}
2782
2784 for (CVType cvt : m_index->ipi().typeArray()) {
2785 switch (cvt.kind()) {
2786 case LF_UDT_SRC_LINE: {
2787 UdtSourceLineRecord udt_src;
2788 llvm::cantFail(TypeDeserializer::deserializeAs(cvt, udt_src));
2789 m_udt_declarations.try_emplace(
2790 udt_src.UDT, UdtDeclaration{/*FileNameIndex=*/udt_src.SourceFile,
2791 /*IsIpiIndex=*/true,
2792 /*Line=*/udt_src.LineNumber});
2793 } break;
2794 case LF_UDT_MOD_SRC_LINE: {
2795 UdtModSourceLineRecord udt_mod_src;
2796 llvm::cantFail(TypeDeserializer::deserializeAs(cvt, udt_mod_src));
2797 // Some types might be contributed by multiple modules. We assume that
2798 // they all point to the same file and line because we can only provide
2799 // one location.
2800 m_udt_declarations.try_emplace(
2801 udt_mod_src.UDT,
2802 UdtDeclaration{/*FileNameIndex=*/udt_mod_src.SourceFile,
2803 /*IsIpiIndex=*/false,
2804 /*Line=*/udt_mod_src.LineNumber});
2805 } break;
2806 default:
2807 break;
2808 }
2809 }
2810}
2811
2812llvm::Expected<Declaration>
2814 std::call_once(m_cached_udt_declarations, [this] { CacheUdtDeclarations(); });
2815
2816 auto it = m_udt_declarations.find(type_id.index);
2817 if (it == m_udt_declarations.end())
2818 return llvm::createStringError("No UDT declaration found");
2819
2820 llvm::StringRef file_name;
2821 if (it->second.IsIpiIndex) {
2822 CVType cvt = m_index->ipi().getType(it->second.FileNameIndex);
2823 if (cvt.kind() != LF_STRING_ID)
2824 return llvm::createStringError("File name was not a LF_STRING_ID");
2825
2826 StringIdRecord sid;
2827 llvm::cantFail(TypeDeserializer::deserializeAs(cvt, sid));
2828 file_name = sid.String;
2829 } else {
2830 // The file name index is an index into the string table
2831 auto string_table = m_index->pdb().getStringTable();
2832 if (!string_table)
2833 return string_table.takeError();
2834
2835 llvm::Expected<llvm::StringRef> string =
2836 string_table->getStringTable().getString(
2837 it->second.FileNameIndex.getIndex());
2838 if (!string)
2839 return string.takeError();
2840 file_name = *string;
2841 }
2842
2843 // rustc sets the filename to "<unknown>" for some files
2844 if (file_name == "\\<unknown>")
2845 return Declaration();
2846
2847 return Declaration(FileSpec(file_name), it->second.Line);
2848}
static llvm::raw_ostream & error(Stream &strm)
#define lldbassert(x)
Definition LLDBAssert.h:16
#define LLDB_LOG(log,...)
The LLDB_LOG* macros defined below are the way to emit log messages.
Definition Log.h:369
#define LLDB_LOG_ERROR(log, error,...)
Definition Log.h:392
static std::unique_ptr< PDBFile > loadMatchingPDBFile(std::string exe_path, llvm::BumpPtrAllocator &allocator)
static std::optional< CVTagRecord > GetNestedTagDefinition(const NestedTypeRecord &Record, const CVTagRecord &parent, TpiStream &tpi)
static lldb::LanguageType TranslateLanguage(PDB_Lang lang)
static std::string GetUnqualifiedTypeName(const TagRecord &record)
static llvm::StringRef GetSimpleTypeName(SimpleTypeKind kind)
static bool IsClassRecord(TypeLeafKind kind)
static bool IsFunctionEpilogue(const CompilandIndexItem &cci, lldb::addr_t addr)
static bool NeedsResolvedCompileUnit(uint32_t resolve_scope)
static bool IsFunctionPrologue(const CompilandIndexItem &cci, lldb::addr_t addr)
static llvm::StringRef DropScope(llvm::StringRef name)
static bool UseNativePDB()
A section + offset based address class.
Definition Address.h:62
lldb::addr_t GetFileAddress() const
Get the file address.
Definition Address.cpp:281
bool IsValid() const
Check if the object state is valid.
Definition Address.h:355
A class that describes a single lexical block.
Definition Block.h:41
RangeList::Entry Range
Definition Block.h:44
lldb::VariableListSP GetBlockVariableList(bool can_create)
Get the variable list for this block only.
Definition Block.cpp:392
Block * FindInnermostBlockByOffset(const lldb::addr_t offset)
Definition Block.cpp:127
void SetBlockInfoHasBeenParsed(bool b, bool set_children)
Definition Block.cpp:479
lldb::BlockSP CreateChild(lldb::user_id_t uid)
Creates a block with the specified UID uid.
Definition Block.cpp:380
Function * CalculateSymbolContextFunction() override
Definition Block.cpp:150
void SetVariableList(lldb::VariableListSP &variable_list_sp)
Set accessor for the variable list.
Definition Block.h:319
Block * GetParent() const
Get the parent block.
Definition Block.cpp:202
bool GetStartAddress(Address &addr)
Definition Block.cpp:317
void SetDidParseVariables(bool b, bool set_children)
Definition Block.cpp:489
bool CanImport(const CompilerType &type)
Returns true iff the given type was copied from another TypeSystemClang and the original type in this...
bool CompleteType(const CompilerType &compiler_type)
A class that describes a compilation unit.
Definition CompileUnit.h:43
void SetVariableList(lldb::VariableListSP &variable_list_sp)
Set accessor for the variable list.
lldb::VariableListSP GetVariableList(bool can_create)
Get the variable list for a compile unit.
const FileSpec & GetPrimaryFile() const
Return the primary source spec associated with this compile unit.
void ResolveSymbolContext(const SourceLocationSpec &src_location_spec, lldb::SymbolContextItem resolve_scope, SymbolContextList &sc_list, RealpathPrefixes *realpath_prefixes=nullptr)
Resolve symbol contexts by file and line.
void SetLineTable(LineTable *line_table)
Set the line table for the compile unit.
void AddFunction(lldb::FunctionSP &function_sp)
Add a function to this compile unit.
size_t GetNumFunctions() const
Returns the number of functions in this compile unit.
lldb::LanguageType GetLanguage()
LineTable * GetLineTable()
Get the line table for the compile unit.
Represents a generic declaration context in a program.
Represents a generic declaration such as a function declaration.
Generic representation of a type in a programming language.
TypeSystemSPWrapper GetTypeSystem() const
Accessors.
lldb::opaque_compiler_type_t GetOpaqueQualType() const
A uniqued constant string class.
Definition ConstString.h:40
llvm::StringRef GetStringRef() const
Get the string value as a llvm::StringRef.
const char * GetCString() const
Get the string value as a C string.
"lldb/Expression/DWARFExpressionList.h" Encapsulates a range map from file address range to a single ...
bool IsValid() const
Return true if the location expression contains data.
void SetFuncFileAddress(lldb::addr_t func_file_addr)
"lldb/Expression/DWARFExpression.h" Encapsulates a DWARF location expression and interprets it.
A class to manage flag bits.
Definition Debugger.h:80
A class that describes the declaration location of a lldb object.
Definition Declaration.h:24
"lldb/Target/ExecutionContext.h" A class that contains an execution context.
A file utility class.
Definition FileSpec.h:57
static bool Match(const FileSpec &pattern, const FileSpec &file)
Match FileSpec pattern against FileSpec file.
Definition FileSpec.cpp:301
const ConstString & GetFilename() const
Filename string const get accessor.
Definition FileSpec.h:251
FileSpec CopyByRemovingLastPathComponent() const
Definition FileSpec.cpp:431
llvm::sys::path::Style Style
Definition FileSpec.h:59
static FileSystem & Instance()
A class that describes a function.
Definition Function.h:400
const Address & GetAddress() const
Return the address of the function (its entry point).
Definition Function.h:453
Block & GetBlock(bool can_create)
Get accessor for the block list.
Definition Function.cpp:382
static void AppendLineEntryToSequence(Sequence &sequence, lldb::addr_t file_addr, uint32_t line, uint16_t column, uint16_t file_idx, bool is_start_of_statement, bool is_start_of_basic_block, bool is_prologue_end, bool is_epilogue_begin, bool is_terminal_entry)
Definition LineTable.cpp:59
A class that handles mangled names.
Definition Mangled.h:34
static bool IsMangledName(llvm::StringRef name)
Definition Mangled.cpp:39
lldb::ModuleSP GetModule() const
Get const accessor for the module pointer.
A class that encapsulates name lookup information.
Definition Module.h:916
lldb::FunctionNameType GetNameTypeMask() const
Definition Module.h:931
ConstString GetLookupName() const
Definition Module.h:927
ConstString GetName() const
Definition Module.h:923
static std::unique_ptr< llvm::pdb::PDBFile > loadPDBFile(std::string PdbPath, llvm::BumpPtrAllocator &Allocator)
static bool RegisterPlugin(llvm::StringRef name, llvm::StringRef description, ABICreateInstance create_callback)
static bool UnregisterPlugin(ABICreateInstance create_callback)
RangeData< lldb::addr_t, uint32_t, std::pair< uint32_t, uint32_t > > Entry
Definition RangeMap.h:462
void Append(const Entry &entry)
Definition RangeMap.h:470
Entry * FindEntryThatContains(B addr)
Definition RangeMap.h:579
"lldb/Core/SourceLocationSpec.h" A source location specifier class.
A stream class that can stream formatted output to a file.
Definition Stream.h:28
A list of support files for a CompileUnit.
const FileSpec & GetFileSpecAtIndex(size_t idx) const
void Append(const FileSpec &file)
Defines a list of symbol context objects.
uint32_t GetSize() const
Get accessor for a symbol context list size.
void Append(const SymbolContext &sc)
Append a new symbol context to the list.
"lldb/Symbol/SymbolContextScope.h" Inherit from this if your object is part of a symbol context and c...
Defines a symbol context baton that can be handed other debug core functions.
Function * function
The Function for a given query.
Block * block
The Block for a given query.
CompileUnit * comp_unit
The CompileUnit for a given query.
LineEntry line_entry
The LineEntry for a given query.
lldb::CompUnitSP GetCompileUnitAtIndex(uint32_t idx) override
ObjectFile * GetObjectFile() override
Definition SymbolFile.h:566
virtual TypeList & GetTypeList()
Definition SymbolFile.h:639
lldb::ObjectFileSP m_objfile_sp
Definition SymbolFile.h:642
void SetCompileUnitAtIndex(uint32_t idx, const lldb::CompUnitSP &cu_sp)
SymbolFileCommon(lldb::ObjectFileSP objfile_sp)
Definition SymbolFile.h:551
uint32_t GetNumCompileUnits() override
lldb::TypeSP MakeType(lldb::user_id_t uid, ConstString name, std::optional< uint64_t > byte_size, SymbolContextScope *context, lldb::user_id_t encoding_uid, Type::EncodingDataType encoding_uid_type, const Declaration &decl, const CompilerType &compiler_qual_type, Type::ResolveState compiler_type_resolve_state, uint32_t opaque_payload=0) override
This function is used to create types that belong to a SymbolFile.
Definition SymbolFile.h:610
virtual std::recursive_mutex & GetModuleMutex() const
Symbols file subclasses should override this to return the Module that owns the TypeSystem that this ...
uint32_t AddSymbol(const Symbol &symbol)
Definition Symtab.cpp:64
uint32_t GetSize() const
Definition TypeList.cpp:60
void Insert(const lldb::TypeSP &type)
Definition TypeList.cpp:27
void Insert(const lldb::TypeSP &type)
Definition TypeMap.cpp:27
A class that contains all state required for type lookups.
Definition Type.h:104
ConstString GetTypeBasename() const
Get the type basename to use when searching the type indexes in each SymbolFile object.
Definition Type.cpp:113
bool ContextMatches(llvm::ArrayRef< lldb_private::CompilerContext > context) const
Check of a CompilerContext array from matching type from a symbol file matches the m_context.
Definition Type.cpp:129
This class tracks the state and results of a TypeQuery.
Definition Type.h:344
bool InsertUnique(const lldb::TypeSP &type_sp)
When types that match a TypeQuery are found, this API is used to insert the matching types.
Definition Type.cpp:194
bool Done(const TypeQuery &query) const
Check if the type matching has found all of the matches that it needs.
Definition Type.cpp:200
bool AlreadySearched(lldb_private::SymbolFile *sym_file)
Check if a SymbolFile object has already been searched by this type match object.
Definition Type.cpp:190
A TypeSystem implementation based on Clang.
Interface for representing a type system.
Definition TypeSystem.h:70
virtual npdb::PdbAstBuilder * GetNativePDBParser()
Definition TypeSystem.h:92
@ eEncodingIsTypedefUID
This type is alias to a type whose UID is m_encoding_uid.
Definition Type.h:434
@ eEncodingIsUID
This type is the type whose UID is m_encoding_uid.
Definition Type.h:423
static std::optional< ParsedName > GetTypeScopeAndBasename(llvm::StringRef name)
Definition Type.cpp:792
void AddVariable(const lldb::VariableSP &var_sp)
RangeVector< lldb::addr_t, lldb::addr_t > RangeList
Definition Variable.h:27
ClangASTImporter & GetClangASTImporter()
clang::DeclContext * GetParentDeclContext(PdbSymUid uid)
CompilerDeclContext ToCompilerDeclContext(clang::DeclContext &context)
clang::DeclContext * FromCompilerDeclContext(CompilerDeclContext context)
CompilerType ToCompilerType(clang::QualType qt)
clang::NamespaceDecl * FindNamespaceDecl(const clang::DeclContext *parent, llvm::StringRef name)
bool CompleteType(clang::QualType qt)
clang::DeclContext * GetOrCreateDeclContextForUid(PdbSymUid uid)
void ParseDeclsForContext(clang::DeclContext &context)
clang::BlockDecl * GetOrCreateBlockDecl(PdbCompilandSymId block_id)
clang::QualType GetOrCreateType(PdbTypeSymId type)
clang::FunctionDecl * GetOrCreateInlinedFunctionDecl(PdbCompilandSymId inlinesite_id)
PdbIndex - Lazy access to the important parts of a PDB file.
Definition PdbIndex.h:47
static llvm::Expected< std::unique_ptr< PdbIndex > > create(llvm::pdb::PDBFile *)
Definition PdbIndex.cpp:43
llvm::pdb::TpiStream & tpi()
Definition PdbIndex.h:124
PdbCompilandId asCompiland() const
PdbCompilandSymId asCompilandSym() const
PdbTypeSymId asTypeSym() const
PdbSymUidKind kind() const
lldb::VariableSP GetOrCreateGlobalVariable(PdbGlobalSymId var_id)
bool ParseLineTable(lldb_private::CompileUnit &comp_unit) override
lldb::TypeSP CreateArrayType(PdbTypeSymId type_id, const llvm::codeview::ArrayRecord &ar, CompilerType ct)
std::optional< ArrayInfo > GetDynamicArrayInfoForUID(lldb::user_id_t type_uid, const lldb_private::ExecutionContext *exe_ctx) override
If type_uid points to an array type, return its characteristics.
void CacheGlobalBaseNames()
Caches the basenames of symbols found in the globals stream.
llvm::Expected< Declaration > ResolveUdtDeclaration(PdbTypeSymId type_id)
lldb::VariableSP CreateGlobalVariable(PdbGlobalSymId var_id)
llvm::Expected< lldb::TypeSystemSP > GetTypeSystemForLanguage(lldb::LanguageType language) override
void InitializeObject() override
Initialize the SymbolFile object.
lldb_private::UniqueCStringMap< uint32_t > m_func_base_names
basename -> Global ID(s)
static SymbolFile * CreateInstance(lldb::ObjectFileSP objfile_sp)
llvm::DenseMap< lldb::user_id_t, lldb::TypeSP > m_types
bool CompleteType(CompilerType &compiler_type) override
lldb::LanguageType ParseLanguage(lldb_private::CompileUnit &comp_unit) override
CompilerDeclContext GetDeclContextForUID(lldb::user_id_t uid) override
void DumpClangAST(Stream &s, llvm::StringRef filter, bool show_color) override
lldb::VariableSP GetOrCreateLocalVariable(PdbCompilandSymId scope_id, PdbCompilandSymId var_id, bool is_param)
size_t ParseVariablesForContext(const SymbolContext &sc) override
size_t ParseFunctions(lldb_private::CompileUnit &comp_unit) override
lldb::TypeSP CreatePointerType(PdbTypeSymId type_id, const llvm::codeview::PointerRecord &pr, CompilerType ct)
lldb::FunctionSP CreateFunction(PdbCompilandSymId func_id, CompileUnit &comp_unit)
llvm::DenseMap< lldb::user_id_t, lldb::BlockSP > m_blocks
bool ParseSupportFiles(lldb_private::CompileUnit &comp_unit, SupportFileList &support_files) override
CompilerDecl GetDeclForUID(lldb::user_id_t uid) override
std::optional< llvm::StringRef > FindMangledFunctionName(PdbCompilandSymId id)
Find the mangled name for a function.
SymbolFileNativePDB(lldb::ObjectFileSP objfile_sp)
lldb::TypeSP GetOrCreateTypedef(PdbGlobalSymId id)
void FindTypesByName(llvm::StringRef name, uint32_t max_matches, TypeMap &types)
lldb::TypeSP CreateTagType(PdbTypeSymId type_id, const llvm::codeview::ClassRecord &cr, CompilerType ct)
lldb::TypeSP GetOrCreateType(PdbTypeSymId type_id)
void GetTypes(SymbolContextScope *sc_scope, lldb::TypeClass type_mask, TypeList &type_list) override
llvm::DenseMap< lldb::user_id_t, lldb::VariableSP > m_local_variables
lldb::VariableSP CreateConstantSymbol(PdbGlobalSymId var_id, const llvm::codeview::CVSymbol &cvs)
lldb::TypeSP CreateType(PdbTypeSymId type_id, CompilerType ct)
lldb_private::UniqueCStringMap< uint32_t > m_func_method_names
method basename -> Global ID(s)
std::optional< llvm::codeview::TypeIndex > GetParentType(llvm::codeview::TypeIndex ti)
lldb_private::UniqueCStringMap< uint32_t > m_global_variable_base_names
global variable basename -> Global ID(s)
void FindFunctions(const Module::LookupInfo &lookup_info, const CompilerDeclContext &parent_decl_ctx, bool include_inlines, SymbolContextList &sc_list) override
std::unique_ptr< llvm::pdb::PDBFile > m_file_up
lldb::VariableSP CreateLocalVariable(PdbCompilandSymId scope_id, PdbCompilandSymId var_id, bool is_param)
lldb::TypeSP CreateProcedureType(PdbTypeSymId type_id, const llvm::codeview::ProcedureRecord &pr, CompilerType ct)
lldb::TypeSP CreateModifierType(PdbTypeSymId type_id, const llvm::codeview::ModifierRecord &mr, CompilerType ct)
uint64_t GetDebugInfoSize(bool load_all_debug_info=false) override
Metrics gathering functions.
std::optional< llvm::StringRef > FindMangledSymbol(SegmentOffset so, llvm::codeview::TypeIndex function_type=llvm::codeview::TypeIndex())
Find a symbol name at a specific address (so).
size_t ParseTypes(lldb_private::CompileUnit &comp_unit) override
Block * GetOrCreateBlock(PdbCompilandSymId block_id)
size_t ParseBlocksRecursive(Function &func) override
lldb::CompUnitSP CreateCompileUnit(const CompilandIndexItem &cci)
std::optional< PdbCompilandSymId > FindSymbolScope(PdbCompilandSymId id)
size_t ParseSymbolArrayInScope(PdbCompilandSymId parent, llvm::function_ref< bool(llvm::codeview::SymbolKind, PdbCompilandSymId)> fn)
size_t ParseVariablesForCompileUnit(CompileUnit &comp_unit, VariableList &variables)
llvm::DenseMap< lldb::user_id_t, lldb::CompUnitSP > m_compilands
Block * CreateBlock(PdbCompilandSymId block_id)
std::vector< CompilerContext > GetContextForType(llvm::codeview::TypeIndex ti)
llvm::Expected< uint32_t > GetFileIndex(const CompilandIndexItem &cii, uint32_t file_id)
lldb::CompUnitSP GetOrCreateCompileUnit(const CompilandIndexItem &cci)
Type * ResolveTypeUID(lldb::user_id_t type_uid) override
llvm::DenseMap< lldb::user_id_t, lldb::FunctionSP > m_functions
bool ParseImportedModules(const SymbolContext &sc, std::vector< lldb_private::SourceModule > &imported_modules) override
llvm::StringRef StripMangledFunctionName(llvm::StringRef mangled, PdbTypeSymId func_ty)
static void DebuggerInitialize(Debugger &debugger)
llvm::DenseMap< lldb::user_id_t, std::shared_ptr< InlineSite > > m_inline_sites
void ParseInlineSite(PdbCompilandSymId inline_site_id, Address func_addr)
lldb::TypeSP CreateClassStructUnion(PdbTypeSymId type_id, const llvm::codeview::TagRecord &record, size_t size, CompilerType ct)
void FindGlobalVariables(ConstString name, const CompilerDeclContext &parent_decl_ctx, uint32_t max_matches, VariableList &variables) override
size_t ParseVariablesForBlock(PdbCompilandSymId block_id)
void ParseDeclsForContext(lldb_private::CompilerDeclContext decl_ctx) override
lldb::FunctionSP GetOrCreateFunction(PdbCompilandSymId func_id, CompileUnit &comp_unit)
llvm::DenseMap< llvm::codeview::TypeIndex, llvm::codeview::TypeIndex > m_parent_types
lldb_private::UniqueCStringMap< uint32_t > m_func_full_names
mangled name/full function name -> Global ID(s)
lldb::CompUnitSP ParseCompileUnitAtIndex(uint32_t index) override
lldb::TypeSP CreateFunctionType(PdbTypeSymId type_id, const llvm::codeview::MemberFunctionRecord &pr, CompilerType ct)
lldb_private::UniqueCStringMap< uint32_t > m_type_base_names
lldb::TypeSP CreateAndCacheType(PdbTypeSymId type_id)
CompilerDeclContext FindNamespace(ConstString name, const CompilerDeclContext &parent_decl_ctx, bool only_root_namespaces) override
Finds a namespace of name name and whose parent context is parent_decl_ctx.
CompilerDeclContext GetDeclContextContainingUID(lldb::user_id_t uid) override
bool ParseDebugMacros(lldb_private::CompileUnit &comp_unit) override
lldb::TypeSP CreateTypedef(PdbGlobalSymId id)
llvm::DenseMap< llvm::codeview::TypeIndex, UdtDeclaration > m_udt_declarations
void FindTypes(const lldb_private::TypeQuery &match, lldb_private::TypeResults &results) override
Find types using a type-matching object that contains all search parameters.
uint32_t ResolveSymbolContext(const Address &so_addr, lldb::SymbolContextItem resolve_scope, SymbolContext &sc) override
llvm::DenseMap< lldb::user_id_t, lldb::VariableSP > m_global_vars
lldb::TypeSP CreateSimpleType(llvm::codeview::TypeIndex ti, CompilerType ct)
#define LLDB_INVALID_UID
#define LLDB_INVALID_ADDRESS
uint64_t toOpaqueUid(const T &cid)
Definition PdbSymUid.h:114
size_t GetTypeSizeForSimpleKind(llvm::codeview::SimpleTypeKind kind)
SegmentOffsetLength GetSegmentOffsetAndLength(const llvm::codeview::CVSymbol &sym)
bool IsTagRecord(llvm::codeview::CVType cvt)
Definition PdbUtil.cpp:517
bool IsValidRecord(const RecordT &sym)
Definition PdbUtil.h:129
DWARFExpression MakeGlobalLocationExpression(uint16_t section, uint32_t offset, lldb::ModuleSP module)
VariableInfo GetVariableLocationInfo(PdbIndex &index, PdbCompilandSymId var_id, Block &func_block, lldb::ModuleSP module)
Definition PdbUtil.cpp:734
bool IsForwardRefUdt(llvm::codeview::CVType cvt)
llvm::pdb::PDB_SymType CVSymToPDBSym(llvm::codeview::SymbolKind kind)
DWARFExpression MakeConstantLocationExpression(llvm::codeview::TypeIndex underlying_ti, llvm::pdb::TpiStream &tpi, const llvm::APSInt &constant, lldb::ModuleSP module)
A class that represents a running process on the host machine.
Log * GetLog(Cat mask)
Retrieve the Log object for the channel associated with the given log enum.
Definition Log.h:332
std::shared_ptr< lldb_private::Function > FunctionSP
std::shared_ptr< lldb_private::Block > BlockSP
std::shared_ptr< lldb_private::ObjectFile > ObjectFileSP
LanguageType
Programming language type.
@ eLanguageTypeUnknown
Unknown or invalid language value.
@ eLanguageTypeRust
Rust.
@ eLanguageTypeObjC_plus_plus
Objective-C++.
@ eLanguageTypeSwift
Swift.
@ eLanguageTypeC
Non-standardized C, such as K&R.
@ eLanguageTypeObjC
Objective-C.
@ eLanguageTypeC_plus_plus
ISO C++:1998.
std::shared_ptr< lldb_private::Type > TypeSP
SymbolType
Symbol types.
std::shared_ptr< lldb_private::VariableList > VariableListSP
std::shared_ptr< lldb_private::SymbolFileType > SymbolFileTypeSP
std::shared_ptr< lldb_private::Variable > VariableSP
uint64_t user_id_t
Definition lldb-types.h:82
uint64_t addr_t
Definition lldb-types.h:80
std::shared_ptr< lldb_private::Module > ModuleSP
std::shared_ptr< lldb_private::CompileUnit > CompUnitSP
@ eValueTypeVariableGlobal
globals variable
@ eValueTypeVariableLocal
function local variables
@ eValueTypeVariableArgument
function argument variables
@ eValueTypeVariableStatic
static variable
@ eValueTypeVariableThreadLocal
thread local storage variable
BaseType GetRangeBase() const
Definition RangeMap.h:45
void SetRangeEnd(BaseType end)
Definition RangeMap.h:80
void SetRangeBase(BaseType b)
Set the start value for the range, and keep the same size.
Definition RangeMap.h:48
lldb::user_id_t GetID() const
Get accessor for the user ID.
Definition UserID.h:47
CompilerContextKind contextKind() const
Definition PdbUtil.h:75
static CVTagRecord create(llvm::codeview::CVType type)
Definition PdbUtil.cpp:198
const llvm::codeview::TagRecord & asTag() const
Definition PdbUtil.h:44
llvm::StringRef name() const
Definition PdbUtil.h:67
Represents a single compile unit.
std::map< llvm::codeview::TypeIndex, llvm::codeview::InlineeSourceLine > m_inline_map
std::optional< llvm::codeview::Compile3Sym > m_compile_opts
llvm::pdb::ModuleDebugStreamRef m_debug_stream
llvm::codeview::StringsAndChecksumsRef m_strings
std::vector< llvm::StringRef > m_file_list
llvm::codeview::TypeIndex index
Definition PdbSymUid.h:76
DWARFExpressionList location
Definition PdbUtil.h:115
llvm::codeview::TypeIndex type
Definition PdbUtil.h:114