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"
82 case PDB_Lang::ObjCpp:
89static std::optional<std::string>
91 llvm::StringRef exe_path) {
94 if (fs.
Exists(original_pdb_path))
95 return std::string(original_pdb_path);
100 const FileSpec original_pdb_spec(original_pdb_path,
102 .value_or(FileSpec::Style::native));
103 const llvm::StringRef pdb_filename = original_pdb_spec.
GetFilename();
114 for (
const FileSpec &search_dir : search_paths) {
123static std::unique_ptr<PDBFile>
126 using namespace llvm::object;
127 auto expected_binary = createBinary(exe_path);
130 if (!expected_binary) {
131 llvm::consumeError(expected_binary.takeError());
134 OwningBinary<Binary> binary = std::move(*expected_binary);
138 auto *obj = llvm::dyn_cast<llvm::object::COFFObjectFile>(binary.getBinary());
141 const llvm::codeview::DebugInfo *pdb_info =
nullptr;
144 llvm::StringRef pdb_file;
145 if (llvm::Error e = obj->getDebugPDBInfo(pdb_info, pdb_file)) {
146 consumeError(std::move(e));
150 std::optional<std::string> resolved_pdb_path =
152 if (!resolved_pdb_path)
161 auto expected_info =
pdb->getPDBInfoStream();
162 if (!expected_info) {
163 llvm::consumeError(expected_info.takeError());
166 llvm::codeview::GUID guid;
167 memcpy(&guid, pdb_info->PDB70.Signature, 16);
169 if (expected_info->getGuid() != guid)
196 case SimpleTypeKind::Boolean128:
198 case SimpleTypeKind::Boolean64:
200 case SimpleTypeKind::Boolean32:
202 case SimpleTypeKind::Boolean16:
204 case SimpleTypeKind::Boolean8:
207 case SimpleTypeKind::Byte:
208 case SimpleTypeKind::UnsignedCharacter:
209 return "unsigned char";
210 case SimpleTypeKind::NarrowCharacter:
212 case SimpleTypeKind::SignedCharacter:
213 case SimpleTypeKind::SByte:
214 return "signed char";
215 case SimpleTypeKind::Character32:
217 case SimpleTypeKind::Character16:
219 case SimpleTypeKind::Character8:
222 case SimpleTypeKind::Complex128:
223 return "_Complex __float128";
224 case SimpleTypeKind::Complex80:
225 return "_Complex long double";
226 case SimpleTypeKind::Complex64:
227 return "_Complex double";
228 case SimpleTypeKind::Complex48:
229 return "_Complex __float48";
230 case SimpleTypeKind::Complex32:
231 case SimpleTypeKind::Complex32PartialPrecision:
232 return "_Complex float";
233 case SimpleTypeKind::Complex16:
234 return "_Complex _Float16";
236 case SimpleTypeKind::Float128:
238 case SimpleTypeKind::Float80:
239 return "long double";
240 case SimpleTypeKind::Float64:
242 case SimpleTypeKind::Float48:
244 case SimpleTypeKind::Float32:
245 case SimpleTypeKind::Float32PartialPrecision:
247 case SimpleTypeKind::Float16:
250 case SimpleTypeKind::Int128Oct:
251 case SimpleTypeKind::Int128:
253 case SimpleTypeKind::Int64:
254 case SimpleTypeKind::Int64Quad:
256 case SimpleTypeKind::Int32Long:
258 case SimpleTypeKind::Int32:
260 case SimpleTypeKind::Int16:
261 case SimpleTypeKind::Int16Short:
264 case SimpleTypeKind::UInt128Oct:
265 case SimpleTypeKind::UInt128:
266 return "unsigned __int128";
267 case SimpleTypeKind::UInt64:
268 case SimpleTypeKind::UInt64Quad:
269 return "unsigned long long";
270 case SimpleTypeKind::UInt32:
272 case SimpleTypeKind::UInt16:
273 case SimpleTypeKind::UInt16Short:
274 return "unsigned short";
275 case SimpleTypeKind::UInt32Long:
276 return "unsigned long";
278 case SimpleTypeKind::HResult:
280 case SimpleTypeKind::Void:
282 case SimpleTypeKind::WideCharacter:
285 case SimpleTypeKind::None:
286 case SimpleTypeKind::NotTranslated:
303static std::optional<CVTagRecord>
322 if (Record.Type.isSimple())
325 CVType cvt = tpi.getType(Record.Type);
334 std::string qname = std::string(parent.
asTag().getUniqueName());
335 if (qname.size() < 4 || child.
asTag().getUniqueName().size() < 4)
341 qname[3] = child.
asTag().getUniqueName()[3];
345 piece += Record.Name;
346 piece.push_back(
'@');
347 qname.insert(4, std::move(piece));
348 if (qname != child.
asTag().UniqueName)
351 return std::move(child);
367 return "Microsoft PDB debug symbol cross-platform file reader.";
383 uint32_t abilities = 0;
391 pdb_file = &
pdb->GetPDBFile();
402 if (!expected_index) {
403 llvm::consumeError(expected_index.takeError());
406 m_index = std::move(*expected_index);
415 if (
m_index->dbi().isStripped())
426 m_index->ParseSectionContribs();
428 auto ts_or_err =
m_objfile_sp->GetModule()->GetTypeSystemForLanguage(
430 if (
auto err = ts_or_err.takeError()) {
432 "Failed to initialize: {0}");
434 if (
auto ts = *ts_or_err)
435 ts->SetSymbolFile(
this);
441 const DbiModuleList &modules =
m_index->dbi().modules();
442 uint32_t count = modules.getModuleCount();
449 DbiModuleDescriptor last = modules.getModuleDescriptor(count - 1);
450 if (last.getModuleName() ==
"* Linker *")
461 if (
auto err = ts_or_err.takeError())
463 auto ts = *ts_or_err;
468 switch (sym.kind()) {
474 return &func->GetBlock(
false);
481 BlockSym block(
static_cast<SymbolRecordKind
>(sym.kind()));
482 cantFail(SymbolDeserializer::deserializeAs<BlockSym>(sym, block));
491 m_index->MakeVirtualAddress(block.Segment, block.CodeOffset);
494 if (block_base >= func_base)
495 child_block->AddRange(
Block::Range(block_base - func_base, block.CodeSize));
498 "S_BLOCK32 at modi: {0:d} offset: {1:d}: adding range "
499 "[{2:x16}-{3:x16}) which has a base that is less than the "
501 "low PC 0x%" PRIx64
". Please file a bug and attach the file at the "
502 "start of this error message",
504 block_base + block.CodeSize, func_base);
507 m_blocks.insert({opaque_block_uid, child_block});
512 comp_unit->GetLineTable();
514 std::shared_ptr<InlineSite> inline_site =
m_inline_sites[opaque_block_uid];
521 for (
size_t i = 0; i < inline_site->ranges.GetSize(); ++i) {
522 auto *entry = inline_site->ranges.GetEntryAtIndex(i);
523 child_block->AddRange(
524 Block::Range(entry->GetRangeBase(), entry->GetByteSize()));
526 child_block->FinalizeRanges();
529 Declaration &decl = inline_site->inline_function_info->GetDeclaration();
530 Declaration &callsite = inline_site->inline_function_info->GetCallSite();
531 child_block->SetInlinedFunctionInfo(
532 inline_site->inline_function_info->GetName().GetCString(),
nullptr,
534 m_blocks.insert({opaque_block_uid, child_block});
538 lldbassert(
false &&
"Symbol is not a block!");
551 lldbassert(sym_record.kind() == S_LPROC32 || sym_record.kind() == S_GPROC32);
563 ProcSym proc(
static_cast<SymbolRecordKind
>(sym_record.kind()));
564 cantFail(SymbolDeserializer::deserializeAs<ProcSym>(sym_record, proc));
565 if (proc.FunctionType == TypeIndex::None())
574 SegmentOffset(proc.Segment, proc.CodeOffset), proc.FunctionType);
575 Mangled mangled(mangled_opt.value_or(proc.Name));
577 FunctionSP func_sp = std::make_shared<Function>(
579 func_type.get(), func_addr,
585 if (
auto err = ts_or_err.takeError())
587 auto ts = *ts_or_err;
590 ts->GetNativePDBParser()->GetOrCreateFunctionDecl(func_id);
605 llvm::SmallString<64> source_file_name =
606 m_index->compilands().GetMainSourceFile(cci);
607 FileSpec fs(llvm::sys::path::convert_to_slash(
608 source_file_name, llvm::sys::path::Style::windows_backslash));
610 CompUnitSP cu_sp = std::make_shared<CompileUnit>(
611 m_objfile_sp->GetModule(),
nullptr, std::make_shared<SupportFile>(fs),
619 const ModifierRecord &mr,
621 TpiStream &stream =
m_index->tpi();
625 if ((mr.Modifiers & ModifierOptions::Const) != ModifierOptions::None)
627 if ((mr.Modifiers & ModifierOptions::Volatile) != ModifierOptions::None)
629 if ((mr.Modifiers & ModifierOptions::Unaligned) != ModifierOptions::None)
630 name +=
"__unaligned ";
632 if (mr.ModifiedType.isSimple())
635 name += computeTypeName(stream.typeCollection(), mr.ModifiedType);
640 llvm::expectedToOptional(modified_type->GetByteSize(
nullptr)),
647 const llvm::codeview::PointerRecord &pr,
653 if (pr.isPointerToMember()) {
654 MemberPointerInfo mpi = pr.getMemberInfo();
667 if (ti == TypeIndex::NullptrT()) {
674 if (ti.getSimpleMode() != SimpleTypeMode::Direct) {
676 uint32_t pointer_size = 0;
677 switch (ti.getSimpleMode()) {
678 case SimpleTypeMode::FarPointer32:
679 case SimpleTypeMode::NearPointer32:
682 case SimpleTypeMode::NearPointer64:
694 if (ti.getSimpleKind() == SimpleTypeKind::NotTranslated)
706 if (!record.hasUniqueName())
709 llvm::ms_demangle::Demangler demangler;
710 std::string_view sv(record.UniqueName.begin(), record.UniqueName.size());
711 llvm::ms_demangle::TagTypeNode *ttn = demangler.parseTagUniqueName(sv);
715 llvm::ms_demangle::IdentifierNode *idn =
716 ttn->QualifiedName->getUnqualifiedIdentifier();
717 return idn->toString();
722 const TagRecord &record,
730 decl = std::move(*maybeDecl);
733 "Failed to resolve declaration for '{1}': {0}", uname);
741 const ClassRecord &cr,
747 const UnionRecord &ur,
753 const EnumRecord &er,
760 decl = std::move(*maybeDecl);
763 "Failed to resolve declaration for '{1}': {0}", uname);
769 llvm::expectedToOptional(underlying_type->GetByteSize(
nullptr)),
nullptr,
775 const ArrayRecord &ar,
784 array_sp->SetEncodingType(element_type.get());
789 const MemberFunctionRecord &mfr,
791 if (mfr.ReturnType.isSimple())
802 const ProcedureRecord &pr,
804 if (pr.ReturnType.isSimple())
815 llvm::codeview::TypeIndex arglist_ti) {
816 if (arglist_ti.isNoneType())
819 CVType arglist_cvt =
m_index->tpi().getType(arglist_ti);
820 if (arglist_cvt.kind() != LF_ARGLIST)
825 TypeDeserializer::deserializeAs<ArgListRecord>(arglist_cvt, alr));
826 for (TypeIndex
id : alr.getIndices())
827 if (!
id.isNoneType() &&
id.isSimple())
832 if (type_id.
index.isSimple())
836 CVType cvt = stream.getType(type_id.
index);
838 if (cvt.kind() == LF_MODIFIER) {
839 ModifierRecord modifier;
841 TypeDeserializer::deserializeAs<ModifierRecord>(cvt, modifier));
845 if (cvt.kind() == LF_POINTER) {
846 PointerRecord pointer;
848 TypeDeserializer::deserializeAs<PointerRecord>(cvt, pointer));
854 llvm::cantFail(TypeDeserializer::deserializeAs<ClassRecord>(cvt, cr));
858 if (cvt.kind() == LF_ENUM) {
860 llvm::cantFail(TypeDeserializer::deserializeAs<EnumRecord>(cvt, er));
864 if (cvt.kind() == LF_UNION) {
866 llvm::cantFail(TypeDeserializer::deserializeAs<UnionRecord>(cvt, ur));
870 if (cvt.kind() == LF_ARRAY) {
872 llvm::cantFail(TypeDeserializer::deserializeAs<ArrayRecord>(cvt, ar));
876 if (cvt.kind() == LF_PROCEDURE) {
878 llvm::cantFail(TypeDeserializer::deserializeAs<ProcedureRecord>(cvt, pr));
881 if (cvt.kind() == LF_MFUNCTION) {
882 MemberFunctionRecord mfr;
883 llvm::cantFail(TypeDeserializer::deserializeAs<MemberFunctionRecord>(cvt, mfr));
893 std::optional<PdbTypeSymId> full_decl_uid;
895 auto expected_full_ti =
896 m_index->tpi().findFullDeclForForwardRef(type_id.
index);
897 if (!expected_full_ti)
898 llvm::consumeError(expected_full_ti.takeError());
899 else if (*expected_full_ti != type_id.
index) {
907 if (full_iter !=
m_types.end()) {
908 TypeSP result = full_iter->second;
917 PdbTypeSymId best_decl_id = full_decl_uid ? *full_decl_uid : type_id;
919 if (
auto err = ts_or_err.takeError())
921 auto ts = *ts_or_err;
959 CVSymbol sym =
m_index->symrecords().readRecord(var_id.
offset);
960 if (sym.kind() == S_CONSTANT)
965 llvm::StringRef name;
967 uint16_t section = 0;
969 bool is_external =
false;
970 switch (sym.kind()) {
975 DataSym ds(sym.kind());
976 llvm::cantFail(SymbolDeserializer::deserializeAs<DataSym>(sym, ds));
981 section = ds.Segment;
982 offset = ds.DataOffset;
983 addr =
m_index->MakeVirtualAddress(ds.Segment, ds.DataOffset);
990 ThreadLocalDataSym tlds(sym.kind());
992 SymbolDeserializer::deserializeAs<ThreadLocalDataSym>(sym, tlds));
995 section = tlds.Segment;
996 offset = tlds.DataOffset;
997 addr =
m_index->MakeVirtualAddress(tlds.Segment, tlds.DataOffset);
1002 llvm_unreachable(
"unreachable!");
1006 std::optional<uint16_t> modi =
m_index->GetModuleIndexForVa(addr);
1017 std::make_shared<SymbolFileType>(*
this,
toOpaqueUid(tid));
1020 if (
auto err = ts_or_err.takeError())
1022 auto ts = *ts_or_err;
1026 ts->GetNativePDBParser()->GetOrCreateVariableDecl(var_id);
1033 std::string global_name(
"::");
1034 global_name += name;
1035 bool artificial =
false;
1036 bool location_is_constant_data =
false;
1037 bool static_member =
false;
1038 VariableSP var_sp = std::make_shared<Variable>(
1039 toOpaqueUid(var_id), name.str().c_str(), global_name.c_str(), type_sp,
1040 scope, comp_unit.get(), ranges, &decl, location, is_external, artificial,
1041 location_is_constant_data, static_member);
1048 const CVSymbol &cvs) {
1049 TpiStream &tpi =
m_index->tpi();
1050 ConstantSym constant(cvs.kind());
1052 llvm::cantFail(SymbolDeserializer::deserializeAs<ConstantSym>(cvs, constant));
1053 std::string global_name(
"::");
1054 global_name += constant.Name;
1057 std::make_shared<SymbolFileType>(*
this,
toOpaqueUid(tid));
1064 constant.Type, tpi, constant.Value, module),
1067 bool external =
false;
1068 bool artificial =
false;
1069 bool location_is_constant_data =
true;
1070 bool static_member =
false;
1071 VariableSP var_sp = std::make_shared<Variable>(
1072 toOpaqueUid(var_id), constant.Name.str().c_str(), global_name.c_str(),
1074 external, artificial, location_is_constant_data, static_member);
1081 if (emplace_result.second) {
1083 emplace_result.first->second = var_sp;
1088 return emplace_result.first->second;
1098 if (emplace_result.second)
1099 emplace_result.first->second =
CreateFunction(func_id, comp_unit);
1101 return emplace_result.first->second;
1107 auto emplace_result =
1109 if (emplace_result.second)
1113 return emplace_result.first->second;
1119 return iter->second.get();
1140 if (index >= UINT16_MAX)
1163 auto *section_list =
1164 m_objfile_sp->GetModule()->GetObjectFile()->GetSectionList();
1168 PublicSym32 last_sym;
1169 size_t last_sym_idx = 0;
1177 auto finish_last_symbol = [&](
const PublicSym32 *next) {
1184 if (next && last_sym.Segment == next->Segment) {
1185 assert(last_sym.Offset <= next->Offset);
1186 last->
SetByteSize(next->Offset - last_sym.Offset);
1189 assert(section_sp->GetByteSize() >= last_sym.Offset);
1190 assert(!next || next->Segment > last_sym.Segment);
1191 last->
SetByteSize(section_sp->GetByteSize() - last_sym.Offset);
1196 for (
auto pid :
m_index->publics().getAddressMap()) {
1198 CVSymbol sym =
m_index->ReadSymbolRecord(global);
1199 auto kind = sym.kind();
1200 if (kind != S_PUB32)
1203 llvm::cantFail(SymbolDeserializer::deserializeAs<PublicSym32>(sym));
1204 finish_last_symbol(&pub);
1206 if (!section_sp || last_sym.Segment != pub.Segment)
1207 section_sp = section_list->FindSectionByID(pub.Segment);
1213 if ((pub.Flags & PublicSymFlags::Function) != PublicSymFlags::None ||
1214 (pub.Flags & PublicSymFlags::Code) != PublicSymFlags::None)
1234 finish_last_symbol(
nullptr);
1246 for (
auto iter = syms.begin(); iter != syms.end(); ++iter) {
1247 if (iter->kind() != S_LPROC32 && iter->kind() != S_GPROC32)
1257 return new_count - count;
1262 uint32_t flags = eSymbolContextCompUnit;
1263 flags |= eSymbolContextVariable;
1264 flags |= eSymbolContextFunction;
1265 flags |= eSymbolContextBlock;
1266 flags |= eSymbolContextLineEntry;
1267 return (resolve_scope & flags) != 0;
1271 const Address &addr, SymbolContextItem resolve_scope, SymbolContext &sc) {
1273 uint32_t resolved_flags = 0;
1277 std::optional<uint16_t> modi =
m_index->GetModuleIndexForVa(file_addr);
1285 resolved_flags |= eSymbolContextCompUnit;
1288 if (resolve_scope & eSymbolContextFunction ||
1289 resolve_scope & eSymbolContextBlock) {
1291 std::vector<SymbolAndUid> matches =
m_index->FindSymbolsByVa(file_addr);
1295 for (
const auto &match : llvm::reverse(matches)) {
1299 PdbCompilandSymId csid = match.uid.asCompilandSym();
1300 CVSymbol cvs =
m_index->ReadSymbolRecord(csid);
1302 if (type != PDB_SymType::Function && type != PDB_SymType::Block)
1304 if (type == PDB_SymType::Function) {
1309 addr_t offset = file_addr - func_base;
1314 if (type == PDB_SymType::Block) {
1322 addr_t offset = file_addr - func_base;
1327 resolved_flags |= eSymbolContextFunction;
1329 resolved_flags |= eSymbolContextBlock;
1334 if (resolve_scope & eSymbolContextLineEntry) {
1337 if (line_table->FindLineEntryByAddress(addr, sc.
line_entry))
1338 resolved_flags |= eSymbolContextLineEntry;
1342 return resolved_flags;
1349 const uint32_t prev_size = sc_list.
GetSize();
1350 if (resolve_scope & eSymbolContextCompUnit) {
1359 if (file_spec_matches_cu_file_spec) {
1365 return sc_list.
GetSize() - prev_size;
1384 std::set<LineTable::Entry, LineTableEntryComparator> line_set;
1389 for (
const DebugSubsectionRecord &dssr :
1391 if (dssr.kind() != DebugSubsectionKind::Lines)
1394 DebugLinesSubsectionRef lines;
1395 llvm::BinaryStreamReader reader(dssr.getRecordData());
1396 if (
auto EC = lines.initialize(reader)) {
1397 llvm::consumeError(std::move(EC));
1401 const LineFragmentHeader *lfh = lines.header();
1402 uint64_t virtual_addr =
1403 m_index->MakeVirtualAddress(lfh->RelocSegment, lfh->RelocOffset);
1407 for (
const LineColumnEntry &group : lines) {
1408 llvm::Expected<uint32_t> file_index_or_err =
1410 if (!file_index_or_err)
1412 uint32_t file_index = file_index_or_err.get();
1416 for (
const LineNumberEntry &entry : group.LineNumbers) {
1417 LineInfo cur_info(entry.Flags);
1419 if (cur_info.isAlwaysStepInto() || cur_info.isNeverStepInto())
1422 uint64_t addr = virtual_addr + entry.Offset;
1424 bool is_statement = cur_info.isStatement();
1428 uint32_t lno = cur_info.getStartLine();
1431 is_prologue, is_epilogue,
false);
1433 auto iter = line_set.find(new_entry);
1434 if (iter != line_set.end() && iter->is_terminal_entry)
1435 line_set.erase(iter);
1436 line_set.insert(new_entry);
1443 line_entry.
data = {file_index, lno};
1445 LineInfo last_line(group.LineNumbers.back().Flags);
1446 line_set.emplace(virtual_addr + lfh->CodeSize, last_line.getEndLine(), 0,
1447 file_index,
false,
false,
false,
false,
true);
1450 line_entry.
SetRangeEnd(virtual_addr + lfh->CodeSize);
1459 const CVSymbolArray &syms = cii->
m_debug_stream.getSymbolArray();
1460 for (
auto iter = syms.begin(); iter != syms.end();) {
1461 if (iter->kind() != S_LPROC32 && iter->kind() != S_GPROC32) {
1466 uint32_t record_offset = iter.offset();
1467 CVSymbol func_record =
1480 if (kind != S_INLINESITE)
1485 for (
const auto &line_entry :
1490 if (!line_entry.is_terminal_entry)
1491 line_set.erase(line_entry);
1492 line_set.insert(line_entry);
1500 iter = syms.at(getScopeEndOffset(func_record));
1506 std::vector<LineTable::Sequence> sequence(1);
1507 for (
const auto &line_entry : line_set) {
1509 sequence.back(), line_entry.file_addr, line_entry.line,
1510 line_entry.column, line_entry.file_idx,
1511 line_entry.is_start_of_statement, line_entry.is_start_of_basic_block,
1512 line_entry.is_prologue_end, line_entry.is_epilogue_begin,
1513 line_entry.is_terminal_entry);
1516 std::make_unique<LineTable>(&comp_unit, std::move(sequence));
1518 if (line_table->GetSize() == 0)
1530llvm::Expected<uint32_t>
1534 return llvm::make_error<RawError>(raw_error_code::no_entry);
1536 const auto &checksums = cii.
m_strings.checksums().getArray();
1537 const auto &strings = cii.
m_strings.strings();
1541 auto iter = checksums.at(file_id);
1542 if (iter == checksums.end())
1543 return llvm::make_error<RawError>(raw_error_code::no_entry);
1545 llvm::Expected<llvm::StringRef> efn = strings.getString(iter->FileNameOffset);
1547 return efn.takeError();
1553 return std::distance(cii.
m_file_list.begin(), fn_iter);
1554 return llvm::make_error<RawError>(raw_error_code::no_entry);
1568 f.starts_with(
"/") ? FileSpec::Style::posix : FileSpec::Style::windows;
1570 support_files.
Append(spec);
1576 const SymbolContext &sc, std::vector<SourceModule> &imported_modules) {
1589 CVSymbol sym = cii->
m_debug_stream.readSymbolAtOffset(
id.offset);
1592 InlineSiteSym inline_site(
static_cast<SymbolRecordKind
>(sym.kind()));
1593 cantFail(SymbolDeserializer::deserializeAs<InlineSiteSym>(sym, inline_site));
1596 std::shared_ptr<InlineSite> inline_site_sp =
1597 std::make_shared<InlineSite>(parent_id);
1600 auto iter = cii->
m_inline_map.find(inline_site.Inlinee);
1603 InlineeSourceLine inlinee_line = iter->second;
1607 llvm::Expected<uint32_t> file_index_or_err =
1609 if (!file_index_or_err)
1611 uint32_t file_offset = file_index_or_err.get();
1613 uint32_t decl_line = inlinee_line.Header->SourceLineNum;
1614 std::unique_ptr<Declaration> decl_up =
1615 std::make_unique<Declaration>(decl_file, decl_line);
1618 uint32_t code_offset = 0;
1619 int32_t line_offset = 0;
1620 std::optional<uint32_t> code_offset_base;
1621 std::optional<uint32_t> code_offset_end;
1622 std::optional<int32_t> cur_line_offset;
1623 std::optional<int32_t> next_line_offset;
1624 std::optional<uint32_t> next_file_offset;
1626 bool is_terminal_entry =
false;
1627 bool is_start_of_statement =
true;
1629 bool is_prologue_end =
true;
1631 auto update_code_offset = [&](uint32_t code_delta) {
1632 if (!code_offset_base)
1633 code_offset_base = code_offset;
1634 else if (!code_offset_end)
1635 code_offset_end = *code_offset_base + code_delta;
1637 auto update_line_offset = [&](int32_t line_delta) {
1638 line_offset += line_delta;
1639 if (!code_offset_base || !cur_line_offset)
1640 cur_line_offset = line_offset;
1642 next_line_offset = line_offset;
1645 auto update_file_offset = [&](uint32_t offset) {
1646 if (!code_offset_base)
1647 file_offset = offset;
1649 next_file_offset = offset;
1652 for (
auto &annot : inline_site.annotations()) {
1653 switch (annot.OpCode) {
1654 case BinaryAnnotationsOpCode::CodeOffset:
1655 case BinaryAnnotationsOpCode::ChangeCodeOffset:
1656 case BinaryAnnotationsOpCode::ChangeCodeOffsetBase:
1657 code_offset += annot.U1;
1658 update_code_offset(annot.U1);
1660 case BinaryAnnotationsOpCode::ChangeLineOffset:
1661 update_line_offset(annot.S1);
1663 case BinaryAnnotationsOpCode::ChangeCodeLength:
1664 update_code_offset(annot.U1);
1665 code_offset += annot.U1;
1666 is_terminal_entry =
true;
1668 case BinaryAnnotationsOpCode::ChangeCodeOffsetAndLineOffset:
1669 code_offset += annot.U1;
1670 update_code_offset(annot.U1);
1671 update_line_offset(annot.S1);
1673 case BinaryAnnotationsOpCode::ChangeCodeLengthAndCodeOffset:
1674 code_offset += annot.U2;
1675 update_code_offset(annot.U2);
1676 update_code_offset(annot.U1);
1677 code_offset += annot.U1;
1678 is_terminal_entry =
true;
1680 case BinaryAnnotationsOpCode::ChangeFile:
1681 update_file_offset(annot.U1);
1688 if (code_offset_base && code_offset_end && cur_line_offset) {
1690 *code_offset_base, *code_offset_end - *code_offset_base,
1691 decl_line + *cur_line_offset));
1693 if (next_file_offset)
1694 file_offset = *next_file_offset;
1695 if (next_line_offset) {
1696 cur_line_offset = next_line_offset;
1697 next_line_offset = std::nullopt;
1699 code_offset_base = is_terminal_entry ? std::nullopt : code_offset_end;
1700 code_offset_end = next_file_offset = std::nullopt;
1702 if (code_offset_base && cur_line_offset) {
1703 if (is_terminal_entry) {
1705 func_base + *code_offset_base, decl_line + *cur_line_offset, 0,
1706 file_offset,
false,
false,
false,
false,
true);
1707 inline_site_sp->line_entries.push_back(line_entry);
1710 decl_line + *cur_line_offset, 0,
1711 file_offset, is_start_of_statement,
false,
1712 is_prologue_end,
false,
false);
1713 inline_site_sp->line_entries.push_back(line_entry);
1714 is_prologue_end =
false;
1715 is_start_of_statement =
false;
1718 if (is_terminal_entry)
1719 is_start_of_statement =
true;
1720 is_terminal_entry =
false;
1723 inline_site_sp->ranges.Sort();
1726 std::unique_ptr<Declaration> callsite_up;
1727 if (!inline_site_sp->ranges.IsEmpty()) {
1728 auto *entry = inline_site_sp->ranges.GetEntryAtIndex(0);
1729 addr_t base_offset = entry->GetRangeBase();
1735 std::shared_ptr<InlineSite> parent_site =
1738 parent_site->inline_function_info->GetDeclaration().GetFile();
1739 if (
auto *parent_entry =
1740 parent_site->ranges.FindEntryThatContains(base_offset)) {
1742 std::make_unique<Declaration>(parent_decl_file, parent_entry->data);
1747 func_base + base_offset)) {
1751 std::make_unique<Declaration>(callsite_file, entry->data.second);
1757 CVType inlinee_cvt =
m_index->ipi().getType(inline_site.Inlinee);
1758 std::string inlinee_name;
1759 if (inlinee_cvt.kind() == LF_MFUNC_ID) {
1760 MemberFuncIdRecord mfr;
1762 TypeDeserializer::deserializeAs<MemberFuncIdRecord>(inlinee_cvt, mfr));
1763 LazyRandomTypeCollection &types =
m_index->tpi().typeCollection();
1764 inlinee_name.append(std::string(types.getTypeName(mfr.ClassType)));
1765 inlinee_name.append(
"::");
1766 inlinee_name.append(mfr.getName().str());
1767 }
else if (inlinee_cvt.kind() == LF_FUNC_ID) {
1769 cantFail(TypeDeserializer::deserializeAs<FuncIdRecord>(inlinee_cvt, fir));
1770 TypeIndex parent_idx = fir.getParentScope();
1771 if (!parent_idx.isNoneType()) {
1772 LazyRandomTypeCollection &ids =
m_index->ipi().typeCollection();
1773 inlinee_name.append(std::string(ids.getTypeName(parent_idx)));
1774 inlinee_name.append(
"::");
1776 inlinee_name.append(fir.getName().str());
1778 inline_site_sp->inline_function_info = std::make_shared<InlineFunctionInfo>(
1779 inlinee_name.c_str(), llvm::StringRef(), decl_up.get(),
1790 std::set<uint64_t> remove_uids;
1792 if (kind == S_GPROC32 || kind == S_LPROC32 || kind == S_BLOCK32 ||
1793 kind == S_INLINESITE) {
1795 if (kind == S_INLINESITE)
1802 for (uint64_t uid : remove_uids) {
1814 CVSymbolArray syms =
1818 for (
auto iter = syms.begin(); iter != syms.end(); ++iter) {
1820 if (fn(iter->kind(), child_id))
1832 auto ts = *ts_or_err;
1836 clang->GetNativePDBParser()->Dump(s, filter, show_color);
1844 std::map<std::pair<uint16_t, uint32_t>, uint32_t> func_addr_ids;
1847 for (
const uint32_t gid :
m_index->globals().getGlobalsTable()) {
1848 CVSymbol sym =
m_index->symrecords().readRecord(gid);
1849 auto kind = sym.kind();
1852 llvm::StringRef name;
1854 case SymbolKind::S_GDATA32:
1855 case SymbolKind::S_LDATA32: {
1857 llvm::cantFail(SymbolDeserializer::deserializeAs<DataSym>(sym));
1861 case SymbolKind::S_GTHREAD32:
1862 case SymbolKind::S_LTHREAD32: {
1863 ThreadLocalDataSym data = llvm::cantFail(
1864 SymbolDeserializer::deserializeAs<ThreadLocalDataSym>(sym));
1868 case SymbolKind::S_CONSTANT: {
1870 llvm::cantFail(SymbolDeserializer::deserializeAs<ConstantSym>(sym));
1878 if (!name.empty()) {
1887 if (kind != S_PROCREF && kind != S_LPROCREF)
1894 llvm::cantFail(SymbolDeserializer::deserializeAs<ProcRefSym>(sym));
1895 if (ref.Name.empty())
1900 m_index->compilands().GetOrCreateCompiland(ref.modi());
1901 auto iter = cci.
m_debug_stream.getSymbolArray().at(ref.SymOffset);
1904 kind = iter->kind();
1905 if (kind != S_GPROC32 && kind != S_LPROC32)
1909 llvm::cantFail(SymbolDeserializer::deserializeAs<ProcSym>(*iter));
1910 if ((proc.Flags & ProcSymFlags::IsUnreachable) != ProcSymFlags::None)
1912 if (proc.Name.empty() || proc.FunctionType.isSimple())
1918 func_addr_ids.emplace(std::make_pair(proc.Segment, proc.CodeOffset), gid);
1921 if (basename.empty())
1922 basename = proc.Name;
1928 auto type =
m_index->tpi().getType(proc.FunctionType);
1929 if (type.kind() == LF_MFUNCTION) {
1930 MemberFunctionRecord mfr;
1932 TypeDeserializer::deserializeAs<MemberFunctionRecord>(type, mfr));
1933 if (!mfr.getThisType().isNoneType())
1939 for (
auto pid :
m_index->publics().getPublicsTable()) {
1941 CVSymbol sym =
m_index->ReadSymbolRecord(global);
1942 auto kind = sym.kind();
1943 if (kind != S_PUB32)
1946 llvm::cantFail(SymbolDeserializer::deserializeAs<PublicSym32>(sym));
1953 auto it = func_addr_ids.find({pub.Segment, pub.Offset});
1954 if (it != func_addr_ids.end())
1976 std::vector<uint32_t> results;
1979 size_t n_matches = 0;
1980 for (uint32_t gid : results) {
1983 if (parent_decl_ctx.
IsValid() &&
1992 if (++n_matches >= max_matches)
2004 if (name_type_mask & eFunctionNameTypeFull)
2007 if (!(name_type_mask & eFunctionNameTypeFull ||
2008 name_type_mask & eFunctionNameTypeBase ||
2009 name_type_mask & eFunctionNameTypeMethod))
2013 std::set<uint32_t> resolved_ids;
2015 std::vector<uint32_t> ids;
2016 if (!Names.GetValues(name, ids))
2019 for (uint32_t
id : ids) {
2020 if (!resolved_ids.insert(
id).second)
2024 if (parent_decl_ctx.
IsValid() &&
2028 CVSymbol sym =
m_index->ReadSymbolRecord(global);
2029 auto kind = sym.kind();
2030 lldbassert(kind == S_PROCREF || kind == S_LPROCREF);
2033 cantFail(SymbolDeserializer::deserializeAs<ProcRefSym>(sym));
2039 m_index->compilands().GetOrCreateCompiland(proc.modi());
2055 if (name_type_mask & eFunctionNameTypeFull)
2057 if (name_type_mask & eFunctionNameTypeBase)
2059 if (name_type_mask & eFunctionNameTypeMethod)
2064 bool include_inlines,
2079 std::vector<uint32_t> matches;
2082 for (uint32_t match_idx : matches) {
2084 if (context.empty())
2093 if (results.
Done(query))
2100 uint32_t max_matches,
2103 std::vector<TypeIndex> matches =
m_index->tpi().findRecordsByName(name);
2104 if (max_matches > 0 && max_matches < matches.size())
2105 matches.resize(max_matches);
2107 for (TypeIndex ti : matches) {
2123 LazyRandomTypeCollection &types =
m_index->tpi().typeCollection();
2126 for (
auto ti = types.getFirst(); ti; ti = types.getNext(*ti)) {
2129 (void)type->GetFullCompilerType();
2133 for (
const uint32_t gid :
m_index->globals().getGlobalsTable()) {
2135 CVSymbol sym =
m_index->ReadSymbolRecord(global);
2136 if (sym.kind() != S_UDT)
2139 UDTSym udt = llvm::cantFail(SymbolDeserializer::deserializeAs<UDTSym>(sym));
2140 bool is_typedef =
true;
2142 CVType cvt =
m_index->tpi().getType(udt.Type);
2144 if (name == udt.Name)
2156 return new_count - old_count;
2164 for (
const uint32_t gid :
m_index->globals().getGlobalsTable()) {
2166 CVSymbol sym =
m_index->ReadSymbolRecord(global);
2171 switch (sym.kind()) {
2172 case SymbolKind::S_GDATA32:
2173 case SymbolKind::S_LDATA32:
2174 case SymbolKind::S_GTHREAD32:
2175 case SymbolKind::S_LTHREAD32: {
2196 Block *func_block = block;
2218 std::string name = var_info.
name.str();
2221 std::make_shared<SymbolFileType>(*
this, type_sp->GetID());
2226 bool external =
false;
2227 bool artificial =
false;
2228 bool location_is_constant_data =
false;
2229 bool static_member =
false;
2231 VariableSP var_sp = std::make_shared<Variable>(
2232 toOpaqueUid(var_id), name.c_str(), name.c_str(), sftype, var_scope, block,
2233 scope_ranges, &decl, var_info.
location, external, artificial,
2234 location_is_constant_data, static_member);
2237 if (
auto err = ts_or_err.takeError())
2239 auto ts = *ts_or_err;
2243 ts->GetNativePDBParser()->GetOrCreateVariableDecl(scope_id, var_id);
2253 return iter->second;
2259 CVSymbol sym =
m_index->ReadSymbolRecord(
id);
2262 UDTSym udt = llvm::cantFail(SymbolDeserializer::deserializeAs<UDTSym>(sym));
2267 if (
auto err = ts_or_err.takeError())
2269 auto ts = *ts_or_err;
2273 auto *typedef_decl = ts->GetNativePDBParser()->GetOrCreateTypedefDecl(
id);
2275 CompilerType ct = target_type->GetForwardCompilerType();
2276 if (
auto *
clang = llvm::dyn_cast_or_null<TypeSystemClang>(ts.get()))
2277 ct =
clang->GetType(
clang->getASTContext().getTypeDeclType(typedef_decl));
2281 llvm::expectedToOptional(target_type->GetByteSize(
nullptr)),
2282 nullptr, target_type->GetID(),
2290 return iter->second;
2304 uint32_t params_remaining = 0;
2305 switch (sym.kind()) {
2308 ProcSym proc(
static_cast<SymbolRecordKind
>(sym.kind()));
2309 cantFail(SymbolDeserializer::deserializeAs<ProcSym>(sym, proc));
2310 CVType signature =
m_index->tpi().getType(proc.FunctionType);
2311 if (signature.kind() == LF_PROCEDURE) {
2312 ProcedureRecord sig;
2313 if (llvm::Error e = TypeDeserializer::deserializeAs<ProcedureRecord>(
2315 llvm::consumeError(std::move(e));
2318 params_remaining = sig.getParameterCount();
2319 }
else if (signature.kind() == LF_MFUNCTION) {
2320 MemberFunctionRecord sig;
2321 if (llvm::Error e = TypeDeserializer::deserializeAs<MemberFunctionRecord>(
2323 llvm::consumeError(std::move(e));
2326 params_remaining = sig.getParameterCount();
2336 lldbassert(
false &&
"Symbol is not a block!");
2342 variables = std::make_shared<VariableList>();
2346 CVSymbolArray syms = limitSymbolArrayToScope(
2352 auto iter = syms.begin();
2353 auto end = syms.end();
2355 while (iter != end) {
2356 uint32_t record_offset = iter.offset();
2357 CVSymbol variable_cvs = *iter;
2363 if (variable_cvs.kind() == S_BLOCK32 ||
2364 variable_cvs.kind() == S_INLINESITE) {
2365 uint32_t block_end = getScopeEndOffset(variable_cvs);
2367 iter = syms.at(block_end);
2371 bool is_param = params_remaining > 0;
2373 switch (variable_cvs.kind()) {
2381 variables->AddVariableIfUnique(variable);
2417 variables = std::make_shared<VariableList>();
2423 llvm_unreachable(
"Unreachable!");
2428 if (
auto err = ts_or_err.takeError())
2430 auto ts = *ts_or_err;
2434 if (
auto decl = ts->GetNativePDBParser()->GetOrCreateDeclForUid(uid))
2442 if (
auto err = ts_or_err.takeError())
2444 auto ts = *ts_or_err;
2449 clang::DeclContext *context =
2460 if (
auto err = ts_or_err.takeError())
2462 auto ts = *ts_or_err;
2475 auto iter =
m_types.find(type_uid);
2482 return &*iter->second;
2487 if (type_id.
index.isNoneType())
2496std::optional<SymbolFile::ArrayInfo>
2499 return std::nullopt;
2505 if (!clang_type_system)
2509 static_cast<PdbAstBuilder *
>(clang_type_system->GetNativePDBParser());
2513 clang::QualType qt =
2520 TypeClass type_mask,
2529 if (
auto err = ts_or_err.takeError())
2531 auto ts = *ts_or_err;
2534 auto *
clang = llvm::dyn_cast_or_null<TypeSystemClang>(ts.get());
2542 clang::DeclContext *decl_context =
nullptr;
2543 if (parent_decl_ctx)
2544 decl_context =
static_cast<clang::DeclContext *
>(
2547 auto *namespace_decl =
2549 if (!namespace_decl)
2552 return clang->CreateDeclContext(namespace_decl);
2555llvm::Expected<lldb::TypeSystemSP>
2557 auto type_system_or_err =
2558 m_objfile_sp->GetModule()->GetTypeSystemForLanguage(language);
2559 if (type_system_or_err)
2560 if (
auto ts = *type_system_or_err)
2561 ts->SetSymbolFile(
this);
2562 return type_system_or_err;
2567 return m_index->pdb().getFileSize();
2571 LazyRandomTypeCollection &types =
m_index->tpi().typeCollection();
2573 llvm::DenseMap<TypeIndex, TypeIndex> forward_to_full;
2574 llvm::DenseMap<TypeIndex, TypeIndex> full_to_forward;
2576 struct RecordIndices {
2581 llvm::StringMap<RecordIndices> record_indices;
2583 for (
auto ti = types.getFirst(); ti; ti = types.getNext(*ti)) {
2584 CVType type = types.getType(*ti);
2590 RecordIndices &indices = record_indices[tag.
asTag().getUniqueName()];
2591 if (tag.
asTag().isForwardRef()) {
2592 indices.forward = *ti;
2600 if (indices.full != TypeIndex::None() &&
2601 indices.forward != TypeIndex::None()) {
2602 forward_to_full[indices.forward] = indices.full;
2603 full_to_forward[indices.full] = indices.forward;
2609 if (tag.
asTag().isForwardRef() || !tag.
asTag().containsNestedClass())
2613 ProcessTpiStream(
PdbIndex &index, TypeIndex parent,
2615 llvm::DenseMap<TypeIndex, TypeIndex> &parents)
2616 : index(index), parents(parents), parent(parent),
2617 parent_cvt(parent_cvt) {}
2620 llvm::DenseMap<TypeIndex, TypeIndex> &parents;
2622 unsigned unnamed_type_index = 1;
2626 llvm::Error visitKnownMember(CVMemberRecord &CVR,
2627 NestedTypeRecord &Record)
override {
2628 std::string unnamed_type_name;
2629 if (Record.Name.empty()) {
2631 llvm::formatv(
"<unnamed-type-$S{0}>", unnamed_type_index).str();
2632 Record.Name = unnamed_type_name;
2633 ++unnamed_type_index;
2635 std::optional<CVTagRecord> tag =
2638 return llvm::ErrorSuccess();
2640 parents[Record.Type] = parent;
2641 return llvm::ErrorSuccess();
2645 CVType field_list_cvt =
m_index->tpi().getType(tag.
asTag().FieldList);
2647 FieldListRecord field_list;
2648 if (llvm::Error
error = TypeDeserializer::deserializeAs<FieldListRecord>(
2649 field_list_cvt, field_list))
2650 llvm::consumeError(std::move(
error));
2651 if (llvm::Error
error = visitMemberRecordStream(field_list.Data, process))
2652 llvm::consumeError(std::move(
error));
2664 std::vector<TypeIndex> full_keys;
2665 std::vector<TypeIndex> fwd_keys;
2667 TypeIndex key = entry.first;
2668 TypeIndex value = entry.second;
2670 auto iter = forward_to_full.find(value);
2671 if (iter != forward_to_full.end())
2672 entry.second = iter->second;
2674 iter = forward_to_full.find(key);
2675 if (iter != forward_to_full.end())
2676 fwd_keys.push_back(key);
2678 full_keys.push_back(key);
2680 for (TypeIndex fwd : fwd_keys) {
2681 TypeIndex full = forward_to_full[fwd];
2685 for (TypeIndex full : full_keys) {
2686 TypeIndex fwd = full_to_forward[full];
2691std::optional<PdbCompilandSymId>
2693 CVSymbol sym =
m_index->ReadSymbolRecord(
id);
2694 if (symbolOpensScope(sym.kind())) {
2697 id.offset = getScopeParentOffset(sym);
2701 return std::nullopt;
2710 auto begin = syms.begin();
2711 auto end = syms.at(
id.offset);
2712 std::vector<PdbCompilandSymId> scope_stack;
2714 while (begin != end) {
2715 if (begin.offset() >
id.offset) {
2717 lldbassert(
false &&
"Invalid compiland symbol id!");
2718 return std::nullopt;
2723 if (symbolOpensScope(begin->kind())) {
2726 uint32_t scope_end = getScopeEndOffset(*begin);
2727 if (scope_end <
id.offset) {
2728 begin = syms.at(scope_end);
2731 scope_stack.emplace_back(
id.modi, begin.offset());
2733 }
else if (symbolEndsScope(begin->kind())) {
2734 scope_stack.pop_back();
2738 if (scope_stack.empty())
2739 return std::nullopt;
2741 return scope_stack.back();
2744std::optional<llvm::codeview::TypeIndex>
2748 return std::nullopt;
2749 return parent_iter->second;
2752std::vector<CompilerContext>
2754 CVType type =
m_index->tpi().getType(ti);
2760 std::optional<Type::ParsedName> parsed_name =
2765 std::vector<CompilerContext> ctx;
2767 for (llvm::StringRef scope : parsed_name->scope) {
2774 for (
auto &el : llvm::reverse(llvm::drop_end(ctx))) {
2780 type =
m_index->tpi().getType(ti);
2781 switch (type.kind()) {
2801std::optional<llvm::StringRef>
2806 return std::nullopt;
2809 if (sym_record.kind() != S_LPROC32 && sym_record.kind() != S_GPROC32)
2810 return std::nullopt;
2812 ProcSym proc(
static_cast<SymbolRecordKind
>(sym_record.kind()));
2813 cantFail(SymbolDeserializer::deserializeAs<ProcSym>(sym_record, proc));
2819std::optional<llvm::StringRef>
2821 TypeIndex function_type) {
2822 auto symbol =
m_index->publics().findByAddress(
m_index->symrecords(),
2825 return std::nullopt;
2827 llvm::StringRef name = symbol->first.Name;
2830 if (!function_type.isNoneType() &&
2831 (symbol->first.Flags & PublicSymFlags::Function) != PublicSymFlags::None)
2848 if (!mangled.starts_with(
'_') ||
2849 m_index->dbi().getMachineType() != PDB_Machine::x86)
2853 PDB_CallingConv cc = PDB_CallingConv::NearC;
2854 if (cvt.kind() == LF_PROCEDURE) {
2855 ProcedureRecord proc;
2856 if (llvm::Error
error =
2857 TypeDeserializer::deserializeAs<ProcedureRecord>(cvt, proc))
2858 llvm::consumeError(std::move(
error));
2860 }
else if (cvt.kind() == LF_MFUNCTION) {
2861 MemberFunctionRecord mfunc;
2862 if (llvm::Error
error =
2863 TypeDeserializer::deserializeAs<MemberFunctionRecord>(cvt, mfunc))
2864 llvm::consumeError(std::move(
error));
2865 cc = mfunc.CallConv;
2872 if (cc == PDB_CallingConv::NearC || cc == PDB_CallingConv::FarC)
2873 return mangled.drop_front();
2879 for (CVType cvt :
m_index->ipi().typeArray()) {
2880 switch (cvt.kind()) {
2881 case LF_UDT_SRC_LINE: {
2882 UdtSourceLineRecord udt_src;
2883 llvm::cantFail(TypeDeserializer::deserializeAs(cvt, udt_src));
2887 udt_src.LineNumber});
2889 case LF_UDT_MOD_SRC_LINE: {
2890 UdtModSourceLineRecord udt_mod_src;
2891 llvm::cantFail(TypeDeserializer::deserializeAs(cvt, udt_mod_src));
2899 udt_mod_src.LineNumber});
2907llvm::Expected<Declaration>
2913 return llvm::createStringError(
"No UDT declaration found");
2915 llvm::StringRef file_name;
2916 if (it->second.IsIpiIndex) {
2917 CVType cvt =
m_index->ipi().getType(it->second.FileNameIndex);
2918 if (cvt.kind() != LF_STRING_ID)
2919 return llvm::createStringError(
"File name was not a LF_STRING_ID");
2922 llvm::cantFail(TypeDeserializer::deserializeAs(cvt, sid));
2923 file_name = sid.String;
2926 auto string_table =
m_index->pdb().getStringTable();
2928 return string_table.takeError();
2930 llvm::Expected<llvm::StringRef>
string =
2931 string_table->getStringTable().getString(
2932 it->second.FileNameIndex.getIndex());
2934 return string.takeError();
2935 file_name = *string;
2939 if (file_name ==
"\\<unknown>")
static llvm::raw_ostream & error(Stream &strm)
#define LLDB_LOG(log,...)
The LLDB_LOG* macros defined below are the way to emit log messages.
#define LLDB_LOG_ERROR(log, error,...)
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 std::optional< std::string > findMatchingPDBFilePath(llvm::StringRef original_pdb_path, llvm::StringRef exe_path)
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.
lldb::addr_t GetFileAddress() const
Get the file address.
bool IsValid() const
Check if the object state is valid.
A class that describes a single lexical block.
lldb::VariableListSP GetBlockVariableList(bool can_create)
Get the variable list for this block only.
Block * FindInnermostBlockByOffset(const lldb::addr_t offset)
void SetBlockInfoHasBeenParsed(bool b, bool set_children)
lldb::BlockSP CreateChild(lldb::user_id_t uid)
Creates a block with the specified UID uid.
Function * CalculateSymbolContextFunction() override
void SetVariableList(lldb::VariableListSP &variable_list_sp)
Set accessor for the variable list.
Block * GetParent() const
Get the parent block.
bool GetStartAddress(Address &addr)
void SetDidParseVariables(bool b, bool set_children)
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.
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.
TypeSystem * GetTypeSystem() const
void * GetOpaqueDeclContext() const
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.
llvm::StringRef GetStringRef() const
Get the string value as a llvm::StringRef.
"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.
A class that describes the declaration location of a lldb object.
"lldb/Target/ExecutionContext.h" A class that contains an execution context.
FileSpec CopyByAppendingPathComponent(llvm::StringRef component) const
static std::optional< Style > GuessPathStyle(llvm::StringRef absolute_path)
Attempt to guess path style for a given path string.
static bool Match(const FileSpec &pattern, const FileSpec &file)
Match FileSpec pattern against FileSpec file.
const ConstString & GetFilename() const
Filename string const get accessor.
size_t GetPath(char *path, size_t max_path_length, bool denormalize=true) const
Extract the full path to the file.
FileSpec CopyByRemovingLastPathComponent() const
llvm::sys::path::Style Style
bool Exists(const FileSpec &file_spec) const
Returns whether the given file exists.
static FileSystem & Instance()
A class that describes a function.
const Address & GetAddress() const
Return the address of the function (its entry point).
Block & GetBlock(bool can_create)
Get accessor for the block list.
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)
A class that handles mangled names.
static bool IsMangledName(llvm::StringRef name)
lldb::ModuleSP GetModule() const
Get const accessor for the module pointer.
A class that encapsulates name lookup information.
lldb::FunctionNameType GetNameTypeMask() const
ConstString GetLookupName() const
ConstString GetName() const
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
void Append(const Entry &entry)
Entry * FindEntryThatContains(B addr)
"lldb/Core/SourceLocationSpec.h" A source location specifier class.
FileSpec GetFileSpec() const
A stream class that can stream formatted output to a file.
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
virtual TypeList & GetTypeList()
lldb::ObjectFileSP m_objfile_sp
void SetCompileUnitAtIndex(uint32_t idx, const lldb::CompUnitSP &cu_sp)
SymbolFileCommon(lldb::ObjectFileSP objfile_sp)
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.
virtual std::recursive_mutex & GetModuleMutex() const
Symbols file subclasses should override this to return the Module that owns the TypeSystem that this ...
void SetByteSize(lldb::addr_t size)
Symbol * SymbolAtIndex(size_t idx)
uint32_t AddSymbol(const Symbol &symbol)
static FileSpecList GetDefaultDebugFileSearchPaths()
void Insert(const lldb::TypeSP &type)
void Insert(const lldb::TypeSP &type)
A class that contains all state required for type lookups.
ConstString GetTypeBasename() const
Get the type basename to use when searching the type indexes in each SymbolFile object.
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.
This class tracks the state and results of a TypeQuery.
bool InsertUnique(const lldb::TypeSP &type_sp)
When types that match a TypeQuery are found, this API is used to insert the matching types.
bool Done(const TypeQuery &query) const
Check if the type matching has found all of the matches that it needs.
bool AlreadySearched(lldb_private::SymbolFile *sym_file)
Check if a SymbolFile object has already been searched by this type match object.
A TypeSystem implementation based on Clang.
Interface for representing a type system.
virtual npdb::PdbAstBuilder * GetNativePDBParser()
@ eEncodingIsTypedefUID
This type is alias to a type whose UID is m_encoding_uid.
@ eEncodingIsUID
This type is the type whose UID is m_encoding_uid.
static std::optional< ParsedName > GetTypeScopeAndBasename(llvm::StringRef name)
void AddVariable(const lldb::VariableSP &var_sp)
RangeVector< lldb::addr_t, lldb::addr_t > RangeList
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.
static llvm::Expected< std::unique_ptr< PdbIndex > > create(llvm::pdb::PDBFile *)
llvm::pdb::TpiStream & tpi()
PdbCompilandId asCompiland() const
PdbCompilandSymId asCompilandSym() const
PdbTypeSymId asTypeSym() const
PdbSymUidKind kind() const
void CreateSimpleArgumentListTypes(llvm::codeview::TypeIndex arglist_ti)
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)
uint32_t CalculateNumCompileUnits() override
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)
llvm::BumpPtrAllocator m_allocator
void GetTypes(SymbolContextScope *sc_scope, lldb::TypeClass type_mask, TypeList &type_list) override
llvm::DenseMap< lldb::user_id_t, lldb::VariableSP > m_local_variables
~SymbolFileNativePDB() override
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)
void AddSymbols(Symtab &symtab) override
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
static llvm::StringRef GetPluginDescriptionStatic()
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)
lldb::addr_t m_obj_load_address
size_t ParseBlocksRecursive(Function &func) override
std::once_flag m_cached_udt_declarations
void CacheUdtDeclarations()
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)
uint32_t CalculateAbilities() override
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
static llvm::StringRef GetPluginNameStatic()
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)
bool m_done_full_type_scan
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.
static char ID
LLVM RTTI support.
uint32_t ResolveSymbolContext(const Address &so_addr, lldb::SymbolContextItem resolve_scope, SymbolContext &sc) override
std::unique_ptr< PdbIndex > m_index
llvm::DenseMap< lldb::user_id_t, lldb::VariableSP > m_global_vars
lldb::TypeSP CreateSimpleType(llvm::codeview::TypeIndex ti, CompilerType ct)
#define LLDB_INVALID_ADDRESS
uint64_t toOpaqueUid(const T &cid)
size_t GetTypeSizeForSimpleKind(llvm::codeview::SimpleTypeKind kind)
SegmentOffsetLength GetSegmentOffsetAndLength(const llvm::codeview::CVSymbol &sym)
bool IsTagRecord(llvm::codeview::CVType cvt)
bool IsValidRecord(const RecordT &sym)
DWARFExpression MakeGlobalLocationExpression(uint16_t section, uint32_t offset, lldb::ModuleSP module)
VariableInfo GetVariableLocationInfo(PdbIndex &index, PdbCompilandSymId var_id, Block &func_block, lldb::ModuleSP module)
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.
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.
@ 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
std::shared_ptr< lldb_private::VariableList > VariableListSP
std::shared_ptr< lldb_private::SymbolFileType > SymbolFileTypeSP
std::shared_ptr< lldb_private::Variable > VariableSP
std::shared_ptr< lldb_private::Section > SectionSP
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
void SetRangeEnd(BaseType end)
void SetRangeBase(BaseType b)
Set the start value for the range, and keep the same size.
lldb::user_id_t GetID() const
Get accessor for the user ID.
CompilerContextKind contextKind() const
static CVTagRecord create(llvm::codeview::CVType type)
const llvm::codeview::TagRecord & asTag() const
llvm::StringRef name() const
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
GlobalLineTable m_global_line_table
llvm::codeview::StringsAndChecksumsRef m_strings
std::vector< llvm::StringRef > m_file_list
llvm::codeview::TypeIndex index
DWARFExpressionList location
llvm::codeview::TypeIndex type