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).
This commit is contained in:
Claudio Takahasi 2011-07-20 17:24:47 -03:00 committed by Johan Hedberg
parent b3eea75304
commit 608bc4a579

View File

@ -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;
}