Skip to content

Commit 06adc24

Browse files
author
theraven
committed
Finished implementing the new runtime APIs.
git-svn-id: svn+ssh://svn.gna.org/svn/gnustep/libs/libobjc2/trunk@33964 72102866-910b-0410-8b05-ffd578937521
1 parent e499e93 commit 06adc24

9 files changed

+506
-7
lines changed

ANNOUNCE

+2-1
Original file line numberDiff line numberDiff line change
@@ -6,12 +6,13 @@ libobjc2). This runtime was designed to support the features of Objective-C 2
66
for use with GNUstep and other Objective-C programs. Highlights of this
77
release include:
88

9+
- Compatibility with the new runtime APIs introduced with Mac OS X 10.7.
910
- Support for small objects (ones hidden inside a pointer). On 32-bit systems,
1011
the runtime permits one small object class, on 64-bit systems it permits 4.
1112
- Support for prototype-style object orientation. You can now add methods, as
1213
well as associated references, to individual objects, as well as cloning
1314
them. The runtime now supports everything required for the JavaScript object
14-
model.
15+
model, including the ability to use blocks as methods.
1516

1617
You may obtain the code for this release from subversion at the following
1718
subversion branch:

Test/PropertyIntrospectionTest.m

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
#import <objc/runtime.h>
2+
#include <stdio.h>
3+
#include <string.h>
4+
#include <assert.h>
5+
6+
@interface Foo
7+
@property (getter=bar, setter=setBar:, nonatomic, copy) id foo;
8+
@end
9+
@interface Foo(Bar)
10+
-(id)bar;
11+
-(void)setBar:(id)b;
12+
@end
13+
@implementation Foo
14+
@synthesize foo;
15+
@end
16+
17+
int main(void)
18+
{
19+
objc_property_t p = class_getProperty(objc_getClass("Foo"), "foo");
20+
unsigned int count;
21+
objc_property_attribute_t *l = property_copyAttributeList(p, &count);
22+
for (unsigned int i=0 ; i<count ; i++)
23+
{
24+
switch (l[i].name[0])
25+
{
26+
case 'T': assert(0==strcmp(l[i].value, "@")); break;
27+
case 'C': assert(0==strcmp(l[i].value, "")); break;
28+
case 'N': assert(0==strcmp(l[i].value, "")); break;
29+
case 'G': assert(0==strcmp(l[i].value, "bar")); break;
30+
case 'S': assert(0==strcmp(l[i].value, "setBar:")); break;
31+
case 'B': assert(0==strcmp(l[i].value, "foo")); break;
32+
}
33+
}
34+
assert(0 == property_copyAttributeList(0, &count));
35+
return 0;
36+
}

Test/ProtocolCreation.m

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
#import <objc/runtime.h>
2+
#include <stdio.h>
3+
#include <string.h>
4+
#include <assert.h>
5+
6+
@protocol Test2 @end
7+
8+
int main(void)
9+
{
10+
Protocol *p = objc_allocateProtocol("Test");
11+
protocol_addMethodDescription(p, @selector(someMethod), "@:", YES, NO);
12+
assert(objc_getProtocol("Test2"));
13+
protocol_addProtocol(p, objc_getProtocol("Test2"));
14+
objc_property_attribute_t attrs[] = { {"T", "@" } };
15+
protocol_addProperty(p, "foo", attrs, 1, YES, YES);
16+
objc_registerProtocol(p);
17+
Protocol *p1 = objc_getProtocol("Test");
18+
assert(p == p1);
19+
struct objc_method_description d = protocol_getMethodDescription(p1, @selector(someMethod), YES, NO);
20+
assert(strcmp(sel_getName(d.name), "someMethod") == 0);
21+
assert(strcmp((d.types), "@:") == 0);
22+
assert(protocol_conformsToProtocol(p1, objc_getProtocol("Test2")));
23+
unsigned int count;
24+
objc_property_t *props = protocol_copyPropertyList(p1, &count);
25+
assert(count == 1);
26+
assert(strcmp("T@,Vfoo", property_getAttributes(*props)) == 0);
27+
return 0;
28+
}

class_table.c

+11
Original file line numberDiff line numberDiff line change
@@ -460,6 +460,17 @@ int objc_getClassList(Class *buffer, int bufferLen)
460460
}
461461
return count;
462462
}
463+
Class *objc_copyClassList(unsigned int *outCount)
464+
{
465+
int count = class_table->table_used;
466+
Class *buffer = calloc(sizeof(Class), count);
467+
if (NULL != outCount)
468+
{
469+
*outCount = count;
470+
}
471+
objc_getClassList(buffer, count);
472+
return buffer;
473+
}
463474

