1 #include "UdtRecordCompleter.h"
3 #include "PdbAstBuilder.h"
8 #include "lldb/Symbol/ClangASTContext.h"
9 #include "lldb/Symbol/ClangASTImporter.h"
10 #include "lldb/Symbol/Type.h"
11 #include "lldb/Utility/LLDBAssert.h"
12 #include "lldb/lldb-enumerations.h"
13 #include "lldb/lldb-forward.h"
15 #include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
16 #include "llvm/DebugInfo/CodeView/TypeIndex.h"
17 #include "llvm/DebugInfo/PDB/Native/TpiStream.h"
18 #include "llvm/DebugInfo/PDB/PDBTypes.h"
20 using namespace llvm::codeview;
21 using namespace llvm::pdb;
23 using namespace lldb_private;
24 using namespace lldb_private::npdb;
26 using Error = llvm::Error;
28 UdtRecordCompleter::UdtRecordCompleter(PdbTypeSymId id,
29 CompilerType &derived_ct,
30 clang::TagDecl &tag_decl,
31 PdbAstBuilder &ast_builder,
33 : m_id(id), m_derived_ct(derived_ct), m_tag_decl(tag_decl),
34 m_ast_builder(ast_builder), m_tpi(tpi) {
35 CVType cvt = m_tpi.getType(m_id.index);
38 llvm::cantFail(TypeDeserializer::deserializeAs<EnumRecord>(cvt, m_cvr.er));
41 llvm::cantFail(TypeDeserializer::deserializeAs<UnionRecord>(cvt, m_cvr.ur));
45 llvm::cantFail(TypeDeserializer::deserializeAs<ClassRecord>(cvt, m_cvr.cr));
48 llvm_unreachable("unreachable!");
52 clang::QualType UdtRecordCompleter::AddBaseClassForTypeIndex(
53 llvm::codeview::TypeIndex ti, llvm::codeview::MemberAccess access) {
54 PdbTypeSymId type_id(ti);
55 clang::QualType qt = m_ast_builder.GetOrCreateType(type_id);
57 CVType udt_cvt = m_tpi.getType(ti);
59 std::unique_ptr<clang::CXXBaseSpecifier> base_spec =
60 m_ast_builder.clang().CreateBaseClassSpecifier(
61 qt.getAsOpaquePtr(), TranslateMemberAccess(access), false,
62 udt_cvt.kind() == LF_CLASS);
63 lldbassert(base_spec);
64 m_bases.push_back(std::move(base_spec));
68 Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
69 BaseClassRecord &base) {
70 clang::QualType base_qt =
71 AddBaseClassForTypeIndex(base.Type, base.getAccess());
74 m_ast_builder.clang().GetAsCXXRecordDecl(base_qt.getAsOpaquePtr());
77 auto offset = clang::CharUnits::fromQuantity(base.getBaseOffset());
78 m_layout.base_offsets.insert(std::make_pair(decl, offset));
80 return llvm::Error::success();
83 Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
84 VirtualBaseClassRecord &base) {
85 AddBaseClassForTypeIndex(base.BaseType, base.getAccess());
87 // FIXME: Handle virtual base offsets.
88 return Error::success();
91 Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
92 ListContinuationRecord &cont) {
93 return Error::success();
96 Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
98 return Error::success();
101 Error UdtRecordCompleter::visitKnownMember(
102 CVMemberRecord &cvr, StaticDataMemberRecord &static_data_member) {
103 clang::QualType member_type =
104 m_ast_builder.GetOrCreateType(PdbTypeSymId(static_data_member.Type));
106 m_ast_builder.CompleteType(member_type);
108 CompilerType member_ct = m_ast_builder.ToCompilerType(member_type);
110 lldb::AccessType access =
111 TranslateMemberAccess(static_data_member.getAccess());
112 ClangASTContext::AddVariableToRecordType(
113 m_derived_ct, static_data_member.Name, member_ct, access);
115 // FIXME: Add a PdbSymUid namespace for field list members and update
116 // the m_uid_to_decl map with this decl.
117 return Error::success();
120 Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
121 NestedTypeRecord &nested) {
122 return Error::success();
125 Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
126 DataMemberRecord &data_member) {
128 uint64_t offset = data_member.FieldOffset * 8;
129 uint32_t bitfield_width = 0;
131 TypeIndex ti(data_member.Type);
132 if (!ti.isSimple()) {
133 CVType cvt = m_tpi.getType(ti);
134 if (cvt.kind() == LF_BITFIELD) {
136 llvm::cantFail(TypeDeserializer::deserializeAs<BitFieldRecord>(cvt, bfr));
137 offset += bfr.BitOffset;
138 bitfield_width = bfr.BitSize;
143 clang::QualType member_qt = m_ast_builder.GetOrCreateType(PdbTypeSymId(ti));
144 m_ast_builder.CompleteType(member_qt);
146 lldb::AccessType access = TranslateMemberAccess(data_member.getAccess());
148 clang::FieldDecl *decl = ClangASTContext::AddFieldToRecordType(
149 m_derived_ct, data_member.Name, m_ast_builder.ToCompilerType(member_qt),
150 access, bitfield_width);
151 // FIXME: Add a PdbSymUid namespace for field list members and update
152 // the m_uid_to_decl map with this decl.
154 m_layout.field_offsets.insert(std::make_pair(decl, offset));
156 return Error::success();
159 Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
160 OneMethodRecord &one_method) {
161 return Error::success();
164 Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
165 OverloadedMethodRecord &overloaded) {
166 return Error::success();
169 Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
170 EnumeratorRecord &enumerator) {
172 llvm::StringRef name = DropNameScope(enumerator.getName());
174 m_ast_builder.clang().AddEnumerationValueToEnumerationType(
175 m_derived_ct, decl, name.str().c_str(), enumerator.Value);
176 return Error::success();
179 void UdtRecordCompleter::complete() {
180 ClangASTContext &clang = m_ast_builder.clang();
181 clang.TransferBaseClasses(m_derived_ct.GetOpaqueQualType(),
184 clang.AddMethodOverridesForCXXRecordType(m_derived_ct.GetOpaqueQualType());
185 ClangASTContext::BuildIndirectFields(m_derived_ct);
186 ClangASTContext::CompleteTagDeclarationDefinition(m_derived_ct);
188 if (auto *record_decl = llvm::dyn_cast<clang::CXXRecordDecl>(&m_tag_decl)) {
189 m_ast_builder.importer().InsertRecordDecl(record_decl, m_layout);