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;
61 static void gtk_object_class_init (GtkObjectClass *klass);
62 static void gtk_object_init (GtkObject *object);
63 static void gtk_object_set_arg (GtkObject *object,
66 static void gtk_object_get_arg (GtkObject *object,
69 static void gtk_object_real_destroy (GtkObject *object);
70 static void gtk_object_finalize (GtkObject *object);
71 static void gtk_object_notify_weaks (GtkObject *object);
72 static void gtk_object_data_init (void);
73 static GtkObjectData* gtk_object_data_new (void);
74 static void gtk_object_data_destroy (GtkObjectData *odata);
75 static guint* gtk_object_data_id_alloc (void);
77 GtkArg* gtk_object_collect_args (guint *nargs,
81 static gint object_signals[LAST_SIGNAL] = { 0 };
83 static gint object_data_init = TRUE;
84 static GHashTable *object_data_ht = NULL;
85 static GMemChunk *object_data_mem_chunk = NULL;
86 static GtkObjectData *object_data_free_list = NULL;
87 static GSList *object_data_id_list = NULL;
88 static gint object_data_id_index = 0;
90 static GHashTable *arg_info_ht = NULL;
92 static const char *user_data_key = "user_data";
95 static gint obj_count = 0;
96 static GSList *living_objs = NULL;
100 gtk_object_debug (void)
106 printf ("living objects (%d):\n", g_slist_length (living_objs));
107 for (node = living_objs; node; node = node->next)
111 obj = (GtkObject*) node->data;
112 printf ("%p: %s %d %s\n",
113 obj, gtk_type_name (GTK_OBJECT_TYPE (obj)),
115 GTK_OBJECT_FLOATING (obj)? "floating" : "");
118 printf ("living objects count = %d\n", obj_count);
121 /*****************************************
122 * gtk_object_init_type:
127 *****************************************/
130 gtk_object_init_type ()
132 GtkType object_type = 0;
133 GtkTypeInfo object_info =
137 sizeof (GtkObjectClass),
138 (GtkClassInitFunc) gtk_object_class_init,
139 (GtkObjectInitFunc) gtk_object_init,
144 object_type = gtk_type_unique (0, &object_info);
145 g_assert (object_type == GTK_TYPE_OBJECT);
147 #ifdef GTK_TRACE_OBJECTS
148 ATEXIT (gtk_object_debug);
149 #endif /* GTK_TRACE_OBJECTS */
153 gtk_object_get_type ()
155 return GTK_TYPE_OBJECT;
158 /*****************************************
159 * gtk_object_class_init:
164 *****************************************/
167 gtk_object_class_init (GtkObjectClass *class)
169 class->signals = NULL;
173 gtk_object_add_arg_type ("GtkObject::user_data", GTK_TYPE_POINTER, ARG_USER_DATA);
174 gtk_object_add_arg_type ("GtkObject::signal", GTK_TYPE_SIGNAL, ARG_SIGNAL);
175 gtk_object_add_arg_type ("GtkObject::object_signal", GTK_TYPE_SIGNAL, ARG_OBJECT_SIGNAL);
177 object_signals[DESTROY] =
178 gtk_signal_new ("destroy",
181 GTK_SIGNAL_OFFSET (GtkObjectClass, destroy),
182 gtk_signal_default_marshaller,
185 gtk_object_class_add_signals (class, object_signals, LAST_SIGNAL);
187 class->destroy = gtk_object_real_destroy;
188 class->finalize = gtk_object_finalize;
191 /*****************************************
192 * gtk_object_real_destroy:
197 *****************************************/
200 gtk_object_real_destroy (GtkObject *object)
202 g_return_if_fail (object != NULL);
203 g_return_if_fail (GTK_IS_OBJECT (object));
205 gtk_signal_handlers_destroy (object);
208 /*****************************************
214 *****************************************/
217 gtk_object_init (GtkObject *object)
219 GTK_OBJECT_FLAGS (object) = GTK_FLOATING;
221 object->ref_count = 1;
222 object->object_data = NULL;
224 #ifdef GTK_TRACE_OBJECTS
226 living_objs = g_slist_prepend (living_objs, object);
227 #endif /* GTK_TRACE_OBJECTS */
230 /*****************************************
231 * gtk_object_set_arg:
236 *****************************************/
239 gtk_object_set_arg (GtkObject *object,
246 gtk_object_set_user_data (object, GTK_VALUE_POINTER (*arg));
249 if ((arg->name[9 + 2 + 6] != ':') || (arg->name[9 + 2 + 7] != ':'))
251 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
254 gtk_signal_connect (object, arg->name + 9 + 2 + 6 + 2,
255 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
256 GTK_VALUE_SIGNAL (*arg).d);
258 case ARG_OBJECT_SIGNAL:
259 if ((arg->name[9 + 2 + 13] != ':') || (arg->name[9 + 2 + 14] != ':'))
261 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
264 gtk_signal_connect_object (object, arg->name + 9 + 2 + 13 + 2,
265 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
266 (GtkObject*) GTK_VALUE_SIGNAL (*arg).d);
269 arg->type = GTK_TYPE_INVALID;
274 /*****************************************
275 * gtk_object_get_arg:
280 *****************************************/
283 gtk_object_get_arg (GtkObject *object,
290 GTK_VALUE_POINTER (*arg) = gtk_object_get_user_data (object);
293 case ARG_OBJECT_SIGNAL:
295 arg->type = GTK_TYPE_INVALID;
300 /*****************************************
301 * gtk_object_class_add_signals:
306 *****************************************/
309 gtk_object_class_add_signals (GtkObjectClass *class,
316 g_return_if_fail (class != NULL);
318 new_signals = g_new (gint, class->nsignals + nsignals);
319 for (i = 0; i < class->nsignals; i++)
320 new_signals[i] = class->signals[i];
321 for (i = 0; i < nsignals; i++)
322 new_signals[class->nsignals + i] = signals[i];
324 g_free (class->signals);
325 class->signals = new_signals;
326 class->nsignals += nsignals;
329 /*****************************************
330 * gtk_object_class_add_user_signal:
335 *****************************************/
338 gtk_object_class_add_user_signal (GtkObjectClass *class,
340 GtkSignalMarshaller marshaller,
350 g_return_val_if_fail (class != NULL, 0);
354 params = g_new (GtkType, nparams);
356 va_start (args, nparams);
358 for (i = 0; i < nparams; i++)
359 params[i] = va_arg (args, GtkType);
366 signal_id = gtk_signal_newv (name,
378 gtk_object_class_add_signals (class, &signal_id, 1);
383 /*****************************************
384 * gtk_object_finalize:
389 *****************************************/
392 gtk_object_finalize (GtkObject *object)
394 GtkObjectData *odata;
396 gtk_object_notify_weaks (object);
398 if (object->object_data)
400 odata = object->object_data;
404 odata->next = object_data_free_list;
405 object_data_free_list = object->object_data;
406 object->object_data = NULL;
412 /*****************************************
418 *****************************************/
421 gtk_object_sink (GtkObject *object)
423 if (GTK_OBJECT_FLOATING (object))
425 GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING);
426 gtk_object_unref (object);
430 /*****************************************
431 * gtk_object_destroy:
436 *****************************************/
439 gtk_object_destroy (GtkObject *object)
441 g_return_if_fail (object != NULL);
442 g_return_if_fail (GTK_IS_OBJECT (object));
444 if (!GTK_OBJECT_DESTROYED (object))
446 GTK_OBJECT_SET_FLAGS (object, GTK_DESTROYED);
447 gtk_signal_emit (object, object_signals[DESTROY]);
451 /*****************************************
454 * Weak refs are very similar to the old "destroy" signal. They allow
455 * one to register a callback that is called when the weakly
456 * referenced object is destroyed.
458 * They are not implemented as a signal because they really are
459 * special and need to be used with great care. Unlike signals, who
460 * should be able to execute any code whatsoever.
462 * A weakref callback is not allowed to retain a reference to the
463 * object. In fact, the object is no longer there at all when it is
466 * A weakref callback is called atmost once.
468 *****************************************/
470 typedef struct _GtkWeakRef GtkWeakRef;
475 GtkDestroyNotify notify;
479 static const gchar *weakrefs_key = "gtk-weakrefs";
482 gtk_object_weakref (GtkObject *object,
483 GtkDestroyNotify notify,
488 g_return_if_fail (object != NULL);
489 g_return_if_fail (notify != NULL);
490 g_return_if_fail (GTK_IS_OBJECT (object));
492 weak = g_new (GtkWeakRef, 1);
493 weak->next = gtk_object_get_data (object, weakrefs_key);
494 weak->notify = notify;
496 gtk_object_set_data (object, weakrefs_key, weak);
500 gtk_object_weakunref (GtkObject *object,
501 GtkDestroyNotify notify,
504 GtkWeakRef *weaks, *w, **wp;
506 g_return_if_fail (object != NULL);
507 g_return_if_fail (GTK_IS_OBJECT (object));
509 weaks = gtk_object_get_data (object, weakrefs_key);
510 for (wp = &weaks; *wp; wp = &(*wp)->next)
513 if (w->notify == notify && w->data == data)
516 gtk_object_set_data (object, weakrefs_key, w->next);
526 gtk_object_notify_weaks (GtkObject *object)
530 w1 = gtk_object_get_data (object, weakrefs_key);
531 gtk_object_set_data (object, weakrefs_key, NULL);
535 w1->notify (w1->data);
542 /*****************************************
548 *****************************************/
551 gtk_object_new (guint type,
560 obj = gtk_type_new (type);
562 va_start (args1, type);
563 va_start (args2, type);
565 args = gtk_object_collect_args (&nargs, args1, args2);
566 gtk_object_setv (obj, nargs, args);
575 /*****************************************
581 *****************************************/
584 gtk_object_newv (guint type,
590 obj = gtk_type_new (type);
591 gtk_object_setv (obj, nargs, args);
596 /*****************************************
602 *****************************************/
605 gtk_object_getv (GtkObject *object,
611 g_return_if_fail (object != NULL);
616 for (i = 0; i < nargs; i++)
623 /* hm, the name cutting shouldn't be needed on gets, but what the heck...
625 lookup_name = g_strdup (args[i].name);
626 d = strchr (lookup_name, ':');
627 if (d && d[1] == ':')
629 d = strchr (d + 2, ':');
633 info = g_hash_table_lookup (arg_info_ht, lookup_name);
640 g_warning ("invalid arg name: \"%s\"\n", lookup_name);
641 args[i].type = GTK_TYPE_INVALID;
642 g_free (lookup_name);
645 else if (!gtk_type_is_a (object->klass->type, info->class_type))
647 g_warning ("invalid arg for %s: \"%s\"\n", gtk_type_name (object->klass->type), lookup_name);
648 args[i].type = GTK_TYPE_INVALID;
649 g_free (lookup_name);
653 g_free (lookup_name);
655 args[i].type = info->type;
656 gtk_type_get_arg (object, info->class_type, &args[i], info->arg_id);
660 /*****************************************
661 * gtk_object_query_args:
666 *****************************************/
668 struct _GtkQueryArgData
673 typedef struct _GtkQueryArgData GtkQueryArgData;
676 gtk_query_arg_foreach (gpointer key,
680 register GtkArgInfo *info;
681 register GtkQueryArgData *data;
686 if (info->class_type == data->class_type)
687 data->arg_list = g_list_prepend (data->arg_list, info);
691 gtk_object_query_args (GtkType class_type,
695 GtkQueryArgData query_data;
697 g_return_val_if_fail (nargs != NULL, NULL);
699 g_return_val_if_fail (gtk_type_is_a (class_type, gtk_object_get_type ()), NULL);
704 query_data.arg_list = NULL;
705 query_data.class_type = class_type;
706 g_hash_table_foreach (arg_info_ht, gtk_query_arg_foreach, &query_data);
708 if (query_data.arg_list)
710 register GList *list;
713 list = query_data.arg_list;
720 g_assert (len == ((GtkObjectClass*) gtk_type_class (class_type))->n_args); /* paranoid */
722 args = g_new0 (GtkArg, len);
732 g_assert (info->seq_id > 0 && info->seq_id <= len); /* paranoid */
734 args[info->seq_id - 1].type = info->type;
735 args[info->seq_id - 1].name = info->name;
739 g_list_free (query_data.arg_list);
747 /*****************************************
753 *****************************************/
756 gtk_object_set (GtkObject *object,
764 g_return_if_fail (object != NULL);
766 va_start (args1, object);
767 va_start (args2, object);
769 args = gtk_object_collect_args (&nargs, args1, args2);
770 gtk_object_setv (object, nargs, args);
777 /*****************************************
783 *****************************************/
786 gtk_object_setv (GtkObject *object,
792 g_return_if_fail (object != NULL);
797 for (i = 0; i < nargs; i++)
804 lookup_name = g_strdup (args[i].name);
805 d = strchr (lookup_name, ':');
806 if (d && d[1] == ':')
808 d = strchr (d + 2, ':');
812 info = g_hash_table_lookup (arg_info_ht, lookup_name);
821 g_warning ("invalid arg name: \"%s\"\n", lookup_name);
824 else if (info->type != args[i].type)
826 g_warning ("invalid arg type for: \"%s\"\n", lookup_name);
829 else if (!gtk_type_is_a (object->klass->type, info->class_type))
831 g_warning ("invalid arg for %s: \"%s\"\n", gtk_type_name (object->klass->type), lookup_name);
835 g_free (lookup_name);
840 gtk_type_set_arg (object, info->class_type, &args[i], info->arg_id);
844 /*****************************************
845 * gtk_object_add_arg_type:
850 *****************************************/
853 gtk_object_add_arg_type (const char *arg_name,
858 gchar class_part[1024];
862 g_return_if_fail (arg_id > 0);
864 arg_part = strchr (arg_name, ':');
865 if (!arg_part || (arg_part[0] != ':') || (arg_part[1] != ':'))
867 g_warning ("invalid arg name: \"%s\"\n", arg_name);
871 strncpy (class_part, arg_name, (glong) (arg_part - arg_name));
872 class_part[(glong) (arg_part - arg_name)] = '\0';
874 class_type = gtk_type_from_name (class_part);
877 g_warning ("invalid class name in arg: \"%s\"\n", arg_name);
881 info = g_new (GtkArgInfo, 1);
882 info->name = g_strdup (arg_name);
883 info->type = arg_type;
884 info->class_type = class_type;
885 info->arg_id = arg_id;
886 info->seq_id = ++((GtkObjectClass*) gtk_type_class (class_type))->n_args;
889 arg_info_ht = g_hash_table_new (g_str_hash, g_str_equal);
891 g_hash_table_insert (arg_info_ht, info->name, info);
894 /*****************************************
895 * gtk_object_get_arg_type:
900 *****************************************/
903 gtk_object_get_arg_type (const char *arg_name)
910 return GTK_TYPE_INVALID;
912 t = strchr (arg_name, ':');
913 if (!t || (t[0] != ':') || (t[1] != ':'))
915 g_warning ("invalid arg name: \"%s\"\n", arg_name);
916 return GTK_TYPE_INVALID;
919 t = strchr (t + 2, ':');
922 strncpy (buffer, arg_name, (long) (t - arg_name));
923 buffer[(long) (t - arg_name)] = '\0';
927 info = g_hash_table_lookup (arg_info_ht, (gpointer) arg_name);
931 return GTK_TYPE_INVALID;
934 /*****************************************
935 * gtk_object_set_data:
940 *****************************************/
943 gtk_object_set_data (GtkObject *object,
947 GtkObjectData *odata;
951 g_return_if_fail (object != NULL);
952 g_return_if_fail (GTK_IS_OBJECT (object));
953 g_return_if_fail (key != NULL);
955 if (object_data_init)
956 gtk_object_data_init ();
958 id = g_hash_table_lookup (object_data_ht, (gpointer) key);
965 odata = object->object_data;
969 if (odata->id == *id)
972 prev->next = odata->next;
973 if (odata == object->object_data)
974 object->object_data = odata->next;
976 gtk_object_data_destroy (odata);
989 id = gtk_object_data_id_alloc ();
990 g_hash_table_insert (object_data_ht, (gpointer) key, id);
993 odata = object->object_data;
996 if (odata->id == *id)
1002 odata = odata->next;
1005 odata = gtk_object_data_new ();
1009 odata->next = object->object_data;
1010 object->object_data = odata;
1014 /*****************************************
1015 * gtk_object_get_data:
1020 *****************************************/
1023 gtk_object_get_data (GtkObject *object,
1026 GtkObjectData *odata;
1029 g_return_val_if_fail (object != NULL, NULL);
1030 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1031 g_return_val_if_fail (key != NULL, NULL);
1033 if (object_data_init)
1034 gtk_object_data_init ();
1036 id = g_hash_table_lookup (object_data_ht, (gpointer) key);
1039 odata = object->object_data;
1042 if (odata->id == *id)
1044 odata = odata->next;
1051 /*****************************************
1052 * gtk_object_remove_data:
1057 *****************************************/
1060 gtk_object_remove_data (GtkObject *object,
1063 g_return_if_fail (object != NULL);
1064 g_return_if_fail (GTK_IS_OBJECT (object));
1065 g_return_if_fail (key != NULL);
1067 gtk_object_set_data (object, key, NULL);
1070 /*****************************************
1071 * gtk_object_set_user_data:
1076 *****************************************/
1079 gtk_object_set_user_data (GtkObject *object,
1082 g_return_if_fail (object != NULL);
1083 g_return_if_fail (GTK_IS_OBJECT (object));
1085 gtk_object_set_data (object, user_data_key, data);
1088 /*****************************************
1089 * gtk_object_get_user_data:
1094 *****************************************/
1097 gtk_object_get_user_data (GtkObject *object)
1099 g_return_val_if_fail (object != NULL, NULL);
1100 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1102 return gtk_object_get_data (object, user_data_key);
1105 /*****************************************
1106 * gtk_object_check_cast:
1111 *****************************************/
1114 gtk_object_check_cast (GtkObject *obj,
1117 if (obj && obj->klass && !gtk_type_is_a (obj->klass->type, cast_type))
1119 gchar *from_name = gtk_type_name (obj->klass->type);
1120 gchar *to_name = gtk_type_name (cast_type);
1122 g_warning ("invalid cast from \"%s\" to \"%s\"",
1123 from_name ? from_name : "(unknown)",
1124 to_name ? to_name : "(unknown)");
1130 /*****************************************
1131 * gtk_object_check_class_cast:
1136 *****************************************/
1139 gtk_object_check_class_cast (GtkObjectClass *klass,
1142 if (klass && !gtk_type_is_a (klass->type, cast_type))
1143 g_warning ("invalid cast from \"%sClass\" to \"%sClass\"",
1144 gtk_type_name (klass->type),
1145 gtk_type_name (cast_type));
1150 /*****************************************
1151 * gtk_object_data_init:
1156 *****************************************/
1159 gtk_object_data_init ()
1161 if (object_data_init)
1163 object_data_init = FALSE;
1165 object_data_ht = g_hash_table_new (g_str_hash, g_str_equal);
1169 /*****************************************
1170 * gtk_object_data_new:
1175 *****************************************/
1177 static GtkObjectData*
1178 gtk_object_data_new ()
1180 GtkObjectData *odata;
1182 if (!object_data_mem_chunk)
1183 object_data_mem_chunk = g_mem_chunk_new ("object data mem chunk",
1184 sizeof (GtkObjectData),
1185 1024, G_ALLOC_AND_FREE);
1187 odata = g_chunk_new (GtkObjectData, object_data_mem_chunk);
1196 /*****************************************
1197 * gtk_object_data_destroy:
1202 *****************************************/
1205 gtk_object_data_destroy (GtkObjectData *odata)
1207 g_return_if_fail (odata != NULL);
1209 g_mem_chunk_free (object_data_mem_chunk, odata);
1212 /*****************************************
1213 * gtk_object_data_id_alloc:
1218 *****************************************/
1221 gtk_object_data_id_alloc ()
1223 static guint next_id = 1;
1226 if (!object_data_id_list ||
1227 (object_data_id_index == OBJECT_DATA_ID_CHUNK))
1229 ids = g_new (guint, OBJECT_DATA_ID_CHUNK);
1230 object_data_id_index = 0;
1231 object_data_id_list = g_slist_prepend (object_data_id_list, ids);
1235 ids = object_data_id_list->data;
1238 ids[object_data_id_index] = next_id++;
1239 return &ids[object_data_id_index++];
1242 /*****************************************
1243 * gtk_object_collect_args:
1248 *****************************************/
1251 gtk_object_collect_args (guint *nargs,
1266 name = va_arg (args1, char *);
1273 type = gtk_object_get_arg_type (name);
1275 switch (GTK_FUNDAMENTAL_TYPE (type))
1277 case GTK_TYPE_INVALID:
1278 g_warning ("invalid arg name: \"%s\" %x\n", name, type);
1279 (void) va_arg (args1, long);
1288 case GTK_TYPE_FLAGS:
1289 (void) va_arg (args1, gint);
1292 case GTK_TYPE_ULONG:
1293 (void) va_arg (args1, glong);
1295 case GTK_TYPE_FLOAT:
1296 (void) va_arg (args1, gfloat);
1298 case GTK_TYPE_DOUBLE:
1299 (void) va_arg (args1, gdouble);
1301 case GTK_TYPE_STRING:
1302 (void) va_arg (args1, gchar*);
1304 case GTK_TYPE_POINTER:
1305 case GTK_TYPE_BOXED:
1306 (void) va_arg (args1, gpointer);
1308 case GTK_TYPE_SIGNAL:
1309 (void) va_arg (args1, GtkFunction);
1310 (void) va_arg (args1, gpointer);
1312 case GTK_TYPE_FOREIGN:
1313 (void) va_arg (args1, gpointer);
1314 (void) va_arg (args1, GtkDestroyNotify);
1316 case GTK_TYPE_CALLBACK:
1317 (void) va_arg (args1, GtkCallbackMarshal);
1318 (void) va_arg (args1, gpointer);
1319 (void) va_arg (args1, GtkDestroyNotify);
1321 case GTK_TYPE_C_CALLBACK:
1322 (void) va_arg (args1, GtkFunction);
1323 (void) va_arg (args1, gpointer);
1326 (void) va_arg (args1, gint);
1327 (void) va_arg (args1, GtkArg*);
1329 case GTK_TYPE_OBJECT:
1330 (void) va_arg (args1, GtkObject*);
1333 g_error ("unsupported type %s in args", gtk_type_name (type));
1345 args = g_new0 (GtkArg, n);
1347 for (i = 0; i < n; i++)
1349 args[i].name = va_arg (args2, char *);
1350 args[i].type = gtk_object_get_arg_type (args[i].name);
1352 switch (GTK_FUNDAMENTAL_TYPE (args[i].type))
1354 case GTK_TYPE_INVALID:
1355 (void) va_arg (args2, long);
1361 GTK_VALUE_CHAR(args[i]) = va_arg (args2, gint);
1364 GTK_VALUE_BOOL(args[i]) = va_arg (args2, gint);
1367 GTK_VALUE_INT(args[i]) = va_arg (args2, gint);
1370 GTK_VALUE_UINT(args[i]) = va_arg (args2, guint);
1373 GTK_VALUE_ENUM(args[i]) = va_arg (args2, gint);
1375 case GTK_TYPE_FLAGS:
1376 GTK_VALUE_FLAGS(args[i]) = va_arg (args2, gint);
1379 GTK_VALUE_LONG(args[i]) = va_arg (args2, glong);
1381 case GTK_TYPE_ULONG:
1382 GTK_VALUE_ULONG(args[i]) = va_arg (args2, gulong);
1384 case GTK_TYPE_FLOAT:
1385 GTK_VALUE_FLOAT(args[i]) = va_arg (args2, gfloat);
1387 case GTK_TYPE_DOUBLE:
1388 GTK_VALUE_DOUBLE(args[i]) = va_arg (args2, gdouble);
1390 case GTK_TYPE_STRING:
1391 GTK_VALUE_STRING(args[i]) = va_arg (args2, gchar*);
1393 case GTK_TYPE_POINTER:
1394 GTK_VALUE_POINTER(args[i]) = va_arg (args2, gpointer);
1396 case GTK_TYPE_BOXED:
1397 GTK_VALUE_BOXED(args[i]) = va_arg (args2, gpointer);
1399 case GTK_TYPE_SIGNAL:
1400 GTK_VALUE_SIGNAL(args[i]).f = va_arg (args2, GtkFunction);
1401 GTK_VALUE_SIGNAL(args[i]).d = va_arg (args2, gpointer);
1403 case GTK_TYPE_FOREIGN:
1404 GTK_VALUE_FOREIGN(args[i]).data = va_arg (args2, gpointer);
1405 GTK_VALUE_FOREIGN(args[i]).notify =
1406 va_arg (args2, GtkDestroyNotify);
1408 case GTK_TYPE_CALLBACK:
1409 GTK_VALUE_CALLBACK(args[i]).marshal =
1410 va_arg (args2, GtkCallbackMarshal);
1411 GTK_VALUE_CALLBACK(args[i]).data = va_arg (args2, gpointer);
1412 GTK_VALUE_CALLBACK(args[i]).notify =
1413 va_arg (args2, GtkDestroyNotify);
1415 case GTK_TYPE_C_CALLBACK:
1416 GTK_VALUE_C_CALLBACK(args[i]).func = va_arg (args2, GtkFunction);
1417 GTK_VALUE_C_CALLBACK(args[i]).func_data =
1418 va_arg (args2, gpointer);
1421 GTK_VALUE_ARGS(args[i]).n_args = va_arg (args2, gint);
1422 GTK_VALUE_ARGS(args[i]).args = va_arg (args2, GtkArg*);
1424 case GTK_TYPE_OBJECT:
1425 GTK_VALUE_OBJECT(args[i]) = va_arg (args2, GtkObject*);
1426 g_assert (GTK_VALUE_OBJECT(args[i]) == NULL ||
1427 GTK_CHECK_TYPE (GTK_VALUE_OBJECT(args[i]),
1431 g_error ("unsupported type %s in args",
1432 gtk_type_name (args[i].type));
1443 #undef gtk_object_ref
1444 #undef gtk_object_unref
1447 gtk_object_ref (GtkObject *object)
1449 g_return_if_fail (object != NULL);
1450 g_return_if_fail (GTK_IS_OBJECT (object));
1452 object->ref_count += 1;
1456 gtk_object_unref (GtkObject *object)
1458 g_return_if_fail (object != NULL);
1459 g_return_if_fail (GTK_IS_OBJECT (object));
1461 if (object->ref_count == 1)
1462 gtk_object_destroy (object);
1464 if (object->ref_count > 0)
1465 object->ref_count -= 1;
1467 if (object->ref_count == 0)
1469 g_assert (g_slist_find (living_objs, object));
1470 living_objs = g_slist_remove (living_objs, object);
1471 object->klass->finalize (object);
1476 static GtkObject *gtk_trace_object = NULL;
1479 gtk_trace_referencing (gpointer *o,
1486 GtkObject *object = (GtkObject*) o;
1488 g_return_if_fail (object != NULL);
1489 g_return_if_fail (GTK_IS_OBJECT (object));
1491 exists = (g_slist_find (living_objs, object) != NULL);
1494 (object == gtk_trace_object ||
1495 gtk_trace_object == (void*)42))
1496 printf ("trace: object_%s: (%s:%p)->ref_count=%d%s (%s_f%02d:%d)\n",
1497 do_ref ? "ref" : "unref",
1498 gtk_type_name (GTK_OBJECT_TYPE (object)),
1501 do_ref ? " + 1" : " - 1 ",
1507 printf ("trace: object_%s(%p): no such object! (%s_f%02d:%d)\n",
1508 do_ref ? "ref" : "unref",
1515 gtk_object_ref (object);
1517 gtk_object_unref (object);