mirror of
https://github.com/gnustep/libs-base.git
synced 2025-05-30 16:30:41 +00:00
Thread-safety fixes.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@39902 72102866-910b-0410-8b05-ffd578937521
This commit is contained in:
parent
0c7237ec08
commit
1b63746799
3 changed files with 148 additions and 93 deletions
|
@ -6,6 +6,8 @@
|
||||||
* Source/NSOperation.m: When starting an operation, have it retain
|
* Source/NSOperation.m: When starting an operation, have it retain
|
||||||
itself in case it'ss removed from the queue and released while
|
itself in case it'ss removed from the queue and released while
|
||||||
running.
|
running.
|
||||||
|
* Headers/Foundation/NSDistributedLock.h: Add lock ivar.
|
||||||
|
* Source/NSDistributedLock.m: Make class thread-safe using lock.
|
||||||
|
|
||||||
2016-06-19 Richard Frith-Macdonald <rfm@gnu.org>
|
2016-06-19 Richard Frith-Macdonald <rfm@gnu.org>
|
||||||
|
|
||||||
|
|
|
@ -27,8 +27,10 @@
|
||||||
#import <GNUstepBase/GSVersionMacros.h>
|
#import <GNUstepBase/GSVersionMacros.h>
|
||||||
|
|
||||||
#import <Foundation/NSObject.h>
|
#import <Foundation/NSObject.h>
|
||||||
#import <Foundation/NSString.h>
|
|
||||||
#import <Foundation/NSDate.h>
|
@class NSDate;
|
||||||
|
@class NSLock;
|
||||||
|
@class NSString;
|
||||||
|
|
||||||
#if defined(__cplusplus)
|
#if defined(__cplusplus)
|
||||||
extern "C" {
|
extern "C" {
|
||||||
|
@ -39,6 +41,7 @@ extern "C" {
|
||||||
#if GS_EXPOSE(NSDistributedLock)
|
#if GS_EXPOSE(NSDistributedLock)
|
||||||
NSString *_lockPath;
|
NSString *_lockPath;
|
||||||
NSDate *_lockTime;
|
NSDate *_lockTime;
|
||||||
|
NSLock *_localLock;
|
||||||
#endif
|
#endif
|
||||||
#if GS_NONFRAGILE
|
#if GS_NONFRAGILE
|
||||||
#else
|
#else
|
||||||
|
|
|
@ -28,8 +28,9 @@
|
||||||
#import "common.h"
|
#import "common.h"
|
||||||
#define EXPOSE_NSDistributedLock_IVARS 1
|
#define EXPOSE_NSDistributedLock_IVARS 1
|
||||||
#import "Foundation/NSDistributedLock.h"
|
#import "Foundation/NSDistributedLock.h"
|
||||||
#import "Foundation/NSFileManager.h"
|
|
||||||
#import "Foundation/NSException.h"
|
#import "Foundation/NSException.h"
|
||||||
|
#import "Foundation/NSFileManager.h"
|
||||||
|
#import "Foundation/NSLock.h"
|
||||||
#import "Foundation/NSValue.h"
|
#import "Foundation/NSValue.h"
|
||||||
#import "GSPrivate.h"
|
#import "GSPrivate.h"
|
||||||
|
|
||||||
|
@ -74,27 +75,39 @@ static NSFileManager *mgr = nil;
|
||||||
*/
|
*/
|
||||||
- (void) breakLock
|
- (void) breakLock
|
||||||
{
|
{
|
||||||
NSDictionary *attributes;
|
[_localLock lock];
|
||||||
|
NS_DURING
|
||||||
DESTROY(_lockTime);
|
|
||||||
attributes = [mgr fileAttributesAtPath: _lockPath traverseLink: YES];
|
|
||||||
if (attributes != nil)
|
|
||||||
{
|
{
|
||||||
NSDate *modDate = [attributes fileModificationDate];
|
NSDictionary *attributes;
|
||||||
|
|
||||||
if ([mgr removeFileAtPath: _lockPath handler: nil] == NO)
|
DESTROY(_lockTime);
|
||||||
|
attributes = [mgr fileAttributesAtPath: _lockPath traverseLink: YES];
|
||||||
|
if (attributes != nil)
|
||||||
{
|
{
|
||||||
NSString *err = [[NSError _last] localizedDescription];
|
NSDate *modDate = [attributes fileModificationDate];
|
||||||
|
|
||||||
attributes = [mgr fileAttributesAtPath: _lockPath traverseLink: YES];
|
if ([mgr removeFileAtPath: _lockPath handler: nil] == NO)
|
||||||
if ([modDate isEqual: [attributes fileModificationDate]] == YES)
|
|
||||||
{
|
{
|
||||||
[NSException raise: NSGenericException
|
NSString *err = [[NSError _last] localizedDescription];
|
||||||
format: @"Failed to remove lock directory '%@' - %@",
|
|
||||||
_lockPath, err];
|
attributes = [mgr fileAttributesAtPath: _lockPath
|
||||||
|
traverseLink: YES];
|
||||||
|
if ([modDate isEqual: [attributes fileModificationDate]] == YES)
|
||||||
|
{
|
||||||
|
[NSException raise: NSGenericException
|
||||||
|
format: @"Failed to remove lock directory '%@' - %@",
|
||||||
|
_lockPath, err];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
NS_HANDLER
|
||||||
|
{
|
||||||
|
[_localLock unlock];
|
||||||
|
[localException raise];
|
||||||
|
}
|
||||||
|
NS_ENDHANDLER
|
||||||
|
[_localLock unlock];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (void) dealloc
|
- (void) dealloc
|
||||||
|
@ -107,21 +120,27 @@ static NSFileManager *mgr = nil;
|
||||||
}
|
}
|
||||||
RELEASE(_lockPath);
|
RELEASE(_lockPath);
|
||||||
RELEASE(_lockTime);
|
RELEASE(_lockTime);
|
||||||
|
RELEASE(_localLock);
|
||||||
[super dealloc];
|
[super dealloc];
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSString*) description
|
- (NSString*) description
|
||||||
{
|
{
|
||||||
|
NSString *result;
|
||||||
|
|
||||||
|
[_localLock lock];
|
||||||
if (_lockTime == nil)
|
if (_lockTime == nil)
|
||||||
{
|
{
|
||||||
return [[super description] stringByAppendingFormat:
|
result = [[super description] stringByAppendingFormat:
|
||||||
@" path '%@' not locked", _lockPath];
|
@" path '%@' not locked", _lockPath];
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
return [[super description] stringByAppendingFormat:
|
result = [[super description] stringByAppendingFormat:
|
||||||
@" path '%@' locked at %@", _lockPath, _lockTime];
|
@" path '%@' locked at %@", _lockPath, _lockTime];
|
||||||
}
|
}
|
||||||
|
[_localLock unlock];
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -137,6 +156,7 @@ static NSFileManager *mgr = nil;
|
||||||
NSString *lockDir;
|
NSString *lockDir;
|
||||||
BOOL isDirectory;
|
BOOL isDirectory;
|
||||||
|
|
||||||
|
_localLock = [NSLock new];
|
||||||
_lockPath = [[aPath stringByStandardizingPath] copy];
|
_lockPath = [[aPath stringByStandardizingPath] copy];
|
||||||
_lockTime = nil;
|
_lockTime = nil;
|
||||||
|
|
||||||
|
@ -189,59 +209,79 @@ static NSFileManager *mgr = nil;
|
||||||
*/
|
*/
|
||||||
- (BOOL) tryLock
|
- (BOOL) tryLock
|
||||||
{
|
{
|
||||||
NSMutableDictionary *attributesToSet;
|
BOOL locked = NO;
|
||||||
NSDictionary *attributes;
|
|
||||||
BOOL locked;
|
|
||||||
|
|
||||||
attributesToSet = [NSMutableDictionary dictionaryWithCapacity: 1];
|
[_localLock lock];
|
||||||
[attributesToSet setObject: [NSNumber numberWithUnsignedInt: 0755]
|
NS_DURING
|
||||||
forKey: NSFilePosixPermissions];
|
|
||||||
|
|
||||||
locked = [mgr createDirectoryAtPath: _lockPath
|
|
||||||
withIntermediateDirectories: YES
|
|
||||||
attributes: attributesToSet
|
|
||||||
error: NULL];
|
|
||||||
if (locked == NO)
|
|
||||||
{
|
{
|
||||||
BOOL dir;
|
NSMutableDictionary *attributesToSet;
|
||||||
|
NSDictionary *attributes;
|
||||||
|
|
||||||
/*
|
if (nil != _lockTime)
|
||||||
* We expect the directory creation to have failed because it already
|
|
||||||
* exists as another processes lock. If the directory doesn't exist,
|
|
||||||
* then either the other process has removed it's lock (and we can retry)
|
|
||||||
* or we have a severe problem!
|
|
||||||
*/
|
|
||||||
if ([mgr fileExistsAtPath: _lockPath isDirectory: &dir] == NO)
|
|
||||||
{
|
{
|
||||||
locked = [mgr createDirectoryAtPath: _lockPath
|
[NSException raise: NSGenericException
|
||||||
withIntermediateDirectories: YES
|
format: @"Attempt to re-lock distributed lock %@",
|
||||||
attributes: attributesToSet
|
_lockPath];
|
||||||
error: NULL];
|
}
|
||||||
if (locked == NO)
|
attributesToSet = [NSMutableDictionary dictionaryWithCapacity: 1];
|
||||||
|
[attributesToSet setObject: [NSNumber numberWithUnsignedInt: 0755]
|
||||||
|
forKey: NSFilePosixPermissions];
|
||||||
|
|
||||||
|
locked = [mgr createDirectoryAtPath: _lockPath
|
||||||
|
withIntermediateDirectories: YES
|
||||||
|
attributes: attributesToSet
|
||||||
|
error: NULL];
|
||||||
|
if (NO == locked)
|
||||||
|
{
|
||||||
|
BOOL dir;
|
||||||
|
|
||||||
|
/* We expect the directory creation to have failed because
|
||||||
|
* it already exists as another processes lock.
|
||||||
|
* If the directory doesn't exist, then either the other
|
||||||
|
* process has removed it's lock (and we can retry)
|
||||||
|
* or we have a severe problem!
|
||||||
|
*/
|
||||||
|
if ([mgr fileExistsAtPath: _lockPath isDirectory: &dir] == NO)
|
||||||
{
|
{
|
||||||
NSLog(@"Failed to create lock directory '%@' - %@",
|
locked = [mgr createDirectoryAtPath: _lockPath
|
||||||
_lockPath, [NSError _last]);
|
withIntermediateDirectories: YES
|
||||||
|
attributes: attributesToSet
|
||||||
|
error: NULL];
|
||||||
|
if (NO == locked)
|
||||||
|
{
|
||||||
|
NSLog(@"Failed to create lock directory '%@' - %@",
|
||||||
|
_lockPath, [NSError _last]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (YES == locked)
|
||||||
|
{
|
||||||
|
attributes = [mgr fileAttributesAtPath: _lockPath
|
||||||
|
traverseLink: YES];
|
||||||
|
if (attributes == nil)
|
||||||
|
{
|
||||||
|
[NSException raise: NSGenericException
|
||||||
|
format: @"Unable to get attributes of lock file we made at %@",
|
||||||
|
_lockPath];
|
||||||
|
}
|
||||||
|
ASSIGN(_lockTime, [attributes fileModificationDate]);
|
||||||
|
if (nil == _lockTime)
|
||||||
|
{
|
||||||
|
[NSException raise: NSGenericException
|
||||||
|
format: @"Unable to get date of lock file we made at %@",
|
||||||
|
_lockPath];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
NS_HANDLER
|
||||||
if (locked == NO)
|
|
||||||
{
|
{
|
||||||
return NO;
|
[_localLock unlock];
|
||||||
}
|
[localException raise];
|
||||||
else
|
|
||||||
{
|
|
||||||
attributes = [mgr fileAttributesAtPath: _lockPath
|
|
||||||
traverseLink: YES];
|
|
||||||
if (attributes == nil)
|
|
||||||
{
|
|
||||||
[NSException raise: NSGenericException
|
|
||||||
format: @"Unable to get attributes of lock file we made at %@",
|
|
||||||
_lockPath];
|
|
||||||
}
|
|
||||||
ASSIGN(_lockTime, [attributes fileModificationDate]);
|
|
||||||
return YES;
|
|
||||||
}
|
}
|
||||||
|
NS_ENDHANDLER
|
||||||
|
[_localLock unlock];
|
||||||
|
return locked;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -251,43 +291,53 @@ static NSFileManager *mgr = nil;
|
||||||
*/
|
*/
|
||||||
- (void) unlock
|
- (void) unlock
|
||||||
{
|
{
|
||||||
NSDictionary *attributes;
|
[_localLock lock];
|
||||||
|
NS_DURING
|
||||||
|
{
|
||||||
|
NSDictionary *attributes;
|
||||||
|
|
||||||
if (_lockTime == nil)
|
if (_lockTime == nil)
|
||||||
{
|
|
||||||
[NSException raise: NSGenericException format: @"not locked by us"];
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Don't remove the lock if it has already been broken by someone
|
|
||||||
* else and re-created. Unfortunately, there is a window between
|
|
||||||
* testing and removing, but we do the bset we can.
|
|
||||||
*/
|
|
||||||
attributes = [mgr fileAttributesAtPath: _lockPath traverseLink: YES];
|
|
||||||
if (attributes == nil)
|
|
||||||
{
|
|
||||||
DESTROY(_lockTime);
|
|
||||||
[NSException raise: NSGenericException
|
|
||||||
format: @"lock '%@' already broken", _lockPath];
|
|
||||||
}
|
|
||||||
if ([_lockTime isEqual: [attributes fileModificationDate]])
|
|
||||||
{
|
|
||||||
DESTROY(_lockTime);
|
|
||||||
if ([mgr removeFileAtPath: _lockPath handler: nil] == NO)
|
|
||||||
{
|
{
|
||||||
[NSException raise: NSGenericException
|
[NSException raise: NSGenericException format: @"not locked by us"];
|
||||||
format: @"Failed to remove lock directory '%@' - %@",
|
}
|
||||||
_lockPath, [NSError _last]];
|
|
||||||
|
/* Don't remove the lock if it has already been broken by someone
|
||||||
|
* else and re-created. Unfortunately, there is a window between
|
||||||
|
* testing and removing, but we do the bset we can.
|
||||||
|
*/
|
||||||
|
attributes = [mgr fileAttributesAtPath: _lockPath traverseLink: YES];
|
||||||
|
if (attributes == nil)
|
||||||
|
{
|
||||||
|
DESTROY(_lockTime);
|
||||||
|
[NSException raise: NSGenericException
|
||||||
|
format: @"lock '%@' already broken", _lockPath];
|
||||||
|
}
|
||||||
|
if ([_lockTime isEqual: [attributes fileModificationDate]])
|
||||||
|
{
|
||||||
|
DESTROY(_lockTime);
|
||||||
|
if ([mgr removeFileAtPath: _lockPath handler: nil] == NO)
|
||||||
|
{
|
||||||
|
[NSException raise: NSGenericException
|
||||||
|
format: @"Failed to remove lock directory '%@' - %@",
|
||||||
|
_lockPath, [NSError _last]];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
DESTROY(_lockTime);
|
||||||
|
[NSException raise: NSGenericException
|
||||||
|
format: @"lock '%@' already broken and in use again",
|
||||||
|
_lockPath];
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
DESTROY(_lockTime);
|
DESTROY(_lockTime);
|
||||||
[NSException raise: NSGenericException
|
|
||||||
format: @"lock '%@' already broken and in use again",
|
|
||||||
_lockPath];
|
|
||||||
}
|
}
|
||||||
DESTROY(_lockTime);
|
NS_HANDLER
|
||||||
|
{
|
||||||
|
[_localLock unlock];
|
||||||
|
[localException raise];
|
||||||
|
}
|
||||||
|
NS_ENDHANDLER
|
||||||
|
[_localLock unlock];
|
||||||
}
|
}
|
||||||
|
|
||||||
@end
|
@end
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue