1998-09-21 09:22:07 +08:00
|
|
|
/* The implementation of class Object for Objective-C.
|
2009-04-09 23:00:19 +08:00
|
|
|
Copyright (C) 1993, 1994, 1995, 1997, 2002, 2009 Free Software Foundation, Inc.
|
1998-09-21 09:22:07 +08:00
|
|
|
|
Makefile.in, [...]: Replace "GNU CC" with "GCC".
2003-05-23 Nathanael Nerode <neroden@gcc.gnu.org>
* Makefile.in, NXConstStr.m, Object.m, Protocol.m, archive.c,
class.c, encoding.c, gc.c, objc/NXConstStr.h, objc/Object.h,
objc/Protocol.h, objc/encoding.h, objc/hash.h, objc/objc-api.h,
objc/objc-list.h, objc/objc.h, ocjc/runtime.h, objc/sarray.h,
objc/thr.h, objc/typedstream.h: Replace "GNU CC" with "GCC".
From-SVN: r67131
2003-05-24 04:04:58 +08:00
|
|
|
This file is part of GCC.
|
1998-09-21 09:22:07 +08:00
|
|
|
|
Makefile.in, [...]: Replace "GNU CC" with "GCC".
2003-05-23 Nathanael Nerode <neroden@gcc.gnu.org>
* Makefile.in, NXConstStr.m, Object.m, Protocol.m, archive.c,
class.c, encoding.c, gc.c, objc/NXConstStr.h, objc/Object.h,
objc/Protocol.h, objc/encoding.h, objc/hash.h, objc/objc-api.h,
objc/objc-list.h, objc/objc.h, ocjc/runtime.h, objc/sarray.h,
objc/thr.h, objc/typedstream.h: Replace "GNU CC" with "GCC".
From-SVN: r67131
2003-05-24 04:04:58 +08:00
|
|
|
GCC is free software; you can redistribute it and/or modify it
|
1998-09-21 09:22:07 +08:00
|
|
|
under the terms of the GNU General Public License as published by the
|
2009-04-09 23:00:19 +08:00
|
|
|
Free Software Foundation; either version 3, or (at your option) any
|
1998-09-21 09:22:07 +08:00
|
|
|
later version.
|
|
|
|
|
Makefile.in, [...]: Replace "GNU CC" with "GCC".
2003-05-23 Nathanael Nerode <neroden@gcc.gnu.org>
* Makefile.in, NXConstStr.m, Object.m, Protocol.m, archive.c,
class.c, encoding.c, gc.c, objc/NXConstStr.h, objc/Object.h,
objc/Protocol.h, objc/encoding.h, objc/hash.h, objc/objc-api.h,
objc/objc-list.h, objc/objc.h, ocjc/runtime.h, objc/sarray.h,
objc/thr.h, objc/typedstream.h: Replace "GNU CC" with "GCC".
From-SVN: r67131
2003-05-24 04:04:58 +08:00
|
|
|
GCC is distributed in the hope that it will be useful, but WITHOUT
|
1998-09-21 09:22:07 +08:00
|
|
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public
|
|
|
|
License for more details.
|
|
|
|
|
2009-04-09 23:00:19 +08:00
|
|
|
Under Section 7 of GPL version 3, you are granted additional
|
|
|
|
permissions described in the GCC Runtime Library Exception, version
|
|
|
|
3.1, as published by the Free Software Foundation.
|
1998-09-21 09:22:07 +08:00
|
|
|
|
2009-04-09 23:00:19 +08:00
|
|
|
You should have received a copy of the GNU General Public License and
|
|
|
|
a copy of the GCC Runtime Library Exception along with this program;
|
|
|
|
see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
|
|
|
|
<http://www.gnu.org/licenses/>. */
|
1998-09-21 09:22:07 +08:00
|
|
|
|
|
|
|
#include <stdarg.h>
|
2008-11-21 20:07:43 +08:00
|
|
|
#include <errno.h>
|
1998-09-21 09:22:07 +08:00
|
|
|
#include "objc/Object.h"
|
|
|
|
#include "objc/Protocol.h"
|
|
|
|
#include "objc/objc-api.h"
|
|
|
|
|
|
|
|
#define MAX_CLASS_NAME_LEN 256
|
|
|
|
|
|
|
|
@implementation Object
|
|
|
|
|
|
|
|
+ initialize
|
|
|
|
{
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- init
|
|
|
|
{
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ new
|
|
|
|
{
|
|
|
|
return [[self alloc] init];
|
|
|
|
}
|
|
|
|
|
|
|
|
+ alloc
|
|
|
|
{
|
|
|
|
return class_create_instance(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
- free
|
|
|
|
{
|
|
|
|
return object_dispose(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
- copy
|
|
|
|
{
|
|
|
|
return [[self shallowCopy] deepen];
|
|
|
|
}
|
|
|
|
|
|
|
|
- shallowCopy
|
|
|
|
{
|
|
|
|
return object_copy(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
- deepen
|
|
|
|
{
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- deepCopy
|
|
|
|
{
|
|
|
|
return [self copy];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (Class)class
|
|
|
|
{
|
|
|
|
return object_get_class(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (Class)superClass
|
|
|
|
{
|
|
|
|
return object_get_super_class(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (MetaClass)metaClass
|
|
|
|
{
|
|
|
|
return object_get_meta_class(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (const char *)name
|
|
|
|
{
|
|
|
|
return object_get_class_name(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
- self
|
|
|
|
{
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (unsigned int)hash
|
|
|
|
{
|
|
|
|
return (size_t)self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isEqual:anObject
|
|
|
|
{
|
|
|
|
return self==anObject;
|
|
|
|
}
|
|
|
|
|
2008-06-10 20:37:51 +08:00
|
|
|
- (int)compare:(id)anotherObject;
|
1998-09-21 09:22:07 +08:00
|
|
|
{
|
|
|
|
if ([self isEqual:anotherObject])
|
|
|
|
return 0;
|
|
|
|
// Ordering objects by their address is pretty useless,
|
|
|
|
// so subclasses should override this is some useful way.
|
2008-11-21 20:07:43 +08:00
|
|
|
else if ((id)self > anotherObject)
|
1998-09-21 09:22:07 +08:00
|
|
|
return 1;
|
|
|
|
else
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isMetaClass
|
|
|
|
{
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isClass
|
|
|
|
{
|
|
|
|
return object_is_class(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isInstance
|
|
|
|
{
|
|
|
|
return object_is_instance(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isKindOf:(Class)aClassObject
|
|
|
|
{
|
|
|
|
Class class;
|
|
|
|
|
|
|
|
for (class = self->isa; class!=Nil; class = class_get_super_class(class))
|
|
|
|
if (class==aClassObject)
|
|
|
|
return YES;
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isMemberOf:(Class)aClassObject
|
|
|
|
{
|
|
|
|
return self->isa==aClassObject;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isKindOfClassNamed:(const char *)aClassName
|
|
|
|
{
|
|
|
|
Class class;
|
|
|
|
|
|
|
|
if (aClassName!=NULL)
|
|
|
|
for (class = self->isa; class!=Nil; class = class_get_super_class(class))
|
|
|
|
if (!strcmp(class_get_class_name(class), aClassName))
|
|
|
|
return YES;
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)isMemberOfClassNamed:(const char *)aClassName
|
|
|
|
{
|
|
|
|
return ((aClassName!=NULL)
|
|
|
|
&&!strcmp(class_get_class_name(self->isa), aClassName));
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (BOOL)instancesRespondTo:(SEL)aSel
|
|
|
|
{
|
|
|
|
return class_get_instance_method(self, aSel)!=METHOD_NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)respondsTo:(SEL)aSel
|
|
|
|
{
|
|
|
|
return ((object_is_instance(self)
|
|
|
|
?class_get_instance_method(self->isa, aSel)
|
|
|
|
:class_get_class_method(self->isa, aSel))!=METHOD_NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (IMP)instanceMethodFor:(SEL)aSel
|
|
|
|
{
|
|
|
|
return method_get_imp(class_get_instance_method(self, aSel));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Indicates if the receiving class or instance conforms to the given protocol
|
|
|
|
// not usually overridden by subclasses
|
|
|
|
//
|
|
|
|
// Modified 9/5/94 to always search the class object's protocol list, rather
|
|
|
|
// than the meta class.
|
|
|
|
|
|
|
|
+ (BOOL) conformsTo: (Protocol*)aProtocol
|
|
|
|
{
|
Object.m: Fix signed/unsigned warning.
* Object.m: Fix signed/unsigned warning.
* Protocol.m: Likewise.
* archive.c: Always include stdlib.h.
(objc_read_short, objc_read_unsigned_short, objc_read_int,
objc_read_long, __objc_read_nbyte_uint, __objc_read_nbyte_ulong):
Fix signed/unsigned warning.
(objc_write_type, objc_read_type, objc_write_types,
objc_read_types): Ensure ctype 8-bit safety.
(__objc_no_write, __objc_no_read): Mark unused parameters.
* class.c (class_table_setup): Specify void arg.
* encoding.c (atoi, objc_sizeof_type, objc_alignof_type,
objc_skip_typespec, objc_skip_offset,
objc_layout_structure_next_member): Ensure ctype 8-bit safety.
(objc_layout_structure_next_member): Ensure variables are
initialized.
* gc.c (__objc_generate_gc_type_description,
class_ivar_set_gcinvisible): Mark unused parameters.
* init.c (__objc_send_load, __objc_destroy_class_tree_node): Mark
unused parameters.
(__objc_init_protocols) Fix signed/unsigned warning.
* nil_method.c (nil_method): Mark unused parameters.
* thr.h (objc_thread_callback): Specify void arg.
* sarray.c (sarray_new, sarray_realloc, sarray_free): Fix
signed/unsigned warning.
(sarray_free): Fix formatting.
* selector.c (sel_types_match): Ensure ctype 8-bit safety.
* sendmsg.c (__objc_init_install_dtable) Mark unused parameters.
From-SVN: r54649
2002-06-16 01:50:14 +08:00
|
|
|
size_t i;
|
1998-09-21 09:22:07 +08:00
|
|
|
struct objc_protocol_list* proto_list;
|
|
|
|
id parent;
|
|
|
|
|
|
|
|
for (proto_list = ((Class)self)->protocols;
|
|
|
|
proto_list; proto_list = proto_list->next)
|
|
|
|
{
|
|
|
|
for (i=0; i < proto_list->count; i++)
|
|
|
|
{
|
|
|
|
if ([proto_list->list[i] conformsTo: aProtocol])
|
|
|
|
return YES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((parent = [self superClass]))
|
|
|
|
return [parent conformsTo: aProtocol];
|
|
|
|
else
|
|
|
|
return NO;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL) conformsTo: (Protocol*)aProtocol
|
|
|
|
{
|
|
|
|
return [[self class] conformsTo:aProtocol];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (IMP)methodFor:(SEL)aSel
|
|
|
|
{
|
|
|
|
return (method_get_imp(object_is_instance(self)
|
|
|
|
?class_get_instance_method(self->isa, aSel)
|
|
|
|
:class_get_class_method(self->isa, aSel)));
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (struct objc_method_description *)descriptionForInstanceMethod:(SEL)aSel
|
|
|
|
{
|
|
|
|
return ((struct objc_method_description *)
|
|
|
|
class_get_instance_method(self, aSel));
|
|
|
|
}
|
|
|
|
|
|
|
|
- (struct objc_method_description *)descriptionForMethod:(SEL)aSel
|
|
|
|
{
|
|
|
|
return ((struct objc_method_description *)
|
|
|
|
(object_is_instance(self)
|
|
|
|
?class_get_instance_method(self->isa, aSel)
|
|
|
|
:class_get_class_method(self->isa, aSel)));
|
|
|
|
}
|
|
|
|
|
|
|
|
- perform:(SEL)aSel
|
|
|
|
{
|
|
|
|
IMP msg = objc_msg_lookup(self, aSel);
|
|
|
|
if (!msg)
|
|
|
|
return [self error:"invalid selector passed to %s", sel_get_name(_cmd)];
|
|
|
|
return (*msg)(self, aSel);
|
|
|
|
}
|
|
|
|
|
|
|
|
- perform:(SEL)aSel with:anObject
|
|
|
|
{
|
|
|
|
IMP msg = objc_msg_lookup(self, aSel);
|
|
|
|
if (!msg)
|
|
|
|
return [self error:"invalid selector passed to %s", sel_get_name(_cmd)];
|
|
|
|
return (*msg)(self, aSel, anObject);
|
|
|
|
}
|
|
|
|
|
|
|
|
- perform:(SEL)aSel with:anObject1 with:anObject2
|
|
|
|
{
|
|
|
|
IMP msg = objc_msg_lookup(self, aSel);
|
|
|
|
if (!msg)
|
|
|
|
return [self error:"invalid selector passed to %s", sel_get_name(_cmd)];
|
|
|
|
return (*msg)(self, aSel, anObject1, anObject2);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (retval_t)forward:(SEL)aSel :(arglist_t)argFrame
|
|
|
|
{
|
2002-06-21 22:36:37 +08:00
|
|
|
(void) argFrame; /* UNUSED */
|
1998-09-21 09:22:07 +08:00
|
|
|
return (retval_t)[self doesNotRecognize: aSel];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (retval_t)performv:(SEL)aSel :(arglist_t)argFrame
|
|
|
|
{
|
|
|
|
return objc_msg_sendv(self, aSel, argFrame);
|
|
|
|
}
|
|
|
|
|
|
|
|
+ poseAs:(Class)aClassObject
|
|
|
|
{
|
|
|
|
return class_pose_as(self, aClassObject);
|
|
|
|
}
|
|
|
|
|
|
|
|
- (Class)transmuteClassTo:(Class)aClassObject
|
|
|
|
{
|
|
|
|
if (object_is_instance(self))
|
|
|
|
if (class_is_class(aClassObject))
|
|
|
|
if (class_get_instance_size(aClassObject)==class_get_instance_size(isa))
|
|
|
|
if ([self isKindOf:aClassObject])
|
|
|
|
{
|
|
|
|
Class old_isa = isa;
|
|
|
|
isa = aClassObject;
|
|
|
|
return old_isa;
|
|
|
|
}
|
|
|
|
return nil;
|
|
|
|
}
|
|
|
|
|
|
|
|
- subclassResponsibility:(SEL)aSel
|
|
|
|
{
|
|
|
|
return [self error:"subclass should override %s", sel_get_name(aSel)];
|
|
|
|
}
|
|
|
|
|
|
|
|
- notImplemented:(SEL)aSel
|
|
|
|
{
|
|
|
|
return [self error:"method %s not implemented", sel_get_name(aSel)];
|
|
|
|
}
|
|
|
|
|
|
|
|
- shouldNotImplement:(SEL)aSel
|
|
|
|
{
|
|
|
|
return [self error:"%s should not implement %s",
|
|
|
|
object_get_class_name(self), sel_get_name(aSel)];
|
|
|
|
}
|
|
|
|
|
|
|
|
- doesNotRecognize:(SEL)aSel
|
|
|
|
{
|
|
|
|
return [self error:"%s does not recognize %s",
|
|
|
|
object_get_class_name(self), sel_get_name(aSel)];
|
|
|
|
}
|
|
|
|
|
|
|
|
- error:(const char *)aString, ...
|
|
|
|
{
|
|
|
|
#define FMT "error: %s (%s)\n%s\n"
|
|
|
|
char fmt[(strlen((char*)FMT)+strlen((char*)object_get_class_name(self))
|
|
|
|
+((aString!=NULL)?strlen((char*)aString):0)+8)];
|
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
sprintf(fmt, FMT, object_get_class_name(self),
|
|
|
|
object_is_instance(self)?"instance":"class",
|
|
|
|
(aString!=NULL)?aString:"");
|
|
|
|
va_start(ap, aString);
|
|
|
|
objc_verror(self, OBJC_ERR_UNKNOWN, fmt, ap);
|
|
|
|
va_end(ap);
|
|
|
|
return nil;
|
|
|
|
#undef FMT
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (int)version
|
|
|
|
{
|
|
|
|
return class_get_version(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
+ setVersion:(int)aVersion
|
|
|
|
{
|
|
|
|
class_set_version(self, aVersion);
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ (int)streamVersion: (TypedStream*)aStream
|
|
|
|
{
|
|
|
|
if (aStream->mode == OBJC_READONLY)
|
|
|
|
return objc_get_stream_class_version (aStream, self);
|
|
|
|
else
|
|
|
|
return class_get_version (self);
|
|
|
|
}
|
|
|
|
|
|
|
|
// These are used to write or read the instance variables
|
|
|
|
// declared in this particular part of the object. Subclasses
|
|
|
|
// should extend these, by calling [super read/write: aStream]
|
|
|
|
// before doing their own archiving. These methods are private, in
|
|
|
|
// the sense that they should only be called from subclasses.
|
|
|
|
|
|
|
|
- read: (TypedStream*)aStream
|
|
|
|
{
|
2002-06-21 22:36:37 +08:00
|
|
|
(void) aStream; /* UNUSED */
|
1998-09-21 09:22:07 +08:00
|
|
|
// [super read: aStream];
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- write: (TypedStream*)aStream
|
|
|
|
{
|
2002-06-21 22:36:37 +08:00
|
|
|
(void) aStream; /* UNUSED */
|
1998-09-21 09:22:07 +08:00
|
|
|
// [super write: aStream];
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- awake
|
|
|
|
{
|
|
|
|
// [super awake];
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|