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 GtkObjectClass *klass);
45 static void gtk_object_set_arg (GtkObject *object,
48 static void gtk_object_get_arg (GtkObject *object,
51 static void gtk_object_shutdown (GtkObject *object);
52 static void gtk_object_real_destroy (GtkObject *object);
53 static void gtk_object_finalize (GtkObject *object);
54 static void gtk_object_notify_weaks (GtkObject *object);
56 static guint object_signals[LAST_SIGNAL] = { 0 };
58 static GHashTable *object_arg_info_ht = NULL;
60 static GQuark quark_user_data = 0;
61 static GQuark quark_weakrefs = 0;
62 static GQuark quark_carg_history = 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;
134 class->construct_args = NULL;
136 /* reset instance specifc methods that don't get inherited */
137 class->get_arg = NULL;
138 class->set_arg = NULL;
142 gtk_object_class_init (GtkObjectClass *class)
144 quark_carg_history = g_quark_from_static_string ("gtk-construct-arg-history");
146 gtk_object_add_arg_type ("GtkObject::user_data",
150 gtk_object_add_arg_type ("GtkObject::signal",
154 gtk_object_add_arg_type ("GtkObject::signal_after",
158 gtk_object_add_arg_type ("GtkObject::object_signal",
162 gtk_object_add_arg_type ("GtkObject::object_signal_after",
165 ARG_OBJECT_SIGNAL_AFTER);
167 object_signals[DESTROY] =
168 gtk_signal_new ("destroy",
171 GTK_SIGNAL_OFFSET (GtkObjectClass, destroy),
172 gtk_marshal_NONE__NONE,
175 gtk_object_class_add_signals (class, object_signals, LAST_SIGNAL);
177 class->get_arg = gtk_object_get_arg;
178 class->set_arg = gtk_object_set_arg;
179 class->shutdown = gtk_object_shutdown;
180 class->destroy = gtk_object_real_destroy;
181 class->finalize = gtk_object_finalize;
185 gtk_object_init (GtkObject *object,
186 GtkObjectClass *klass)
188 gboolean needs_construction = FALSE;
190 GTK_OBJECT_FLAGS (object) = GTK_FLOATING;
193 needs_construction |= klass->construct_args != NULL;
194 klass = gtk_type_parent_class (klass->type);
196 while (klass && !needs_construction);
197 if (!needs_construction)
198 GTK_OBJECT_FLAGS (object) |= GTK_CONSTRUCTED;
200 object->ref_count = 1;
201 g_datalist_init (&object->object_data);
203 #ifdef G_ENABLE_DEBUG
204 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
209 living_objs_ht = g_hash_table_new (g_direct_hash, NULL);
211 g_hash_table_insert (living_objs_ht, object, object);
213 #endif /* G_ENABLE_DEBUG */
216 /********************************************
217 * Functions to end a GtkObject's life time
219 ********************************************/
221 gtk_object_destroy (GtkObject *object)
223 g_return_if_fail (object != NULL);
224 g_return_if_fail (GTK_IS_OBJECT (object));
225 g_return_if_fail (GTK_OBJECT_CONSTRUCTED (object));
227 if (!GTK_OBJECT_DESTROYED (object))
229 /* we will hold a reference on the object in this place, so
230 * to ease all classes shutdown and destroy implementations.
231 * i.e. they don't have to bother about referencing at all.
233 gtk_object_ref (object);
234 object->klass->shutdown (object);
235 gtk_object_unref (object);
240 gtk_object_shutdown (GtkObject *object)
242 GTK_OBJECT_SET_FLAGS (object, GTK_DESTROYED);
243 gtk_signal_emit (object, object_signals[DESTROY]);
247 gtk_object_real_destroy (GtkObject *object)
249 if (GTK_OBJECT_CONNECTED (object))
250 gtk_signal_handlers_destroy (object);
254 gtk_object_finalize (GtkObject *object)
256 gtk_object_notify_weaks (object);
258 g_datalist_clear (&object->object_data);
260 gtk_type_free (GTK_OBJECT_TYPE (object), object);
263 /*****************************************
264 * GtkObject argument handlers
266 *****************************************/
269 gtk_object_set_arg (GtkObject *object,
280 gtk_object_set_user_data (object, GTK_VALUE_POINTER (*arg));
282 case ARG_OBJECT_SIGNAL_AFTER:
284 case ARG_OBJECT_SIGNAL:
286 case ARG_SIGNAL_AFTER:
290 arg_name = gtk_arg_name_strip_type (arg->name);
292 arg_name[n] == ':' &&
293 arg_name[n + 1] == ':' &&
294 arg_name[n + 2] != 0)
296 gtk_signal_connect_full (object,
298 GTK_VALUE_SIGNAL (*arg).f, NULL,
299 GTK_VALUE_SIGNAL (*arg).d,
301 (arg_id == ARG_OBJECT_SIGNAL ||
302 arg_id == ARG_OBJECT_SIGNAL_AFTER),
303 (arg_id == ARG_OBJECT_SIGNAL_AFTER ||
304 arg_id == ARG_SIGNAL_AFTER));
307 g_warning ("gtk_object_set_arg(): invalid signal argument: \"%s\"\n", arg->name);
315 gtk_object_get_arg (GtkObject *object,
322 GTK_VALUE_POINTER (*arg) = gtk_object_get_user_data (object);
325 case ARG_OBJECT_SIGNAL:
327 arg->type = GTK_TYPE_INVALID;
332 /*****************************************
333 * gtk_object_class_add_signals:
338 *****************************************/
341 gtk_object_class_add_signals (GtkObjectClass *class,
345 g_return_if_fail (GTK_IS_OBJECT_CLASS (class));
348 g_return_if_fail (signals != NULL);
350 class->signals = g_renew (guint, class->signals, class->nsignals + nsignals);
351 memcpy (class->signals + class->nsignals, signals, nsignals * sizeof (guint));
352 class->nsignals += nsignals;
356 gtk_object_class_add_user_signal (GtkObjectClass *class,
358 GtkSignalMarshaller marshaller,
368 g_return_val_if_fail (class != NULL, 0);
370 g_message ("gtk_object_class_add_user_signal() is deprecated");
374 params = g_new (GtkType, nparams);
376 va_start (args, nparams);
378 for (i = 0; i < nparams; i++)
379 params[i] = va_arg (args, GtkType);
386 signal_id = gtk_signal_newv (name,
398 gtk_object_class_add_signals (class, &signal_id, 1);
404 gtk_object_class_user_signal_new (GtkObjectClass *class,
406 GtkSignalRunType signal_flags,
407 GtkSignalMarshaller marshaller,
417 g_return_val_if_fail (class != NULL, 0);
421 params = g_new (GtkType, nparams);
423 va_start (args, nparams);
425 for (i = 0; i < nparams; i++)
426 params[i] = va_arg (args, GtkType);
433 signal_id = gtk_signal_newv (name,
445 gtk_object_class_add_signals (class, &signal_id, 1);
451 gtk_object_class_user_signal_newv (GtkObjectClass *class,
453 GtkSignalRunType signal_flags,
454 GtkSignalMarshaller marshaller,
461 g_return_val_if_fail (class != NULL, 0);
464 g_return_val_if_fail (params != NULL, 0);
466 signal_id = gtk_signal_newv (name,
476 gtk_object_class_add_signals (class, &signal_id, 1);
481 /*****************************************
487 *****************************************/
490 gtk_object_sink (GtkObject *object)
492 g_return_if_fail (object != NULL);
493 g_return_if_fail (GTK_IS_OBJECT (object));
495 if (GTK_OBJECT_FLOATING (object))
497 GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING);
498 gtk_object_unref (object);
502 /*****************************************
505 * Weak refs are very similar to the old "destroy" signal. They allow
506 * one to register a callback that is called when the weakly
507 * referenced object is finalized.
509 * They are not implemented as a signal because they really are
510 * special and need to be used with great care. Unlike signals, which
511 * should be able to execute any code whatsoever.
513 * A weakref callback is not allowed to retain a reference to the
514 * object. Object data keys may be retrieved in a weak reference
517 * A weakref callback is called at most once.
519 *****************************************/
521 typedef struct _GtkWeakRef GtkWeakRef;
526 GtkDestroyNotify notify;
531 gtk_object_weakref (GtkObject *object,
532 GtkDestroyNotify notify,
537 g_return_if_fail (object != NULL);
538 g_return_if_fail (notify != NULL);
539 g_return_if_fail (GTK_IS_OBJECT (object));
542 quark_weakrefs = g_quark_from_static_string ("gtk-weakrefs");
544 weak = g_new (GtkWeakRef, 1);
545 weak->next = gtk_object_get_data_by_id (object, quark_weakrefs);
546 weak->notify = notify;
548 gtk_object_set_data_by_id (object, quark_weakrefs, weak);
552 gtk_object_weakunref (GtkObject *object,
553 GtkDestroyNotify notify,
556 GtkWeakRef *weaks, *w, **wp;
558 g_return_if_fail (object != NULL);
559 g_return_if_fail (GTK_IS_OBJECT (object));
564 weaks = gtk_object_get_data_by_id (object, quark_weakrefs);
565 for (wp = &weaks; *wp; wp = &(*wp)->next)
568 if (w->notify == notify && w->data == data)
571 gtk_object_set_data_by_id (object, quark_weakrefs, w->next);
581 gtk_object_notify_weaks (GtkObject *object)
587 w1 = gtk_object_get_data_by_id (object, quark_weakrefs);
591 w1->notify (w1->data);
599 /****************************************************
600 * GtkObject argument mechanism and object creation
602 ****************************************************/
605 gtk_object_new (GtkType object_type,
606 const gchar *first_arg_name,
611 GSList *arg_list = NULL;
612 GSList *info_list = NULL;
615 g_return_val_if_fail (GTK_FUNDAMENTAL_TYPE (object_type) == GTK_TYPE_OBJECT, NULL);
617 object = gtk_type_new (object_type);
619 va_start (var_args, first_arg_name);
620 error = gtk_object_args_collect (GTK_OBJECT_TYPE (object),
629 g_warning ("gtk_object_new(): %s", error);
637 slist_arg = arg_list;
638 slist_info = info_list;
641 gtk_object_arg_set (object, slist_arg->data, slist_info->data);
642 slist_arg = slist_arg->next;
643 slist_info = slist_info->next;
645 gtk_args_collect_cleanup (arg_list, info_list);
648 if (!GTK_OBJECT_CONSTRUCTED (object))
649 gtk_object_default_construct (object);
655 gtk_object_newv (GtkType object_type,
662 g_return_val_if_fail (GTK_FUNDAMENTAL_TYPE (object_type) == GTK_TYPE_OBJECT, NULL);
664 g_return_val_if_fail (args != NULL, NULL);
666 object = gtk_type_new (object_type);
668 for (max_args = args + n_args; args < max_args; args++)
669 gtk_object_arg_set (object, args, NULL);
671 if (!GTK_OBJECT_CONSTRUCTED (object))
672 gtk_object_default_construct (object);
678 gtk_object_setv (GtkObject *object,
684 g_return_if_fail (object != NULL);
685 g_return_if_fail (GTK_IS_OBJECT (object));
687 g_return_if_fail (args != NULL);
689 for (max_args = args + n_args; args < max_args; args++)
690 gtk_object_arg_set (object, args, NULL);
694 gtk_object_getv (GtkObject *object,
700 g_return_if_fail (object != NULL);
701 g_return_if_fail (GTK_IS_OBJECT (object));
703 g_return_if_fail (args != NULL);
705 for (max_args = args + n_args; args < max_args; args++)
706 gtk_object_arg_get (object, args, NULL);
710 gtk_object_set (GtkObject *object,
711 const gchar *first_arg_name,
715 GSList *arg_list = NULL;
716 GSList *info_list = NULL;
719 g_return_if_fail (object != NULL);
720 g_return_if_fail (GTK_IS_OBJECT (object));
722 va_start (var_args, first_arg_name);
723 error = gtk_object_args_collect (GTK_OBJECT_TYPE (object),
732 g_warning ("gtk_object_set(): %s", error);
740 slist_arg = arg_list;
741 slist_info = info_list;
744 gtk_object_arg_set (object, slist_arg->data, slist_info->data);
745 slist_arg = slist_arg->next;
746 slist_info = slist_info->next;
748 gtk_args_collect_cleanup (arg_list, info_list);
753 gtk_object_arg_set (GtkObject *object,
757 GtkObjectClass *oclass;
759 g_return_if_fail (object != NULL);
760 g_return_if_fail (GTK_IS_OBJECT (object));
761 g_return_if_fail (arg != NULL);
767 error = gtk_arg_get_info (GTK_OBJECT_TYPE (object),
773 g_warning ("gtk_object_arg_set(): %s", error);
779 if (info->arg_flags & GTK_ARG_CONSTRUCT_ONLY &&
780 GTK_OBJECT_CONSTRUCTED (object))
782 g_warning ("gtk_object_arg_set(): cannot set argument \"%s\" for constructed object",
786 if (!(info->arg_flags & GTK_ARG_WRITABLE))
788 g_warning ("gtk_object_arg_set(): argument \"%s\" is not writable",
792 if (info->type != arg->type)
794 g_warning ("gtk_object_arg_set(): argument \"%s\" has invalid type `%s'",
796 gtk_type_name (arg->type));
800 oclass = gtk_type_class (info->class_type);
801 g_assert (oclass->set_arg != NULL);
802 oclass->set_arg (object, arg, info->arg_id);
803 if (!GTK_OBJECT_CONSTRUCTED (object) &&
804 (info->arg_flags & GTK_ARG_CONSTRUCT_ONLY ||
805 info->arg_flags & GTK_ARG_CONSTRUCT))
809 slist = gtk_object_get_data_by_id (object, quark_carg_history);
810 gtk_object_set_data_by_id (object,
812 g_slist_prepend (slist, info));
817 gtk_object_arg_get (GtkObject *object,
821 GtkObjectClass *oclass;
823 g_return_if_fail (object != NULL);
824 g_return_if_fail (GTK_IS_OBJECT (object));
825 g_return_if_fail (arg != NULL);
831 error = gtk_arg_get_info (GTK_OBJECT_TYPE (object),
837 g_warning ("gtk_object_arg_get(): %s", error);
839 arg->type = GTK_TYPE_INVALID;
844 if (! (info->arg_flags & GTK_ARG_READABLE))
846 g_warning ("gtk_object_arg_get(): argument \"%s\" is not readable",
848 arg->type = GTK_TYPE_INVALID;
852 oclass = gtk_type_class (info->class_type);
853 g_assert (oclass->get_arg != NULL);
854 arg->type = info->type;
855 oclass->get_arg (object, arg, info->arg_id);
859 gtk_object_default_construct (GtkObject *object)
863 g_return_if_fail (object != NULL);
864 g_return_if_fail (GTK_IS_OBJECT (object));
866 if (!GTK_OBJECT_CONSTRUCTED (object))
868 for (slist = object->klass->construct_args;
869 slist && !GTK_OBJECT_CONSTRUCTED (object);
876 history = gtk_object_get_data_by_id (object, quark_carg_history);
877 if (!g_slist_find (history, info))
881 /* default application */
882 arg.type = info->type;
883 arg.name = info->name;
884 switch (gtk_type_get_varargs_type (arg.type))
887 GTK_VALUE_FLOAT (arg) = 0.0;
889 case GTK_TYPE_DOUBLE:
890 GTK_VALUE_DOUBLE (arg) = 0.0;
893 case GTK_TYPE_STRING:
894 case GTK_TYPE_POINTER:
895 case GTK_TYPE_OBJECT:
896 GTK_VALUE_POINTER (arg) = NULL;
899 memset (&arg.d, 0, sizeof (arg.d));
902 gtk_object_arg_set (object, &arg, info);
906 if (!GTK_OBJECT_CONSTRUCTED (object))
907 gtk_object_constructed (object);
912 gtk_object_constructed (GtkObject *object)
914 g_return_if_fail (object != NULL);
915 g_return_if_fail (GTK_IS_OBJECT (object));
916 g_return_if_fail (GTK_OBJECT_CONSTRUCTED (object) == FALSE);
918 g_slist_free (gtk_object_get_data_by_id (object, quark_carg_history));
919 gtk_object_set_data_by_id (object, quark_carg_history, NULL);
920 GTK_OBJECT_FLAGS (object) |= GTK_CONSTRUCTED;
924 gtk_object_add_arg_type (const char *arg_name,
931 g_return_if_fail (arg_name != NULL);
932 g_return_if_fail (arg_type > GTK_TYPE_NONE);
933 g_return_if_fail (arg_id > 0);
934 g_return_if_fail ((arg_flags & GTK_ARG_CHILD_ARG) == 0);
935 if (arg_flags & GTK_ARG_CONSTRUCT)
936 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) == GTK_ARG_READWRITE);
938 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) != 0);
939 if (arg_flags & GTK_ARG_CONSTRUCT_ONLY)
940 g_return_if_fail ((arg_flags & GTK_ARG_WRITABLE) == GTK_ARG_WRITABLE);
942 if (!object_arg_info_ht)
943 object_arg_info_ht = g_hash_table_new (gtk_arg_info_hash,
946 info = gtk_arg_type_new_static (GTK_TYPE_OBJECT,
948 GTK_STRUCT_OFFSET (GtkObjectClass, n_args),
954 (info->arg_flags & GTK_ARG_CONSTRUCT ||
955 info->arg_flags & GTK_ARG_CONSTRUCT_ONLY))
957 GtkObjectClass *class;
959 class = gtk_type_class (info->class_type);
960 if (info->arg_flags & GTK_ARG_CONSTRUCT_ONLY)
961 class->construct_args = g_slist_prepend (class->construct_args, info);
963 class->construct_args = g_slist_append (class->construct_args, info);
968 gtk_object_args_collect (GtkType object_type,
970 GSList **info_list_p,
971 const gchar *first_arg_name,
974 return gtk_args_collect (object_type,
983 gtk_object_arg_get_info (GtkType object_type,
984 const gchar *arg_name,
987 return gtk_arg_get_info (object_type,
994 gtk_object_query_args (GtkType class_type,
998 g_return_val_if_fail (n_args != NULL, NULL);
1000 g_return_val_if_fail (GTK_FUNDAMENTAL_TYPE (class_type) == GTK_TYPE_OBJECT, NULL);
1002 return gtk_args_query (class_type, object_arg_info_ht, arg_flags, n_args);
1005 /*****************************************
1006 * GtkObject object_data mechanism
1008 *****************************************/
1011 gtk_object_set_data_by_id (GtkObject *object,
1015 g_return_if_fail (object != NULL);
1016 g_return_if_fail (GTK_IS_OBJECT (object));
1018 g_datalist_id_set_data (&object->object_data, data_id, data);
1022 gtk_object_set_data (GtkObject *object,
1026 g_return_if_fail (object != NULL);
1027 g_return_if_fail (GTK_IS_OBJECT (object));
1028 g_return_if_fail (key != NULL);
1030 g_datalist_set_data (&object->object_data, key, data);
1034 gtk_object_set_data_by_id_full (GtkObject *object,
1037 GtkDestroyNotify destroy)
1039 g_return_if_fail (object != NULL);
1040 g_return_if_fail (GTK_IS_OBJECT (object));
1042 g_datalist_id_set_data_full (&object->object_data, data_id, data, destroy);
1046 gtk_object_set_data_full (GtkObject *object,
1049 GtkDestroyNotify destroy)
1051 g_return_if_fail (object != NULL);
1052 g_return_if_fail (GTK_IS_OBJECT (object));
1053 g_return_if_fail (key != NULL);
1055 g_datalist_set_data_full (&object->object_data, key, data, destroy);
1059 gtk_object_get_data_by_id (GtkObject *object,
1062 g_return_val_if_fail (object != NULL, NULL);
1063 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1065 return g_datalist_id_get_data (&object->object_data, data_id);
1069 gtk_object_get_data (GtkObject *object,
1072 g_return_val_if_fail (object != NULL, NULL);
1073 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1074 g_return_val_if_fail (key != NULL, NULL);
1076 return g_datalist_get_data (&object->object_data, key);
1080 gtk_object_remove_data_by_id (GtkObject *object,
1083 g_return_if_fail (object != NULL);
1084 g_return_if_fail (GTK_IS_OBJECT (object));
1086 g_datalist_id_remove_data (&object->object_data, data_id);
1090 gtk_object_remove_data (GtkObject *object,
1093 g_return_if_fail (object != NULL);
1094 g_return_if_fail (GTK_IS_OBJECT (object));
1095 g_return_if_fail (key != NULL);
1097 g_datalist_remove_data (&object->object_data, key);
1101 gtk_object_remove_no_notify_by_id (GtkObject *object,
1104 g_return_if_fail (object != NULL);
1105 g_return_if_fail (GTK_IS_OBJECT (object));
1107 g_datalist_id_remove_no_notify (&object->object_data, key_id);
1111 gtk_object_remove_no_notify (GtkObject *object,
1114 g_return_if_fail (object != NULL);
1115 g_return_if_fail (GTK_IS_OBJECT (object));
1116 g_return_if_fail (key != NULL);
1118 g_datalist_remove_no_notify (&object->object_data, key);
1122 gtk_object_set_user_data (GtkObject *object,
1125 g_return_if_fail (object != NULL);
1126 g_return_if_fail (GTK_IS_OBJECT (object));
1128 if (!quark_user_data)
1129 quark_user_data = g_quark_from_static_string ("user_data");
1131 g_datalist_id_set_data (&object->object_data, quark_user_data, data);
1135 gtk_object_get_user_data (GtkObject *object)
1137 g_return_val_if_fail (object != NULL, NULL);
1138 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1140 return g_datalist_id_get_data (&object->object_data, quark_user_data);
1143 /*******************************************
1144 * GtkObject referencing and unreferencing
1146 *******************************************/
1148 #undef gtk_object_ref
1149 #undef gtk_object_unref
1152 gtk_object_ref (GtkObject *object)
1154 g_return_if_fail (object != NULL);
1155 g_return_if_fail (GTK_IS_OBJECT (object));
1156 g_return_if_fail (object->ref_count > 0);
1158 object->ref_count += 1;
1162 gtk_object_unref (GtkObject *object)
1164 g_return_if_fail (object != NULL);
1165 g_return_if_fail (GTK_IS_OBJECT (object));
1166 g_return_if_fail (object->ref_count > 0);
1168 if (object->ref_count == 1)
1170 gtk_object_destroy (object);
1172 g_return_if_fail (object->ref_count > 0);
1175 object->ref_count -= 1;
1177 if (object->ref_count == 0)
1179 #ifdef G_ENABLE_DEBUG
1180 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1182 g_assert (g_hash_table_lookup (living_objs_ht, object) == object);
1183 g_hash_table_remove (living_objs_ht, object);
1186 #endif /* G_ENABLE_DEBUG */
1187 object->klass->finalize (object);
1191 static GtkObject *gtk_trace_object = NULL;
1193 gtk_trace_referencing (GtkObject *object,
1199 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1201 gboolean exists = TRUE;
1203 g_return_if_fail (object != NULL);
1204 g_return_if_fail (GTK_IS_OBJECT (object));
1206 #ifdef G_ENABLE_DEBUG
1207 exists = g_hash_table_lookup (living_objs_ht, object) != NULL;
1208 #endif /* G_ENABLE_DEBUG */
1211 (object == gtk_trace_object ||
1212 gtk_trace_object == (void*)42))
1213 fprintf (stdout, "trace: object_%s: (%s:%p)->ref_count=%d %s (%s:%d)\n",
1214 do_ref ? "ref" : "unref",
1215 gtk_type_name (GTK_OBJECT_TYPE (object)),
1218 do_ref ? "+ 1" : "- 1",
1222 fprintf (stdout, "trace: object_%s(%p): no such object! (%s:%d)\n",
1223 do_ref ? "ref" : "unref",
1230 gtk_object_ref (object);
1232 gtk_object_unref (object);