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 Free
16 * Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
22 #include "gtkobject.h"
23 #include "gtksignal.h"
26 #define OBJECT_DATA_ID_CHUNK 1024
41 typedef struct _GtkObjectData GtkObjectData;
42 typedef struct _GtkArgInfo GtkArgInfo;
48 GtkDestroyNotify destroy;
63 static void gtk_object_class_init (GtkObjectClass *klass);
64 static void gtk_object_init (GtkObject *object);
65 static void gtk_object_set_arg (GtkObject *object,
68 static void gtk_object_get_arg (GtkObject *object,
71 static void gtk_object_shutdown (GtkObject *object);
72 static void gtk_object_real_destroy (GtkObject *object);
73 static void gtk_object_finalize (GtkObject *object);
74 static void gtk_object_notify_weaks (gpointer data);
75 static void gtk_object_data_destroy (GtkObjectData *odata);
76 static guint* gtk_object_data_id_alloc (void);
78 GtkArg* gtk_object_collect_args (guint *nargs,
82 static guint object_signals[LAST_SIGNAL] = { 0 };
84 static GHashTable *object_data_ht = NULL;
85 static GMemChunk *object_data_mem_chunk = NULL;
86 static GSList *object_data_id_list = NULL;
87 static guint object_data_id_index = 0;
89 static GHashTable *arg_info_ht = NULL;
91 static const gchar *user_data_key = "user_data";
92 static guint user_data_key_id = 0;
96 static guint obj_count = 0;
97 static GHashTable *living_objs_ht = NULL;
99 gtk_object_debug_foreach (gpointer key, gpointer value, gpointer user_data)
103 object = (GtkObject*) value;
104 g_print ("GTK-DEBUG: %p: %s ref_count=%d%s%s\n",
106 gtk_type_name (GTK_OBJECT_TYPE (object)),
108 GTK_OBJECT_FLOATING (object) ? " (floating)" : "",
109 GTK_OBJECT_DESTROYED (object) ? " (destroyed)" : "");
112 gtk_object_debug (void)
114 g_hash_table_foreach (living_objs_ht, gtk_object_debug_foreach, NULL);
116 g_print ("GTK-DEBUG: living objects count = %d\n", obj_count);
119 gtk_object_pointer_hash (const gpointer v)
127 #endif /* G_ENABLE_DEBUG */
129 /****************************************************
130 * GtkObject type, class and instance initialization
132 ****************************************************/
135 gtk_object_init_type ()
137 GtkType object_type = 0;
138 GtkTypeInfo object_info =
142 sizeof (GtkObjectClass),
143 (GtkClassInitFunc) gtk_object_class_init,
144 (GtkObjectInitFunc) gtk_object_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 ()
161 return GTK_TYPE_OBJECT;
165 gtk_object_class_init (GtkObjectClass *class)
167 class->signals = NULL;
171 gtk_object_add_arg_type ("GtkObject::user_data",
175 gtk_object_add_arg_type ("GtkObject::signal",
179 gtk_object_add_arg_type ("GtkObject::object_signal",
184 object_signals[DESTROY] =
185 gtk_signal_new ("destroy",
188 GTK_SIGNAL_OFFSET (GtkObjectClass, destroy),
189 gtk_signal_default_marshaller,
192 gtk_object_class_add_signals (class, object_signals, LAST_SIGNAL);
194 class->shutdown = gtk_object_shutdown;
195 class->destroy = gtk_object_real_destroy;
196 class->finalize = gtk_object_finalize;
200 gtk_object_init (GtkObject *object)
202 GTK_OBJECT_FLAGS (object) = GTK_FLOATING;
204 object->ref_count = 1;
205 object->object_data = NULL;
207 #ifdef G_ENABLE_DEBUG
208 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
213 living_objs_ht = g_hash_table_new (gtk_object_pointer_hash, NULL);
215 g_hash_table_insert (living_objs_ht, object, object);
217 #endif /* G_ENABLE_DEBUG */
220 /********************************************
221 * Functions to end a GtkObject's life time
223 ********************************************/
226 gtk_object_destroy (GtkObject *object)
228 g_return_if_fail (object != NULL);
229 g_return_if_fail (GTK_IS_OBJECT (object));
231 if (!GTK_OBJECT_DESTROYED (object))
233 /* we will hold a reference on the object in this place, so
234 * to ease all classes shutdown and destroy implementations.
235 * i.e. they don't have to bother about referencing at all.
237 gtk_object_ref (object);
238 object->klass->shutdown (object);
239 gtk_object_unref (object);
244 gtk_object_shutdown (GtkObject *object)
246 GTK_OBJECT_SET_FLAGS (object, GTK_DESTROYED);
247 gtk_signal_emit (object, object_signals[DESTROY]);
251 gtk_object_real_destroy (GtkObject *object)
253 gtk_signal_handlers_destroy (object);
257 gtk_object_finalize (GtkObject *object)
259 GtkObjectData *odata, *next;
261 odata = object->object_data;
265 gtk_object_data_destroy (odata);
272 /*****************************************
273 * GtkObject argument handlers
275 *****************************************/
278 gtk_object_set_arg (GtkObject *object,
285 gtk_object_set_user_data (object, GTK_VALUE_POINTER (*arg));
288 if ((arg->name[9 + 2 + 6] != ':') || (arg->name[9 + 2 + 7] != ':'))
290 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
291 arg->type = GTK_TYPE_INVALID;
294 gtk_signal_connect (object, arg->name + 9 + 2 + 6 + 2,
295 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
296 GTK_VALUE_SIGNAL (*arg).d);
298 case ARG_OBJECT_SIGNAL:
299 if ((arg->name[9 + 2 + 13] != ':') || (arg->name[9 + 2 + 14] != ':'))
301 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
302 arg->type = GTK_TYPE_INVALID;
305 gtk_signal_connect_object (object, arg->name + 9 + 2 + 13 + 2,
306 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
307 (GtkObject*) GTK_VALUE_SIGNAL (*arg).d);
310 arg->type = GTK_TYPE_INVALID;
316 gtk_object_get_arg (GtkObject *object,
323 GTK_VALUE_POINTER (*arg) = gtk_object_get_user_data (object);
326 case ARG_OBJECT_SIGNAL:
328 arg->type = GTK_TYPE_INVALID;
333 /*****************************************
334 * gtk_object_class_add_signals:
339 *****************************************/
342 gtk_object_class_add_signals (GtkObjectClass *class,
349 g_return_if_fail (class != NULL);
351 new_signals = g_new (guint, class->nsignals + nsignals);
352 for (i = 0; i < class->nsignals; i++)
353 new_signals[i] = class->signals[i];
354 for (i = 0; i < nsignals; i++)
355 new_signals[class->nsignals + i] = signals[i];
357 g_free (class->signals);
358 class->signals = new_signals;
359 class->nsignals += nsignals;
362 /*****************************************
363 * gtk_object_class_add_user_signal:
368 *****************************************/
371 gtk_object_class_add_user_signal (GtkObjectClass *class,
373 GtkSignalMarshaller marshaller,
383 g_return_val_if_fail (class != NULL, 0);
387 params = g_new (GtkType, nparams);
389 va_start (args, nparams);
391 for (i = 0; i < nparams; i++)
392 params[i] = va_arg (args, GtkType);
399 signal_id = gtk_signal_newv (name,
411 gtk_object_class_add_signals (class, &signal_id, 1);
416 /*****************************************
422 *****************************************/
425 gtk_object_sink (GtkObject *object)
427 g_return_if_fail (object != NULL);
428 g_return_if_fail (GTK_IS_OBJECT (object));
430 if (GTK_OBJECT_FLOATING (object))
432 GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING);
433 gtk_object_unref (object);
437 /*****************************************
440 * Weak refs are very similar to the old "destroy" signal. They allow
441 * one to register a callback that is called when the weakly
442 * referenced object is finalized.
444 * They are not implemented as a signal because they really are
445 * special and need to be used with great care. Unlike signals, who
446 * should be able to execute any code whatsoever.
448 * A weakref callback is not allowed to retain a reference to the
449 * object. In fact, the object is no longer there at all when it is
452 * A weakref callback is called atmost once.
454 *****************************************/
456 typedef struct _GtkWeakRef GtkWeakRef;
461 GtkDestroyNotify notify;
465 static const gchar *weakrefs_key = "gtk-weakrefs";
468 gtk_object_weakref (GtkObject *object,
469 GtkDestroyNotify notify,
474 g_return_if_fail (object != NULL);
475 g_return_if_fail (notify != NULL);
476 g_return_if_fail (GTK_IS_OBJECT (object));
478 weak = g_new (GtkWeakRef, 1);
479 weak->next = gtk_object_get_data (object, weakrefs_key);
480 weak->notify = notify;
482 gtk_object_set_data_full (object, weakrefs_key, weak,
483 gtk_object_notify_weaks);
487 gtk_object_weakunref (GtkObject *object,
488 GtkDestroyNotify notify,
491 GtkWeakRef *weaks, *w, **wp;
493 g_return_if_fail (object != NULL);
494 g_return_if_fail (GTK_IS_OBJECT (object));
496 weaks = gtk_object_get_data (object, weakrefs_key);
497 for (wp = &weaks; *wp; wp = &(*wp)->next)
500 if (w->notify == notify && w->data == data)
503 gtk_object_set_data_full (object, weakrefs_key, w->next,
504 gtk_object_notify_weaks);
514 gtk_object_notify_weaks (gpointer data)
518 w1 = (GtkWeakRef *)data;
522 w1->notify (w1->data);
529 /*****************************************
535 *****************************************/
538 gtk_object_new (GtkType type,
547 obj = gtk_type_new (type);
549 va_start (args1, type);
550 va_start (args2, type);
552 args = gtk_object_collect_args (&nargs, args1, args2);
553 gtk_object_setv (obj, nargs, args);
562 /*****************************************
568 *****************************************/
571 gtk_object_newv (GtkType type,
577 obj = gtk_type_new (type);
578 gtk_object_setv (obj, nargs, args);
583 /*****************************************
589 *****************************************/
592 gtk_object_getv (GtkObject *object,
598 g_return_if_fail (object != NULL);
599 g_return_if_fail (GTK_IS_OBJECT (object));
604 for (i = 0; i < nargs; i++)
611 /* hm, the name cutting shouldn't be needed on gets, but what the heck...
613 lookup_name = g_strdup (args[i].name);
614 d = strchr (lookup_name, ':');
615 if (d && d[1] == ':')
617 d = strchr (d + 2, ':');
621 info = g_hash_table_lookup (arg_info_ht, lookup_name);
628 g_warning ("invalid arg name: \"%s\"\n", lookup_name);
629 args[i].type = GTK_TYPE_INVALID;
630 g_free (lookup_name);
633 else if (!gtk_type_is_a (object->klass->type, info->class_type))
635 g_warning ("invalid arg for %s: \"%s\"\n", gtk_type_name (object->klass->type), lookup_name);
636 args[i].type = GTK_TYPE_INVALID;
637 g_free (lookup_name);
640 else if (! (info->arg_flags & GTK_ARG_READABLE))
642 g_warning ("arg is not supplied for read-access: \"%s\"\n", lookup_name);
643 args[i].type = GTK_TYPE_INVALID;
644 g_free (lookup_name);
648 g_free (lookup_name);
650 args[i].type = info->type;
651 gtk_type_get_arg (object, info->class_type, &args[i], info->arg_id);
655 /*****************************************
656 * gtk_object_query_args:
661 *****************************************/
663 struct _GtkQueryArgData
668 typedef struct _GtkQueryArgData GtkQueryArgData;
671 gtk_query_arg_foreach (gpointer key,
675 register GtkArgInfo *info;
676 register GtkQueryArgData *data;
681 if (info->class_type == data->class_type)
682 data->arg_list = g_list_prepend (data->arg_list, info);
686 gtk_object_query_args (GtkType class_type,
691 GtkQueryArgData query_data;
695 g_return_val_if_fail (nargs != NULL, NULL);
697 g_return_val_if_fail (gtk_type_is_a (class_type, gtk_object_get_type ()), NULL);
702 /* make sure the types class has been initialized, because
703 * the argument setup happens in the gtk_*_class_init() functions.
705 gtk_type_class (class_type);
707 query_data.arg_list = NULL;
708 query_data.class_type = class_type;
709 g_hash_table_foreach (arg_info_ht, gtk_query_arg_foreach, &query_data);
711 if (query_data.arg_list)
713 register GList *list;
716 list = query_data.arg_list;
723 g_assert (len == ((GtkObjectClass*) gtk_type_class (class_type))->n_args); /* paranoid */
725 args = g_new0 (GtkArg, len);
728 *arg_flags = g_new (guint32, len);
737 g_assert (info->seq_id > 0 && info->seq_id <= len); /* paranoid */
739 args[info->seq_id - 1].type = info->type;
740 args[info->seq_id - 1].name = info->name;
742 (*arg_flags)[info->seq_id - 1] = info->arg_flags;
746 g_list_free (query_data.arg_list);
754 /*****************************************
760 *****************************************/
763 gtk_object_set (GtkObject *object,
771 g_return_if_fail (object != NULL);
773 va_start (args1, object);
774 va_start (args2, object);
776 args = gtk_object_collect_args (&nargs, args1, args2);
777 gtk_object_setv (object, nargs, args);
784 /*****************************************
790 *****************************************/
793 gtk_object_setv (GtkObject *object,
799 g_return_if_fail (object != NULL);
800 g_return_if_fail (GTK_OBJECT (object));
805 for (i = 0; i < nargs; i++)
812 lookup_name = g_strdup (args[i].name);
813 d = strchr (lookup_name, ':');
814 if (d && d[1] == ':')
816 d = strchr (d + 2, ':');
820 info = g_hash_table_lookup (arg_info_ht, lookup_name);
829 g_warning ("invalid arg name: \"%s\"\n", lookup_name);
832 else if (info->type != args[i].type)
834 g_warning ("invalid arg type for: \"%s\"\n", lookup_name);
837 else if (!gtk_type_is_a (object->klass->type, info->class_type))
839 g_warning ("invalid arg for %s: \"%s\"\n", gtk_type_name (object->klass->type), lookup_name);
842 else if (! (info->arg_flags & GTK_ARG_WRITABLE))
844 g_warning ("arg is not supplied for write-access: \"%s\"\n", lookup_name);
848 g_free (lookup_name);
853 gtk_type_set_arg (object, info->class_type, &args[i], info->arg_id);
857 /*****************************************
858 * gtk_object_add_arg_type:
863 *****************************************/
866 gtk_object_add_arg_type (const char *arg_name,
872 gchar class_part[1024];
876 g_return_if_fail (arg_name != NULL);
877 g_return_if_fail (arg_type > GTK_TYPE_NONE);
878 g_return_if_fail (arg_id > 0);
879 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) != 0);
881 arg_part = strchr (arg_name, ':');
882 if (!arg_part || (arg_part[0] != ':') || (arg_part[1] != ':'))
884 g_warning ("invalid arg name: \"%s\"\n", arg_name);
888 strncpy (class_part, arg_name, (glong) (arg_part - arg_name));
889 class_part[(glong) (arg_part - arg_name)] = '\0';
891 class_type = gtk_type_from_name (class_part);
894 g_warning ("invalid class name in arg: \"%s\"\n", arg_name);
898 info = g_new (GtkArgInfo, 1);
899 info->name = g_strdup (arg_name);
900 info->type = arg_type;
901 info->class_type = class_type;
902 info->arg_flags = arg_flags & (GTK_ARG_READABLE | GTK_ARG_WRITABLE);
903 info->arg_id = arg_id;
904 info->seq_id = ++((GtkObjectClass*) gtk_type_class (class_type))->n_args;
907 arg_info_ht = g_hash_table_new (g_str_hash, g_str_equal);
909 g_hash_table_insert (arg_info_ht, info->name, info);
912 /*****************************************
913 * gtk_object_get_arg_type:
918 *****************************************/
921 gtk_object_get_arg_type (const gchar *arg_name)
928 return GTK_TYPE_INVALID;
930 t = strchr (arg_name, ':');
931 if (!t || (t[0] != ':') || (t[1] != ':'))
933 g_warning ("invalid arg name: \"%s\"\n", arg_name);
934 return GTK_TYPE_INVALID;
937 t = strchr (t + 2, ':');
940 strncpy (buffer, arg_name, (long) (t - arg_name));
941 buffer[(long) (t - arg_name)] = '\0';
945 info = g_hash_table_lookup (arg_info_ht, (gpointer) arg_name);
949 return GTK_TYPE_INVALID;
952 /*****************************************
953 * GtkObject object_data mechanism
955 *****************************************/
958 gtk_object_set_data_by_id (GtkObject *object,
962 g_return_if_fail (data_id > 0);
964 gtk_object_set_data_by_id_full (object, data_id, data, NULL);
968 gtk_object_set_data (GtkObject *object,
972 g_return_if_fail (key != NULL);
974 gtk_object_set_data_by_id_full (object, gtk_object_data_force_id (key), data, NULL);
978 gtk_object_set_data_by_id_full (GtkObject *object,
981 GtkDestroyNotify destroy)
983 GtkObjectData *odata;
986 g_return_if_fail (object != NULL);
987 g_return_if_fail (GTK_IS_OBJECT (object));
988 g_return_if_fail (data_id > 0);
993 odata = object->object_data;
997 if (odata->id == data_id)
1000 prev->next = odata->next;
1001 if (odata == object->object_data)
1002 object->object_data = odata->next;
1004 gtk_object_data_destroy (odata);
1009 odata = odata->next;
1014 odata = object->object_data;
1017 if (odata->id == data_id)
1020 odata->destroy (odata->data);
1023 odata->destroy = destroy;
1027 odata = odata->next;
1030 if (!object_data_mem_chunk)
1031 object_data_mem_chunk = g_mem_chunk_new ("object data mem chunk",
1032 sizeof (GtkObjectData),
1033 1024, G_ALLOC_AND_FREE);
1035 odata = g_chunk_new (GtkObjectData, object_data_mem_chunk);
1036 odata->id = data_id;
1038 odata->destroy = destroy;
1039 odata->next = object->object_data;
1041 object->object_data = odata;
1046 gtk_object_set_data_full (GtkObject *object,
1049 GtkDestroyNotify destroy)
1051 g_return_if_fail (key != NULL);
1053 gtk_object_set_data_by_id_full (object, gtk_object_data_force_id (key), data, destroy);
1057 gtk_object_data_force_id (const gchar *key)
1061 g_return_val_if_fail (key != NULL, 0);
1063 if (!object_data_ht)
1064 object_data_ht = g_hash_table_new (g_str_hash, g_str_equal);
1066 id = g_hash_table_lookup (object_data_ht, (gpointer) key);
1069 id = gtk_object_data_id_alloc ();
1070 g_hash_table_insert (object_data_ht, g_strdup (key), id);
1077 gtk_object_get_data_by_id (GtkObject *object,
1080 GtkObjectData *odata;
1082 g_return_val_if_fail (object != NULL, NULL);
1083 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1087 odata = object->object_data;
1090 if (odata->id == data_id)
1092 odata = odata->next;
1100 gtk_object_get_data (GtkObject *object,
1105 g_return_val_if_fail (key != NULL, NULL);
1107 id = gtk_object_data_try_key (key);
1109 return gtk_object_get_data_by_id (object, id);
1115 gtk_object_data_try_key (const gchar *key)
1117 g_return_val_if_fail (key != NULL, 0);
1123 id = g_hash_table_lookup (object_data_ht, (gpointer) key);
1132 gtk_object_remove_data_by_id (GtkObject *object,
1136 gtk_object_set_data_by_id_full (object, data_id, NULL, NULL);
1140 gtk_object_remove_data (GtkObject *object,
1145 g_return_if_fail (key != NULL);
1147 id = gtk_object_data_try_key (key);
1149 gtk_object_set_data_by_id_full (object, id, NULL, NULL);
1153 gtk_object_data_destroy (GtkObjectData *odata)
1155 g_return_if_fail (odata != NULL);
1158 odata->destroy (odata->data);
1160 g_mem_chunk_free (object_data_mem_chunk, odata);
1164 gtk_object_data_id_alloc ()
1166 static guint next_id = 1;
1169 if (!object_data_id_list ||
1170 (object_data_id_index == OBJECT_DATA_ID_CHUNK))
1172 ids = g_new (guint, OBJECT_DATA_ID_CHUNK);
1173 object_data_id_index = 0;
1174 object_data_id_list = g_slist_prepend (object_data_id_list, ids);
1178 ids = object_data_id_list->data;
1181 ids[object_data_id_index] = next_id++;
1182 return &ids[object_data_id_index++];
1185 /*****************************************
1186 * gtk_object_set_user_data:
1191 *****************************************/
1194 gtk_object_set_user_data (GtkObject *object,
1197 if (!user_data_key_id)
1198 user_data_key_id = gtk_object_data_force_id (user_data_key);
1200 gtk_object_set_data_by_id_full (object, user_data_key_id, data, NULL);
1203 /*****************************************
1204 * gtk_object_get_user_data:
1209 *****************************************/
1212 gtk_object_get_user_data (GtkObject *object)
1214 if (user_data_key_id)
1215 return gtk_object_get_data_by_id (object, user_data_key_id);
1220 /*****************************************
1221 * gtk_object_check_cast:
1226 *****************************************/
1229 gtk_object_descriptive_type_name (GtkType type)
1233 name = gtk_type_name (type);
1241 gtk_object_check_cast (GtkObject *obj,
1246 g_warning ("invalid cast from (NULL) pointer to `%s'",
1247 gtk_object_descriptive_type_name (cast_type));
1252 g_warning ("invalid unclassed pointer in cast to `%s'",
1253 gtk_object_descriptive_type_name (cast_type));
1256 if (obj->klass->type < GTK_TYPE_OBJECT)
1258 g_warning ("invalid class type `%s' in cast to `%s'",
1259 gtk_object_descriptive_type_name (obj->klass->type),
1260 gtk_object_descriptive_type_name (cast_type));
1263 if (!gtk_type_is_a (obj->klass->type, cast_type))
1265 g_warning ("invalid cast from `%s' to `%s'",
1266 gtk_object_descriptive_type_name (obj->klass->type),
1267 gtk_object_descriptive_type_name (cast_type));
1274 /*****************************************
1275 * gtk_object_check_class_cast:
1280 *****************************************/
1283 gtk_object_check_class_cast (GtkObjectClass *klass,
1288 g_warning ("invalid class cast from (NULL) pointer to `%s'",
1289 gtk_object_descriptive_type_name (cast_type));
1292 if (klass->type < GTK_TYPE_OBJECT)
1294 g_warning ("invalid class type `%s' in class cast to `%s'",
1295 gtk_object_descriptive_type_name (klass->type),
1296 gtk_object_descriptive_type_name (cast_type));
1299 if (!gtk_type_is_a (klass->type, cast_type))
1301 g_warning ("invalid class cast from `%s' to `%s'",
1302 gtk_object_descriptive_type_name (klass->type),
1303 gtk_object_descriptive_type_name (cast_type));
1310 /*****************************************
1311 * gtk_object_collect_args:
1316 *****************************************/
1319 gtk_object_collect_args (guint *nargs,
1334 name = va_arg (args1, char *);
1341 type = gtk_object_get_arg_type (name);
1343 switch (GTK_FUNDAMENTAL_TYPE (type))
1345 case GTK_TYPE_INVALID:
1346 g_warning ("invalid arg name: \"%s\" %x\n", name, type);
1347 (void) va_arg (args1, long);
1356 case GTK_TYPE_FLAGS:
1357 (void) va_arg (args1, gint);
1360 case GTK_TYPE_ULONG:
1361 (void) va_arg (args1, glong);
1363 case GTK_TYPE_FLOAT:
1364 (void) va_arg (args1, gfloat);
1366 case GTK_TYPE_DOUBLE:
1367 (void) va_arg (args1, gdouble);
1369 case GTK_TYPE_STRING:
1370 (void) va_arg (args1, gchar*);
1372 case GTK_TYPE_POINTER:
1373 case GTK_TYPE_BOXED:
1374 (void) va_arg (args1, gpointer);
1376 case GTK_TYPE_SIGNAL:
1377 (void) va_arg (args1, GtkFunction);
1378 (void) va_arg (args1, gpointer);
1380 case GTK_TYPE_FOREIGN:
1381 (void) va_arg (args1, gpointer);
1382 (void) va_arg (args1, GtkDestroyNotify);
1384 case GTK_TYPE_CALLBACK:
1385 (void) va_arg (args1, GtkCallbackMarshal);
1386 (void) va_arg (args1, gpointer);
1387 (void) va_arg (args1, GtkDestroyNotify);
1389 case GTK_TYPE_C_CALLBACK:
1390 (void) va_arg (args1, GtkFunction);
1391 (void) va_arg (args1, gpointer);
1394 (void) va_arg (args1, gint);
1395 (void) va_arg (args1, GtkArg*);
1397 case GTK_TYPE_OBJECT:
1398 (void) va_arg (args1, GtkObject*);
1401 g_error ("unsupported type %s in args", gtk_type_name (type));
1413 args = g_new0 (GtkArg, n);
1415 for (i = 0; i < n; i++)
1417 args[i].name = va_arg (args2, char *);
1418 args[i].type = gtk_object_get_arg_type (args[i].name);
1420 switch (GTK_FUNDAMENTAL_TYPE (args[i].type))
1422 case GTK_TYPE_INVALID:
1423 (void) va_arg (args2, long);
1429 GTK_VALUE_CHAR(args[i]) = va_arg (args2, gint);
1432 GTK_VALUE_BOOL(args[i]) = va_arg (args2, gint);
1435 GTK_VALUE_INT(args[i]) = va_arg (args2, gint);
1438 GTK_VALUE_UINT(args[i]) = va_arg (args2, guint);
1441 GTK_VALUE_ENUM(args[i]) = va_arg (args2, gint);
1443 case GTK_TYPE_FLAGS:
1444 GTK_VALUE_FLAGS(args[i]) = va_arg (args2, gint);
1447 GTK_VALUE_LONG(args[i]) = va_arg (args2, glong);
1449 case GTK_TYPE_ULONG:
1450 GTK_VALUE_ULONG(args[i]) = va_arg (args2, gulong);
1452 case GTK_TYPE_FLOAT:
1453 GTK_VALUE_FLOAT(args[i]) = va_arg (args2, gfloat);
1455 case GTK_TYPE_DOUBLE:
1456 GTK_VALUE_DOUBLE(args[i]) = va_arg (args2, gdouble);
1458 case GTK_TYPE_STRING:
1459 GTK_VALUE_STRING(args[i]) = va_arg (args2, gchar*);
1461 case GTK_TYPE_POINTER:
1462 GTK_VALUE_POINTER(args[i]) = va_arg (args2, gpointer);
1464 case GTK_TYPE_BOXED:
1465 GTK_VALUE_BOXED(args[i]) = va_arg (args2, gpointer);
1467 case GTK_TYPE_SIGNAL:
1468 GTK_VALUE_SIGNAL(args[i]).f = va_arg (args2, GtkFunction);
1469 GTK_VALUE_SIGNAL(args[i]).d = va_arg (args2, gpointer);
1471 case GTK_TYPE_FOREIGN:
1472 GTK_VALUE_FOREIGN(args[i]).data = va_arg (args2, gpointer);
1473 GTK_VALUE_FOREIGN(args[i]).notify =
1474 va_arg (args2, GtkDestroyNotify);
1476 case GTK_TYPE_CALLBACK:
1477 GTK_VALUE_CALLBACK(args[i]).marshal =
1478 va_arg (args2, GtkCallbackMarshal);
1479 GTK_VALUE_CALLBACK(args[i]).data = va_arg (args2, gpointer);
1480 GTK_VALUE_CALLBACK(args[i]).notify =
1481 va_arg (args2, GtkDestroyNotify);
1483 case GTK_TYPE_C_CALLBACK:
1484 GTK_VALUE_C_CALLBACK(args[i]).func = va_arg (args2, GtkFunction);
1485 GTK_VALUE_C_CALLBACK(args[i]).func_data =
1486 va_arg (args2, gpointer);
1489 GTK_VALUE_ARGS(args[i]).n_args = va_arg (args2, gint);
1490 GTK_VALUE_ARGS(args[i]).args = va_arg (args2, GtkArg*);
1492 case GTK_TYPE_OBJECT:
1493 GTK_VALUE_OBJECT(args[i]) = va_arg (args2, GtkObject*);
1494 g_assert (GTK_VALUE_OBJECT(args[i]) == NULL ||
1495 GTK_CHECK_TYPE (GTK_VALUE_OBJECT(args[i]),
1499 g_error ("unsupported type %s in args",
1500 gtk_type_name (args[i].type));
1511 #undef gtk_object_ref
1512 #undef gtk_object_unref
1515 gtk_object_ref (GtkObject *object)
1517 g_return_if_fail (object != NULL);
1518 g_return_if_fail (GTK_IS_OBJECT (object));
1520 object->ref_count += 1;
1524 gtk_object_unref (GtkObject *object)
1526 g_return_if_fail (object != NULL);
1527 g_return_if_fail (GTK_IS_OBJECT (object));
1529 if (object->ref_count == 1)
1530 gtk_object_destroy (object);
1532 if (object->ref_count > 0)
1533 object->ref_count -= 1;
1535 if (object->ref_count == 0)
1537 #ifdef G_ENABLE_DEBUG
1538 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1540 g_assert (g_hash_table_lookup (living_objs_ht, object) == object);
1541 g_hash_table_remove (living_objs_ht, object);
1544 #endif /* G_ENABLE_DEBUG */
1545 object->klass->finalize (object);
1550 #ifdef G_ENABLE_DEBUG
1551 static GtkObject *gtk_trace_object = NULL;
1553 gtk_trace_referencing (gpointer *o,
1560 GtkObject *object = (GtkObject*) o;
1562 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1564 g_return_if_fail (object != NULL);
1565 g_return_if_fail (GTK_IS_OBJECT (object));
1567 exists = g_hash_table_lookup (living_objs_ht, object) != NULL;
1570 (object == gtk_trace_object ||
1571 gtk_trace_object == (void*)42))
1572 printf ("trace: object_%s: (%s:%p)->ref_count=%d%s (%s_f%02d:%d)\n",
1573 do_ref ? "ref" : "unref",
1574 gtk_type_name (GTK_OBJECT_TYPE (object)),
1577 do_ref ? " + 1" : " - 1 ",
1583 printf ("trace: object_%s(%p): no such object! (%s_f%02d:%d)\n",
1584 do_ref ? "ref" : "unref",
1592 gtk_object_ref (object);
1594 gtk_object_unref (object);
1596 #endif /* G_ENABLE_DEBUG */