NSNumber rewrite

git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/base/trunk@6366 72102866-910b-0410-8b05-ffd578937521
This commit is contained in:
Richard Frith-MacDonald 2000-03-23 18:57:43 +00:00
parent 0fedfd92c4
commit 7c5dc20462
5 changed files with 1794 additions and 278 deletions

View file

@ -1,9 +1,18 @@
2000-03-23 Richard Frith-Macdonald <rfm@gnu.org>
NSNumber - more extensive rewrite/optimisation ...
* Headers/gnustep/base/NSValue.h: simplified
* Source/NSNumber.m: Implemented efficient abstract methods and
improved caching.
* Source/NSConcreteNumber.m: More efficient implementations including
some previously missing methods.
2000-03-23 Adam Fedor <fedor@gnu.org>
* configure.in: Check for vasprintf function
* config/config.vasprintf.c: New file.
2000-03-19 Richard Frith-Macdonald <rfm@gnu.org>
2000-03-22 Richard Frith-Macdonald <rfm@gnu.org>
NSNumber optimisation ...
* Headers/gnustep/base/NSValue.h: Added a couple of GNUstep specific

View file

@ -114,7 +114,6 @@
- (NSComparisonResult) compare: (NSNumber*)otherNumber;
- (BOOL) isEqualToNumber: (NSNumber*)otherNumber;
@end
#ifndef NO_GNUSTEP
@ -131,22 +130,8 @@
* Cache info for internal use by NSNumber concrete subclasses.
*/
typedef struct {
int typeOrder;
int typeNext;
NSComparisonResult (*compValue)(NSNumber*, SEL, NSNumber*);
BOOL (*boolValue)(NSNumber*, SEL);
char (*charValue)(NSNumber*, SEL);
unsigned char (*unsignedCharValue)(NSNumber*, SEL);
short (*shortValue)(NSNumber*, SEL);
unsigned short (*unsignedShortValue)(NSNumber*, SEL);
int (*intValue)(NSNumber*, SEL);
unsigned int (*unsignedIntValue)(NSNumber*, SEL);
long (*longValue)(NSNumber*, SEL);
unsigned long (*unsignedLongValue)(NSNumber*, SEL);
long long (*longLongValue)(NSNumber*, SEL);
unsigned long long (*unsignedLongLongValue)(NSNumber*, SEL);
float (*floatValue)(NSNumber*, SEL);
double (*doubleValue)(NSNumber*, SEL);
int typeLevel;
void (*getValue)(NSNumber*, SEL, void*);
} GSNumberInfo;
GSNumberInfo *GSNumberInfoFromObject(NSNumber *o);

View file

