Add connection logic to the Battery Plugin. When the driver is loaded, it will request a connection to the remote device and release the connection request when destroyed. --- profiles/batterystate/batterystate.c | 78 +++++++++++++++++++++++++++++++++- profiles/batterystate/batterystate.h | 3 +- profiles/batterystate/manager.c | 22 +++++++++- 3 files changed, 99 insertions(+), 4 deletions(-) diff --git a/profiles/batterystate/batterystate.c b/profiles/batterystate/batterystate.c index 04c2e5e..40663f6 100644 --- a/profiles/batterystate/batterystate.c +++ b/profiles/batterystate/batterystate.c @@ -29,17 +29,29 @@ #include "adapter.h" #include "device.h" +#include "gattrib.h" +#include "attio.h" #include "att.h" #include "gattrib.h" #include "gatt.h" #include "batterystate.h" +#include "log.h" struct battery { struct btd_device *dev; /* Device reference */ + GAttrib *attrib; /* GATT connection */ + guint attioid; /* Att watcher id */ + struct att_range *svc_range; /* Battery range */ + GSList *chars; /* Characteristics */ }; static GSList *servers; +struct characteristic { + struct gatt_char attr; /* Characteristic */ + struct battery *batt; /* Parent Battery Service */ +}; + static gint cmp_device(gconstpointer a, gconstpointer b) { const struct battery *batt = a; @@ -55,20 +67,84 @@ static void batterystate_free(gpointer user_data) { struct battery *batt = user_data; + if (batt->chars != NULL) + g_slist_free_full(batt->chars, g_free); + + if (batt->attioid > 0) + btd_device_remove_attio_callback(batt->dev, batt->attioid); + + if (batt->attrib != NULL) + g_attrib_unref(batt->attrib); + btd_device_unref(batt->dev); g_free(batt); } +static void configure_batterystate_cb(GSList *characteristics, guint8 status, + gpointer user_data) +{ + struct battery *batt = user_data; + GSList *l; + + if (status != 0) { + error("Discover batterystate characteristics: %s", + att_ecode2str(status)); + return; + } -int batterystate_register(struct btd_device *device) + for (l = characteristics; l; l = l->next) { + struct gatt_char *c = l->data; + struct characteristic *ch; + + ch = g_new0(struct characteristic, 1); + ch->attr.handle = c->handle; + ch->attr.properties = c->properties; + ch->attr.value_handle = c->value_handle; + memcpy(ch->attr.uuid, c->uuid, MAX_LEN_UUID_STR + 1); + ch->batt = batt; + + batt->chars = g_slist_append(batt->chars, ch); + } +} + +static void attio_connected_cb(GAttrib *attrib, gpointer user_data) +{ + struct battery *batt = user_data; + + batt->attrib = g_attrib_ref(attrib); + + if (batt->chars == NULL) { + gatt_discover_char(batt->attrib, batt->svc_range->start, + batt->svc_range->end, NULL, + configure_batterystate_cb, batt); + } +} + +static void attio_disconnected_cb(gpointer user_data) +{ + struct battery *batt = user_data; + + g_attrib_unref(batt->attrib); + batt->attrib = NULL; +} + +int batterystate_register(struct btd_device *device, + struct gatt_primary *prim) { struct battery *batt; batt = g_new0(struct battery, 1); batt->dev = btd_device_ref(device); + batt->svc_range = g_new0(struct att_range, 1); + batt->svc_range->start = prim->range.start; + batt->svc_range->end = prim->range.end; + servers = g_slist_prepend(servers, batt); + batt->attioid = btd_device_add_attio_callback(device, + attio_connected_cb, attio_disconnected_cb, + batt); return 0; } diff --git a/profiles/batterystate/batterystate.h b/profiles/batterystate/batterystate.h index 9aedae7..2d30028 100644 --- a/profiles/batterystate/batterystate.h +++ b/profiles/batterystate/batterystate.h @@ -19,6 +19,5 @@ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * */ - -int batterystate_register(struct btd_device *device); +int batterystate_register(struct btd_device *device, struct gatt_primary *prim); void batterystate_unregister(struct btd_device *device); diff --git a/profiles/batterystate/manager.c b/profiles/batterystate/manager.c index 6718acf..62076ac 100644 --- a/profiles/batterystate/manager.c +++ b/profiles/batterystate/manager.c @@ -34,9 +34,29 @@ #define BATTERY_SERVICE_UUID "0000180f-0000-1000-8000-00805f9b34fb" +static gint primary_uuid_cmp(gconstpointer a, gconstpointer b) +{ + const struct gatt_primary *prim = a; + const char *uuid = b; + + return g_strcmp0(prim->uuid, uuid); +} + static int batterystate_driver_probe(struct btd_device *device, GSList *uuids) { - return batterystate_register(device); + struct gatt_primary *prim; + GSList *primaries, *l; + + primaries = btd_device_get_primaries(device); + + l = g_slist_find_custom(primaries, BATTERY_SERVICE_UUID, + primary_uuid_cmp); + if (l == NULL) + return -EINVAL; + + prim = l->data; + + return batterystate_register(device, prim); } static void batterystate_driver_remove(struct btd_device *device) -- 1.7.9.5 -- To unsubscribe from this list: send the line "unsubscribe linux-bluetooth" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html