ObjFW  Diff

Differences From Artifact [66376a6ccb]:

  • File src/runtime/class.m — part of check-in [ef6d69931e] at 2019-02-07 00:46:41 on branch trunk — Make style consistent between ObjFW and ObjFW_RT

    ObjFW_RT used to be a separate project that followed the BSD style, as
    it was written in pure C, while ObjFW's style is based on the BSD style
    with changes to make it a better fit for Objective-C. This commit
    changes ObjFW_RT to use the same style as ObjFW. (user: js, size: 20544) [annotate] [blame] [check-ins using]

To Artifact [f8bb9979ba]:


30
31
32
33
34
35
36
37

38
39
40
41
42
43

44
45
46
47
48
49


50
51

52
53
54
55
56

57
58
59
60
61
62
63
64
65
66

67
68
69
70
71
72
73
74

75
76
77
78

79
80
81
82
83
84
85
86
87
88

89
90
91
92
93
94
95
30
31
32
33
34
35
36

37
38
39
40
41
42

43
44
45
46
47


48
49
50

51
52
53
54
55

56
57
58
59
60
61
62
63
64
65

66
67
68
69
70
71
72
73

74
75
76
77

78
79
80
81
82
83
84
85
86
87

88
89
90
91
92
93
94
95







-
+





-
+




-
-
+
+

-
+




-
+









-
+







-
+



-
+









-
+







static Class *loadQueue = NULL;
static size_t loadQueueCount = 0;
static struct objc_dtable *emptyDTable = NULL;
static unsigned lookupsUntilFastPath = 128;
static struct objc_sparsearray *fastPath = NULL;

static void
registerClass(struct objc_abi_class *cls)
registerClass(struct objc_abi_class *rawClass)
{
	if (classes == NULL)
		classes = objc_hashtable_new(
		    objc_hash_string, objc_equal_string, 2);

	objc_hashtable_set(classes, cls->name, cls);
	objc_hashtable_set(classes, rawClass->name, rawClass);

	if (emptyDTable == NULL)
		emptyDTable = objc_dtable_new();

	cls->DTable = emptyDTable;
	cls->metaclass->DTable = emptyDTable;
	rawClass->DTable = emptyDTable;
	rawClass->metaclass->DTable = emptyDTable;

	if (strcmp(cls->name, "Protocol") != 0)
	if (strcmp(rawClass->name, "Protocol") != 0)
		classesCount++;
}

bool
class_registerAlias_np(Class cls, const char *name)
class_registerAlias_np(Class class, const char *name)
{
	objc_global_mutex_lock();

	if (classes == NULL) {
		objc_global_mutex_unlock();

		return NO;
	}

	objc_hashtable_set(classes, name, (Class)((uintptr_t)cls | 1));
	objc_hashtable_set(classes, name, (Class)((uintptr_t)class | 1));

	objc_global_mutex_unlock();

	return YES;
}

