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