From: Jefferson Delfes <jefferson.delfes@xxxxxxxxxxxxx> The adapter will keep a list of broadcasters which set new values of advertising. The new broadcaster is added on the list when the function SetServiceData() or SetManufacturerData() is called. --- src/adapter.c | 135 +++++++++++++++++++++++++++++++++++++++++++++++++++++++++ src/eir.h | 3 ++ 2 files changed, 138 insertions(+) diff --git a/src/adapter.c b/src/adapter.c index 94c813c..3041d6c 100644 --- a/src/adapter.c +++ b/src/adapter.c @@ -117,6 +117,17 @@ struct observer_watcher { char *path; /* DBus path */ }; +struct bcast_session { + struct btd_adapter *adapter; + guint id; + char *owner; + uint8_t data_type; + uint16_t data_id; + uint8_t data_len; + /* Reserve 2 octets for ServiceUUID/CompanyId */ + uint8_t data[EIR_DATA_MAX_LEN - sizeof(uint16_t)]; +}; + struct btd_adapter { uint16_t dev_id; gboolean up; @@ -147,6 +158,7 @@ struct btd_adapter { GSList *mode_sessions; /* Request Mode sessions */ GSList *disc_sessions; /* Discovery sessions */ GSList *observers; /* Observer watchers */ + GSList *bcast_sessions; /* Broadcast sessions */ guint discov_id; /* Discovery timer */ gboolean discovering; /* Discovery active */ gboolean discov_suspended; /* Discovery suspended */ @@ -1869,10 +1881,84 @@ static DBusMessage *unregister_manufobserver(DBusConnection *conn, return dbus_message_new_method_return(msg); } +static void free_bcast_session(gpointer user_data) +{ + struct bcast_session *session = user_data; + + btd_adapter_unref(session->adapter); + + g_dbus_remove_watch(connection, session->id); + + g_free(session->owner); + g_free(session); +} + +static gint cmp_bcast_session(gconstpointer a, gconstpointer b) +{ + const struct bcast_session *session = a; + const struct bcast_session *match = b; + int ret; + + ret = g_strcmp0(session->owner, match->owner); + if (ret) + return ret; + + ret = session->data_type - match->data_type; + if (ret) + return ret; + + return session->data_id - match->data_id; +} + +static struct bcast_session *find_bcast_session(GSList *list, + const char *sender, uint8_t type, uint16_t data_id) +{ + struct bcast_session *match; + GSList *l; + + match = g_new0(struct bcast_session, 1); + match->owner = g_strdup(sender); + match->data_type = type; + match->data_id = data_id; + + l = g_slist_find_custom(list, match, cmp_bcast_session); + g_free(match->owner); + g_free(match); + + return (l ? l->data : NULL); +} + +static void bcast_session_exit(DBusConnection *conn, void *user_data) +{ + struct bcast_session *session = user_data; + struct btd_adapter *adapter = session->adapter; + uint8_t type = session->data_type; + + if (type == EIR_SVC_DATA) + DBG("Service Data Broadcaster watcher %s disconnected", + session->owner); + else + DBG("Manufacturer Data Broadcasterer watcher %s disconnected", + session->owner); + + adapter->bcast_sessions = g_slist_remove(adapter->bcast_sessions, + session); + free_bcast_session(session); + + /* FIXME: Stop advertising, send data blob from others apps to kernel + * with same type and restart advertising*/ +} + +static void update_adv_data(struct btd_adapter *adapter, + struct bcast_session *session, uint8_t *data, int size) +{ +} + static DBusMessage *set_service_data(DBusConnection *conn, DBusMessage *msg, void *data) { struct btd_adapter *adapter = data; + struct bcast_session *session; const char *sender; uint16_t uuid; uint8_t *sdata; @@ -1886,9 +1972,28 @@ static DBusMessage *set_service_data(DBusConnection *conn, sender = dbus_message_get_sender(msg); + session = find_bcast_session(adapter->bcast_sessions, sender, + EIR_SVC_DATA, uuid); + if (session) { + update_adv_data(adapter, session, sdata, ssize); + goto done; + } + + session = g_new(struct bcast_session, 1); + session->data_id = uuid; + session->owner = g_strdup(sender); + session->adapter = btd_adapter_ref(adapter); + session->id = g_dbus_add_disconnect_watch(conn, sender, + bcast_session_exit, session, + NULL); + + adapter->bcast_sessions = g_slist_prepend(adapter->bcast_sessions, + session); + DBG("Service Data Broadcaster registered for hci%d at %s", adapter->dev_id, sender); +done: return dbus_message_new_method_return(msg); } @@ -1896,6 +2001,7 @@ static DBusMessage *set_manufacturer_data(DBusConnection *conn, DBusMessage *msg, void *data) { struct btd_adapter *adapter = data; + struct bcast_session *session; const char *sender; uint16_t company_id; uint8_t *mdata; @@ -1909,9 +2015,28 @@ static DBusMessage *set_manufacturer_data(DBusConnection *conn, sender = dbus_message_get_sender(msg); + session = find_bcast_session(adapter->bcast_sessions, sender, + EIR_MANUF_DATA, company_id); + if (session != NULL) { + update_adv_data(adapter, session, mdata, msize); + goto done; + } + + session = g_new(struct bcast_session, 1); + session->data_id = company_id; + session->owner = g_strdup(sender); + session->adapter = btd_adapter_ref(adapter); + session->id = g_dbus_add_disconnect_watch(conn, sender, + bcast_session_exit, session, + NULL); + + adapter->bcast_sessions = g_slist_prepend(adapter->bcast_sessions, + session); + DBG("Manufacturer Specific Data Broadcaster registered for hci%d at %s", adapter->dev_id, sender); +done: return dbus_message_new_method_return(msg); } @@ -2698,6 +2823,16 @@ static void adapter_free(gpointer user_data) g_slist_free_full(adapter->observers, destroy_observer); } + if (adapter->bcast_sessions) { + int err; + + err = mgmt_set_broadcaster(adapter->dev_id, FALSE); + if (err < 0) + error("Failed to set Broadcaster: %s (%d)", + strerror(-err), -err); + g_slist_free_full(adapter->bcast_sessions, free_bcast_session); + } + g_free(adapter->path); g_free(adapter->name); g_free(adapter); diff --git a/src/eir.h b/src/eir.h index f8e6c7c..428fc36 100644 --- a/src/eir.h +++ b/src/eir.h @@ -38,6 +38,9 @@ #define EIR_GAP_APPEARANCE 0x19 /* GAP appearance */ #define EIR_MANUF_DATA 0xFF /* manufacturer specific data */ +/* Reserve 2 octets for length and data type */ +#define EIR_DATA_MAX_LEN (HCI_MAX_EIR_LENGTH - 2) + struct uuid_info { uuid_t uuid; uint8_t svc_hint; -- 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