libs-base/Source/Additions/NSObject+GNUstepBase.m
rfm f4e03baeea build on apple
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@33791 72102866-910b-0410-8b05-ffd578937521
2011-08-26 11:21:32 +00:00

323 lines
6.5 KiB
Objective-C

/* Implementation of extension methods to base additions
Copyright (C) 2010 Free Software Foundation, Inc.
Written by: Richard Frith-Macdonald <rfm@gnu.org>
This file is part of the GNUstep Base Library.
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Library General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free
Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02111 USA.
*/
#import "common.h"
#import "Foundation/NSArray.h"
#import "Foundation/NSException.h"
#import "Foundation/NSLock.h"
#import "GNUstepBase/NSObject+GNUstepBase.h"
#import "GNUstepBase/NSDebug+GNUstepBase.h"
/**
* Extension methods for the NSObject class
*/
@implementation NSObject (GNUstepBase)
+ (id) notImplemented: (SEL)selector
{
[NSException raise: NSGenericException
format: @"method %@ not implemented in %s(class)",
selector ? (id)NSStringFromSelector(selector) : (id)@"(null)",
NSStringFromClass(self)];
return nil;
}
- (NSComparisonResult) compare: (id)anObject
{
NSLog(@"WARNING: The -compare: method for NSObject is deprecated.");
if (anObject == self)
{
return NSOrderedSame;
}
if (anObject == nil)
{
[NSException raise: NSInvalidArgumentException
format: @"nil argument for compare:"];
}
if ([self isEqual: anObject])
{
return NSOrderedSame;
}
/*
* Ordering objects by their address is pretty useless,
* so subclasses should override this is some useful way.
*/
if ((id)self > anObject)
{
return NSOrderedDescending;
}
else
{
return NSOrderedAscending;
}
}
- (BOOL) isInstance
{
GSOnceMLog(@"Warning, the -isInstance method is deprecated. "
@"Use 'class_isMetaClass([self class]) ? NO : YES' instead");
return class_isMetaClass([self class]) ? NO : YES;
}
- (id) makeImmutableCopyOnFail: (BOOL)force
{
if (force == YES)
{
return AUTORELEASE([self copy]);
}
return self;
}
- (id) notImplemented: (SEL)aSel
{
char c = (class_isMetaClass(object_getClass(self)) ? '+' : '-');
[NSException
raise: NSInvalidArgumentException
format: @"[%@%c%@] not implemented",
NSStringFromClass([self class]), c,
aSel ? (id)NSStringFromSelector(aSel) : (id)@"(null)"];
return nil;
}
- (id) shouldNotImplement: (SEL)aSel
{
char c = (class_isMetaClass(object_getClass(self)) ? '+' : '-');
[NSException
raise: NSInvalidArgumentException
format: @"[%@%c%@] should not be implemented",
NSStringFromClass([self class]), c,
aSel ? (id)NSStringFromSelector(aSel) : (id)@"(null)"];
return nil;
}
- (id) subclassResponsibility: (SEL)aSel
{
char c = (class_isMetaClass(object_getClass(self)) ? '+' : '-');
[NSException raise: NSInvalidArgumentException
format: @"[%@%c%@] should be overridden by subclass",
NSStringFromClass([self class]), c,
aSel ? (id)NSStringFromSelector(aSel) : (id)@"(null)"];
return nil;
}
@end
#if defined(GNUSTEP)
struct exitLink {
struct exitLink *next;
id obj; // Object to release or class for atExit
SEL sel; // Selector for atExit or 0 if releasing
id *at; // Address of static variable or NULL
};
static struct exitLink *exited = 0;
static BOOL enabled = NO;
static BOOL shouldCleanUp = NO;
static void
handleExit()
{
while (exited != 0)
{
struct exitLink *tmp = exited;
exited = tmp->next;
if (0 != tmp->sel)
{
Method method;
IMP msg;
method = class_getClassMethod(tmp->obj, tmp->sel);
msg = method_getImplementation(method);
if (0 != msg)
{
(*msg)(tmp->obj, tmp->sel);
}
}
else if (YES == shouldCleanUp)
{
if (0 != tmp->at)
{
tmp->obj = *(tmp->at);
*(tmp->at) = nil;
}
[tmp->obj release];
}
free(tmp);
}
}
@implementation NSObject(GSCleanup)
+ (id) leakAt: (id*)anAddress
{
struct exitLink *l;
l = (struct exitLink*)malloc(sizeof(struct exitLink));
l->at = anAddress;
l->obj = [*anAddress retain];
l->sel = 0;
[gnustep_global_lock lock];
l->next = exited;
exited = l;
[gnustep_global_lock unlock];
return l->obj;
}
+ (id) leak: (id)anObject
{
struct exitLink *l;
l = (struct exitLink*)malloc(sizeof(struct exitLink));
l->at = 0;
l->obj = [anObject retain];
l->sel = 0;
[gnustep_global_lock lock];
l->next = exited;
exited = l;
[gnustep_global_lock unlock];
return l->obj;
}
+ (BOOL) registerAtExit
{
return [self registerAtExit: @selector(atExit)];
}
+ (BOOL) registerAtExit: (SEL)sel
{
Method m;
Class s;
struct exitLink *l;
if (0 == sel)
{
sel = @selector(atExit);
}
m = class_getClassMethod(self, sel);
if (0 == m)
{
return NO; // method not implemented.
}
s = class_getSuperclass(self);
if (0 != s && class_getClassMethod(s, sel) == m)
{
return NO; // method not implemented in this class
}
[gnustep_global_lock lock];
for (l = exited; l != 0; l = l->next)
{
if (l->obj == self && sel_isEqual(l->sel, sel))
{
[gnustep_global_lock unlock];
return NO; // Already registered
}
}
l = (struct exitLink*)malloc(sizeof(struct exitLink));
l->obj = self;
l->sel = sel;
l->at = 0;
l->next = exited;
exited = l;
if (NO == enabled)
{
atexit(handleExit);
enabled = YES;
}
[gnustep_global_lock unlock];
return YES;
}
+ (void) setShouldCleanUp: (BOOL)aFlag
{
if (YES == aFlag)
{
[gnustep_global_lock lock];
if (NO == enabled)
{
atexit(handleExit);
enabled = YES;
}
[gnustep_global_lock unlock];
shouldCleanUp = YES;
}
else
{
shouldCleanUp = NO;
}
}
+ (BOOL) shouldCleanUp
{
return shouldCleanUp;
}
@end
#else
/* Dummy implementation
*/
@implementation NSObject(GSCleanup)
+ (id) leakAt: (id*)anAddress
{
[*anAddress retain];
}
+ (id) leak: (id)anObject
{
return [anObject retain];
}
+ (BOOL) registerAtExit
{
return [self registerAtExit: @selector(atExit)];
}
+ (BOOL) registerAtExit: (SEL)sel
{
return NO;
}
+ (void) setShouldCleanUp: (BOOL)aFlag
{
return;
}
+ (BOOL) shouldCleanUp
{
return NO;
}
#endif