1 /* GNU Objective C Runtime ivar related functions.
2 Copyright (C) 2010 Free Software Foundation, Inc.
3 Contributed by Nicola Pero
5 This file is part of GCC.
7 GCC is free software; you can redistribute it and/or modify it under the
8 terms of the GNU General Public License as published by the Free Software
9 Foundation; either version 3, or (at your option) any later version.
11 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
12 WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
13 FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
16 Under Section 7 of GPL version 3, you are granted additional
17 permissions described in the GCC Runtime Library Exception, version
18 3.1, as published by the Free Software Foundation.
20 You should have received a copy of the GNU General Public License and
21 a copy of the GCC Runtime Library Exception along with this program;
22 see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
23 <http://www.gnu.org/licenses/>. */
25 #include "objc-private/common.h"
26 #include "objc/runtime.h"
27 #include "objc-private/module-abi-8.h" /* For runtime structures */
29 #include "objc-private/runtime.h" /* the kitchen sink */
30 #include <string.h> /* For strcmp. */
31 #include <stdlib.h> /* For malloc. */
34 class_getInstanceVariable (Class class_
, const char *name
)
36 if (class_
!= Nil
&& name
!= NULL
&& ! CLS_IS_IN_CONSTRUCTION (class_
))
40 struct objc_ivar_list
*ivars
= class_
->ivars
;
45 for (i
= 0; i
< ivars
->ivar_count
; i
++)
47 struct objc_ivar
*ivar
= &(ivars
->ivar_list
[i
]);
49 if (!strcmp (ivar
->ivar_name
, name
))
55 class_
= class_getSuperclass (class_
);
62 class_getClassVariable (Class class_
, const char *name
)
67 /* Logically, since a class is an instance of its meta-class, and
68 since its class methods are the instance methods of the
69 meta-class, class variables should be instance variables of the
70 meta-class. That is different from the normal use of having
71 'static' variables in the class implementation file, because
72 every class would have its own variables.
74 Anyway, it is all speculative at this stage, but if we get class
75 variables in Objective-C, it is conceivable that this
76 implementation should work. */
77 return class_getInstanceVariable (class_
->class_pointer
, name
);
81 object_getIndexedIvars (id object
)
87 return (void *)(((char *)object
)
88 + object
->class_pointer
->instance_size
);
93 object_getInstanceVariable (id object
, const char *name
, void **returnValue
)
95 if (object
== nil
|| name
== NULL
)
99 struct objc_ivar
* variable
= class_getInstanceVariable (object
->class_pointer
, name
);
101 if (variable
!= NULL
&& returnValue
!= NULL
)
103 char *location
= (char *)object
+ variable
->ivar_offset
;
105 *returnValue
= *((id
*)location
);
113 object_setInstanceVariable (id object
, const char *name
, void *newValue
)
115 if (object
== nil
|| name
== NULL
)
119 struct objc_ivar
* variable
= class_getInstanceVariable (object
->class_pointer
, name
);
121 if (variable
!= NULL
)
123 char *location
= (char *)object
+ variable
->ivar_offset
;
125 *((id
*)location
) = (id
)newValue
;
132 id
object_getIvar (id object
, struct objc_ivar
* variable
)
134 if (object
== nil
|| variable
== NULL
)
138 char *location
= (char *)object
+ variable
->ivar_offset
;
140 return *((id
*)location
);
144 void object_setIvar (id object
, struct objc_ivar
* variable
, id value
)
146 if (object
== nil
|| variable
== NULL
)
150 char *location
= (char *)object
+ variable
->ivar_offset
;
152 *((id
*)location
) = value
;
156 const char * ivar_getName (struct objc_ivar
* variable
)
158 if (variable
== NULL
)
161 return variable
->ivar_name
;
164 ptrdiff_t ivar_getOffset (struct objc_ivar
* variable
)
166 if (variable
== NULL
)
169 return (ptrdiff_t)(variable
->ivar_offset
);
172 const char * ivar_getTypeEncoding (struct objc_ivar
* variable
)
174 if (variable
== NULL
)
177 return variable
->ivar_type
;
180 struct objc_ivar
** class_copyIvarList (Class class_
, unsigned int *numberOfReturnedIvars
)
182 unsigned int count
= 0;
183 struct objc_ivar
**returnValue
= NULL
;
184 struct objc_ivar_list
* ivar_list
;
186 if (class_
== Nil
|| CLS_IS_IN_CONSTRUCTION (class_
))
188 if (numberOfReturnedIvars
)
189 *numberOfReturnedIvars
= 0;
193 /* Count how many ivars we have. */
194 ivar_list
= class_
->ivars
;
195 count
= ivar_list
->ivar_count
;
201 /* Allocate enough memory to hold them. */
202 returnValue
= (struct objc_ivar
**)(malloc (sizeof (struct objc_ivar
*) * (count
+ 1)));
204 /* Copy the ivars. */
205 for (i
= 0; i
< count
; i
++)
207 returnValue
[i
] = &(ivar_list
->ivar_list
[i
]);
210 returnValue
[i
] = NULL
;
213 if (numberOfReturnedIvars
)
214 *numberOfReturnedIvars
= count
;
220 class_addIvar (Class class_
, const char * ivar_name
, size_t size
,
221 unsigned char alignment
, const char *type
)
223 struct objc_ivar_list
*ivars
;
226 || (! CLS_IS_IN_CONSTRUCTION (class_
))
228 || (strcmp (ivar_name
, "") == 0)
233 /* Check if the class has an instance variable with that name
235 ivars
= class_
->ivars
;
241 for (i
= 0; i
< ivars
->ivar_count
; i
++)
243 struct objc_ivar
*ivar
= &(ivars
->ivar_list
[i
]);
245 if (strcmp (ivar
->ivar_name
, ivar_name
) == 0)
252 /* Ok, no direct ivars. Check superclasses. */
253 if (class_getInstanceVariable (objc_getClass ((char *)(class_
->super_class
)),
257 /* Good. Create space for the new instance variable. */
260 int ivar_count
= ivars
->ivar_count
+ 1;
261 int new_size
= sizeof (struct objc_ivar_list
)
262 + (ivar_count
- 1) * sizeof (struct objc_ivar
);
264 ivars
= (struct objc_ivar_list
*) objc_realloc (ivars
, new_size
);
265 ivars
->ivar_count
= ivar_count
;
266 class_
->ivars
= ivars
;
270 int new_size
= sizeof (struct objc_ivar_list
);
272 ivars
= (struct objc_ivar_list
*) objc_malloc (new_size
);
273 ivars
->ivar_count
= 1;
274 class_
->ivars
= ivars
;
277 /* Now ivars is set to a list of instance variables of the right
280 struct objc_ivar
*ivar
= &(ivars
->ivar_list
[ivars
->ivar_count
- 1]);
283 ivar
->ivar_name
= objc_malloc (strlen (ivar_name
) + 1);
284 strcpy ((char *)ivar
->ivar_name
, ivar_name
);
286 ivar
->ivar_type
= objc_malloc (strlen (type
) + 1);
287 strcpy ((char *)ivar
->ivar_type
, type
);
289 /* The new instance variable is placed at the end of the existing
290 instance_size, at the first byte that is aligned with
292 misalignment
= class_
->instance_size
% alignment
;
294 if (misalignment
== 0)
295 ivar
->ivar_offset
= class_
->instance_size
;
297 ivar
->ivar_offset
= class_
->instance_size
- misalignment
+ alignment
;
299 class_
->instance_size
= ivar
->ivar_offset
+ objc_sizeof_type (ivar
->ivar_type
);
307 property_getName (struct objc_property
* property
__attribute__ ((__unused__
)))
309 if (property
== NULL
)
313 /* The current ABI does not have any information on properties. */
318 property_getAttributes (struct objc_property
* property
__attribute__ ((__unused__
)))
320 if (property
== NULL
)
324 /* The current ABI does not have any information on properties. */
328 struct objc_property
*
329 class_getProperty (Class class_
__attribute__ ((__unused__
)),
330 const char *propertyName
__attribute__ ((__unused__
)))
332 if (class_
== NULL
|| propertyName
== NULL
)
336 /* The current ABI does not have any information on class properties. */
340 struct objc_property
**
341 class_copyPropertyList (Class class_
__attribute__ ((__unused__
)),
342 unsigned int *numberOfReturnedProperties
__attribute__ ((__unused__
)))
346 if (numberOfReturnedProperties
)
347 *numberOfReturnedProperties
= 0;
352 /* The current ABI does not have any information on class properties. */
353 if (numberOfReturnedProperties
)
354 *numberOfReturnedProperties
= 0;
360 class_getIvarLayout (Class class_
__attribute__ ((__unused__
)))
366 class_getWeakIvarLayout (Class class_
__attribute__ ((__unused__
)))
372 class_setIvarLayout (Class class_
__attribute__ ((__unused__
)),
373 const char *layout
__attribute__ ((__unused__
)))
379 class_setWeakIvarLayout (Class class_
__attribute__ ((__unused__
)),
380 const char *layout
__attribute__ ((__unused__
)))