20#include "llvm/ADT/STLExtras.h"
21#include "llvm/DebugInfo/CodeView/SymbolDeserializer.h"
22#include "llvm/DebugInfo/CodeView/TypeDeserializer.h"
23#include "llvm/DebugInfo/CodeView/TypeIndex.h"
24#include "llvm/DebugInfo/PDB/Native/GlobalsStream.h"
25#include "llvm/DebugInfo/PDB/Native/TpiStream.h"
26#include "llvm/DebugInfo/PDB/PDBTypes.h"
40 llvm::DenseMap<clang::Decl *, DeclStatus> &decl_to_status,
42 llvm::SmallSet<std::pair<llvm::StringRef, CompilerType>, 8>>
44 : m_id(
id), m_derived_ct(derived_ct), m_tag_decl(tag_decl),
45 m_ast_builder(ast_builder), m_index(index),
46 m_decl_to_status(decl_to_status), m_cxx_record_map(cxx_record_map) {
50 m_cvr.
er.Options = ClassOptions::None;
51 llvm::cantFail(TypeDeserializer::deserializeAs<EnumRecord>(cvt,
m_cvr.
er));
54 m_cvr.
ur.Options = ClassOptions::None;
55 llvm::cantFail(TypeDeserializer::deserializeAs<UnionRecord>(cvt,
m_cvr.
ur));
61 m_cvr.
cr.Options = ClassOptions::None;
62 llvm::cantFail(TypeDeserializer::deserializeAs<ClassRecord>(cvt,
m_cvr.
cr));
67 llvm_unreachable(
"unreachable!");
72 llvm::codeview::TypeIndex ti, llvm::codeview::MemberAccess access,
73 std::optional<uint64_t> vtable_idx) {
79 std::unique_ptr<clang::CXXBaseSpecifier> base_spec =
82 vtable_idx.has_value(), udt_cvt.kind() == LF_CLASS);
87 std::make_pair(vtable_idx.value_or(0), std::move(base_spec)));
93 MemberAccess access, MethodOptions options,
94 MemberAttributes attrs) {
95 clang::QualType method_qt =
97 if (method_qt.isNull())
105 if (iter->getSecond().contains({name, method_ct})) {
111 bool is_artificial = (options & MethodOptions::CompilerGenerated) ==
112 MethodOptions::CompilerGenerated;
114 derived_opaque_ty, name.data(),
nullptr, method_ct,
115 access_type, attrs.isVirtual(), attrs.isStatic(),
false,
false,
false,
121Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
122 BaseClassRecord &base) {
123 clang::QualType base_qt =
126 if (base_qt.isNull())
127 return llvm::Error::success();
132 auto offset = clang::CharUnits::fromQuantity(base.getBaseOffset());
135 return llvm::Error::success();
138Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
139 VirtualBaseClassRecord &base) {
142 return Error::success();
145Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
146 ListContinuationRecord &cont) {
147 return Error::success();
150Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
151 VFPtrRecord &vfptr) {
152 return Error::success();
155Error UdtRecordCompleter::visitKnownMember(
156 CVMemberRecord &cvr, StaticDataMemberRecord &static_data_member) {
157 clang::QualType member_type =
159 if (member_type.isNull())
160 return llvm::Error::success();
167 m_derived_ct, static_data_member.Name, member_ct, access);
172 std::string qual_name = decl->getQualifiedNameAsString();
177 for (
const auto &result : results) {
178 if (result.second.kind() == SymbolKind::S_CONSTANT) {
179 ConstantSym constant(SymbolRecordKind::ConstantSym);
180 cantFail(SymbolDeserializer::deserializeAs<ConstantSym>(result.second,
183 clang::QualType qual_type = decl->getType();
184 unsigned type_width = decl->getASTContext().getIntWidth(qual_type);
185 unsigned constant_width = constant.Value.getBitWidth();
187 if (qual_type->isIntegralOrEnumerationType()) {
188 if (type_width >= constant_width) {
190 decl, constant.Value.extOrTrunc(type_width));
193 "Class '{0}' has a member '{1}' of type '{2}' ({3} bits) "
194 "which resolves to a wider constant value ({4} bits). "
195 "Ignoring constant.",
197 member_ct.
GetTypeName(), type_width, constant_width);
201 switch (basic_type_enum) {
205 if (type_width == constant_width) {
208 ? llvm::APFloat(constant.Value.bitsToFloat())
209 : llvm::APFloat(constant.Value.bitsToDouble()));
210 decl->setConstexpr(
true);
214 "Class '{0}' has a member '{1}' of type '{2}' ({3} bits) "
215 "which resolves to a constant value of mismatched width "
216 "({4} bits). Ignoring constant.",
218 member_ct.
GetTypeName(), type_width, constant_width);
232 return Error::success();
235Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
236 NestedTypeRecord &nested) {
237 return Error::success();
240Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
241 DataMemberRecord &data_member) {
243 uint64_t offset = data_member.FieldOffset * 8;
244 uint32_t bitfield_width = 0;
246 TypeIndex ti(data_member.Type);
247 if (!ti.isSimple()) {
249 if (cvt.kind() == LF_BITFIELD) {
251 llvm::cantFail(TypeDeserializer::deserializeAs<BitFieldRecord>(cvt, bfr));
252 offset += bfr.BitOffset;
253 bitfield_width = bfr.BitSize;
259 if (member_qt.isNull())
260 return Error::success();
266 return Error::success();
269 return Error::success();
272Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
273 OneMethodRecord &one_method) {
274 AddMethod(one_method.Name, one_method.Type, one_method.getAccess(),
275 one_method.getOptions(), one_method.Attrs);
277 return Error::success();
280Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
281 OverloadedMethodRecord &overloaded) {
282 TypeIndex method_list_idx = overloaded.MethodList;
284 CVType method_list_type =
m_index.
tpi().getType(method_list_idx);
285 assert(method_list_type.kind() == LF_METHODLIST);
287 MethodOverloadListRecord method_list;
288 llvm::cantFail(TypeDeserializer::deserializeAs<MethodOverloadListRecord>(
289 method_list_type, method_list));
291 for (
const OneMethodRecord &method : method_list.Methods)
292 AddMethod(overloaded.Name, method.Type, method.getAccess(),
293 method.getOptions(), method.Attrs);
295 return Error::success();
298Error UdtRecordCompleter::visitKnownMember(CVMemberRecord &cvr,
299 EnumeratorRecord &enumerator) {
304 m_derived_ct, decl, name.str().c_str(), enumerator.Value);
305 return Error::success();
310 llvm::stable_sort(
m_bases, llvm::less_first());
312 std::vector<std::unique_ptr<clang::CXXBaseSpecifier>> bases;
315 bases.push_back(std::move(ib.second));
321 for (
const auto &base_class : bases) {
322 clang::TypeSourceInfo *type_source_info =
323 base_class->getTypeSourceInfo();
324 if (type_source_info) {
326 clang.GetType(type_source_info->getType()));
337 if (
auto *record_decl = llvm::dyn_cast<clang::CXXRecordDecl>(&
m_tag_decl)) {
346 clang::DeclContext *parent_decl_ctx) {
348 clang.GetSymbolFile()->GetBackingSymbolFile());
349 clang::FieldDecl *field_decl =
nullptr;
350 uint64_t bit_size = 0;
351 switch (field->
kind) {
362 ? clang::TagTypeKind::Struct
363 : clang::TagTypeKind::Union;
372 clang::DeclContext *decl_ctx =
clang.GetDeclContextForType(record_ct);
373 for (
const auto &member : field->
fields) {
377 uint64_t member_bit_size =
AddMember(
clang, member.get(), member_offset,
378 record_ct, layout, decl_ctx);
380 bit_size = std::max(bit_size, member_offset + member_bit_size);
382 bit_size = std::max(bit_size, member_bit_size);
386 clang::RecordDecl *record_decl =
clang.GetAsRecordDecl(record_ct);
400 parent_layout.
field_offsets.insert({field_decl, bit_offset});
406 clang::DeclContext *decl_ctx =
418 llvm::StringRef name, uint64_t offset, uint64_t field_size,
420 fields_map[offset].push_back(std::make_unique<Member>(
421 name, offset, field_size, qt, access, bitfield_width));
445 std::map<uint64_t, std::vector<Member *>> end_offset_map;
446 for (
auto &pair : fields_map) {
447 uint64_t offset = pair.first;
448 auto &fields = pair.second;
451 if (offset > start_offset) {
456 auto iter = end_offset_map.lower_bound(offset);
457 if (iter == end_offset_map.end())
459 else if (iter->first > offset) {
460 if (iter == end_offset_map.begin())
464 if (iter->second.empty())
466 parent = iter->second.back();
467 iter->second.pop_back();
474 if (fields.size() == 1) {
475 uint64_t end_offset = offset + fields.back()->bit_size;
476 parent->
fields.push_back(std::move(fields.back()));
478 end_offset_map[end_offset].push_back(parent);
481 "If parent is union, it must be the top level record.");
482 end_offset_map[end_offset].push_back(parent->
fields.back().get());
487 parent = parent->
fields.back().get();
491 "If parent is union, it must be the top level record.");
493 for (
auto &field : fields) {
494 int64_t bit_size = field->bit_size;
495 parent->
fields.push_back(std::move(field));
496 end_offset_map[offset + bit_size].push_back(
497 parent->
fields.back().get());
#define LLDB_LOG(log,...)
The LLDB_LOG* macros defined below are the way to emit log messages.
void SetRecordLayout(clang::RecordDecl *decl, const LayoutInfo &layout)
Sets the layout for the given RecordDecl.
Generic representation of a type in a programming language.
lldb::BasicType GetBasicTypeEnumeration() const
bool IsCompleteType() const
lldb::opaque_compiler_type_t GetOpaqueQualType() const
ConstString GetTypeName(bool BaseOnly=false) const
A class that describes the declaration location of a lldb object.
A TypeSystem implementation based on Clang.
static clang::FieldDecl * AddFieldToRecordType(const CompilerType &type, llvm::StringRef name, const CompilerType &field_type, lldb::AccessType access, uint32_t bitfield_bit_size)
static void SetFloatingInitializerForVariable(clang::VarDecl *var, const llvm::APFloat &init_value)
Initializes a variable with a floating point value.
static void SetIntegerInitializerForVariable(clang::VarDecl *var, const llvm::APInt &init_value)
Initializes a variable with an integer value.
std::unique_ptr< clang::CXXBaseSpecifier > CreateBaseClassSpecifier(lldb::opaque_compiler_type_t type, lldb::AccessType access, bool is_virtual, bool base_of_class)
static void BuildIndirectFields(const CompilerType &type)
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::EnumConstantDecl * AddEnumerationValueToEnumerationType(const CompilerType &enum_type, const Declaration &decl, const char *name, int64_t enum_value, uint32_t enum_value_bit_size)
static clang::CXXRecordDecl * GetAsCXXRecordDecl(lldb::opaque_compiler_type_t type)
static bool CompleteTagDeclarationDefinition(const CompilerType &type)
static bool StartTagDeclarationDefinition(const CompilerType &type)
static clang::VarDecl * AddVariableToRecordType(const CompilerType &type, llvm::StringRef name, const CompilerType &var_type, lldb::AccessType access)
static void RequireCompleteType(CompilerType type)
Complete a type from debug info, or mark it as forcefully completed if there is no definition of the ...
ClangASTImporter & GetClangASTImporter()
CompilerType ToCompilerType(clang::QualType qt)
TypeSystemClang & clang()
clang::DeclContext * GetOrCreateDeclContextForUid(PdbSymUid uid)
clang::QualType GetOrCreateType(PdbTypeSymId type)
PdbIndex - Lazy access to the important parts of a PDB file.
llvm::pdb::GlobalsStream & globals()
llvm::pdb::SymbolStream & symrecords()
llvm::pdb::TpiStream & tpi()
lldb::user_id_t anonymous_id
std::vector< IndexedBase > m_bases
llvm::DenseMap< lldb::opaque_compiler_type_t, llvm::SmallSet< std::pair< llvm::StringRef, CompilerType >, 8 > > & m_cxx_record_map
ClangASTImporter::LayoutInfo m_layout
clang::TagDecl & m_tag_decl
CompilerType & m_derived_ct
llvm::DenseMap< clang::Decl *, DeclStatus > & m_decl_to_status
uint64_t AddMember(TypeSystemClang &clang, Member *field, uint64_t bit_offset, CompilerType parent_ct, ClangASTImporter::LayoutInfo &parent_layout, clang::DeclContext *decl_ctx)
clang::QualType AddBaseClassForTypeIndex(llvm::codeview::TypeIndex ti, llvm::codeview::MemberAccess access, std::optional< uint64_t > vtable_idx=std::optional< uint64_t >())
void AddMethod(llvm::StringRef name, llvm::codeview::TypeIndex type_idx, llvm::codeview::MemberAccess access, llvm::codeview::MethodOptions options, llvm::codeview::MemberAttributes attrs)
PdbAstBuilder & m_ast_builder
union lldb_private::npdb::UdtRecordCompleter::UdtTagRecord m_cvr
UdtRecordCompleter(PdbTypeSymId id, CompilerType &derived_ct, clang::TagDecl &tag_decl, PdbAstBuilder &ast_builder, PdbIndex &index, llvm::DenseMap< clang::Decl *, DeclStatus > &decl_to_status, llvm::DenseMap< lldb::opaque_compiler_type_t, llvm::SmallSet< std::pair< llvm::StringRef, CompilerType >, 8 > > &cxx_record_map)
llvm::StringRef DropNameScope(llvm::StringRef name)
lldb::AccessType TranslateMemberAccess(llvm::codeview::MemberAccess access)
size_t GetSizeOfType(PdbTypeSymId id, llvm::pdb::TpiStream &tpi)
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.
void * opaque_compiler_type_t
BasicType
Basic types enumeration for the public API SBType::GetBasicType().
@ eLanguageTypeC_plus_plus
ISO C++:1998.
llvm::DenseMap< const clang::FieldDecl *, uint64_t > field_offsets
llvm::codeview::TypeIndex index
llvm::SmallVector< MemberUP, 1 > fields
enum lldb_private::npdb::UdtRecordCompleter::Member::Kind kind
void CollectMember(llvm::StringRef name, uint64_t offset, uint64_t field_size, clang::QualType qt, lldb::AccessType access, uint64_t bitfield_width)
std::map< uint64_t, llvm::SmallVector< MemberUP, 1 > > fields_map
llvm::codeview::EnumRecord er
llvm::codeview::ClassRecord cr
llvm::codeview::UnionRecord ur