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"
25 #define GTK_OBJECT_DEBUG 1
27 #define OBJECT_DATA_ID_CHUNK 1024
42 typedef struct _GtkObjectData GtkObjectData;
43 typedef struct _GtkArgInfo GtkArgInfo;
62 static void gtk_object_class_init (GtkObjectClass *klass);
63 static void gtk_object_init (GtkObject *object);
64 static void gtk_object_set_arg (GtkObject *object,
67 static void gtk_object_get_arg (GtkObject *object,
70 static void gtk_object_real_destroy (GtkObject *object);
71 static void gtk_object_finalize (GtkObject *object);
72 static void gtk_object_notify_weaks (GtkObject *object);
73 static void gtk_object_data_init (void);
74 static GtkObjectData* gtk_object_data_new (void);
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 gint object_signals[LAST_SIGNAL] = { 0 };
84 static gint object_data_init = TRUE;
85 static GHashTable *object_data_ht = NULL;
86 static GMemChunk *object_data_mem_chunk = NULL;
87 static GtkObjectData *object_data_free_list = 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";
96 #ifdef GTK_OBJECT_DEBUG
97 static int obj_count = 0;
98 static GSList *living_objs = NULL;
101 gtk_object_debug (void)
108 printf ("living objects (%d):\n", obj_count);
109 for (node = living_objs; node; node = node->next)
113 obj = (GtkObject*) node->data;
114 printf ("%p: %s %d %s\n",
115 obj, gtk_type_name (GTK_OBJECT_TYPE (obj)),
117 GTK_OBJECT_FLOATING (obj)? "floating" : "");
120 printf ("%d living objects\n", obj_count);
122 #endif GTK_OBJECT_DEBUG
124 /*****************************************
125 * gtk_object_init_type:
130 *****************************************/
133 gtk_object_init_type ()
135 GtkType object_type = 0;
136 GtkTypeInfo object_info =
140 sizeof (GtkObjectClass),
141 (GtkClassInitFunc) gtk_object_class_init,
142 (GtkObjectInitFunc) gtk_object_init,
147 object_type = gtk_type_unique (0, &object_info);
148 g_assert (object_type == GTK_TYPE_OBJECT);
150 #ifdef GTK_OBJECT_DEBUG
151 ATEXIT (gtk_object_debug);
152 #endif GTK_OBJECT_DEBUG
156 gtk_object_get_type ()
158 return GTK_TYPE_OBJECT;
161 /*****************************************
162 * gtk_object_class_init:
167 *****************************************/
170 gtk_object_class_init (GtkObjectClass *class)
172 class->signals = NULL;
176 gtk_object_add_arg_type ("GtkObject::user_data", GTK_TYPE_POINTER, ARG_USER_DATA);
177 gtk_object_add_arg_type ("GtkObject::signal", GTK_TYPE_SIGNAL, ARG_SIGNAL);
178 gtk_object_add_arg_type ("GtkObject::object_signal", GTK_TYPE_SIGNAL, ARG_OBJECT_SIGNAL);
180 object_signals[DESTROY] =
181 gtk_signal_new ("destroy",
184 GTK_SIGNAL_OFFSET (GtkObjectClass, destroy),
185 gtk_signal_default_marshaller,
188 gtk_object_class_add_signals (class, object_signals, LAST_SIGNAL);
190 class->destroy = gtk_object_real_destroy;
191 class->finalize = gtk_object_finalize;
194 /*****************************************
195 * gtk_object_real_destroy:
200 *****************************************/
203 gtk_object_real_destroy (GtkObject *object)
205 g_return_if_fail (object != NULL);
206 g_return_if_fail (GTK_IS_OBJECT (object));
208 /* FIXME: gtk_signal_handlers_destroy (object); */
210 /* object->klass = gtk_type_class (gtk_destroyed_get_type ()); */
213 /*****************************************
219 *****************************************/
222 gtk_object_init (GtkObject *object)
224 GTK_OBJECT_FLAGS (object) = GTK_FLOATING;
226 object->ref_count = 1;
227 object->object_data = NULL;
229 #ifdef GTK_OBJECT_DEBUG
231 living_objs = g_slist_prepend (living_objs, object);
232 #endif GTK_OBJECT_DEBUG
235 /*****************************************
236 * gtk_object_set_arg:
241 *****************************************/
244 gtk_object_set_arg (GtkObject *object,
251 gtk_object_set_user_data (object, GTK_VALUE_POINTER (*arg));
254 if ((arg->name[9 + 2 + 6] != ':') || (arg->name[9 + 2 + 7] != ':'))
256 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
259 gtk_signal_connect (object, arg->name + 9 + 2 + 6 + 2,
260 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
261 GTK_VALUE_SIGNAL (*arg).d);
263 case ARG_OBJECT_SIGNAL:
264 if ((arg->name[9 + 2 + 13] != ':') || (arg->name[9 + 2 + 14] != ':'))
266 g_warning ("invalid signal argument: \"%s\"\n", arg->name);
269 gtk_signal_connect_object (object, arg->name + 9 + 2 + 13 + 2,
270 (GtkSignalFunc) GTK_VALUE_SIGNAL (*arg).f,
271 (GtkObject*) GTK_VALUE_SIGNAL (*arg).d);
274 arg->type = GTK_TYPE_INVALID;
279 /*****************************************
280 * gtk_object_get_arg:
285 *****************************************/
288 gtk_object_get_arg (GtkObject *object,
295 GTK_VALUE_POINTER (*arg) = gtk_object_get_user_data (object);
298 case ARG_OBJECT_SIGNAL:
300 arg->type = GTK_TYPE_INVALID;
305 /*****************************************
306 * gtk_object_class_add_signals:
311 *****************************************/
314 gtk_object_class_add_signals (GtkObjectClass *class,
321 g_return_if_fail (class != NULL);
323 new_signals = g_new (gint, class->nsignals + nsignals);
324 for (i = 0; i < class->nsignals; i++)
325 new_signals[i] = class->signals[i];
326 for (i = 0; i < nsignals; i++)
327 new_signals[class->nsignals + i] = signals[i];
329 g_free (class->signals);
330 class->signals = new_signals;
331 class->nsignals += nsignals;
334 /*****************************************
335 * gtk_object_class_add_user_signal:
340 *****************************************/
343 gtk_object_class_add_user_signal (GtkObjectClass *class,
345 GtkSignalMarshaller marshaller,
355 g_return_val_if_fail (class != NULL, 0);
359 params = g_new (GtkType, nparams);
361 va_start (args, nparams);
363 for (i = 0; i < nparams; i++)
364 params[i] = va_arg (args, GtkType);
371 signal_id = gtk_signal_newv (name,
383 gtk_object_class_add_signals (class, &signal_id, 1);
388 /*****************************************
394 *****************************************/
397 gtk_object_ref (GtkObject *object)
399 g_return_if_fail (object != NULL);
400 g_return_if_fail (GTK_IS_OBJECT (object));
402 object->ref_count += 1;
405 /*****************************************
411 *****************************************/
414 gtk_object_unref (GtkObject *object)
416 g_return_if_fail (object != NULL);
417 g_return_if_fail (GTK_IS_OBJECT (object));
419 if (object->ref_count == 1)
420 gtk_object_destroy (object);
422 if (object->ref_count > 0)
423 object->ref_count -= 1;
425 if (object->ref_count == 0)
429 #ifdef GTK_OBJECT_DEBUG
430 g_assert (g_slist_find (living_objs, object));
431 living_objs = g_slist_remove (living_objs, object);
433 fprintf (stderr, "finalizing %p %s\n", object, gtk_type_name (object->klass->type));
435 #endif GTK_OBJECT_DEBUG
437 object->klass->finalize (object);
441 /*****************************************
442 * gtk_object_finalize:
447 *****************************************/
450 gtk_object_finalize (GtkObject *object)
452 GtkObjectData *odata;
454 gtk_object_notify_weaks (object);
456 if (object->object_data)
458 odata = object->object_data;
462 odata->next = object_data_free_list;
463 object_data_free_list = object->object_data;
464 object->object_data = NULL;
470 /*****************************************
476 *****************************************/
479 gtk_object_sink (GtkObject *object)
481 if (GTK_OBJECT_FLOATING (object))
483 GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING);
484 gtk_object_unref (object);
488 /*****************************************
489 * gtk_object_destroy:
494 *****************************************/
497 gtk_object_destroy (GtkObject *object)
499 g_return_if_fail (object != NULL);
500 g_return_if_fail (GTK_IS_OBJECT (object));
502 if (!GTK_OBJECT_DESTROYED (object))
504 GTK_OBJECT_SET_FLAGS (object, GTK_DESTROYED);
505 gtk_signal_emit (object, object_signals[DESTROY]);
509 /*****************************************
512 * Weak refs are very similar to the old "destroy" signal. They allow
513 * one to register a callback that is called when the weakly
514 * referenced object is destroyed.
516 * They are not implemented as a signal because they really are
517 * special and need to be used with great care. Unlike signals, who
518 * should be able to execute any code whatsoever.
520 * A weakref callback is not allowed to retain a reference to the
521 * object. In fact, the object is no longer there at all when it is
524 * A weakref callback is called atmost once.
526 *****************************************/
528 typedef struct _GtkWeakRef GtkWeakRef;
533 GtkDestroyNotify notify;
537 static const gchar *weakrefs_key = "gtk-weakrefs";
540 gtk_object_weakref (GtkObject *object,
541 GtkDestroyNotify notify,
546 g_return_if_fail (object != NULL);
547 g_return_if_fail (notify != NULL);
548 g_return_if_fail (GTK_IS_OBJECT (object));
550 weak = g_new (GtkWeakRef, 1);
551 weak->next = gtk_object_get_data (object, weakrefs_key);
552 weak->notify = notify;
554 gtk_object_set_data (object, weakrefs_key, weak);
558 gtk_object_weakunref (GtkObject *object,
559 GtkDestroyNotify notify,
562 GtkWeakRef *weaks, *w, **wp;
564 g_return_if_fail (object != NULL);
565 g_return_if_fail (GTK_IS_OBJECT (object));
567 weaks = gtk_object_get_data (object, weakrefs_key);
568 for (wp = &weaks; *wp; wp = &(*wp)->next)
571 if (w->notify == notify && w->data == data)
574 gtk_object_set_data (object, weakrefs_key, w->next);
584 gtk_object_notify_weaks (GtkObject *object)
588 w1 = gtk_object_get_data (object, weakrefs_key);
589 gtk_object_set_data (object, weakrefs_key, NULL);
593 w1->notify (w1->data);
600 /*****************************************
606 *****************************************/
609 gtk_object_new (guint type,
618 obj = gtk_type_new (type);
620 va_start (args1, type);
621 va_start (args2, type);
623 args = gtk_object_collect_args (&nargs, args1, args2);
624 gtk_object_setv (obj, nargs, args);
633 /*****************************************
639 *****************************************/
642 gtk_object_newv (guint type,
648 obj = gtk_type_new (type);
649 gtk_object_setv (obj, nargs, args);
654 /*****************************************
660 *****************************************/
663 gtk_object_getv (GtkObject *object,
669 g_return_if_fail (object != NULL);
674 for (i = 0; i < nargs; i++)
681 /* hm, the name cutting shouldn't be needed on gets, but what the heck...
683 lookup_name = g_strdup (args[i].name);
684 d = strchr (lookup_name, ':');
685 if (d && d[1] == ':')
687 d = strchr (d + 2, ':');
691 info = g_hash_table_lookup (arg_info_ht, lookup_name);
698 g_warning ("invalid arg name: \"%s\"\n", lookup_name);
699 args[i].type = GTK_TYPE_INVALID;
700 g_free (lookup_name);
703 else if (!gtk_type_is_a (object->klass->type, info->class_type))
705 g_warning ("invalid arg for %s: \"%s\"\n", gtk_type_name (object->klass->type), lookup_name);
706 args[i].type = GTK_TYPE_INVALID;
707 g_free (lookup_name);
711 g_free (lookup_name);
713 args[i].type = info->type;
714 gtk_type_get_arg (object, info->class_type, &args[i], info->arg_id);
718 /*****************************************
719 * gtk_object_query_args:
724 *****************************************/
726 struct _GtkQueryArgData
731 typedef struct _GtkQueryArgData GtkQueryArgData;
734 gtk_query_arg_foreach (gpointer key,
738 register GtkArgInfo *info;
739 register GtkQueryArgData *data;
744 if (info->class_type == data->class_type)
745 data->arg_list = g_list_prepend (data->arg_list, info);
749 gtk_object_query_args (GtkType class_type,
753 GtkQueryArgData query_data;
755 g_return_val_if_fail (nargs != NULL, NULL);
757 g_return_val_if_fail (gtk_type_is_a (class_type, gtk_object_get_type ()), NULL);
762 query_data.arg_list = NULL;
763 query_data.class_type = class_type;
764 g_hash_table_foreach (arg_info_ht, gtk_query_arg_foreach, &query_data);
766 if (query_data.arg_list)
768 register GList *list;
771 list = query_data.arg_list;
778 g_assert (len == ((GtkObjectClass*) gtk_type_class (class_type))->n_args); /* paranoid */
780 args = g_new0 (GtkArg, len);
790 g_assert (info->seq_id > 0 && info->seq_id <= len); /* paranoid */
792 args[info->seq_id - 1].type = info->type;
793 args[info->seq_id - 1].name = info->name;
797 g_list_free (query_data.arg_list);
805 /*****************************************
811 *****************************************/
814 gtk_object_set (GtkObject *object,
822 g_return_if_fail (object != NULL);
824 va_start (args1, object);
825 va_start (args2, object);
827 args = gtk_object_collect_args (&nargs, args1, args2);
828 gtk_object_setv (object, nargs, args);
835 /*****************************************
841 *****************************************/
844 gtk_object_setv (GtkObject *object,
850 g_return_if_fail (object != NULL);
855 for (i = 0; i < nargs; i++)
862 lookup_name = g_strdup (args[i].name);
863 d = strchr (lookup_name, ':');
864 if (d && d[1] == ':')
866 d = strchr (d + 2, ':');
870 info = g_hash_table_lookup (arg_info_ht, lookup_name);
879 g_warning ("invalid arg name: \"%s\"\n", lookup_name);
882 else if (info->type != args[i].type)
884 g_warning ("invalid arg type for: \"%s\"\n", lookup_name);
887 else if (!gtk_type_is_a (object->klass->type, info->class_type))
889 g_warning ("invalid arg for %s: \"%s\"\n", gtk_type_name (object->klass->type), lookup_name);
893 g_free (lookup_name);
898 gtk_type_set_arg (object, info->class_type, &args[i], info->arg_id);
902 /*****************************************
903 * gtk_object_add_arg_type:
908 *****************************************/
911 gtk_object_add_arg_type (const char *arg_name,
916 gchar class_part[1024];
920 g_return_if_fail (arg_id > 0);
922 arg_part = strchr (arg_name, ':');
923 if (!arg_part || (arg_part[0] != ':') || (arg_part[1] != ':'))
925 g_warning ("invalid arg name: \"%s\"\n", arg_name);
929 strncpy (class_part, arg_name, (glong) (arg_part - arg_name));
930 class_part[(glong) (arg_part - arg_name)] = '\0';
932 class_type = gtk_type_from_name (class_part);
935 g_warning ("invalid class name in arg: \"%s\"\n", arg_name);
939 info = g_new (GtkArgInfo, 1);
940 info->name = g_strdup (arg_name);
941 info->type = arg_type;
942 info->class_type = class_type;
943 info->arg_id = arg_id;
944 info->seq_id = ++((GtkObjectClass*) gtk_type_class (class_type))->n_args;
947 arg_info_ht = g_hash_table_new (g_str_hash, g_str_equal);
949 g_hash_table_insert (arg_info_ht, info->name, info);
952 /*****************************************
953 * gtk_object_get_arg_type:
958 *****************************************/
961 gtk_object_get_arg_type (const char *arg_name)
968 return GTK_TYPE_INVALID;
970 t = strchr (arg_name, ':');
971 if (!t || (t[0] != ':') || (t[1] != ':'))
973 g_warning ("invalid arg name: \"%s\"\n", arg_name);
974 return GTK_TYPE_INVALID;
977 t = strchr (t + 2, ':');
980 strncpy (buffer, arg_name, (long) (t - arg_name));
981 buffer[(long) (t - arg_name)] = '\0';
985 info = g_hash_table_lookup (arg_info_ht, (gpointer) arg_name);
989 return GTK_TYPE_INVALID;
992 /*****************************************
993 * gtk_object_set_data:
998 *****************************************/
1001 gtk_object_set_data (GtkObject *object,
1005 GtkObjectData *odata;
1006 GtkObjectData *prev;
1009 g_return_if_fail (object != NULL);
1010 g_return_if_fail (GTK_IS_OBJECT (object));
1011 g_return_if_fail (key != NULL);
1013 if (object_data_init)
1014 gtk_object_data_init ();
1016 id = g_hash_table_lookup (object_data_ht, (gpointer) key);
1023 odata = object->object_data;
1027 if (odata->id == *id)
1030 prev->next = odata->next;
1031 if (odata == object->object_data)
1032 object->object_data = odata->next;
1034 gtk_object_data_destroy (odata);
1039 odata = odata->next;
1047 id = gtk_object_data_id_alloc ();
1048 g_hash_table_insert (object_data_ht, (gpointer) key, id);
1051 odata = object->object_data;
1054 if (odata->id == *id)
1060 odata = odata->next;
1063 odata = gtk_object_data_new ();
1067 odata->next = object->object_data;
1068 object->object_data = odata;
1072 /*****************************************
1073 * gtk_object_get_data:
1078 *****************************************/
1081 gtk_object_get_data (GtkObject *object,
1084 GtkObjectData *odata;
1087 g_return_val_if_fail (object != NULL, NULL);
1088 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1089 g_return_val_if_fail (key != NULL, NULL);
1091 if (object_data_init)
1092 gtk_object_data_init ();
1094 id = g_hash_table_lookup (object_data_ht, (gpointer) key);
1097 odata = object->object_data;
1100 if (odata->id == *id)
1102 odata = odata->next;
1109 /*****************************************
1110 * gtk_object_remove_data:
1115 *****************************************/
1118 gtk_object_remove_data (GtkObject *object,
1121 g_return_if_fail (object != NULL);
1122 g_return_if_fail (GTK_IS_OBJECT (object));
1123 g_return_if_fail (key != NULL);
1125 gtk_object_set_data (object, key, NULL);
1128 /*****************************************
1129 * gtk_object_set_user_data:
1134 *****************************************/
1137 gtk_object_set_user_data (GtkObject *object,
1140 g_return_if_fail (object != NULL);
1141 g_return_if_fail (GTK_IS_OBJECT (object));
1143 gtk_object_set_data (object, user_data_key, data);
1146 /*****************************************
1147 * gtk_object_get_user_data:
1152 *****************************************/
1155 gtk_object_get_user_data (GtkObject *object)
1157 g_return_val_if_fail (object != NULL, NULL);
1158 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1160 return gtk_object_get_data (object, user_data_key);
1163 /*****************************************
1164 * gtk_object_check_cast:
1169 *****************************************/
1172 gtk_object_check_cast (GtkObject *obj,
1175 if (obj && obj->klass && !gtk_type_is_a (obj->klass->type, cast_type))
1177 gchar *from_name = gtk_type_name (obj->klass->type);
1178 gchar *to_name = gtk_type_name (cast_type);
1180 g_warning ("invalid cast from \"%s\" to \"%s\"",
1181 from_name ? from_name : "(unknown)",
1182 to_name ? to_name : "(unknown)");
1188 /*****************************************
1189 * gtk_object_check_class_cast:
1194 *****************************************/
1197 gtk_object_check_class_cast (GtkObjectClass *klass,
1200 if (klass && !gtk_type_is_a (klass->type, cast_type))
1201 g_warning ("invalid cast from \"%sClass\" to \"%sClass\"",
1202 gtk_type_name (klass->type),
1203 gtk_type_name (cast_type));
1208 /*****************************************
1209 * gtk_object_data_init:
1214 *****************************************/
1217 gtk_object_data_init ()
1219 if (object_data_init)
1221 object_data_init = FALSE;
1223 object_data_ht = g_hash_table_new (g_str_hash, g_str_equal);
1227 /*****************************************
1228 * gtk_object_data_new:
1233 *****************************************/
1235 static GtkObjectData*
1236 gtk_object_data_new ()
1238 GtkObjectData *odata;
1240 if (!object_data_mem_chunk)
1241 object_data_mem_chunk = g_mem_chunk_new ("object data mem chunk",
1242 sizeof (GtkObjectData),
1243 1024, G_ALLOC_AND_FREE);
1245 odata = g_chunk_new (GtkObjectData, object_data_mem_chunk);
1254 /*****************************************
1255 * gtk_object_data_destroy:
1260 *****************************************/
1263 gtk_object_data_destroy (GtkObjectData *odata)
1265 g_return_if_fail (odata != NULL);
1267 g_mem_chunk_free (object_data_mem_chunk, odata);
1270 /*****************************************
1271 * gtk_object_data_id_alloc:
1276 *****************************************/
1279 gtk_object_data_id_alloc ()
1281 static guint next_id = 1;
1284 if (!object_data_id_list ||
1285 (object_data_id_index == OBJECT_DATA_ID_CHUNK))
1287 ids = g_new (guint, OBJECT_DATA_ID_CHUNK);
1288 object_data_id_index = 0;
1289 object_data_id_list = g_slist_prepend (object_data_id_list, ids);
1293 ids = object_data_id_list->data;
1296 ids[object_data_id_index] = next_id++;
1297 return &ids[object_data_id_index++];
1300 /*****************************************
1301 * gtk_object_collect_args:
1306 *****************************************/
1309 gtk_object_collect_args (guint *nargs,
1324 name = va_arg (args1, char *);
1331 type = gtk_object_get_arg_type (name);
1333 switch (GTK_FUNDAMENTAL_TYPE (type))
1335 case GTK_TYPE_INVALID:
1336 g_warning ("invalid arg name: \"%s\" %x\n", name, type);
1337 (void) va_arg (args1, long);
1346 case GTK_TYPE_FLAGS:
1347 (void) va_arg (args1, gint);
1350 case GTK_TYPE_ULONG:
1351 (void) va_arg (args1, glong);
1353 case GTK_TYPE_FLOAT:
1354 (void) va_arg (args1, gfloat);
1356 case GTK_TYPE_DOUBLE:
1357 (void) va_arg (args1, gdouble);
1359 case GTK_TYPE_STRING:
1360 (void) va_arg (args1, gchar*);
1362 case GTK_TYPE_POINTER:
1363 case GTK_TYPE_BOXED:
1364 (void) va_arg (args1, gpointer);
1366 case GTK_TYPE_SIGNAL:
1367 (void) va_arg (args1, GtkFunction);
1368 (void) va_arg (args1, gpointer);
1370 case GTK_TYPE_FOREIGN:
1371 (void) va_arg (args1, gpointer);
1372 (void) va_arg (args1, GtkDestroyNotify);
1374 case GTK_TYPE_CALLBACK:
1375 (void) va_arg (args1, GtkCallbackMarshal);
1376 (void) va_arg (args1, gpointer);
1377 (void) va_arg (args1, GtkDestroyNotify);
1379 case GTK_TYPE_C_CALLBACK:
1380 (void) va_arg (args1, GtkFunction);
1381 (void) va_arg (args1, gpointer);
1384 (void) va_arg (args1, gint);
1385 (void) va_arg (args1, GtkArg*);
1387 case GTK_TYPE_OBJECT:
1388 (void) va_arg (args1, GtkObject*);
1391 g_error ("unsupported type %s in args", gtk_type_name (type));
1403 args = g_new0 (GtkArg, n);
1405 for (i = 0; i < n; i++)
1407 args[i].name = va_arg (args2, char *);
1408 args[i].type = gtk_object_get_arg_type (args[i].name);
1410 switch (GTK_FUNDAMENTAL_TYPE (args[i].type))
1412 case GTK_TYPE_INVALID:
1413 (void) va_arg (args2, long);
1419 GTK_VALUE_CHAR(args[i]) = va_arg (args2, gint);
1422 GTK_VALUE_BOOL(args[i]) = va_arg (args2, gint);
1425 GTK_VALUE_INT(args[i]) = va_arg (args2, gint);
1428 GTK_VALUE_UINT(args[i]) = va_arg (args2, guint);
1431 GTK_VALUE_ENUM(args[i]) = va_arg (args2, gint);
1433 case GTK_TYPE_FLAGS:
1434 GTK_VALUE_FLAGS(args[i]) = va_arg (args2, gint);
1437 GTK_VALUE_LONG(args[i]) = va_arg (args2, glong);
1439 case GTK_TYPE_ULONG:
1440 GTK_VALUE_ULONG(args[i]) = va_arg (args2, gulong);
1442 case GTK_TYPE_FLOAT:
1443 GTK_VALUE_FLOAT(args[i]) = va_arg (args2, gfloat);
1445 case GTK_TYPE_DOUBLE:
1446 GTK_VALUE_DOUBLE(args[i]) = va_arg (args2, gdouble);
1448 case GTK_TYPE_STRING:
1449 GTK_VALUE_STRING(args[i]) = va_arg (args2, gchar*);
1451 case GTK_TYPE_POINTER:
1452 GTK_VALUE_POINTER(args[i]) = va_arg (args2, gpointer);
1454 case GTK_TYPE_BOXED:
1455 GTK_VALUE_BOXED(args[i]) = va_arg (args2, gpointer);
1457 case GTK_TYPE_SIGNAL:
1458 GTK_VALUE_SIGNAL(args[i]).f = va_arg (args2, GtkFunction);
1459 GTK_VALUE_SIGNAL(args[i]).d = va_arg (args2, gpointer);
1461 case GTK_TYPE_FOREIGN:
1462 GTK_VALUE_FOREIGN(args[i]).data = va_arg (args2, gpointer);
1463 GTK_VALUE_FOREIGN(args[i]).notify =
1464 va_arg (args2, GtkDestroyNotify);
1466 case GTK_TYPE_CALLBACK:
1467 GTK_VALUE_CALLBACK(args[i]).marshal =
1468 va_arg (args2, GtkCallbackMarshal);
1469 GTK_VALUE_CALLBACK(args[i]).data = va_arg (args2, gpointer);
1470 GTK_VALUE_CALLBACK(args[i]).notify =
1471 va_arg (args2, GtkDestroyNotify);
1473 case GTK_TYPE_C_CALLBACK:
1474 GTK_VALUE_C_CALLBACK(args[i]).func = va_arg (args2, GtkFunction);
1475 GTK_VALUE_C_CALLBACK(args[i]).func_data =
1476 va_arg (args2, gpointer);
1479 GTK_VALUE_ARGS(args[i]).n_args = va_arg (args2, gint);
1480 GTK_VALUE_ARGS(args[i]).args = va_arg (args2, GtkArg*);
1482 case GTK_TYPE_OBJECT:
1483 GTK_VALUE_OBJECT(args[i]) = va_arg (args2, GtkObject*);
1484 g_assert (GTK_VALUE_OBJECT(args[i]) == NULL ||
1485 GTK_CHECK_TYPE (GTK_VALUE_OBJECT(args[i]),
1489 g_error ("unsupported type %s in args",
1490 gtk_type_name (args[i].type));