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_real_destroy (GtkObject *object);
72 static void gtk_object_finalize (GtkObject *object);
73 static void gtk_object_notify_weaks (gpointer data);
74 static void gtk_object_data_init (void);
75 static GtkObjectData* gtk_object_data_new (void);
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 gint object_signals[LAST_SIGNAL] = { 0 };
85 static gint object_data_init = TRUE;
86 static GHashTable *object_data_ht = NULL;
87 static GMemChunk *object_data_mem_chunk = NULL;
88 static GSList *object_data_id_list = NULL;
89 static gint object_data_id_index = 0;
91 static GHashTable *arg_info_ht = NULL;
93 static const char *user_data_key = "user_data";
97 static gint 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 ("%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 ("living objects count = %d\n", obj_count);
120 gtk_object_pointer_hash (const gpointer v)
128 #endif /* G_ENABLE_DEBUG */
130 /*****************************************
131 * gtk_object_init_type:
136 *****************************************/
139 gtk_object_init_type ()
141 GtkType object_type = 0;
142 GtkTypeInfo object_info =
146 sizeof (GtkObjectClass),
147 (GtkClassInitFunc) gtk_object_class_init,
148 (GtkObjectInitFunc) gtk_object_init,
153 object_type = gtk_type_unique (0, &object_info);
154 g_assert (object_type == GTK_TYPE_OBJECT);
156 #ifdef G_ENABLE_DEBUG
157 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
158 ATEXIT (gtk_object_debug);
159 #endif /* G_ENABLE_DEBUG */
163 gtk_object_get_type ()
165 return GTK_TYPE_OBJECT;
168 /*****************************************
169 * gtk_object_class_init:
174 *****************************************/
177 gtk_object_class_init (GtkObjectClass *class)
179 class->signals = NULL;
183 gtk_object_add_arg_type ("GtkObject::user_data",
187 gtk_object_add_arg_type ("GtkObject::signal",
191 gtk_object_add_arg_type ("GtkObject::object_signal",
196 object_signals[DESTROY] =
197 gtk_signal_new ("destroy",
200 GTK_SIGNAL_OFFSET (GtkObjectClass, destroy),
201 gtk_signal_default_marshaller,
204 gtk_object_class_add_signals (class, object_signals, LAST_SIGNAL);
206 class->destroy = gtk_object_real_destroy;
207 class->finalize = gtk_object_finalize;
210 /*****************************************
211 * gtk_object_real_destroy:
216 *****************************************/
219 gtk_object_real_destroy (GtkObject *object)
221 g_return_if_fail (object != NULL);
222 g_return_if_fail (GTK_IS_OBJECT (object));
224 gtk_signal_handlers_destroy (object);
227 /*****************************************
233 *****************************************/
236 gtk_object_init (GtkObject *object)
238 GTK_OBJECT_FLAGS (object) = GTK_FLOATING;
240 object->ref_count = 1;
241 object->object_data = NULL;
243 #ifdef G_ENABLE_DEBUG
244 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
249 living_objs_ht = g_hash_table_new (gtk_object_pointer_hash, NULL);
251 g_hash_table_insert (living_objs_ht, object, object);
253 #endif /* G_ENABLE_DEBUG */
256 /*****************************************
257 * gtk_object_set_arg:
262 *****************************************/
265 gtk_object_set_arg (GtkObject *object,
272 gtk_object_set_user_data (object, GTK_VALUE_POINTER (*arg));
275 if ((arg->name[9 + 2 + 6] != ':') || (arg->name[9 + 2 + 7] != ':'))
277 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
278 arg->type = GTK_TYPE_INVALID;
281 gtk_signal_connect (object, arg->name + 9 + 2 + 6 + 2,
282 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
283 GTK_VALUE_SIGNAL (*arg).d);
285 case ARG_OBJECT_SIGNAL:
286 if ((arg->name[9 + 2 + 13] != ':') || (arg->name[9 + 2 + 14] != ':'))
288 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
289 arg->type = GTK_TYPE_INVALID;
292 gtk_signal_connect_object (object, arg->name + 9 + 2 + 13 + 2,
293 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
294 (GtkObject*) GTK_VALUE_SIGNAL (*arg).d);
297 arg->type = GTK_TYPE_INVALID;
302 /*****************************************
303 * gtk_object_get_arg:
308 *****************************************/
311 gtk_object_get_arg (GtkObject *object,
318 GTK_VALUE_POINTER (*arg) = gtk_object_get_user_data (object);
321 case ARG_OBJECT_SIGNAL:
323 arg->type = GTK_TYPE_INVALID;
328 /*****************************************
329 * gtk_object_class_add_signals:
334 *****************************************/
337 gtk_object_class_add_signals (GtkObjectClass *class,
344 g_return_if_fail (class != NULL);
346 new_signals = g_new (gint, class->nsignals + nsignals);
347 for (i = 0; i < class->nsignals; i++)
348 new_signals[i] = class->signals[i];
349 for (i = 0; i < nsignals; i++)
350 new_signals[class->nsignals + i] = signals[i];
352 g_free (class->signals);
353 class->signals = new_signals;
354 class->nsignals += nsignals;
357 /*****************************************
358 * gtk_object_class_add_user_signal:
363 *****************************************/
366 gtk_object_class_add_user_signal (GtkObjectClass *class,
368 GtkSignalMarshaller marshaller,
378 g_return_val_if_fail (class != NULL, 0);
382 params = g_new (GtkType, nparams);
384 va_start (args, nparams);
386 for (i = 0; i < nparams; i++)
387 params[i] = va_arg (args, GtkType);
394 signal_id = gtk_signal_newv (name,
406 gtk_object_class_add_signals (class, &signal_id, 1);
411 /*****************************************
412 * gtk_object_finalize:
417 *****************************************/
420 gtk_object_finalize (GtkObject *object)
422 GtkObjectData *odata, *next;
424 odata = object->object_data;
428 gtk_object_data_destroy (odata);
435 /*****************************************
441 *****************************************/
444 gtk_object_sink (GtkObject *object)
446 g_return_if_fail (object != NULL);
447 g_return_if_fail (GTK_IS_OBJECT (object));
449 if (GTK_OBJECT_FLOATING (object))
451 GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING);
452 gtk_object_unref (object);
456 /*****************************************
457 * gtk_object_destroy:
462 *****************************************/
465 gtk_object_destroy (GtkObject *object)
467 g_return_if_fail (object != NULL);
468 g_return_if_fail (GTK_IS_OBJECT (object));
470 if (!GTK_OBJECT_DESTROYED (object))
472 GTK_OBJECT_SET_FLAGS (object, GTK_DESTROYED);
473 gtk_signal_emit (object, object_signals[DESTROY]);
477 /*****************************************
480 * Weak refs are very similar to the old "destroy" signal. They allow
481 * one to register a callback that is called when the weakly
482 * referenced object is finalized.
484 * They are not implemented as a signal because they really are
485 * special and need to be used with great care. Unlike signals, who
486 * should be able to execute any code whatsoever.
488 * A weakref callback is not allowed to retain a reference to the
489 * object. In fact, the object is no longer there at all when it is
492 * A weakref callback is called atmost once.
494 *****************************************/
496 typedef struct _GtkWeakRef GtkWeakRef;
501 GtkDestroyNotify notify;
505 static const gchar *weakrefs_key = "gtk-weakrefs";
508 gtk_object_weakref (GtkObject *object,
509 GtkDestroyNotify notify,
514 g_return_if_fail (object != NULL);
515 g_return_if_fail (notify != NULL);
516 g_return_if_fail (GTK_IS_OBJECT (object));
518 weak = g_new (GtkWeakRef, 1);
519 weak->next = gtk_object_get_data (object, weakrefs_key);
520 weak->notify = notify;
522 gtk_object_set_data_full (object, weakrefs_key, weak,
523 gtk_object_notify_weaks);
527 gtk_object_weakunref (GtkObject *object,
528 GtkDestroyNotify notify,
531 GtkWeakRef *weaks, *w, **wp;
533 g_return_if_fail (object != NULL);
534 g_return_if_fail (GTK_IS_OBJECT (object));
536 weaks = gtk_object_get_data (object, weakrefs_key);
537 for (wp = &weaks; *wp; wp = &(*wp)->next)
540 if (w->notify == notify && w->data == data)
543 gtk_object_set_data_full (object, weakrefs_key, w->next,
544 gtk_object_notify_weaks);
554 gtk_object_notify_weaks (gpointer data)
558 w1 = (GtkWeakRef *)data;
562 w1->notify (w1->data);
569 /*****************************************
575 *****************************************/
578 gtk_object_new (guint type,
587 obj = gtk_type_new (type);
589 va_start (args1, type);
590 va_start (args2, type);
592 args = gtk_object_collect_args (&nargs, args1, args2);
593 gtk_object_setv (obj, nargs, args);
602 /*****************************************
608 *****************************************/
611 gtk_object_newv (guint type,
617 obj = gtk_type_new (type);
618 gtk_object_setv (obj, nargs, args);
623 /*****************************************
629 *****************************************/
632 gtk_object_getv (GtkObject *object,
638 g_return_if_fail (object != NULL);
643 for (i = 0; i < nargs; i++)
650 /* hm, the name cutting shouldn't be needed on gets, but what the heck...
652 lookup_name = g_strdup (args[i].name);
653 d = strchr (lookup_name, ':');
654 if (d && d[1] == ':')
656 d = strchr (d + 2, ':');
660 info = g_hash_table_lookup (arg_info_ht, lookup_name);
667 g_warning ("invalid arg name: \"%s\"\n", lookup_name);
668 args[i].type = GTK_TYPE_INVALID;
669 g_free (lookup_name);
672 else if (!gtk_type_is_a (object->klass->type, info->class_type))
674 g_warning ("invalid arg for %s: \"%s\"\n", gtk_type_name (object->klass->type), lookup_name);
675 args[i].type = GTK_TYPE_INVALID;
676 g_free (lookup_name);
679 else if (!info->arg_flags & GTK_ARG_READABLE)
681 g_warning ("arg is not supplied for read-access: \"%s\"\n", lookup_name);
682 args[i].type = GTK_TYPE_INVALID;
683 g_free (lookup_name);
687 g_free (lookup_name);
689 args[i].type = info->type;
690 gtk_type_get_arg (object, info->class_type, &args[i], info->arg_id);
694 /*****************************************
695 * gtk_object_query_args:
700 *****************************************/
702 struct _GtkQueryArgData
707 typedef struct _GtkQueryArgData GtkQueryArgData;
710 gtk_query_arg_foreach (gpointer key,
714 register GtkArgInfo *info;
715 register GtkQueryArgData *data;
720 if (info->class_type == data->class_type)
721 data->arg_list = g_list_prepend (data->arg_list, info);
725 gtk_object_query_args (GtkType class_type,
730 GtkQueryArgData query_data;
734 g_return_val_if_fail (nargs != NULL, NULL);
736 g_return_val_if_fail (gtk_type_is_a (class_type, gtk_object_get_type ()), NULL);
741 /* make sure the types class has been initialized, because
742 * the argument setup happens in the gtk_*_class_init() functions.
744 gtk_type_class (class_type);
746 query_data.arg_list = NULL;
747 query_data.class_type = class_type;
748 g_hash_table_foreach (arg_info_ht, gtk_query_arg_foreach, &query_data);
750 if (query_data.arg_list)
752 register GList *list;
755 list = query_data.arg_list;
762 g_assert (len == ((GtkObjectClass*) gtk_type_class (class_type))->n_args); /* paranoid */
764 args = g_new0 (GtkArg, len);
767 *arg_flags = g_new (guint32, len);
776 g_assert (info->seq_id > 0 && info->seq_id <= len); /* paranoid */
778 args[info->seq_id - 1].type = info->type;
779 args[info->seq_id - 1].name = info->name;
781 (*arg_flags)[info->seq_id - 1] = info->arg_flags;
785 g_list_free (query_data.arg_list);
793 /*****************************************
799 *****************************************/
802 gtk_object_set (GtkObject *object,
810 g_return_if_fail (object != NULL);
812 va_start (args1, object);
813 va_start (args2, object);
815 args = gtk_object_collect_args (&nargs, args1, args2);
816 gtk_object_setv (object, nargs, args);
823 /*****************************************
829 *****************************************/
832 gtk_object_setv (GtkObject *object,
838 g_return_if_fail (object != NULL);
843 for (i = 0; i < nargs; i++)
850 lookup_name = g_strdup (args[i].name);
851 d = strchr (lookup_name, ':');
852 if (d && d[1] == ':')
854 d = strchr (d + 2, ':');
858 info = g_hash_table_lookup (arg_info_ht, lookup_name);
867 g_warning ("invalid arg name: \"%s\"\n", lookup_name);
870 else if (info->type != args[i].type)
872 g_warning ("invalid arg type for: \"%s\"\n", lookup_name);
875 else if (!gtk_type_is_a (object->klass->type, info->class_type))
877 g_warning ("invalid arg for %s: \"%s\"\n", gtk_type_name (object->klass->type), lookup_name);
880 else if (!info->arg_flags & GTK_ARG_WRITABLE)
882 g_warning ("arg is not supplied for write-access: \"%s\"\n", lookup_name);
886 g_free (lookup_name);
891 gtk_type_set_arg (object, info->class_type, &args[i], info->arg_id);
895 /*****************************************
896 * gtk_object_add_arg_type:
901 *****************************************/
904 gtk_object_add_arg_type (const char *arg_name,
910 gchar class_part[1024];
914 g_return_if_fail (arg_name != NULL);
915 g_return_if_fail (arg_type > GTK_TYPE_NONE);
916 g_return_if_fail (arg_id > 0);
917 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) != 0);
919 arg_part = strchr (arg_name, ':');
920 if (!arg_part || (arg_part[0] != ':') || (arg_part[1] != ':'))
922 g_warning ("invalid arg name: \"%s\"\n", arg_name);
926 strncpy (class_part, arg_name, (glong) (arg_part - arg_name));
927 class_part[(glong) (arg_part - arg_name)] = '\0';
929 class_type = gtk_type_from_name (class_part);
932 g_warning ("invalid class name in arg: \"%s\"\n", arg_name);
936 info = g_new (GtkArgInfo, 1);
937 info->name = g_strdup (arg_name);
938 info->type = arg_type;
939 info->class_type = class_type;
940 info->arg_flags = arg_flags & (GTK_ARG_READABLE | GTK_ARG_WRITABLE);
941 info->arg_id = arg_id;
942 info->seq_id = ++((GtkObjectClass*) gtk_type_class (class_type))->n_args;
945 arg_info_ht = g_hash_table_new (g_str_hash, g_str_equal);
947 g_hash_table_insert (arg_info_ht, info->name, info);
950 /*****************************************
951 * gtk_object_get_arg_type:
956 *****************************************/
959 gtk_object_get_arg_type (const char *arg_name)
966 return GTK_TYPE_INVALID;
968 t = strchr (arg_name, ':');
969 if (!t || (t[0] != ':') || (t[1] != ':'))
971 g_warning ("invalid arg name: \"%s\"\n", arg_name);
972 return GTK_TYPE_INVALID;
975 t = strchr (t + 2, ':');
978 strncpy (buffer, arg_name, (long) (t - arg_name));
979 buffer[(long) (t - arg_name)] = '\0';
983 info = g_hash_table_lookup (arg_info_ht, (gpointer) arg_name);
987 return GTK_TYPE_INVALID;
990 /*****************************************
991 * gtk_object_set_data:
996 *****************************************/
999 gtk_object_set_data (GtkObject *object,
1003 gtk_object_set_data_full (object, key, data, NULL);
1006 /*****************************************
1007 * gtk_object_set_data_full:
1012 *****************************************/
1015 gtk_object_set_data_full (GtkObject *object,
1018 GtkDestroyNotify destroy)
1020 GtkObjectData *odata;
1021 GtkObjectData *prev;
1024 g_return_if_fail (object != NULL);
1025 g_return_if_fail (GTK_IS_OBJECT (object));
1026 g_return_if_fail (key != NULL);
1028 if (object_data_init)
1029 gtk_object_data_init ();
1031 id = g_hash_table_lookup (object_data_ht, (gpointer) key);
1038 odata = object->object_data;
1042 if (odata->id == *id)
1045 prev->next = odata->next;
1046 if (odata == object->object_data)
1047 object->object_data = odata->next;
1049 gtk_object_data_destroy (odata);
1054 odata = odata->next;
1062 id = gtk_object_data_id_alloc ();
1063 g_hash_table_insert (object_data_ht, (gpointer) key, id);
1066 odata = object->object_data;
1069 if (odata->id == *id)
1072 odata->destroy = destroy;
1076 odata = odata->next;
1079 odata = gtk_object_data_new ();
1082 odata->destroy = destroy;
1084 odata->next = object->object_data;
1085 object->object_data = odata;
1089 /*****************************************
1090 * gtk_object_get_data:
1095 *****************************************/
1098 gtk_object_get_data (GtkObject *object,
1101 GtkObjectData *odata;
1104 g_return_val_if_fail (object != NULL, NULL);
1105 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1106 g_return_val_if_fail (key != NULL, NULL);
1108 if (object_data_init)
1109 gtk_object_data_init ();
1111 id = g_hash_table_lookup (object_data_ht, (gpointer) key);
1114 odata = object->object_data;
1117 if (odata->id == *id)
1119 odata = odata->next;
1126 /*****************************************
1127 * gtk_object_remove_data:
1132 *****************************************/
1135 gtk_object_remove_data (GtkObject *object,
1138 g_return_if_fail (object != NULL);
1139 g_return_if_fail (GTK_IS_OBJECT (object));
1140 g_return_if_fail (key != NULL);
1142 gtk_object_set_data_full (object, key, NULL, NULL);
1145 /*****************************************
1146 * gtk_object_set_user_data:
1151 *****************************************/
1154 gtk_object_set_user_data (GtkObject *object,
1157 g_return_if_fail (object != NULL);
1158 g_return_if_fail (GTK_IS_OBJECT (object));
1160 gtk_object_set_data_full (object, user_data_key, data, NULL);
1163 /*****************************************
1164 * gtk_object_get_user_data:
1169 *****************************************/
1172 gtk_object_get_user_data (GtkObject *object)
1174 g_return_val_if_fail (object != NULL, NULL);
1175 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1177 return gtk_object_get_data (object, user_data_key);
1180 /*****************************************
1181 * gtk_object_check_cast:
1186 *****************************************/
1189 gtk_object_descriptive_type_name (GtkType type)
1193 name = gtk_type_name (type);
1201 gtk_object_check_cast (GtkObject *obj,
1206 g_warning ("invalid cast from (NULL) pointer to `%s'",
1207 gtk_object_descriptive_type_name (cast_type));
1212 g_warning ("invalid unclassed pointer in cast to `%s'",
1213 gtk_object_descriptive_type_name (cast_type));
1216 if (obj->klass->type < GTK_TYPE_OBJECT)
1218 g_warning ("invalid class type `%s' in cast to `%s'",
1219 gtk_object_descriptive_type_name (obj->klass->type),
1220 gtk_object_descriptive_type_name (cast_type));
1223 if (!gtk_type_is_a (obj->klass->type, cast_type))
1225 g_warning ("invalid cast from `%s' to `%s'",
1226 gtk_object_descriptive_type_name (obj->klass->type),
1227 gtk_object_descriptive_type_name (cast_type));
1234 /*****************************************
1235 * gtk_object_check_class_cast:
1240 *****************************************/
1243 gtk_object_check_class_cast (GtkObjectClass *klass,
1248 g_warning ("invalid class cast from (NULL) pointer to `%s'",
1249 gtk_object_descriptive_type_name (cast_type));
1252 if (klass->type < GTK_TYPE_OBJECT)
1254 g_warning ("invalid class type `%s' in class cast to `%s'",
1255 gtk_object_descriptive_type_name (klass->type),
1256 gtk_object_descriptive_type_name (cast_type));
1259 if (!gtk_type_is_a (klass->type, cast_type))
1261 g_warning ("invalid class cast from `%s' to `%s'",
1262 gtk_object_descriptive_type_name (klass->type),
1263 gtk_object_descriptive_type_name (cast_type));
1270 /*****************************************
1271 * gtk_object_data_init:
1276 *****************************************/
1279 gtk_object_data_init ()
1281 if (object_data_init)
1283 object_data_init = FALSE;
1285 object_data_ht = g_hash_table_new (g_str_hash, g_str_equal);
1289 /*****************************************
1290 * gtk_object_data_new:
1295 *****************************************/
1297 static GtkObjectData*
1298 gtk_object_data_new ()
1300 GtkObjectData *odata;
1302 if (!object_data_mem_chunk)
1303 object_data_mem_chunk = g_mem_chunk_new ("object data mem chunk",
1304 sizeof (GtkObjectData),
1305 1024, G_ALLOC_AND_FREE);
1307 odata = g_chunk_new (GtkObjectData, object_data_mem_chunk);
1311 odata->destroy = NULL;
1317 /*****************************************
1318 * gtk_object_data_destroy:
1323 *****************************************/
1326 gtk_object_data_destroy (GtkObjectData *odata)
1328 g_return_if_fail (odata != NULL);
1331 odata->destroy (odata->data);
1333 g_mem_chunk_free (object_data_mem_chunk, odata);
1336 /*****************************************
1337 * gtk_object_data_id_alloc:
1342 *****************************************/
1345 gtk_object_data_id_alloc ()
1347 static guint next_id = 1;
1350 if (!object_data_id_list ||
1351 (object_data_id_index == OBJECT_DATA_ID_CHUNK))
1353 ids = g_new (guint, OBJECT_DATA_ID_CHUNK);
1354 object_data_id_index = 0;
1355 object_data_id_list = g_slist_prepend (object_data_id_list, ids);
1359 ids = object_data_id_list->data;
1362 ids[object_data_id_index] = next_id++;
1363 return &ids[object_data_id_index++];
1366 /*****************************************
1367 * gtk_object_collect_args:
1372 *****************************************/
1375 gtk_object_collect_args (guint *nargs,
1390 name = va_arg (args1, char *);
1397 type = gtk_object_get_arg_type (name);
1399 switch (GTK_FUNDAMENTAL_TYPE (type))
1401 case GTK_TYPE_INVALID:
1402 g_warning ("invalid arg name: \"%s\" %x\n", name, type);
1403 (void) va_arg (args1, long);
1412 case GTK_TYPE_FLAGS:
1413 (void) va_arg (args1, gint);
1416 case GTK_TYPE_ULONG:
1417 (void) va_arg (args1, glong);
1419 case GTK_TYPE_FLOAT:
1420 (void) va_arg (args1, gfloat);
1422 case GTK_TYPE_DOUBLE:
1423 (void) va_arg (args1, gdouble);
1425 case GTK_TYPE_STRING:
1426 (void) va_arg (args1, gchar*);
1428 case GTK_TYPE_POINTER:
1429 case GTK_TYPE_BOXED:
1430 (void) va_arg (args1, gpointer);
1432 case GTK_TYPE_SIGNAL:
1433 (void) va_arg (args1, GtkFunction);
1434 (void) va_arg (args1, gpointer);
1436 case GTK_TYPE_FOREIGN:
1437 (void) va_arg (args1, gpointer);
1438 (void) va_arg (args1, GtkDestroyNotify);
1440 case GTK_TYPE_CALLBACK:
1441 (void) va_arg (args1, GtkCallbackMarshal);
1442 (void) va_arg (args1, gpointer);
1443 (void) va_arg (args1, GtkDestroyNotify);
1445 case GTK_TYPE_C_CALLBACK:
1446 (void) va_arg (args1, GtkFunction);
1447 (void) va_arg (args1, gpointer);
1450 (void) va_arg (args1, gint);
1451 (void) va_arg (args1, GtkArg*);
1453 case GTK_TYPE_OBJECT:
1454 (void) va_arg (args1, GtkObject*);
1457 g_error ("unsupported type %s in args", gtk_type_name (type));
1469 args = g_new0 (GtkArg, n);
1471 for (i = 0; i < n; i++)
1473 args[i].name = va_arg (args2, char *);
1474 args[i].type = gtk_object_get_arg_type (args[i].name);
1476 switch (GTK_FUNDAMENTAL_TYPE (args[i].type))
1478 case GTK_TYPE_INVALID:
1479 (void) va_arg (args2, long);
1485 GTK_VALUE_CHAR(args[i]) = va_arg (args2, gint);
1488 GTK_VALUE_BOOL(args[i]) = va_arg (args2, gint);
1491 GTK_VALUE_INT(args[i]) = va_arg (args2, gint);
1494 GTK_VALUE_UINT(args[i]) = va_arg (args2, guint);
1497 GTK_VALUE_ENUM(args[i]) = va_arg (args2, gint);
1499 case GTK_TYPE_FLAGS:
1500 GTK_VALUE_FLAGS(args[i]) = va_arg (args2, gint);
1503 GTK_VALUE_LONG(args[i]) = va_arg (args2, glong);
1505 case GTK_TYPE_ULONG:
1506 GTK_VALUE_ULONG(args[i]) = va_arg (args2, gulong);
1508 case GTK_TYPE_FLOAT:
1509 GTK_VALUE_FLOAT(args[i]) = va_arg (args2, gfloat);
1511 case GTK_TYPE_DOUBLE:
1512 GTK_VALUE_DOUBLE(args[i]) = va_arg (args2, gdouble);
1514 case GTK_TYPE_STRING:
1515 GTK_VALUE_STRING(args[i]) = va_arg (args2, gchar*);
1517 case GTK_TYPE_POINTER:
1518 GTK_VALUE_POINTER(args[i]) = va_arg (args2, gpointer);
1520 case GTK_TYPE_BOXED:
1521 GTK_VALUE_BOXED(args[i]) = va_arg (args2, gpointer);
1523 case GTK_TYPE_SIGNAL:
1524 GTK_VALUE_SIGNAL(args[i]).f = va_arg (args2, GtkFunction);
1525 GTK_VALUE_SIGNAL(args[i]).d = va_arg (args2, gpointer);
1527 case GTK_TYPE_FOREIGN:
1528 GTK_VALUE_FOREIGN(args[i]).data = va_arg (args2, gpointer);
1529 GTK_VALUE_FOREIGN(args[i]).notify =
1530 va_arg (args2, GtkDestroyNotify);
1532 case GTK_TYPE_CALLBACK:
1533 GTK_VALUE_CALLBACK(args[i]).marshal =
1534 va_arg (args2, GtkCallbackMarshal);
1535 GTK_VALUE_CALLBACK(args[i]).data = va_arg (args2, gpointer);
1536 GTK_VALUE_CALLBACK(args[i]).notify =
1537 va_arg (args2, GtkDestroyNotify);
1539 case GTK_TYPE_C_CALLBACK:
1540 GTK_VALUE_C_CALLBACK(args[i]).func = va_arg (args2, GtkFunction);
1541 GTK_VALUE_C_CALLBACK(args[i]).func_data =
1542 va_arg (args2, gpointer);
1545 GTK_VALUE_ARGS(args[i]).n_args = va_arg (args2, gint);
1546 GTK_VALUE_ARGS(args[i]).args = va_arg (args2, GtkArg*);
1548 case GTK_TYPE_OBJECT:
1549 GTK_VALUE_OBJECT(args[i]) = va_arg (args2, GtkObject*);
1550 g_assert (GTK_VALUE_OBJECT(args[i]) == NULL ||
1551 GTK_CHECK_TYPE (GTK_VALUE_OBJECT(args[i]),
1555 g_error ("unsupported type %s in args",
1556 gtk_type_name (args[i].type));
1567 #undef gtk_object_ref
1568 #undef gtk_object_unref
1571 gtk_object_ref (GtkObject *object)
1573 g_return_if_fail (object != NULL);
1574 g_return_if_fail (GTK_IS_OBJECT (object));
1576 object->ref_count += 1;
1580 gtk_object_unref (GtkObject *object)
1582 g_return_if_fail (object != NULL);
1583 g_return_if_fail (GTK_IS_OBJECT (object));
1585 if (object->ref_count == 1)
1586 gtk_object_destroy (object);
1588 if (object->ref_count > 0)
1589 object->ref_count -= 1;
1591 if (object->ref_count == 0)
1593 #ifdef G_ENABLE_DEBUG
1594 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1596 g_assert (g_hash_table_lookup (living_objs_ht, object) == object);
1597 g_hash_table_remove (living_objs_ht, object);
1600 #endif /* G_ENABLE_DEBUG */
1601 object->klass->finalize (object);
1606 #ifdef G_ENABLE_DEBUG
1607 static GtkObject *gtk_trace_object = NULL;
1609 gtk_trace_referencing (gpointer *o,
1616 GtkObject *object = (GtkObject*) o;
1618 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1620 g_return_if_fail (object != NULL);
1621 g_return_if_fail (GTK_IS_OBJECT (object));
1623 exists = g_hash_table_lookup (living_objs_ht, object) != NULL;
1626 (object == gtk_trace_object ||
1627 gtk_trace_object == (void*)42))
1628 printf ("trace: object_%s: (%s:%p)->ref_count=%d%s (%s_f%02d:%d)\n",
1629 do_ref ? "ref" : "unref",
1630 gtk_type_name (GTK_OBJECT_TYPE (object)),
1633 do_ref ? " + 1" : " - 1 ",
1639 printf ("trace: object_%s(%p): no such object! (%s_f%02d:%d)\n",
1640 do_ref ? "ref" : "unref",
1648 gtk_object_ref (object);
1650 gtk_object_unref (object);
1652 #endif /* G_ENABLE_DEBUG */