#define _GNU_SOURCE /* GHC's semi-public Rts API */ #include #include #include #include #include #include #include static int print_debug_info () { static int __print_debug_info = -1; if (__print_debug_info == -1) { __print_debug_info = getenv ("HASKELL_GI_DEBUG_MEM") != NULL; } return __print_debug_info; } /* A mutex protecting the log file handle. We make it recursive, i.e. refcounted, so it is OK to lock repeatedly in the same thread. */ static pthread_mutex_t log_mutex #if defined(PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP) = PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP; #elif defined(PTHREAD_RECURSIVE_MUTEX_INITIALIZER) = PTHREAD_RECURSIVE_MUTEX_INITIALIZER; #else ; __attribute__ ((constructor)) static void init_log_mutex() { pthread_mutexattr_t attr; pthread_mutexattr_init(&attr); pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE); pthread_mutex_init(&log_mutex, &attr); pthread_mutexattr_destroy(&attr); } #endif /* Give the current thread exclusive access to the log */ static void lock_log() { pthread_mutex_lock(&log_mutex); } /* Decrease the refcount of the mutex protecting access to the log from other threads */ static void unlock_log() { pthread_mutex_unlock(&log_mutex); } /* Print the given message to the log. The passed in string does not need to be zero-terminated. The message is only printed if the HASKELL_GI_DEBUG_MEM variable is set. */ void dbg_log_with_len (const char *msg, int len) { if (print_debug_info()) { lock_log(); fwrite(msg, len, 1, stderr); unlock_log(); } } /* Print the given printf-style message to the log. The message is only printed if the HASKELL_GI_DEBUG_MEM variable is set. */ __attribute__ ((format (printf, 1, 2))) static void dbg_log (const char *msg, ...) { va_list args; va_start(args, msg); if (print_debug_info()) { lock_log(); vfprintf(stderr, msg, args); unlock_log(); } va_end(args); } int check_object_type (void *instance, GType type) { int result; if (instance != NULL) { result = !!G_TYPE_CHECK_INSTANCE_TYPE(instance, type); } else { result = 0; dbg_log("Check failed: got a null pointer\n"); } return result; } GType _haskell_gi_g_value_get_type (GValue *gvalue) { return G_VALUE_TYPE (gvalue); } /* Information about a boxed type to free */ typedef struct { GType gtype; gpointer boxed; } BoxedFreeInfo; /* Auxiliary function for freeing boxed types in the main loop. See the annotation in g_object_unref_in_main_loop() below. */ static gboolean main_loop_boxed_free_helper (gpointer _info) { BoxedFreeInfo *info = (BoxedFreeInfo*)_info; if (print_debug_info()) { GThread *self = g_thread_self (); lock_log(); dbg_log("Freeing a boxed object at %p from idle callback [thread: %p]\n", info->boxed, self); dbg_log("\tIt is of type %s\n", g_type_name(info->gtype)); } g_boxed_free (info->gtype, info->boxed); if (print_debug_info()) { dbg_log("\tdone freeing %p.\n", info->boxed); unlock_log(); } g_free(info); return FALSE; /* Do not invoke again */ } void boxed_free_helper (GType gtype, void *boxed) { BoxedFreeInfo *info = g_malloc(sizeof(BoxedFreeInfo)); info->gtype = gtype; info->boxed = boxed; g_idle_add (main_loop_boxed_free_helper, info); } void dbg_g_object_disown (GObject *obj) { GType gtype; if (print_debug_info()) { lock_log(); GThread *self = g_thread_self(); dbg_log("Disowning a GObject at %p [thread: %p]\n", obj, self); gtype = G_TYPE_FROM_INSTANCE (obj); dbg_log("\tIt is of type %s\n", g_type_name(gtype)); dbg_log("\tIts refcount before disowning is %d\n", (int)obj->ref_count); unlock_log(); } } static void print_object_dbg_info (GObject *obj) { GThread *self = g_thread_self(); GType gtype; dbg_log("Unref of %p from idle callback [thread: %p]\n", obj, self); gtype = G_TYPE_FROM_INSTANCE (obj); dbg_log("\tIt is of type %s\n", g_type_name(gtype)); dbg_log("\tIts refcount before unref is %d\n", (int)obj->ref_count); } /* We schedule all GObject deletions to happen in the main loop. The reason is that for some types the destructor is not thread safe, and assumes that it is being run from the same thread as the main loop that created the object. */ static gboolean g_object_unref_in_main_loop (gpointer obj) { if (print_debug_info()) { lock_log(); print_object_dbg_info ((GObject*)obj); } g_object_unref (obj); if (print_debug_info()) { fprintf(stderr, "\tUnref done\n"); unlock_log(); } return FALSE; /* Do not invoke again */ } void dbg_g_object_unref (GObject *obj) { g_idle_add(g_object_unref_in_main_loop, obj); } /** * dbg_g_object_new: * @gtype: #GType for the object to construct. * @n_props: Number of parameters for g_object_new_with_properties(). * @names: Names of the properties to be set. * @values: Parameters for g_object_new_with_properties(). * * Allocate a #GObject of #GType @gtype, with the given @params. The * returned object is never floating, and we always own a reference to * it. (It might not be the only existing to the object, but it is in * any case safe to call g_object_unref() when we are not wrapping the * object ourselves anymore.) * * Returns: A new #GObject. */ gpointer dbg_g_object_new (GType gtype, guint n_props, const char *names[], const GValue values[]) { gpointer result; if (print_debug_info()) { GThread *self = g_thread_self(); lock_log(); dbg_log("Creating a new GObject of type %s [thread: %p]\n", g_type_name(gtype), self); } #if GLIB_CHECK_VERSION(2,54,0) result = g_object_new_with_properties (gtype, n_props, names, values); #else { GParameter params[n_props]; int i; for (i=0; icallback); hs_free_fun_ptr (closure->callback); dbg_log("\tDone.\n"); unlock_log(); } /* Check whether the given closure is floating */ gboolean haskell_gi_g_closure_is_floating (GClosure *closure) { return !!(closure->floating); }