Path: blob/main/contrib/llvm-project/lldb/source/Plugins/Language/ObjC/NSArray.cpp
39644 views
//===-- NSArray.cpp -------------------------------------------------------===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//78#include "clang/AST/ASTContext.h"9#include "clang/Basic/TargetInfo.h"1011#include "Cocoa.h"1213#include "Plugins/LanguageRuntime/ObjC/AppleObjCRuntime/AppleObjCRuntime.h"14#include "Plugins/TypeSystem/Clang/TypeSystemClang.h"1516#include "lldb/Core/ValueObject.h"17#include "lldb/Core/ValueObjectConstResult.h"18#include "lldb/DataFormatters/FormattersHelpers.h"19#include "lldb/Expression/FunctionCaller.h"20#include "lldb/Target/Language.h"21#include "lldb/Target/Target.h"22#include "lldb/Utility/DataBufferHeap.h"23#include "lldb/Utility/Endian.h"24#include "lldb/Utility/Status.h"25#include "lldb/Utility/Stream.h"2627using namespace lldb;28using namespace lldb_private;29using namespace lldb_private::formatters;3031namespace lldb_private {32namespace formatters {33std::map<ConstString, CXXFunctionSummaryFormat::Callback> &34NSArray_Additionals::GetAdditionalSummaries() {35static std::map<ConstString, CXXFunctionSummaryFormat::Callback> g_map;36return g_map;37}3839std::map<ConstString, CXXSyntheticChildren::CreateFrontEndCallback> &40NSArray_Additionals::GetAdditionalSynthetics() {41static std::map<ConstString, CXXSyntheticChildren::CreateFrontEndCallback>42g_map;43return g_map;44}4546class NSArrayMSyntheticFrontEndBase : public SyntheticChildrenFrontEnd {47public:48NSArrayMSyntheticFrontEndBase(lldb::ValueObjectSP valobj_sp);4950~NSArrayMSyntheticFrontEndBase() override = default;5152llvm::Expected<uint32_t> CalculateNumChildren() override;5354lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override;5556lldb::ChildCacheState Update() override = 0;5758bool MightHaveChildren() override;5960size_t GetIndexOfChildWithName(ConstString name) override;6162protected:63virtual lldb::addr_t GetDataAddress() = 0;6465virtual uint64_t GetUsedCount() = 0;6667virtual uint64_t GetOffset() = 0;6869virtual uint64_t GetSize() = 0;7071ExecutionContextRef m_exe_ctx_ref;72uint8_t m_ptr_size = 8;73CompilerType m_id_type;74};7576template <typename D32, typename D64>77class GenericNSArrayMSyntheticFrontEnd : public NSArrayMSyntheticFrontEndBase {78public:79GenericNSArrayMSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp);8081~GenericNSArrayMSyntheticFrontEnd() override;8283lldb::ChildCacheState Update() override;8485protected:86lldb::addr_t GetDataAddress() override;8788uint64_t GetUsedCount() override;8990uint64_t GetOffset() override;9192uint64_t GetSize() override;9394private:95D32 *m_data_32;96D64 *m_data_64;97};9899namespace Foundation1010 {100namespace {101struct DataDescriptor_32 {102uint32_t _used;103uint32_t _offset;104uint32_t _size : 28;105uint64_t _priv1 : 4;106uint32_t _priv2;107uint32_t _data;108};109110struct DataDescriptor_64 {111uint64_t _used;112uint64_t _offset;113uint64_t _size : 60;114uint64_t _priv1 : 4;115uint32_t _priv2;116uint64_t _data;117};118}119120using NSArrayMSyntheticFrontEnd =121GenericNSArrayMSyntheticFrontEnd<DataDescriptor_32, DataDescriptor_64>;122}123124namespace Foundation1428 {125namespace {126struct DataDescriptor_32 {127uint32_t _used;128uint32_t _offset;129uint32_t _size;130uint32_t _data;131};132133struct DataDescriptor_64 {134uint64_t _used;135uint64_t _offset;136uint64_t _size;137uint64_t _data;138};139}140141using NSArrayMSyntheticFrontEnd =142GenericNSArrayMSyntheticFrontEnd<DataDescriptor_32, DataDescriptor_64>;143}144145namespace Foundation1437 {146template <typename PtrType>147struct DataDescriptor {148PtrType _cow;149// __deque150PtrType _data;151uint32_t _offset;152uint32_t _size;153uint32_t _muts;154uint32_t _used;155};156157using NSArrayMSyntheticFrontEnd =158GenericNSArrayMSyntheticFrontEnd<159DataDescriptor<uint32_t>, DataDescriptor<uint64_t>>;160161template <typename DD>162uint64_t163__NSArrayMSize_Impl(lldb_private::Process &process,164lldb::addr_t valobj_addr, Status &error) {165const lldb::addr_t start_of_descriptor =166valobj_addr + process.GetAddressByteSize();167DD descriptor = DD();168process.ReadMemory(start_of_descriptor, &descriptor,169sizeof(descriptor), error);170if (error.Fail()) {171return 0;172}173return descriptor._used;174}175176uint64_t177__NSArrayMSize(lldb_private::Process &process, lldb::addr_t valobj_addr,178Status &error) {179if (process.GetAddressByteSize() == 4) {180return __NSArrayMSize_Impl<DataDescriptor<uint32_t>>(process, valobj_addr,181error);182} else {183return __NSArrayMSize_Impl<DataDescriptor<uint64_t>>(process, valobj_addr,184error);185}186}187188}189190namespace CallStackArray {191struct DataDescriptor_32 {192uint32_t _data;193uint32_t _used;194uint32_t _offset;195const uint32_t _size = 0;196};197198struct DataDescriptor_64 {199uint64_t _data;200uint64_t _used;201uint64_t _offset;202const uint64_t _size = 0;203};204205using NSCallStackArraySyntheticFrontEnd =206GenericNSArrayMSyntheticFrontEnd<DataDescriptor_32, DataDescriptor_64>;207} // namespace CallStackArray208209template <typename D32, typename D64, bool Inline>210class GenericNSArrayISyntheticFrontEnd : public SyntheticChildrenFrontEnd {211public:212GenericNSArrayISyntheticFrontEnd(lldb::ValueObjectSP valobj_sp);213214~GenericNSArrayISyntheticFrontEnd() override;215216llvm::Expected<uint32_t> CalculateNumChildren() override;217218lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override;219220lldb::ChildCacheState Update() override;221222bool MightHaveChildren() override;223224size_t GetIndexOfChildWithName(ConstString name) override;225226private:227ExecutionContextRef m_exe_ctx_ref;228uint8_t m_ptr_size = 8;229230D32 *m_data_32;231D64 *m_data_64;232CompilerType m_id_type;233};234235namespace Foundation1300 {236struct IDD32 {237uint32_t used;238uint32_t list;239};240241struct IDD64 {242uint64_t used;243uint64_t list;244};245246using NSArrayISyntheticFrontEnd =247GenericNSArrayISyntheticFrontEnd<IDD32, IDD64, true>;248}249250namespace Foundation1430 {251using NSArrayISyntheticFrontEnd =252Foundation1428::NSArrayMSyntheticFrontEnd;253}254255namespace Foundation1436 {256struct IDD32 {257uint32_t used;258uint32_t list; // in Inline cases, this is the first element259};260261struct IDD64 {262uint64_t used;263uint64_t list; // in Inline cases, this is the first element264};265266using NSArrayI_TransferSyntheticFrontEnd =267GenericNSArrayISyntheticFrontEnd<IDD32, IDD64, false>;268269using NSArrayISyntheticFrontEnd =270GenericNSArrayISyntheticFrontEnd<IDD32, IDD64, true>;271272using NSFrozenArrayMSyntheticFrontEnd =273Foundation1437::NSArrayMSyntheticFrontEnd;274275uint64_t276__NSFrozenArrayMSize(lldb_private::Process &process, lldb::addr_t valobj_addr,277Status &error) {278return Foundation1437::__NSArrayMSize(process, valobj_addr, error);279}280}281282namespace ConstantArray {283284struct ConstantArray32 {285uint64_t used;286uint32_t list;287};288289struct ConstantArray64 {290uint64_t used;291uint64_t list;292};293294using NSConstantArraySyntheticFrontEnd =295GenericNSArrayISyntheticFrontEnd<ConstantArray32, ConstantArray64, false>;296} // namespace ConstantArray297298class NSArray0SyntheticFrontEnd : public SyntheticChildrenFrontEnd {299public:300NSArray0SyntheticFrontEnd(lldb::ValueObjectSP valobj_sp);301302~NSArray0SyntheticFrontEnd() override = default;303304llvm::Expected<uint32_t> CalculateNumChildren() override;305306lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override;307308lldb::ChildCacheState Update() override;309310bool MightHaveChildren() override;311312size_t GetIndexOfChildWithName(ConstString name) override;313};314315class NSArray1SyntheticFrontEnd : public SyntheticChildrenFrontEnd {316public:317NSArray1SyntheticFrontEnd(lldb::ValueObjectSP valobj_sp);318319~NSArray1SyntheticFrontEnd() override = default;320321llvm::Expected<uint32_t> CalculateNumChildren() override;322323lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override;324325lldb::ChildCacheState Update() override;326327bool MightHaveChildren() override;328329size_t GetIndexOfChildWithName(ConstString name) override;330};331} // namespace formatters332} // namespace lldb_private333334bool lldb_private::formatters::NSArraySummaryProvider(335ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {336static constexpr llvm::StringLiteral g_TypeHint("NSArray");337338ProcessSP process_sp = valobj.GetProcessSP();339if (!process_sp)340return false;341342ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);343344if (!runtime)345return false;346347ObjCLanguageRuntime::ClassDescriptorSP descriptor(348runtime->GetClassDescriptor(valobj));349350if (!descriptor || !descriptor->IsValid())351return false;352353uint32_t ptr_size = process_sp->GetAddressByteSize();354355lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);356357if (!valobj_addr)358return false;359360uint64_t value = 0;361362ConstString class_name(descriptor->GetClassName());363364static const ConstString g_NSArrayI("__NSArrayI");365static const ConstString g_NSArrayM("__NSArrayM");366static const ConstString g_NSArrayI_Transfer("__NSArrayI_Transfer");367static const ConstString g_NSFrozenArrayM("__NSFrozenArrayM");368static const ConstString g_NSArray0("__NSArray0");369static const ConstString g_NSArray1("__NSSingleObjectArrayI");370static const ConstString g_NSArrayCF("__NSCFArray");371static const ConstString g_NSArrayMLegacy("__NSArrayM_Legacy");372static const ConstString g_NSArrayMImmutable("__NSArrayM_Immutable");373static const ConstString g_NSCallStackArray("_NSCallStackArray");374static const ConstString g_NSConstantArray("NSConstantArray");375376if (class_name.IsEmpty())377return false;378379if (class_name == g_NSArrayI) {380Status error;381value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,382ptr_size, 0, error);383if (error.Fail())384return false;385} else if (class_name == g_NSConstantArray) {386Status error;387value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size, 8,3880, error);389if (error.Fail())390return false;391} else if (class_name == g_NSArrayM) {392AppleObjCRuntime *apple_runtime =393llvm::dyn_cast_or_null<AppleObjCRuntime>(runtime);394Status error;395if (apple_runtime && apple_runtime->GetFoundationVersion() >= 1437) {396value = Foundation1437::__NSArrayMSize(*process_sp, valobj_addr, error);397} else {398value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,399ptr_size, 0, error);400}401if (error.Fail())402return false;403} else if (class_name == g_NSArrayI_Transfer) {404Status error;405value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,406ptr_size, 0, error);407if (error.Fail())408return false;409} else if (class_name == g_NSFrozenArrayM) {410Status error;411value = Foundation1436::__NSFrozenArrayMSize(*process_sp, valobj_addr, error);412if (error.Fail())413return false;414} else if (class_name == g_NSArrayMLegacy) {415Status error;416value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,417ptr_size, 0, error);418if (error.Fail())419return false;420} else if (class_name == g_NSArrayMImmutable) {421Status error;422value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,423ptr_size, 0, error);424if (error.Fail())425return false;426} else if (class_name == g_NSArray0) {427value = 0;428} else if (class_name == g_NSArray1) {429value = 1;430} else if (class_name == g_NSArrayCF || class_name == g_NSCallStackArray) {431// __NSCFArray and _NSCallStackArray store the number of elements as a432// pointer-sized value at offset `2 * ptr_size`.433Status error;434value = process_sp->ReadUnsignedIntegerFromMemory(435valobj_addr + 2 * ptr_size, ptr_size, 0, error);436if (error.Fail())437return false;438} else {439auto &map(NSArray_Additionals::GetAdditionalSummaries());440auto iter = map.find(class_name), end = map.end();441if (iter != end)442return iter->second(valobj, stream, options);443else444return false;445}446447llvm::StringRef prefix, suffix;448if (Language *language = Language::FindPlugin(options.GetLanguage()))449std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);450451stream << prefix;452stream.Printf("%" PRIu64 " %s%s", value, "element", value == 1 ? "" : "s");453stream << suffix;454return true;455}456457lldb_private::formatters::NSArrayMSyntheticFrontEndBase::458NSArrayMSyntheticFrontEndBase(lldb::ValueObjectSP valobj_sp)459: SyntheticChildrenFrontEnd(*valobj_sp), m_exe_ctx_ref(), m_id_type() {460if (valobj_sp) {461TypeSystemClangSP scratch_ts_sp = ScratchTypeSystemClang::GetForTarget(462*valobj_sp->GetExecutionContextRef().GetTargetSP());463if (scratch_ts_sp)464m_id_type = CompilerType(465scratch_ts_sp->weak_from_this(),466scratch_ts_sp->getASTContext().ObjCBuiltinIdTy.getAsOpaquePtr());467if (valobj_sp->GetProcessSP())468m_ptr_size = valobj_sp->GetProcessSP()->GetAddressByteSize();469}470}471472template <typename D32, typename D64>473lldb_private::formatters::474GenericNSArrayMSyntheticFrontEnd<D32, D64>::475GenericNSArrayMSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp)476: NSArrayMSyntheticFrontEndBase(valobj_sp), m_data_32(nullptr),477m_data_64(nullptr) {}478479llvm::Expected<uint32_t> lldb_private::formatters::480NSArrayMSyntheticFrontEndBase::CalculateNumChildren() {481return GetUsedCount();482}483484lldb::ValueObjectSP485lldb_private::formatters::NSArrayMSyntheticFrontEndBase::GetChildAtIndex(486uint32_t idx) {487if (idx >= CalculateNumChildrenIgnoringErrors())488return lldb::ValueObjectSP();489lldb::addr_t object_at_idx = GetDataAddress();490size_t pyhs_idx = idx;491pyhs_idx += GetOffset();492if (GetSize() <= pyhs_idx)493pyhs_idx -= GetSize();494object_at_idx += (pyhs_idx * m_ptr_size);495StreamString idx_name;496idx_name.Printf("[%" PRIu64 "]", (uint64_t)idx);497return CreateValueObjectFromAddress(idx_name.GetString(), object_at_idx,498m_exe_ctx_ref, m_id_type);499}500501template <typename D32, typename D64>502lldb::ChildCacheState503lldb_private::formatters::GenericNSArrayMSyntheticFrontEnd<D32, D64>::Update() {504ValueObjectSP valobj_sp = m_backend.GetSP();505m_ptr_size = 0;506delete m_data_32;507m_data_32 = nullptr;508delete m_data_64;509m_data_64 = nullptr;510if (!valobj_sp)511return lldb::ChildCacheState::eRefetch;512m_exe_ctx_ref = valobj_sp->GetExecutionContextRef();513Status error;514error.Clear();515lldb::ProcessSP process_sp(valobj_sp->GetProcessSP());516if (!process_sp)517return lldb::ChildCacheState::eRefetch;518m_ptr_size = process_sp->GetAddressByteSize();519uint64_t data_location = valobj_sp->GetValueAsUnsigned(0) + m_ptr_size;520if (m_ptr_size == 4) {521m_data_32 = new D32();522process_sp->ReadMemory(data_location, m_data_32, sizeof(D32),523error);524} else {525m_data_64 = new D64();526process_sp->ReadMemory(data_location, m_data_64, sizeof(D64),527error);528}529530return error.Success() ? lldb::ChildCacheState::eReuse531: lldb::ChildCacheState::eRefetch;532}533534bool535lldb_private::formatters::NSArrayMSyntheticFrontEndBase::MightHaveChildren() {536return true;537}538539size_t540lldb_private::formatters::NSArrayMSyntheticFrontEndBase::GetIndexOfChildWithName(541ConstString name) {542const char *item_name = name.GetCString();543uint32_t idx = ExtractIndexFromString(item_name);544if (idx < UINT32_MAX && idx >= CalculateNumChildrenIgnoringErrors())545return UINT32_MAX;546return idx;547}548549template <typename D32, typename D64>550lldb_private::formatters::GenericNSArrayMSyntheticFrontEnd<D32, D64>::551GenericNSArrayMSyntheticFrontEnd::~GenericNSArrayMSyntheticFrontEnd() {552delete m_data_32;553m_data_32 = nullptr;554delete m_data_64;555m_data_64 = nullptr;556}557558template <typename D32, typename D64>559lldb::addr_t560lldb_private::formatters::561GenericNSArrayMSyntheticFrontEnd<D32, D64>::562GenericNSArrayMSyntheticFrontEnd::GetDataAddress() {563if (!m_data_32 && !m_data_64)564return LLDB_INVALID_ADDRESS;565return m_data_32 ? m_data_32->_data : m_data_64->_data;566}567568template <typename D32, typename D64>569uint64_t570lldb_private::formatters::571GenericNSArrayMSyntheticFrontEnd<D32, D64>::572GenericNSArrayMSyntheticFrontEnd::GetUsedCount() {573if (!m_data_32 && !m_data_64)574return 0;575return m_data_32 ? m_data_32->_used : m_data_64->_used;576}577578template <typename D32, typename D64>579uint64_t580lldb_private::formatters::581GenericNSArrayMSyntheticFrontEnd<D32, D64>::582GenericNSArrayMSyntheticFrontEnd::GetOffset() {583if (!m_data_32 && !m_data_64)584return 0;585return m_data_32 ? m_data_32->_offset : m_data_64->_offset;586}587588template <typename D32, typename D64>589uint64_t590lldb_private::formatters::591GenericNSArrayMSyntheticFrontEnd<D32, D64>::592GenericNSArrayMSyntheticFrontEnd::GetSize() {593if (!m_data_32 && !m_data_64)594return 0;595return m_data_32 ? m_data_32->_size : m_data_64->_size;596}597598template <typename D32, typename D64, bool Inline>599lldb_private::formatters::GenericNSArrayISyntheticFrontEnd<D32, D64, Inline>::600GenericNSArrayISyntheticFrontEnd(lldb::ValueObjectSP valobj_sp)601: SyntheticChildrenFrontEnd(*valobj_sp), m_exe_ctx_ref(),602m_data_32(nullptr), m_data_64(nullptr) {603if (valobj_sp) {604CompilerType type = valobj_sp->GetCompilerType();605if (type) {606TypeSystemClangSP scratch_ts_sp = ScratchTypeSystemClang::GetForTarget(607*valobj_sp->GetExecutionContextRef().GetTargetSP());608if (scratch_ts_sp)609m_id_type = scratch_ts_sp->GetType(610scratch_ts_sp->getASTContext().ObjCBuiltinIdTy);611}612}613}614615template <typename D32, typename D64, bool Inline>616lldb_private::formatters::GenericNSArrayISyntheticFrontEnd<D32, D64, Inline>::617GenericNSArrayISyntheticFrontEnd::~GenericNSArrayISyntheticFrontEnd() {618delete m_data_32;619m_data_32 = nullptr;620delete m_data_64;621m_data_64 = nullptr;622}623624template <typename D32, typename D64, bool Inline>625size_t626lldb_private::formatters::GenericNSArrayISyntheticFrontEnd<D32, D64, Inline>::627GetIndexOfChildWithName(ConstString name) {628const char *item_name = name.GetCString();629uint32_t idx = ExtractIndexFromString(item_name);630if (idx < UINT32_MAX && idx >= CalculateNumChildrenIgnoringErrors())631return UINT32_MAX;632return idx;633}634635template <typename D32, typename D64, bool Inline>636llvm::Expected<uint32_t>637lldb_private::formatters::GenericNSArrayISyntheticFrontEnd<638D32, D64, Inline>::CalculateNumChildren() {639return m_data_32 ? m_data_32->used : m_data_64->used;640}641642template <typename D32, typename D64, bool Inline>643lldb::ChildCacheState644lldb_private::formatters::GenericNSArrayISyntheticFrontEnd<D32, D64,645Inline>::Update() {646ValueObjectSP valobj_sp = m_backend.GetSP();647m_ptr_size = 0;648delete m_data_32;649m_data_32 = nullptr;650delete m_data_64;651m_data_64 = nullptr;652if (!valobj_sp)653return lldb::ChildCacheState::eRefetch;654m_exe_ctx_ref = valobj_sp->GetExecutionContextRef();655Status error;656error.Clear();657lldb::ProcessSP process_sp(valobj_sp->GetProcessSP());658if (!process_sp)659return lldb::ChildCacheState::eRefetch;660m_ptr_size = process_sp->GetAddressByteSize();661uint64_t data_location = valobj_sp->GetValueAsUnsigned(0) + m_ptr_size;662if (m_ptr_size == 4) {663m_data_32 = new D32();664process_sp->ReadMemory(data_location, m_data_32, sizeof(D32),665error);666} else {667m_data_64 = new D64();668process_sp->ReadMemory(data_location, m_data_64, sizeof(D64),669error);670}671672return error.Success() ? lldb::ChildCacheState::eReuse673: lldb::ChildCacheState::eRefetch;674}675676template <typename D32, typename D64, bool Inline>677bool678lldb_private::formatters::GenericNSArrayISyntheticFrontEnd<D32, D64, Inline>::679MightHaveChildren() {680return true;681}682683template <typename D32, typename D64, bool Inline>684lldb::ValueObjectSP685lldb_private::formatters::GenericNSArrayISyntheticFrontEnd<D32, D64, Inline>::686GetChildAtIndex(uint32_t idx) {687if (idx >= CalculateNumChildrenIgnoringErrors())688return lldb::ValueObjectSP();689lldb::addr_t object_at_idx;690if (Inline) {691object_at_idx = m_backend.GetSP()->GetValueAsUnsigned(0) + m_ptr_size;692object_at_idx += m_ptr_size == 4 ? sizeof(D32) : sizeof(D64); // skip the data header693object_at_idx -= m_ptr_size; // we treat the last entry in the data header as the first pointer694} else {695object_at_idx = m_data_32 ? m_data_32->list : m_data_64->list;696}697object_at_idx += (idx * m_ptr_size);698699ProcessSP process_sp = m_exe_ctx_ref.GetProcessSP();700if (!process_sp)701return lldb::ValueObjectSP();702Status error;703if (error.Fail())704return lldb::ValueObjectSP();705StreamString idx_name;706idx_name.Printf("[%" PRIu64 "]", (uint64_t)idx);707return CreateValueObjectFromAddress(idx_name.GetString(), object_at_idx,708m_exe_ctx_ref, m_id_type);709}710711lldb_private::formatters::NSArray0SyntheticFrontEnd::NSArray0SyntheticFrontEnd(712lldb::ValueObjectSP valobj_sp)713: SyntheticChildrenFrontEnd(*valobj_sp) {}714715size_t716lldb_private::formatters::NSArray0SyntheticFrontEnd::GetIndexOfChildWithName(717ConstString name) {718return UINT32_MAX;719}720721llvm::Expected<uint32_t>722lldb_private::formatters::NSArray0SyntheticFrontEnd::CalculateNumChildren() {723return 0;724}725726lldb::ChildCacheState727lldb_private::formatters::NSArray0SyntheticFrontEnd::Update() {728return lldb::ChildCacheState::eRefetch;729}730731bool lldb_private::formatters::NSArray0SyntheticFrontEnd::MightHaveChildren() {732return false;733}734735lldb::ValueObjectSP736lldb_private::formatters::NSArray0SyntheticFrontEnd::GetChildAtIndex(737uint32_t idx) {738return lldb::ValueObjectSP();739}740741lldb_private::formatters::NSArray1SyntheticFrontEnd::NSArray1SyntheticFrontEnd(742lldb::ValueObjectSP valobj_sp)743: SyntheticChildrenFrontEnd(*valobj_sp.get()) {}744745size_t746lldb_private::formatters::NSArray1SyntheticFrontEnd::GetIndexOfChildWithName(747ConstString name) {748static const ConstString g_zero("[0]");749750if (name == g_zero)751return 0;752753return UINT32_MAX;754}755756llvm::Expected<uint32_t>757lldb_private::formatters::NSArray1SyntheticFrontEnd::CalculateNumChildren() {758return 1;759}760761lldb::ChildCacheState762lldb_private::formatters::NSArray1SyntheticFrontEnd::Update() {763return lldb::ChildCacheState::eRefetch;764}765766bool lldb_private::formatters::NSArray1SyntheticFrontEnd::MightHaveChildren() {767return true;768}769770lldb::ValueObjectSP771lldb_private::formatters::NSArray1SyntheticFrontEnd::GetChildAtIndex(772uint32_t idx) {773static const ConstString g_zero("[0]");774775if (idx == 0) {776TypeSystemClangSP scratch_ts_sp =777ScratchTypeSystemClang::GetForTarget(*m_backend.GetTargetSP());778if (scratch_ts_sp) {779CompilerType id_type(scratch_ts_sp->GetBasicType(lldb::eBasicTypeObjCID));780return m_backend.GetSyntheticChildAtOffset(781m_backend.GetProcessSP()->GetAddressByteSize(), id_type, true,782g_zero);783}784}785return lldb::ValueObjectSP();786}787788SyntheticChildrenFrontEnd *789lldb_private::formatters::NSArraySyntheticFrontEndCreator(790CXXSyntheticChildren *synth, lldb::ValueObjectSP valobj_sp) {791if (!valobj_sp)792return nullptr;793794lldb::ProcessSP process_sp(valobj_sp->GetProcessSP());795if (!process_sp)796return nullptr;797AppleObjCRuntime *runtime = llvm::dyn_cast_or_null<AppleObjCRuntime>(798ObjCLanguageRuntime::Get(*process_sp));799if (!runtime)800return nullptr;801802CompilerType valobj_type(valobj_sp->GetCompilerType());803Flags flags(valobj_type.GetTypeInfo());804805if (flags.IsClear(eTypeIsPointer)) {806Status error;807valobj_sp = valobj_sp->AddressOf(error);808if (error.Fail() || !valobj_sp)809return nullptr;810}811812ObjCLanguageRuntime::ClassDescriptorSP descriptor(813runtime->GetClassDescriptor(*valobj_sp));814815if (!descriptor || !descriptor->IsValid())816return nullptr;817818ConstString class_name(descriptor->GetClassName());819820static const ConstString g_NSArrayI("__NSArrayI");821static const ConstString g_NSConstantArray("NSConstantArray");822static const ConstString g_NSArrayI_Transfer("__NSArrayI_Transfer");823static const ConstString g_NSFrozenArrayM("__NSFrozenArrayM");824static const ConstString g_NSArrayM("__NSArrayM");825static const ConstString g_NSArray0("__NSArray0");826static const ConstString g_NSArray1("__NSSingleObjectArrayI");827static const ConstString g_NSArrayMLegacy("__NSArrayM_Legacy");828static const ConstString g_NSArrayMImmutable("__NSArrayM_Immutable");829static const ConstString g_NSCallStackArray("_NSCallStackArray");830831if (class_name.IsEmpty())832return nullptr;833834if (class_name == g_NSArrayI) {835if (runtime->GetFoundationVersion() >= 1436)836return (new Foundation1436::NSArrayISyntheticFrontEnd(valobj_sp));837if (runtime->GetFoundationVersion() >= 1430)838return (new Foundation1430::NSArrayISyntheticFrontEnd(valobj_sp));839return (new Foundation1300::NSArrayISyntheticFrontEnd(valobj_sp));840} else if (class_name == g_NSArrayI_Transfer) {841return (new Foundation1436::NSArrayI_TransferSyntheticFrontEnd(valobj_sp));842} else if (class_name == g_NSConstantArray) {843return new ConstantArray::NSConstantArraySyntheticFrontEnd(valobj_sp);844} else if (class_name == g_NSFrozenArrayM) {845return (new Foundation1436::NSFrozenArrayMSyntheticFrontEnd(valobj_sp));846} else if (class_name == g_NSArray0) {847return (new NSArray0SyntheticFrontEnd(valobj_sp));848} else if (class_name == g_NSArray1) {849return (new NSArray1SyntheticFrontEnd(valobj_sp));850} else if (class_name == g_NSArrayM) {851if (runtime->GetFoundationVersion() >= 1437)852return (new Foundation1437::NSArrayMSyntheticFrontEnd(valobj_sp));853if (runtime->GetFoundationVersion() >= 1428)854return (new Foundation1428::NSArrayMSyntheticFrontEnd(valobj_sp));855if (runtime->GetFoundationVersion() >= 1100)856return (new Foundation1010::NSArrayMSyntheticFrontEnd(valobj_sp));857} else if (class_name == g_NSCallStackArray) {858return (new CallStackArray::NSCallStackArraySyntheticFrontEnd(valobj_sp));859} else {860auto &map(NSArray_Additionals::GetAdditionalSynthetics());861auto iter = map.find(class_name), end = map.end();862if (iter != end)863return iter->second(synth, valobj_sp);864}865866return nullptr;867}868869870