377 lines
9.4 KiB
C
377 lines
9.4 KiB
C
/* GNU Objective C Runtime ivar related functions.
|
|
Copyright (C) 2010-2020 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;
|
|
}
|