123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377 |
- /* GNU Objective C Runtime ivar related functions.
- Copyright (C) 2010-2015 Free Software Foundation, Inc.
- Contributed by Nicola Pero
- This file is part of GCC.
- GCC is free software; you can redistribute it and/or modify it under the
- terms of the GNU General Public License as published by the Free Software
- Foundation; either version 3, or (at your option) any later version.
- GCC 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 General Public License for more
- details.
- 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.
- 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/>. */
- #include "objc-private/common.h"
- #include "objc/runtime.h"
- #include "objc-private/module-abi-8.h" /* For runtime structures */
- #include "objc/thr.h"
- #include "objc-private/runtime.h" /* the kitchen sink */
- #include <string.h> /* For strcmp. */
- #include <stdlib.h> /* For malloc. */
- struct objc_ivar *
- class_getInstanceVariable (Class class_, const char *name)
- {
- if (class_ != Nil && name != NULL && ! CLS_IS_IN_CONSTRUCTION (class_))
- {
- while (class_ != Nil)
- {
- struct objc_ivar_list *ivars = class_->ivars;
- if (ivars != NULL)
- {
- int i;
-
- for (i = 0; i < ivars->ivar_count; i++)
- {
- struct objc_ivar *ivar = &(ivars->ivar_list[i]);
-
- if (!strcmp (ivar->ivar_name, name))
- return ivar;
- }
- }
- class_ = class_getSuperclass (class_);
- }
- }
- return NULL;
- }
- struct objc_ivar *
- class_getClassVariable (Class class_, const char *name)
- {
- if (class_ == Nil)
- return NULL;
- /* Logically, since a class is an instance of its meta-class, and
- since its class methods are the instance methods of the
- meta-class, class variables should be instance variables of the
- meta-class. That is different from the normal use of having
- 'static' variables in the class implementation file, because
- every class would have its own variables.
- Anyway, it is all speculative at this stage, but if we get class
- variables in Objective-C, it is conceivable that this
- implementation should work. */
- return class_getInstanceVariable (class_->class_pointer, name);
- }
- void *
- object_getIndexedIvars (id object)
- {
- if (object == nil)
- return NULL;
- else
- return (void *)(((char *)object)
- + object->class_pointer->instance_size);
- }
- struct objc_ivar *
- object_getInstanceVariable (id object, const char *name, void **returnValue)
- {
- if (object == nil || name == NULL)
- return NULL;
- else
- {
- struct objc_ivar * variable = class_getInstanceVariable (object->class_pointer, name);
- if (variable != NULL && returnValue != NULL)
- {
- char *location = (char *)object + variable->ivar_offset;
-
- *returnValue = *((id *)location);
- }
- return variable;
- }
- }
- struct objc_ivar *
- object_setInstanceVariable (id object, const char *name, void *newValue)
- {
- if (object == nil || name == NULL)
- return NULL;
- else
- {
- struct objc_ivar * variable = class_getInstanceVariable (object->class_pointer, name);
- if (variable != NULL)
- {
- char *location = (char *)object + variable->ivar_offset;
-
- *((id *)location) = (id)newValue;
- }
- return variable;
- }
- }
- id object_getIvar (id object, struct objc_ivar * variable)
- {
- if (object == nil || variable == NULL)
- return nil;
- else
- {
- char *location = (char *)object + variable->ivar_offset;
- return *((id *)location);
- }
- }
- void object_setIvar (id object, struct objc_ivar * variable, id value)
- {
- if (object == nil || variable == NULL)
- return;
- else
- {
- char *location = (char *)object + variable->ivar_offset;
- *((id *)location) = value;
- }
- }
- const char * ivar_getName (struct objc_ivar * variable)
- {
- if (variable == NULL)
- return NULL;
- return variable->ivar_name;
- }
- ptrdiff_t ivar_getOffset (struct objc_ivar * variable)
- {
- if (variable == NULL)
- return 0;
- return (ptrdiff_t)(variable->ivar_offset);
- }
- const char * ivar_getTypeEncoding (struct objc_ivar * variable)
- {
- if (variable == NULL)
- return NULL;
- return variable->ivar_type;
- }
- struct objc_ivar ** class_copyIvarList (Class class_, unsigned int *numberOfReturnedIvars)
- {
- unsigned int count = 0;
- struct objc_ivar **returnValue = NULL;
- struct objc_ivar_list* ivar_list;
- if (class_ == Nil || CLS_IS_IN_CONSTRUCTION (class_) || !class_->ivars)
- {
- if (numberOfReturnedIvars)
- *numberOfReturnedIvars = 0;
- return NULL;
- }
-
- /* Count how many ivars we have. */
- ivar_list = class_->ivars;
- count = ivar_list->ivar_count;
- if (count != 0)
- {
- unsigned int i = 0;
-
- /* Allocate enough memory to hold them. */
- returnValue = (struct objc_ivar **)(malloc (sizeof (struct objc_ivar *) * (count + 1)));
-
- /* Copy the ivars. */
- for (i = 0; i < count; i++)
- returnValue[i] = &(ivar_list->ivar_list[i]);
-
- returnValue[i] = NULL;
- }
-
- if (numberOfReturnedIvars)
- *numberOfReturnedIvars = count;
- return returnValue;
- }
- BOOL
- class_addIvar (Class class_, const char * ivar_name, size_t size,
- unsigned char log_2_of_alignment, const char *type)
- {
- struct objc_ivar_list *ivars;
- if (class_ == Nil
- || (! CLS_IS_IN_CONSTRUCTION (class_))
- || ivar_name == NULL
- || (strcmp (ivar_name, "") == 0)
- || size == 0
- || type == NULL)
- return NO;
- /* Check if the class has an instance variable with that name
- already. */
- ivars = class_->ivars;
- if (ivars != NULL)
- {
- int i;
-
- for (i = 0; i < ivars->ivar_count; i++)
- {
- struct objc_ivar *ivar = &(ivars->ivar_list[i]);
-
- if (strcmp (ivar->ivar_name, ivar_name) == 0)
- return NO;
- }
- }
- /* Ok, no direct ivars. Check superclasses. */
- if (class_getInstanceVariable (objc_getClass ((char *)(class_->super_class)),
- ivar_name))
- return NO;
- /* Good. Create space for the new instance variable. */
- if (ivars)
- {
- int ivar_count = ivars->ivar_count + 1;
- int new_size = sizeof (struct objc_ivar_list)
- + (ivar_count - 1) * sizeof (struct objc_ivar);
-
- ivars = (struct objc_ivar_list*) objc_realloc (ivars, new_size);
- ivars->ivar_count = ivar_count;
- class_->ivars = ivars;
- }
- else
- {
- int new_size = sizeof (struct objc_ivar_list);
-
- ivars = (struct objc_ivar_list*) objc_malloc (new_size);
- ivars->ivar_count = 1;
- class_->ivars = ivars;
- }
-
- /* Now ivars is set to a list of instance variables of the right
- size. */
- {
- struct objc_ivar *ivar = &(ivars->ivar_list[ivars->ivar_count - 1]);
- unsigned int alignment = 1 << log_2_of_alignment;
- int misalignment;
-
- ivar->ivar_name = objc_malloc (strlen (ivar_name) + 1);
- strcpy ((char *)ivar->ivar_name, ivar_name);
- ivar->ivar_type = objc_malloc (strlen (type) + 1);
- strcpy ((char *)ivar->ivar_type, type);
- /* The new instance variable is placed at the end of the existing
- instance_size, at the first byte that is aligned with
- alignment. */
- misalignment = class_->instance_size % alignment;
-
- if (misalignment == 0)
- ivar->ivar_offset = class_->instance_size;
- else
- ivar->ivar_offset = class_->instance_size - misalignment + alignment;
-
- class_->instance_size = ivar->ivar_offset + size;
- }
-
- return YES;
- }
- const char *
- property_getName (struct objc_property * property __attribute__ ((__unused__)))
- {
- if (property == NULL)
- return NULL;
- /* TODO: New ABI. */
- /* The current ABI does not have any information on properties. */
- return NULL;
- }
- const char *
- property_getAttributes (struct objc_property * property __attribute__ ((__unused__)))
- {
- if (property == NULL)
- return NULL;
- /* TODO: New ABI. */
- /* The current ABI does not have any information on properties. */
- return NULL;
- }
- struct objc_property *
- class_getProperty (Class class_ __attribute__ ((__unused__)),
- const char *propertyName __attribute__ ((__unused__)))
- {
- if (class_ == NULL || propertyName == NULL)
- return NULL;
- /* TODO: New ABI. */
- /* The current ABI does not have any information on class properties. */
- return NULL;
- }
- struct objc_property **
- class_copyPropertyList (Class class_ __attribute__ ((__unused__)),
- unsigned int *numberOfReturnedProperties __attribute__ ((__unused__)))
- {
- if (class_ == Nil)
- {
- if (numberOfReturnedProperties)
- *numberOfReturnedProperties = 0;
- return NULL;
- }
- /* TODO: New ABI. */
- /* The current ABI does not have any information on class properties. */
- if (numberOfReturnedProperties)
- *numberOfReturnedProperties = 0;
- return NULL;
- }
- const char *
- class_getIvarLayout (Class class_ __attribute__ ((__unused__)))
- {
- return NULL;
- }
- const char *
- class_getWeakIvarLayout (Class class_ __attribute__ ((__unused__)))
- {
- return NULL;
- }
- void
- class_setIvarLayout (Class class_ __attribute__ ((__unused__)),
- const char *layout __attribute__ ((__unused__)))
- {
- return;
- }
- void
- class_setWeakIvarLayout (Class class_ __attribute__ ((__unused__)),
- const char *layout __attribute__ ((__unused__)))
- {
- return;
- }
|