From 608bc4a5795e5fcc8461b845648b167b9968ee07 Mon Sep 17 00:00:00 2001 From: Claudio Takahasi Date: Wed, 20 Jul 2011 17:24:47 -0300 Subject: [PATCH] Manage GAttrib refs based on registered callbacks Connection should not be dropped if there is at least one ATT connection callback registered. After discovering primary services over the LE link, GAttrib ref needs to be held to allow notifying an active connection while registering connection callbacks(inside plugins probing). --- src/device.c | 40 ++++++++++++++++++++++++++++------------ 1 file changed, 28 insertions(+), 12 deletions(-) diff --git a/src/device.c b/src/device.c index 821ecba40..e0a618296 100644 --- a/src/device.c +++ b/src/device.c @@ -96,6 +96,7 @@ struct browse_req { DBusConnection *conn; DBusMessage *msg; GIOChannel *io; + GAttrib *attrib; struct btd_device *device; GSList *match_uuids; GSList *profiles_added; @@ -163,6 +164,8 @@ static void browse_request_free(struct browse_req *req, gboolean shutdown) { if (req->listener_id) g_dbus_remove_watch(req->conn, req->listener_id); + if (req->attrib) + g_attrib_unref(req->attrib); if (req->io) { if (shutdown) g_io_channel_shutdown(req->io, FALSE, NULL); @@ -1642,9 +1645,21 @@ static void primary_cb(GSList *services, guint8 status, gpointer user_data) uuids = g_slist_append(uuids, prim->uuid); } + /* + * Profiles may register attio callbacks in the probing callback. + * GAttrib reference can be released if the registered callbacks + * list is emtpy. + */ + device->attrib = g_attrib_ref(req->attrib); + device_register_services(req->conn, device, g_slist_copy(services), -1); device_probe_drivers(device, uuids); + if (device->attios == NULL) { + g_attrib_unref(device->attrib); + device->attrib = NULL; + } + g_slist_free(uuids); create_device_reply(device, req); @@ -1683,16 +1698,16 @@ static void att_connect_cb(GIOChannel *io, GError *gerr, gpointer user_data) device->attioid = 0; } - device->attrib = g_attrib_new(io); - g_attrib_set_destroy_function(device->attrib, attrib_destroyed, device); - - if (req) - gatt_discover_primary(device->attrib, NULL, primary_cb, req); - else if (device->attios) + if (req) { + req->attrib = g_attrib_new(io); + gatt_discover_primary(req->attrib, NULL, primary_cb, req); + } else if (device->attios) { + device->attrib = g_attrib_new(io); + g_attrib_set_destroy_function(device->attrib, attrib_destroyed, + device); g_slist_foreach(device->attios, attio_connected, device->attrib); - - g_io_channel_unref(io); + } } static gboolean att_auto_connect(gpointer user_data) @@ -1733,6 +1748,8 @@ static gboolean att_auto_connect(gpointer user_data) return TRUE; } + g_io_channel_unref(io); + return TRUE; } @@ -2566,18 +2583,17 @@ guint btd_device_add_attio_callback(struct btd_device *device, attio->dcfunc = dcfunc; attio->user_data = user_data; - device->attios = g_slist_append(device->attios, attio); - if (device->attrib && cfunc) cfunc(device->attrib, user_data); - - if (device->attioid == 0 && device->attrib == NULL) { + else if (device->attioid == 0) { att_auto_connect(device); device->attioid = g_timeout_add_seconds(AUTOCONNECT_INTERVAL, att_auto_connect, device); } + device->attios = g_slist_append(device->attios, attio); + return attio->id; } -- 2.47.3