blob: 79bb29bf9a0b1fec75423f44ff8bd304b45bbf95 [file]
// SPDX-License-Identifier: LGPL-2.1-or-later
/*
*
* BlueZ - Bluetooth protocol stack for Linux
*
* Copyright (C) 2024 Asymptotic Inc.
*
* Author: Arun Raghavan <arun@asymptotic.io>
*
*
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdbool.h>
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
#include "bluetooth/bluetooth.h"
#include "bluetooth/uuid.h"
#include "src/shared/util.h"
#include "src/shared/att.h"
#include "src/log.h"
#include "src/shared/queue.h"
#include "src/shared/gatt-db.h"
#include "src/shared/gatt-client.h"
#include "asha.h"
/* We use strings instead of uint128_t to maintain readability */
#define ASHA_CHRC_READ_ONLY_PROPERTIES_UUID "6333651e-c481-4a3e-9169-7c902aad37bb"
#define ASHA_CHRC_AUDIO_CONTROL_POINT_UUID "f0d4de7e-4a88-476c-9d9f-1937b0996cc0"
#define ASHA_CHRC_AUDIO_STATUS_UUID "38663f1a-e711-4cac-b641-326b56404837"
#define ASHA_CHRC_VOLUME_UUID "00e4ca9e-ab14-41e4-8823-f9e70c7e91df"
#define ASHA_CHRC_LE_PSM_OUT_UUID "2d410339-82b6-42aa-b34e-e2e01df8cc1a"
static struct queue *asha_devices;
static unsigned int bt_asha_status(struct bt_asha *asha, bool connected);
static bool match_hisyncid(const void *data, const void *user_data)
{
const struct bt_asha_set *set = data;
const struct bt_asha *asha = user_data;
return (memcmp(set->hisyncid, asha->hisyncid, 8) == 0);
}
static struct bt_asha_set *find_asha_set(struct bt_asha *asha)
{
return queue_find(asha_devices, match_hisyncid, asha);
}
static uint8_t is_other_connected(struct bt_asha *asha)
{
struct bt_asha_set *set = find_asha_set(asha);
if (set) {
if (asha->right_side && set->left) {
DBG("ASHA right and left side connected");
return 1;
}
if (!asha->right_side && set->right) {
DBG("ASHA left and right side connected");
return 1;
}
}
if (asha->right_side)
DBG("ASHA right side connected");
else
DBG("ASHA left side connected");
return 0;
}
static void update_asha_set(struct bt_asha *asha, bool connected)
{
struct bt_asha_set *set;
set = queue_find(asha_devices, match_hisyncid, asha);
if (connected) {
if (!set) {
set = new0(struct bt_asha_set, 1);
memcpy(set->hisyncid, asha->hisyncid, 8);
queue_push_tail(asha_devices, set);
DBG("Created ASHA set");
}
if (asha->right_side) {
set->right = asha;
DBG("Right side registered for ASHA set");
} else {
set->left = asha;
DBG("Left side registered for ASHA set");
}
} else {
if (!set) {
error("Missing ASHA set");
return;
}
if (asha->right_side && set->right) {
set->right = NULL;
DBG("Right side unregistered for ASHA set");
} else if (!asha->right_side && set->left) {
set->left = NULL;
DBG("Left side unregistered for ASHA set");
}
if (!set->right && !set->left) {
if (queue_remove(asha_devices, set)) {
free(set);
DBG("Freeing ASHA set");
}
if (!queue_peek_tail(asha_devices)) {
queue_destroy(asha_devices, NULL);
asha_devices = NULL;
}
}
}
}
static int asha_set_send_status(struct bt_asha *asha, bool other_connected)
{
struct bt_asha_set *set;
int ret = 0;
set = queue_find(asha_devices, match_hisyncid, asha);
if (set) {
if (asha->right_side && set->left) {
ret = bt_asha_status(set->left, other_connected);
DBG("ASHA left side update: %d, ret: %d",
other_connected, ret);
}
if (!asha->right_side && set->right) {
ret = bt_asha_status(set->right, other_connected);
DBG("ASHA right side update: %d, ret: %d",
other_connected, ret);
}
}
return ret;
}
struct bt_asha *bt_asha_new(void)
{
struct bt_asha *asha;
asha = new0(struct bt_asha, 1);
return asha;
}
void bt_asha_reset(struct bt_asha *asha)
{
if (asha->status_notify_id) {
bt_gatt_client_unregister_notify(asha->client,
asha->status_notify_id);
}
gatt_db_unref(asha->db);
asha->db = NULL;
bt_gatt_client_unref(asha->client);
asha->client = NULL;
bt_asha_state_reset(asha);
asha->psm = 0;
memset(asha->hisyncid, 0, sizeof(asha->hisyncid));
asha->attach_cb = NULL;
asha->attach_cb_data = NULL;
update_asha_set(asha, false);
}
void bt_asha_state_reset(struct bt_asha *asha)
{
asha->state = ASHA_STOPPED;
asha->state_cb = NULL;
asha->state_cb_data = NULL;
}
void bt_asha_free(struct bt_asha *asha)
{
update_asha_set(asha, false);
gatt_db_unref(asha->db);
bt_gatt_client_unref(asha->client);
free(asha);
}
static void asha_acp_sent(bool success, uint8_t err, void *user_data)
{
struct bt_asha *asha = user_data;
if (success) {
DBG("AudioControlPoint command successfully sent");
} else {
error("Failed to send AudioControlPoint command: %d", err);
if (asha->state_cb)
asha->state_cb(-1, asha->state_cb_data);
bt_asha_state_reset(asha);
}
}
static int asha_send_acp(struct bt_asha *asha, uint8_t *cmd,
unsigned int len, bt_asha_cb_t cb, void *user_data)
{
if (!bt_gatt_client_write_value(asha->client, asha->acp_handle, cmd,
len, asha_acp_sent, asha, NULL)) {
error("Error writing ACP command");
return -1;
}
asha->state_cb = cb;
asha->state_cb_data = user_data;
return 0;
}
static int asha_send_acp_without_response(struct bt_asha *asha,
uint8_t *cmd, unsigned int len)
{
if (!bt_gatt_client_write_without_response(asha->client,
asha->acp_handle, false, cmd, len)) {
error("Error writing ACP command");
return -1;
}
return 0;
}
unsigned int bt_asha_start(struct bt_asha *asha, bt_asha_cb_t cb,
void *user_data)
{
uint8_t other_connected = is_other_connected(asha);
uint8_t acp_start_cmd[] = {
0x01, /* START */
0x01, /* G.722, 16 kHz */
0, /* Unknown media type */
asha->volume, /* Volume */
other_connected,
};
int ret;
if (asha->state != ASHA_STOPPED) {
error("ASHA device start failed. Bad state %d", asha->state);
return 0;
}
ret = asha_send_acp(asha, acp_start_cmd, sizeof(acp_start_cmd), cb,
user_data);
if (ret < 0)
return ret;
asha->state = ASHA_STARTING;
return 0;
}
unsigned int bt_asha_stop(struct bt_asha *asha)
{
uint8_t acp_stop_cmd[] = {
0x02, /* STOP */
};
int ret;
if (asha->state != ASHA_STARTED)
return 0;
asha->state = ASHA_STOPPED;
ret = asha_send_acp(asha, acp_stop_cmd, sizeof(acp_stop_cmd), NULL,
NULL);
asha_set_send_status(asha, false);
/* We reset our state without waiting for a response */
bt_asha_state_reset(asha);
DBG("ASHA stop done");
return ret;
}
static unsigned int bt_asha_status(struct bt_asha *asha, bool other_connected)
{
uint8_t status = other_connected ? 1 : 0;
uint8_t acp_status_cmd[] = {
0x03, /* STATUS */
status,
};
int ret;
if (asha->state != ASHA_STARTED) {
const char *side = asha->right_side ? "right" : "left";
DBG("ASHA %s device not started for status update", side);
return 0;
}
ret = asha_send_acp_without_response(asha, acp_status_cmd,
sizeof(acp_status_cmd));
if (ret < 0)
return ret;
return 0;
}
bool bt_asha_set_volume(struct bt_asha *asha, int8_t volume)
{
if (!bt_gatt_client_write_without_response(asha->client,
asha->volume_handle, false,
(const uint8_t *)&volume, 1)) {
error("Error writing volume");
return false;
}
asha->volume = volume;
return true;
}
static bool uuid_cmp(const char *uuid1, const bt_uuid_t *uuid2)
{
bt_uuid_t lhs;
bt_string_to_uuid(&lhs, uuid1);
return bt_uuid_cmp(&lhs, uuid2) == 0;
}
static void check_probe_done(struct bt_asha *asha)
{
uint8_t zeroes[8] = { 0, };
/* Once we have ROPs & PSM, we should be good to go */
if (asha->psm == 0 || memcmp(asha->hisyncid, zeroes,
sizeof(zeroes)) == 0)
return;
if (asha->attach_cb) {
asha->attach_cb(asha->attach_cb_data);
asha->attach_cb = NULL;
asha->attach_cb_data = NULL;
}
}
static void read_psm(bool success,
uint8_t att_ecode,
const uint8_t *value,
uint16_t length,
void *user_data)
{
struct bt_asha *asha = user_data;
if (!success) {
DBG("Reading PSM failed with ATT error: %u", att_ecode);
return;
}
if (length != 2) {
DBG("Reading PSM failed: unexpected length %u", length);
return;
}
asha->psm = get_le16(value);
DBG("Got PSM: %u", asha->psm);
check_probe_done(asha);
}
static void read_rops(bool success,
uint8_t att_ecode,
const uint8_t *value,
uint16_t length,
void *user_data)
{
struct bt_asha *asha = user_data;
if (!success) {
DBG("Reading ROPs failed with ATT error: %u", att_ecode);
return;
}
if (length != 17) {
DBG("Reading ROPs failed: unexpected length %u", length);
return;
}
if (value[0] != 0x01) {
DBG("Unexpected ASHA version: %u", value[0]);
return;
}
/* Device Capabilities */
asha->right_side = (value[1] & 0x1) != 0;
asha->binaural = (value[1] & 0x2) != 0;
asha->csis_supported = (value[1] & 0x4) != 0;
/* HiSyncId: 2 byte company id, 6 byte ID shared by left and right */
memcpy(asha->hisyncid, &value[2], 8);
/* FeatureMap */
asha->coc_streaming_supported = (value[10] & 0x1) != 0;
/* RenderDelay */
asha->render_delay = get_le16(&value[11]);
/* byte 13 & 14 are reserved */
/* Codec IDs */
asha->codec_ids = get_le16(&value[15]);
DBG("Got ROPS: side %u, binaural %u, csis: %u, delay %u, codecs: %u",
asha->right_side, asha->binaural, asha->csis_supported,
asha->render_delay, asha->codec_ids);
check_probe_done(asha);
}
static void audio_status_register(uint16_t att_ecode, void *user_data)
{
if (att_ecode)
DBG("AudioStatusPoint register failed 0x%04x", att_ecode);
else
DBG("AudioStatusPoint register succeeded");
}
static void audio_status_notify(uint16_t value_handle, const uint8_t *value,
uint16_t length, void *user_data)
{
struct bt_asha *asha = user_data;
uint8_t status = *value;
/* Back these up to survive the reset paths */
bt_asha_cb_t state_cb = asha->state_cb;
bt_asha_cb_t state_cb_data = asha->state_cb_data;
DBG("ASHA status %u", status);
if (asha->state == ASHA_STARTING) {
if (status == 0) {
asha->state = ASHA_STARTED;
DBG("ASHA start complete");
update_asha_set(asha, true);
asha_set_send_status(asha, true);
} else {
bt_asha_state_reset(asha);
DBG("ASHA start failed");
}
}
if (state_cb) {
state_cb(status, state_cb_data);
asha->state_cb = NULL;
asha->state_cb_data = NULL;
}
}
static void handle_characteristic(struct gatt_db_attribute *attr,
void *user_data)
{
struct bt_asha *asha = user_data;
uint16_t value_handle;
bt_uuid_t uuid;
char uuid_str[MAX_LEN_UUID_STR];
if (!gatt_db_attribute_get_char_data(attr, NULL, &value_handle, NULL,
NULL, &uuid)) {
error("Failed to obtain characteristic data");
return;
}
bt_uuid_to_string(&uuid, uuid_str, sizeof(uuid_str));
if (uuid_cmp(ASHA_CHRC_LE_PSM_OUT_UUID, &uuid)) {
DBG("Got chrc %s/0x%x: LE_PSM_ID", uuid_str, value_handle);
if (!bt_gatt_client_read_value(asha->client, value_handle,
read_psm, asha, NULL))
DBG("Failed to send request to read battery level");
} else if (uuid_cmp(ASHA_CHRC_READ_ONLY_PROPERTIES_UUID, &uuid)) {
DBG("Got chrc %s/0x%x: READ_ONLY_PROPERTIES", uuid_str,
value_handle);
if (!bt_gatt_client_read_value(asha->client, value_handle,
read_rops, asha, NULL))
DBG("Failed to send request for readonly properties");
} else if (uuid_cmp(ASHA_CHRC_AUDIO_CONTROL_POINT_UUID, &uuid)) {
DBG("Got chrc %s/0x%x: AUDIO_CONTROL_POINT", uuid_str,
value_handle);
/* Store this for later writes */
asha->acp_handle = value_handle;
} else if (uuid_cmp(ASHA_CHRC_VOLUME_UUID, &uuid)) {
DBG("Got chrc %s/0x%x: VOLUME", uuid_str, value_handle);
/* Store this for later writes */
asha->volume_handle = value_handle;
} else if (uuid_cmp(ASHA_CHRC_AUDIO_STATUS_UUID, &uuid)) {
DBG("Got chrc %s/0x%x: AUDIO_STATUS", uuid_str, value_handle);
asha->status_notify_id =
bt_gatt_client_register_notify(asha->client,
value_handle, audio_status_register,
audio_status_notify, asha, NULL);
if (!asha->status_notify_id)
DBG("Failed to send request to notify AudioStatus");
} else {
DBG("Unsupported characteristic: %s", uuid_str);
}
}
static void foreach_asha_service(struct gatt_db_attribute *attr,
void *user_data)
{
struct bt_asha *asha = user_data;
DBG("Found ASHA GATT service");
asha->attr = attr;
gatt_db_service_set_claimed(attr, true);
gatt_db_service_foreach_char(asha->attr, handle_characteristic, asha);
}
bool bt_asha_attach(struct bt_asha *asha, struct gatt_db *db,
struct bt_gatt_client *client, bt_asha_attach_cb_t attach_cb,
void *cb_user_data)
{
bt_uuid_t asha_uuid;
asha->db = gatt_db_ref(db);
asha->client = bt_gatt_client_clone(client);
asha->attach_cb = attach_cb;
asha->attach_cb_data = cb_user_data;
bt_uuid16_create(&asha_uuid, ASHA_SERVICE);
gatt_db_foreach_service(db, &asha_uuid, foreach_asha_service, asha);
if (!asha->attr) {
error("ASHA attribute not found");
bt_asha_reset(asha);
return false;
}
if (!asha_devices)
asha_devices = queue_new();
return true;
}