464475
Class class_getSuperclass(Class cls)
465476
{

encoding2.c

+1
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@ const char *objc_skip_argspec(const char *type)
4141

4242
PRIVATE size_t lengthOfTypeEncoding(const char *types)
4343
{
44+
if ((NULL == types) || ('\0' == types[0])) { return 0; }
4445
const char *end = objc_skip_typespec(types);
4546
size_t length = end - types;
4647
return length;

objc/runtime.h

+91
Original file line numberDiff line numberDiff line change
@@ -163,6 +163,26 @@ struct objc_method_description
163163
const char *types;
164164
};
165165

166+
/**
167+
* The objc_property_attribute_t type is used to store attributes for
168+
* properties. This is used to store a decomposed version of the property
169+
* encoding, with each flag stored in the name and each value in the value.
170+
*
171+
* All of the strings that these refer to are internal to the runtime and
172+
* should not be freed.
173+
*/
174+
typedef struct
175+
{
176+
/**
177+
* The flag that this attribute describes. All current flags are single characters,
178+
*/
179+
const char *name;
180+
/**
181+
*/
182+
const char *value;
183+
} objc_property_attribute_t;
184+
185+
166186

167187
#ifndef YES
168188
# define YES ((BOOL)1)
@@ -520,6 +540,12 @@ id objc_getClass(const char *name);
520540
* Otherwise, it copies classes and returns the number copied.
521541
*/
522542
int objc_getClassList(Class *buffer, int bufferLen);
543+
/**
544+
* Returns a copy of the list of all classes in the system. The caller is
545+
* responsible for freeing this list. The number of classes is returned in the
546+
* parameter.
547+
*/
548+
Class *objc_copyClassList(unsigned int *outCount);
523549

524550
/**
525551
* Returns the metaclass with the specified name. This is equivalent to
@@ -544,6 +570,41 @@ id objc_lookUpClass(const char *name);
544570
* Returns the protocol with the specified name.
545571
*/
546572
Protocol *objc_getProtocol(const char *name);
573+
/**
574+
* Allocates a new protocol. This returns NULL if a protocol with the same
575+
* name already exists in the system.
576+
*
577+
* Protocols are immutable after they have been registered, so may only be
578+
* modified between calling this function and calling objc_registerProtocol().
579+
*/
580+
Protocol *objc_allocateProtocol(const char *name);
581+
/**
582+
* Registers a protocol with the runtime. After this point, the protocol may
583+
* not be modified.
584+
*/
585+
void objc_registerProtocol(Protocol *proto);
586+
/**
587+
* Adds a method to the protocol.
588+
*/
589+
void protocol_addMethodDescription(Protocol *aProtocol,
590+
SEL name,
591+
const char *types,
592+
BOOL isRequiredMethod,
593+
BOOL isInstanceMethod);
594+
/**
595+
* Adds a protocol to the protocol.
596+
*/
597+
void protocol_addProtocol(Protocol *aProtocol, Protocol *addition);
598+
/**
599+
* Adds a property to the protocol.
600+
*/
601+
void protocol_addProperty(Protocol *aProtocol,
602+
const char *name,
603+
const objc_property_attribute_t *attributes,
604+
unsigned int attributeCount,
605+
BOOL isRequiredProperty,
606+
BOOL isInstanceProperty);
607+
547608

548609
/**
549610
* Registers a new class and its metaclass with the runtime. This function
@@ -600,6 +661,36 @@ const char *property_getName(objc_property_t property);
600661
*/
601662
const char *property_getAttributes(objc_property_t property);
602663

664+
/**
665+
* Returns an array of attributes for this property.
666+
*/
667+
objc_property_attribute_t *property_copyAttributeList(objc_property_t property,
668+
unsigned int *outCount);
669+
/**
670+
* Adds a property to the class, given a specified set of attributes. Note
671+
* that this only sets the property metadata. The property accessor methods
672+
* must already be created.
673+
*/
674+
BOOL class_addProperty(Class cls,
675+
const char *name,
676+
const objc_property_attribute_t *attributes,
677+
unsigned int attributeCount);
678+
679+
/**
680+
* Replaces property metadata. If the property does not exist, then this is
681+
* equivalent to calling class_addProperty().
682+
*/
683+
void class_replaceProperty(Class cls,
684+
const char *name,
685+
const objc_property_attribute_t *attributes,
686+
unsigned int attributeCount);
687+
688+
/**
689+
* Returns a copy of a single attribute.
690+
*/
691+
char *property_copyAttributeValue(objc_property_t property,
692+
const char *attributeName);
693+
603694
/**
604695
* Testswhether a protocol conforms to another protocol.
605696
*/

properties.h

+8-3
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
#include "visibility.h"
12

23
enum PropertyAttributeKind
34
{
@@ -55,7 +56,7 @@ struct objc_property
5556
* Attributes for this property. Made by ORing together
5657
* PropertyAttributeKinds.
5758
*/
58-
const char attributes;
59+
char attributes;
5960
/**
6061
* Flag set if the property is synthesized.
6162
*/
@@ -88,8 +89,7 @@ struct objc_property_list
8889
*/
8990
int count;
9091
/*
91-
* UNUSED. In future, categories will be allowed to add properties and
92-
* then this will be used to link the declarations together.
92+
* The next property in a linked list.
9393
*/
9494
struct objc_property_list *next;
9595
/**
@@ -98,3 +98,8 @@ struct objc_property_list
9898
struct objc_property properties[];
9999
};
100100

101+
/**
102+
* Constructs a property description from a list of attributes.
103+
*/
104+
PRIVATE struct objc_property propertyFromAttrs(const objc_property_attribute_t *attributes,
105+
unsigned int attributeCount);

0 commit comments

Comments
 (0)