1dda28197Spatrick //===-- AppleObjCClassDescriptorV2.cpp ------------------------------------===//
2061da546Spatrick //
3061da546Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4061da546Spatrick // See https://llvm.org/LICENSE.txt for license information.
5061da546Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6061da546Spatrick //
7061da546Spatrick //===----------------------------------------------------------------------===//
8061da546Spatrick 
9061da546Spatrick #include "AppleObjCClassDescriptorV2.h"
10061da546Spatrick 
11061da546Spatrick #include "lldb/Expression/FunctionCaller.h"
12*f6aab3d8Srobert #include "lldb/Target/ABI.h"
13*f6aab3d8Srobert #include "lldb/Utility/LLDBLog.h"
14061da546Spatrick #include "lldb/Utility/Log.h"
15061da546Spatrick 
16061da546Spatrick using namespace lldb;
17061da546Spatrick using namespace lldb_private;
18061da546Spatrick 
Read_objc_class(Process * process,std::unique_ptr<objc_class_t> & objc_class) const19061da546Spatrick bool ClassDescriptorV2::Read_objc_class(
20061da546Spatrick     Process *process, std::unique_ptr<objc_class_t> &objc_class) const {
21dda28197Spatrick   objc_class = std::make_unique<objc_class_t>();
22061da546Spatrick 
23061da546Spatrick   bool ret = objc_class->Read(process, m_objc_class_ptr);
24061da546Spatrick 
25061da546Spatrick   if (!ret)
26061da546Spatrick     objc_class.reset();
27061da546Spatrick 
28061da546Spatrick   return ret;
29061da546Spatrick }
30061da546Spatrick 
GetClassDataMask(Process * process)31061da546Spatrick static lldb::addr_t GetClassDataMask(Process *process) {
32061da546Spatrick   switch (process->GetAddressByteSize()) {
33061da546Spatrick   case 4:
34061da546Spatrick     return 0xfffffffcUL;
35061da546Spatrick   case 8:
36061da546Spatrick     return 0x00007ffffffffff8UL;
37061da546Spatrick   default:
38061da546Spatrick     break;
39061da546Spatrick   }
40061da546Spatrick 
41061da546Spatrick   return LLDB_INVALID_ADDRESS;
42061da546Spatrick }
43061da546Spatrick 
Read(Process * process,lldb::addr_t addr)44061da546Spatrick bool ClassDescriptorV2::objc_class_t::Read(Process *process,
45061da546Spatrick                                            lldb::addr_t addr) {
46061da546Spatrick   size_t ptr_size = process->GetAddressByteSize();
47061da546Spatrick 
48061da546Spatrick   size_t objc_class_size = ptr_size    // uintptr_t isa;
49061da546Spatrick                            + ptr_size  // Class superclass;
50061da546Spatrick                            + ptr_size  // void *cache;
51061da546Spatrick                            + ptr_size  // IMP *vtable;
52061da546Spatrick                            + ptr_size; // uintptr_t data_NEVER_USE;
53061da546Spatrick 
54061da546Spatrick   DataBufferHeap objc_class_buf(objc_class_size, '\0');
55061da546Spatrick   Status error;
56061da546Spatrick 
57061da546Spatrick   process->ReadMemory(addr, objc_class_buf.GetBytes(), objc_class_size, error);
58061da546Spatrick   if (error.Fail()) {
59061da546Spatrick     return false;
60061da546Spatrick   }
61061da546Spatrick 
62061da546Spatrick   DataExtractor extractor(objc_class_buf.GetBytes(), objc_class_size,
63061da546Spatrick                           process->GetByteOrder(),
64061da546Spatrick                           process->GetAddressByteSize());
65061da546Spatrick 
66061da546Spatrick   lldb::offset_t cursor = 0;
67061da546Spatrick 
68061da546Spatrick   m_isa = extractor.GetAddress_unchecked(&cursor);        // uintptr_t isa;
69061da546Spatrick   m_superclass = extractor.GetAddress_unchecked(&cursor); // Class superclass;
70061da546Spatrick   m_cache_ptr = extractor.GetAddress_unchecked(&cursor);  // void *cache;
71061da546Spatrick   m_vtable_ptr = extractor.GetAddress_unchecked(&cursor); // IMP *vtable;
72061da546Spatrick   lldb::addr_t data_NEVER_USE =
73061da546Spatrick       extractor.GetAddress_unchecked(&cursor); // uintptr_t data_NEVER_USE;
74061da546Spatrick 
75061da546Spatrick   m_flags = (uint8_t)(data_NEVER_USE & (lldb::addr_t)3);
76061da546Spatrick   m_data_ptr = data_NEVER_USE & GetClassDataMask(process);
77061da546Spatrick 
78*f6aab3d8Srobert   if (ABISP abi_sp = process->GetABI()) {
79*f6aab3d8Srobert     m_isa = abi_sp->FixCodeAddress(m_isa);
80*f6aab3d8Srobert     m_superclass = abi_sp->FixCodeAddress(m_superclass);
81*f6aab3d8Srobert     m_data_ptr = abi_sp->FixCodeAddress(m_data_ptr);
82*f6aab3d8Srobert   }
83061da546Spatrick   return true;
84061da546Spatrick }
85061da546Spatrick 
Read(Process * process,lldb::addr_t addr)86061da546Spatrick bool ClassDescriptorV2::class_rw_t::Read(Process *process, lldb::addr_t addr) {
87061da546Spatrick   size_t ptr_size = process->GetAddressByteSize();
88061da546Spatrick 
89061da546Spatrick   size_t size = sizeof(uint32_t)   // uint32_t flags;
90061da546Spatrick                 + sizeof(uint32_t) // uint32_t version;
91061da546Spatrick                 + ptr_size         // const class_ro_t *ro;
92061da546Spatrick                 + ptr_size         // union { method_list_t **method_lists;
93061da546Spatrick                                    // method_list_t *method_list; };
94061da546Spatrick                 + ptr_size         // struct chained_property_list *properties;
95061da546Spatrick                 + ptr_size         // const protocol_list_t **protocols;
96061da546Spatrick                 + ptr_size         // Class firstSubclass;
97061da546Spatrick                 + ptr_size;        // Class nextSiblingClass;
98061da546Spatrick 
99061da546Spatrick   DataBufferHeap buffer(size, '\0');
100061da546Spatrick   Status error;
101061da546Spatrick 
102061da546Spatrick   process->ReadMemory(addr, buffer.GetBytes(), size, error);
103061da546Spatrick   if (error.Fail()) {
104061da546Spatrick     return false;
105061da546Spatrick   }
106061da546Spatrick 
107061da546Spatrick   DataExtractor extractor(buffer.GetBytes(), size, process->GetByteOrder(),
108061da546Spatrick                           process->GetAddressByteSize());
109061da546Spatrick 
110061da546Spatrick   lldb::offset_t cursor = 0;
111061da546Spatrick 
112061da546Spatrick   m_flags = extractor.GetU32_unchecked(&cursor);
113061da546Spatrick   m_version = extractor.GetU32_unchecked(&cursor);
114061da546Spatrick   m_ro_ptr = extractor.GetAddress_unchecked(&cursor);
115*f6aab3d8Srobert   if (ABISP abi_sp = process->GetABI())
116*f6aab3d8Srobert     m_ro_ptr = abi_sp->FixCodeAddress(m_ro_ptr);
117061da546Spatrick   m_method_list_ptr = extractor.GetAddress_unchecked(&cursor);
118061da546Spatrick   m_properties_ptr = extractor.GetAddress_unchecked(&cursor);
119061da546Spatrick   m_firstSubclass = extractor.GetAddress_unchecked(&cursor);
120061da546Spatrick   m_nextSiblingClass = extractor.GetAddress_unchecked(&cursor);
121061da546Spatrick 
122061da546Spatrick   if (m_ro_ptr & 1) {
123061da546Spatrick     DataBufferHeap buffer(ptr_size, '\0');
124061da546Spatrick     process->ReadMemory(m_ro_ptr ^ 1, buffer.GetBytes(), ptr_size, error);
125061da546Spatrick     if (error.Fail())
126061da546Spatrick       return false;
127061da546Spatrick     cursor = 0;
128061da546Spatrick     DataExtractor extractor(buffer.GetBytes(), ptr_size,
129061da546Spatrick                             process->GetByteOrder(),
130061da546Spatrick                             process->GetAddressByteSize());
131061da546Spatrick     m_ro_ptr = extractor.GetAddress_unchecked(&cursor);
132*f6aab3d8Srobert     if (ABISP abi_sp = process->GetABI())
133*f6aab3d8Srobert       m_ro_ptr = abi_sp->FixCodeAddress(m_ro_ptr);
134061da546Spatrick   }
135061da546Spatrick 
136061da546Spatrick   return true;
137061da546Spatrick }
138061da546Spatrick 
Read(Process * process,lldb::addr_t addr)139061da546Spatrick bool ClassDescriptorV2::class_ro_t::Read(Process *process, lldb::addr_t addr) {
140061da546Spatrick   size_t ptr_size = process->GetAddressByteSize();
141061da546Spatrick 
142061da546Spatrick   size_t size = sizeof(uint32_t)   // uint32_t flags;
143061da546Spatrick                 + sizeof(uint32_t) // uint32_t instanceStart;
144061da546Spatrick                 + sizeof(uint32_t) // uint32_t instanceSize;
145061da546Spatrick                 + (ptr_size == 8 ? sizeof(uint32_t)
146061da546Spatrick                                  : 0) // uint32_t reserved; // __LP64__ only
147061da546Spatrick                 + ptr_size            // const uint8_t *ivarLayout;
148061da546Spatrick                 + ptr_size            // const char *name;
149061da546Spatrick                 + ptr_size            // const method_list_t *baseMethods;
150061da546Spatrick                 + ptr_size            // const protocol_list_t *baseProtocols;
151061da546Spatrick                 + ptr_size            // const ivar_list_t *ivars;
152061da546Spatrick                 + ptr_size            // const uint8_t *weakIvarLayout;
153061da546Spatrick                 + ptr_size;           // const property_list_t *baseProperties;
154061da546Spatrick 
155061da546Spatrick   DataBufferHeap buffer(size, '\0');
156061da546Spatrick   Status error;
157061da546Spatrick 
158061da546Spatrick   process->ReadMemory(addr, buffer.GetBytes(), size, error);
159061da546Spatrick   if (error.Fail()) {
160061da546Spatrick     return false;
161061da546Spatrick   }
162061da546Spatrick 
163061da546Spatrick   DataExtractor extractor(buffer.GetBytes(), size, process->GetByteOrder(),
164061da546Spatrick                           process->GetAddressByteSize());
165061da546Spatrick 
166061da546Spatrick   lldb::offset_t cursor = 0;
167061da546Spatrick 
168061da546Spatrick   m_flags = extractor.GetU32_unchecked(&cursor);
169061da546Spatrick   m_instanceStart = extractor.GetU32_unchecked(&cursor);
170061da546Spatrick   m_instanceSize = extractor.GetU32_unchecked(&cursor);
171061da546Spatrick   if (ptr_size == 8)
172061da546Spatrick     m_reserved = extractor.GetU32_unchecked(&cursor);
173061da546Spatrick   else
174061da546Spatrick     m_reserved = 0;
175061da546Spatrick   m_ivarLayout_ptr = extractor.GetAddress_unchecked(&cursor);
176061da546Spatrick   m_name_ptr = extractor.GetAddress_unchecked(&cursor);
177061da546Spatrick   m_baseMethods_ptr = extractor.GetAddress_unchecked(&cursor);
178061da546Spatrick   m_baseProtocols_ptr = extractor.GetAddress_unchecked(&cursor);
179061da546Spatrick   m_ivars_ptr = extractor.GetAddress_unchecked(&cursor);
180061da546Spatrick   m_weakIvarLayout_ptr = extractor.GetAddress_unchecked(&cursor);
181061da546Spatrick   m_baseProperties_ptr = extractor.GetAddress_unchecked(&cursor);
182061da546Spatrick 
183061da546Spatrick   DataBufferHeap name_buf(1024, '\0');
184061da546Spatrick 
185061da546Spatrick   process->ReadCStringFromMemory(m_name_ptr, (char *)name_buf.GetBytes(),
186061da546Spatrick                                  name_buf.GetByteSize(), error);
187061da546Spatrick 
188061da546Spatrick   if (error.Fail()) {
189061da546Spatrick     return false;
190061da546Spatrick   }
191061da546Spatrick 
192061da546Spatrick   m_name.assign((char *)name_buf.GetBytes());
193061da546Spatrick 
194061da546Spatrick   return true;
195061da546Spatrick }
196061da546Spatrick 
Read_class_row(Process * process,const objc_class_t & objc_class,std::unique_ptr<class_ro_t> & class_ro,std::unique_ptr<class_rw_t> & class_rw) const197061da546Spatrick bool ClassDescriptorV2::Read_class_row(
198061da546Spatrick     Process *process, const objc_class_t &objc_class,
199061da546Spatrick     std::unique_ptr<class_ro_t> &class_ro,
200061da546Spatrick     std::unique_ptr<class_rw_t> &class_rw) const {
201061da546Spatrick   class_ro.reset();
202061da546Spatrick   class_rw.reset();
203061da546Spatrick 
204061da546Spatrick   Status error;
205061da546Spatrick   uint32_t class_row_t_flags = process->ReadUnsignedIntegerFromMemory(
206061da546Spatrick       objc_class.m_data_ptr, sizeof(uint32_t), 0, error);
207061da546Spatrick   if (!error.Success())
208061da546Spatrick     return false;
209061da546Spatrick 
210061da546Spatrick   if (class_row_t_flags & RW_REALIZED) {
211dda28197Spatrick     class_rw = std::make_unique<class_rw_t>();
212061da546Spatrick 
213061da546Spatrick     if (!class_rw->Read(process, objc_class.m_data_ptr)) {
214061da546Spatrick       class_rw.reset();
215061da546Spatrick       return false;
216061da546Spatrick     }
217061da546Spatrick 
218dda28197Spatrick     class_ro = std::make_unique<class_ro_t>();
219061da546Spatrick 
220061da546Spatrick     if (!class_ro->Read(process, class_rw->m_ro_ptr)) {
221061da546Spatrick       class_rw.reset();
222061da546Spatrick       class_ro.reset();
223061da546Spatrick       return false;
224061da546Spatrick     }
225061da546Spatrick   } else {
226dda28197Spatrick     class_ro = std::make_unique<class_ro_t>();
227061da546Spatrick 
228061da546Spatrick     if (!class_ro->Read(process, objc_class.m_data_ptr)) {
229061da546Spatrick       class_ro.reset();
230061da546Spatrick       return false;
231061da546Spatrick     }
232061da546Spatrick   }
233061da546Spatrick 
234061da546Spatrick   return true;
235061da546Spatrick }
236061da546Spatrick 
Read(Process * process,lldb::addr_t addr)237061da546Spatrick bool ClassDescriptorV2::method_list_t::Read(Process *process,
238061da546Spatrick                                             lldb::addr_t addr) {
239061da546Spatrick   size_t size = sizeof(uint32_t)    // uint32_t entsize_NEVER_USE;
240061da546Spatrick                 + sizeof(uint32_t); // uint32_t count;
241061da546Spatrick 
242061da546Spatrick   DataBufferHeap buffer(size, '\0');
243061da546Spatrick   Status error;
244061da546Spatrick 
245*f6aab3d8Srobert   if (ABISP abi_sp = process->GetABI())
246*f6aab3d8Srobert     addr = abi_sp->FixCodeAddress(addr);
247061da546Spatrick   process->ReadMemory(addr, buffer.GetBytes(), size, error);
248061da546Spatrick   if (error.Fail()) {
249061da546Spatrick     return false;
250061da546Spatrick   }
251061da546Spatrick 
252061da546Spatrick   DataExtractor extractor(buffer.GetBytes(), size, process->GetByteOrder(),
253061da546Spatrick                           process->GetAddressByteSize());
254061da546Spatrick 
255061da546Spatrick   lldb::offset_t cursor = 0;
256061da546Spatrick 
257dda28197Spatrick   uint32_t entsize = extractor.GetU32_unchecked(&cursor);
258dda28197Spatrick   m_is_small = (entsize & 0x80000000) != 0;
259dda28197Spatrick   m_has_direct_selector = (entsize & 0x40000000) != 0;
260dda28197Spatrick   m_entsize = entsize & 0xfffc;
261061da546Spatrick   m_count = extractor.GetU32_unchecked(&cursor);
262061da546Spatrick   m_first_ptr = addr + cursor;
263061da546Spatrick 
264061da546Spatrick   return true;
265061da546Spatrick }
266061da546Spatrick 
Read(Process * process,lldb::addr_t addr,lldb::addr_t relative_selector_base_addr,bool is_small,bool has_direct_sel)267dda28197Spatrick bool ClassDescriptorV2::method_t::Read(Process *process, lldb::addr_t addr,
268be691f3bSpatrick                                        lldb::addr_t relative_selector_base_addr,
269dda28197Spatrick                                        bool is_small, bool has_direct_sel) {
270dda28197Spatrick   size_t ptr_size = process->GetAddressByteSize();
271dda28197Spatrick   size_t size = GetSize(process, is_small);
272061da546Spatrick 
273061da546Spatrick   DataBufferHeap buffer(size, '\0');
274061da546Spatrick   Status error;
275061da546Spatrick 
276061da546Spatrick   process->ReadMemory(addr, buffer.GetBytes(), size, error);
277061da546Spatrick   if (error.Fail()) {
278061da546Spatrick     return false;
279061da546Spatrick   }
280061da546Spatrick 
281061da546Spatrick   DataExtractor extractor(buffer.GetBytes(), size, process->GetByteOrder(),
282dda28197Spatrick                           ptr_size);
283061da546Spatrick   lldb::offset_t cursor = 0;
284061da546Spatrick 
285dda28197Spatrick   if (is_small) {
286dda28197Spatrick     uint32_t nameref_offset = extractor.GetU32_unchecked(&cursor);
287dda28197Spatrick     uint32_t types_offset = extractor.GetU32_unchecked(&cursor);
288dda28197Spatrick     uint32_t imp_offset = extractor.GetU32_unchecked(&cursor);
289dda28197Spatrick 
290dda28197Spatrick     m_name_ptr = addr + nameref_offset;
291dda28197Spatrick 
292dda28197Spatrick     if (!has_direct_sel) {
293dda28197Spatrick       // The SEL offset points to a SELRef. We need to dereference twice.
294dda28197Spatrick       m_name_ptr = process->ReadUnsignedIntegerFromMemory(m_name_ptr, ptr_size,
295dda28197Spatrick                                                           0, error);
296dda28197Spatrick       if (!error.Success())
297dda28197Spatrick         return false;
298be691f3bSpatrick     } else if (relative_selector_base_addr != LLDB_INVALID_ADDRESS) {
299be691f3bSpatrick       m_name_ptr = relative_selector_base_addr + nameref_offset;
300dda28197Spatrick     }
301dda28197Spatrick     m_types_ptr = addr + 4 + types_offset;
302dda28197Spatrick     m_imp_ptr = addr + 8 + imp_offset;
303dda28197Spatrick   } else {
304061da546Spatrick     m_name_ptr = extractor.GetAddress_unchecked(&cursor);
305061da546Spatrick     m_types_ptr = extractor.GetAddress_unchecked(&cursor);
306061da546Spatrick     m_imp_ptr = extractor.GetAddress_unchecked(&cursor);
307dda28197Spatrick   }
308061da546Spatrick 
309061da546Spatrick   process->ReadCStringFromMemory(m_name_ptr, m_name, error);
310061da546Spatrick   if (error.Fail()) {
311061da546Spatrick     return false;
312061da546Spatrick   }
313061da546Spatrick 
314061da546Spatrick   process->ReadCStringFromMemory(m_types_ptr, m_types, error);
315061da546Spatrick   return !error.Fail();
316061da546Spatrick }
317061da546Spatrick 
Read(Process * process,lldb::addr_t addr)318061da546Spatrick bool ClassDescriptorV2::ivar_list_t::Read(Process *process, lldb::addr_t addr) {
319061da546Spatrick   size_t size = sizeof(uint32_t)    // uint32_t entsize;
320061da546Spatrick                 + sizeof(uint32_t); // uint32_t count;
321061da546Spatrick 
322061da546Spatrick   DataBufferHeap buffer(size, '\0');
323061da546Spatrick   Status error;
324061da546Spatrick 
325061da546Spatrick   process->ReadMemory(addr, buffer.GetBytes(), size, error);
326061da546Spatrick   if (error.Fail()) {
327061da546Spatrick     return false;
328061da546Spatrick   }
329061da546Spatrick 
330061da546Spatrick   DataExtractor extractor(buffer.GetBytes(), size, process->GetByteOrder(),
331061da546Spatrick                           process->GetAddressByteSize());
332061da546Spatrick 
333061da546Spatrick   lldb::offset_t cursor = 0;
334061da546Spatrick 
335061da546Spatrick   m_entsize = extractor.GetU32_unchecked(&cursor);
336061da546Spatrick   m_count = extractor.GetU32_unchecked(&cursor);
337061da546Spatrick   m_first_ptr = addr + cursor;
338061da546Spatrick 
339061da546Spatrick   return true;
340061da546Spatrick }
341061da546Spatrick 
Read(Process * process,lldb::addr_t addr)342061da546Spatrick bool ClassDescriptorV2::ivar_t::Read(Process *process, lldb::addr_t addr) {
343061da546Spatrick   size_t size = GetSize(process);
344061da546Spatrick 
345061da546Spatrick   DataBufferHeap buffer(size, '\0');
346061da546Spatrick   Status error;
347061da546Spatrick 
348061da546Spatrick   process->ReadMemory(addr, buffer.GetBytes(), size, error);
349061da546Spatrick   if (error.Fail()) {
350061da546Spatrick     return false;
351061da546Spatrick   }
352061da546Spatrick 
353061da546Spatrick   DataExtractor extractor(buffer.GetBytes(), size, process->GetByteOrder(),
354061da546Spatrick                           process->GetAddressByteSize());
355061da546Spatrick 
356061da546Spatrick   lldb::offset_t cursor = 0;
357061da546Spatrick 
358061da546Spatrick   m_offset_ptr = extractor.GetAddress_unchecked(&cursor);
359061da546Spatrick   m_name_ptr = extractor.GetAddress_unchecked(&cursor);
360061da546Spatrick   m_type_ptr = extractor.GetAddress_unchecked(&cursor);
361061da546Spatrick   m_alignment = extractor.GetU32_unchecked(&cursor);
362061da546Spatrick   m_size = extractor.GetU32_unchecked(&cursor);
363061da546Spatrick 
364061da546Spatrick   process->ReadCStringFromMemory(m_name_ptr, m_name, error);
365061da546Spatrick   if (error.Fail()) {
366061da546Spatrick     return false;
367061da546Spatrick   }
368061da546Spatrick 
369061da546Spatrick   process->ReadCStringFromMemory(m_type_ptr, m_type, error);
370061da546Spatrick   return !error.Fail();
371061da546Spatrick }
372061da546Spatrick 
Describe(std::function<void (ObjCLanguageRuntime::ObjCISA)> const & superclass_func,std::function<bool (const char *,const char *)> const & instance_method_func,std::function<bool (const char *,const char *)> const & class_method_func,std::function<bool (const char *,const char *,lldb::addr_t,uint64_t)> const & ivar_func) const373061da546Spatrick bool ClassDescriptorV2::Describe(
374061da546Spatrick     std::function<void(ObjCLanguageRuntime::ObjCISA)> const &superclass_func,
375061da546Spatrick     std::function<bool(const char *, const char *)> const &instance_method_func,
376061da546Spatrick     std::function<bool(const char *, const char *)> const &class_method_func,
377061da546Spatrick     std::function<bool(const char *, const char *, lldb::addr_t,
378061da546Spatrick                        uint64_t)> const &ivar_func) const {
379061da546Spatrick   lldb_private::Process *process = m_runtime.GetProcess();
380061da546Spatrick 
381061da546Spatrick   std::unique_ptr<objc_class_t> objc_class;
382061da546Spatrick   std::unique_ptr<class_ro_t> class_ro;
383061da546Spatrick   std::unique_ptr<class_rw_t> class_rw;
384061da546Spatrick 
385061da546Spatrick   if (!Read_objc_class(process, objc_class))
386061da546Spatrick     return false;
387061da546Spatrick   if (!Read_class_row(process, *objc_class, class_ro, class_rw))
388061da546Spatrick     return false;
389061da546Spatrick 
390061da546Spatrick   static ConstString NSObject_name("NSObject");
391061da546Spatrick 
392061da546Spatrick   if (m_name != NSObject_name && superclass_func)
393061da546Spatrick     superclass_func(objc_class->m_superclass);
394061da546Spatrick 
395061da546Spatrick   if (instance_method_func) {
396061da546Spatrick     std::unique_ptr<method_list_t> base_method_list;
397061da546Spatrick 
398dda28197Spatrick     base_method_list = std::make_unique<method_list_t>();
399061da546Spatrick     if (!base_method_list->Read(process, class_ro->m_baseMethods_ptr))
400061da546Spatrick       return false;
401061da546Spatrick 
402dda28197Spatrick     bool is_small = base_method_list->m_is_small;
403dda28197Spatrick     bool has_direct_selector = base_method_list->m_has_direct_selector;
404dda28197Spatrick 
405dda28197Spatrick     if (base_method_list->m_entsize != method_t::GetSize(process, is_small))
406061da546Spatrick       return false;
407061da546Spatrick 
408be691f3bSpatrick     std::unique_ptr<method_t> method = std::make_unique<method_t>();
409be691f3bSpatrick     lldb::addr_t relative_selector_base_addr =
410be691f3bSpatrick         m_runtime.GetRelativeSelectorBaseAddr();
411061da546Spatrick     for (uint32_t i = 0, e = base_method_list->m_count; i < e; ++i) {
412dda28197Spatrick       method->Read(process,
413dda28197Spatrick                    base_method_list->m_first_ptr +
414dda28197Spatrick                        (i * base_method_list->m_entsize),
415be691f3bSpatrick                    relative_selector_base_addr, is_small, has_direct_selector);
416061da546Spatrick 
417061da546Spatrick       if (instance_method_func(method->m_name.c_str(), method->m_types.c_str()))
418061da546Spatrick         break;
419061da546Spatrick     }
420061da546Spatrick   }
421061da546Spatrick 
422061da546Spatrick   if (class_method_func) {
423061da546Spatrick     AppleObjCRuntime::ClassDescriptorSP metaclass(GetMetaclass());
424061da546Spatrick 
425061da546Spatrick     // We don't care about the metaclass's superclass, or its class methods.
426061da546Spatrick     // Its instance methods are our class methods.
427061da546Spatrick 
428061da546Spatrick     if (metaclass) {
429061da546Spatrick       metaclass->Describe(
430061da546Spatrick           std::function<void(ObjCLanguageRuntime::ObjCISA)>(nullptr),
431061da546Spatrick           class_method_func,
432061da546Spatrick           std::function<bool(const char *, const char *)>(nullptr),
433061da546Spatrick           std::function<bool(const char *, const char *, lldb::addr_t,
434061da546Spatrick                              uint64_t)>(nullptr));
435061da546Spatrick     }
436061da546Spatrick   }
437061da546Spatrick 
438061da546Spatrick   if (ivar_func) {
439061da546Spatrick     if (class_ro->m_ivars_ptr != 0) {
440061da546Spatrick       ivar_list_t ivar_list;
441061da546Spatrick       if (!ivar_list.Read(process, class_ro->m_ivars_ptr))
442061da546Spatrick         return false;
443061da546Spatrick 
444061da546Spatrick       if (ivar_list.m_entsize != ivar_t::GetSize(process))
445061da546Spatrick         return false;
446061da546Spatrick 
447061da546Spatrick       ivar_t ivar;
448061da546Spatrick 
449061da546Spatrick       for (uint32_t i = 0, e = ivar_list.m_count; i < e; ++i) {
450061da546Spatrick         ivar.Read(process, ivar_list.m_first_ptr + (i * ivar_list.m_entsize));
451061da546Spatrick 
452061da546Spatrick         if (ivar_func(ivar.m_name.c_str(), ivar.m_type.c_str(),
453061da546Spatrick                       ivar.m_offset_ptr, ivar.m_size))
454061da546Spatrick           break;
455061da546Spatrick       }
456061da546Spatrick     }
457061da546Spatrick   }
458061da546Spatrick 
459061da546Spatrick   return true;
460061da546Spatrick }
461061da546Spatrick 
GetClassName()462061da546Spatrick ConstString ClassDescriptorV2::GetClassName() {
463061da546Spatrick   if (!m_name) {
464061da546Spatrick     lldb_private::Process *process = m_runtime.GetProcess();
465061da546Spatrick 
466061da546Spatrick     if (process) {
467061da546Spatrick       std::unique_ptr<objc_class_t> objc_class;
468061da546Spatrick       std::unique_ptr<class_ro_t> class_ro;
469061da546Spatrick       std::unique_ptr<class_rw_t> class_rw;
470061da546Spatrick 
471061da546Spatrick       if (!Read_objc_class(process, objc_class))
472061da546Spatrick         return m_name;
473061da546Spatrick       if (!Read_class_row(process, *objc_class, class_ro, class_rw))
474061da546Spatrick         return m_name;
475061da546Spatrick 
476061da546Spatrick       m_name = ConstString(class_ro->m_name.c_str());
477061da546Spatrick     }
478061da546Spatrick   }
479061da546Spatrick   return m_name;
480061da546Spatrick }
481061da546Spatrick 
GetSuperclass()482061da546Spatrick ObjCLanguageRuntime::ClassDescriptorSP ClassDescriptorV2::GetSuperclass() {
483061da546Spatrick   lldb_private::Process *process = m_runtime.GetProcess();
484061da546Spatrick 
485061da546Spatrick   if (!process)
486061da546Spatrick     return ObjCLanguageRuntime::ClassDescriptorSP();
487061da546Spatrick 
488061da546Spatrick   std::unique_ptr<objc_class_t> objc_class;
489061da546Spatrick 
490061da546Spatrick   if (!Read_objc_class(process, objc_class))
491061da546Spatrick     return ObjCLanguageRuntime::ClassDescriptorSP();
492061da546Spatrick 
493061da546Spatrick   return m_runtime.ObjCLanguageRuntime::GetClassDescriptorFromISA(
494061da546Spatrick       objc_class->m_superclass);
495061da546Spatrick }
496061da546Spatrick 
GetMetaclass() const497061da546Spatrick ObjCLanguageRuntime::ClassDescriptorSP ClassDescriptorV2::GetMetaclass() const {
498061da546Spatrick   lldb_private::Process *process = m_runtime.GetProcess();
499061da546Spatrick 
500061da546Spatrick   if (!process)
501061da546Spatrick     return ObjCLanguageRuntime::ClassDescriptorSP();
502061da546Spatrick 
503061da546Spatrick   std::unique_ptr<objc_class_t> objc_class;
504061da546Spatrick 
505061da546Spatrick   if (!Read_objc_class(process, objc_class))
506061da546Spatrick     return ObjCLanguageRuntime::ClassDescriptorSP();
507061da546Spatrick 
508061da546Spatrick   lldb::addr_t candidate_isa = m_runtime.GetPointerISA(objc_class->m_isa);
509061da546Spatrick 
510061da546Spatrick   return ObjCLanguageRuntime::ClassDescriptorSP(
511061da546Spatrick       new ClassDescriptorV2(m_runtime, candidate_isa, nullptr));
512061da546Spatrick }
513061da546Spatrick 
GetInstanceSize()514061da546Spatrick uint64_t ClassDescriptorV2::GetInstanceSize() {
515061da546Spatrick   lldb_private::Process *process = m_runtime.GetProcess();
516061da546Spatrick 
517061da546Spatrick   if (process) {
518061da546Spatrick     std::unique_ptr<objc_class_t> objc_class;
519061da546Spatrick     std::unique_ptr<class_ro_t> class_ro;
520061da546Spatrick     std::unique_ptr<class_rw_t> class_rw;
521061da546Spatrick 
522061da546Spatrick     if (!Read_objc_class(process, objc_class))
523061da546Spatrick       return 0;
524061da546Spatrick     if (!Read_class_row(process, *objc_class, class_ro, class_rw))
525061da546Spatrick       return 0;
526061da546Spatrick 
527061da546Spatrick     return class_ro->m_instanceSize;
528061da546Spatrick   }
529061da546Spatrick 
530061da546Spatrick   return 0;
531061da546Spatrick }
532061da546Spatrick 
iVarsStorage()533be691f3bSpatrick ClassDescriptorV2::iVarsStorage::iVarsStorage() : m_ivars(), m_mutex() {}
534061da546Spatrick 
size()535061da546Spatrick size_t ClassDescriptorV2::iVarsStorage::size() { return m_ivars.size(); }
536061da546Spatrick 
537061da546Spatrick ClassDescriptorV2::iVarDescriptor &ClassDescriptorV2::iVarsStorage::
operator [](size_t idx)538061da546Spatrick operator[](size_t idx) {
539061da546Spatrick   return m_ivars[idx];
540061da546Spatrick }
541061da546Spatrick 
fill(AppleObjCRuntimeV2 & runtime,ClassDescriptorV2 & descriptor)542061da546Spatrick void ClassDescriptorV2::iVarsStorage::fill(AppleObjCRuntimeV2 &runtime,
543061da546Spatrick                                            ClassDescriptorV2 &descriptor) {
544061da546Spatrick   if (m_filled)
545061da546Spatrick     return;
546061da546Spatrick   std::lock_guard<std::recursive_mutex> guard(m_mutex);
547*f6aab3d8Srobert   Log *log = GetLog(LLDBLog::Types);
548061da546Spatrick   LLDB_LOGV(log, "class_name = {0}", descriptor.GetClassName());
549061da546Spatrick   m_filled = true;
550061da546Spatrick   ObjCLanguageRuntime::EncodingToTypeSP encoding_to_type_sp(
551061da546Spatrick       runtime.GetEncodingToType());
552061da546Spatrick   Process *process(runtime.GetProcess());
553061da546Spatrick   if (!encoding_to_type_sp)
554061da546Spatrick     return;
555061da546Spatrick   descriptor.Describe(nullptr, nullptr, nullptr, [this, process,
556061da546Spatrick                                                   encoding_to_type_sp,
557061da546Spatrick                                                   log](const char *name,
558061da546Spatrick                                                        const char *type,
559061da546Spatrick                                                        lldb::addr_t offset_ptr,
560061da546Spatrick                                                        uint64_t size) -> bool {
561061da546Spatrick     const bool for_expression = false;
562061da546Spatrick     const bool stop_loop = false;
563061da546Spatrick     LLDB_LOGV(log, "name = {0}, encoding = {1}, offset_ptr = {2:x}, size = {3}",
564061da546Spatrick               name, type, offset_ptr, size);
565061da546Spatrick     CompilerType ivar_type =
566061da546Spatrick         encoding_to_type_sp->RealizeType(type, for_expression);
567061da546Spatrick     if (ivar_type) {
568061da546Spatrick       LLDB_LOGV(log,
569061da546Spatrick                 "name = {0}, encoding = {1}, offset_ptr = {2:x}, size = "
570061da546Spatrick                 "{3}, type_size = {4}",
571061da546Spatrick                 name, type, offset_ptr, size,
572*f6aab3d8Srobert                 ivar_type.GetByteSize(nullptr).value_or(0));
573061da546Spatrick       Scalar offset_scalar;
574061da546Spatrick       Status error;
575061da546Spatrick       const int offset_ptr_size = 4;
576061da546Spatrick       const bool is_signed = false;
577061da546Spatrick       size_t read = process->ReadScalarIntegerFromMemory(
578061da546Spatrick           offset_ptr, offset_ptr_size, is_signed, offset_scalar, error);
579061da546Spatrick       if (error.Success() && 4 == read) {
580061da546Spatrick         LLDB_LOGV(log, "offset_ptr = {0:x} --> {1}", offset_ptr,
581061da546Spatrick                   offset_scalar.SInt());
582061da546Spatrick         m_ivars.push_back(
583061da546Spatrick             {ConstString(name), ivar_type, size, offset_scalar.SInt()});
584061da546Spatrick       } else
585061da546Spatrick         LLDB_LOGV(log, "offset_ptr = {0:x} --> read fail, read = %{1}",
586061da546Spatrick                   offset_ptr, read);
587061da546Spatrick     }
588061da546Spatrick     return stop_loop;
589061da546Spatrick   });
590061da546Spatrick }
591061da546Spatrick 
GetIVarInformation()592061da546Spatrick void ClassDescriptorV2::GetIVarInformation() {
593061da546Spatrick   m_ivars_storage.fill(m_runtime, *this);
594061da546Spatrick }
595