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.