You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
168 lines
4.7 KiB
Objective-C
168 lines
4.7 KiB
Objective-C
#include "Test.h"
|
|
|
|
void direct_saturation_test();
|
|
|
|
@interface TestWithDelloc : Test
|
|
@end
|
|
|
|
@implementation TestWithDelloc
|
|
- (void)dealloc
|
|
{
|
|
id obj = nil;
|
|
objc_storeStrong(&obj, self);
|
|
assert(obj == self);
|
|
assert(object_getRetainCount_np(obj) == 0);
|
|
[super dealloc];
|
|
}
|
|
@end
|
|
|
|
int main()
|
|
{
|
|
id obj = [Test new];
|
|
assert(object_getRetainCount_np(obj) == 1);
|
|
|
|
for (int i = 0; i < 2; i++)
|
|
{
|
|
size_t count = object_getRetainCount_np(obj);
|
|
id ret = objc_retain_fast_np(obj);
|
|
assert(ret == obj);
|
|
assert(object_getRetainCount_np(obj) == ++count);
|
|
}
|
|
|
|
for (int i = 0; i < 2; i++)
|
|
{
|
|
size_t count = object_getRetainCount_np(obj);
|
|
BOOL destroy = objc_release_fast_no_destroy_np(obj);
|
|
assert(destroy == NO);
|
|
assert(object_getRetainCount_np(obj) == --count);
|
|
}
|
|
|
|
{
|
|
// Final release should prevent further retains and releases.
|
|
assert(objc_release_fast_no_destroy_np(obj) == YES);
|
|
assert(object_getRetainCount_np(obj) == 0);
|
|
assert(objc_retain_fast_np(obj) == obj);
|
|
assert(object_getRetainCount_np(obj) == 0);
|
|
assert(objc_release_fast_no_destroy_np(obj) == NO);
|
|
assert(object_getRetainCount_np(obj) == 0);
|
|
}
|
|
|
|
object_dispose(obj);
|
|
obj = [Test new];
|
|
|
|
{
|
|
// Should not be able to delete weak refs until final release.
|
|
id weak;
|
|
assert(objc_initWeak(&weak, obj) == obj);
|
|
assert(weak != nil);
|
|
assert(objc_loadWeakRetained(&weak) == obj);
|
|
assert(objc_release_fast_no_destroy_np(obj) == NO);
|
|
// Assumes a return of NO means no effect on obj at all.
|
|
assert(objc_delete_weak_refs(obj) == NO);
|
|
assert(objc_loadWeakRetained(&weak) == obj);
|
|
assert(objc_release_fast_no_destroy_np(obj) == NO);
|
|
// This will also call objc_delete_weak_refs() and succeed.
|
|
assert(objc_release_fast_no_destroy_np(obj) == YES);
|
|
objc_destroyWeak(&weak);
|
|
// Check what happens when the weak refs were already deleted.
|
|
assert(objc_delete_weak_refs(obj) == YES);
|
|
}
|
|
|
|
object_dispose(obj);
|
|
// Check we can use strong references inside a dealloc method.
|
|
obj = [TestWithDelloc new];
|
|
[obj release];
|
|
obj = nil;
|
|
|
|
direct_saturation_test();
|
|
return 0;
|
|
}
|
|
|
|
|
|
// ----------------
|
|
// This test has knowledge of the implementation details of the ARC
|
|
// reference counting and may need modification if the details change.
|
|
|
|
const long refcount_shift = 1;
|
|
const size_t weak_mask = ((size_t)1)<<((sizeof(size_t)*8)-refcount_shift);
|
|
const size_t refcount_mask = ~weak_mask;
|
|
const size_t refcount_max = refcount_mask - 1;
|
|
|
|
size_t get_refcount(id obj)
|
|
{
|
|
size_t *refCount = ((size_t*)obj) - 1;
|
|
return *refCount & refcount_mask;
|
|
}
|
|
|
|
void set_refcount(id obj, size_t count)
|
|
{
|
|
size_t *refCount = ((size_t*)obj) - 1;
|
|
*refCount = (*refCount & weak_mask) | (count & refcount_mask);
|
|
}
|
|
|
|
void direct_saturation_test()
|
|
{
|
|
{
|
|
id obj = [Test new];
|
|
// sanity check
|
|
objc_retain_fast_np(obj);
|
|
assert(object_getRetainCount_np(obj) == 2);
|
|
assert(get_refcount(obj) == 1);
|
|
|
|
// Check the behaviour close to the maximum refcount.
|
|
set_refcount(obj, refcount_max - 3);
|
|
assert(object_getRetainCount_np(obj) == refcount_max - 2);
|
|
|
|
assert(objc_retain_fast_np(obj) == obj);
|
|
assert(object_getRetainCount_np(obj) == refcount_max - 1);
|
|
|
|
id weak;
|
|
assert(objc_initWeak(&weak, obj) == obj);
|
|
assert(weak != nil);
|
|
assert(objc_loadWeakRetained(&weak) == obj);
|
|
assert(object_getRetainCount_np(obj) == refcount_max);
|
|
|
|
// This retain should cause the count to saturate.
|
|
assert(objc_retain_fast_np(obj) == obj);
|
|
assert(object_getRetainCount_np(obj) == refcount_max + 1);
|
|
|
|
// A saturated count is no longer affected by retains or releases.
|
|
assert(objc_release_fast_no_destroy_np(obj) == NO);
|
|
assert(object_getRetainCount_np(obj) == refcount_max + 1);
|
|
assert(objc_retain_fast_np(obj) == obj);
|
|
assert(object_getRetainCount_np(obj) == refcount_max + 1);
|
|
|
|
// Nor should any weak refs be deleted.
|
|
assert(objc_delete_weak_refs(obj) == NO);
|
|
assert(objc_loadWeakRetained(&weak) == obj);
|
|
assert(object_getRetainCount_np(obj) == refcount_max + 1);
|
|
|
|
// Cleanup (can skip this if it becomes an issue)
|
|
objc_destroyWeak(&weak);
|
|
set_refcount(obj, 0);
|
|
objc_release_fast_no_destroy_np(obj);
|
|
object_dispose(obj);
|
|
}
|
|
|
|
{
|
|
id obj = [Test new];
|
|
set_refcount(obj, refcount_max - 2);
|
|
assert(objc_retain_fast_np(obj) == obj);
|
|
assert(objc_retain_fast_np(obj) == obj);
|
|
assert(object_getRetainCount_np(obj) == refcount_max + 1);
|
|
|
|
// Check we can init a weak ref to an object with a saturated count.
|
|
id weak;
|
|
assert(objc_initWeak(&weak, obj) == obj);
|
|
assert(weak != nil);
|
|
assert(objc_loadWeakRetained(&weak) == obj);
|
|
assert(object_getRetainCount_np(obj) == refcount_max + 1);
|
|
|
|
// Cleanup (can skip this if it becomes an issue)
|
|
objc_destroyWeak(&weak);
|
|
set_refcount(obj, 0);
|
|
objc_release_fast_no_destroy_np(obj);
|
|
object_dispose(obj);
|
|
}
|
|
}
|