1 /* GTK - The GIMP Toolkit
2 * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Library General Public
6 * License as published by the Free Software Foundation; either
7 * version 2 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Library General Public License for more details.
14 * You should have received a copy of the GNU Library General Public
15 * License along with this library; if not, write to the
16 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
17 * Boston, MA 02111-1307, USA.
22 #include "gtkobject.h"
23 #include "gtksignal.h"
36 ARG_OBJECT_SIGNAL_AFTER
40 void gtk_object_init_type (void);
41 static void gtk_object_base_class_init (GtkObjectClass *klass);
42 static void gtk_object_class_init (GtkObjectClass *klass);
43 static void gtk_object_init (GtkObject *object);
44 static void gtk_object_set_arg (GtkObject *object,
47 static void gtk_object_get_arg (GtkObject *object,
50 static void gtk_object_shutdown (GtkObject *object);
51 static void gtk_object_real_destroy (GtkObject *object);
52 static void gtk_object_finalize (GtkObject *object);
53 static void gtk_object_notify_weaks (GtkObject *object);
55 static guint object_signals[LAST_SIGNAL] = { 0 };
57 static GHashTable *object_arg_info_ht = NULL;
59 static const gchar *user_data_key = "user_data";
60 static guint user_data_key_id = 0;
61 static const gchar *weakrefs_key = "gtk-weakrefs";
62 static guint weakrefs_key_id = 0;
66 static guint obj_count = 0;
67 static GHashTable *living_objs_ht = NULL;
69 gtk_object_debug_foreach (gpointer key, gpointer value, gpointer user_data)
73 object = (GtkObject*) value;
74 g_message ("[%p] %s\tref_count=%d%s%s",
76 gtk_type_name (GTK_OBJECT_TYPE (object)),
78 GTK_OBJECT_FLOATING (object) ? " (floating)" : "",
79 GTK_OBJECT_DESTROYED (object) ? " (destroyed)" : "");
82 gtk_object_debug (void)
85 g_hash_table_foreach (living_objs_ht, gtk_object_debug_foreach, NULL);
87 g_message ("living objects count = %d", obj_count);
89 #endif /* G_ENABLE_DEBUG */
91 /****************************************************
92 * GtkObject type, class and instance initialization
94 ****************************************************/
97 gtk_object_init_type (void)
99 GtkType object_type = 0;
100 GtkTypeInfo object_info =
104 sizeof (GtkObjectClass),
105 (GtkClassInitFunc) gtk_object_class_init,
106 (GtkObjectInitFunc) gtk_object_init,
107 /* reserved_1 */ NULL,
108 /* reserved_2 */ NULL,
109 (GtkClassInitFunc) gtk_object_base_class_init,
112 object_type = gtk_type_unique (0, &object_info);
113 g_assert (object_type == GTK_TYPE_OBJECT);
115 #ifdef G_ENABLE_DEBUG
116 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
117 g_atexit (gtk_object_debug);
118 #endif /* G_ENABLE_DEBUG */
122 gtk_object_get_type (void)
124 return GTK_TYPE_OBJECT;
128 gtk_object_base_class_init (GtkObjectClass *class)
130 /* reset instance specific fields that don't get inherited */
131 class->signals = NULL;
135 /* reset instance specifc methods that don't get inherited */
136 class->get_arg = NULL;
137 class->set_arg = NULL;
141 gtk_object_class_init (GtkObjectClass *class)
143 gtk_object_add_arg_type ("GtkObject::user_data",
147 gtk_object_add_arg_type ("GtkObject::signal",
151 gtk_object_add_arg_type ("GtkObject::signal_after",
155 gtk_object_add_arg_type ("GtkObject::object_signal",
159 gtk_object_add_arg_type ("GtkObject::object_signal_after",
162 ARG_OBJECT_SIGNAL_AFTER);
164 object_signals[DESTROY] =
165 gtk_signal_new ("destroy",
168 GTK_SIGNAL_OFFSET (GtkObjectClass, destroy),
169 gtk_marshal_NONE__NONE,
172 gtk_object_class_add_signals (class, object_signals, LAST_SIGNAL);
174 class->get_arg = gtk_object_get_arg;
175 class->set_arg = gtk_object_set_arg;
176 class->shutdown = gtk_object_shutdown;
177 class->destroy = gtk_object_real_destroy;
178 class->finalize = gtk_object_finalize;
182 gtk_object_init (GtkObject *object)
184 GTK_OBJECT_FLAGS (object) = GTK_FLOATING;
186 object->ref_count = 1;
187 g_datalist_init (&object->object_data);
189 #ifdef G_ENABLE_DEBUG
190 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
195 living_objs_ht = g_hash_table_new (g_direct_hash, NULL);
197 g_hash_table_insert (living_objs_ht, object, object);
199 #endif /* G_ENABLE_DEBUG */
202 /********************************************
203 * Functions to end a GtkObject's life time
205 ********************************************/
207 gtk_object_destroy (GtkObject *object)
209 g_return_if_fail (object != NULL);
210 g_return_if_fail (GTK_IS_OBJECT (object));
212 if (!GTK_OBJECT_DESTROYED (object))
214 /* we will hold a reference on the object in this place, so
215 * to ease all classes shutdown and destroy implementations.
216 * i.e. they don't have to bother about referencing at all.
218 gtk_object_ref (object);
219 object->klass->shutdown (object);
220 gtk_object_unref (object);
225 gtk_object_shutdown (GtkObject *object)
227 GTK_OBJECT_SET_FLAGS (object, GTK_DESTROYED);
228 gtk_signal_emit (object, object_signals[DESTROY]);
232 gtk_object_real_destroy (GtkObject *object)
234 if (GTK_OBJECT_CONNECTED (object))
235 gtk_signal_handlers_destroy (object);
239 gtk_object_finalize (GtkObject *object)
241 gtk_object_notify_weaks (object);
243 g_datalist_clear (&object->object_data);
245 gtk_type_free (GTK_OBJECT_TYPE (object), object);
248 /*****************************************
249 * GtkObject argument handlers
251 *****************************************/
254 gtk_object_set_arg (GtkObject *object,
265 gtk_object_set_user_data (object, GTK_VALUE_POINTER (*arg));
267 case ARG_OBJECT_SIGNAL_AFTER:
269 case ARG_OBJECT_SIGNAL:
271 case ARG_SIGNAL_AFTER:
275 arg_name = gtk_arg_name_strip_type (arg->name);
277 arg_name[n] == ':' &&
278 arg_name[n + 1] == ':' &&
279 arg_name[n + 2] != 0)
281 gtk_signal_connect_full (object,
283 GTK_VALUE_SIGNAL (*arg).f, NULL,
284 GTK_VALUE_SIGNAL (*arg).d,
286 (arg_id == ARG_OBJECT_SIGNAL ||
287 arg_id == ARG_OBJECT_SIGNAL_AFTER),
288 (arg_id == ARG_OBJECT_SIGNAL_AFTER ||
289 arg_id == ARG_SIGNAL_AFTER));
292 g_warning ("gtk_object_set_arg(): invalid signal argument: \"%s\"\n", arg->name);
300 gtk_object_get_arg (GtkObject *object,
307 GTK_VALUE_POINTER (*arg) = gtk_object_get_user_data (object);
310 case ARG_OBJECT_SIGNAL:
312 arg->type = GTK_TYPE_INVALID;
317 /*****************************************
318 * gtk_object_class_add_signals:
323 *****************************************/
326 gtk_object_class_add_signals (GtkObjectClass *class,
330 g_return_if_fail (GTK_IS_OBJECT_CLASS (class));
333 g_return_if_fail (signals != NULL);
335 class->signals = g_renew (guint, class->signals, class->nsignals + nsignals);
336 memcpy (class->signals + class->nsignals, signals, nsignals * sizeof (guint));
337 class->nsignals += nsignals;
341 gtk_object_class_add_user_signal (GtkObjectClass *class,
343 GtkSignalMarshaller marshaller,
353 g_return_val_if_fail (class != NULL, 0);
357 params = g_new (GtkType, nparams);
359 va_start (args, nparams);
361 for (i = 0; i < nparams; i++)
362 params[i] = va_arg (args, GtkType);
369 signal_id = gtk_signal_newv (name,
381 gtk_object_class_add_signals (class, &signal_id, 1);
387 gtk_object_class_user_signal_new (GtkObjectClass *class,
389 GtkSignalRunType signal_flags,
390 GtkSignalMarshaller marshaller,
400 g_return_val_if_fail (class != NULL, 0);
404 params = g_new (GtkType, nparams);
406 va_start (args, nparams);
408 for (i = 0; i < nparams; i++)
409 params[i] = va_arg (args, GtkType);
416 signal_id = gtk_signal_newv (name,
428 gtk_object_class_add_signals (class, &signal_id, 1);
434 gtk_object_class_user_signal_newv (GtkObjectClass *class,
436 GtkSignalRunType signal_flags,
437 GtkSignalMarshaller marshaller,
444 g_return_val_if_fail (class != NULL, 0);
447 g_return_val_if_fail (params != NULL, 0);
449 signal_id = gtk_signal_newv (name,
459 gtk_object_class_add_signals (class, &signal_id, 1);
464 /*****************************************
470 *****************************************/
473 gtk_object_sink (GtkObject *object)
475 g_return_if_fail (object != NULL);
476 g_return_if_fail (GTK_IS_OBJECT (object));
478 if (GTK_OBJECT_FLOATING (object))
480 GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING);
481 gtk_object_unref (object);
485 /*****************************************
488 * Weak refs are very similar to the old "destroy" signal. They allow
489 * one to register a callback that is called when the weakly
490 * referenced object is finalized.
492 * They are not implemented as a signal because they really are
493 * special and need to be used with great care. Unlike signals, which
494 * should be able to execute any code whatsoever.
496 * A weakref callback is not allowed to retain a reference to the
497 * object. Object data keys may be retrieved in a weak reference
500 * A weakref callback is called at most once.
502 *****************************************/
504 typedef struct _GtkWeakRef GtkWeakRef;
509 GtkDestroyNotify notify;
514 gtk_object_weakref (GtkObject *object,
515 GtkDestroyNotify notify,
520 g_return_if_fail (object != NULL);
521 g_return_if_fail (notify != NULL);
522 g_return_if_fail (GTK_IS_OBJECT (object));
524 if (!weakrefs_key_id)
525 weakrefs_key_id = g_quark_from_static_string (weakrefs_key);
527 weak = g_new (GtkWeakRef, 1);
528 weak->next = gtk_object_get_data_by_id (object, weakrefs_key_id);
529 weak->notify = notify;
531 gtk_object_set_data_by_id (object, weakrefs_key_id, weak);
535 gtk_object_weakunref (GtkObject *object,
536 GtkDestroyNotify notify,
539 GtkWeakRef *weaks, *w, **wp;
541 g_return_if_fail (object != NULL);
542 g_return_if_fail (GTK_IS_OBJECT (object));
544 if (!weakrefs_key_id)
547 weaks = gtk_object_get_data_by_id (object, weakrefs_key_id);
548 for (wp = &weaks; *wp; wp = &(*wp)->next)
551 if (w->notify == notify && w->data == data)
554 gtk_object_set_data_by_id (object, weakrefs_key_id, w->next);
564 gtk_object_notify_weaks (GtkObject *object)
570 w1 = gtk_object_get_data_by_id (object, weakrefs_key_id);
574 w1->notify (w1->data);
582 /****************************************************
583 * GtkObject argument mechanism and object creation
585 ****************************************************/
588 gtk_object_new (GtkType object_type,
589 const gchar *first_arg_name,
594 GSList *arg_list = NULL;
595 GSList *info_list = NULL;
598 g_return_val_if_fail (GTK_FUNDAMENTAL_TYPE (object_type) == GTK_TYPE_OBJECT, NULL);
600 object = gtk_type_new (object_type);
602 va_start (var_args, first_arg_name);
603 error = gtk_object_args_collect (GTK_OBJECT_TYPE (object),
612 g_warning ("gtk_object_new(): %s", error);
620 slist_arg = arg_list;
621 slist_info = info_list;
624 gtk_object_arg_set (object, slist_arg->data, slist_info->data);
625 slist_arg = slist_arg->next;
626 slist_info = slist_info->next;
628 gtk_args_collect_cleanup (arg_list, info_list);
635 gtk_object_newv (GtkType object_type,
642 g_return_val_if_fail (GTK_FUNDAMENTAL_TYPE (object_type) == GTK_TYPE_OBJECT, NULL);
644 g_return_val_if_fail (args != NULL, NULL);
646 object = gtk_type_new (object_type);
648 for (max_args = args + n_args; args < max_args; args++)
649 gtk_object_arg_set (object, args, NULL);
655 gtk_object_setv (GtkObject *object,
661 g_return_if_fail (object != NULL);
662 g_return_if_fail (GTK_IS_OBJECT (object));
664 g_return_if_fail (args != NULL);
666 for (max_args = args + n_args; args < max_args; args++)
667 gtk_object_arg_set (object, args, NULL);
671 gtk_object_getv (GtkObject *object,
677 g_return_if_fail (object != NULL);
678 g_return_if_fail (GTK_IS_OBJECT (object));
680 g_return_if_fail (args != NULL);
682 for (max_args = args + n_args; args < max_args; args++)
683 gtk_object_arg_get (object, args, NULL);
687 gtk_object_set (GtkObject *object,
688 const gchar *first_arg_name,
692 GSList *arg_list = NULL;
693 GSList *info_list = NULL;
696 g_return_if_fail (object != NULL);
697 g_return_if_fail (GTK_IS_OBJECT (object));
699 va_start (var_args, first_arg_name);
700 error = gtk_object_args_collect (GTK_OBJECT_TYPE (object),
709 g_warning ("gtk_object_set(): %s", error);
717 slist_arg = arg_list;
718 slist_info = info_list;
721 gtk_object_arg_set (object, slist_arg->data, slist_info->data);
722 slist_arg = slist_arg->next;
723 slist_info = slist_info->next;
725 gtk_args_collect_cleanup (arg_list, info_list);
730 gtk_object_arg_set (GtkObject *object,
734 GtkObjectClass *oclass;
736 g_return_if_fail (object != NULL);
737 g_return_if_fail (GTK_IS_OBJECT (object));
738 g_return_if_fail (arg != NULL);
744 error = gtk_arg_get_info (GTK_OBJECT_TYPE (object),
750 g_warning ("gtk_object_arg_set(): %s", error);
756 if (! (info->arg_flags & GTK_ARG_WRITABLE))
758 g_warning ("gtk_object_arg_set(): argument \"%s\" is not writable",
762 if (info->type != arg->type)
764 g_warning ("gtk_object_arg_set(): argument \"%s\" has invalid type `%s'",
766 gtk_type_name (arg->type));
770 oclass = gtk_type_class (info->class_type);
771 g_assert (oclass->set_arg != NULL);
772 oclass->set_arg (object, arg, info->arg_id);
776 gtk_object_arg_get (GtkObject *object,
780 GtkObjectClass *oclass;
782 g_return_if_fail (object != NULL);
783 g_return_if_fail (GTK_IS_OBJECT (object));
784 g_return_if_fail (arg != NULL);
790 error = gtk_arg_get_info (GTK_OBJECT_TYPE (object),
796 g_warning ("gtk_object_arg_get(): %s", error);
798 arg->type = GTK_TYPE_INVALID;
803 if (! (info->arg_flags & GTK_ARG_READABLE))
805 g_warning ("gtk_object_arg_get(): argument \"%s\" is not readable",
807 arg->type = GTK_TYPE_INVALID;
811 oclass = gtk_type_class (info->class_type);
812 g_assert (oclass->get_arg != NULL);
813 arg->type = info->type;
814 oclass->get_arg (object, arg, info->arg_id);
818 gtk_object_add_arg_type (const char *arg_name,
823 g_return_if_fail (arg_name != NULL);
824 g_return_if_fail (arg_type > GTK_TYPE_NONE);
825 g_return_if_fail (arg_id > 0);
826 g_return_if_fail ((arg_flags & GTK_ARG_CHILD_ARG) == 0);
827 if (arg_flags & GTK_ARG_CONSTRUCT)
828 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) == GTK_ARG_READWRITE);
830 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) != 0);
832 if (!object_arg_info_ht)
833 object_arg_info_ht = g_hash_table_new (gtk_arg_info_hash,
836 gtk_arg_type_new_static (GTK_TYPE_OBJECT,
838 GTK_STRUCT_OFFSET (GtkObjectClass, n_args),
846 gtk_object_args_collect (GtkType object_type,
848 GSList **info_list_p,
849 const gchar *first_arg_name,
852 return gtk_args_collect (object_type,
861 gtk_object_arg_get_info (GtkType object_type,
862 const gchar *arg_name,
865 return gtk_arg_get_info (object_type,
872 gtk_object_query_args (GtkType class_type,
876 g_return_val_if_fail (n_args != NULL, NULL);
878 g_return_val_if_fail (GTK_FUNDAMENTAL_TYPE (class_type) == GTK_TYPE_OBJECT, NULL);
880 return gtk_args_query (class_type, object_arg_info_ht, arg_flags, n_args);
883 /*****************************************
884 * GtkObject object_data mechanism
886 *****************************************/
889 gtk_object_set_data_by_id (GtkObject *object,
893 g_return_if_fail (object != NULL);
894 g_return_if_fail (GTK_IS_OBJECT (object));
896 g_datalist_id_set_data (&object->object_data, data_id, data);
900 gtk_object_set_data (GtkObject *object,
904 g_return_if_fail (object != NULL);
905 g_return_if_fail (GTK_IS_OBJECT (object));
906 g_return_if_fail (key != NULL);
908 g_datalist_set_data (&object->object_data, key, data);
912 gtk_object_set_data_by_id_full (GtkObject *object,
915 GtkDestroyNotify destroy)
917 g_return_if_fail (object != NULL);
918 g_return_if_fail (GTK_IS_OBJECT (object));
920 g_datalist_id_set_data_full (&object->object_data, data_id, data, destroy);
924 gtk_object_set_data_full (GtkObject *object,
927 GtkDestroyNotify destroy)
929 g_return_if_fail (object != NULL);
930 g_return_if_fail (GTK_IS_OBJECT (object));
931 g_return_if_fail (key != NULL);
933 g_datalist_set_data_full (&object->object_data, key, data, destroy);
937 gtk_object_get_data_by_id (GtkObject *object,
940 g_return_val_if_fail (object != NULL, NULL);
941 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
943 return g_datalist_id_get_data (&object->object_data, data_id);
947 gtk_object_get_data (GtkObject *object,
950 g_return_val_if_fail (object != NULL, NULL);
951 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
952 g_return_val_if_fail (key != NULL, NULL);
954 return g_datalist_get_data (&object->object_data, key);
958 gtk_object_remove_data_by_id (GtkObject *object,
961 g_return_if_fail (object != NULL);
962 g_return_if_fail (GTK_IS_OBJECT (object));
964 g_datalist_id_remove_data (&object->object_data, data_id);
968 gtk_object_remove_data (GtkObject *object,
971 g_return_if_fail (object != NULL);
972 g_return_if_fail (GTK_IS_OBJECT (object));
973 g_return_if_fail (key != NULL);
975 g_datalist_remove_data (&object->object_data, key);
979 gtk_object_remove_no_notify_by_id (GtkObject *object,
982 g_return_if_fail (object != NULL);
983 g_return_if_fail (GTK_IS_OBJECT (object));
985 g_datalist_id_remove_no_notify (&object->object_data, key_id);
989 gtk_object_set_data_destroy (GtkObject *object,
992 g_return_if_fail (object != NULL);
993 g_return_if_fail (GTK_IS_OBJECT (object));
994 g_return_if_fail (key != NULL);
996 g_datalist_remove_no_notify (&object->object_data, key);
1000 gtk_object_set_user_data (GtkObject *object,
1003 g_return_if_fail (object != NULL);
1004 g_return_if_fail (GTK_IS_OBJECT (object));
1006 if (!user_data_key_id)
1007 user_data_key_id = g_quark_from_static_string (user_data_key);
1009 g_datalist_id_set_data (&object->object_data, user_data_key_id, data);
1013 gtk_object_get_user_data (GtkObject *object)
1015 g_return_val_if_fail (object != NULL, NULL);
1016 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1018 return g_datalist_id_get_data (&object->object_data, user_data_key_id);
1021 /*******************************************
1022 * GtkObject referencing and unreferencing
1024 *******************************************/
1026 #undef gtk_object_ref
1027 #undef gtk_object_unref
1030 gtk_object_ref (GtkObject *object)
1032 g_return_if_fail (object != NULL);
1033 g_return_if_fail (GTK_IS_OBJECT (object));
1034 g_return_if_fail (object->ref_count > 0);
1036 object->ref_count += 1;
1040 gtk_object_unref (GtkObject *object)
1042 g_return_if_fail (object != NULL);
1043 g_return_if_fail (GTK_IS_OBJECT (object));
1044 g_return_if_fail (object->ref_count > 0);
1046 if (object->ref_count == 1)
1048 gtk_object_destroy (object);
1050 g_return_if_fail (object->ref_count > 0);
1053 object->ref_count -= 1;
1055 if (object->ref_count == 0)
1057 #ifdef G_ENABLE_DEBUG
1058 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1060 g_assert (g_hash_table_lookup (living_objs_ht, object) == object);
1061 g_hash_table_remove (living_objs_ht, object);
1064 #endif /* G_ENABLE_DEBUG */
1065 object->klass->finalize (object);
1069 static GtkObject *gtk_trace_object = NULL;
1071 gtk_trace_referencing (GtkObject *object,
1077 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1079 gboolean exists = TRUE;
1081 g_return_if_fail (object != NULL);
1082 g_return_if_fail (GTK_IS_OBJECT (object));
1084 #ifdef G_ENABLE_DEBUG
1085 exists = g_hash_table_lookup (living_objs_ht, object) != NULL;
1086 #endif /* G_ENABLE_DEBUG */
1089 (object == gtk_trace_object ||
1090 gtk_trace_object == (void*)42))
1091 fprintf (stdout, "trace: object_%s: (%s:%p)->ref_count=%d %s (%s:%d)\n",
1092 do_ref ? "ref" : "unref",
1093 gtk_type_name (GTK_OBJECT_TYPE (object)),
1096 do_ref ? "+ 1" : "- 1",
1100 fprintf (stdout, "trace: object_%s(%p): no such object! (%s:%d)\n",
1101 do_ref ? "ref" : "unref",
1108 gtk_object_ref (object);
1110 gtk_object_unref (object);