ftp.nice.ch/Attic/openStep/implementation/gnustep/sources/libFoundation.0.7.tgz#/libFoundation-0.7/libFoundation/Foundation/NSLock.m

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

/* 
   NSLock.m

   Copyright (C) 1995, 1996 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/NSLock.h>

#include <Foundation/NSException.h>
#include <Foundation/exceptions/GeneralExceptions.h>

/*
 * NSLock - lock of depth limited to 1
 */

@implementation NSLock

- init
{
    mutex = objc_mutex_allocate();
    if (!mutex) {
	[self release];
	return nil;
    }
    return self;
}

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

- (void)lock
{
    if (objc_mutex_lock(mutex) > 1)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to lock a simple lock (NSLock) multiple times"]);
}

- (BOOL)tryLock
{
    int depth = objc_mutex_trylock(mutex);
    
    if (depth > 1)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to lock a simple lock (NSLock) multiple times"]);
    
    return depth == 1 ? YES : NO;
}

- (BOOL)lockBeforeDate:(NSDate *)limit
{
    [self notImplemented:_cmd];
    return NO;
}

- (void)unlock
{
    if (objc_mutex_unlock(mutex) < 0)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to unlock a lock not owned by the current thread"]);
}

@end /* NSLock */

/*
 * NSLock - lock of unlimited (MAX_INT) depth
 */

@implementation NSRecursiveLock:NSObject

- init
{
    mutex = objc_mutex_allocate();
    if (!mutex) {
	[self release];
	return nil;
    }
    return self;
}

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

- (void)lock
{
    if (objc_mutex_lock(mutex) < 0)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to lock an invalid lock"]);
}

- (BOOL)tryLock
{
    return objc_mutex_trylock(mutex) > 0 ? YES : NO;
}

- (BOOL)lockBeforeDate:(NSDate *)limit
{
    [self notImplemented:_cmd];
    return NO;
}

- (void)unlock
{
    if (objc_mutex_unlock(mutex) < 0)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to unlock a lock not owned by the current thread"]);
}

@end /* NSRecursiveLock:NSObject */

/*
 * Condition lock
 */

@implementation NSConditionLock:NSObject

- init
{
    return [self initWithCondition:0];
}

- initWithCondition:(int)aCondition
{
    mutex = objc_mutex_allocate();
    if (!mutex) {
	[self release];
	return nil;
    }
    condition = objc_condition_allocate();
    if (!condition) {
	[self release];
	return nil;
    }
    value = aCondition;
    return self;
}

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

- (int)condition
{
    return value;
}

- (void)lock
{
    if (objc_mutex_lock(mutex) < 0)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to lock an invalid lock"]);
}

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

    // Return on error
    if (depth == -1)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to lock an invalid lock"]);
	    
    // Error if recursive lock and condition is false 
    if ((depth > 1) && (value != aCondition))
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to lock a condition lock multiple times"
	    @"with a different condition value"]);

    // Wait condition
    while (value != aCondition)
	    objc_condition_wait(condition, mutex);
}

- (BOOL)lockBeforeDate:(NSDate *)limit
{
    [self notImplemented:_cmd];
    return NO;
}

- (BOOL)lockWhenCondition:(int)aCondition beforeDate:(NSDate *)limit
{
    [self notImplemented:_cmd];
    return NO;
}

- (BOOL)tryLock
{
    return objc_mutex_trylock(mutex) > 0 ? YES : NO;
}

- (BOOL)tryLockWhenCondition:(int)aCondition
{
    int depth;
    
    // Try to lock the mutex
    depth = objc_mutex_trylock(mutex);

    // Return on error
    if (depth == -1)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to lock an invalid lock"]);
	    
    // Return on no lock
    if (depth == 0)
	return NO;
    
    if (value != aCondition) {
	objc_mutex_unlock(mutex);
	return NO;
    }
    
    return YES;
}

- (void)unlock
{
    if (objc_mutex_unlock(mutex) < 0)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to unlock a lock not owned by the current thread"]);
}

- (void)unlockWithCondition:(int)aCondition
{
    int depth;
    
    // Try to lock the mutex again
    depth = objc_mutex_trylock(mutex);
    
    // Another thread has the lock so abort
    if (depth == -1)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to lock an invalid lock"]);
    
    // If the depth is only 1 then we just acquired
    // the lock above, bogus unlock so abort
    if (depth == 1)
	THROW([[InvalidUseOfMethodException alloc] initWithFormat:
	    @"attempt to unlock with condition a lock "
	    @"not owned by the current thread"]);
    
    value = aCondition;
    
    // Wake up threads waiting a condition to happen
    objc_condition_broadcast(condition);
    // This is a valid unlock so set the condition and unlock twice
    objc_mutex_unlock(mutex);
    objc_mutex_unlock(mutex);
}

@end /* NSConditionLock:NSObject */

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