3#include "llvm/DebugInfo/CodeView/CVTypeVisitor.h"
4#include "llvm/DebugInfo/CodeView/LazyRandomTypeCollection.h"
5#include "llvm/DebugInfo/CodeView/RecordName.h"
6#include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
7#include "llvm/DebugInfo/CodeView/SymbolRecord.h"
8#include "llvm/DebugInfo/CodeView/SymbolRecordHelpers.h"
9#include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
10#include "llvm/DebugInfo/CodeView/TypeVisitorCallbacks.h"
11#include "llvm/DebugInfo/PDB/Native/DbiStream.h"
12#include "llvm/DebugInfo/PDB/Native/PublicsStream.h"
13#include "llvm/DebugInfo/PDB/Native/SymbolStream.h"
14#include "llvm/DebugInfo/PDB/Native/TpiStream.h"
15#include "llvm/Demangle/MicrosoftDemangle.h"
38 TypeIndex func_type_index,
39 clang::FunctionDecl *&function_decl,
42 : m_index(m_index), m_clang(m_clang), func_type_index(func_type_index),
43 function_decl(function_decl), parent_ty(parent_ty),
44 proc_name(proc_name), func_ct(func_ct) {}
47 TypeIndex func_type_index;
48 clang::FunctionDecl *&function_decl;
50 llvm::StringRef proc_name;
53 llvm::Error visitKnownMember(CVMemberRecord &cvr,
54 OverloadedMethodRecord &overloaded)
override {
55 TypeIndex method_list_idx = overloaded.MethodList;
57 CVType method_list_type = m_index.
tpi().getType(method_list_idx);
58 assert(method_list_type.kind() == LF_METHODLIST);
60 MethodOverloadListRecord method_list;
61 llvm::cantFail(TypeDeserializer::deserializeAs<MethodOverloadListRecord>(
62 method_list_type, method_list));
64 for (
const OneMethodRecord &method : method_list.Methods) {
65 if (method.getType().getIndex() == func_type_index.getIndex())
66 AddMethod(overloaded.Name, method.getAccess(), method.getOptions(),
70 return llvm::Error::success();
73 llvm::Error visitKnownMember(CVMemberRecord &cvr,
74 OneMethodRecord &record)
override {
75 AddMethod(record.getName(), record.getAccess(), record.getOptions(),
77 return llvm::Error::success();
80 void AddMethod(llvm::StringRef name, MemberAccess access,
81 MethodOptions options, MemberAttributes attrs) {
82 if (name != proc_name || function_decl)
85 bool is_virtual = attrs.isVirtual();
86 bool is_static = attrs.isStatic();
87 bool is_artificial = (options & MethodOptions::CompilerGenerated) ==
88 MethodOptions::CompilerGenerated;
91 nullptr, func_ct, access_type,
92 is_virtual, is_static,
94 false, is_artificial);
101 case TypeRecordKind::Class:
102 return clang::TagTypeKind::Class;
103 case TypeRecordKind::Struct:
104 return clang::TagTypeKind::Struct;
105 case TypeRecordKind::Union:
106 return clang::TagTypeKind::Union;
107 case TypeRecordKind::Interface:
108 return clang::TagTypeKind::Interface;
109 case TypeRecordKind::Enum:
110 return clang::TagTypeKind::Enum;
112 lldbassert(
false &&
"Invalid tag record kind!");
113 return clang::TagTypeKind::Struct;
120 return args.back() == TypeIndex::None();
125 for (llvm::ms_demangle::Node *n : scopes) {
126 auto *idn =
static_cast<llvm::ms_demangle::IdentifierNode *
>(n);
127 if (idn->TemplateParams)
133static std::optional<clang::CallingConv>
135 using CC = llvm::codeview::CallingConvention;
140 return clang::CallingConv::CC_C;
143 return clang::CallingConv::CC_X86Pascal;
146 return clang::CallingConv::CC_X86FastCall;
147 case CC::NearStdCall:
149 return clang::CallingConv::CC_X86StdCall;
151 return clang::CallingConv::CC_X86ThisCall;
153 return clang::CallingConv::CC_X86VectorCall;
160 return name ==
"`anonymous namespace'" || name ==
"`anonymous-namespace'";
169std::pair<clang::DeclContext *, std::string>
174 if (!record.hasUniqueName())
177 llvm::ms_demangle::Demangler demangler;
178 std::string_view sv(record.UniqueName.begin(), record.UniqueName.size());
179 llvm::ms_demangle::TagTypeNode *ttn = demangler.parseTagUniqueName(sv);
183 llvm::ms_demangle::IdentifierNode *idn =
184 ttn->QualifiedName->getUnqualifiedIdentifier();
185 std::string uname = idn->toString(llvm::ms_demangle::OF_NoTagSpecifier);
187 llvm::ms_demangle::NodeArrayNode *name_components =
188 ttn->QualifiedName->Components;
189 llvm::ArrayRef<llvm::ms_demangle::Node *> scopes(name_components->Nodes,
190 name_components->Count - 1);
197 std::optional<TypeIndex> parent_index = pdb->
GetParentType(ti);
200 return {context, uname};
208 return {context, std::string(record.Name)};
210 for (llvm::ms_demangle::Node *scope : scopes) {
211 auto *nii =
static_cast<llvm::ms_demangle::NamedIdentifierNode *
>(scope);
212 std::string str = nii->toString();
215 return {context, uname};
223 if (parent_qt.isNull())
224 return {
nullptr,
""};
226 context = clang::TagDecl::castToDeclContext(parent_qt->getAsTagDecl());
227 return {context, uname};
252 clang::Decl *scope_decl = clang::Decl::castFromDeclContext(scope);
258 switch (cvs.kind()) {
277std::optional<CompilerDecl>
282 clang::Decl *result =
nullptr;
283 switch (uid.
kind()) {
291 if (
auto *tag = qt->getAsTagDecl()) {
319 return clang::Decl::castToDeclContext(decl);
322std::pair<clang::DeclContext *, std::string>
328 llvm::ArrayRef<MSVCUndecoratedNameSpecifier> specs = parser.
GetSpecifiers();
332 llvm::StringRef uname = specs.back().GetBaseName();
333 specs = specs.drop_back();
335 return {context, std::string(name)};
337 llvm::StringRef scope_name = specs.back().GetFullName();
340 std::vector<TypeIndex> types = index.
tpi().findRecordsByName(scope_name);
341 while (!types.empty()) {
345 clang::TagDecl *tag = qt->getAsTagDecl();
347 return {clang::TagDecl::castToDeclContext(tag), std::string(uname)};
353 std::string ns_name = spec.GetBaseName().str();
356 return {context, std::string(uname)};
365 switch (uid.
kind()) {
367 std::optional<PdbCompilandSymId> scope =
396 switch (global.kind()) {
397 case SymbolKind::S_GDATA32:
398 case SymbolKind::S_LDATA32:
400 case SymbolKind::S_PROCREF:
401 case SymbolKind::S_LPROCREF: {
402 ProcRefSym ref{global.kind()};
404 SymbolDeserializer::deserializeAs<ProcRefSym>(global, ref));
408 case SymbolKind::S_CONSTANT:
409 case SymbolKind::S_UDT:
425 clang::TagDecl *tag = qt->getAsTagDecl();
426 if (qt->isArrayType()) {
427 const clang::Type *element_type = qt->getArrayElementTypeNoTypeQual();
428 tag = element_type->getAsTagDecl();
456 CVType cvt = index.
tpi().getType(tag_ti);
457 if (cvt.kind() == LF_MODIFIER)
461 cvt = index.
tpi().getType(best_ti.
index);
471 CVType field_list_cvt = index.
tpi().getType(field_list_ti);
472 if (field_list_cvt.kind() != LF_FIELDLIST)
474 FieldListRecord field_list;
475 if (llvm::Error
error = TypeDeserializer::deserializeAs<FieldListRecord>(
476 field_list_cvt, field_list))
477 llvm::consumeError(std::move(
error));
485 llvm::codeview::visitMemberRecordStream(field_list.Data, completer);
490 llvm::consumeError(std::move(
error));
497 if (ti == TypeIndex::NullptrT())
500 if (ti.getSimpleMode() != SimpleTypeMode::Direct) {
502 if (direct_type.isNull())
507 if (ti.getSimpleKind() == SimpleTypeKind::NotTranslated)
522 if (pointee_type.isNull())
525 if (pointer.isPointerToMember()) {
526 MemberPointerInfo mpi = pointer.getMemberInfo();
528 if (class_type.isNull())
530 if (clang::TagDecl *tag = class_type->getAsTagDecl()) {
531 clang::MSInheritanceAttr::Spelling spelling;
532 switch (mpi.Representation) {
533 case llvm::codeview::PointerToMemberRepresentation::SingleInheritanceData:
534 case llvm::codeview::PointerToMemberRepresentation::
535 SingleInheritanceFunction:
537 clang::MSInheritanceAttr::Spelling::Keyword_single_inheritance;
539 case llvm::codeview::PointerToMemberRepresentation::
540 MultipleInheritanceData:
541 case llvm::codeview::PointerToMemberRepresentation::
542 MultipleInheritanceFunction:
544 clang::MSInheritanceAttr::Spelling::Keyword_multiple_inheritance;
546 case llvm::codeview::PointerToMemberRepresentation::
547 VirtualInheritanceData:
548 case llvm::codeview::PointerToMemberRepresentation::
549 VirtualInheritanceFunction:
551 clang::MSInheritanceAttr::Spelling::Keyword_virtual_inheritance;
553 case llvm::codeview::PointerToMemberRepresentation::Unknown:
555 clang::MSInheritanceAttr::Spelling::Keyword_unspecified_inheritance;
558 spelling = clang::MSInheritanceAttr::Spelling::SpellingNotCalculated;
561 tag->addAttr(clang::MSInheritanceAttr::CreateImplicit(
565 pointee_type, class_type.getTypePtr());
568 clang::QualType pointer_type;
569 if (pointer.getMode() == PointerMode::LValueReference)
571 else if (pointer.getMode() == PointerMode::RValueReference)
576 if ((pointer.getOptions() & PointerOptions::Const) != PointerOptions::None)
577 pointer_type.addConst();
579 if ((pointer.getOptions() & PointerOptions::Volatile) != PointerOptions::None)
580 pointer_type.addVolatile();
582 if ((pointer.getOptions() & PointerOptions::Restrict) != PointerOptions::None)
583 pointer_type.addRestrict();
590 clang::QualType unmodified_type =
GetOrCreateType(modifier.ModifiedType);
591 if (unmodified_type.isNull())
594 if ((modifier.Modifiers & ModifierOptions::Const) != ModifierOptions::None)
595 unmodified_type.addConst();
596 if ((modifier.Modifiers & ModifierOptions::Volatile) != ModifierOptions::None)
597 unmodified_type.addVolatile();
599 return unmodified_type;
603 const TagRecord &record) {
604 clang::DeclContext *context =
nullptr;
630 clang::QualType result =
644clang::NamespaceDecl *
646 clang::DeclContext &context) {
655 return llvm::dyn_cast<clang::BlockDecl>(decl);
659 clang::BlockDecl *block_decl =
672 clang::DeclContext &scope) {
693 return llvm::dyn_cast<clang::VarDecl>(decl);
708 return llvm::dyn_cast<clang::VarDecl>(decl);
718clang::TypedefNameDecl *
721 return llvm::dyn_cast<clang::TypedefNameDecl>(decl);
728 UDTSym udt = llvm::cantFail(SymbolDeserializer::deserializeAs<UDTSym>(sym));
734 if (qt.isNull() || !scope)
755 if (type.
index.isSimple())
761 CVType cvt = index.
tpi().getType(type.
index);
763 if (cvt.kind() == LF_MODIFIER) {
764 ModifierRecord modifier;
766 TypeDeserializer::deserializeAs<ModifierRecord>(cvt, modifier));
770 if (cvt.kind() == LF_POINTER) {
771 PointerRecord pointer;
773 TypeDeserializer::deserializeAs<PointerRecord>(cvt, pointer));
786 if (cvt.kind() == LF_ARRAY) {
788 llvm::cantFail(TypeDeserializer::deserializeAs<ArrayRecord>(cvt, ar));
792 if (cvt.kind() == LF_PROCEDURE) {
794 llvm::cantFail(TypeDeserializer::deserializeAs<ProcedureRecord>(cvt, pr));
798 if (cvt.kind() == LF_MFUNCTION) {
799 MemberFunctionRecord mfr;
801 TypeDeserializer::deserializeAs<MemberFunctionRecord>(cvt, mfr));
809 if (type.
index.isNoneType())
841 clang::TagDecl *tag = qt->getAsTagDecl();
853 llvm::StringRef func_name,
TypeIndex func_ti,
855 clang::StorageClass func_storage,
856 bool is_inline, clang::DeclContext *parent) {
857 clang::FunctionDecl *function_decl =
nullptr;
858 if (parent->isRecord()) {
862 clang::QualType parent_qt = llvm::cast<clang::TypeDecl>(parent)
864 ->getCanonicalTypeInternal();
870 if (iter->getSecond().contains({func_name, func_ct})) {
875 CVType cvt = index.
tpi().getType(func_ti);
876 MemberFunctionRecord func_record(
static_cast<TypeRecordKind
>(cvt.kind()));
877 llvm::cantFail(TypeDeserializer::deserializeAs<MemberFunctionRecord>(
879 TypeIndex class_index = func_record.getClassType();
881 CVType parent_cvt = index.
tpi().getType(class_index);
884 if (tag_record.isForwardRef()) {
885 llvm::Expected<TypeIndex> eti =
886 index.
tpi().findFullDeclForForwardRef(class_index);
891 if (!tag_record.FieldList.isSimple()) {
892 CVType field_list_cvt = index.
tpi().getType(tag_record.FieldList);
893 FieldListRecord field_list;
894 if (llvm::Error
error = TypeDeserializer::deserializeAs<FieldListRecord>(
895 field_list_cvt, field_list))
896 llvm::consumeError(std::move(
error));
897 CreateMethodDecl process(index,
m_clang, func_ti, function_decl,
898 parent_opaque_ty, func_name, func_ct);
899 if (llvm::Error err = visitMemberRecordStream(field_list.Data, process))
900 llvm::consumeError(std::move(err));
903 if (!function_decl) {
905 parent_opaque_ty, func_name,
919 return function_decl;
930 InlineSiteSym inline_site(
static_cast<SymbolRecordKind
>(sym.kind()));
931 cantFail(SymbolDeserializer::deserializeAs<InlineSiteSym>(sym, inline_site));
938 return llvm::dyn_cast<clang::FunctionDecl>(decl);
939 clang::FunctionDecl *function_decl =
941 if (function_decl ==
nullptr)
946 uint64_t inlinesite_uid =
toOpaqueUid(inlinesite_id);
949 status.
uid = inlinesite_uid;
958 return function_decl;
968 CVType func_cvt = index.
ipi().getType(func_tid.
index);
969 llvm::StringRef func_name;
971 clang::DeclContext *parent =
nullptr;
972 switch (func_cvt.kind()) {
974 MemberFuncIdRecord mfr;
976 TypeDeserializer::deserializeAs<MemberFuncIdRecord>(func_cvt, mfr));
977 func_name = mfr.getName();
978 func_ti = mfr.getFunctionType();
985 cantFail(TypeDeserializer::deserializeAs<FuncIdRecord>(func_cvt, fir));
986 func_name = fir.getName();
987 func_ti = fir.getFunctionType();
989 if (!fir.ParentScope.isNoneType()) {
990 CVType parent_cvt = index.
ipi().getType(fir.ParentScope);
991 if (parent_cvt.kind() == LF_STRING_ID) {
994 TypeDeserializer::deserializeAs<StringIdRecord>(parent_cvt, sir));
1001 lldbassert(
false &&
"Invalid function id type!");
1004 if (func_qt.isNull() || !parent)
1007 uint32_t param_count =
1008 llvm::cast<clang::FunctionProtoType>(func_qt)->getNumParams();
1010 clang::SC_None,
true, parent);
1013clang::FunctionDecl *
1016 return llvm::dyn_cast<clang::FunctionDecl>(decl);
1021 std::string context_name;
1022 if (clang::NamespaceDecl *ns = llvm::dyn_cast<clang::NamespaceDecl>(parent)) {
1023 context_name = ns->getQualifiedNameAsString();
1024 }
else if (clang::TagDecl *tag = llvm::dyn_cast<clang::TagDecl>(parent)) {
1025 context_name = tag->getQualifiedNameAsString();
1032 ProcSym proc(
static_cast<SymbolRecordKind
>(cvs.kind()));
1033 llvm::cantFail(SymbolDeserializer::deserializeAs<ProcSym>(cvs, proc));
1040 clang::StorageClass storage = clang::SC_None;
1041 if (proc.Kind == SymbolRecordKind::ProcSym)
1042 storage = clang::SC_Static;
1044 const clang::FunctionProtoType *func_type =
1045 llvm::dyn_cast<clang::FunctionProtoType>(qt);
1049 llvm::StringRef proc_name = proc.Name;
1050 proc_name.consume_front(context_name);
1051 proc_name.consume_front(
"::");
1052 clang::FunctionDecl *function_decl =
1054 func_type->getNumParams(), storage,
false, parent);
1055 if (function_decl ==
nullptr)
1065 return function_decl;
1069 clang::FunctionDecl &function_decl,
1070 uint32_t param_count) {
1075 CVSymbolArray scope =
1079 auto begin = scope.begin();
1080 auto end = scope.end();
1081 std::vector<clang::ParmVarDecl *> params;
1082 for (uint32_t i = 0; i < param_count && begin != end;) {
1083 uint32_t record_offset = begin.offset();
1084 CVSymbol sym = *begin++;
1087 llvm::StringRef param_name;
1088 switch (sym.kind()) {
1090 RegRelativeSym reg(SymbolRecordKind::RegRelativeSym);
1091 cantFail(SymbolDeserializer::deserializeAs<RegRelativeSym>(sym, reg));
1092 param_type = reg.Type;
1093 param_name = reg.Name;
1097 RegisterSym reg(SymbolRecordKind::RegisterSym);
1098 cantFail(SymbolDeserializer::deserializeAs<RegisterSym>(sym, reg));
1099 param_type = reg.Index;
1100 param_name = reg.Name;
1104 LocalSym local(SymbolRecordKind::LocalSym);
1105 cantFail(SymbolDeserializer::deserializeAs<LocalSym>(sym, local));
1106 if ((local.Flags & LocalSymFlags::IsParameter) == LocalSymFlags::None)
1108 param_type = local.Type;
1109 param_name = local.Name;
1131 param_type_ct, clang::SC_None,
true);
1135 params.push_back(param);
1139 if (!params.empty() && params.size() == param_count)
1144 const EnumRecord &er) {
1145 clang::DeclContext *decl_context =
nullptr;
1152 if (underlying_type.isNull())
1173 if (element_type.isNull() || element_size == 0)
1175 uint64_t element_count = ar.Size / element_size;
1178 element_count,
false);
1184 llvm::codeview::CallingConvention calling_convention) {
1188 TpiStream &stream = index.
tpi();
1189 CVType args_cvt = stream.getType(args_type_idx);
1192 TypeDeserializer::deserializeAs<ArgListRecord>(args_cvt, args));
1194 llvm::ArrayRef<TypeIndex> arg_indices = llvm::ArrayRef(args.ArgIndices);
1197 arg_indices = arg_indices.drop_back();
1199 std::vector<CompilerType> arg_types;
1200 arg_types.reserve(arg_indices.size());
1202 for (
TypeIndex arg_index : arg_indices) {
1204 if (arg_type.isNull())
1210 if (return_type.isNull())
1213 std::optional<clang::CallingConv> cc =
1220 return_ct, arg_types.data(), arg_types.size(), is_variadic, 0, *cc);
1222 return clang::QualType::getFromOpaquePtr(
1227 return llvm::isa<clang::TagDecl>(&context);
1231 return llvm::isa<clang::FunctionDecl>(&context);
1235 return llvm::isa<clang::BlockDecl>(&context);
1239 clang::NamespaceDecl *ns = llvm::dyn_cast<clang::NamespaceDecl>(&context);
1242 std::string qname = ns->getQualifiedNameAsString();
1247 for (
const CVType &cvt : index.
tpi().typeArray()) {
1259 clang::DeclContext *context =
nullptr;
1262 if (!context || !context->isNamespace())
1265 clang::NamespaceDecl *ns = llvm::cast<clang::NamespaceDecl>(context);
1266 llvm::StringRef ns_name = ns->getName();
1267 if (ns_name.starts_with(qname)) {
1268 ns_name = ns_name.drop_front(qname.size());
1269 if (ns_name.starts_with(
"::"))
1283 for (
const CVType &cvt : index.
tpi().typeArray()) {
1300 uint32_t module_count = index.
dbi().modules().getModuleCount();
1301 for (uint16_t modi = 0; modi < module_count; ++modi) {
1303 const CVSymbolArray &symbols = cii.
m_debug_stream.getSymbolArray();
1304 auto iter = symbols.begin();
1305 while (iter != symbols.end()) {
1308 switch (iter->kind()) {
1312 iter = symbols.at(getScopeEndOffset(*iter));
1331 const CVSymbolArray &symbols) {
1332 clang::FunctionDecl *func_decl = llvm::dyn_cast<clang::FunctionDecl>(&decl);
1335 unsigned int params = func_decl->getNumParams();
1339 CVSymbolArray result = symbols;
1341 while (!result.empty()) {
1345 CVSymbol sym = *result.begin();
1346 result.drop_front();
1361 lldbassert(sym.kind() == S_GPROC32 || sym.kind() == S_LPROC32 ||
1362 sym.kind() == S_BLOCK32 || sym.kind() == S_INLINESITE);
1365 CVSymbolArray symbols =
1370 if (sym.kind() == S_GPROC32 || sym.kind() == S_LPROC32)
1374 symbols.drop_front();
1375 auto begin = symbols.begin();
1376 while (begin != symbols.end()) {
1379 if (begin->kind() == S_BLOCK32 || begin->kind() == S_INLINESITE) {
1381 begin = symbols.at(getScopeEndOffset(*begin));
1389 clang::Decl *decl = clang::Decl::castFromDeclContext(&context);
1395 if (
auto *tag = llvm::dyn_cast<clang::TagDecl>(&context)) {
1413 if (context.isTranslationUnit()) {
1419 if (context.isNamespace()) {
1435 return {
m_clang.weak_from_this(), qt.getAsOpaquePtr()};
static llvm::raw_ostream & error(Stream &strm)
static bool isFunctionDecl(clang::DeclContext &context)
static bool isBlockDecl(clang::DeclContext &context)
static CVSymbolArray skipFunctionParameters(clang::Decl &decl, const CVSymbolArray &symbols)
static std::optional< clang::CallingConv > TranslateCallingConvention(llvm::codeview::CallingConvention conv)
static bool isTagDecl(clang::DeclContext &context)
static bool isLocalVariableType(SymbolKind K)
static bool IsAnonymousNamespaceName(llvm::StringRef name)
static clang::TagTypeKind TranslateUdtKind(const TagRecord &cr)
static bool IsCVarArgsFunction(llvm::ArrayRef< TypeIndex > args)
static bool AnyScopesHaveTemplateParams(llvm::ArrayRef< llvm::ms_demangle::Node * > scopes)
llvm::ArrayRef< MSVCUndecoratedNameSpecifier > GetSpecifiers() const
Represents a generic declaration context in a program.
void * GetOpaqueDeclContext() const
Represents a generic declaration such as a function declaration.
Generic representation of a type in a programming language.
lldb::opaque_compiler_type_t GetOpaqueQualType() const
CompilerType CreateTypedef(const char *name, const CompilerDeclContext &decl_ctx, uint32_t payload) const
Create a typedef to this type using "name" as the name of the typedef this type is valid and the type...
A class that describes the declaration location of a lldb object.
A stream class that can stream formatted output to a file.
llvm::raw_ostream & AsRawOstream()
Returns a raw_ostream that forwards the data to this Stream object.
virtual SymbolFile * GetBackingSymbolFile()
SymbolFileOnDemand class overrides this to return the underlying backing SymbolFile implementation th...
A TypeSystem implementation based on Clang.
clang::TranslationUnitDecl * GetTranslationUnitDecl()
static clang::TypedefNameDecl * GetAsTypedefDecl(const CompilerType &type)
CompilerType GetBasicType(lldb::BasicType type)
CompilerType CreateArrayType(const CompilerType &element_type, std::optional< size_t > element_count, bool is_vector)
CompilerType GetType(clang::QualType qt)
Creates a CompilerType from the given QualType with the current TypeSystemClang instance as the Compi...
CompilerDecl GetCompilerDecl(clang::Decl *decl)
Creates a CompilerDecl from the given Decl with the current TypeSystemClang instance as its typesyste...
void Dump(llvm::raw_ostream &output) override
CompilerDeclContext CreateDeclContext(clang::DeclContext *ctx)
Creates a CompilerDeclContext from the given DeclContext with the current TypeSystemClang instance as...
clang::FunctionDecl * CreateFunctionDeclaration(clang::DeclContext *decl_ctx, OptionalClangModuleID owning_module, llvm::StringRef name, const CompilerType &function_Type, clang::StorageClass storage, bool is_inline)
clang::CXXMethodDecl * AddMethodToCXXRecordType(lldb::opaque_compiler_type_t type, llvm::StringRef name, const char *mangled_name, const CompilerType &method_type, lldb::AccessType access, bool is_virtual, bool is_static, bool is_inline, bool is_explicit, bool is_attr_used, bool is_artificial)
clang::NamespaceDecl * GetUniqueNamespaceDeclaration(const char *name, clang::DeclContext *decl_ctx, OptionalClangModuleID owning_module, bool is_inline=false)
CompilerType CreateRecordType(clang::DeclContext *decl_ctx, OptionalClangModuleID owning_module, lldb::AccessType access_type, llvm::StringRef name, int kind, lldb::LanguageType language, std::optional< ClangASTMetadata > metadata=std::nullopt, bool exports_symbols=false)
static bool SetHasExternalStorage(lldb::opaque_compiler_type_t type, bool has_extern)
CompilerType CreateEnumerationType(llvm::StringRef name, clang::DeclContext *decl_ctx, OptionalClangModuleID owning_module, const Declaration &decl, const CompilerType &integer_qual_type, bool is_scoped)
clang::ParmVarDecl * CreateParameterDeclaration(clang::DeclContext *decl_ctx, OptionalClangModuleID owning_module, const char *name, const CompilerType ¶m_type, int storage, bool add_decl=false)
static bool StartTagDeclarationDefinition(const CompilerType &type)
CompilerType CreateFunctionType(const CompilerType &result_type, const CompilerType *args, unsigned num_args, bool is_variadic, unsigned type_quals, clang::CallingConv cc=clang::CC_C, clang::RefQualifierKind ref_qual=clang::RQ_None)
void SetFunctionParameters(clang::FunctionDecl *function_decl, llvm::ArrayRef< clang::ParmVarDecl * > params)
clang::ASTContext & getASTContext() const
Returns the clang::ASTContext instance managed by this TypeSystemClang.
clang::VarDecl * CreateVariableDeclaration(clang::DeclContext *decl_context, OptionalClangModuleID owning_module, const char *name, clang::QualType type)
clang::BlockDecl * CreateBlockDeclaration(clang::DeclContext *ctx, OptionalClangModuleID owning_module)
virtual SymbolFile * GetSymbolFile() const
CompilandIndexItem & GetOrCreateCompiland(uint16_t modi)
const CompilandIndexItem * GetCompiland(uint16_t modi) const
clang::QualType CreatePointerType(const llvm::codeview::PointerRecord &pointer)
bool CompleteTagDecl(clang::TagDecl &tag)
void ParseBlockChildren(PdbCompilandSymId block_id)
llvm::DenseMap< lldb::opaque_compiler_type_t, llvm::SmallSet< std::pair< llvm::StringRef, CompilerType >, 8 > > m_cxx_record_map
clang::FunctionDecl * CreateFunctionDecl(PdbCompilandSymId func_id, llvm::StringRef func_name, TypeIndex func_ti, CompilerType func_ct, uint32_t param_count, clang::StorageClass func_storage, bool is_inline, clang::DeclContext *parent)
lldb_private::CompilerDeclContext GetTranslationUnitDecl()
clang::QualType CreateModifierType(const llvm::codeview::ModifierRecord &modifier)
void ParseDeclsForSimpleContext(clang::DeclContext &context)
CompilerDecl ToCompilerDecl(clang::Decl &decl)
llvm::DenseMap< lldb::user_id_t, clang::QualType > m_uid_to_type
clang::QualType CreateFunctionType(TypeIndex args_type_idx, TypeIndex return_type_idx, llvm::codeview::CallingConvention calling_convention)
clang::DeclContext * GetParentDeclContext(PdbSymUid uid)
TypeSystemClang & m_clang
void ParseNamespace(clang::DeclContext &parent)
llvm::codeview::TypeIndex TypeIndex
CompilerDeclContext ToCompilerDeclContext(clang::DeclContext &context)
std::pair< clang::DeclContext *, std::string > CreateDeclInfoForUndecoratedName(llvm::StringRef uname)
clang::Decl * TryGetDecl(PdbSymUid uid) const
clang::DeclContext * FromCompilerDeclContext(CompilerDeclContext context)
clang::VarDecl * GetOrCreateVariableDecl(PdbCompilandSymId scope_id, PdbCompilandSymId var_id)
clang::Decl * GetOrCreateSymbolForId(PdbCompilandSymId id)
CompilerType ToCompilerType(clang::QualType qt)
clang::NamespaceDecl * GetOrCreateNamespaceDecl(const char *name, clang::DeclContext &context)
void Dump(Stream &stream)
PdbAstBuilder(TypeSystemClang &clang)
std::pair< clang::DeclContext *, std::string > CreateDeclInfoForType(const llvm::codeview::TagRecord &record, TypeIndex ti)
clang::QualType GetBasicType(lldb::BasicType type)
bool CompleteType(clang::QualType qt)
llvm::DenseSet< clang::NamespaceDecl * > m_parsed_namespaces
llvm::DenseMap< clang::Decl *, DeclStatus > m_decl_to_status
clang::DeclContext * GetOrCreateDeclContextForUid(PdbSymUid uid)
void ParseDeclsForContext(clang::DeclContext &context)
llvm::DenseMap< lldb::user_id_t, clang::Decl * > m_uid_to_decl
clang::BlockDecl * GetOrCreateBlockDecl(PdbCompilandSymId block_id)
clang::Decl * FromCompilerDecl(CompilerDecl decl)
clang::QualType CreateSimpleType(TypeIndex ti)
void CreateFunctionParameters(PdbCompilandSymId func_id, clang::FunctionDecl &function_decl, uint32_t param_count)
clang::FunctionDecl * GetOrCreateFunctionDecl(PdbCompilandSymId func_id)
llvm::once_flag m_parse_all_types
clang::QualType CreateArrayType(const llvm::codeview::ArrayRecord &array)
clang::QualType CreateRecordType(PdbTypeSymId id, const llvm::codeview::TagRecord &record)
clang::FunctionDecl * CreateFunctionDeclFromId(PdbTypeSymId func_tid, PdbCompilandSymId func_sid)
llvm::once_flag m_parse_functions_and_non_local_vars
clang::VarDecl * CreateVariableDecl(PdbSymUid uid, llvm::codeview::CVSymbol sym, clang::DeclContext &scope)
clang::QualType GetOrCreateType(PdbTypeSymId type)
clang::QualType CreateEnumType(PdbTypeSymId id, const llvm::codeview::EnumRecord &record)
clang::FunctionDecl * GetOrCreateInlinedFunctionDecl(PdbCompilandSymId inlinesite_id)
void ParseAllFunctionsAndNonLocalVars()
std::optional< lldb_private::CompilerDecl > GetOrCreateDeclForUid(PdbSymUid uid)
clang::QualType CreateType(PdbTypeSymId type)
clang::TypedefNameDecl * GetOrCreateTypedefDecl(PdbGlobalSymId id)
PdbIndex - Lazy access to the important parts of a PDB file.
llvm::pdb::TpiStream & ipi()
CompileUnitIndex & compilands()
llvm::pdb::DbiStream & dbi()
llvm::codeview::CVSymbol ReadSymbolRecord(PdbCompilandSymId cu_sym) const
llvm::pdb::TpiStream & tpi()
PdbGlobalSymId asGlobalSym() const
PdbCompilandSymId asCompilandSym() const
PdbTypeSymId asTypeSym() const
PdbSymUidKind kind() const
std::optional< llvm::codeview::TypeIndex > GetParentType(llvm::codeview::TypeIndex ti)
std::optional< PdbCompilandSymId > FindSymbolScope(PdbCompilandSymId id)
llvm::StringRef DropNameScope(llvm::StringRef name)
uint64_t toOpaqueUid(const T &cid)
lldb::BasicType GetCompilerTypeForSimpleKind(llvm::codeview::SimpleTypeKind kind)
VariableInfo GetVariableNameInfo(llvm::codeview::CVSymbol symbol)
bool IsTagRecord(llvm::codeview::CVType cvt)
llvm::codeview::TypeIndex LookThroughModifierRecord(llvm::codeview::CVType modifier)
bool IsForwardRefUdt(llvm::codeview::CVType cvt)
lldb::AccessType TranslateMemberAccess(llvm::codeview::MemberAccess access)
llvm::codeview::TypeIndex GetFieldListIndex(llvm::codeview::CVType cvt)
size_t GetSizeOfType(PdbTypeSymId id, llvm::pdb::TpiStream &tpi)
PdbTypeSymId GetBestPossibleDecl(PdbTypeSymId id, llvm::pdb::TpiStream &tpi)
A class that represents a running process on the host machine.
void * opaque_compiler_type_t
BasicType
Basic types enumeration for the public API SBType::GetBasicType().
@ eLanguageTypeC_plus_plus
ISO C++:1998.
static clang::Decl * GetDecl(const CompilerDecl &decl)
Returns the clang::Decl of the given CompilerDecl.
const llvm::codeview::UnionRecord & asUnion() const
static CVTagRecord create(llvm::codeview::CVType type)
const llvm::codeview::ClassRecord & asClass() const
const llvm::codeview::EnumRecord & asEnum() const
const llvm::codeview::TagRecord & asTag() const
Represents a single compile unit.
llvm::pdb::ModuleDebugStreamRef m_debug_stream
llvm::codeview::TypeIndex index
llvm::codeview::TypeIndex type