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_base_class_init (GtkObjectClass *klass);
66 static void gtk_object_class_init (GtkObjectClass *klass);
67 static void gtk_object_init (GtkObject *object);
68 static void gtk_object_set_arg (GtkObject *object,
71 static void gtk_object_get_arg (GtkObject *object,
74 static void gtk_object_shutdown (GtkObject *object);
75 static void gtk_object_real_destroy (GtkObject *object);
76 static void gtk_object_finalize (GtkObject *object);
77 static void gtk_object_notify_weaks (GtkObject *object);
79 GtkArg* gtk_object_collect_args (guint *nargs,
80 GtkType (*) (const gchar*),
84 static guint object_signals[LAST_SIGNAL] = { 0 };
86 static GHashTable *arg_info_ht = NULL;
88 static const gchar *user_data_key = "user_data";
89 static guint user_data_key_id = 0;
90 static const gchar *weakrefs_key = "gtk-weakrefs";
91 static guint weakrefs_key_id = 0;
93 static GtkObjectData *gtk_object_data_free_list = NULL;
95 #define GTK_OBJECT_DATA_DESTROY( odata ) { \
97 odata->destroy (odata->data); \
98 odata->next = gtk_object_data_free_list; \
99 gtk_object_data_free_list = odata; \
103 #ifdef G_ENABLE_DEBUG
104 static guint obj_count = 0;
105 static GHashTable *living_objs_ht = NULL;
107 gtk_object_debug_foreach (gpointer key, gpointer value, gpointer user_data)
111 object = (GtkObject*) value;
112 g_print ("GTK-DEBUG: %p: %s ref_count=%d%s%s\n",
114 gtk_type_name (GTK_OBJECT_TYPE (object)),
116 GTK_OBJECT_FLOATING (object) ? " (floating)" : "",
117 GTK_OBJECT_DESTROYED (object) ? " (destroyed)" : "");
120 gtk_object_debug (void)
122 g_hash_table_foreach (living_objs_ht, gtk_object_debug_foreach, NULL);
124 g_print ("GTK-DEBUG: living objects count = %d\n", obj_count);
126 #endif /* G_ENABLE_DEBUG */
128 /****************************************************
129 * GtkObject type, class and instance initialization
131 ****************************************************/
134 gtk_object_init_type (void)
136 GtkType object_type = 0;
137 GtkTypeInfo object_info =
141 sizeof (GtkObjectClass),
142 (GtkClassInitFunc) gtk_object_class_init,
143 (GtkObjectInitFunc) gtk_object_init,
144 /* reversed_1 */ NULL,
145 /* reversed_2 */ NULL,
146 (GtkClassInitFunc) gtk_object_base_class_init,
149 object_type = gtk_type_unique (0, &object_info);
150 g_assert (object_type == GTK_TYPE_OBJECT);
152 #ifdef G_ENABLE_DEBUG
153 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
154 ATEXIT (gtk_object_debug);
155 #endif /* G_ENABLE_DEBUG */
159 gtk_object_get_type (void)
161 return GTK_TYPE_OBJECT;
165 gtk_object_base_class_init (GtkObjectClass *class)
167 /* reset instance specific fields that don't get inhrited */
168 class->signals = NULL;
172 /* reset instance specifc methods that don't get inherited */
173 class->get_arg = NULL;
174 class->set_arg = NULL;
178 gtk_object_class_init (GtkObjectClass *class)
180 gtk_object_add_arg_type ("GtkObject::user_data",
184 gtk_object_add_arg_type ("GtkObject::signal",
188 gtk_object_add_arg_type ("GtkObject::object_signal",
193 object_signals[DESTROY] =
194 gtk_signal_new ("destroy",
197 GTK_SIGNAL_OFFSET (GtkObjectClass, destroy),
198 gtk_signal_default_marshaller,
201 gtk_object_class_add_signals (class, object_signals, LAST_SIGNAL);
203 class->get_arg = gtk_object_get_arg;
204 class->set_arg = gtk_object_set_arg;
205 class->shutdown = gtk_object_shutdown;
206 class->destroy = gtk_object_real_destroy;
207 class->finalize = gtk_object_finalize;
211 gtk_object_init (GtkObject *object)
213 GTK_OBJECT_FLAGS (object) = GTK_FLOATING;
215 object->ref_count = 1;
216 object->object_data = NULL;
218 #ifdef G_ENABLE_DEBUG
219 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
224 living_objs_ht = g_hash_table_new (g_direct_hash, NULL);
226 g_hash_table_insert (living_objs_ht, object, object);
228 #endif /* G_ENABLE_DEBUG */
231 /********************************************
232 * Functions to end a GtkObject's life time
234 ********************************************/
236 gtk_object_destroy (GtkObject *object)
238 g_return_if_fail (object != NULL);
239 g_return_if_fail (GTK_IS_OBJECT (object));
241 if (!GTK_OBJECT_DESTROYED (object))
243 /* we will hold a reference on the object in this place, so
244 * to ease all classes shutdown and destroy implementations.
245 * i.e. they don't have to bother about referencing at all.
247 gtk_object_ref (object);
248 object->klass->shutdown (object);
249 gtk_object_unref (object);
254 gtk_object_shutdown (GtkObject *object)
256 GTK_OBJECT_SET_FLAGS (object, GTK_DESTROYED);
257 gtk_signal_emit (object, object_signals[DESTROY]);
261 gtk_object_real_destroy (GtkObject *object)
263 if (GTK_OBJECT_CONNECTED (object))
264 gtk_signal_handlers_destroy (object);
268 gtk_object_finalize (GtkObject *object)
270 gtk_object_notify_weaks (object);
272 while (object->object_data)
274 GtkObjectData *odata;
276 odata = object->object_data;
277 object->object_data = odata->next;
278 GTK_OBJECT_DATA_DESTROY (odata);
281 gtk_type_free (GTK_OBJECT_TYPE (object), object);
284 /*****************************************
285 * GtkObject argument handlers
287 *****************************************/
290 gtk_object_set_arg (GtkObject *object,
297 gtk_object_set_user_data (object, GTK_VALUE_POINTER (*arg));
300 if ((arg->name[9 + 2 + 6] != ':') || (arg->name[9 + 2 + 7] != ':'))
302 g_warning ("gtk_object_set_arg(): invalid signal argument: \"%s\"\n", arg->name);
305 gtk_signal_connect (object, arg->name + 9 + 2 + 6 + 2,
306 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
307 GTK_VALUE_SIGNAL (*arg).d);
309 case ARG_OBJECT_SIGNAL:
310 if ((arg->name[9 + 2 + 13] != ':') || (arg->name[9 + 2 + 14] != ':'))
312 g_warning ("gtk_object_set_arg(): invalid signal argument: \"%s\"\n", arg->name);
315 gtk_signal_connect_object (object, arg->name + 9 + 2 + 13 + 2,
316 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
317 (GtkObject*) GTK_VALUE_SIGNAL (*arg).d);
325 gtk_object_get_arg (GtkObject *object,
332 GTK_VALUE_POINTER (*arg) = gtk_object_get_user_data (object);
335 case ARG_OBJECT_SIGNAL:
337 arg->type = GTK_TYPE_INVALID;
342 /*****************************************
343 * gtk_object_class_add_signals:
348 *****************************************/
351 gtk_object_class_add_signals (GtkObjectClass *class,
358 g_return_if_fail (class != NULL);
360 new_signals = g_new (guint, class->nsignals + nsignals);
361 for (i = 0; i < class->nsignals; i++)
362 new_signals[i] = class->signals[i];
363 for (i = 0; i < nsignals; i++)
364 new_signals[class->nsignals + i] = signals[i];
366 g_free (class->signals);
367 class->signals = new_signals;
368 class->nsignals += nsignals;
372 gtk_object_class_add_user_signal (GtkObjectClass *class,
374 GtkSignalMarshaller marshaller,
384 g_return_val_if_fail (class != NULL, 0);
388 params = g_new (GtkType, nparams);
390 va_start (args, nparams);
392 for (i = 0; i < nparams; i++)
393 params[i] = va_arg (args, GtkType);
400 signal_id = gtk_signal_newv (name,
412 gtk_object_class_add_signals (class, &signal_id, 1);
418 gtk_object_class_user_signal_new (GtkObjectClass *class,
420 GtkSignalRunType signal_flags,
421 GtkSignalMarshaller marshaller,
431 g_return_val_if_fail (class != NULL, 0);
435 params = g_new (GtkType, nparams);
437 va_start (args, nparams);
439 for (i = 0; i < nparams; i++)
440 params[i] = va_arg (args, GtkType);
447 signal_id = gtk_signal_newv (name,
459 gtk_object_class_add_signals (class, &signal_id, 1);
465 gtk_object_class_user_signal_newv (GtkObjectClass *class,
467 GtkSignalRunType signal_flags,
468 GtkSignalMarshaller marshaller,
475 g_return_val_if_fail (class != NULL, 0);
478 g_return_val_if_fail (params != NULL, 0);
480 signal_id = gtk_signal_newv (name,
490 gtk_object_class_add_signals (class, &signal_id, 1);
495 /*****************************************
501 *****************************************/
504 gtk_object_sink (GtkObject *object)
506 g_return_if_fail (object != NULL);
507 g_return_if_fail (GTK_IS_OBJECT (object));
509 if (GTK_OBJECT_FLOATING (object))
511 GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING);
512 gtk_object_unref (object);
516 /*****************************************
519 * Weak refs are very similar to the old "destroy" signal. They allow
520 * one to register a callback that is called when the weakly
521 * referenced object is finalized.
523 * They are not implemented as a signal because they really are
524 * special and need to be used with great care. Unlike signals, which
525 * should be able to execute any code whatsoever.
527 * A weakref callback is not allowed to retain a reference to the
528 * object. Object data keys may be retrieved in a weak reference
531 * A weakref callback is called at most once.
533 *****************************************/
535 typedef struct _GtkWeakRef GtkWeakRef;
540 GtkDestroyNotify notify;
545 gtk_object_weakref (GtkObject *object,
546 GtkDestroyNotify notify,
551 g_return_if_fail (object != NULL);
552 g_return_if_fail (notify != NULL);
553 g_return_if_fail (GTK_IS_OBJECT (object));
555 if (!weakrefs_key_id)
556 weakrefs_key_id = g_quark_from_static_string (weakrefs_key);
558 weak = g_new (GtkWeakRef, 1);
559 weak->next = gtk_object_get_data_by_id (object, weakrefs_key_id);
560 weak->notify = notify;
562 gtk_object_set_data_by_id (object, weakrefs_key_id, weak);
566 gtk_object_weakunref (GtkObject *object,
567 GtkDestroyNotify notify,
570 GtkWeakRef *weaks, *w, **wp;
572 g_return_if_fail (object != NULL);
573 g_return_if_fail (GTK_IS_OBJECT (object));
575 if (!weakrefs_key_id)
578 weaks = gtk_object_get_data_by_id (object, weakrefs_key_id);
579 for (wp = &weaks; *wp; wp = &(*wp)->next)
582 if (w->notify == notify && w->data == data)
585 gtk_object_set_data_by_id (object, weakrefs_key_id, w->next);
595 gtk_object_notify_weaks (GtkObject *object)
601 w1 = gtk_object_get_data_by_id (object, weakrefs_key_id);
605 w1->notify (w1->data);
613 /*****************************************
619 *****************************************/
622 gtk_object_new (GtkType type,
631 obj = gtk_type_new (type);
633 va_start (args1, type);
634 va_start (args2, type);
636 args = gtk_object_collect_args (&nargs, gtk_object_get_arg_type, args1, args2);
637 gtk_object_setv (obj, nargs, args);
646 /*****************************************
652 *****************************************/
655 gtk_object_newv (GtkType type,
661 obj = gtk_type_new (type);
662 gtk_object_setv (obj, nargs, args);
667 /*****************************************
673 *****************************************/
676 gtk_object_getv (GtkObject *object,
682 g_return_if_fail (object != NULL);
683 g_return_if_fail (GTK_IS_OBJECT (object));
688 for (i = 0; i < nargs; i++)
693 GtkObjectClass *oclass;
696 /* hm, the name cutting shouldn't be needed on gets, but what the heck...
698 lookup_name = g_strdup (args[i].name);
699 d = strchr (lookup_name, ':');
700 if (d && d[1] == ':')
702 d = strchr (d + 2, ':');
706 info = g_hash_table_lookup (arg_info_ht, lookup_name);
713 g_warning ("gtk_object_getv(): invalid arg name: \"%s\"\n",
715 args[i].type = GTK_TYPE_INVALID;
716 g_free (lookup_name);
719 else if (!gtk_type_is_a (object->klass->type, info->class_type))
721 g_warning ("gtk_object_getv(): invalid arg for %s: \"%s\"\n",
722 gtk_type_name (object->klass->type), lookup_name);
723 args[i].type = GTK_TYPE_INVALID;
724 g_free (lookup_name);
727 else if (! (info->arg_flags & GTK_ARG_READABLE))
729 g_warning ("gtk_object_getv(): arg is not supplied for read-access: \"%s\"\n",
731 args[i].type = GTK_TYPE_INVALID;
732 g_free (lookup_name);
736 g_free (lookup_name);
738 args[i].type = info->type;
740 oclass = gtk_type_class (info->class_type);
741 if (oclass && oclass->get_arg)
742 oclass->get_arg (object, &args[i], info->arg_id);
744 args[i].type = GTK_TYPE_INVALID;
747 gtk_type_get_arg (object, info->class_type, &args[i], info->arg_id);
752 /*****************************************
753 * gtk_object_query_args:
758 *****************************************/
760 struct _GtkQueryArgData
765 typedef struct _GtkQueryArgData GtkQueryArgData;
768 gtk_query_arg_foreach (gpointer key,
772 register GtkArgInfo *info;
773 register GtkQueryArgData *data;
778 if (info->class_type == data->class_type)
779 data->arg_list = g_list_prepend (data->arg_list, info);
783 gtk_object_query_args (GtkType class_type,
788 GtkQueryArgData query_data;
792 g_return_val_if_fail (nargs != NULL, NULL);
794 g_return_val_if_fail (gtk_type_is_a (class_type, GTK_TYPE_OBJECT), NULL);
799 /* make sure the types class has been initialized, because
800 * the argument setup happens in the gtk_*_class_init() functions.
802 gtk_type_class (class_type);
804 query_data.arg_list = NULL;
805 query_data.class_type = class_type;
806 g_hash_table_foreach (arg_info_ht, gtk_query_arg_foreach, &query_data);
808 if (query_data.arg_list)
810 register GList *list;
813 list = query_data.arg_list;
820 g_assert (len == ((GtkObjectClass*) gtk_type_class (class_type))->n_args); /* paranoid */
822 args = g_new0 (GtkArg, len);
825 *arg_flags = g_new (guint32, len);
834 g_assert (info->seq_id > 0 && info->seq_id <= len); /* paranoid */
836 args[info->seq_id - 1].type = info->type;
837 args[info->seq_id - 1].name = info->name;
839 (*arg_flags)[info->seq_id - 1] = info->arg_flags;
843 g_list_free (query_data.arg_list);
851 /*****************************************
857 *****************************************/
860 gtk_object_set (GtkObject *object,
868 g_return_if_fail (object != NULL);
870 va_start (args1, object);
871 va_start (args2, object);
873 args = gtk_object_collect_args (&nargs, gtk_object_get_arg_type, args1, args2);
874 gtk_object_setv (object, nargs, args);
881 /*****************************************
887 *****************************************/
890 gtk_object_setv (GtkObject *object,
896 g_return_if_fail (object != NULL);
897 g_return_if_fail (GTK_OBJECT (object));
902 for (i = 0; i < nargs; i++)
908 GtkObjectClass *oclass;
910 lookup_name = g_strdup (args[i].name);
911 d = strchr (lookup_name, ':');
912 if (d && d[1] == ':')
914 d = strchr (d + 2, ':');
918 info = g_hash_table_lookup (arg_info_ht, lookup_name);
927 g_warning ("gtk_object_setv(): invalid arg name: \"%s\"\n",
931 else if (info->type != args[i].type)
933 g_warning ("gtk_object_setv(): invalid arg type for: \"%s\"\n",
937 else if (!gtk_type_is_a (object->klass->type, info->class_type))
939 g_warning ("gtk_object_setv(): invalid arg for %s: \"%s\"\n",
940 gtk_type_name (object->klass->type), lookup_name);
943 else if (! (info->arg_flags & GTK_ARG_WRITABLE))
945 g_warning ("gtk_object_setv(): arg is not supplied for write-access: \"%s\"\n",
950 g_free (lookup_name);
955 oclass = gtk_type_class (info->class_type);
956 if (oclass && oclass->set_arg)
957 oclass->set_arg (object, &args[i], info->arg_id);
960 gtk_type_set_arg (object, info->class_type, &args[i], info->arg_id);
965 /*****************************************
966 * gtk_object_add_arg_type:
971 *****************************************/
974 gtk_object_add_arg_type (const char *arg_name,
980 gchar class_part[1024];
984 g_return_if_fail (arg_name != NULL);
985 g_return_if_fail (arg_type > GTK_TYPE_NONE);
986 g_return_if_fail (arg_id > 0);
987 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) != 0);
988 g_return_if_fail ((arg_flags & GTK_ARG_CHILD_ARG) == 0);
990 arg_flags &= GTK_ARG_MASK;
992 arg_part = strchr (arg_name, ':');
993 if (!arg_part || (arg_part[0] != ':') || (arg_part[1] != ':'))
995 g_warning ("invalid arg name: \"%s\"\n", arg_name);
999 strncpy (class_part, arg_name, (glong) (arg_part - arg_name));
1000 class_part[(glong) (arg_part - arg_name)] = '\0';
1002 class_type = gtk_type_from_name (class_part);
1005 g_warning ("gtk_object_add_arg_type(): invalid class name in arg: \"%s\"\n", arg_name);
1009 info = g_new (GtkArgInfo, 1);
1010 info->name = g_strdup (arg_name);
1011 info->type = arg_type;
1012 info->class_type = class_type;
1013 info->arg_flags = arg_flags;
1014 info->arg_id = arg_id;
1015 info->seq_id = ++((GtkObjectClass*) gtk_type_class (class_type))->n_args;
1018 arg_info_ht = g_hash_table_new (g_str_hash, g_str_equal);
1020 g_hash_table_insert (arg_info_ht, info->name, info);
1023 /*****************************************
1024 * gtk_object_get_arg_type:
1029 *****************************************/
1032 gtk_object_get_arg_type (const gchar *arg_name)
1038 g_return_val_if_fail (arg_name != NULL, 0);
1041 return GTK_TYPE_INVALID;
1043 if (!arg_name || strlen (arg_name) > 120)
1047 g_warning ("gtk_object_get_arg_type(): argument `arg_name' exceeds maximum size.");
1048 return GTK_TYPE_INVALID;
1051 t = strchr (arg_name, ':');
1052 if (!t || (t[0] != ':') || (t[1] != ':'))
1054 g_warning ("gtk_object_get_arg_type(): invalid arg name: \"%s\"\n", arg_name);
1055 return GTK_TYPE_INVALID;
1058 t = strchr (t + 2, ':');
1061 strncpy (buffer, arg_name, (long) (t - arg_name));
1062 buffer[(long) (t - arg_name)] = '\0';
1066 info = g_hash_table_lookup (arg_info_ht, arg_name);
1070 return GTK_TYPE_INVALID;
1073 /*****************************************
1074 * GtkObject object_data mechanism
1076 *****************************************/
1079 gtk_object_set_data_by_id (GtkObject *object,
1083 g_return_if_fail (data_id > 0);
1085 gtk_object_set_data_by_id_full (object, data_id, data, NULL);
1089 gtk_object_set_data (GtkObject *object,
1093 g_return_if_fail (key != NULL);
1095 gtk_object_set_data_by_id_full (object, gtk_object_data_force_id (key), data, NULL);
1099 gtk_object_set_data_by_id_full (GtkObject *object,
1102 GtkDestroyNotify destroy)
1104 GtkObjectData *odata;
1106 g_return_if_fail (object != NULL);
1107 g_return_if_fail (GTK_IS_OBJECT (object));
1108 g_return_if_fail (data_id > 0);
1110 odata = object->object_data;
1113 GtkObjectData *prev;
1119 if (odata->id == data_id)
1122 prev->next = odata->next;
1124 object->object_data = odata->next;
1126 GTK_OBJECT_DATA_DESTROY (odata);
1131 odata = odata->next;
1138 if (odata->id == data_id)
1140 register GtkDestroyNotify dfunc;
1141 register gpointer ddata;
1143 dfunc = odata->destroy;
1144 ddata = odata->data;
1145 odata->destroy = destroy;
1148 /* we need to have updated all structures prior to
1149 * invokation of the destroy function
1157 odata = odata->next;
1160 if (gtk_object_data_free_list)
1162 odata = gtk_object_data_free_list;
1163 gtk_object_data_free_list = odata->next;
1167 GtkObjectData *odata_block;
1170 odata_block = g_new0 (GtkObjectData, GTK_OBJECT_DATA_BLOCK_SIZE);
1171 for (i = 1; i < GTK_OBJECT_DATA_BLOCK_SIZE; i++)
1173 (odata_block + i)->next = gtk_object_data_free_list;
1174 gtk_object_data_free_list = (odata_block + i);
1177 odata = odata_block;
1180 odata->id = data_id;
1182 odata->destroy = destroy;
1183 odata->next = object->object_data;
1185 object->object_data = odata;
1190 gtk_object_set_data_full (GtkObject *object,
1193 GtkDestroyNotify destroy)
1195 g_return_if_fail (key != NULL);
1197 gtk_object_set_data_by_id_full (object, gtk_object_data_force_id (key), data, destroy);
1201 gtk_object_get_data_by_id (GtkObject *object,
1204 GtkObjectData *odata;
1206 g_return_val_if_fail (object != NULL, NULL);
1207 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1211 odata = object->object_data;
1214 if (odata->id == data_id)
1216 odata = odata->next;
1224 gtk_object_get_data (GtkObject *object,
1229 g_return_val_if_fail (key != NULL, NULL);
1231 id = gtk_object_data_try_key (key);
1233 return gtk_object_get_data_by_id (object, id);
1239 gtk_object_remove_data_by_id (GtkObject *object,
1243 gtk_object_set_data_by_id_full (object, data_id, NULL, NULL);
1247 gtk_object_remove_data (GtkObject *object,
1252 g_return_if_fail (key != NULL);
1254 id = gtk_object_data_try_key (key);
1256 gtk_object_set_data_by_id_full (object, id, NULL, NULL);
1259 /*****************************************
1260 * gtk_object_set_user_data:
1265 *****************************************/
1268 gtk_object_set_user_data (GtkObject *object,
1271 if (!user_data_key_id)
1272 user_data_key_id = g_quark_from_static_string (user_data_key);
1274 gtk_object_set_data_by_id_full (object, user_data_key_id, data, NULL);
1277 /*****************************************
1278 * gtk_object_get_user_data:
1283 *****************************************/
1286 gtk_object_get_user_data (GtkObject *object)
1288 if (user_data_key_id)
1289 return gtk_object_get_data_by_id (object, user_data_key_id);
1294 /*****************************************
1295 * gtk_object_check_cast:
1300 *****************************************/
1303 gtk_object_descriptive_type_name (GtkType type)
1307 name = gtk_type_name (type);
1315 gtk_object_check_cast (GtkObject *obj,
1320 g_warning ("invalid cast from (NULL) pointer to `%s'",
1321 gtk_object_descriptive_type_name (cast_type));
1326 g_warning ("invalid unclassed pointer in cast to `%s'",
1327 gtk_object_descriptive_type_name (cast_type));
1330 if (obj->klass->type < GTK_TYPE_OBJECT)
1332 g_warning ("invalid class type `%s' in cast to `%s'",
1333 gtk_object_descriptive_type_name (obj->klass->type),
1334 gtk_object_descriptive_type_name (cast_type));
1337 if (!gtk_type_is_a (obj->klass->type, cast_type))
1339 g_warning ("invalid cast from `%s' to `%s'",
1340 gtk_object_descriptive_type_name (obj->klass->type),
1341 gtk_object_descriptive_type_name (cast_type));
1348 /*****************************************
1349 * gtk_object_check_class_cast:
1354 *****************************************/
1357 gtk_object_check_class_cast (GtkObjectClass *klass,
1362 g_warning ("invalid class cast from (NULL) pointer to `%s'",
1363 gtk_object_descriptive_type_name (cast_type));
1366 if (klass->type < GTK_TYPE_OBJECT)
1368 g_warning ("invalid class type `%s' in class cast to `%s'",
1369 gtk_object_descriptive_type_name (klass->type),
1370 gtk_object_descriptive_type_name (cast_type));
1373 if (!gtk_type_is_a (klass->type, cast_type))
1375 g_warning ("invalid class cast from `%s' to `%s'",
1376 gtk_object_descriptive_type_name (klass->type),
1377 gtk_object_descriptive_type_name (cast_type));
1384 /*****************************************
1385 * gtk_object_collect_args:
1390 *****************************************/
1393 gtk_object_collect_args (guint *nargs,
1394 GtkType (*get_arg_type) (const gchar*),
1409 name = va_arg (args1, char *);
1416 type = get_arg_type (name);
1418 switch (GTK_FUNDAMENTAL_TYPE (type))
1420 case GTK_TYPE_INVALID:
1421 g_warning ("GTK: invalid arg name: \"%s\" %x\n", name, type);
1422 (void) va_arg (args1, long);
1431 case GTK_TYPE_FLAGS:
1432 (void) va_arg (args1, gint);
1435 case GTK_TYPE_ULONG:
1436 (void) va_arg (args1, glong);
1438 case GTK_TYPE_FLOAT:
1439 (void) va_arg (args1, gfloat);
1441 case GTK_TYPE_DOUBLE:
1442 (void) va_arg (args1, gdouble);
1444 case GTK_TYPE_STRING:
1445 (void) va_arg (args1, gchar*);
1447 case GTK_TYPE_POINTER:
1448 case GTK_TYPE_BOXED:
1449 (void) va_arg (args1, gpointer);
1451 case GTK_TYPE_SIGNAL:
1452 (void) va_arg (args1, GtkFunction);
1453 (void) va_arg (args1, gpointer);
1455 case GTK_TYPE_FOREIGN:
1456 (void) va_arg (args1, gpointer);
1457 (void) va_arg (args1, GtkDestroyNotify);
1459 case GTK_TYPE_CALLBACK:
1460 (void) va_arg (args1, GtkCallbackMarshal);
1461 (void) va_arg (args1, gpointer);
1462 (void) va_arg (args1, GtkDestroyNotify);
1464 case GTK_TYPE_C_CALLBACK:
1465 (void) va_arg (args1, GtkFunction);
1466 (void) va_arg (args1, gpointer);
1469 (void) va_arg (args1, gint);
1470 (void) va_arg (args1, GtkArg*);
1472 case GTK_TYPE_OBJECT:
1473 (void) va_arg (args1, GtkObject*);
1476 g_error ("unsupported type %s in args", gtk_type_name (type));
1488 args = g_new0 (GtkArg, n);
1490 for (i = 0; i < n; i++)
1492 args[i].name = va_arg (args2, char *);
1493 args[i].type = get_arg_type (args[i].name);
1495 switch (GTK_FUNDAMENTAL_TYPE (args[i].type))
1497 case GTK_TYPE_INVALID:
1498 (void) va_arg (args2, long);
1504 GTK_VALUE_CHAR(args[i]) = va_arg (args2, gint);
1507 GTK_VALUE_BOOL(args[i]) = va_arg (args2, gint);
1510 GTK_VALUE_INT(args[i]) = va_arg (args2, gint);
1513 GTK_VALUE_UINT(args[i]) = va_arg (args2, guint);
1516 GTK_VALUE_ENUM(args[i]) = va_arg (args2, gint);
1518 case GTK_TYPE_FLAGS:
1519 GTK_VALUE_FLAGS(args[i]) = va_arg (args2, gint);
1522 GTK_VALUE_LONG(args[i]) = va_arg (args2, glong);
1524 case GTK_TYPE_ULONG:
1525 GTK_VALUE_ULONG(args[i]) = va_arg (args2, gulong);
1527 case GTK_TYPE_FLOAT:
1528 GTK_VALUE_FLOAT(args[i]) = va_arg (args2, gfloat);
1530 case GTK_TYPE_DOUBLE:
1531 GTK_VALUE_DOUBLE(args[i]) = va_arg (args2, gdouble);
1533 case GTK_TYPE_STRING:
1534 GTK_VALUE_STRING(args[i]) = va_arg (args2, gchar*);
1536 case GTK_TYPE_POINTER:
1537 GTK_VALUE_POINTER(args[i]) = va_arg (args2, gpointer);
1539 case GTK_TYPE_BOXED:
1540 GTK_VALUE_BOXED(args[i]) = va_arg (args2, gpointer);
1542 case GTK_TYPE_SIGNAL:
1543 GTK_VALUE_SIGNAL(args[i]).f = va_arg (args2, GtkFunction);
1544 GTK_VALUE_SIGNAL(args[i]).d = va_arg (args2, gpointer);
1546 case GTK_TYPE_FOREIGN:
1547 GTK_VALUE_FOREIGN(args[i]).data = va_arg (args2, gpointer);
1548 GTK_VALUE_FOREIGN(args[i]).notify =
1549 va_arg (args2, GtkDestroyNotify);
1551 case GTK_TYPE_CALLBACK:
1552 GTK_VALUE_CALLBACK(args[i]).marshal =
1553 va_arg (args2, GtkCallbackMarshal);
1554 GTK_VALUE_CALLBACK(args[i]).data = va_arg (args2, gpointer);
1555 GTK_VALUE_CALLBACK(args[i]).notify =
1556 va_arg (args2, GtkDestroyNotify);
1558 case GTK_TYPE_C_CALLBACK:
1559 GTK_VALUE_C_CALLBACK(args[i]).func = va_arg (args2, GtkFunction);
1560 GTK_VALUE_C_CALLBACK(args[i]).func_data =
1561 va_arg (args2, gpointer);
1564 GTK_VALUE_ARGS(args[i]).n_args = va_arg (args2, gint);
1565 GTK_VALUE_ARGS(args[i]).args = va_arg (args2, GtkArg*);
1567 case GTK_TYPE_OBJECT:
1568 GTK_VALUE_OBJECT(args[i]) = va_arg (args2, GtkObject*);
1569 g_assert (GTK_VALUE_OBJECT(args[i]) == NULL ||
1570 GTK_CHECK_TYPE (GTK_VALUE_OBJECT(args[i]),
1574 g_error ("unsupported type %s in args",
1575 gtk_type_name (args[i].type));
1586 #undef gtk_object_ref
1587 #undef gtk_object_unref
1590 gtk_object_ref (GtkObject *object)
1592 g_return_if_fail (object != NULL);
1593 g_return_if_fail (GTK_IS_OBJECT (object));
1595 object->ref_count += 1;
1599 gtk_object_unref (GtkObject *object)
1601 g_return_if_fail (object != NULL);
1602 g_return_if_fail (GTK_IS_OBJECT (object));
1604 if (object->ref_count == 1)
1605 gtk_object_destroy (object);
1607 if (object->ref_count > 0)
1608 object->ref_count -= 1;
1610 if (object->ref_count == 0)
1612 #ifdef G_ENABLE_DEBUG
1613 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1615 g_assert (g_hash_table_lookup (living_objs_ht, object) == object);
1616 g_hash_table_remove (living_objs_ht, object);
1619 #endif /* G_ENABLE_DEBUG */
1620 object->klass->finalize (object);
1624 static GtkObject *gtk_trace_object = NULL;
1626 gtk_trace_referencing (GtkObject *object,
1632 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1634 gboolean exists = TRUE;
1636 g_return_if_fail (object != NULL);
1637 g_return_if_fail (GTK_IS_OBJECT (object));
1639 #ifdef G_ENABLE_DEBUG
1640 exists = g_hash_table_lookup (living_objs_ht, object) != NULL;
1641 #endif /* G_ENABLE_DEBUG */
1644 (object == gtk_trace_object ||
1645 gtk_trace_object == (void*)42))
1646 fprintf (stdout, "trace: object_%s: (%s:%p)->ref_count=%d %s (%s:%d)\n",
1647 do_ref ? "ref" : "unref",
1648 gtk_type_name (GTK_OBJECT_TYPE (object)),
1651 do_ref ? "+ 1" : "- 1",
1655 fprintf (stdout, "trace: object_%s(%p): no such object! (%s:%d)\n",
1656 do_ref ? "ref" : "unref",
1663 gtk_object_ref (object);
1665 gtk_object_unref (object);