weakrace.m   [plain text]


// TEST_CONFIG MEM=mrc

#include "test.h"
#include <objc/NSObject.h>

static semaphore_t go1;
static semaphore_t go2;
static semaphore_t done;

#define VARCOUNT 100000
static id obj;
static id vars[VARCOUNT];


void *destroyer(void *arg __unused)
{
    while (1) {
        semaphore_wait(go1);
        for (int i = 0; i < VARCOUNT; i++) {
            objc_destroyWeak(&vars[i]);
        }
        semaphore_signal(done);
    }
}


void *deallocator(void *arg __unused)
{
    while (1) {
        semaphore_wait(go2);
        [obj release];
        semaphore_signal(done);
    }
}


void cycle(void)
{
    // rdar://12896779 objc_destroyWeak() versus weak clear in dealloc

    // Clean up from previous cycle - objc_destroyWeak() doesn't set var to nil
    for (int i = 0; i < VARCOUNT; i++) {
        vars[i] = nil;
    }

    obj = [NSObject new];
    for (int i = 0; i < VARCOUNT; i++) {
        objc_storeWeak(&vars[i], obj);
    }

    // let destroyer start before deallocator runs
    semaphore_signal(go1);
    sched_yield();
    semaphore_signal(go2);
    
    semaphore_wait(done);
    semaphore_wait(done);
}


int main()
{
    semaphore_create(mach_task_self(), &go1, 0, 0);
    semaphore_create(mach_task_self(), &go2, 0, 0);
    semaphore_create(mach_task_self(), &done, 0, 0);

    pthread_t th[2];
    pthread_create(&th[1], NULL, deallocator, NULL);
    pthread_create(&th[1], NULL, destroyer, NULL);

    for (int i = 0; i < 100; i++) {
        cycle();
    }

    succeed(__FILE__);
}