static void
registerSelectors(struct objc_abi_class *cls)
registerSelectors(struct objc_abi_class *rawClass)
{
	struct objc_abi_method_list *methodList;

	for (methodList = cls->methodList; methodList != NULL;
	for (methodList = rawClass->methodList; methodList != NULL;
	    methodList = methodList->next)
		for (unsigned int i = 0; i < methodList->count; i++)
			objc_register_selector((struct objc_abi_selector *)
			    &methodList->methods[i]);
}

Class
objc_classname_to_class(const char *name, bool cache)
{
	Class cls;
	Class class;

	if (classes == NULL)
		return Nil;

	/*
	 * Fast path
	 *
108
109
110
111
112
113
114
115

116
117
118


119
120
121
122
123

124
125
126
127
128
129

130
131
132
133

134
135
136
137

138
139
140
141

142
143
144
145
146
147

148
149
150
151

152
153
154
155

156
157
158
159
160
161
162
163
164
165
166

167
168

169
170
171
172


173
174

175
176

177
178
179
180

181
182
183
184

185
186
187
188


189
190
191


192
193

194
195
196

197
198
199
200

201
202
203

204
205
206
207
208
209
210
211
212



213
214
215
216
217
218
219


220
221
222
223
224

225
226
227
228
229


230
231
232

233
234
235


236
237
238
239
240

241
242
243


244
245

246
247

248
249
250


251
252
253
254
255

256
257

258
259
260

261
262
263

264
265
266


267
268
269
270
271
272
273






274
275
276
277
278


279
280
281
282

283
284

285
286

287
288
289
290



291
292
293
294
295
296
297
298
299
300
301


302
303
304


305
306

307
308

309
310
311


312
313
314
315

316
317

318
319
320
321


322
323
324


325
326
327


328
329
330
331
332
333
334


335
336

337
338
339
340

341
342

343
344
345
346
347
348
349
350
351
352

353
354
355
356
357

358
359

360
361
362
363
364

365
366
367
368
369
370
371
108
109
110
111
112
113
114

115
116


117
118
119
120
121
122

123
124
125
126
127
128

129
130
131
132

133
134
135
136

137
138
139
140

141
142
143
144
145
146

147
148
149
150

151
152
153
154

155
156
157
158
159
160
161
162
163
164
165

166
167

168
169
170


171
172
173

174
175

176
177
178
179

180
181
182
183

184
185
186


187
188
189


190
191
192

193
194
195

196
197
198
199

200
201
202

203
204
205
206
207
208
209
210


211
212
213
214
215
216
217
218


219
220
221
222
223
224

225
226
227
228


229
230
231
232

233
234


235
236
237
238
239
240

241
242


243
244
245

246
247

248
249


250
251
252
253
254
255

256
257

258
259
260

261
262
263

264
265


266
267
268






269
270
271
272
273
274
275
276
277


278
279
280
281
282

283
284

285
286

287
288
289


290
291
292
293
294
295
296
297
298
299
300
301


302
303
304


305
306
307

308
309

310
311


312
313
314
315
316

317
318

319
320
321


322
323
324


325
326
327


328
329
330
331
332
333
334


335
336
337

338
339
340
341

342
343

344
345
346
347
348
349
350
351
352
353

354
355
356
357
358

359
360

361
362
363
364
365

366
367
368
369
370
371
372
373







-
+

-
-
+
+




-
+





-
+



-
+



-
+



-
+





-
+



-
+



-
+










-
+

-
+


-
-
+
+

-
+

-
+



-
+



-
+


-
-
+
+

-
-
+
+

-
+


-
+



-
+


-
+







-
-
+
+
+





-
-
+
+




-
+



-
-
+
+


-
+

-
-
+
+




-
+

-
-
+
+

-
+

-
+

-
-
+
+




-
+

-
+


-
+


-
+

-
-
+
+

-
-
-
-
-
-
+
+
+
+
+
+



-
-
+
+



-
+

-
+

-
+


-
-
+
+
+









-
-
+
+

-
-
+
+

-
+

-
+

-
-
+
+



-
+

-
+


-
-
+
+

-
-
+
+

-
-
+
+





-
-
+
+

-
+



-
+

-
+









-
+




-
+

-
+




-
+







	 * Runtime internal usage does not use the fast path and does not count
	 * as a call into objc_classname_to_class(). The reason for this is
	 * that if the runtime calls into objc_classname_to_class(), it already
	 * has the lock and thus the performance gain would be small, but it
	 * would waste memory.
	 */
	if (cache && fastPath != NULL) {
		cls = objc_sparsearray_get(fastPath, (uintptr_t)name);
		class = objc_sparsearray_get(fastPath, (uintptr_t)name);

		if (cls != Nil)
			return cls;
		if (class != Nil)
			return class;
	}

	objc_global_mutex_lock();

	cls = (Class)((uintptr_t)objc_hashtable_get(classes, name) & ~1);
	class = (Class)((uintptr_t)objc_hashtable_get(classes, name) & ~1);

	if (cache && fastPath == NULL && --lookupsUntilFastPath == 0)
		fastPath = objc_sparsearray_new(sizeof(uintptr_t));

	if (cache && fastPath != NULL)
		objc_sparsearray_set(fastPath, (uintptr_t)name, cls);
		objc_sparsearray_set(fastPath, (uintptr_t)name, class);

	objc_global_mutex_unlock();

	return cls;
	return class;
}

static void
callMethod(Class cls, const char *method)
callMethod(Class class, const char *method)
{
	SEL selector = sel_registerName(method);

	for (struct objc_method_list *methodList = cls->isa->methodList;
	for (struct objc_method_list *methodList = class->isa->methodList;
	    methodList != NULL; methodList = methodList->next)
		for (unsigned int i = 0; i < methodList->count; i++)
			if (sel_isEqual((SEL)&methodList->methods[i].selector,
			    selector))
				((void (*)(id, SEL))methodList->methods[i]
				    .implementation)(cls, selector);
				    .implementation)(class, selector);
}

static bool
hasLoad(Class cls)
hasLoad(Class class)
{
	SEL selector = sel_registerName("load");

	for (struct objc_method_list *methodList = cls->isa->methodList;
	for (struct objc_method_list *methodList = class->isa->methodList;
	    methodList != NULL; methodList = methodList->next)
		for (size_t i = 0; i < methodList->count; i++)
			if (sel_isEqual((SEL)&methodList->methods[i].selector,
			    selector))
				return true;

	return false;
}

static void
callLoad(Class cls)
callLoad(Class class)
{
	if (cls->info & OBJC_CLASS_INFO_LOADED)
	if (class->info & OBJC_CLASS_INFO_LOADED)
		return;

	if (cls->superclass != Nil)
		callLoad(cls->superclass);
	if (class->superclass != Nil)
		callLoad(class->superclass);

	callMethod(cls, "load");
	callMethod(class, "load");

	cls->info |= OBJC_CLASS_INFO_LOADED;
	class->info |= OBJC_CLASS_INFO_LOADED;
}

void
objc_update_dtable(Class cls)
objc_update_dtable(Class class)
{
	struct objc_category **categories;

	if (!(cls->info & OBJC_CLASS_INFO_DTABLE))
	if (!(class->info & OBJC_CLASS_INFO_DTABLE))
		return;

	if (cls->DTable == emptyDTable)
		cls->DTable = objc_dtable_new();
	if (class->DTable == emptyDTable)
		class->DTable = objc_dtable_new();

	if (cls->superclass != Nil)
		objc_dtable_copy(cls->DTable, cls->superclass->DTable);
	if (class->superclass != Nil)
		objc_dtable_copy(class->DTable, class->superclass->DTable);

	for (struct objc_method_list *methodList = cls->methodList;
	for (struct objc_method_list *methodList = class->methodList;
	    methodList != NULL; methodList = methodList->next)
		for (unsigned int i = 0; i < methodList->count; i++)
			objc_dtable_set(cls->DTable,
			objc_dtable_set(class->DTable,
			    (uint32_t)methodList->methods[i].selector.UID,
			    methodList->methods[i].implementation);

	if ((categories = objc_categories_for_class(cls)) != NULL) {
	if ((categories = objc_categories_for_class(class)) != NULL) {
		for (unsigned int i = 0; categories[i] != NULL; i++) {
			struct objc_method_list *methodList =
			    (cls->info & OBJC_CLASS_INFO_CLASS
			    (class->info & OBJC_CLASS_INFO_CLASS
			    ? categories[i]->instanceMethods
			    : categories[i]->classMethods);

			for (; methodList != NULL;
			    methodList = methodList->next)
				for (unsigned int j = 0;
				    j < methodList->count; j++)
					objc_dtable_set(cls->DTable, (uint32_t)
					    methodList->methods[j].selector.UID,
					objc_dtable_set(class->DTable,
					    (uint32_t)methodList->methods[j]
					    .selector.UID,
					    methodList->methods[j]
					    .implementation);
		}
	}

	if (cls->subclassList != NULL)
		for (Class *iter = cls->subclassList; *iter != NULL; iter++)
	if (class->subclassList != NULL)
		for (Class *iter = class->subclassList; *iter != NULL; iter++)
			objc_update_dtable(*iter);
}

static void
addSubclass(Class cls)
addSubclass(Class class)
{
	size_t i;

	if (cls->superclass->subclassList == NULL) {
		if ((cls->superclass->subclassList =
	if (class->superclass->subclassList == NULL) {
		if ((class->superclass->subclassList =
		    malloc(2 * sizeof(Class))) == NULL)
			OBJC_ERROR("Not enough memory for subclass list of "
			    "class %s!", cls->superclass->name);
			    "class %s!", class->superclass->name);

		cls->superclass->subclassList[0] = cls;
		cls->superclass->subclassList[1] = Nil;
		class->superclass->subclassList[0] = class;
		class->superclass->subclassList[1] = Nil;

		return;
	}

	for (i = 0; cls->superclass->subclassList[i] != Nil; i++);
	for (i = 0; class->superclass->subclassList[i] != Nil; i++);

	cls->superclass->subclassList =
	    realloc(cls->superclass->subclassList, (i + 2) * sizeof(Class));
	class->superclass->subclassList =
	    realloc(class->superclass->subclassList, (i + 2) * sizeof(Class));

	if (cls->superclass->subclassList == NULL)
	if (class->superclass->subclassList == NULL)
		OBJC_ERROR("Not enough memory for subclass list of class %s\n",
		    cls->superclass->name);
		    class->superclass->name);

	cls->superclass->subclassList[i] = cls;
	cls->superclass->subclassList[i + 1] = Nil;
	class->superclass->subclassList[i] = class;
	class->superclass->subclassList[i + 1] = Nil;
}


static void
updateIVarOffsets(Class cls)
updateIVarOffsets(Class class)
{
	if (!(cls->info & OBJC_CLASS_INFO_NEW_ABI))
	if (!(class->info & OBJC_CLASS_INFO_NEW_ABI))
		return;

	if (cls->instanceSize > 0)
	if (class->instanceSize > 0)
		return;

	cls->instanceSize = -cls->instanceSize;
	class->instanceSize = -class->instanceSize;

	if (cls->superclass != Nil) {
		cls->instanceSize += cls->superclass->instanceSize;
	if (class->superclass != Nil) {
		class->instanceSize += class->superclass->instanceSize;

		if (cls->iVars != NULL) {
			for (unsigned int i = 0; i < cls->iVars->count; i++) {
				cls->iVars->iVars[i].offset +=
				    cls->superclass->instanceSize;
				*cls->iVarOffsets[i] =
				    cls->iVars->iVars[i].offset;
		if (class->iVars != NULL) {
			for (unsigned int i = 0; i < class->iVars->count; i++) {
				class->iVars->iVars[i].offset +=
				    class->superclass->instanceSize;
				*class->iVarOffsets[i] =
				    class->iVars->iVars[i].offset;
			}
		}
	} else
		for (unsigned int i = 0; i < cls->iVars->count; i++)
			*cls->iVarOffsets[i] = cls->iVars->iVars[i].offset;
		for (unsigned int i = 0; i < class->iVars->count; i++)
			*class->iVarOffsets[i] = class->iVars->iVars[i].offset;
}

static void
setupClass(Class cls)
setupClass(Class class)
{
	const char *superclass;
	const char *superclassName;

	if (cls->info & OBJC_CLASS_INFO_SETUP)
	if (class->info & OBJC_CLASS_INFO_SETUP)
		return;

	if ((superclass = ((struct objc_abi_class *)cls)->superclass) != NULL) {
		Class super = objc_classname_to_class(superclass, false);
	superclassName = ((struct objc_abi_class *)class)->superclass;
	if (superclassName != NULL) {
		Class super = objc_classname_to_class(superclassName, false);

		if (super == Nil)
			return;

		setupClass(super);

		if (!(super->info & OBJC_CLASS_INFO_SETUP))
			return;

		cls->superclass = super;
		cls->isa->superclass = super->isa;
		class->superclass = super;
		class->isa->superclass = super->isa;

		addSubclass(cls);
		addSubclass(cls->isa);
		addSubclass(class);
		addSubclass(class->isa);
	} else
		cls->isa->superclass = cls;
		class->isa->superclass = class;

	updateIVarOffsets(cls);
	updateIVarOffsets(class);

	cls->info |= OBJC_CLASS_INFO_SETUP;
	cls->isa->info |= OBJC_CLASS_INFO_SETUP;
	class->info |= OBJC_CLASS_INFO_SETUP;
	class->isa->info |= OBJC_CLASS_INFO_SETUP;
}

static void
initializeClass(Class cls)
initializeClass(Class class)
{
	if (cls->info & OBJC_CLASS_INFO_INITIALIZED)
	if (class->info & OBJC_CLASS_INFO_INITIALIZED)
		return;

	if (cls->superclass)
		initializeClass(cls->superclass);
	if (class->superclass)
		initializeClass(class->superclass);

	cls->info |= OBJC_CLASS_INFO_DTABLE;
	cls->isa->info |= OBJC_CLASS_INFO_DTABLE;
	class->info |= OBJC_CLASS_INFO_DTABLE;
	class->isa->info |= OBJC_CLASS_INFO_DTABLE;

	objc_update_dtable(cls);
	objc_update_dtable(cls->isa);
	objc_update_dtable(class);
	objc_update_dtable(class->isa);

	/*
	 * Set it first to prevent calling it recursively due to message sends
	 * in the initialize method
	 */
	cls->info |= OBJC_CLASS_INFO_INITIALIZED;
	cls->isa->info |= OBJC_CLASS_INFO_INITIALIZED;
	class->info |= OBJC_CLASS_INFO_INITIALIZED;
	class->isa->info |= OBJC_CLASS_INFO_INITIALIZED;

	callMethod(cls, "initialize");
	callMethod(class, "initialize");
}

void
objc_initialize_class(Class cls)
objc_initialize_class(Class class)
{
	if (cls->info & OBJC_CLASS_INFO_INITIALIZED)
	if (class->info & OBJC_CLASS_INFO_INITIALIZED)
		return;

	objc_global_mutex_lock();

	/*
	 * It's possible that two threads try to initialize a class at the same
	 * time. Make sure that the thread which held the lock did not already
	 * initialize it.
	 */
	if (cls->info & OBJC_CLASS_INFO_INITIALIZED) {
	if (class->info & OBJC_CLASS_INFO_INITIALIZED) {
		objc_global_mutex_unlock();
		return;
	}

	setupClass(cls);
	setupClass(class);

	if (!(cls->info & OBJC_CLASS_INFO_SETUP)) {
	if (!(class->info & OBJC_CLASS_INFO_SETUP)) {
		objc_global_mutex_unlock();
		return;
	}

	initializeClass(cls);
	initializeClass(class);

	objc_global_mutex_unlock();
}

static void
processLoadQueue()
{
394
395
396
397
398
399
400
401

402
403
404
405
406



407
408
409
410

411
412
413


414
415
416


417
418
419
420
421
422
423
424
425

426
427
428

429
430
431
432
433
434
435
436
437

438
439
440
441
442
443
444


445
446
447
448
449
450
451
452





453
454
455
456
457
458

459
460
461
462
463
464
465

466
467
468
469

470
471
472
473

474
475
476
477



478
479
480
481


482
483
484


485
486

487
488
489
490
491
492
493
494
495
496

497
498

499
500
501
502


503
504
505
506

507
508
509
510

511
512
513

514
515
516
517
518
519
520
521
522
523
524
525

526
527

528
529
530

531
532
533
534
535
536
537
396
397
398
399
400
401
402

403
404
405



406
407
408
409
410
411

412
413


414
415
416


417
418
419
420
421
422
423
424
425
426

427
428
429

430
431
432
433
434
435
436
437
438

439
440
441
442
443
444


445
446
447
448
449





450
451
452
453
454
455
456
457
458
459

460
461
462
463
464
465
466

467
468
469
470

471
472
473
474

475
476



477
478
479
480
481


482
483
484


485
486
487

488
489
490
491
492
493
494
495
496
497

498
499

500
501
502


503
504
505
506
507

508
509
510
511

512
513
514

515
516
517
518
519
520
521
522
523
524
525
526

527
528

529
530
531

532
533
534
535
536
537
538
539







-
+


-
-
-
+
+
+



-
+

-
-
+
+

-
-
+
+








-
+


-
+








-
+





-
-
+
+



-
-
-
-
-
+
+
+
+
+





-
+






-
+



-
+



-
+

-
-
-
+
+
+


-
-
+
+

-
-
+
+

-
+









-
+

-
+


-
-
+
+



-
+



-
+


-
+











-
+

-
+


-
+







	}
}

void
objc_register_all_classes(struct objc_abi_symtab *symtab)
{
	for (uint16_t i = 0; i < symtab->classDefsCount; i++) {
		struct objc_abi_class *cls =
		struct objc_abi_class *rawClass =
		    (struct objc_abi_class *)symtab->defs[i];

		registerClass(cls);
		registerSelectors(cls);
		registerSelectors(cls->metaclass);
		registerClass(rawClass);
		registerSelectors(rawClass);
		registerSelectors(rawClass->metaclass);
	}

	for (uint16_t i = 0; i < symtab->classDefsCount; i++) {
		Class cls = (Class)symtab->defs[i];
		Class class = (Class)symtab->defs[i];

		if (hasLoad(cls)) {
			setupClass(cls);
		if (hasLoad(class)) {
			setupClass(class);

			if (cls->info & OBJC_CLASS_INFO_SETUP)
				callLoad(cls);
			if (class->info & OBJC_CLASS_INFO_SETUP)
				callLoad(class);
			else {
				loadQueue = realloc(loadQueue,
				    sizeof(Class) * (loadQueueCount + 1));

				if (loadQueue == NULL)
					OBJC_ERROR("Not enough memory for load "
					    "queue!");

				loadQueue[loadQueueCount++] = cls;
				loadQueue[loadQueueCount++] = class;
			}
		} else
			cls->info |= OBJC_CLASS_INFO_LOADED;
			class->info |= OBJC_CLASS_INFO_LOADED;
	}

	processLoadQueue();
}

Class
objc_allocateClassPair(Class superclass, const char *name, size_t extraBytes)
{
	struct objc_class *cls, *metaclass;
	struct objc_class *class, *metaclass;
	Class iter, rootclass = Nil;

	if (extraBytes > LONG_MAX)
		OBJC_ERROR("extra_bytes out of range!")

	if ((cls = calloc(1, sizeof(*cls))) == NULL ||
	    (metaclass = calloc(1, sizeof(*cls))) == NULL)
	if ((class = calloc(1, sizeof(*class))) == NULL ||
	    (metaclass = calloc(1, sizeof(*class))) == NULL)
		OBJC_ERROR("Not enough memory to allocate class pair for class "
		    "%s!", name)

	cls->isa = metaclass;
	cls->superclass = superclass;
	cls->name = name;
	cls->info = OBJC_CLASS_INFO_CLASS;
	cls->instanceSize = (superclass != Nil ?
	class->isa = metaclass;
	class->superclass = superclass;
	class->name = name;
	class->info = OBJC_CLASS_INFO_CLASS;
	class->instanceSize = (superclass != Nil ?
	    superclass->instanceSize : 0) + (long)extraBytes;

	for (iter = superclass; iter != Nil; iter = iter->superclass)
		rootclass = iter;

	metaclass->isa = (rootclass != Nil ? rootclass->isa : cls);
	metaclass->isa = (rootclass != Nil ? rootclass->isa : class);
	metaclass->superclass = (superclass != Nil ? superclass->isa : Nil);
	metaclass->name = name;
	metaclass->info = OBJC_CLASS_INFO_CLASS;
	metaclass->instanceSize = (superclass != Nil ?
	    superclass->isa->instanceSize : 0) + (long)extraBytes;

	return cls;
	return class;
}

void
objc_registerClassPair(Class cls)
objc_registerClassPair(Class class)
{
	objc_global_mutex_lock();

	registerClass((struct objc_abi_class *)cls);
	registerClass((struct objc_abi_class *)class);

	if (cls->superclass != Nil) {
		addSubclass(cls);
		addSubclass(cls->isa);
	if (class->superclass != Nil) {
		addSubclass(class);
		addSubclass(class->isa);
	}

	cls->info |= OBJC_CLASS_INFO_SETUP;
	cls->isa->info |= OBJC_CLASS_INFO_SETUP;
	class->info |= OBJC_CLASS_INFO_SETUP;
	class->isa->info |= OBJC_CLASS_INFO_SETUP;

	if (hasLoad(cls))
		callLoad(cls);
	if (hasLoad(class))
		callLoad(class);
	else
		cls->info |= OBJC_CLASS_INFO_LOADED;
		class->info |= OBJC_CLASS_INFO_LOADED;

	processLoadQueue();

	objc_global_mutex_unlock();
}

Class
objc_lookUpClass(const char *name)
{
	Class cls;
	Class class;

	if ((cls = objc_classname_to_class(name, true)) == NULL)
	if ((class = objc_classname_to_class(name, true)) == NULL)
		return Nil;

	if (cls->info & OBJC_CLASS_INFO_SETUP)
		return cls;
	if (class->info & OBJC_CLASS_INFO_SETUP)
		return class;

	objc_global_mutex_lock();

	setupClass(cls);
	setupClass(class);

	objc_global_mutex_unlock();

	if (!(cls->info & OBJC_CLASS_INFO_SETUP))
	if (!(class->info & OBJC_CLASS_INFO_SETUP))
		return Nil;

	return cls;
	return class;
}

Class
objc_getClass(const char *name)
{
	return objc_lookUpClass(name);
}

Class
objc_getRequiredClass(const char *name)
{
	Class cls;
	Class class;

	if ((cls = objc_getClass(name)) == Nil)
	if ((class = objc_getClass(name)) == Nil)
		OBJC_ERROR("Class %s not found!", name);

	return cls;
	return class;
}

Class
objc_lookup_class(const char *name)
{
	return objc_getClass(name);
}
552
553
554
555
556
557
558
559

560
561
562
563
564
565
566
567
568
569
570
571
572

573
574

575
576
577

578
579
580
581
582
583
584
585
586

587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602


603
604
605
606
607
608
609
610

611
612

613
614
615

616
617
618
619

620
621

622
623
624

625
626
627
628

629
630

631
632
633

634
635
636
637

638
639

640
641
642

643
644
645
646

647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662

663
664
665

666
667
668
669
670

671
672
673
674
675
676
677
678
679
680

681
682
683

684
685
686
687
688

689
690
691
692

693
694
695

696
697
698
699
700
701
702
703
704
705
706
707
708
709
710

711
712
713
714
715
716
717
718
719
720
721


722
723
724
725
726
727
728
729

730
731
732
733
734
735

736
737

738
739
740
741

742
743
744
745

746
747
748
749
750

751
752
753
754
755
756
757
758
759


760
761
762
763
764
765

766
767
768
769
770
771
772
773


774
775
776
777
778
779
780
781
782
783
784

785
786
787
788
789
790
791
792

793
794
795

796
797
798

799
800
801
802
803
804
805
554
555
556
557
558
559
560

561
562
563
564
565
566
567
568
569
570
571
572
573

574
575

576
577
578

579
580
581
582
583
584
585
586
587

588
589
590
591
592
593
594
595
596
597
598
599
600
601
602


603
604
605
606
607
608
609
610
611

612
613

614
615
616

617
618
619
620

621
622

623
624
625

626
627
628
629

630
631

632
633
634

635
636
637
638

639
640

641
642
643

644
645
646
647

648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663

664
665
666

667
668
669
670
671

672
673
674
675
676
677
678
679
680
681

682
683
684

685
686
687
688
689

690
691
692
693

694
695
696

697
698
699
700
701
702
703
704
705
706
707
708
709
710
711

712
713
714
715
716
717
718
719
720
721
722

723
724
725
726
727
728
729
730
731

732
733
734
735
736
737

738
739

740
741
742
743

744
745
746
747

748
749
750
751
752

753
754
755
756
757
758
759
760


761
762
763
764
765
766
767

768
769
770
771
772
773
774


775
776
777
778
779
780
781
782
783
784
785
786

787
788
789
790
791
792
793
794

795
796
797

798
799
800

801
802
803
804
805
806
807
808







-
+












-
+

-
+


-
+








-
+














-
-
+
+







-
+

-
+


-
+



-
+

-
+


-
+



-
+

-
+


-
+



-
+

-
+


-
+



-
+















-
+


-
+




-
+









-
+


-
+




-
+



-
+


-
+














-
+










-
+
+







-
+





-
+

-
+



-
+



-
+




-
+







-
-
+
+





-
+






-
-
+
+










-
+







-
+


-
+


-
+







		return classesCount;

	if (classesCount < count)
		count = classesCount;

	j = 0;
	for (uint32_t i = 0; i < classes->size; i++) {
		void *cls;
		void *class;

		if (j >= count) {
			objc_global_mutex_unlock();
			return j;
		}

		if (classes->data[i] == NULL)
			continue;

		if (strcmp(classes->data[i]->key, "Protocol") == 0)
			continue;

		cls = (Class)classes->data[i]->object;
		class = (Class)classes->data[i]->object;

		if (cls == Nil || (uintptr_t)cls & 1)
		if (class == Nil || (uintptr_t)class & 1)
			continue;

		buffer[j++] = cls;
		buffer[j++] = class;
	}

	objc_global_mutex_unlock();

	return j;
}

Class *
objc_copyClassList(unsigned int *len)
objc_copyClassList(unsigned int *length)
{
	Class *ret;
	unsigned int count;

	objc_global_mutex_lock();

	if ((ret = malloc((classesCount + 1) * sizeof(Class))) == NULL)
		OBJC_ERROR("Failed to allocate memory for class list!");

	count = objc_getClassList(ret, classesCount);
	OF_ENSURE(count == classesCount);

	ret[count] = Nil;

	if (len != NULL)
		*len = count;
	if (length != NULL)
		*length = count;

	objc_global_mutex_unlock();

	return ret;
}

bool
class_isMetaClass(Class cls)
class_isMetaClass(Class class)
{
	if (cls == Nil)
	if (class == Nil)
		return false;

	return (cls->info & OBJC_CLASS_INFO_METACLASS);
	return (class->info & OBJC_CLASS_INFO_METACLASS);
}

const char *
class_getName(Class cls)
class_getName(Class class)
{
	if (cls == Nil)
	if (class == Nil)
		return "";

	return cls->name;
	return class->name;
}

Class
class_getSuperclass(Class cls)
class_getSuperclass(Class class)
{
	if (cls == Nil)
	if (class == Nil)
		return Nil;

	return cls->superclass;
	return class->superclass;
}

unsigned long
class_getInstanceSize(Class cls)
class_getInstanceSize(Class class)
{
	if (cls == Nil)
	if (class == Nil)
		return 0;

	return cls->instanceSize;
	return class->instanceSize;
}

IMP
class_getMethodImplementation(Class cls, SEL selector)
class_getMethodImplementation(Class class, SEL selector)
{
	/*
	 * We use a dummy object here so that the normal lookup is used, even
	 * though we don't have an object. Doing so is safe, as objc_msg_lookup
	 * does not access the object, but only its class.
	 *
	 * Just looking it up in the dispatch table could result in returning
	 * NULL instead of the forwarding handler, it would also mean
	 * +[resolveClassMethod:] / +[resolveInstanceMethod:] would not be
	 * called.
	 */
	struct {
		Class isa;
	} dummy;

	if (cls == Nil)
	if (class == Nil)
		return NULL;

	dummy.isa = cls;
	dummy.isa = class;
	return objc_msg_lookup((id)&dummy, selector);
}

IMP
class_getMethodImplementation_stret(Class cls, SEL selector)
class_getMethodImplementation_stret(Class class, SEL selector)
{
	/*
	 * Same as above, but use objc_msg_lookup_stret instead, so that the
	 * correct forwarding handler is returned.
	 */
	struct {
		Class isa;
	} dummy;

	if (cls == Nil)
	if (class == Nil)
		return NULL;

	dummy.isa = cls;
	dummy.isa = class;
	return objc_msg_lookup_stret((id)&dummy, selector);
}

static struct objc_method *
getMethod(Class cls, SEL selector)
getMethod(Class class, SEL selector)
{
	struct objc_category **categories;

	if ((categories = objc_categories_for_class(cls)) != NULL) {
	if ((categories = objc_categories_for_class(class)) != NULL) {
		for (; *categories != NULL; categories++) {
			struct objc_method_list *methodList =
			    (cls->info & OBJC_CLASS_INFO_METACLASS
			    (class->info & OBJC_CLASS_INFO_METACLASS
			    ? (*categories)->classMethods
			    : (*categories)->instanceMethods);

			for (; methodList != NULL;
			    methodList = methodList->next)
				for (unsigned int i = 0;
				    i < methodList->count; i++)
					if (sel_isEqual((SEL)
					    &methodList->methods[i].selector,
					    selector))
						return &methodList->methods[i];
		}
	}

	for (struct objc_method_list *methodList = cls->methodList;
	for (struct objc_method_list *methodList = class->methodList;
	    methodList != NULL; methodList = methodList->next)
		for (unsigned int i = 0; i < methodList->count; i++)
			if (sel_isEqual((SEL)&methodList->methods[i].selector,
			    selector))
				return &methodList->methods[i];

	return NULL;
}

static void
addMethod(Class cls, SEL selector, IMP implementation, const char *typeEncoding)
addMethod(Class class, SEL selector, IMP implementation,
    const char *typeEncoding)
{
	struct objc_method_list *methodList;

	/* FIXME: We need a way to free this at objc_exit() */
	if ((methodList = malloc(sizeof(struct objc_method_list))) == NULL)
		OBJC_ERROR("Not enough memory to replace method!");

	methodList->next = cls->methodList;
	methodList->next = class->methodList;
	methodList->count = 1;
	methodList->methods[0].selector.UID = selector->UID;
	methodList->methods[0].selector.typeEncoding = typeEncoding;
	methodList->methods[0].implementation = implementation;

	cls->methodList = methodList;
	class->methodList = methodList;

	objc_update_dtable(cls);
	objc_update_dtable(class);
}

const char *
class_getMethodTypeEncoding(Class cls, SEL selector)
class_getMethodTypeEncoding(Class class, SEL selector)
{
	struct objc_method *method;

	if (cls == Nil)
	if (class == Nil)
		return NULL;

	objc_global_mutex_lock();

	if ((method = getMethod(cls, selector)) != NULL) {
	if ((method = getMethod(class, selector)) != NULL) {
		const char *ret = method->selector.typeEncoding;
		objc_global_mutex_unlock();
		return ret;
	}

	objc_global_mutex_unlock();

	if (cls->superclass != Nil)
		return class_getMethodTypeEncoding(cls->superclass, selector);
	if (class->superclass != Nil)
		return class_getMethodTypeEncoding(class->superclass, selector);

	return NULL;
}

bool
class_addMethod(Class cls, SEL selector, IMP implementation,
class_addMethod(Class class, SEL selector, IMP implementation,
    const char *typeEncoding)
{
	bool ret;

	objc_global_mutex_lock();

	if (getMethod(cls, selector) == NULL) {
		addMethod(cls, selector, implementation, typeEncoding);
	if (getMethod(class, selector) == NULL) {
		addMethod(class, selector, implementation, typeEncoding);
		ret = true;
	} else
		ret = false;

	objc_global_mutex_unlock();

	return ret;
}

IMP
class_replaceMethod(Class cls, SEL selector, IMP implementation,
class_replaceMethod(Class class, SEL selector, IMP implementation,
    const char *typeEncoding)
{
	struct objc_method *method;
	IMP oldImplementation;

	objc_global_mutex_lock();

	if ((method = getMethod(cls, selector)) != NULL) {
	if ((method = getMethod(class, selector)) != NULL) {
		oldImplementation = method->implementation;
		method->implementation = implementation;
		objc_update_dtable(cls);
		objc_update_dtable(class);
	} else {
		oldImplementation = NULL;
		addMethod(cls, selector, implementation, typeEncoding);
		addMethod(class, selector, implementation, typeEncoding);
	}

	objc_global_mutex_unlock();

	return oldImplementation;
}

813
814
815
816
817
818
819
820

821
822
823
824
825
826
827
828
829
830
831

832
833
834
835
836
837

838
839

840
841
842
843

844
845

846
847
848


849
850
851
852

853
854

855
856
857
858
859
860
861

862
863
864
865

866
867

868
869
870
871
872
873



874
875
876
877


878
879

880
881
882


883
884

885
886
887
888

889
890
891


892
893
894


895
896

897
898

899
900
901
902


903
904
905
906
907
908
909
910
911
912
913
914

915
916

917
918
919

920
921
922
923
924
925
926
816
817
818
819
820
821
822

823
824
825
826
827
828
829
830
831
832
833

834
835
836
837
838
839

840
841

842
843
844
845

846
847

848
849


850
851
852
853
854

855
856

857
858
859
860
861
862
863

864
865
866
867

868
869

870
871
872
873



874
875
876
877
878


879
880
881

882
883


884
885
886

887
888
889
890

891
892


893
894
895


896
897
898

899
900

901
902
903


904
905
906
907
908
909
910
911
912
913
914
915
916

917
918

919
920
921

922
923
924
925
926
927
928
929







-
+










-
+





-
+

-
+



-
+

-
+

-
-
+
+



-
+

-
+






-
+



-
+

-
+



-
-
-
+
+
+


-
-
+
+

-
+

-
-
+
+

-
+



-
+

-
-
+
+

-
-
+
+

-
+

-
+


-
-
+
+











-
+

-
+


-
+








	object = (struct objc_object *)object_;

	return object->isa;
}

Class
object_setClass(id object_, Class cls)
object_setClass(id object_, Class class)
{
	struct objc_object *object;
	Class old;

	if (object_ == nil)
		return Nil;

	object = (struct objc_object *)object_;

	old = object->isa;
	object->isa = cls;
	object->isa = class;

	return old;
}

const char *
object_getClassName(id obj)
object_getClassName(id object)
{
	return class_getName(object_getClass(obj));
	return class_getName(object_getClass(object));
}

static void
unregisterClass(Class rcls)
unregisterClass(Class class)
{
	struct objc_abi_class *cls = (struct objc_abi_class *)rcls;
	struct objc_abi_class *rawClass = (struct objc_abi_class *)class;

	if ((rcls->info & OBJC_CLASS_INFO_SETUP) && rcls->superclass != Nil &&
	    rcls->superclass->subclassList != NULL) {
	if ((class->info & OBJC_CLASS_INFO_SETUP) && class->superclass != Nil &&
	    class->superclass->subclassList != NULL) {
		size_t i = SIZE_MAX, count = 0;
		Class *tmp;

		for (tmp = rcls->superclass->subclassList;
		for (tmp = class->superclass->subclassList;
		    *tmp != Nil; tmp++) {
			if (*tmp == rcls)
			if (*tmp == class)
				i = count;

			count++;
		}

		if (count > 0 && i < SIZE_MAX) {
			tmp = rcls->superclass->subclassList;
			tmp = class->superclass->subclassList;
			tmp[i] = tmp[count - 1];
			tmp[count - 1] = NULL;

			if ((tmp = realloc(rcls->superclass->subclassList,
			if ((tmp = realloc(class->superclass->subclassList,
			    count * sizeof(Class))) != NULL)
				rcls->superclass->subclassList = tmp;
				class->superclass->subclassList = tmp;
		}
	}

	if (rcls->subclassList != NULL) {
		free(rcls->subclassList);
		rcls->subclassList = NULL;
	if (class->subclassList != NULL) {
		free(class->subclassList);
		class->subclassList = NULL;
	}

	if (rcls->DTable != NULL && rcls->DTable != emptyDTable)
		objc_dtable_free(rcls->DTable);
	if (class->DTable != NULL && class->DTable != emptyDTable)
		objc_dtable_free(class->DTable);

	rcls->DTable = NULL;
	class->DTable = NULL;

	if ((rcls->info & OBJC_CLASS_INFO_SETUP) && rcls->superclass != Nil)
		cls->superclass = rcls->superclass->name;
	if ((class->info & OBJC_CLASS_INFO_SETUP) && class->superclass != Nil)
		rawClass->superclass = class->superclass->name;

	rcls->info &= ~OBJC_CLASS_INFO_SETUP;
	class->info &= ~OBJC_CLASS_INFO_SETUP;
}

void
objc_unregister_class(Class cls)
objc_unregister_class(Class class)
{
	while (cls->subclassList != NULL && cls->subclassList[0] != Nil)
		objc_unregister_class(cls->subclassList[0]);
	while (class->subclassList != NULL && class->subclassList[0] != Nil)
		objc_unregister_class(class->subclassList[0]);

	if (cls->info & OBJC_CLASS_INFO_LOADED)
		callMethod(cls, "unload");
	if (class->info & OBJC_CLASS_INFO_LOADED)
		callMethod(class, "unload");

	objc_hashtable_delete(classes, cls->name);
	objc_hashtable_delete(classes, class->name);

	if (strcmp(class_getName(cls), "Protocol") != 0)
	if (strcmp(class_getName(class), "Protocol") != 0)
		classesCount--;

	unregisterClass(cls);
	unregisterClass(cls->isa);
	unregisterClass(class);
	unregisterClass(class->isa);
}

void
objc_unregister_all_classes(void)
{
	if (classes == NULL)
		return;

	for (uint32_t i = 0; i < classes->size; i++) {
		if (classes->data[i] != NULL &&
		    classes->data[i] != &objc_deleted_bucket) {
			void *cls = (Class)classes->data[i]->object;
			void *class = (Class)classes->data[i]->object;

			if (cls == Nil || (uintptr_t)cls & 1)
			if (class == Nil || (uintptr_t)class & 1)
				continue;

			objc_unregister_class(cls);
			objc_unregister_class(class);

			/*
			 * The table might have been resized, so go back to the
			 * start again.
			 *
			 * Due to the i++ in the for loop, we need to set it to
			 * UINT32_MAX so that it will get increased at the end