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.
23 #include "gtkobject.h"
24 #include "gtksignal.h"
27 #define GTK_OBJECT_DATA_ID_BLOCK_SIZE (1024)
28 #define GTK_OBJECT_DATA_BLOCK_SIZE (1024)
40 ARG_OBJECT_SIGNAL_AFTER
44 typedef struct _GtkObjectData GtkObjectData;
50 GtkDestroyNotify destroy;
55 void gtk_object_init_type (void);
56 static void gtk_object_base_class_init (GtkObjectClass *klass);
57 static void gtk_object_class_init (GtkObjectClass *klass);
58 static void gtk_object_init (GtkObject *object);
59 static void gtk_object_set_arg (GtkObject *object,
62 static void gtk_object_get_arg (GtkObject *object,
65 static void gtk_object_shutdown (GtkObject *object);
66 static void gtk_object_real_destroy (GtkObject *object);
67 static void gtk_object_finalize (GtkObject *object);
68 static void gtk_object_notify_weaks (GtkObject *object);
70 static guint object_signals[LAST_SIGNAL] = { 0 };
72 static GHashTable *object_arg_info_ht = NULL;
74 static const gchar *user_data_key = "user_data";
75 static guint user_data_key_id = 0;
76 static const gchar *weakrefs_key = "gtk-weakrefs";
77 static guint weakrefs_key_id = 0;
79 static GtkObjectData *gtk_object_data_free_list = NULL;
81 #define GTK_OBJECT_DATA_DESTROY( odata ) { \
83 odata->destroy (odata->data); \
84 odata->next = gtk_object_data_free_list; \
85 gtk_object_data_free_list = odata; \
90 static guint obj_count = 0;
91 static GHashTable *living_objs_ht = NULL;
93 gtk_object_debug_foreach (gpointer key, gpointer value, gpointer user_data)
97 object = (GtkObject*) value;
98 g_message ("[%p] %s\tref_count=%d%s%s",
100 gtk_type_name (GTK_OBJECT_TYPE (object)),
102 GTK_OBJECT_FLOATING (object) ? " (floating)" : "",
103 GTK_OBJECT_DESTROYED (object) ? " (destroyed)" : "");
106 gtk_object_debug (void)
108 g_hash_table_foreach (living_objs_ht, gtk_object_debug_foreach, NULL);
110 g_message ("living objects count = %d", obj_count);
112 #endif /* G_ENABLE_DEBUG */
114 /****************************************************
115 * GtkObject type, class and instance initialization
117 ****************************************************/
120 gtk_object_init_type (void)
122 GtkType object_type = 0;
123 GtkTypeInfo object_info =
127 sizeof (GtkObjectClass),
128 (GtkClassInitFunc) gtk_object_class_init,
129 (GtkObjectInitFunc) gtk_object_init,
130 /* reserved_1 */ NULL,
131 /* reserved_2 */ NULL,
132 (GtkClassInitFunc) gtk_object_base_class_init,
135 object_type = gtk_type_unique (0, &object_info);
136 g_assert (object_type == GTK_TYPE_OBJECT);
138 #ifdef G_ENABLE_DEBUG
139 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
140 ATEXIT (gtk_object_debug);
141 #endif /* G_ENABLE_DEBUG */
145 gtk_object_get_type (void)
147 return GTK_TYPE_OBJECT;
151 gtk_object_base_class_init (GtkObjectClass *class)
153 /* reset instance specific fields that don't get inhrited */
154 class->signals = NULL;
158 /* reset instance specifc methods that don't get inherited */
159 class->get_arg = NULL;
160 class->set_arg = NULL;
164 gtk_object_class_init (GtkObjectClass *class)
166 gtk_object_add_arg_type ("GtkObject::user_data",
170 gtk_object_add_arg_type ("GtkObject::signal",
174 gtk_object_add_arg_type ("GtkObject::signal_after",
178 gtk_object_add_arg_type ("GtkObject::object_signal",
182 gtk_object_add_arg_type ("GtkObject::object_signal_after",
185 ARG_OBJECT_SIGNAL_AFTER);
187 object_signals[DESTROY] =
188 gtk_signal_new ("destroy",
191 GTK_SIGNAL_OFFSET (GtkObjectClass, destroy),
192 gtk_marshal_NONE__NONE,
195 gtk_object_class_add_signals (class, object_signals, LAST_SIGNAL);
197 class->get_arg = gtk_object_get_arg;
198 class->set_arg = gtk_object_set_arg;
199 class->shutdown = gtk_object_shutdown;
200 class->destroy = gtk_object_real_destroy;
201 class->finalize = gtk_object_finalize;
205 gtk_object_init (GtkObject *object)
207 GTK_OBJECT_FLAGS (object) = GTK_FLOATING;
209 object->ref_count = 1;
210 object->object_data = NULL;
212 #ifdef G_ENABLE_DEBUG
213 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
218 living_objs_ht = g_hash_table_new (g_direct_hash, NULL);
220 g_hash_table_insert (living_objs_ht, object, object);
222 #endif /* G_ENABLE_DEBUG */
225 /********************************************
226 * Functions to end a GtkObject's life time
228 ********************************************/
230 gtk_object_destroy (GtkObject *object)
232 g_return_if_fail (object != NULL);
233 g_return_if_fail (GTK_IS_OBJECT (object));
235 if (!GTK_OBJECT_DESTROYED (object))
237 /* we will hold a reference on the object in this place, so
238 * to ease all classes shutdown and destroy implementations.
239 * i.e. they don't have to bother about referencing at all.
241 gtk_object_ref (object);
242 object->klass->shutdown (object);
243 gtk_object_unref (object);
248 gtk_object_shutdown (GtkObject *object)
250 GTK_OBJECT_SET_FLAGS (object, GTK_DESTROYED);
251 gtk_signal_emit (object, object_signals[DESTROY]);
255 gtk_object_real_destroy (GtkObject *object)
257 if (GTK_OBJECT_CONNECTED (object))
258 gtk_signal_handlers_destroy (object);
262 gtk_object_finalize (GtkObject *object)
264 gtk_object_notify_weaks (object);
266 while (object->object_data)
268 GtkObjectData *odata;
270 odata = object->object_data;
271 object->object_data = odata->next;
272 GTK_OBJECT_DATA_DESTROY (odata);
275 gtk_type_free (GTK_OBJECT_TYPE (object), object);
278 /*****************************************
279 * GtkObject argument handlers
281 *****************************************/
284 gtk_object_set_arg (GtkObject *object,
295 gtk_object_set_user_data (object, GTK_VALUE_POINTER (*arg));
297 case ARG_OBJECT_SIGNAL_AFTER:
299 case ARG_OBJECT_SIGNAL:
301 case ARG_SIGNAL_AFTER:
305 arg_name = gtk_arg_name_strip_type (arg->name);
307 arg_name[n] == ':' &&
308 arg_name[n + 1] == ':' &&
309 arg_name[n + 2] != 0)
311 gtk_signal_connect_full (object,
313 GTK_VALUE_SIGNAL (*arg).f, NULL,
314 GTK_VALUE_SIGNAL (*arg).d,
316 (arg_id == ARG_OBJECT_SIGNAL ||
317 arg_id == ARG_OBJECT_SIGNAL_AFTER),
318 (arg_id == ARG_OBJECT_SIGNAL_AFTER ||
319 arg_id == ARG_SIGNAL_AFTER));
322 g_warning ("gtk_object_set_arg(): invalid signal argument: \"%s\"\n", arg->name);
330 gtk_object_get_arg (GtkObject *object,
337 GTK_VALUE_POINTER (*arg) = gtk_object_get_user_data (object);
340 case ARG_OBJECT_SIGNAL:
342 arg->type = GTK_TYPE_INVALID;
347 /*****************************************
348 * gtk_object_class_add_signals:
353 *****************************************/
356 gtk_object_class_add_signals (GtkObjectClass *class,
363 g_return_if_fail (class != NULL);
365 new_signals = g_new (guint, class->nsignals + nsignals);
366 for (i = 0; i < class->nsignals; i++)
367 new_signals[i] = class->signals[i];
368 for (i = 0; i < nsignals; i++)
369 new_signals[class->nsignals + i] = signals[i];
371 g_free (class->signals);
372 class->signals = new_signals;
373 class->nsignals += nsignals;
377 gtk_object_class_add_user_signal (GtkObjectClass *class,
379 GtkSignalMarshaller marshaller,
389 g_return_val_if_fail (class != NULL, 0);
393 params = g_new (GtkType, nparams);
395 va_start (args, nparams);
397 for (i = 0; i < nparams; i++)
398 params[i] = va_arg (args, GtkType);
405 signal_id = gtk_signal_newv (name,
417 gtk_object_class_add_signals (class, &signal_id, 1);
423 gtk_object_class_user_signal_new (GtkObjectClass *class,
425 GtkSignalRunType signal_flags,
426 GtkSignalMarshaller marshaller,
436 g_return_val_if_fail (class != NULL, 0);
440 params = g_new (GtkType, nparams);
442 va_start (args, nparams);
444 for (i = 0; i < nparams; i++)
445 params[i] = va_arg (args, GtkType);
452 signal_id = gtk_signal_newv (name,
464 gtk_object_class_add_signals (class, &signal_id, 1);
470 gtk_object_class_user_signal_newv (GtkObjectClass *class,
472 GtkSignalRunType signal_flags,
473 GtkSignalMarshaller marshaller,
480 g_return_val_if_fail (class != NULL, 0);
483 g_return_val_if_fail (params != NULL, 0);
485 signal_id = gtk_signal_newv (name,
495 gtk_object_class_add_signals (class, &signal_id, 1);
500 /*****************************************
506 *****************************************/
509 gtk_object_sink (GtkObject *object)
511 g_return_if_fail (object != NULL);
512 g_return_if_fail (GTK_IS_OBJECT (object));
514 if (GTK_OBJECT_FLOATING (object))
516 GTK_OBJECT_UNSET_FLAGS (object, GTK_FLOATING);
517 gtk_object_unref (object);
521 /*****************************************
524 * Weak refs are very similar to the old "destroy" signal. They allow
525 * one to register a callback that is called when the weakly
526 * referenced object is finalized.
528 * They are not implemented as a signal because they really are
529 * special and need to be used with great care. Unlike signals, which
530 * should be able to execute any code whatsoever.
532 * A weakref callback is not allowed to retain a reference to the
533 * object. Object data keys may be retrieved in a weak reference
536 * A weakref callback is called at most once.
538 *****************************************/
540 typedef struct _GtkWeakRef GtkWeakRef;
545 GtkDestroyNotify notify;
550 gtk_object_weakref (GtkObject *object,
551 GtkDestroyNotify notify,
556 g_return_if_fail (object != NULL);
557 g_return_if_fail (notify != NULL);
558 g_return_if_fail (GTK_IS_OBJECT (object));
560 if (!weakrefs_key_id)
561 weakrefs_key_id = g_quark_from_static_string (weakrefs_key);
563 weak = g_new (GtkWeakRef, 1);
564 weak->next = gtk_object_get_data_by_id (object, weakrefs_key_id);
565 weak->notify = notify;
567 gtk_object_set_data_by_id (object, weakrefs_key_id, weak);
571 gtk_object_weakunref (GtkObject *object,
572 GtkDestroyNotify notify,
575 GtkWeakRef *weaks, *w, **wp;
577 g_return_if_fail (object != NULL);
578 g_return_if_fail (GTK_IS_OBJECT (object));
580 if (!weakrefs_key_id)
583 weaks = gtk_object_get_data_by_id (object, weakrefs_key_id);
584 for (wp = &weaks; *wp; wp = &(*wp)->next)
587 if (w->notify == notify && w->data == data)
590 gtk_object_set_data_by_id (object, weakrefs_key_id, w->next);
600 gtk_object_notify_weaks (GtkObject *object)
606 w1 = gtk_object_get_data_by_id (object, weakrefs_key_id);
610 w1->notify (w1->data);
618 /****************************************************
619 * GtkObject argument mechanism and object creation
621 ****************************************************/
624 gtk_object_new (GtkType object_type,
629 GSList *arg_list = NULL;
630 GSList *info_list = NULL;
633 g_return_val_if_fail (GTK_FUNDAMENTAL_TYPE (object_type) == GTK_TYPE_OBJECT, NULL);
635 object = gtk_type_new (object_type);
637 va_start (var_args, object_type);
638 error = gtk_object_args_collect (GTK_OBJECT_TYPE (object),
646 g_warning ("gtk_object_new(): %s", error);
654 slist_arg = arg_list;
655 slist_info = info_list;
658 gtk_object_arg_set (object, slist_arg->data, slist_info->data);
659 slist_arg = slist_arg->next;
660 slist_info = slist_info->next;
662 gtk_args_collect_cleanup (arg_list, info_list);
669 gtk_object_newv (GtkType object_type,
676 g_return_val_if_fail (GTK_FUNDAMENTAL_TYPE (object_type) == GTK_TYPE_OBJECT, NULL);
678 g_return_val_if_fail (args != NULL, NULL);
680 object = gtk_type_new (object_type);
682 for (max_args = args + n_args; args < max_args; args++)
683 gtk_object_arg_set (object, args, NULL);
689 gtk_object_setv (GtkObject *object,
695 g_return_if_fail (object != NULL);
696 g_return_if_fail (GTK_IS_OBJECT (object));
698 g_return_if_fail (args != NULL);
700 for (max_args = args + n_args; args < max_args; args++)
701 gtk_object_arg_set (object, args, NULL);
705 gtk_object_getv (GtkObject *object,
711 g_return_if_fail (object != NULL);
712 g_return_if_fail (GTK_IS_OBJECT (object));
714 g_return_if_fail (args != NULL);
716 for (max_args = args + n_args; args < max_args; args++)
717 gtk_object_arg_get (object, args, NULL);
721 gtk_object_set (GtkObject *object,
725 GSList *arg_list = NULL;
726 GSList *info_list = NULL;
729 g_return_if_fail (object != NULL);
730 g_return_if_fail (GTK_IS_OBJECT (object));
732 va_start (var_args, object);
733 error = gtk_object_args_collect (GTK_OBJECT_TYPE (object),
741 g_warning ("gtk_object_set(): %s", error);
749 slist_arg = arg_list;
750 slist_info = info_list;
753 gtk_object_arg_set (object, slist_arg->data, slist_info->data);
754 slist_arg = slist_arg->next;
755 slist_info = slist_info->next;
757 gtk_args_collect_cleanup (arg_list, info_list);
762 gtk_object_arg_set (GtkObject *object,
766 GtkObjectClass *oclass;
768 g_return_if_fail (object != NULL);
769 g_return_if_fail (GTK_IS_OBJECT (object));
770 g_return_if_fail (arg != NULL);
776 error = gtk_arg_get_info (GTK_OBJECT_TYPE (object),
782 g_warning ("gtk_object_arg_set(): %s", error);
788 if (! (info->arg_flags & GTK_ARG_WRITABLE))
790 g_warning ("gtk_object_arg_set(): argument \"%s\" is not writable",
794 if (info->type != arg->type)
796 g_warning ("gtk_object_arg_set(): argument \"%s\" has invalid type `%s'",
798 gtk_type_name (arg->type));
802 oclass = gtk_type_class (info->class_type);
803 g_assert (oclass->set_arg != NULL);
804 oclass->set_arg (object, arg, info->arg_id);
808 gtk_object_arg_get (GtkObject *object,
812 GtkObjectClass *oclass;
814 g_return_if_fail (object != NULL);
815 g_return_if_fail (GTK_IS_OBJECT (object));
816 g_return_if_fail (arg != NULL);
822 error = gtk_arg_get_info (GTK_OBJECT_TYPE (object),
828 g_warning ("gtk_object_arg_get(): %s", error);
830 arg->type = GTK_TYPE_INVALID;
835 if (! (info->arg_flags & GTK_ARG_READABLE))
837 g_warning ("gtk_object_arg_get(): argument \"%s\" is not readable",
839 arg->type = GTK_TYPE_INVALID;
843 oclass = gtk_type_class (info->class_type);
844 g_assert (oclass->get_arg != NULL);
845 arg->type = info->type;
846 oclass->get_arg (object, arg, info->arg_id);
850 gtk_object_add_arg_type (const char *arg_name,
855 g_return_if_fail (arg_name != NULL);
856 g_return_if_fail (arg_type > GTK_TYPE_NONE);
857 g_return_if_fail (arg_id > 0);
858 g_return_if_fail ((arg_flags & GTK_ARG_CHILD_ARG) == 0);
859 if (arg_flags & GTK_ARG_CONSTRUCT)
860 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) == GTK_ARG_READWRITE);
862 g_return_if_fail ((arg_flags & GTK_ARG_READWRITE) != 0);
864 if (!object_arg_info_ht)
865 object_arg_info_ht = g_hash_table_new (gtk_arg_info_hash,
868 gtk_arg_type_new_static (GTK_TYPE_OBJECT,
870 GTK_STRUCT_OFFSET (GtkObjectClass, n_args),
878 gtk_object_args_collect (GtkType object_type,
880 GSList **info_list_p,
883 return gtk_args_collect (object_type,
891 gtk_object_arg_get_info (GtkType object_type,
892 const gchar *arg_name,
895 return gtk_arg_get_info (object_type,
902 gtk_object_query_args (GtkType class_type,
906 g_return_val_if_fail (n_args != NULL, NULL);
908 g_return_val_if_fail (GTK_FUNDAMENTAL_TYPE (class_type) == GTK_TYPE_OBJECT, NULL);
910 return gtk_args_query (class_type, object_arg_info_ht, arg_flags, n_args);
913 /********************************************************
914 * GtkObject and GtkObjectClass cast checking functions
916 ********************************************************/
919 gtk_object_descriptive_type_name (GtkType type)
923 name = gtk_type_name (type);
931 gtk_object_check_cast (GtkObject *obj,
936 g_warning ("invalid cast from (NULL) pointer to `%s'",
937 gtk_object_descriptive_type_name (cast_type));
942 g_warning ("invalid unclassed pointer in cast to `%s'",
943 gtk_object_descriptive_type_name (cast_type));
946 if (obj->klass->type < GTK_TYPE_OBJECT)
948 g_warning ("invalid class type `%s' in cast to `%s'",
949 gtk_object_descriptive_type_name (obj->klass->type),
950 gtk_object_descriptive_type_name (cast_type));
953 if (!gtk_type_is_a (obj->klass->type, cast_type))
955 g_warning ("invalid cast from `%s' to `%s'",
956 gtk_object_descriptive_type_name (obj->klass->type),
957 gtk_object_descriptive_type_name (cast_type));
965 gtk_object_check_class_cast (GtkObjectClass *klass,
970 g_warning ("invalid class cast from (NULL) pointer to `%s'",
971 gtk_object_descriptive_type_name (cast_type));
974 if (klass->type < GTK_TYPE_OBJECT)
976 g_warning ("invalid class type `%s' in class cast to `%s'",
977 gtk_object_descriptive_type_name (klass->type),
978 gtk_object_descriptive_type_name (cast_type));
981 if (!gtk_type_is_a (klass->type, cast_type))
983 g_warning ("invalid class cast from `%s' to `%s'",
984 gtk_object_descriptive_type_name (klass->type),
985 gtk_object_descriptive_type_name (cast_type));
992 /*****************************************
993 * GtkObject object_data mechanism
995 *****************************************/
998 gtk_object_set_data_by_id (GtkObject *object,
1002 g_return_if_fail (data_id > 0);
1004 gtk_object_set_data_by_id_full (object, data_id, data, NULL);
1008 gtk_object_set_data (GtkObject *object,
1012 g_return_if_fail (key != NULL);
1014 gtk_object_set_data_by_id_full (object, gtk_object_data_force_id (key), data, NULL);
1018 gtk_object_set_data_by_id_full (GtkObject *object,
1021 GtkDestroyNotify destroy)
1023 GtkObjectData *odata;
1025 g_return_if_fail (object != NULL);
1026 g_return_if_fail (GTK_IS_OBJECT (object));
1027 g_return_if_fail (data_id > 0);
1029 odata = object->object_data;
1032 GtkObjectData *prev;
1038 if (odata->id == data_id)
1041 prev->next = odata->next;
1043 object->object_data = odata->next;
1045 GTK_OBJECT_DATA_DESTROY (odata);
1050 odata = odata->next;
1057 if (odata->id == data_id)
1059 register GtkDestroyNotify dfunc;
1060 register gpointer ddata;
1062 dfunc = odata->destroy;
1063 ddata = odata->data;
1064 odata->destroy = destroy;
1067 /* we need to have updated all structures prior to
1068 * invokation of the destroy function
1076 odata = odata->next;
1079 if (gtk_object_data_free_list)
1081 odata = gtk_object_data_free_list;
1082 gtk_object_data_free_list = odata->next;
1086 GtkObjectData *odata_block;
1089 odata_block = g_new0 (GtkObjectData, GTK_OBJECT_DATA_BLOCK_SIZE);
1090 for (i = 1; i < GTK_OBJECT_DATA_BLOCK_SIZE; i++)
1092 (odata_block + i)->next = gtk_object_data_free_list;
1093 gtk_object_data_free_list = (odata_block + i);
1096 odata = odata_block;
1099 odata->id = data_id;
1101 odata->destroy = destroy;
1102 odata->next = object->object_data;
1104 object->object_data = odata;
1109 gtk_object_set_data_full (GtkObject *object,
1112 GtkDestroyNotify destroy)
1114 g_return_if_fail (key != NULL);
1116 gtk_object_set_data_by_id_full (object, gtk_object_data_force_id (key), data, destroy);
1120 gtk_object_get_data_by_id (GtkObject *object,
1123 GtkObjectData *odata;
1125 g_return_val_if_fail (object != NULL, NULL);
1126 g_return_val_if_fail (GTK_IS_OBJECT (object), NULL);
1130 odata = object->object_data;
1133 if (odata->id == data_id)
1135 odata = odata->next;
1143 gtk_object_get_data (GtkObject *object,
1148 g_return_val_if_fail (key != NULL, NULL);
1150 id = gtk_object_data_try_key (key);
1152 return gtk_object_get_data_by_id (object, id);
1158 gtk_object_remove_data_by_id (GtkObject *object,
1162 gtk_object_set_data_by_id_full (object, data_id, NULL, NULL);
1166 gtk_object_remove_data (GtkObject *object,
1171 g_return_if_fail (key != NULL);
1173 id = gtk_object_data_try_key (key);
1175 gtk_object_set_data_by_id_full (object, id, NULL, NULL);
1179 gtk_object_set_user_data (GtkObject *object,
1182 if (!user_data_key_id)
1183 user_data_key_id = g_quark_from_static_string (user_data_key);
1185 gtk_object_set_data_by_id_full (object, user_data_key_id, data, NULL);
1189 gtk_object_get_user_data (GtkObject *object)
1191 if (user_data_key_id)
1192 return gtk_object_get_data_by_id (object, user_data_key_id);
1197 /*******************************************
1198 * GtkObject referencing and unreferencing
1200 *******************************************/
1202 #undef gtk_object_ref
1203 #undef gtk_object_unref
1206 gtk_object_ref (GtkObject *object)
1208 g_return_if_fail (object != NULL);
1209 g_return_if_fail (GTK_IS_OBJECT (object));
1211 object->ref_count += 1;
1215 gtk_object_unref (GtkObject *object)
1217 g_return_if_fail (object != NULL);
1218 g_return_if_fail (GTK_IS_OBJECT (object));
1220 if (object->ref_count == 1)
1221 gtk_object_destroy (object);
1223 if (object->ref_count > 0)
1224 object->ref_count -= 1;
1226 if (object->ref_count == 0)
1228 #ifdef G_ENABLE_DEBUG
1229 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1231 g_assert (g_hash_table_lookup (living_objs_ht, object) == object);
1232 g_hash_table_remove (living_objs_ht, object);
1235 #endif /* G_ENABLE_DEBUG */
1236 object->klass->finalize (object);
1240 static GtkObject *gtk_trace_object = NULL;
1242 gtk_trace_referencing (GtkObject *object,
1248 if (gtk_debug_flags & GTK_DEBUG_OBJECTS)
1250 gboolean exists = TRUE;
1252 g_return_if_fail (object != NULL);
1253 g_return_if_fail (GTK_IS_OBJECT (object));
1255 #ifdef G_ENABLE_DEBUG
1256 exists = g_hash_table_lookup (living_objs_ht, object) != NULL;
1257 #endif /* G_ENABLE_DEBUG */
1260 (object == gtk_trace_object ||
1261 gtk_trace_object == (void*)42))
1262 fprintf (stdout, "trace: object_%s: (%s:%p)->ref_count=%d %s (%s:%d)\n",
1263 do_ref ? "ref" : "unref",
1264 gtk_type_name (GTK_OBJECT_TYPE (object)),
1267 do_ref ? "+ 1" : "- 1",
1271 fprintf (stdout, "trace: object_%s(%p): no such object! (%s:%d)\n",
1272 do_ref ? "ref" : "unref",
1279 gtk_object_ref (object);
1281 gtk_object_unref (object);