9#include "clang/AST/ASTContext.h"
10#include "clang/Basic/TargetInfo.h"
34std::map<ConstString, CXXFunctionSummaryFormat::Callback> &
36 static std::map<ConstString, CXXFunctionSummaryFormat::Callback> g_map;
40std::map<ConstString, CXXSyntheticChildren::CreateFrontEndCallback> &
42 static std::map<ConstString, CXXSyntheticChildren::CreateFrontEndCallback>
73template <
typename D32,
typename D64>
98 struct DataDescriptor_32 {
107 struct DataDescriptor_64 {
123 struct DataDescriptor_32 {
130 struct DataDescriptor_64 {
143 template <
typename PtrType>
158 template <
typename DD>
164 DD descriptor = DD();
165 process.
ReadMemory(start_of_descriptor, &descriptor,
166 sizeof(descriptor),
error);
170 return descriptor._used;
206template <
typename D32,
typename D64,
bool Inline>
327 static constexpr llvm::StringLiteral g_TypeHint(
"NSArray");
341 if (!descriptor || !descriptor->IsValid())
344 uint32_t ptr_size = process_sp->GetAddressByteSize();
353 ConstString class_name(descriptor->GetClassName());
357 static const ConstString g_NSArrayI_Transfer(
"__NSArrayI_Transfer");
358 static const ConstString g_NSFrozenArrayM(
"__NSFrozenArrayM");
360 static const ConstString g_NSArray1(
"__NSSingleObjectArrayI");
361 static const ConstString g_NSArrayCF(
"__NSCFArray");
362 static const ConstString g_NSArrayMLegacy(
"__NSArrayM_Legacy");
363 static const ConstString g_NSArrayMImmutable(
"__NSArrayM_Immutable");
364 static const ConstString g_NSCallStackArray(
"_NSCallStackArray");
365 static const ConstString g_NSConstantArray(
"NSConstantArray");
370 if (class_name == g_NSArrayI) {
372 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,
376 }
else if (class_name == g_NSConstantArray) {
378 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size, 8,
382 }
else if (class_name == g_NSArrayM) {
384 llvm::dyn_cast_or_null<AppleObjCRuntime>(runtime);
389 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,
394 }
else if (class_name == g_NSArrayI_Transfer) {
396 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,
400 }
else if (class_name == g_NSFrozenArrayM) {
405 }
else if (class_name == g_NSArrayMLegacy) {
407 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,
411 }
else if (class_name == g_NSArrayMImmutable) {
413 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,
417 }
else if (class_name == g_NSArray0) {
419 }
else if (class_name == g_NSArray1) {
421 }
else if (class_name == g_NSArrayCF || class_name == g_NSCallStackArray) {
425 value = process_sp->ReadUnsignedIntegerFromMemory(
426 valobj_addr + 2 * ptr_size, ptr_size, 0,
error);
431 auto iter = map.find(class_name), end = map.end();
433 return iter->second(valobj, stream, options);
438 llvm::StringRef prefix, suffix;
440 std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);
443 stream.
Printf(
"%" PRIu64
" %s%s", value,
"element", value == 1 ?
"" :
"s");
453 *valobj_sp->GetExecutionContextRef().GetTargetSP());
456 scratch_ts_sp->weak_from_this(),
457 scratch_ts_sp->getASTContext().ObjCBuiltinIdTy.getAsOpaquePtr());
458 if (valobj_sp->GetProcessSP())
459 m_ptr_size = valobj_sp->GetProcessSP()->GetAddressByteSize();
463template <
typename D32,
typename D64>
481 size_t pyhs_idx = idx;
487 idx_name.
Printf(
"[%" PRIu64
"]", (uint64_t)idx);
492template <
typename D32,
typename D64>
509 m_ptr_size = process_sp->GetAddressByteSize();
510 uint64_t data_location = valobj_sp->GetValueAsUnsigned(0) +
m_ptr_size;
513 process_sp->ReadMemory(data_location,
m_data_32,
sizeof(D32),
517 process_sp->ReadMemory(data_location,
m_data_64,
sizeof(D64),
525template <
typename D32,
typename D64>
534template <
typename D32,
typename D64>
536lldb_private::formatters::
537 GenericNSArrayMSyntheticFrontEnd<D32, D64>::
538 GenericNSArrayMSyntheticFrontEnd::GetDataAddress() {
539 if (!m_data_32 && !m_data_64)
541 return m_data_32 ? m_data_32->_data : m_data_64->_data;
544template <
typename D32,
typename D64>
546lldb_private::formatters::
547 GenericNSArrayMSyntheticFrontEnd<D32, D64>::
548 GenericNSArrayMSyntheticFrontEnd::GetUsedCount() {
549 if (!m_data_32 && !m_data_64)
551 return m_data_32 ? m_data_32->_used : m_data_64->_used;
554template <
typename D32,
typename D64>
556lldb_private::formatters::
557 GenericNSArrayMSyntheticFrontEnd<D32, D64>::
558 GenericNSArrayMSyntheticFrontEnd::GetOffset() {
559 if (!m_data_32 && !m_data_64)
561 return m_data_32 ? m_data_32->_offset : m_data_64->_offset;
564template <
typename D32,
typename D64>
566lldb_private::formatters::
567 GenericNSArrayMSyntheticFrontEnd<D32, D64>::
568 GenericNSArrayMSyntheticFrontEnd::GetSize() {
569 if (!m_data_32 && !m_data_64)
571 return m_data_32 ? m_data_32->_size : m_data_64->_size;
574template <
typename D32,
typename D64,
bool Inline>
583 *valobj_sp->GetExecutionContextRef().GetTargetSP());
586 scratch_ts_sp->getASTContext().ObjCBuiltinIdTy);
591template <
typename D32,
typename D64,
bool Inline>
600template <
typename D32,
typename D64,
bool Inline>
601llvm::Expected<uint32_t>
607template <
typename D32,
typename D64,
bool Inline>
625 m_ptr_size = process_sp->GetAddressByteSize();
626 uint64_t data_location = valobj_sp->GetValueAsUnsigned(0) +
m_ptr_size;
629 process_sp->ReadMemory(data_location,
m_data_32,
sizeof(D32),
633 process_sp->ReadMemory(data_location,
m_data_64,
sizeof(D64),
641template <
typename D32,
typename D64,
bool Inline>
650 object_at_idx +=
m_ptr_size == 4 ?
sizeof(D32) :
sizeof(D64);
664 idx_name.
Printf(
"[%" PRIu64
"]", (uint64_t)idx);
673llvm::Expected<size_t>
679llvm::Expected<uint32_t>
703llvm::Expected<size_t>
714llvm::Expected<uint32_t>
734 return m_backend.GetSyntheticChildAtOffset(
735 m_backend.GetProcessSP()->GetAddressByteSize(), id_type,
true,
757 Flags flags(valobj_type.GetTypeInfo());
759 if (flags.IsClear(eTypeIsPointer)) {
761 valobj_sp = valobj_sp->AddressOf(
error);
762 if (
error.Fail() || !valobj_sp)
769 if (!descriptor || !descriptor->IsValid())
772 ConstString class_name(descriptor->GetClassName());
775 static const ConstString g_NSConstantArray(
"NSConstantArray");
776 static const ConstString g_NSArrayI_Transfer(
"__NSArrayI_Transfer");
777 static const ConstString g_NSFrozenArrayM(
"__NSFrozenArrayM");
780 static const ConstString g_NSArray1(
"__NSSingleObjectArrayI");
781 static const ConstString g_NSArrayMLegacy(
"__NSArrayM_Legacy");
782 static const ConstString g_NSArrayMImmutable(
"__NSArrayM_Immutable");
783 static const ConstString g_NSCallStackArray(
"_NSCallStackArray");
785 if (class_name.IsEmpty())
788 if (class_name == g_NSArrayI) {
794 }
else if (class_name == g_NSArrayI_Transfer) {
796 }
else if (class_name == g_NSConstantArray) {
798 }
else if (class_name == g_NSFrozenArrayM) {
800 }
else if (class_name == g_NSArray0) {
802 }
else if (class_name == g_NSArray1) {
804 }
else if (class_name == g_NSArrayM) {
811 }
else if (class_name == g_NSCallStackArray) {
815 auto iter = map.find(class_name), end = map.end();
817 return iter->second(synth, valobj_sp);
static llvm::raw_ostream & error(Stream &strm)
static std::optional< size_t > CalculateNumChildren(CompilerType container_elem_type, uint64_t num_elements, CompilerType element_type)
Calculates the number of elements stored in a container (with element type 'container_elem_type') as ...
uint32_t GetFoundationVersion()
Generic representation of a type in a programming language.
A uniqued constant string class.
bool IsEmpty() const
Test for empty string.
Execution context objects refer to objects in the execution of the program that is being debugged.
static Language * FindPlugin(lldb::LanguageType language)
std::shared_ptr< ClassDescriptor > ClassDescriptorSP
static ObjCLanguageRuntime * Get(Process &process)
virtual ClassDescriptorSP GetClassDescriptor(ValueObject &in_value)
A plug-in interface definition class for debugging a process.
virtual size_t ReadMemory(lldb::addr_t vm_addr, void *buf, size_t size, Status &error)
Read of memory from a process.
uint32_t GetAddressByteSize() const
static lldb::TypeSystemClangSP GetForTarget(Target &target, std::optional< IsolatedASTKind > ast_kind=DefaultAST, bool create_on_demand=true)
Returns the scratch TypeSystemClang for the given target.
llvm::StringRef GetString() const
A stream class that can stream formatted output to a file.
size_t Printf(const char *format,...) __attribute__((format(printf
Output printf formatted output to the stream.
uint32_t CalculateNumChildrenIgnoringErrors(uint32_t max=UINT32_MAX)
SyntheticChildrenFrontEnd(ValueObject &backend)
lldb::ValueObjectSP CreateValueObjectFromAddress(llvm::StringRef name, uint64_t address, const ExecutionContext &exe_ctx, CompilerType type, bool do_deref=true)
lldb::LanguageType GetLanguage() const
lldb::ProcessSP GetProcessSP() const
virtual uint64_t GetValueAsUnsigned(uint64_t fail_value, bool *success=nullptr)
#define LLDB_INVALID_ADDRESS
A class that represents a running process on the host machine.
ChildCacheState
Specifies if children need to be re-computed after a call to SyntheticChildrenFrontEnd::Update.
@ eRefetch
Children need to be recomputed dynamically.
@ eReuse
Children did not change and don't need to be recomputed; re-use what we computed the last time we cal...
std::shared_ptr< lldb_private::ValueObject > ValueObjectSP
std::shared_ptr< lldb_private::Process > ProcessSP
std::shared_ptr< lldb_private::TypeSystemClang > TypeSystemClangSP