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 OBJECT_DATA_ID_CHUNK 1024
42 typedef struct _GtkObjectData GtkObjectData;
43 typedef struct _GtkArgInfo GtkArgInfo;
49 GtkDestroyNotify destroy;
64 static void gtk_object_class_init (GtkObjectClass *klass);
65 static void gtk_object_init (GtkObject *object);
66 static void gtk_object_set_arg (GtkObject *object,
69 static void gtk_object_get_arg (GtkObject *object,
72 static void gtk_object_shutdown (GtkObject *object);
73 static void gtk_object_real_destroy (GtkObject *object);
74 static void gtk_object_finalize (GtkObject *object);
75 static void gtk_object_notify_weaks (gpointer data);
76 static void gtk_object_data_destroy (GtkObjectData *odata);
77 static guint* gtk_object_data_id_alloc (void);
79 GtkArg* gtk_object_collect_args (guint *nargs,
83 static guint object_signals[LAST_SIGNAL] = { 0 };
85 static GHashTable *object_data_ht = NULL;
86 static GMemChunk *object_data_mem_chunk = NULL;
87 static GSList *object_data_id_list = NULL;
88 static guint object_data_id_index = 0;
90 static GHashTable *arg_info_ht = NULL;
92 static const gchar *user_data_key = "user_data";
93 static guint user_data_key_id = 0;
97 static guint obj_count = 0;
98 static GHashTable *living_objs_ht = NULL;
100 gtk_object_debug_foreach (gpointer key, gpointer value, gpointer user_data)
104 object = (GtkObject*) value;
105 g_print ("GTK-DEBUG: %p: %s ref_count=%d%s%s\n",
107 gtk_type_name (GTK_OBJECT_TYPE (object)),
109 GTK_OBJECT_FLOATING (object) ? " (floating)" : "",
110 GTK_OBJECT_DESTROYED (object) ? " (destroyed)" : "");
113 gtk_object_debug (void)
115 g_hash_table_foreach (living_objs_ht, gtk_object_debug_foreach, NULL);
117 g_print ("GTK-DEBUG: living objects count = %d\n", obj_count);
120 gtk_object_pointer_hash (const gpointer v)
128 #endif /* G_ENABLE_DEBUG */
130 /****************************************************
131 * GtkObject type, class and instance initialization
133 ****************************************************/
136 gtk_object_init_type ()
138 GtkType object_type = 0;
139 GtkTypeInfo object_info =
143 sizeof (GtkObjectClass),
144 (GtkClassInitFunc) gtk_object_class_init,
145 (GtkObjectInitFunc) gtk_object_init,
150 object_type = gtk_type_unique (0, &object_info);
151 g_assert (object_type == GTK_TYPE_OBJECT);
153 #ifdef G_ENABLE_DEBUG
154 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
155 ATEXIT (gtk_object_debug);
156 #endif /* G_ENABLE_DEBUG */
160 gtk_object_get_type ()
162 return GTK_TYPE_OBJECT;
166 gtk_object_class_init (GtkObjectClass *class)
168 class->signals = NULL;
172 gtk_object_add_arg_type ("GtkObject::user_data",
176 gtk_object_add_arg_type ("GtkObject::signal",
180 gtk_object_add_arg_type ("GtkObject::object_signal",
185 object_signals[DESTROY] =
186 gtk_signal_new ("destroy",
189 GTK_SIGNAL_OFFSET (GtkObjectClass, destroy),
190 gtk_signal_default_marshaller,
193 gtk_object_class_add_signals (class, object_signals, LAST_SIGNAL);
195 class->shutdown = gtk_object_shutdown;
196 class->destroy = gtk_object_real_destroy;
197 class->finalize = gtk_object_finalize;
201 gtk_object_init (GtkObject *object)
203 GTK_OBJECT_FLAGS (object) = GTK_FLOATING;
205 object->ref_count = 1;
206 object->object_data = NULL;
208 #ifdef G_ENABLE_DEBUG
209 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
214 living_objs_ht = g_hash_table_new (gtk_object_pointer_hash, NULL);
216 g_hash_table_insert (living_objs_ht, object, object);
218 #endif /* G_ENABLE_DEBUG */
221 /********************************************
222 * Functions to end a GtkObject's life time
224 ********************************************/
227 gtk_object_destroy (GtkObject *object)
229 g_return_if_fail (object != NULL);
230 g_return_if_fail (GTK_IS_OBJECT (object));
232 if (!GTK_OBJECT_DESTROYED (object))
234 /* we will hold a reference on the object in this place, so
235 * to ease all classes shutdown and destroy implementations.
236 * i.e. they don't have to bother about referencing at all.
238 gtk_object_ref (object);
239 object->klass->shutdown (object);
240 gtk_object_unref (object);
245 gtk_object_shutdown (GtkObject *object)
247 GTK_OBJECT_SET_FLAGS (object, GTK_DESTROYED);
248 gtk_signal_emit (object, object_signals[DESTROY]);
252 gtk_object_real_destroy (GtkObject *object)
254 gtk_signal_handlers_destroy (object);
258 gtk_object_finalize (GtkObject *object)
260 GtkObjectData *odata, *next;
262 odata = object->object_data;
266 gtk_object_data_destroy (odata);
273 /*****************************************
274 * GtkObject argument handlers
276 *****************************************/
279 gtk_object_set_arg (GtkObject *object,
286 gtk_object_set_user_data (object, GTK_VALUE_POINTER (*arg));
289 if ((arg->name[9 + 2 + 6] != ':') || (arg->name[9 + 2 + 7] != ':'))
291 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
292 arg->type = GTK_TYPE_INVALID;
295 gtk_signal_connect (object, arg->name + 9 + 2 + 6 + 2,
296 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
297 GTK_VALUE_SIGNAL (*arg).d);
299 case ARG_OBJECT_SIGNAL:
300 if ((arg->name[9 + 2 + 13] != ':') || (arg->name[9 + 2 + 14] != ':'))
302 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
303 arg->type = GTK_TYPE_INVALID;
306 gtk_signal_connect_object (object, arg->name + 9 + 2 + 13 + 2,
307 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
308 (GtkObject*) GTK_VALUE_SIGNAL (*arg).d);
311 arg->type = GTK_TYPE_INVALID;
317 gtk_object_get_arg (GtkObject *object,
324 GTK_VALUE_POINTER (*arg) = gtk_object_get_user_data (object);
327 case ARG_OBJECT_SIGNAL:
329 arg->type = GTK_TYPE_INVALID;
334 /*****************************************
335 * gtk_object_class_add_signals:
340 *****************************************/
343 gtk_object_class_add_signals (GtkObjectClass *class,
350 g_return_if_fail (class != NULL);
352 new_signals = g_new (guint, class->nsignals + nsignals);
353 for (i = 0; i < class->nsignals; i++)
354 new_signals[i] = class->signals[i];
355 for (i = 0; i < nsignals; i++)
356 new_signals[class->nsignals + i] = signals[i];
358 g_free (class->signals);
359 class->signals = new_signals;
360 class->nsignals += nsignals;
363 /*****************************************
364 * gtk_object_class_add_user_signal:
369 *****************************************/
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);
417 /*****************************************
423 *****************************************/
426 gtk_object_sink (GtkObject *object)
428 g_return_if_fail (object != NULL);
429 g_return_if_fail (GTK_IS_OBJECT (object));
431 if (GTK_OBJECT_FLOATING (object))
433 GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING);
434 gtk_object_unref (object);
438 /*****************************************
441 * Weak refs are very similar to the old "destroy" signal. They allow
442 * one to register a callback that is called when the weakly
443 * referenced object is finalized.
445 * They are not implemented as a signal because they really are
446 * special and need to be used with great care. Unlike signals, who
447 * should be able to execute any code whatsoever.
449 * A weakref callback is not allowed to retain a reference to the
450 * object. In fact, the object is no longer there at all when it is
453 * A weakref callback is called atmost once.
455 *****************************************/
457 typedef struct _GtkWeakRef GtkWeakRef;
462 GtkDestroyNotify notify;
466 static const gchar *weakrefs_key = "gtk-weakrefs";
469 gtk_object_weakref (GtkObject *object,
470 GtkDestroyNotify notify,
475 g_return_if_fail (object != NULL);
476 g_return_if_fail (notify != NULL);
477 g_return_if_fail (GTK_IS_OBJECT (object));
479 weak = g_new (GtkWeakRef, 1);
480 weak->next = gtk_object_get_data (object, weakrefs_key);
481 weak->notify = notify;
483 gtk_object_set_data_full (object, weakrefs_key, weak,
484 gtk_object_notify_weaks);
488 gtk_object_weakunref (GtkObject *object,
489 GtkDestroyNotify notify,
492 GtkWeakRef *weaks, *w, **wp;
494 g_return_if_fail (object != NULL);
495 g_return_if_fail (GTK_IS_OBJECT (object));
497 weaks = gtk_object_get_data (object, weakrefs_key);
498 for (wp = &weaks; *wp; wp = &(*wp)->next)
501 if (w->notify == notify && w->data == data)
504 gtk_object_set_data_full (object, weakrefs_key, w->next,
505 gtk_object_notify_weaks);
515 gtk_object_notify_weaks (gpointer data)
519 w1 = (GtkWeakRef *)data;
523 w1->notify (w1->data);
530 /*****************************************
536 *****************************************/
539 gtk_object_new (GtkType type,
548 obj = gtk_type_new (type);
550 va_start (args1, type);
551 va_start (args2, type);
553 args = gtk_object_collect_args (&nargs, args1, args2);
554 gtk_object_setv (obj, nargs, args);
563 /*****************************************
569 *****************************************/
572 gtk_object_newv (GtkType type,
578 obj = gtk_type_new (type);
579 gtk_object_setv (obj, nargs, args);
584 /*****************************************
590 *****************************************/
593 gtk_object_getv (GtkObject *object,
599 g_return_if_fail (object != NULL);
600 g_return_if_fail (GTK_IS_OBJECT (object));
605 for (i = 0; i < nargs; i++)
612 /* hm, the name cutting shouldn't be needed on gets, but what the heck...
614 lookup_name = g_strdup (args[i].name);
615 d = strchr (lookup_name, ':');
616 if (d && d[1] == ':')
618 d = strchr (d + 2, ':');
622 info = g_hash_table_lookup (arg_info_ht, lookup_name);
629 g_warning ("invalid arg name: \"%s\"\n", lookup_name);
630 args[i].type = GTK_TYPE_INVALID;
631 g_free (lookup_name);
634 else if (!gtk_type_is_a (object->klass->type, info->class_type))
636 g_warning ("invalid arg for %s: \"%s\"\n", gtk_type_name (object->klass->type), lookup_name);
637 args[i].type = GTK_TYPE_INVALID;
638 g_free (lookup_name);
641 else if (! (info->arg_flags & GTK_ARG_READABLE))
643 g_warning ("arg is not supplied for read-access: \"%s\"\n", lookup_name);
644 args[i].type = GTK_TYPE_INVALID;
645 g_free (lookup_name);
649 g_free (lookup_name);
651 args[i].type = info->type;
652 gtk_type_get_arg (object, info->class_type, &args[i], info->arg_id);
656 /*****************************************
657 * gtk_object_query_args:
662 *****************************************/
664 struct _GtkQueryArgData
669 typedef struct _GtkQueryArgData GtkQueryArgData;
672 gtk_query_arg_foreach (gpointer key,
676 register GtkArgInfo *info;
677 register GtkQueryArgData *data;
682 if (info->class_type == data->class_type)
683 data->arg_list = g_list_prepend (data->arg_list, info);
687 gtk_object_query_args (GtkType class_type,
692 GtkQueryArgData query_data;
696 g_return_val_if_fail (nargs != NULL, NULL);
698 g_return_val_if_fail (gtk_type_is_a (class_type, gtk_object_get_type ()), NULL);
703 /* make sure the types class has been initialized, because
704 * the argument setup happens in the gtk_*_class_init() functions.
706 gtk_type_class (class_type);
708 query_data.arg_list = NULL;
709 query_data.class_type = class_type;
710 g_hash_table_foreach (arg_info_ht, gtk_query_arg_foreach, &query_data);
712 if (query_data.arg_list)
714 register GList *list;
717 list = query_data.arg_list;
724 g_assert (len == ((GtkObjectClass*) gtk_type_class (class_type))->n_args); /* paranoid */
726 args = g_new0 (GtkArg, len);
729 *arg_flags = g_new (guint32, len);
738 g_assert (info->seq_id > 0 && info->seq_id <= len); /* paranoid */
740 args[info->seq_id - 1].type = info->type;
741 args[info->seq_id - 1].name = info->name;
743 (*arg_flags)[info->seq_id - 1] = info->arg_flags;
747 g_list_free (query_data.arg_list);
755 /*****************************************
761 *****************************************/
764 gtk_object_set (GtkObject *object,
772 g_return_if_fail (object != NULL);
774 va_start (args1, object);
775 va_start (args2, object);
777 args = gtk_object_collect_args (&nargs, args1, args2);
778 gtk_object_setv (object, nargs, args);
785 /*****************************************
791 *****************************************/
794 gtk_object_setv (GtkObject *object,
800 g_return_if_fail (object != NULL);
801 g_return_if_fail (GTK_OBJECT (object));
806 for (i = 0; i < nargs; i++)
813 lookup_name = g_strdup (args[i].name);
814 d = strchr (lookup_name, ':');
815 if (d && d[1] == ':')
817 d = strchr (d + 2, ':');
821 info = g_hash_table_lookup (arg_info_ht, lookup_name);
830 g_warning ("invalid arg name: \"%s\"\n", lookup_name);
833 else if (info->type != args[i].type)
835 g_warning ("invalid arg type for: \"%s\"\n", lookup_name);
838 else if (!gtk_type_is_a (object->klass->type, info->class_type))
840 g_warning ("invalid arg for %s: \"%s\"\n", gtk_type_name (object->klass->type), lookup_name);
843 else if (! (info->arg_flags & GTK_ARG_WRITABLE))
845 g_warning ("arg is not supplied for write-access: \"%s\"\n", lookup_name);
849 g_free (lookup_name);
854 gtk_type_set_arg (object, info->class_type, &args[i], info->arg_id);
858 /*****************************************
859 * gtk_object_add_arg_type:
864 *****************************************/
867 gtk_object_add_arg_type (const char *arg_name,
873 gchar class_part[1024];
877 g_return_if_fail (arg_name != NULL);
878 g_return_if_fail (arg_type > GTK_TYPE_NONE);
879 g_return_if_fail (arg_id > 0);
880 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) != 0);
882 arg_part = strchr (arg_name, ':');
883 if (!arg_part || (arg_part[0] != ':') || (arg_part[1] != ':'))
885 g_warning ("invalid arg name: \"%s\"\n", arg_name);
889 strncpy (class_part, arg_name, (glong) (arg_part - arg_name));
890 class_part[(glong) (arg_part - arg_name)] = '\0';
892 class_type = gtk_type_from_name (class_part);
895 g_warning ("invalid class name in arg: \"%s\"\n", arg_name);
899 info = g_new (GtkArgInfo, 1);
900 info->name = g_strdup (arg_name);
901 info->type = arg_type;
902 info->class_type = class_type;
903 info->arg_flags = arg_flags & (GTK_ARG_READABLE | GTK_ARG_WRITABLE);
904 info->arg_id = arg_id;
905 info->seq_id = ++((GtkObjectClass*) gtk_type_class (class_type))->n_args;
908 arg_info_ht = g_hash_table_new (g_str_hash, g_str_equal);
910 g_hash_table_insert (arg_info_ht, info->name, info);
913 /*****************************************
914 * gtk_object_get_arg_type:
919 *****************************************/
922 gtk_object_get_arg_type (const gchar *arg_name)
929 return GTK_TYPE_INVALID;
931 t = strchr (arg_name, ':');
932 if (!t || (t[0] != ':') || (t[1] != ':'))
934 g_warning ("invalid arg name: \"%s\"\n", arg_name);
935 return GTK_TYPE_INVALID;
938 t = strchr (t + 2, ':');
941 strncpy (buffer, arg_name, (long) (t - arg_name));
942 buffer[(long) (t - arg_name)] = '\0';
946 info = g_hash_table_lookup (arg_info_ht, (gpointer) arg_name);
950 return GTK_TYPE_INVALID;
953 /*****************************************
954 * GtkObject object_data mechanism
956 *****************************************/
959 gtk_object_set_data_by_id (GtkObject *object,
963 g_return_if_fail (data_id > 0);
965 gtk_object_set_data_by_id_full (object, data_id, data, NULL);
969 gtk_object_set_data (GtkObject *object,
973 g_return_if_fail (key != NULL);
975 gtk_object_set_data_by_id_full (object, gtk_object_data_force_id (key), data, NULL);
979 gtk_object_set_data_by_id_full (GtkObject *object,
982 GtkDestroyNotify destroy)
984 GtkObjectData *odata;
987 g_return_if_fail (object != NULL);
988 g_return_if_fail (GTK_IS_OBJECT (object));
989 g_return_if_fail (data_id > 0);
994 odata = object->object_data;
998 if (odata->id == data_id)
1001 prev->next = odata->next;
1002 if (odata == object->object_data)
1003 object->object_data = odata->next;
1005 gtk_object_data_destroy (odata);
1010 odata = odata->next;
1015 odata = object->object_data;
1018 if (odata->id == data_id)
1021 odata->destroy (odata->data);
1024 odata->destroy = destroy;
1028 odata = odata->next;
1031 if (!object_data_mem_chunk)
1032 object_data_mem_chunk = g_mem_chunk_new ("object data mem chunk",
1033 sizeof (GtkObjectData),
1034 1024, G_ALLOC_AND_FREE);
1036 odata = g_chunk_new (GtkObjectData, object_data_mem_chunk);
1037 odata->id = data_id;
1039 odata->destroy = destroy;
1040 odata->next = object->object_data;
1042 object->object_data = odata;
1047 gtk_object_set_data_full (GtkObject *object,
1050 GtkDestroyNotify destroy)
1052 g_return_if_fail (key != NULL);
1054 gtk_object_set_data_by_id_full (object, gtk_object_data_force_id (key), data, destroy);
1058 gtk_object_data_force_id (const gchar *key)
1062 g_return_val_if_fail (key != NULL, 0);
1064 if (!object_data_ht)
1065 object_data_ht = g_hash_table_new (g_str_hash, g_str_equal);
1067 id = g_hash_table_lookup (object_data_ht, (gpointer) key);
1070 id = gtk_object_data_id_alloc ();
1071 g_hash_table_insert (object_data_ht, g_strdup (key), id);
1078 gtk_object_get_data_by_id (GtkObject *object,
1081 GtkObjectData *odata;
1083 g_return_val_if_fail (object != NULL, NULL);
1084 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1088 odata = object->object_data;
1091 if (odata->id == data_id)
1093 odata = odata->next;
1101 gtk_object_get_data (GtkObject *object,
1106 g_return_val_if_fail (key != NULL, NULL);
1108 id = gtk_object_data_try_key (key);
1110 return gtk_object_get_data_by_id (object, id);
1116 gtk_object_data_try_key (const gchar *key)
1118 g_return_val_if_fail (key != NULL, 0);
1124 id = g_hash_table_lookup (object_data_ht, (gpointer) key);
1133 gtk_object_remove_data_by_id (GtkObject *object,
1137 gtk_object_set_data_by_id_full (object, data_id, NULL, NULL);
1141 gtk_object_remove_data (GtkObject *object,
1146 g_return_if_fail (key != NULL);
1148 id = gtk_object_data_try_key (key);
1150 gtk_object_set_data_by_id_full (object, id, NULL, NULL);
1154 gtk_object_data_destroy (GtkObjectData *odata)
1156 g_return_if_fail (odata != NULL);
1159 odata->destroy (odata->data);
1161 g_mem_chunk_free (object_data_mem_chunk, odata);
1165 gtk_object_data_id_alloc ()
1167 static guint next_id = 1;
1170 if (!object_data_id_list ||
1171 (object_data_id_index == OBJECT_DATA_ID_CHUNK))
1173 ids = g_new (guint, OBJECT_DATA_ID_CHUNK);
1174 object_data_id_index = 0;
1175 object_data_id_list = g_slist_prepend (object_data_id_list, ids);
1179 ids = object_data_id_list->data;
1182 ids[object_data_id_index] = next_id++;
1183 return &ids[object_data_id_index++];
1186 /*****************************************
1187 * gtk_object_set_user_data:
1192 *****************************************/
1195 gtk_object_set_user_data (GtkObject *object,
1198 if (!user_data_key_id)
1199 user_data_key_id = gtk_object_data_force_id (user_data_key);
1201 gtk_object_set_data_by_id_full (object, user_data_key_id, data, NULL);
1204 /*****************************************
1205 * gtk_object_get_user_data:
1210 *****************************************/
1213 gtk_object_get_user_data (GtkObject *object)
1215 if (user_data_key_id)
1216 return gtk_object_get_data_by_id (object, user_data_key_id);
1221 /*****************************************
1222 * gtk_object_check_cast:
1227 *****************************************/
1230 gtk_object_descriptive_type_name (GtkType type)
1234 name = gtk_type_name (type);
1242 gtk_object_check_cast (GtkObject *obj,
1247 g_warning ("invalid cast from (NULL) pointer to `%s'",
1248 gtk_object_descriptive_type_name (cast_type));
1253 g_warning ("invalid unclassed pointer in cast to `%s'",
1254 gtk_object_descriptive_type_name (cast_type));
1257 if (obj->klass->type < GTK_TYPE_OBJECT)
1259 g_warning ("invalid class type `%s' in cast to `%s'",
1260 gtk_object_descriptive_type_name (obj->klass->type),
1261 gtk_object_descriptive_type_name (cast_type));
1264 if (!gtk_type_is_a (obj->klass->type, cast_type))
1266 g_warning ("invalid cast from `%s' to `%s'",
1267 gtk_object_descriptive_type_name (obj->klass->type),
1268 gtk_object_descriptive_type_name (cast_type));
1275 /*****************************************
1276 * gtk_object_check_class_cast:
1281 *****************************************/
1284 gtk_object_check_class_cast (GtkObjectClass *klass,
1289 g_warning ("invalid class cast from (NULL) pointer to `%s'",
1290 gtk_object_descriptive_type_name (cast_type));
1293 if (klass->type < GTK_TYPE_OBJECT)
1295 g_warning ("invalid class type `%s' in class cast to `%s'",
1296 gtk_object_descriptive_type_name (klass->type),
1297 gtk_object_descriptive_type_name (cast_type));
1300 if (!gtk_type_is_a (klass->type, cast_type))
1302 g_warning ("invalid class cast from `%s' to `%s'",
1303 gtk_object_descriptive_type_name (klass->type),
1304 gtk_object_descriptive_type_name (cast_type));
1311 /*****************************************
1312 * gtk_object_collect_args:
1317 *****************************************/
1320 gtk_object_collect_args (guint *nargs,
1335 name = va_arg (args1, char *);
1342 type = gtk_object_get_arg_type (name);
1344 switch (GTK_FUNDAMENTAL_TYPE (type))
1346 case GTK_TYPE_INVALID:
1347 g_warning ("invalid arg name: \"%s\" %x\n", name, type);
1348 (void) va_arg (args1, long);
1357 case GTK_TYPE_FLAGS:
1358 (void) va_arg (args1, gint);
1361 case GTK_TYPE_ULONG:
1362 (void) va_arg (args1, glong);
1364 case GTK_TYPE_FLOAT:
1365 (void) va_arg (args1, gfloat);
1367 case GTK_TYPE_DOUBLE:
1368 (void) va_arg (args1, gdouble);
1370 case GTK_TYPE_STRING:
1371 (void) va_arg (args1, gchar*);
1373 case GTK_TYPE_POINTER:
1374 case GTK_TYPE_BOXED:
1375 (void) va_arg (args1, gpointer);
1377 case GTK_TYPE_SIGNAL:
1378 (void) va_arg (args1, GtkFunction);
1379 (void) va_arg (args1, gpointer);
1381 case GTK_TYPE_FOREIGN:
1382 (void) va_arg (args1, gpointer);
1383 (void) va_arg (args1, GtkDestroyNotify);
1385 case GTK_TYPE_CALLBACK:
1386 (void) va_arg (args1, GtkCallbackMarshal);
1387 (void) va_arg (args1, gpointer);
1388 (void) va_arg (args1, GtkDestroyNotify);
1390 case GTK_TYPE_C_CALLBACK:
1391 (void) va_arg (args1, GtkFunction);
1392 (void) va_arg (args1, gpointer);
1395 (void) va_arg (args1, gint);
1396 (void) va_arg (args1, GtkArg*);
1398 case GTK_TYPE_OBJECT:
1399 (void) va_arg (args1, GtkObject*);
1402 g_error ("unsupported type %s in args", gtk_type_name (type));
1414 args = g_new0 (GtkArg, n);
1416 for (i = 0; i < n; i++)
1418 args[i].name = va_arg (args2, char *);
1419 args[i].type = gtk_object_get_arg_type (args[i].name);
1421 switch (GTK_FUNDAMENTAL_TYPE (args[i].type))
1423 case GTK_TYPE_INVALID:
1424 (void) va_arg (args2, long);
1430 GTK_VALUE_CHAR(args[i]) = va_arg (args2, gint);
1433 GTK_VALUE_BOOL(args[i]) = va_arg (args2, gint);
1436 GTK_VALUE_INT(args[i]) = va_arg (args2, gint);
1439 GTK_VALUE_UINT(args[i]) = va_arg (args2, guint);
1442 GTK_VALUE_ENUM(args[i]) = va_arg (args2, gint);
1444 case GTK_TYPE_FLAGS:
1445 GTK_VALUE_FLAGS(args[i]) = va_arg (args2, gint);
1448 GTK_VALUE_LONG(args[i]) = va_arg (args2, glong);
1450 case GTK_TYPE_ULONG:
1451 GTK_VALUE_ULONG(args[i]) = va_arg (args2, gulong);
1453 case GTK_TYPE_FLOAT:
1454 GTK_VALUE_FLOAT(args[i]) = va_arg (args2, gfloat);
1456 case GTK_TYPE_DOUBLE:
1457 GTK_VALUE_DOUBLE(args[i]) = va_arg (args2, gdouble);
1459 case GTK_TYPE_STRING:
1460 GTK_VALUE_STRING(args[i]) = va_arg (args2, gchar*);
1462 case GTK_TYPE_POINTER:
1463 GTK_VALUE_POINTER(args[i]) = va_arg (args2, gpointer);
1465 case GTK_TYPE_BOXED:
1466 GTK_VALUE_BOXED(args[i]) = va_arg (args2, gpointer);
1468 case GTK_TYPE_SIGNAL:
1469 GTK_VALUE_SIGNAL(args[i]).f = va_arg (args2, GtkFunction);
1470 GTK_VALUE_SIGNAL(args[i]).d = va_arg (args2, gpointer);
1472 case GTK_TYPE_FOREIGN:
1473 GTK_VALUE_FOREIGN(args[i]).data = va_arg (args2, gpointer);
1474 GTK_VALUE_FOREIGN(args[i]).notify =
1475 va_arg (args2, GtkDestroyNotify);
1477 case GTK_TYPE_CALLBACK:
1478 GTK_VALUE_CALLBACK(args[i]).marshal =
1479 va_arg (args2, GtkCallbackMarshal);
1480 GTK_VALUE_CALLBACK(args[i]).data = va_arg (args2, gpointer);
1481 GTK_VALUE_CALLBACK(args[i]).notify =
1482 va_arg (args2, GtkDestroyNotify);
1484 case GTK_TYPE_C_CALLBACK:
1485 GTK_VALUE_C_CALLBACK(args[i]).func = va_arg (args2, GtkFunction);
1486 GTK_VALUE_C_CALLBACK(args[i]).func_data =
1487 va_arg (args2, gpointer);
1490 GTK_VALUE_ARGS(args[i]).n_args = va_arg (args2, gint);
1491 GTK_VALUE_ARGS(args[i]).args = va_arg (args2, GtkArg*);
1493 case GTK_TYPE_OBJECT:
1494 GTK_VALUE_OBJECT(args[i]) = va_arg (args2, GtkObject*);
1495 g_assert (GTK_VALUE_OBJECT(args[i]) == NULL ||
1496 GTK_CHECK_TYPE (GTK_VALUE_OBJECT(args[i]),
1500 g_error ("unsupported type %s in args",
1501 gtk_type_name (args[i].type));
1512 #undef gtk_object_ref
1513 #undef gtk_object_unref
1516 gtk_object_ref (GtkObject *object)
1518 g_return_if_fail (object != NULL);
1519 g_return_if_fail (GTK_IS_OBJECT (object));
1521 object->ref_count += 1;
1525 gtk_object_unref (GtkObject *object)
1527 g_return_if_fail (object != NULL);
1528 g_return_if_fail (GTK_IS_OBJECT (object));
1530 if (object->ref_count == 1)
1531 gtk_object_destroy (object);
1533 if (object->ref_count > 0)
1534 object->ref_count -= 1;
1536 if (object->ref_count == 0)
1538 #ifdef G_ENABLE_DEBUG
1539 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1541 g_assert (g_hash_table_lookup (living_objs_ht, object) == object);
1542 g_hash_table_remove (living_objs_ht, object);
1545 #endif /* G_ENABLE_DEBUG */
1546 object->klass->finalize (object);
1551 #ifdef G_ENABLE_DEBUG
1552 static GtkObject *gtk_trace_object = NULL;
1554 gtk_trace_referencing (gpointer *o,
1561 GtkObject *object = (GtkObject*) o;
1563 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1565 g_return_if_fail (object != NULL);
1566 g_return_if_fail (GTK_IS_OBJECT (object));
1568 exists = g_hash_table_lookup (living_objs_ht, object) != NULL;
1571 (object == gtk_trace_object ||
1572 gtk_trace_object == (void*)42))
1573 printf ("trace: object_%s: (%s:%p)->ref_count=%d%s (%s_f%02d:%d)\n",
1574 do_ref ? "ref" : "unref",
1575 gtk_type_name (GTK_OBJECT_TYPE (object)),
1578 do_ref ? " + 1" : " - 1 ",
1584 printf ("trace: object_%s(%p): no such object! (%s_f%02d:%d)\n",
1585 do_ref ? "ref" : "unref",
1593 gtk_object_ref (object);
1595 gtk_object_unref (object);
1597 #endif /* G_ENABLE_DEBUG */