mirror of
https://github.com/gnustep/libs-base.git
synced 2025-04-27 18:50:47 +00:00
Richard: I'm unsure about three of these, which were fixes in memset() calls in: - NSConcreteMapTable.m - NSConcreteHashTable.m - Additions/NSData+GNUstepBase.m Please can you check them? I think they are intended to zero the entire object (rather than the first word), but the lack of comments makes me unsure. Most changes were just tweaks to variable types. I've also removed some dead code from NSInvocation. This was small group of things that were marked for internal use only, but not actually referenced in the code anywhere. Other improvements: - NSArray / NSDictionary fixed up to use the 10.7 (ARC-friendly) prototypes. - getObjects:andKeys: implemented for NSDictionary (10.5 method) - NSPointerArray and NSHashTable now properly support weak objects. - Tests for weak objects in collections. git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@33621 72102866-910b-0410-8b05-ffd578937521
408 lines
9.3 KiB
Objective-C
408 lines
9.3 KiB
Objective-C
/* Implementation of garbage collecting dictionary classes
|
||
|
||
Copyright (C) 2002 Free Software Foundation, Inc.
|
||
|
||
Written by: Richard Frith-Macdonald <rfm@gnu.org>
|
||
Inspired by gc classes of Ovidiu Predescu and Mircea Oancea
|
||
|
||
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"
|
||
#ifndef NeXT_Foundation_LIBRARY
|
||
#import "Foundation/NSException.h"
|
||
#endif
|
||
|
||
#import "GNUstepBase/GSObjCRuntime.h"
|
||
#import "GNUstepBase/GCObject.h"
|
||
|
||
typedef struct {
|
||
id object;
|
||
BOOL isGCObject;
|
||
} GCInfo;
|
||
|
||
@interface _GCDictionaryKeyEnumerator : NSEnumerator
|
||
{
|
||
@public
|
||
GCDictionary *dict;
|
||
NSMapEnumerator enumerator;
|
||
}
|
||
- (id) nextObject;
|
||
@end
|
||
|
||
@interface _GCDictionaryObjectEnumerator : _GCDictionaryKeyEnumerator
|
||
- (id) nextObject;
|
||
@end
|
||
|
||
@implementation _GCDictionaryKeyEnumerator
|
||
- (id) copyWithZone: (NSZone*)z
|
||
{
|
||
return [self retain];
|
||
}
|
||
- (void) dealloc
|
||
{
|
||
NSEndMapTableEnumeration(&enumerator);
|
||
DESTROY(dict);
|
||
[super dealloc];
|
||
}
|
||
- (id) nextObject
|
||
{
|
||
GCInfo *keyStruct;
|
||
GCInfo *valueStruct;
|
||
|
||
return NSNextMapEnumeratorPair(&enumerator,
|
||
(void**)&keyStruct, (void**)&valueStruct) ? keyStruct->object : nil;
|
||
}
|
||
@end
|
||
|
||
@implementation _GCDictionaryObjectEnumerator
|
||
- (id) nextObject
|
||
{
|
||
GCInfo *keyStruct;
|
||
GCInfo *valueStruct;
|
||
|
||
return NSNextMapEnumeratorPair(&enumerator,
|
||
(void**)&keyStruct, (void**)&valueStruct) ? valueStruct->object : nil;
|
||
}
|
||
@end
|
||
|
||
@implementation GCDictionary
|
||
|
||
static unsigned
|
||
_GCHashObject(NSMapTable *table, const GCInfo *objectStruct)
|
||
{
|
||
return [objectStruct->object hash];
|
||
}
|
||
|
||
static BOOL
|
||
_GCCompareObjects(NSMapTable *table, const GCInfo *o1, const GCInfo *o2)
|
||
{
|
||
return [o1->object isEqual: o2->object];
|
||
}
|
||
|
||
static void
|
||
_GCRetainObjects(NSMapTable *table, const void *ptr)
|
||
{
|
||
#if !GS_WITH_GC
|
||
GCInfo *objectStruct = (GCInfo*)ptr;
|
||
|
||
[objectStruct->object retain];
|
||
#endif
|
||
}
|
||
|
||
static void
|
||
_GCReleaseObjects(NSMapTable *table, const void *ptr)
|
||
{
|
||
GCInfo *objectStruct = (GCInfo*)ptr;
|
||
|
||
if ([GCObject gcIsCollecting])
|
||
{
|
||
if (objectStruct->isGCObject == NO)
|
||
{
|
||
DESTROY(objectStruct->object);
|
||
}
|
||
}
|
||
else
|
||
{
|
||
DESTROY(objectStruct->object);
|
||
}
|
||
NSZoneFree(NSDefaultMallocZone(), objectStruct);
|
||
}
|
||
|
||
static NSString*
|
||
_GCDescribeObjects(NSMapTable *table, const GCInfo *objectStruct)
|
||
{
|
||
return [objectStruct->object description];
|
||
}
|
||
|
||
static const NSMapTableKeyCallBacks GCInfoMapKeyCallBacks = {
|
||
(NSUInteger(*)(NSMapTable *, const void *))_GCHashObject,
|
||
(BOOL(*)(NSMapTable *, const void *, const void *))_GCCompareObjects,
|
||
(void (*)(NSMapTable *, const void *))_GCRetainObjects,
|
||
(void (*)(NSMapTable *, void *))_GCReleaseObjects,
|
||
(NSString *(*)(NSMapTable *, const void *))_GCDescribeObjects,
|
||
(const void *)NULL
|
||
};
|
||
|
||
static const NSMapTableValueCallBacks GCInfoValueCallBacks = {
|
||
(void (*)(NSMapTable *, const void *))_GCRetainObjects,
|
||
(void (*)(NSMapTable *, void *))_GCReleaseObjects,
|
||
(NSString *(*)(NSMapTable *, const void *))_GCDescribeObjects
|
||
};
|
||
|
||
static Class gcClass = 0;
|
||
|
||
+ (void) initialize
|
||
{
|
||
if (gcClass == 0)
|
||
{
|
||
gcClass = [GCObject class];
|
||
GSObjCAddClassBehavior(self, gcClass);
|
||
}
|
||
}
|
||
|
||
- (id) copyWithZone: (NSZone*)zone
|
||
{
|
||
if (NSShouldRetainWithZone(self, zone))
|
||
{
|
||
return [self retain];
|
||
}
|
||
return [[GCDictionary allocWithZone: zone] initWithDictionary: self];
|
||
}
|
||
|
||
- (NSUInteger) count
|
||
{
|
||
return NSCountMapTable(_map);
|
||
}
|
||
|
||
- (void) dealloc
|
||
{
|
||
[GCObject gcObjectWillBeDeallocated: (GCObject*)self];
|
||
NSFreeMapTable(_map);
|
||
[super dealloc];
|
||
}
|
||
|
||
- (void) gcDecrementRefCountOfContainedObjects
|
||
{
|
||
NSMapEnumerator enumerator = NSEnumerateMapTable(_map);
|
||
GCInfo *keyStruct;
|
||
GCInfo *valueStruct;
|
||
|
||
gc.flags.visited = 0;
|
||
while (NSNextMapEnumeratorPair(&enumerator,
|
||
(void**)&keyStruct, (void**)&valueStruct))
|
||
{
|
||
if (keyStruct->isGCObject)
|
||
{
|
||
[keyStruct->object gcDecrementRefCount];
|
||
}
|
||
if (valueStruct->isGCObject)
|
||
{
|
||
[valueStruct->object gcDecrementRefCount];
|
||
}
|
||
}
|
||
NSEndMapTableEnumeration(&enumerator);
|
||
}
|
||
|
||
- (BOOL) gcIncrementRefCountOfContainedObjects
|
||
{
|
||
NSMapEnumerator enumerator;
|
||
GCInfo *keyStruct;
|
||
GCInfo *valueStruct;
|
||
|
||
if (gc.flags.visited == 1)
|
||
{
|
||
return NO;
|
||
}
|
||
gc.flags.visited = 1;
|
||
|
||
enumerator = NSEnumerateMapTable(_map);
|
||
while (NSNextMapEnumeratorPair(&enumerator,
|
||
(void**)&keyStruct, (void**)&valueStruct))
|
||
{
|
||
if (keyStruct->isGCObject)
|
||
{
|
||
[keyStruct->object gcIncrementRefCount];
|
||
[keyStruct->object gcIncrementRefCountOfContainedObjects];
|
||
}
|
||
if (valueStruct->isGCObject)
|
||
{
|
||
[valueStruct->object gcIncrementRefCount];
|
||
[valueStruct->object gcIncrementRefCountOfContainedObjects];
|
||
}
|
||
}
|
||
NSEndMapTableEnumeration(&enumerator);
|
||
return YES;
|
||
}
|
||
|
||
- (id) initWithDictionary: (NSDictionary*)dictionary
|
||
{
|
||
id keys = [dictionary keyEnumerator];
|
||
id key;
|
||
NSUInteger size = ([dictionary count] * 4) / 3;
|
||
NSZone *z = NSDefaultMallocZone();
|
||
|
||
_map = NSCreateMapTableWithZone(GCInfoMapKeyCallBacks,
|
||
GCInfoValueCallBacks, size, z);
|
||
|
||
while ((key = [keys nextObject]) != nil)
|
||
{
|
||
GCInfo *keyStruct;
|
||
GCInfo *valueStruct;
|
||
id value;
|
||
|
||
keyStruct = NSZoneMalloc(z, sizeof(GCInfo));
|
||
valueStruct = NSZoneMalloc(z, sizeof(GCInfo));
|
||
value = [dictionary objectForKey: key];
|
||
keyStruct->object = key;
|
||
keyStruct->isGCObject = [key isKindOfClass: gcClass];
|
||
valueStruct->object = value;
|
||
valueStruct->isGCObject = [value isKindOfClass: gcClass];
|
||
NSMapInsert(_map, keyStruct, valueStruct);
|
||
}
|
||
|
||
return self;
|
||
}
|
||
|
||
- (id) initWithObjects: (const id[])objects
|
||
forKeys: (const id[])keys
|
||
count: (NSUInteger)count
|
||
{
|
||
NSUInteger size = (count * 4) / 3;
|
||
NSZone *z = NSDefaultMallocZone();
|
||
|
||
_map = NSCreateMapTableWithZone(GCInfoMapKeyCallBacks,
|
||
GCInfoValueCallBacks, size, z);
|
||
|
||
while (count-- > 0)
|
||
{
|
||
GCInfo *keyStruct;
|
||
GCInfo *valueStruct;
|
||
|
||
if (!keys[count] || !objects[count])
|
||
{
|
||
DESTROY(self);
|
||
[NSException raise: NSInvalidArgumentException
|
||
format: @"Nil object added in dictionary"];
|
||
}
|
||
keyStruct = NSZoneMalloc(z, sizeof(GCInfo));
|
||
valueStruct = NSZoneMalloc(z, sizeof(GCInfo));
|
||
keyStruct->object = keys[count];
|
||
keyStruct->isGCObject = [keys[count] isKindOfClass: gcClass];
|
||
valueStruct->object = objects[count];
|
||
valueStruct->isGCObject
|
||
= [objects[count] isKindOfClass: gcClass];
|
||
NSMapInsert(_map, keyStruct, valueStruct);
|
||
}
|
||
return self;
|
||
}
|
||
|
||
/**
|
||
* We use the same initial instance variable layout as a GCObject and
|
||
* ue the <em>behavior</em> mechanism to inherit methods from that class
|
||
* to implement a form of multiple inheritance. We need to implement
|
||
* this method to make this apparent at runtime.
|
||
*/
|
||
- (BOOL) isKindOfClass: (Class)c
|
||
{
|
||
if (c == gcClass)
|
||
{
|
||
return YES;
|
||
}
|
||
return [super isKindOfClass: c];
|
||
}
|
||
|
||
- (NSEnumerator*) keyEnumerator
|
||
{
|
||
_GCDictionaryKeyEnumerator *e;
|
||
|
||
e = [_GCDictionaryKeyEnumerator alloc];
|
||
e->dict = [self retain];
|
||
e->enumerator = NSEnumerateMapTable(_map);
|
||
return AUTORELEASE(e);
|
||
}
|
||
|
||
- (NSEnumerator*) objectEnumerator
|
||
{
|
||
_GCDictionaryObjectEnumerator *e;
|
||
|
||
e = [_GCDictionaryObjectEnumerator alloc];
|
||
e->dict = [self retain];
|
||
e->enumerator = NSEnumerateMapTable(_map);
|
||
return AUTORELEASE(e);
|
||
}
|
||
|
||
- (id) mutableCopyWithZone: (NSZone*)zone
|
||
{
|
||
return [[GCMutableDictionary allocWithZone: zone] initWithDictionary: self];
|
||
}
|
||
|
||
- (id) objectForKey: (id)key
|
||
{
|
||
GCInfo keyStruct = { key, 0 };
|
||
GCInfo *valueStruct;
|
||
|
||
valueStruct = NSMapGet(_map, (void**)&keyStruct);
|
||
return valueStruct ? valueStruct->object : nil;
|
||
}
|
||
|
||
@end
|
||
|
||
|
||
|
||
@implementation GCMutableDictionary
|
||
|
||
+ (void) initialize
|
||
{
|
||
static BOOL beenHere = NO;
|
||
|
||
if (beenHere == NO)
|
||
{
|
||
beenHere = YES;
|
||
GSObjCAddClassBehavior(self, [GCDictionary class]);
|
||
}
|
||
}
|
||
|
||
- (id) init
|
||
{
|
||
return [self initWithCapacity: 0];
|
||
}
|
||
|
||
- (id) initWithCapacity: (NSUInteger)aNumItems
|
||
{
|
||
NSUInteger size = (aNumItems * 4) / 3;
|
||
|
||
_map = NSCreateMapTableWithZone(GCInfoMapKeyCallBacks,
|
||
GCInfoValueCallBacks, size, [self zone]);
|
||
return self;
|
||
}
|
||
|
||
- (id) copyWithZone: (NSZone*)zone
|
||
{
|
||
return [[GCDictionary allocWithZone: zone] initWithDictionary: self];
|
||
}
|
||
|
||
- (void) setObject: (id)anObject forKey: (id)aKey
|
||
{
|
||
GCInfo *keyStruct;
|
||
GCInfo *valueStruct;
|
||
NSZone *z = NSDefaultMallocZone();
|
||
|
||
keyStruct = NSZoneMalloc(z, sizeof(GCInfo));
|
||
valueStruct = NSZoneMalloc(z, sizeof(GCInfo));
|
||
keyStruct->object = aKey;
|
||
keyStruct->isGCObject = [aKey isKindOfClass: gcClass];
|
||
valueStruct->object = anObject;
|
||
valueStruct->isGCObject = [anObject isKindOfClass: gcClass];
|
||
NSMapInsert(_map, keyStruct, valueStruct);
|
||
}
|
||
|
||
- (void) removeObjectForKey: (id)key
|
||
{
|
||
GCInfo keyStruct = { key, 0 };
|
||
|
||
NSMapRemove(_map, (void**)&keyStruct);
|
||
}
|
||
|
||
- (void) removeAllObjects
|
||
{
|
||
NSResetMapTable(_map);
|
||
}
|
||
|
||
@end
|