Hi Marcin, On Mon, Oct 28, 2013 at 01:30:09PM +0100, Marcin Kraglak wrote: > Use mgmt interface for adding and removing uuids to adapter. > This is needed for proper changing adapter's class of device. > --- > android/adapter.c | 141 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ > 1 file changed, 141 insertions(+) > > diff --git a/android/adapter.c b/android/adapter.c > index b1cb5f8..bed0423 100644 > --- a/android/adapter.c > +++ b/android/adapter.c > @@ -532,10 +532,105 @@ error: > ipc_send_rsp(io, HAL_SERVICE_ID_BLUETOOTH, status); > } > > +static int uuid_cmp(const void *a, const void *b) > +{ > + const sdp_record_t *rec = a; > + const uuid_t *uuid = b; > + > + return sdp_uuid_cmp(&rec->svclass, uuid); > +} > + > +static void uuid_to_uuid128(uuid_t *uuid128, const uuid_t *uuid) > +{ > + if (uuid->type == SDP_UUID16) > + sdp_uuid16_to_uuid128(uuid128, uuid); > + else if (uuid->type == SDP_UUID32) > + sdp_uuid32_to_uuid128(uuid128, uuid); > + else > + memcpy(uuid128, uuid, sizeof(*uuid)); > +} > + > +static uint8_t get_uuid_mask(const uuid_t *uuid) > +{ > + if (uuid->type != SDP_UUID16) > + return 0; > + > + switch (uuid->value.uuid16) { > + case DIALUP_NET_SVCLASS_ID: > + case CIP_SVCLASS_ID: > + return 0x42; /* Telephony & Networking */ > + case IRMC_SYNC_SVCLASS_ID: > + case OBEX_OBJPUSH_SVCLASS_ID: > + case OBEX_FILETRANS_SVCLASS_ID: > + case IRMC_SYNC_CMD_SVCLASS_ID: > + case PBAP_PSE_SVCLASS_ID: > + return 0x10; /* Object Transfer */ > + case HEADSET_SVCLASS_ID: > + case HANDSFREE_SVCLASS_ID: > + return 0x20; /* Audio */ > + case CORDLESS_TELEPHONY_SVCLASS_ID: > + case INTERCOM_SVCLASS_ID: > + case FAX_SVCLASS_ID: > + case SAP_SVCLASS_ID: > + /* > + * Setting the telephony bit for the handsfree audio gateway > + * role is not required by the HFP specification, but the > + * Nokia 616 carkit is just plain broken! It will refuse > + * pairing without this bit set. > + */ > + case HANDSFREE_AGW_SVCLASS_ID: > + return 0x40; /* Telephony */ > + case AUDIO_SOURCE_SVCLASS_ID: > + case VIDEO_SOURCE_SVCLASS_ID: > + return 0x08; /* Capturing */ > + case AUDIO_SINK_SVCLASS_ID: > + case VIDEO_SINK_SVCLASS_ID: > + return 0x04; /* Rendering */ > + case PANU_SVCLASS_ID: > + case NAP_SVCLASS_ID: > + case GN_SVCLASS_ID: > + return 0x02; /* Networking */ > + default: > + return 0; > + } > +} Can those functions be reused between our android/adapter and src/adapter? Best regards Andrei Emeltchenko > + > +static void add_uuid_complete(uint8_t status, uint16_t length, > + const void *param, void *user_data) > +{ > + if (status != MGMT_STATUS_SUCCESS) { > + error("Failed to add UUID: %s (0x%02x)", > + mgmt_errstr(status), status); > + return; > + } > + /*TODO inform that adapter uuids has been changed*/ > +} > + > +static int add_uuid(uuid_t *uuid, uint8_t svc_hint) > +{ > + struct mgmt_cp_add_uuid cp; > + uuid_t uuid128; > + uint128_t uint128; > + > + uuid_to_uuid128(&uuid128, uuid); > + > + ntoh128((uint128_t *) uuid128.value.uuid128.data, &uint128); > + htob128(&uint128, (uint128_t *) cp.uuid); > + cp.svc_hint = svc_hint; > + > + if (mgmt_send(adapter->mgmt, MGMT_OP_ADD_UUID, > + 0, sizeof(cp), &cp, > + add_uuid_complete, adapter, NULL) > 0) > + return 0; > + > + return -EIO; > +} > + > static void adapter_service_insert(sdp_record_t *rec) > { > sdp_list_t *browse_list = NULL; > uuid_t browse_uuid; > + bool new_uuid; > > /* skip record without a browse group */ > if (sdp_get_browse_groups(rec, &browse_list) < 0) { > @@ -552,6 +647,19 @@ static void adapter_service_insert(sdp_record_t *rec) > adapter->services = sdp_list_insert_sorted( > adapter->services, rec, record_sort); > > + if (!sdp_list_find(adapter->services, &rec->svclass, uuid_cmp)) > + new_uuid = true; > + else > + new_uuid = false; > + > + adapter->services = sdp_list_insert_sorted(adapter->services, rec, > + record_sort); > + > + if (new_uuid) { > + uint8_t svc_hint = get_uuid_mask(&rec->svclass); > + add_uuid(&rec->svclass, svc_hint); > + } > + > done: > sdp_list_free(browse_list, free); > } > @@ -569,6 +677,36 @@ int bt_adapter_service_add(sdp_record_t *rec) > return ret; > } > > +static void remove_uuid_complete(uint8_t status, uint16_t length, > + const void *param, void *user_data) > +{ > + if (status != MGMT_STATUS_SUCCESS) { > + error("Failed to remove UUID: %s (0x%02x)", > + mgmt_errstr(status), status); > + return; > + } > + /*TODO inform that adapter uuids has been changed*/ > +} > + > +static int remove_uuid(uuid_t *uuid) > +{ > + struct mgmt_cp_remove_uuid cp; > + uuid_t uuid128; > + uint128_t uint128; > + > + uuid_to_uuid128(&uuid128, uuid); > + > + ntoh128((uint128_t *) uuid128.value.uuid128.data, &uint128); > + htob128(&uint128, (uint128_t *) cp.uuid); > + > + if (mgmt_send(adapter->mgmt, MGMT_OP_REMOVE_UUID, > + 0, sizeof(cp), &cp, > + remove_uuid_complete, adapter, NULL) > 0) > + return 0; > + > + return -EIO; > +} > + > void bt_adapter_service_remove(uint32_t handle) > { > sdp_record_t *rec = sdp_record_find(handle); > @@ -578,6 +716,9 @@ void bt_adapter_service_remove(uint32_t handle) > > adapter->services = sdp_list_remove(adapter->services, rec); > > + if (!sdp_list_find(adapter->services, &rec->svclass, uuid_cmp)) > + remove_uuid(&rec->svclass); > + > remove_record_from_server(rec->handle); > } > > -- > 1.8.2.2 > > -- > 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 -- 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