ftp.nice.ch/Attic/openStep/implementation/gnustep/sources/libFoundation.0.7.tgz#/libFoundation-0.7/FoundationTestsuite/tests/basic/multithreading.m

This is multithreading.m in view mode; [Download] [Up]

/* 
   multithreading.m

   Copyright (C) 1995, 1996, 1997 Ovidiu Predescu and Mircea Oancea.
   All rights reserved.

   Author: Mircea Oancea <mircea@jupiter.elcom.pub.ro>

   This file is part of libFoundation.

   Permission to use, copy, modify, and distribute this software and its
   documentation for any purpose and without fee is hereby granted, provided
   that the above copyright notice appear in all copies and that both that
   copyright notice and this permission notice appear in supporting
   documentation.

   We disclaim all warranties with regard to this software, including all
   implied warranties of merchantability and fitness, in no event shall
   we be liable for any special, indirect or consequential damages or any
   damages whatsoever resulting from loss of use, data or profits, whether in
   an action of contract, negligence or other tortious action, arising out of
   or in connection with the use or performance of this software.
*/

#include <Foundation/NSObject.h>
#include <extensions/objc-runtime.h>

/*
 * Lock for printf
 */
 
static objc_mutex_t io_lock = NULL;
static int verbose = 0;

/*
 *   Condition lock
 *   The condition is the bitwise and of user value and condition's predicate
 */

@interface BitmapConditionLock : NSObject
{
    objc_mutex_t     mutex;
    objc_condition_t condition;
    volatile int      predicate;
}
- (id)initWithCondition:(int)value;
- (int)condition;
- (void)lockWhenCondition:(int)value;
- (void)unlockWithCondition:(int)value;
@end;

@implementation BitmapConditionLock

- (id)init
{
    [super init];
    mutex = objc_mutex_allocate();
    condition = objc_condition_allocate();
    predicate = 0;
    return self;
}

- (id)initWithCondition:(int)value
{
    [self init];
    predicate = value;
    return self;
}

- (void)dealloc
{
    objc_mutex_deallocate(mutex);
    objc_condition_deallocate(condition);
    [super dealloc];
}

- (int)condition
{
    return predicate;
}

- (void)lockWhenCondition:(int)value
{
    int depth;
    
    // Try to lock the mutex
    depth = objc_mutex_lock(mutex);

    // Return on error
    if (depth == -1)
	return;
	    
    // Error if recursive lock and condition is false 
    if ((depth > 1) && (predicate & value))
	return;

    // Wait condition
    while (!(predicate & value))
	objc_condition_wait(condition, mutex);
}

- (void)unlockWithCondition:(int)value
{
    int depth;
    
    // Try to lock the mutex again
    depth = objc_mutex_trylock(mutex);
    
    // Another thread has the lock so abort
    if (depth == -1)
	return;
    
    // If the depth is only 1 then we just acquired
    // the lock above, bogus unlock so abort
    if (depth == 1)
	return;
    
    // This is a valid unlock so set the condition
    // and unlock twice
    predicate = value;
    objc_condition_broadcast(condition);
    objc_mutex_unlock(mutex);
    objc_mutex_unlock(mutex);
}

@end

/*
*   Store Object
*/

@interface Store : NSObject
{
    BitmapConditionLock* lock;
    unsigned capacity;
    volatile int count;
}
- initWithCapacity:(unsigned)aCapacity;
- (int)addObject;
- (int)removeObject;
- (int)count;
@end;

typedef enum {
    StoreEmpty    = 1,
    StoreHasData  = 2,
    StoreFull     = 4
} StoreState;

@implementation Store

- initWithCapacity:(unsigned)aCapacity
{
    [super init];
    lock = [[BitmapConditionLock alloc] initWithCondition:StoreEmpty];
    capacity = aCapacity;
    count = 0;
    return self;
}

- (void)dealloc
{
    [lock release];
    [super dealloc];
}

- (int)addObject
{
    [lock lockWhenCondition:StoreEmpty|StoreHasData];
    count++;
    [lock unlockWithCondition:count<capacity ? StoreHasData : StoreFull];
    return count;
}

- (int)removeObject
{
    [lock lockWhenCondition:StoreFull|StoreHasData];
    count--;
    [lock unlockWithCondition:count ? StoreHasData : StoreEmpty];
    return count;
}

