1 /* GTK - The GIMP Toolkit
2 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
23 #include "gtkobject.h"
24 #include "gtksignal.h"
27 #define GTK_OBJECT_DATA_ID_BLOCK_SIZE (1024)
28 #define GTK_OBJECT_DATA_BLOCK_SIZE (1024)
42 typedef struct _GtkObjectData GtkObjectData;
43 typedef struct _GtkArgInfo GtkArgInfo;
49 GtkDestroyNotify destroy;
64 void gtk_object_init_type (void);
65 static void gtk_object_class_init (GtkObjectClass *klass);
66 static void gtk_object_init (GtkObject *object);
67 static void gtk_object_set_arg (GtkObject *object,
70 static void gtk_object_get_arg (GtkObject *object,
73 static void gtk_object_shutdown (GtkObject *object);
74 static void gtk_object_real_destroy (GtkObject *object);
75 static void gtk_object_finalize (GtkObject *object);
76 static void gtk_object_notify_weaks (GtkObject *object);
78 GtkArg* gtk_object_collect_args (guint *nargs,
79 GtkType (*) (const gchar*),
83 static guint object_signals[LAST_SIGNAL] = { 0 };
85 static GHashTable *arg_info_ht = NULL;
87 static const gchar *user_data_key = "user_data";
88 static guint user_data_key_id = 0;
89 static const gchar *weakrefs_key = "gtk-weakrefs";
90 static guint weakrefs_key_id = 0;
92 static GtkObjectData *gtk_object_data_free_list = NULL;
94 #define GTK_OBJECT_DATA_DESTROY( odata ) { \
96 odata->destroy (odata->data); \
97 odata->next = gtk_object_data_free_list; \
98 gtk_object_data_free_list = odata; \
102 #ifdef G_ENABLE_DEBUG
103 static guint obj_count = 0;
104 static GHashTable *living_objs_ht = NULL;
106 gtk_object_debug_foreach (gpointer key, gpointer value, gpointer user_data)
110 object = (GtkObject*) value;
111 g_print ("GTK-DEBUG: %p: %s ref_count=%d%s%s\n",
113 gtk_type_name (GTK_OBJECT_TYPE (object)),
115 GTK_OBJECT_FLOATING (object) ? " (floating)" : "",
116 GTK_OBJECT_DESTROYED (object) ? " (destroyed)" : "");
119 gtk_object_debug (void)
121 g_hash_table_foreach (living_objs_ht, gtk_object_debug_foreach, NULL);
123 g_print ("GTK-DEBUG: living objects count = %d\n", obj_count);
125 #endif /* G_ENABLE_DEBUG */
127 /****************************************************
128 * GtkObject type, class and instance initialization
130 ****************************************************/
133 gtk_object_init_type (void)
135 GtkType object_type = 0;
136 GtkTypeInfo object_info =
140 sizeof (GtkObjectClass),
141 (GtkClassInitFunc) gtk_object_class_init,
142 (GtkObjectInitFunc) gtk_object_init,
147 object_type = gtk_type_unique (0, &object_info);
148 g_assert (object_type == GTK_TYPE_OBJECT);
150 #ifdef G_ENABLE_DEBUG
151 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
152 ATEXIT (gtk_object_debug);
153 #endif /* G_ENABLE_DEBUG */
157 gtk_object_get_type (void)
159 return GTK_TYPE_OBJECT;
163 gtk_object_class_init (GtkObjectClass *class)
165 class->signals = NULL;
169 gtk_object_add_arg_type ("GtkObject::user_data",
173 gtk_object_add_arg_type ("GtkObject::signal",
177 gtk_object_add_arg_type ("GtkObject::object_signal",
182 object_signals[DESTROY] =
183 gtk_signal_new ("destroy",
186 GTK_SIGNAL_OFFSET (GtkObjectClass, destroy),
187 gtk_signal_default_marshaller,
190 gtk_object_class_add_signals (class, object_signals, LAST_SIGNAL);
192 class->shutdown = gtk_object_shutdown;
193 class->destroy = gtk_object_real_destroy;
194 class->finalize = gtk_object_finalize;
198 gtk_object_init (GtkObject *object)
200 GTK_OBJECT_FLAGS (object) = GTK_FLOATING;
202 object->ref_count = 1;
203 object->object_data = NULL;
205 #ifdef G_ENABLE_DEBUG
206 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
211 living_objs_ht = g_hash_table_new (g_direct_hash, NULL);
213 g_hash_table_insert (living_objs_ht, object, object);
215 #endif /* G_ENABLE_DEBUG */
218 /********************************************
219 * Functions to end a GtkObject's life time
221 ********************************************/
223 gtk_object_destroy (GtkObject *object)
225 g_return_if_fail (object != NULL);
226 g_return_if_fail (GTK_IS_OBJECT (object));
228 if (!GTK_OBJECT_DESTROYED (object))
230 /* we will hold a reference on the object in this place, so
231 * to ease all classes shutdown and destroy implementations.
232 * i.e. they don't have to bother about referencing at all.
234 gtk_object_ref (object);
235 object->klass->shutdown (object);
236 gtk_object_unref (object);
241 gtk_object_shutdown (GtkObject *object)
243 GTK_OBJECT_SET_FLAGS (object, GTK_DESTROYED);
244 gtk_signal_emit (object, object_signals[DESTROY]);
248 gtk_object_real_destroy (GtkObject *object)
250 if (GTK_OBJECT_CONNECTED (object))
251 gtk_signal_handlers_destroy (object);
255 gtk_object_finalize (GtkObject *object)
257 gtk_object_notify_weaks (object);
259 while (object->object_data)
261 GtkObjectData *odata;
263 odata = object->object_data;
264 object->object_data = odata->next;
265 GTK_OBJECT_DATA_DESTROY (odata);
268 gtk_type_free (GTK_OBJECT_TYPE (object), object);
271 /*****************************************
272 * GtkObject argument handlers
274 *****************************************/
277 gtk_object_set_arg (GtkObject *object,
284 gtk_object_set_user_data (object, GTK_VALUE_POINTER (*arg));
287 if ((arg->name[9 + 2 + 6] != ':') || (arg->name[9 + 2 + 7] != ':'))
289 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
292 gtk_signal_connect (object, arg->name + 9 + 2 + 6 + 2,
293 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
294 GTK_VALUE_SIGNAL (*arg).d);
296 case ARG_OBJECT_SIGNAL:
297 if ((arg->name[9 + 2 + 13] != ':') || (arg->name[9 + 2 + 14] != ':'))
299 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
302 gtk_signal_connect_object (object, arg->name + 9 + 2 + 13 + 2,
303 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
304 (GtkObject*) GTK_VALUE_SIGNAL (*arg).d);
312 gtk_object_get_arg (GtkObject *object,
319 GTK_VALUE_POINTER (*arg) = gtk_object_get_user_data (object);
322 case ARG_OBJECT_SIGNAL:
324 arg->type = GTK_TYPE_INVALID;
329 /*****************************************
330 * gtk_object_class_add_signals:
335 *****************************************/
338 gtk_object_class_add_signals (GtkObjectClass *class,
345 g_return_if_fail (class != NULL);
347 new_signals = g_new (guint, class->nsignals + nsignals);
348 for (i = 0; i < class->nsignals; i++)
349 new_signals[i] = class->signals[i];
350 for (i = 0; i < nsignals; i++)
351 new_signals[class->nsignals + i] = signals[i];
353 g_free (class->signals);
354 class->signals = new_signals;
355 class->nsignals += nsignals;
359 gtk_object_class_add_user_signal (GtkObjectClass *class,
361 GtkSignalMarshaller marshaller,
371 g_return_val_if_fail (class != NULL, 0);
375 params = g_new (GtkType, nparams);
377 va_start (args, nparams);
379 for (i = 0; i < nparams; i++)
380 params[i] = va_arg (args, GtkType);
387 signal_id = gtk_signal_newv (name,
399 gtk_object_class_add_signals (class, &signal_id, 1);
405 gtk_object_class_user_signal_new (GtkObjectClass *class,
407 GtkSignalRunType signal_flags,
408 GtkSignalMarshaller marshaller,
418 g_return_val_if_fail (class != NULL, 0);
422 params = g_new (GtkType, nparams);
424 va_start (args, nparams);
426 for (i = 0; i < nparams; i++)
427 params[i] = va_arg (args, GtkType);
434 signal_id = gtk_signal_newv (name,
446 gtk_object_class_add_signals (class, &signal_id, 1);
452 gtk_object_class_user_signal_newv (GtkObjectClass *class,
454 GtkSignalRunType signal_flags,
455 GtkSignalMarshaller marshaller,
462 g_return_val_if_fail (class != NULL, 0);
465 g_return_val_if_fail (params != NULL, 0);
467 signal_id = gtk_signal_newv (name,
477 gtk_object_class_add_signals (class, &signal_id, 1);
482 /*****************************************
488 *****************************************/
491 gtk_object_sink (GtkObject *object)
493 g_return_if_fail (object != NULL);
494 g_return_if_fail (GTK_IS_OBJECT (object));
496 if (GTK_OBJECT_FLOATING (object))
498 GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING);
499 gtk_object_unref (object);
503 /*****************************************
506 * Weak refs are very similar to the old "destroy" signal. They allow
507 * one to register a callback that is called when the weakly
508 * referenced object is finalized.
510 * They are not implemented as a signal because they really are
511 * special and need to be used with great care. Unlike signals, which
512 * should be able to execute any code whatsoever.
514 * A weakref callback is not allowed to retain a reference to the
515 * object. Object data keys may be retrieved in a weak reference
518 * A weakref callback is called at most once.
520 *****************************************/
522 typedef struct _GtkWeakRef GtkWeakRef;
527 GtkDestroyNotify notify;
532 gtk_object_weakref (GtkObject *object,
533 GtkDestroyNotify notify,
538 g_return_if_fail (object != NULL);
539 g_return_if_fail (notify != NULL);
540 g_return_if_fail (GTK_IS_OBJECT (object));
542 if (!weakrefs_key_id)
543 weakrefs_key_id = g_quark_from_static_string (weakrefs_key);
545 weak = g_new (GtkWeakRef, 1);
546 weak->next = gtk_object_get_data_by_id (object, weakrefs_key_id);
547 weak->notify = notify;
549 gtk_object_set_data_by_id (object, weakrefs_key_id, weak);
553 gtk_object_weakunref (GtkObject *object,
554 GtkDestroyNotify notify,
557 GtkWeakRef *weaks, *w, **wp;
559 g_return_if_fail (object != NULL);
560 g_return_if_fail (GTK_IS_OBJECT (object));
562 if (!weakrefs_key_id)
565 weaks = gtk_object_get_data_by_id (object, weakrefs_key_id);
566 for (wp = &weaks; *wp; wp = &(*wp)->next)
569 if (w->notify == notify && w->data == data)
572 gtk_object_set_data_by_id (object, weakrefs_key_id, w->next);
582 gtk_object_notify_weaks (GtkObject *object)
588 w1 = gtk_object_get_data_by_id (object, weakrefs_key_id);
592 w1->notify (w1->data);
600 /*****************************************
606 *****************************************/
609 gtk_object_new (GtkType type,
618 obj = gtk_type_new (type);
620 va_start (args1, type);
621 va_start (args2, type);
623 args = gtk_object_collect_args (&nargs, gtk_object_get_arg_type, args1, args2);
624 gtk_object_setv (obj, nargs, args);
633 /*****************************************
639 *****************************************/
642 gtk_object_newv (GtkType type,
648 obj = gtk_type_new (type);
649 gtk_object_setv (obj, nargs, args);
654 /*****************************************
660 *****************************************/
663 gtk_object_getv (GtkObject *object,
669 g_return_if_fail (object != NULL);
670 g_return_if_fail (GTK_IS_OBJECT (object));
675 for (i = 0; i < nargs; i++)
682 /* hm, the name cutting shouldn't be needed on gets, but what the heck...
684 lookup_name = g_strdup (args[i].name);
685 d = strchr (lookup_name, ':');
686 if (d && d[1] == ':')
688 d = strchr (d + 2, ':');
692 info = g_hash_table_lookup (arg_info_ht, lookup_name);
699 g_warning ("gtk_object_getv(): invalid arg name: \"%s\"\n",
701 args[i].type = GTK_TYPE_INVALID;
702 g_free (lookup_name);
705 else if (!gtk_type_is_a (object->klass->type, info->class_type))
707 g_warning ("gtk_object_getv(): invalid arg for %s: \"%s\"\n",
708 gtk_type_name (object->klass->type), lookup_name);
709 args[i].type = GTK_TYPE_INVALID;
710 g_free (lookup_name);
713 else if (! (info->arg_flags & GTK_ARG_READABLE))
715 g_warning ("gtk_object_getv(): arg is not supplied for read-access: \"%s\"\n",
717 args[i].type = GTK_TYPE_INVALID;
718 g_free (lookup_name);
722 g_free (lookup_name);
724 args[i].type = info->type;
725 gtk_type_get_arg (object, info->class_type, &args[i], info->arg_id);
729 /*****************************************
730 * gtk_object_query_args:
735 *****************************************/
737 struct _GtkQueryArgData
742 typedef struct _GtkQueryArgData GtkQueryArgData;
745 gtk_query_arg_foreach (gpointer key,
749 register GtkArgInfo *info;
750 register GtkQueryArgData *data;
755 if (info->class_type == data->class_type)
756 data->arg_list = g_list_prepend (data->arg_list, info);
760 gtk_object_query_args (GtkType class_type,
765 GtkQueryArgData query_data;
769 g_return_val_if_fail (nargs != NULL, NULL);
771 g_return_val_if_fail (gtk_type_is_a (class_type, GTK_TYPE_OBJECT), NULL);
776 /* make sure the types class has been initialized, because
777 * the argument setup happens in the gtk_*_class_init() functions.
779 gtk_type_class (class_type);
781 query_data.arg_list = NULL;
782 query_data.class_type = class_type;
783 g_hash_table_foreach (arg_info_ht, gtk_query_arg_foreach, &query_data);
785 if (query_data.arg_list)
787 register GList *list;
790 list = query_data.arg_list;
797 g_assert (len == ((GtkObjectClass*) gtk_type_class (class_type))->n_args); /* paranoid */
799 args = g_new0 (GtkArg, len);
802 *arg_flags = g_new (guint32, len);
811 g_assert (info->seq_id > 0 && info->seq_id <= len); /* paranoid */
813 args[info->seq_id - 1].type = info->type;
814 args[info->seq_id - 1].name = info->name;
816 (*arg_flags)[info->seq_id - 1] = info->arg_flags;
820 g_list_free (query_data.arg_list);
828 /*****************************************
834 *****************************************/
837 gtk_object_set (GtkObject *object,
845 g_return_if_fail (object != NULL);
847 va_start (args1, object);
848 va_start (args2, object);
850 args = gtk_object_collect_args (&nargs, gtk_object_get_arg_type, args1, args2);
851 gtk_object_setv (object, nargs, args);
858 /*****************************************
864 *****************************************/
867 gtk_object_setv (GtkObject *object,
873 g_return_if_fail (object != NULL);
874 g_return_if_fail (GTK_OBJECT (object));
879 for (i = 0; i < nargs; i++)
886 lookup_name = g_strdup (args[i].name);
887 d = strchr (lookup_name, ':');
888 if (d && d[1] == ':')
890 d = strchr (d + 2, ':');
894 info = g_hash_table_lookup (arg_info_ht, lookup_name);
903 g_warning ("gtk_object_setv(): invalid arg name: \"%s\"\n",
907 else if (info->type != args[i].type)
909 g_warning ("gtk_object_setv(): invalid arg type for: \"%s\"\n",
913 else if (!gtk_type_is_a (object->klass->type, info->class_type))
915 g_warning ("gtk_object_setv(): invalid arg for %s: \"%s\"\n",
916 gtk_type_name (object->klass->type), lookup_name);
919 else if (! (info->arg_flags & GTK_ARG_WRITABLE))
921 g_warning ("gtk_object_setv(): arg is not supplied for write-access: \"%s\"\n",
926 g_free (lookup_name);
931 gtk_type_set_arg (object, info->class_type, &args[i], info->arg_id);
935 /*****************************************
936 * gtk_object_add_arg_type:
941 *****************************************/
944 gtk_object_add_arg_type (const char *arg_name,
950 gchar class_part[1024];
954 g_return_if_fail (arg_name != NULL);
955 g_return_if_fail (arg_type > GTK_TYPE_NONE);
956 g_return_if_fail (arg_id > 0);
957 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) != 0);
958 g_return_if_fail ((arg_flags & GTK_ARG_CHILD_ARG) == 0);
960 arg_flags &= GTK_ARG_MASK;
962 arg_part = strchr (arg_name, ':');
963 if (!arg_part || (arg_part[0] != ':') || (arg_part[1] != ':'))
965 g_warning ("invalid arg name: \"%s\"\n", arg_name);
969 strncpy (class_part, arg_name, (glong) (arg_part - arg_name));
970 class_part[(glong) (arg_part - arg_name)] = '\0';
972 class_type = gtk_type_from_name (class_part);
975 g_warning ("gtk_object_add_arg_type(): invalid class name in arg: \"%s\"\n", arg_name);
979 info = g_new (GtkArgInfo, 1);
980 info->name = g_strdup (arg_name);
981 info->type = arg_type;
982 info->class_type = class_type;
983 info->arg_flags = arg_flags;
984 info->arg_id = arg_id;
985 info->seq_id = ++((GtkObjectClass*) gtk_type_class (class_type))->n_args;
988 arg_info_ht = g_hash_table_new (g_str_hash, g_str_equal);
990 g_hash_table_insert (arg_info_ht, info->name, info);
993 /*****************************************
994 * gtk_object_get_arg_type:
999 *****************************************/
1002 gtk_object_get_arg_type (const gchar *arg_name)
1008 g_return_val_if_fail (arg_name != NULL, 0);
1011 return GTK_TYPE_INVALID;
1013 if (!arg_name || strlen (arg_name) > 120)
1017 g_warning ("gtk_object_get_arg_type(): argument `arg_name' exceeds maximum size.");
1018 return GTK_TYPE_INVALID;
1021 t = strchr (arg_name, ':');
1022 if (!t || (t[0] != ':') || (t[1] != ':'))
1024 g_warning ("gtk_object_get_arg_type(): invalid arg name: \"%s\"\n", arg_name);
1025 return GTK_TYPE_INVALID;
1028 t = strchr (t + 2, ':');
1031 strncpy (buffer, arg_name, (long) (t - arg_name));
1032 buffer[(long) (t - arg_name)] = '\0';
1036 info = g_hash_table_lookup (arg_info_ht, arg_name);
1040 return GTK_TYPE_INVALID;
1043 /*****************************************
1044 * GtkObject object_data mechanism
1046 *****************************************/
1049 gtk_object_set_data_by_id (GtkObject *object,
1053 g_return_if_fail (data_id > 0);
1055 gtk_object_set_data_by_id_full (object, data_id, data, NULL);
1059 gtk_object_set_data (GtkObject *object,
1063 g_return_if_fail (key != NULL);
1065 gtk_object_set_data_by_id_full (object, gtk_object_data_force_id (key), data, NULL);
1069 gtk_object_set_data_by_id_full (GtkObject *object,
1072 GtkDestroyNotify destroy)
1074 GtkObjectData *odata;
1076 g_return_if_fail (object != NULL);
1077 g_return_if_fail (GTK_IS_OBJECT (object));
1078 g_return_if_fail (data_id > 0);
1080 odata = object->object_data;
1083 GtkObjectData *prev;
1089 if (odata->id == data_id)
1092 prev->next = odata->next;
1094 object->object_data = odata->next;
1096 GTK_OBJECT_DATA_DESTROY (odata);
1101 odata = odata->next;
1108 if (odata->id == data_id)
1110 register GtkDestroyNotify dfunc;
1111 register gpointer ddata;
1113 dfunc = odata->destroy;
1114 ddata = odata->data;
1115 odata->destroy = destroy;
1118 /* we need to have updated all structures prior to
1119 * invokation of the destroy function
1127 odata = odata->next;
1130 if (gtk_object_data_free_list)
1132 odata = gtk_object_data_free_list;
1133 gtk_object_data_free_list = odata->next;
1137 GtkObjectData *odata_block;
1140 odata_block = g_new0 (GtkObjectData, GTK_OBJECT_DATA_BLOCK_SIZE);
1141 for (i = 1; i < GTK_OBJECT_DATA_BLOCK_SIZE; i++)
1143 (odata_block + i)->next = gtk_object_data_free_list;
1144 gtk_object_data_free_list = (odata_block + i);
1147 odata = odata_block;
1150 odata->id = data_id;
1152 odata->destroy = destroy;
1153 odata->next = object->object_data;
1155 object->object_data = odata;
1160 gtk_object_set_data_full (GtkObject *object,
1163 GtkDestroyNotify destroy)
1165 g_return_if_fail (key != NULL);
1167 gtk_object_set_data_by_id_full (object, gtk_object_data_force_id (key), data, destroy);
1171 gtk_object_get_data_by_id (GtkObject *object,
1174 GtkObjectData *odata;
1176 g_return_val_if_fail (object != NULL, NULL);
1177 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1181 odata = object->object_data;
1184 if (odata->id == data_id)
1186 odata = odata->next;
1194 gtk_object_get_data (GtkObject *object,
1199 g_return_val_if_fail (key != NULL, NULL);
1201 id = gtk_object_data_try_key (key);
1203 return gtk_object_get_data_by_id (object, id);
1209 gtk_object_remove_data_by_id (GtkObject *object,
1213 gtk_object_set_data_by_id_full (object, data_id, NULL, NULL);
1217 gtk_object_remove_data (GtkObject *object,
1222 g_return_if_fail (key != NULL);
1224 id = gtk_object_data_try_key (key);
1226 gtk_object_set_data_by_id_full (object, id, NULL, NULL);
1229 /*****************************************
1230 * gtk_object_set_user_data:
1235 *****************************************/
1238 gtk_object_set_user_data (GtkObject *object,
1241 if (!user_data_key_id)
1242 user_data_key_id = g_quark_from_static_string (user_data_key);
1244 gtk_object_set_data_by_id_full (object, user_data_key_id, data, NULL);
1247 /*****************************************
1248 * gtk_object_get_user_data:
1253 *****************************************/
1256 gtk_object_get_user_data (GtkObject *object)
1258 if (user_data_key_id)
1259 return gtk_object_get_data_by_id (object, user_data_key_id);
1264 /*****************************************
1265 * gtk_object_check_cast:
1270 *****************************************/
1273 gtk_object_descriptive_type_name (GtkType type)
1277 name = gtk_type_name (type);
1285 gtk_object_check_cast (GtkObject *obj,
1290 g_warning ("invalid cast from (NULL) pointer to `%s'",
1291 gtk_object_descriptive_type_name (cast_type));
1296 g_warning ("invalid unclassed pointer in cast to `%s'",
1297 gtk_object_descriptive_type_name (cast_type));
1300 if (obj->klass->type < GTK_TYPE_OBJECT)
1302 g_warning ("invalid class type `%s' in cast to `%s'",
1303 gtk_object_descriptive_type_name (obj->klass->type),
1304 gtk_object_descriptive_type_name (cast_type));
1307 if (!gtk_type_is_a (obj->klass->type, cast_type))
1309 g_warning ("invalid cast from `%s' to `%s'",
1310 gtk_object_descriptive_type_name (obj->klass->type),
1311 gtk_object_descriptive_type_name (cast_type));
1318 /*****************************************
1319 * gtk_object_check_class_cast:
1324 *****************************************/
1327 gtk_object_check_class_cast (GtkObjectClass *klass,
1332 g_warning ("invalid class cast from (NULL) pointer to `%s'",
1333 gtk_object_descriptive_type_name (cast_type));
1336 if (klass->type < GTK_TYPE_OBJECT)
1338 g_warning ("invalid class type `%s' in class cast to `%s'",
1339 gtk_object_descriptive_type_name (klass->type),
1340 gtk_object_descriptive_type_name (cast_type));
1343 if (!gtk_type_is_a (klass->type, cast_type))
1345 g_warning ("invalid class cast from `%s' to `%s'",
1346 gtk_object_descriptive_type_name (klass->type),
1347 gtk_object_descriptive_type_name (cast_type));
1354 /*****************************************
1355 * gtk_object_collect_args:
1360 *****************************************/
1363 gtk_object_collect_args (guint *nargs,
1364 GtkType (*get_arg_type) (const gchar*),
1379 name = va_arg (args1, char *);
1386 type = get_arg_type (name);
1388 switch (GTK_FUNDAMENTAL_TYPE (type))
1390 case GTK_TYPE_INVALID:
1391 g_warning ("GTK: invalid arg name: \"%s\" %x\n", name, type);
1392 (void) va_arg (args1, long);
1401 case GTK_TYPE_FLAGS:
1402 (void) va_arg (args1, gint);
1405 case GTK_TYPE_ULONG:
1406 (void) va_arg (args1, glong);
1408 case GTK_TYPE_FLOAT:
1409 (void) va_arg (args1, gfloat);
1411 case GTK_TYPE_DOUBLE:
1412 (void) va_arg (args1, gdouble);
1414 case GTK_TYPE_STRING:
1415 (void) va_arg (args1, gchar*);
1417 case GTK_TYPE_POINTER:
1418 case GTK_TYPE_BOXED:
1419 (void) va_arg (args1, gpointer);
1421 case GTK_TYPE_SIGNAL:
1422 (void) va_arg (args1, GtkFunction);
1423 (void) va_arg (args1, gpointer);
1425 case GTK_TYPE_FOREIGN:
1426 (void) va_arg (args1, gpointer);
1427 (void) va_arg (args1, GtkDestroyNotify);
1429 case GTK_TYPE_CALLBACK:
1430 (void) va_arg (args1, GtkCallbackMarshal);
1431 (void) va_arg (args1, gpointer);
1432 (void) va_arg (args1, GtkDestroyNotify);
1434 case GTK_TYPE_C_CALLBACK:
1435 (void) va_arg (args1, GtkFunction);
1436 (void) va_arg (args1, gpointer);
1439 (void) va_arg (args1, gint);
1440 (void) va_arg (args1, GtkArg*);
1442 case GTK_TYPE_OBJECT:
1443 (void) va_arg (args1, GtkObject*);
1446 g_error ("unsupported type %s in args", gtk_type_name (type));
1458 args = g_new0 (GtkArg, n);
1460 for (i = 0; i < n; i++)
1462 args[i].name = va_arg (args2, char *);
1463 args[i].type = get_arg_type (args[i].name);
1465 switch (GTK_FUNDAMENTAL_TYPE (args[i].type))
1467 case GTK_TYPE_INVALID:
1468 (void) va_arg (args2, long);
1474 GTK_VALUE_CHAR(args[i]) = va_arg (args2, gint);
1477 GTK_VALUE_BOOL(args[i]) = va_arg (args2, gint);
1480 GTK_VALUE_INT(args[i]) = va_arg (args2, gint);
1483 GTK_VALUE_UINT(args[i]) = va_arg (args2, guint);
1486 GTK_VALUE_ENUM(args[i]) = va_arg (args2, gint);
1488 case GTK_TYPE_FLAGS:
1489 GTK_VALUE_FLAGS(args[i]) = va_arg (args2, gint);
1492 GTK_VALUE_LONG(args[i]) = va_arg (args2, glong);
1494 case GTK_TYPE_ULONG:
1495 GTK_VALUE_ULONG(args[i]) = va_arg (args2, gulong);
1497 case GTK_TYPE_FLOAT:
1498 GTK_VALUE_FLOAT(args[i]) = va_arg (args2, gfloat);
1500 case GTK_TYPE_DOUBLE:
1501 GTK_VALUE_DOUBLE(args[i]) = va_arg (args2, gdouble);
1503 case GTK_TYPE_STRING:
1504 GTK_VALUE_STRING(args[i]) = va_arg (args2, gchar*);
1506 case GTK_TYPE_POINTER:
1507 GTK_VALUE_POINTER(args[i]) = va_arg (args2, gpointer);
1509 case GTK_TYPE_BOXED:
1510 GTK_VALUE_BOXED(args[i]) = va_arg (args2, gpointer);
1512 case GTK_TYPE_SIGNAL:
1513 GTK_VALUE_SIGNAL(args[i]).f = va_arg (args2, GtkFunction);
1514 GTK_VALUE_SIGNAL(args[i]).d = va_arg (args2, gpointer);
1516 case GTK_TYPE_FOREIGN:
1517 GTK_VALUE_FOREIGN(args[i]).data = va_arg (args2, gpointer);
1518 GTK_VALUE_FOREIGN(args[i]).notify =
1519 va_arg (args2, GtkDestroyNotify);
1521 case GTK_TYPE_CALLBACK:
1522 GTK_VALUE_CALLBACK(args[i]).marshal =
1523 va_arg (args2, GtkCallbackMarshal);
1524 GTK_VALUE_CALLBACK(args[i]).data = va_arg (args2, gpointer);
1525 GTK_VALUE_CALLBACK(args[i]).notify =
1526 va_arg (args2, GtkDestroyNotify);
1528 case GTK_TYPE_C_CALLBACK:
1529 GTK_VALUE_C_CALLBACK(args[i]).func = va_arg (args2, GtkFunction);
1530 GTK_VALUE_C_CALLBACK(args[i]).func_data =
1531 va_arg (args2, gpointer);
1534 GTK_VALUE_ARGS(args[i]).n_args = va_arg (args2, gint);
1535 GTK_VALUE_ARGS(args[i]).args = va_arg (args2, GtkArg*);
1537 case GTK_TYPE_OBJECT:
1538 GTK_VALUE_OBJECT(args[i]) = va_arg (args2, GtkObject*);
1539 g_assert (GTK_VALUE_OBJECT(args[i]) == NULL ||
1540 GTK_CHECK_TYPE (GTK_VALUE_OBJECT(args[i]),
1544 g_error ("unsupported type %s in args",
1545 gtk_type_name (args[i].type));
1556 #undef gtk_object_ref
1557 #undef gtk_object_unref
1560 gtk_object_ref (GtkObject *object)
1562 g_return_if_fail (object != NULL);
1563 g_return_if_fail (GTK_IS_OBJECT (object));
1565 object->ref_count += 1;
1569 gtk_object_unref (GtkObject *object)
1571 g_return_if_fail (object != NULL);
1572 g_return_if_fail (GTK_IS_OBJECT (object));
1574 if (object->ref_count == 1)
1575 gtk_object_destroy (object);
1577 if (object->ref_count > 0)
1578 object->ref_count -= 1;
1580 if (object->ref_count == 0)
1582 #ifdef G_ENABLE_DEBUG
1583 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1585 g_assert (g_hash_table_lookup (living_objs_ht, object) == object);
1586 g_hash_table_remove (living_objs_ht, object);
1589 #endif /* G_ENABLE_DEBUG */
1590 object->klass->finalize (object);
1594 static GtkObject *gtk_trace_object = NULL;
1596 gtk_trace_referencing (GtkObject *object,
1602 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1604 gboolean exists = TRUE;
1606 g_return_if_fail (object != NULL);
1607 g_return_if_fail (GTK_IS_OBJECT (object));
1609 #ifdef G_ENABLE_DEBUG
1610 exists = g_hash_table_lookup (living_objs_ht, object) != NULL;
1611 #endif /* G_ENABLE_DEBUG */
1614 (object == gtk_trace_object ||
1615 gtk_trace_object == (void*)42))
1616 fprintf (stdout, "trace: object_%s: (%s:%p)->ref_count=%d %s (%s:%d)\n",
1617 do_ref ? "ref" : "unref",
1618 gtk_type_name (GTK_OBJECT_TYPE (object)),
1621 do_ref ? "+ 1" : "- 1",
1625 fprintf (stdout, "trace: object_%s(%p): no such object! (%s:%d)\n",
1626 do_ref ? "ref" : "unref",
1633 gtk_object_ref (object);
1635 gtk_object_unref (object);