diff options
Diffstat (limited to 'libempathy')
-rw-r--r-- | libempathy/empathy-contact-factory.c | 185 |
1 files changed, 85 insertions, 100 deletions
diff --git a/libempathy/empathy-contact-factory.c b/libempathy/empathy-contact-factory.c index 8f5c51a5c..28f5f2d79 100644 --- a/libempathy/empathy-contact-factory.c +++ b/libempathy/empathy-contact-factory.c @@ -48,7 +48,7 @@ struct _EmpathyContactFactoryPriv { typedef struct { EmpathyContactFactory *factory; McAccount *account; - guint nb_pending_calls; + guint refcount; TpConn *tp_conn; DBusGProxy *aliasing_iface; @@ -126,18 +126,79 @@ contact_factory_account_data_find_by_id (ContactFactoryAccountData *account_data return l ? l->data : NULL; } -static void contact_factory_account_data_free (gpointer data); +static void contact_factory_account_data_disconnect (ContactFactoryAccountData *account_data); static void -contact_factory_account_data_return_call (ContactFactoryAccountData *account_data) +contact_factory_weak_notify (gpointer data, + GObject *where_the_object_was) { - if (--account_data->nb_pending_calls == 0 && - account_data->contacts == NULL) { - contact_factory_account_data_free (account_data); + ContactFactoryAccountData *account_data = data; + + empathy_debug (DEBUG_DOMAIN, "Remove finalized contact %p", + where_the_object_was); + + account_data->contacts = g_list_remove (account_data->contacts, + where_the_object_was); + if (!account_data->contacts) { + EmpathyContactFactoryPriv *priv = GET_PRIV (account_data->factory); + + g_hash_table_remove (priv->accounts, account_data->account); } } static void +contact_factory_remove_foreach (gpointer data, + gpointer user_data) +{ + ContactFactoryAccountData *account_data = user_data; + EmpathyContact *contact = data; + + g_object_weak_unref (G_OBJECT (contact), + contact_factory_weak_notify, + account_data); +} + +static ContactFactoryAccountData * +contact_factory_account_data_ref (ContactFactoryAccountData *account_data) +{ + account_data->refcount++; + + return account_data; +} + +static void +contact_factory_account_data_unref (ContactFactoryAccountData *account_data) +{ + account_data->refcount--; + if (account_data->refcount > 0) { + return; + } + + empathy_debug (DEBUG_DOMAIN, "Account data finalized: %p (%s)", + account_data, + mc_account_get_normalized_name (account_data->account)); + + contact_factory_account_data_disconnect (account_data); + + if (account_data->contacts) { + g_list_foreach (account_data->contacts, + contact_factory_remove_foreach, + account_data); + g_list_free (account_data->contacts); + } + + if (account_data->account) { + g_object_unref (account_data->account); + } + + if (account_data->tp_conn) { + g_object_unref (account_data->tp_conn); + } + + g_slice_free (ContactFactoryAccountData, account_data); +} + +static void contact_factory_presences_table_foreach (const gchar *state_str, GHashTable *presences_table, EmpathyPresence **presence) @@ -216,7 +277,7 @@ contact_factory_get_presence_cb (DBusGProxy *proxy, g_hash_table_destroy (handle_table); OUT: - contact_factory_account_data_return_call (account_data); + contact_factory_account_data_unref (account_data); } static void @@ -241,7 +302,7 @@ contact_factory_set_aliases_cb (DBusGProxy *proxy, error->message); } - contact_factory_account_data_return_call (account_data); + contact_factory_account_data_unref (account_data); } static void @@ -280,7 +341,7 @@ contact_factory_request_aliases_cb (DBusGProxy *proxy, g_strfreev (contact_names); OUT: - contact_factory_account_data_return_call (data->account_data); + contact_factory_account_data_unref (data->account_data); g_free (data->handles); g_slice_free (RequestAliasesData, data); } @@ -364,7 +425,7 @@ contact_factory_request_avatars_cb (DBusGProxy *proxy, error->message); } - contact_factory_account_data_return_call (account_data); + contact_factory_account_data_unref (account_data); } typedef struct { @@ -454,17 +515,16 @@ contact_factory_get_known_avatar_tokens_cb (DBusGProxy *proxy, /* Request needed avatars */ if (data.handles->len > 0) { - account_data->nb_pending_calls++; tp_conn_iface_avatars_request_avatars_async (account_data->avatars_iface, data.handles, contact_factory_request_avatars_cb, - account_data); + contact_factory_account_data_ref (account_data)); } g_hash_table_destroy (tokens); g_array_free (data.handles, TRUE); OUT: - contact_factory_account_data_return_call (account_data); + contact_factory_account_data_unref (account_data); } static void @@ -486,11 +546,10 @@ contact_factory_avatar_updated_cb (DBusGProxy *proxy, handles = g_array_new (FALSE, FALSE, sizeof (guint)); g_array_append_val (handles, handle); - account_data->nb_pending_calls++; tp_conn_iface_avatars_request_avatars_async (account_data->avatars_iface, handles, contact_factory_request_avatars_cb, - account_data); + contact_factory_account_data_ref (account_data)); g_array_free (handles, TRUE); } @@ -570,7 +629,7 @@ contact_factory_get_capabilities_cb (DBusGProxy *proxy, g_ptr_array_free (capabilities, TRUE); OUT: - contact_factory_account_data_return_call (account_data); + contact_factory_account_data_unref (account_data); } static void @@ -607,19 +666,17 @@ contact_factory_request_everything (ContactFactoryAccountData *account_data, GArray *handles) { if (account_data->presence_iface) { - account_data->nb_pending_calls++; tp_conn_iface_presence_get_presence_async (account_data->presence_iface, handles, contact_factory_get_presence_cb, - account_data); + contact_factory_account_data_ref (account_data)); } if (account_data->aliasing_iface) { RequestAliasesData *data; - account_data->nb_pending_calls++; data = g_slice_new (RequestAliasesData); - data->account_data = account_data; + data->account_data = contact_factory_account_data_ref (account_data); data->handles = g_memdup (handles->data, handles->len * sizeof (guint)); tp_conn_iface_aliasing_request_aliases_async (account_data->aliasing_iface, @@ -629,19 +686,17 @@ contact_factory_request_everything (ContactFactoryAccountData *account_data, } if (account_data->avatars_iface) { - account_data->nb_pending_calls++; tp_conn_iface_avatars_get_known_avatar_tokens_async (account_data->avatars_iface, handles, contact_factory_get_known_avatar_tokens_cb, - account_data); + contact_factory_account_data_ref (account_data)); } if (account_data->capabilities_iface) { - account_data->nb_pending_calls++; tp_conn_iface_capabilities_get_capabilities_async (account_data->capabilities_iface, handles, contact_factory_get_capabilities_cb, - account_data); + contact_factory_account_data_ref (account_data)); } } @@ -679,7 +734,7 @@ contact_factory_request_handles_cb (DBusGProxy *proxy, OUT: g_list_foreach (data->contacts, (GFunc) g_object_unref, NULL); g_list_free (data->contacts); - contact_factory_account_data_return_call (data->account_data); + contact_factory_account_data_unref (data->account_data); g_slice_free (RequestHandlesData, data); } @@ -846,7 +901,7 @@ contact_factory_account_data_update (ContactFactoryAccountData *account_data) /* Request new handles for all contacts */ if (account_data->contacts) { data = g_slice_new (RequestHandlesData); - data->account_data = account_data; + data->account_data = contact_factory_account_data_ref (account_data); data->contacts = g_list_copy (account_data->contacts); g_list_foreach (data->contacts, (GFunc) g_object_ref, NULL); @@ -858,7 +913,6 @@ contact_factory_account_data_update (ContactFactoryAccountData *account_data) i++; } - account_data->nb_pending_calls++; tp_conn_request_handles_async (DBUS_G_PROXY (account_data->tp_conn), TP_HANDLE_TYPE_CONTACT, contact_ids, @@ -868,38 +922,6 @@ contact_factory_account_data_update (ContactFactoryAccountData *account_data) } } -static void -contact_factory_weak_notify (gpointer data, - GObject *where_the_object_was) -{ - ContactFactoryAccountData *account_data = data; - - empathy_debug (DEBUG_DOMAIN, "Remove finalized contact %p", - where_the_object_was); - - account_data->contacts = g_list_remove (account_data->contacts, - where_the_object_was); - if (!account_data->contacts) { - EmpathyContactFactoryPriv *priv; - - priv = GET_PRIV (account_data->factory); - - g_hash_table_remove (priv->accounts, account_data->account); - } -} - -static void -contact_factory_remove_foreach (gpointer data, - gpointer user_data) -{ - ContactFactoryAccountData *account_data = user_data; - EmpathyContact *contact = data; - - g_object_weak_unref (G_OBJECT (contact), - contact_factory_weak_notify, - account_data); -} - static ContactFactoryAccountData * contact_factory_account_data_new (EmpathyContactFactory *factory, McAccount *account) @@ -909,6 +931,7 @@ contact_factory_account_data_new (EmpathyContactFactory *factory, account_data = g_slice_new0 (ContactFactoryAccountData); account_data->factory = factory; account_data->account = g_object_ref (account); + account_data->refcount = 1; contact_factory_account_data_update (account_data); @@ -916,42 +939,6 @@ contact_factory_account_data_new (EmpathyContactFactory *factory, } static void -contact_factory_account_data_free (gpointer data) -{ - ContactFactoryAccountData *account_data = data; - - contact_factory_account_data_disconnect (account_data); - - if (account_data->contacts) { - g_list_foreach (account_data->contacts, - contact_factory_remove_foreach, - account_data); - g_list_free (account_data->contacts); - account_data->contacts = NULL; - } - - if (account_data->account) { - g_object_unref (account_data->account); - account_data->account = NULL; - } - - if (account_data->tp_conn) { - g_object_unref (account_data->tp_conn); - account_data->tp_conn = NULL; - account_data->aliasing_iface = NULL; - account_data->avatars_iface = NULL; - account_data->presence_iface = NULL; - account_data->capabilities_iface = NULL; - } - - /* Keep the struct alive if we have calls in flight, it will be - * destroyed once all calls returned. */ - if (account_data->nb_pending_calls == 0) { - g_slice_free (ContactFactoryAccountData, account_data); - } -} - -static void contact_factory_status_changed_cb (MissionControl *mc, TelepathyConnectionStatus status, McPresence presence, @@ -1072,9 +1059,8 @@ empathy_contact_factory_get_from_id (EmpathyContactFactory *factory, RequestHandlesData *data; const gchar *contact_ids[] = {id, NULL}; - account_data->nb_pending_calls++; data = g_slice_new (RequestHandlesData); - data->account_data = account_data; + data->account_data = contact_factory_account_data_ref (account_data); data->contacts = g_list_prepend (NULL, g_object_ref (contact)); tp_conn_request_handles_async (DBUS_G_PROXY (account_data->tp_conn), TP_HANDLE_TYPE_CONTACT, @@ -1238,11 +1224,10 @@ empathy_contact_factory_set_name (EmpathyContactFactory *factory, GUINT_TO_POINTER (handle), g_strdup (name)); - account_data->nb_pending_calls++; tp_conn_iface_aliasing_set_aliases_async (account_data->aliasing_iface, new_alias, contact_factory_set_aliases_cb, - account_data); + contact_factory_account_data_ref (account_data)); g_hash_table_destroy (new_alias); } @@ -1286,7 +1271,7 @@ empathy_contact_factory_init (EmpathyContactFactory *factory) priv->accounts = g_hash_table_new_full (empathy_account_hash, empathy_account_equal, g_object_unref, - contact_factory_account_data_free); + (GDestroyNotify) contact_factory_account_data_unref); dbus_g_proxy_connect_signal (DBUS_G_PROXY (priv->mc), "AccountStatusChanged", |