- (int)count
{
    return count;
}

@end;

/*
*   Producer/Consumer
*/

@interface Worker : NSObject
{
    int myId;
    Store* store;
    volatile BOOL done;
}
- initWithId:(int)anId store:aStore;
- (void)consume:(int)count;
- (void)produce:(int)count;
- (void)wait;
@end

@implementation Worker

- initWithId:(int)anId store:aStore 
{
    [super init];
    myId = anId;
    store = aStore;
    done = NO;
    return self;
}

- (void)dealloc
{
    [super dealloc];
}

- (void)produce:(int)count
{
    int storeCount;
    
    if (verbose) {
	objc_mutex_lock(io_lock);
	printf("%2d: add: Start.\n", myId);
	objc_mutex_unlock(io_lock);
    }
    
    while (count>0) {
	storeCount = [store addObject];
	if (verbose) {
	    objc_mutex_lock(io_lock);
	    printf("%2d: add: %3d store has: %3d\n", myId, count, storeCount);
	    objc_mutex_unlock(io_lock);
	}
	count--;
    }
    if (verbose) {
	objc_mutex_lock(io_lock);
	printf("%2d: add: Done.\n", myId);
	objc_mutex_unlock(io_lock);
    }
    done = YES;
}

- (void)consume:(int)count
{
    int storeCount;

    if (verbose) {
	objc_mutex_lock(io_lock);
	printf("%2d: del: Start.\n", myId);
	objc_mutex_unlock(io_lock);
    }
    while (count>0) {
	storeCount = [store removeObject];
	if (verbose) {
	    objc_mutex_lock(io_lock);
	    printf("%2d: del: %3d store has: %3d\n", myId, count, storeCount);
	    objc_mutex_unlock(io_lock);
	}
	count--;
    }
    if (verbose) {
	objc_mutex_lock(io_lock);
	printf("%2d: del: Done.\n", myId);
	objc_mutex_unlock(io_lock);
    }
    done = YES;
}

- (void)wait
{
    while (!done);
}

@end

@interface TestThreads : NSObject
+ (BOOL)runBitmapConditionTest:(int)sc 
  producers:(int)np items:(int)pi 
  consumers:(int)nc items:(int)ci;
@end

@implementation TestThreads

int items_to_consume, items_to_produce;

static void consumeFunc(id obj)
{
    [obj consume:items_to_consume];
}

static void produceFunc(id obj)
{
    [obj produce:items_to_produce];
}

+ (BOOL)runBitmapConditionTest:(int)sc 
  producers:(int)np items:(int)pi 
  consumers:(int)nc items:(int)ci
{
    int nprod  = np;
    int ncons  = nc;
    int scap   = sc;
    int pitems = pi;
    int citems = ci;
    
    id store, producers[nprod], consumers[ncons];
    int i;

    printf("Starting test: store capacity %d "
	    "\n\tproducers %d each %d items, "
	    "\n\tconsumers %d each %d items\n",
	    scap, nprod, pitems, ncons, citems);

    if (nprod*pitems != ncons*citems) {
	printf("Error: (producers*prod_items) "
		"must be equal to (consumers*cons_items)\n");
	return 0;
    }

    io_lock = objc_mutex_allocate();

    store = [[Store alloc] initWithCapacity:scap];
    
    for (i=0; i<ncons; i++)
	consumers[i] = [[Worker alloc] initWithId:i store:store];
    for (i=0; i<nprod; i++)
	producers[i] = [[Worker alloc] initWithId:i store:store];

    items_to_consume = citems;
    printf("Starting consumers.\n");
    for (i=0; i<ncons; i++)
	objc_thread_create((void (*)(void *arg))consumeFunc, consumers[i]);
	//objc_thread_detach(@selector(consume:), consumers[i], (id)citems);

    items_to_produce = pitems;
    printf("Starting producers.\n");
    for (i=0; i<nprod; i++)
	objc_thread_create((void (*)(void *arg))produceFunc, producers[i]);
	//objc_thread_create(@selector(produce:), producers[i], (id)pitems);

    for (i=0; i<ncons; i++) {
	[consumers[i] wait];
	[consumers[i] release];
    }
    for (i=0; i<nprod; i++) {
	[producers[i] wait];
	[producers[i] release];
    }

    [store release];

    printf("thread test Ok.\n");
    return 1;
}

@end

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.