LLDB mainline
NSArray.cpp
Go to the documentation of this file.
1//===-- NSArray.cpp -------------------------------------------------------===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8
9#include "clang/AST/ASTContext.h"
10#include "clang/Basic/TargetInfo.h"
11
12#include "Cocoa.h"
13
16
20#include "lldb/Target/Target.h"
22#include "lldb/Utility/Endian.h"
23#include "lldb/Utility/Status.h"
24#include "lldb/Utility/Stream.h"
27
28using namespace lldb;
29using namespace lldb_private;
30using namespace lldb_private::formatters;
31
32namespace lldb_private {
33namespace formatters {
34std::map<ConstString, CXXFunctionSummaryFormat::Callback> &
36 static std::map<ConstString, CXXFunctionSummaryFormat::Callback> g_map;
37 return g_map;
38}
39
40std::map<ConstString, CXXSyntheticChildren::CreateFrontEndCallback> &
42 static std::map<ConstString, CXXSyntheticChildren::CreateFrontEndCallback>
43 g_map;
44 return g_map;
45}
46
48public:
50
51 ~NSArrayMSyntheticFrontEndBase() override = default;
52
53 llvm::Expected<uint32_t> CalculateNumChildren() override;
54
55 lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override;
56
58
59protected:
61
62 virtual uint64_t GetUsedCount() = 0;
63
64 virtual uint64_t GetOffset() = 0;
65
66 virtual uint64_t GetSize() = 0;
67
69 uint8_t m_ptr_size = 8;
71};
72
73template <typename D32, typename D64>
75public:
77
79
81
82protected:
84
85 uint64_t GetUsedCount() override;
86
87 uint64_t GetOffset() override;
88
89 uint64_t GetSize() override;
90
91private:
94};
95
96namespace Foundation1010 {
97 namespace {
98 struct DataDescriptor_32 {
99 uint32_t _used;
100 uint32_t _offset;
101 uint32_t _size : 28;
102 uint64_t _priv1 : 4;
103 uint32_t _priv2;
104 uint32_t _data;
105 };
106
107 struct DataDescriptor_64 {
108 uint64_t _used;
109 uint64_t _offset;
110 uint64_t _size : 60;
111 uint64_t _priv1 : 4;
112 uint32_t _priv2;
113 uint64_t _data;
114 };
115 }
116
119}
120
121namespace Foundation1428 {
122 namespace {
123 struct DataDescriptor_32 {
124 uint32_t _used;
125 uint32_t _offset;
126 uint32_t _size;
127 uint32_t _data;
128 };
129
130 struct DataDescriptor_64 {
131 uint64_t _used;
132 uint64_t _offset;
133 uint64_t _size;
134 uint64_t _data;
135 };
136 }
137
140}
141
142namespace Foundation1437 {
143 template <typename PtrType>
145 PtrType _cow;
146 // __deque
147 PtrType _data;
148 uint32_t _offset;
149 uint32_t _size;
150 uint32_t _muts;
151 uint32_t _used;
152 };
153
157
158 template <typename DD>
159 uint64_t
161 lldb::addr_t valobj_addr, Status &error) {
162 const lldb::addr_t start_of_descriptor =
163 valobj_addr + process.GetAddressByteSize();
164 DD descriptor = DD();
165 process.ReadMemory(start_of_descriptor, &descriptor,
166 sizeof(descriptor), error);
167 if (error.Fail()) {
168 return 0;
169 }
170 return descriptor._used;
171 }
172
173 uint64_t
175 Status &error) {
176 if (process.GetAddressByteSize() == 4) {
177 return __NSArrayMSize_Impl<DataDescriptor<uint32_t>>(process, valobj_addr,
178 error);
179 } else {
180 return __NSArrayMSize_Impl<DataDescriptor<uint64_t>>(process, valobj_addr,
181 error);
182 }
183 }
184
185}
186
187namespace CallStackArray {
189 uint32_t _data;
190 uint32_t _used;
191 uint32_t _offset;
192 const uint32_t _size = 0;
193};
194
196 uint64_t _data;
197 uint64_t _used;
198 uint64_t _offset;
199 const uint64_t _size = 0;
200};
201
204} // namespace CallStackArray
205
206template <typename D32, typename D64, bool Inline>
227
228namespace Foundation1300 {
229 struct IDD32 {
230 uint32_t used;
231 uint32_t list;
232 };
233
234 struct IDD64 {
235 uint64_t used;
236 uint64_t list;
237 };
238
241}
242
247
248namespace Foundation1436 {
249 struct IDD32 {
250 uint32_t used;
251 uint32_t list; // in Inline cases, this is the first element
252 };
253
254 struct IDD64 {
255 uint64_t used;
256 uint64_t list; // in Inline cases, this is the first element
257 };
258
261
264
267
268 uint64_t
270 Status &error) {
271 return Foundation1437::__NSArrayMSize(process, valobj_addr, error);
272 }
273}
274
275namespace ConstantArray {
276
278 uint64_t used;
279 uint32_t list;
280};
281
283 uint64_t used;
284 uint64_t list;
285};
286
289} // namespace ConstantArray
290
292public:
294
295 ~NSArray0SyntheticFrontEnd() override = default;
296
297 llvm::Expected<uint32_t> CalculateNumChildren() override;
298
299 lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override;
300
301 lldb::ChildCacheState Update() override;
302
303 bool MightHaveChildren() override;
304
305 llvm::Expected<size_t> GetIndexOfChildWithName(ConstString name) override;
306};
307
309public:
311
312 ~NSArray1SyntheticFrontEnd() override = default;
313
314 llvm::Expected<uint32_t> CalculateNumChildren() override;
315
316 lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override;
317
318 lldb::ChildCacheState Update() override;
319
320 llvm::Expected<size_t> GetIndexOfChildWithName(ConstString name) override;
321};
322} // namespace formatters
323} // namespace lldb_private
324
326 ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options) {
327 static constexpr llvm::StringLiteral g_TypeHint("NSArray");
328
329 ProcessSP process_sp = valobj.GetProcessSP();
330 if (!process_sp)
331 return false;
332
333 ObjCLanguageRuntime *runtime = ObjCLanguageRuntime::Get(*process_sp);
334
335 if (!runtime)
336 return false;
337
339 runtime->GetClassDescriptor(valobj));
340
341 if (!descriptor || !descriptor->IsValid())
342 return false;
343
344 uint32_t ptr_size = process_sp->GetAddressByteSize();
345
346 lldb::addr_t valobj_addr = valobj.GetValueAsUnsigned(0);
347
348 if (!valobj_addr)
349 return false;
350
351 uint64_t value = 0;
352
353 ConstString class_name(descriptor->GetClassName());
354
355 static const ConstString g_NSArrayI("__NSArrayI");
356 static const ConstString g_NSArrayM("__NSArrayM");
357 static const ConstString g_NSArrayI_Transfer("__NSArrayI_Transfer");
358 static const ConstString g_NSFrozenArrayM("__NSFrozenArrayM");
359 static const ConstString g_NSArray0("__NSArray0");
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");
366
367 if (class_name.IsEmpty())
368 return false;
369
370 if (class_name == g_NSArrayI) {
372 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,
373 ptr_size, 0, error);
374 if (error.Fail())
375 return false;
376 } else if (class_name == g_NSConstantArray) {
378 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size, 8,
379 0, error);
380 if (error.Fail())
381 return false;
382 } else if (class_name == g_NSArrayM) {
383 AppleObjCRuntime *apple_runtime =
384 llvm::dyn_cast_or_null<AppleObjCRuntime>(runtime);
386 if (apple_runtime && apple_runtime->GetFoundationVersion() >= 1437) {
387 value = Foundation1437::__NSArrayMSize(*process_sp, valobj_addr, error);
388 } else {
389 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,
390 ptr_size, 0, error);
391 }
392 if (error.Fail())
393 return false;
394 } else if (class_name == g_NSArrayI_Transfer) {
396 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,
397 ptr_size, 0, error);
398 if (error.Fail())
399 return false;
400 } else if (class_name == g_NSFrozenArrayM) {
402 value = Foundation1436::__NSFrozenArrayMSize(*process_sp, valobj_addr, error);
403 if (error.Fail())
404 return false;
405 } else if (class_name == g_NSArrayMLegacy) {
407 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,
408 ptr_size, 0, error);
409 if (error.Fail())
410 return false;
411 } else if (class_name == g_NSArrayMImmutable) {
413 value = process_sp->ReadUnsignedIntegerFromMemory(valobj_addr + ptr_size,
414 ptr_size, 0, error);
415 if (error.Fail())
416 return false;
417 } else if (class_name == g_NSArray0) {
418 value = 0;
419 } else if (class_name == g_NSArray1) {
420 value = 1;
421 } else if (class_name == g_NSArrayCF || class_name == g_NSCallStackArray) {
422 // __NSCFArray and _NSCallStackArray store the number of elements as a
423 // pointer-sized value at offset `2 * ptr_size`.
425 value = process_sp->ReadUnsignedIntegerFromMemory(
426 valobj_addr + 2 * ptr_size, ptr_size, 0, error);
427 if (error.Fail())
428 return false;
429 } else {
431 auto iter = map.find(class_name), end = map.end();
432 if (iter != end)
433 return iter->second(valobj, stream, options);
434 else
435 return false;
436 }
437
438 llvm::StringRef prefix, suffix;
439 if (Language *language = Language::FindPlugin(options.GetLanguage()))
440 std::tie(prefix, suffix) = language->GetFormatterPrefixSuffix(g_TypeHint);
441
442 stream << prefix;
443 stream.Printf("%" PRIu64 " %s%s", value, "element", value == 1 ? "" : "s");
444 stream << suffix;
445 return true;
446}
447
451 if (valobj_sp) {
453 *valobj_sp->GetExecutionContextRef().GetTargetSP());
454 if (scratch_ts_sp)
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();
460 }
461}
462
463template <typename D32, typename D64>
469
474
477 uint32_t idx) {
479 return lldb::ValueObjectSP();
480 lldb::addr_t object_at_idx = GetDataAddress();
481 size_t pyhs_idx = idx;
482 pyhs_idx += GetOffset();
483 if (GetSize() <= pyhs_idx)
484 pyhs_idx -= GetSize();
485 object_at_idx += (pyhs_idx * m_ptr_size);
486 StreamString idx_name;
487 idx_name.Printf("[%" PRIu64 "]", (uint64_t)idx);
488 return CreateValueObjectFromAddress(idx_name.GetString(), object_at_idx,
490}
491
492template <typename D32, typename D64>
495 ValueObjectSP valobj_sp = m_backend.GetSP();
496 m_ptr_size = 0;
497 delete m_data_32;
498 m_data_32 = nullptr;
499 delete m_data_64;
500 m_data_64 = nullptr;
501 if (!valobj_sp)
503 m_exe_ctx_ref = valobj_sp->GetExecutionContextRef();
505 error.Clear();
506 lldb::ProcessSP process_sp(valobj_sp->GetProcessSP());
507 if (!process_sp)
509 m_ptr_size = process_sp->GetAddressByteSize();
510 uint64_t data_location = valobj_sp->GetValueAsUnsigned(0) + m_ptr_size;
511 if (m_ptr_size == 4) {
512 m_data_32 = new D32();
513 process_sp->ReadMemory(data_location, m_data_32, sizeof(D32),
514 error);
515 } else {
516 m_data_64 = new D64();
517 process_sp->ReadMemory(data_location, m_data_64, sizeof(D64),
518 error);
519 }
520
521 return error.Success() ? lldb::ChildCacheState::eReuse
523}
524
525template <typename D32, typename D64>
528 delete m_data_32;
529 m_data_32 = nullptr;
530 delete m_data_64;
531 m_data_64 = nullptr;
532}
533
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;
542}
543
544template <typename D32, typename D64>
545uint64_t
546lldb_private::formatters::
547 GenericNSArrayMSyntheticFrontEnd<D32, D64>::
548 GenericNSArrayMSyntheticFrontEnd::GetUsedCount() {
549 if (!m_data_32 && !m_data_64)
550 return 0;
551 return m_data_32 ? m_data_32->_used : m_data_64->_used;
552}
553
554template <typename D32, typename D64>
555uint64_t
556lldb_private::formatters::
557 GenericNSArrayMSyntheticFrontEnd<D32, D64>::
558 GenericNSArrayMSyntheticFrontEnd::GetOffset() {
559 if (!m_data_32 && !m_data_64)
560 return 0;
561 return m_data_32 ? m_data_32->_offset : m_data_64->_offset;
562}
563
564template <typename D32, typename D64>
565uint64_t
566lldb_private::formatters::
567 GenericNSArrayMSyntheticFrontEnd<D32, D64>::
568 GenericNSArrayMSyntheticFrontEnd::GetSize() {
569 if (!m_data_32 && !m_data_64)
570 return 0;
571 return m_data_32 ? m_data_32->_size : m_data_64->_size;
572}
573
574template <typename D32, typename D64, bool Inline>
578 m_data_32(nullptr), m_data_64(nullptr) {
579 if (valobj_sp) {
580 CompilerType type = valobj_sp->GetCompilerType();
581 if (type) {
583 *valobj_sp->GetExecutionContextRef().GetTargetSP());
584 if (scratch_ts_sp)
585 m_id_type = scratch_ts_sp->GetType(
586 scratch_ts_sp->getASTContext().ObjCBuiltinIdTy);
587 }
588 }
589}
590
591template <typename D32, typename D64, bool Inline>
594 delete m_data_32;
595 m_data_32 = nullptr;
596 delete m_data_64;
597 m_data_64 = nullptr;
598}
599
600template <typename D32, typename D64, bool Inline>
601llvm::Expected<uint32_t>
603 D32, D64, Inline>::CalculateNumChildren() {
604 return m_data_32 ? m_data_32->used : m_data_64->used;
605}
606
607template <typename D32, typename D64, bool Inline>
610 Inline>::Update() {
611 ValueObjectSP valobj_sp = m_backend.GetSP();
612 m_ptr_size = 0;
613 delete m_data_32;
614 m_data_32 = nullptr;
615 delete m_data_64;
616 m_data_64 = nullptr;
617 if (!valobj_sp)
619 m_exe_ctx_ref = valobj_sp->GetExecutionContextRef();
621 error.Clear();
622 lldb::ProcessSP process_sp(valobj_sp->GetProcessSP());
623 if (!process_sp)
625 m_ptr_size = process_sp->GetAddressByteSize();
626 uint64_t data_location = valobj_sp->GetValueAsUnsigned(0) + m_ptr_size;
627 if (m_ptr_size == 4) {
628 m_data_32 = new D32();
629 process_sp->ReadMemory(data_location, m_data_32, sizeof(D32),
630 error);
631 } else {
632 m_data_64 = new D64();
633 process_sp->ReadMemory(data_location, m_data_64, sizeof(D64),
634 error);
635 }
636
637 return error.Success() ? lldb::ChildCacheState::eReuse
639}
640
641template <typename D32, typename D64, bool Inline>
644 GetChildAtIndex(uint32_t idx) {
646 return lldb::ValueObjectSP();
647 lldb::addr_t object_at_idx;
648 if (Inline) {
649 object_at_idx = m_backend.GetSP()->GetValueAsUnsigned(0) + m_ptr_size;
650 object_at_idx += m_ptr_size == 4 ? sizeof(D32) : sizeof(D64); // skip the data header
651 object_at_idx -= m_ptr_size; // we treat the last entry in the data header as the first pointer
652 } else {
653 object_at_idx = m_data_32 ? m_data_32->list : m_data_64->list;
654 }
655 object_at_idx += (idx * m_ptr_size);
656
657 ProcessSP process_sp = m_exe_ctx_ref.GetProcessSP();
658 if (!process_sp)
659 return lldb::ValueObjectSP();
661 if (error.Fail())
662 return lldb::ValueObjectSP();
663 StreamString idx_name;
664 idx_name.Printf("[%" PRIu64 "]", (uint64_t)idx);
665 return CreateValueObjectFromAddress(idx_name.GetString(), object_at_idx,
667}
668
672
673llvm::Expected<size_t>
678
679llvm::Expected<uint32_t>
683
688
692
698
702
703llvm::Expected<size_t>
705 ConstString name) {
706 static const ConstString g_zero("[0]");
707
708 if (name == g_zero)
709 return 0;
710
711 return UINT32_MAX;
712}
713
714llvm::Expected<uint32_t>
718
723
726 uint32_t idx) {
727 static const ConstString g_zero("[0]");
728
729 if (idx == 0) {
730 TypeSystemClangSP scratch_ts_sp =
732 if (scratch_ts_sp) {
733 CompilerType id_type(scratch_ts_sp->GetBasicType(lldb::eBasicTypeObjCID));
734 return m_backend.GetSyntheticChildAtOffset(
735 m_backend.GetProcessSP()->GetAddressByteSize(), id_type, true,
736 g_zero);
737 }
738 }
739 return lldb::ValueObjectSP();
740}
741
744 CXXSyntheticChildren *synth, lldb::ValueObjectSP valobj_sp) {
745 if (!valobj_sp)
746 return nullptr;
747
748 lldb::ProcessSP process_sp(valobj_sp->GetProcessSP());
749 if (!process_sp)
750 return nullptr;
751 AppleObjCRuntime *runtime = llvm::dyn_cast_or_null<AppleObjCRuntime>(
752 ObjCLanguageRuntime::Get(*process_sp));
753 if (!runtime)
754 return nullptr;
755
756 CompilerType valobj_type(valobj_sp->GetCompilerType());
757 Flags flags(valobj_type.GetTypeInfo());
758
759 if (flags.IsClear(eTypeIsPointer)) {
761 valobj_sp = valobj_sp->AddressOf(error);
762 if (error.Fail() || !valobj_sp)
763 return nullptr;
764 }
765
767 runtime->GetClassDescriptor(*valobj_sp));
768
769 if (!descriptor || !descriptor->IsValid())
770 return nullptr;
771
772 ConstString class_name(descriptor->GetClassName());
773
774 static const ConstString g_NSArrayI("__NSArrayI");
775 static const ConstString g_NSConstantArray("NSConstantArray");
776 static const ConstString g_NSArrayI_Transfer("__NSArrayI_Transfer");
777 static const ConstString g_NSFrozenArrayM("__NSFrozenArrayM");
778 static const ConstString g_NSArrayM("__NSArrayM");
779 static const ConstString g_NSArray0("__NSArray0");
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");
784
785 if (class_name.IsEmpty())
786 return nullptr;
787
788 if (class_name == g_NSArrayI) {
789 if (runtime->GetFoundationVersion() >= 1436)
790 return (new Foundation1436::NSArrayISyntheticFrontEnd(valobj_sp));
791 if (runtime->GetFoundationVersion() >= 1430)
792 return (new Foundation1430::NSArrayISyntheticFrontEnd(valobj_sp));
793 return (new Foundation1300::NSArrayISyntheticFrontEnd(valobj_sp));
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) {
801 return (new NSArray0SyntheticFrontEnd(valobj_sp));
802 } else if (class_name == g_NSArray1) {
803 return (new NSArray1SyntheticFrontEnd(valobj_sp));
804 } else if (class_name == g_NSArrayM) {
805 if (runtime->GetFoundationVersion() >= 1437)
806 return (new Foundation1437::NSArrayMSyntheticFrontEnd(valobj_sp));
807 if (runtime->GetFoundationVersion() >= 1428)
808 return (new Foundation1428::NSArrayMSyntheticFrontEnd(valobj_sp));
809 if (runtime->GetFoundationVersion() >= 1100)
810 return (new Foundation1010::NSArrayMSyntheticFrontEnd(valobj_sp));
811 } else if (class_name == g_NSCallStackArray) {
813 } else {
815 auto iter = map.find(class_name), end = map.end();
816 if (iter != end)
817 return iter->second(synth, valobj_sp);
818 }
819
820 return nullptr;
821}
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 ...
Generic representation of a type in a programming language.
A uniqued constant string class.
Definition ConstString.h:40
bool IsEmpty() const
Test for empty string.
Execution context objects refer to objects in the execution of the program that is being debugged.
A class to manage flags.
Definition Flags.h:22
static Language * FindPlugin(lldb::LanguageType language)
Definition Language.cpp:84
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.
Definition Process.h:354
virtual size_t ReadMemory(lldb::addr_t vm_addr, void *buf, size_t size, Status &error)
Read of memory from a process.
Definition Process.cpp:1907
uint32_t GetAddressByteSize() const
Definition Process.cpp:3725
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.
An error handling class.
Definition Status.h:118
llvm::StringRef GetString() const
A stream class that can stream formatted output to a file.
Definition Stream.h:28
size_t Printf(const char *format,...) __attribute__((format(printf
Output printf formatted output to the stream.
Definition Stream.cpp:134
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)
GenericNSArrayISyntheticFrontEnd(lldb::ValueObjectSP valobj_sp)
Definition NSArray.cpp:576
lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override
Definition NSArray.cpp:644
lldb::ChildCacheState Update() override
This function is assumed to always succeed and if it fails, the front-end should know to deal with it...
Definition NSArray.cpp:610
llvm::Expected< uint32_t > CalculateNumChildren() override
Definition NSArray.cpp:603
lldb::ChildCacheState Update() override
This function is assumed to always succeed and if it fails, the front-end should know to deal with it...
Definition NSArray.cpp:494
GenericNSArrayMSyntheticFrontEnd(lldb::ValueObjectSP valobj_sp)
Definition NSArray.cpp:466
llvm::Expected< size_t > GetIndexOfChildWithName(ConstString name) override
Determine the index of a named child.
Definition NSArray.cpp:674
lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override
Definition NSArray.cpp:694
NSArray0SyntheticFrontEnd(lldb::ValueObjectSP valobj_sp)
Definition NSArray.cpp:669
lldb::ChildCacheState Update() override
This function is assumed to always succeed and if it fails, the front-end should know to deal with it...
Definition NSArray.cpp:685
llvm::Expected< uint32_t > CalculateNumChildren() override
Definition NSArray.cpp:680
llvm::Expected< uint32_t > CalculateNumChildren() override
Definition NSArray.cpp:715
lldb::ChildCacheState Update() override
This function is assumed to always succeed and if it fails, the front-end should know to deal with it...
Definition NSArray.cpp:720
NSArray1SyntheticFrontEnd(lldb::ValueObjectSP valobj_sp)
Definition NSArray.cpp:699
llvm::Expected< size_t > GetIndexOfChildWithName(ConstString name) override
Determine the index of a named child.
Definition NSArray.cpp:704
lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override
Definition NSArray.cpp:725
llvm::Expected< uint32_t > CalculateNumChildren() override
Definition NSArray.cpp:471
NSArrayMSyntheticFrontEndBase(lldb::ValueObjectSP valobj_sp)
Definition NSArray.cpp:449
lldb::ValueObjectSP GetChildAtIndex(uint32_t idx) override
Definition NSArray.cpp:476
lldb::ChildCacheState Update() override=0
This function is assumed to always succeed and if it fails, the front-end should know to deal with it...
static std::map< ConstString, CXXFunctionSummaryFormat::Callback > & GetAdditionalSummaries()
Definition NSArray.cpp:35
static std::map< ConstString, CXXSyntheticChildren::CreateFrontEndCallback > & GetAdditionalSynthetics()
Definition NSArray.cpp:41
#define LLDB_INVALID_ADDRESS
#define UINT32_MAX
GenericNSArrayMSyntheticFrontEnd< DataDescriptor_32, DataDescriptor_64 > NSCallStackArraySyntheticFrontEnd
Definition NSArray.cpp:202
GenericNSArrayISyntheticFrontEnd< ConstantArray32, ConstantArray64, false > NSConstantArraySyntheticFrontEnd
Definition NSArray.cpp:287
GenericNSArrayMSyntheticFrontEnd< DataDescriptor_32, DataDescriptor_64 > NSArrayMSyntheticFrontEnd
Definition NSArray.cpp:117
GenericNSArrayISyntheticFrontEnd< IDD32, IDD64, true > NSArrayISyntheticFrontEnd
Definition NSArray.cpp:239
GenericNSArrayMSyntheticFrontEnd< DataDescriptor_32, DataDescriptor_64 > NSArrayMSyntheticFrontEnd
Definition NSArray.cpp:138
Foundation1428::NSArrayMSyntheticFrontEnd NSArrayISyntheticFrontEnd
Definition NSArray.cpp:244
Foundation1437::NSArrayMSyntheticFrontEnd NSFrozenArrayMSyntheticFrontEnd
Definition NSArray.cpp:265
GenericNSArrayISyntheticFrontEnd< IDD32, IDD64, true > NSArrayISyntheticFrontEnd
Definition NSArray.cpp:262
GenericNSArrayISyntheticFrontEnd< IDD32, IDD64, false > NSArrayI_TransferSyntheticFrontEnd
Definition NSArray.cpp:259
uint64_t __NSFrozenArrayMSize(lldb_private::Process &process, lldb::addr_t valobj_addr, Status &error)
Definition NSArray.cpp:269
GenericNSArrayMSyntheticFrontEnd< DataDescriptor< uint32_t >, DataDescriptor< uint64_t > > NSArrayMSyntheticFrontEnd
Definition NSArray.cpp:154
uint64_t __NSArrayMSize(lldb_private::Process &process, lldb::addr_t valobj_addr, Status &error)
Definition NSArray.cpp:174
uint64_t __NSArrayMSize_Impl(lldb_private::Process &process, lldb::addr_t valobj_addr, Status &error)
Definition NSArray.cpp:160
SyntheticChildrenFrontEnd * NSArraySyntheticFrontEndCreator(CXXSyntheticChildren *, lldb::ValueObjectSP)
bool NSArraySummaryProvider(ValueObject &valobj, Stream &stream, const TypeSummaryOptions &options)
Definition NSArray.cpp:325
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
uint64_t addr_t
Definition lldb-types.h:80