@ -1,10 +1,12 @@
# line 1 "NSConcreteNumber.m" /* So gdb knows which file we are in */
/* NSConcreteNumber - Object encapsulation of numbers
Copyright (C) 1993, 1994, 1996 Free Software Foundation, Inc.
Copyright (C) 1993, 1994, 1996, 2000 Free Software Foundation, Inc.
Written by: Adam Fedor <fedor@boulder.colorado.edu>
Date: Mar 1995
Rewrite: Richard Frith-Macdonald <rfm@gnu.org>
Date: Mar 2000
This file is part of the GNUstep Base Library.
@ -36,113 +38,58 @@
defined to a number from 0 to 12 cooresponding to each number type */
#if TYPE_ORDER == 0
# define NumberTemplate NSBoolNumber
# define TYPE_METHOD boolValue
# define TYPE_FORMAT @"%u"
# define NEXT_ORDER 1
# define NEXT_TYPE char
# define NEXT_METHOD charValue
# define TYPE_TYPE BOOL
#elif TYPE_ORDER == 1
# define NumberTemplate NSCharNumber
# define TYPE_METHOD charValue
# define TYPE_FORMAT @"%c"
# define NEXT_ORDER 3
# define NEXT_TYPE short
# define NEXT_METHOD shortValue
# define TYPE_TYPE char
#elif TYPE_ORDER == 2
# define NumberTemplate NSUCharNumber
# define TYPE_METHOD unsignedCharValue
# define TYPE_FORMAT @"%c"
# define NEXT_ORDER 3
# define NEXT_TYPE short
# define NEXT_METHOD shortValue
# define TYPE_TYPE unsigned char
#elif TYPE_ORDER == 3
# define NumberTemplate NSShortNumber
# define TYPE_METHOD shortValue
# define TYPE_FORMAT @"%hd"
# define NEXT_ORDER 5
# define NEXT_TYPE int
# define NEXT_METHOD intValue
# define TYPE_TYPE short
#elif TYPE_ORDER == 4
# define NumberTemplate NSUShortNumber
# define TYPE_METHOD unsignedShortValue
# define TYPE_FORMAT @"%hu"
# define NEXT_ORDER 5
# define NEXT_TYPE int
# define NEXT_METHOD intValue
# define TYPE_TYPE unsigned short
#elif TYPE_ORDER == 5
# define NumberTemplate NSIntNumber
# define TYPE_METHOD intValue
# define TYPE_FORMAT @"%d"
# define NEXT_ORDER 7
# define NEXT_TYPE long
# define NEXT_METHOD longValue
# define TYPE_TYPE int
#elif TYPE_ORDER == 6
# define NumberTemplate NSUIntNumber
# define TYPE_METHOD unsignedIntValue
# define TYPE_FORMAT @"%u"
# define NEXT_ORDER 7
# define NEXT_TYPE long
# define NEXT_METHOD longValue
# define TYPE_TYPE unsigned int
#elif TYPE_ORDER == 7
# define NumberTemplate NSLongNumber
# define TYPE_METHOD longValue
# define TYPE_FORMAT @"%ld"
# define NEXT_ORDER 9
# define NEXT_TYPE long long
# define NEXT_METHOD longLongValue
# define TYPE_TYPE long
#elif TYPE_ORDER == 8
# define NumberTemplate NSULongNumber
# define TYPE_METHOD unsignedLongValue
# define TYPE_FORMAT @"%lu"
# define NEXT_ORDER 9
# define NEXT_TYPE long long
# define NEXT_METHOD longLongValue
# define TYPE_TYPE unsigned long
#elif TYPE_ORDER == 9
# define NumberTemplate NSLongLongNumber
# define TYPE_METHOD longLongValue
# define TYPE_FORMAT @"%lld"
# define NEXT_ORDER 11
# define NEXT_TYPE float
# define NEXT_METHOD floatValue
# define TYPE_TYPE long long
#elif TYPE_ORDER == 10
# define NumberTemplate NSULongLongNumber
# define TYPE_METHOD unsignedLongLongValue
# define TYPE_FORMAT @"%llu"
# define NEXT_ORDER 11
# define NEXT_TYPE float
# define NEXT_METHOD floatValue
# define TYPE_TYPE unsigned long long
#elif TYPE_ORDER == 11
# define NumberTemplate NSFloatNumber
# define TYPE_METHOD floatValue
# define TYPE_FORMAT @"%f"
# define NEXT_ORDER 12
# define NEXT_TYPE double
# define NEXT_METHOD doubleValue
# define TYPE_TYPE float
#elif TYPE_ORDER == 12
# define NumberTemplate NSDoubleNumber
# define TYPE_METHOD doubleValue
# define TYPE_FORMAT @"%g"
# define NEXT_ORDER 12
# define NEXT_TYPE double
# define NEXT_METHOD doubleValue
# define TYPE_TYPE double
#endif
@interface NSNumber (Private)
- (int) _typeNext;
- (int) _typeOrder;
@end
@implementation NumberTemplate (Private)
- (int) _typeNext
{
return NEXT_ORDER;
}
- (int) _typeOrder
{
return TYPE_ORDER;
}
@end
@implementation NumberTemplate
- (id) initWithBytes: (const void*)value objCType: (const char*)type
@ -192,67 +139,67 @@
- (BOOL) boolValue
{
return data;
return (BOOL)data;
}
- (char) charValue
{
return data;
return (char)data;
}
- (double) doubleValue
{
return data;
return (double)data;
}
- (float) floatValue
{
return data;
return (float)data;
}
- (int) intValue
{
return data;
return (int)data;
}
- (long long) longLongValue
{
return data;
return (long long)data;
}
- (long) longValue
{
return data;
return (long)data;
}
- (short) shortValue
{
return data;
return (short)data;
}
- (unsigned char) unsignedCharValue
{
return data;
return (unsigned char)data;
}
- (unsigned int) unsignedIntValue
{
return data;
return (unsigned int)data;
}
- (unsigned long long) unsignedLongLongValue
{
return data;
return (unsigned long long)data;
}
- (unsigned long) unsignedLongValue
{
return data;
return (unsigned long)data;
}
- (unsigned short) unsignedShortValue
{
return data;
return (unsigned short)data;
}
- (NSComparisonResult) compare: (NSNumber*)other
@ -265,68 +212,168 @@
}
info = GSNumberInfoFromObject(other);
/*
* If the two types are the same, or the other type can be promoted to ours.
*/
if (TYPE_ORDER == info->typeOrder || TYPE_ORDER >= info->typeNext)
switch (info->typeLevel)
{
typedef __typeof__(data) _dt;
_dt other_data = (*(info->TYPE_METHOD))(other, @selector(TYPE_METHOD));
if (data == other_data)
case 0:
{
return NSOrderedSame;
}
else if (data < other_data)
{
return NSOrderedAscending;
}
else
{
return NSOrderedDescending;
}
}
BOOL oData;
/*
* If we and the other object both promote to the same type
*/
if (NEXT_ORDER == info->typeNext)
{
int res;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
case 1:
{
char oData;
res = ((NEXT_TYPE)data)
- (*(info->NEXT_METHOD))(other, @selector(NEXT_METHOD));
if (res == 0)
{
return NSOrderedSame;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
else if (res < 0)
case 2:
{
return NSOrderedAscending;
}
else
{
return NSOrderedDescending;
}
}
else
{
NSComparisonResult r;
unsigned char oData;
/*
* We must be promoted to match the other.
*/
r = (*(info->compValue))(other, @selector(compare:), self);
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
case 3:
{
short oData;
if (r == NSOrderedAscending)
{
return NSOrderedDescending;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
if (r == NSOrderedDescending)
case 4:
{
return NSOrderedAscending;
unsigned short oData;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
return r;
case 5:
{
int oData;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
case 6:
{
unsigned int oData;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
case 7:
{
long oData;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
case 8:
{
unsigned long oData;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
case 9:
{
long long oData;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
case 10:
{
unsigned long long oData;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
case 11:
{
float oData;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
case 12:
{
double oData;
(*(info->getValue))(other, @selector(getValue:), (void*)&oData);
if (data == oData)
return NSOrderedSame;
else if (data < oData)
return NSOrderedAscending;
else
return NSOrderedDescending;
}
default:
[NSException raise: NSInvalidArgumentException
format: @"number type value for comparison"];
return NSOrderedSame;
}
}
@ -339,22 +386,47 @@
#endif
}
- (id) copy
{
if (NSShouldRetainWithZone(self, NSDefaultMallocZone()))
return RETAIN(self);
else
return NSCopyObject(self, 0, NSDefaultMallocZone());
}
- (id) copyWithZone: (NSZone*)zone
{
if (NSShouldRetainWithZone(self, zone))
return RETAIN(self);
else
return NSCopyObject(self, 0, zone);
}
// Override these from NSValue
- (void) getValue: (void*)value
{
if (value == 0)
{
[NSException raise: NSInvalidArgumentException
format: @"Cannot copy value into NULL pointer"];
format: @"Cannot copy value into NULL pointer"];
/* NOT REACHED */
}
memcpy(value, &data, objc_sizeof_type([self objCType]));
memcpy(value, &data, objc_sizeof_type(@encode(TYPE_TYPE)));
}
- (const char*) objCType
{
typedef __typeof__(data) _dt;
return @encode(_dt);
return @encode(TYPE_TYPE);
}
- (id) nonretainedObjectValue
{
return (id)(void*)&data;
}
- (void*) pointerValue
{
return (void*)&data;
}
// NSCoding
@ -370,16 +442,14 @@
return [super replacementObjectForPortCoder: aCoder];
}
- (void) encodeWithCoder: coder
- (void) encodeWithCoder: (NSCoder*)coder
{
const char *type = [self objCType];
[coder encodeValueOfObjCType: type at: &data];
[coder encodeValueOfObjCType: @encode(TYPE_TYPE) at: &data];
}
- (id) initWithCoder: coder
- (id) initWithCoder: (NSCoder*)coder
{
const char *type = [self objCType];
[coder decodeValueOfObjCType: type at: &data];
[coder decodeValueOfObjCType: @encode(TYPE_TYPE) at: &data];
return self;
}

File diff suppressed because it is too large Load diff

View file

@ -1777,27 +1777,45 @@ handle_printf_atsign (FILE *stream,
}
/* Returns a new string with the path component given in aString
appended to the receiver. Raises an exception if aString starts with
a '/'. Checks the receiver to see if the last letter is a '/', if it
is not, a '/' is appended before appending aString */
* appended to the receiver.
* Removes trailing separators and multiple separators.
*/
- (NSString*) stringByAppendingPathComponent: (NSString*)aString
{
unsigned length;
unsigned length = [self length];
unsigned aLength = [aString length];
unichar buf[length+aLength+1];
if ([aString length] == 0)
return AUTORELEASE([self copy]);
length = [self length];
if (length == 0)
return AUTORELEASE([aString copy]);
[self getCharacters: buf];
if (aLength > 0)
{
buf[length++] = pathSepChar;
[aString getCharacters: &buf[length]];
}
length += aLength;
while (length > 1 && pathSepMember(buf[length-1]) == YES)
{
length--;
}
aLength = length - 1;
while (aLength > 1)
{
if (pathSepMember(buf[aLength]) == YES)
{
if (pathSepMember(buf[aLength-1]) == YES)
{
unsigned pos;
if (pathSepMember([aString characterAtIndex: 0]) == YES)
[NSException raise: NSGenericException
format: @"attempt to append illegal path component"];
if (pathSepMember([self characterAtIndex: length-1]) == YES)
return [self stringByAppendingString: aString];
else
return [self stringByAppendingFormat: @"%@%@", pathSepString, aString];
for (pos = aLength+1; pos < length; pos++)
{
buf[pos-1] = buf[pos];
}
length--;
}
}
aLength--;
}
return [NSString stringWithCharacters: buf length: length];
}
/* Returns a new string with the path extension given in aString
@ -2192,30 +2210,34 @@ handle_printf_atsign (FILE *stream,
- (NSArray*) pathComponents
{
NSMutableArray *a;
NSArray *r;
NSMutableArray *a;
NSArray *r;
a = [[self componentsSeparatedByString: pathSepString] mutableCopy];
if ([a count] > 0) {
int i;
a = [[self componentsSeparatedByString: pathSepString] mutableCopy];
if ([a count] > 0)
{
int i;
/* If the path began with a '/' then the first path component must
* be a '/' rather than an empty string so that our output could be
* fed into [+pathWithComponents: ]
*/
if ([[a objectAtIndex: 0] length] == 0) {
[a replaceObjectAtIndex: 0 withObject: pathSepString];
/* If the path began with a '/' then the first path component must
* be a '/' rather than an empty string so that our output could be
* fed into [+pathWithComponents: ]
*/
if ([[a objectAtIndex: 0] length] == 0)
{
[a replaceObjectAtIndex: 0 withObject: pathSepString];
}
/* Any empty path components (except a trailing one) must be removed. */
for (i = [a count] - 2; i > 0; i--) {
if ([[a objectAtIndex: i] length] == 0) {
[a removeObjectAtIndex: i];
/* Any empty path components (except a trailing one) must be removed. */
for (i = [a count] - 2; i > 0; i--)
{
if ([[a objectAtIndex: i] length] == 0)
{
[a removeObjectAtIndex: i];
}
}
}
r = [a copy];
RELEASE(a);
return AUTORELEASE(r);
r = [a copy];
RELEASE(a);
return AUTORELEASE(r);
}
- (NSArray*) stringsByAppendingPaths: (NSArray*)paths