From ead7f4028a1033d02e675e23c653f63218027349 Mon Sep 17 00:00:00 2001 From: Iulia Tanasescu Date: Fri, 13 Dec 2024 13:31:13 +0200 Subject: [PATCH] bap: Do not allow BIS QoS mismatch For Broadcast Source streams, QoS parameters are used for configuring the PA and BIG. All parameters provided to the LE Create BIG command are the same for all BISes (Core v5.3, vol.4, part E, page 2573). Likewise, since the PA train is associated with the BIG, it is unique for all included BISes. Thus, a stream should not be configured if the QoS parameters do not match the global BIG configuration. This commit adds a QoS check before a new stream is configured, to make sure that all streams share the same settings. --- profiles/audio/bap.c | 86 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 86 insertions(+) diff --git a/profiles/audio/bap.c b/profiles/audio/bap.c index f382b43a5..77df9455a 100644 --- a/profiles/audio/bap.c +++ b/profiles/audio/bap.c @@ -905,6 +905,83 @@ static void setup_free(void *data) free(setup); } +static bool match_io_qos(const struct bt_bap_io_qos *io_qos, + const struct bt_bap_io_qos *match) +{ + if (io_qos->interval != match->interval) + return false; + + if (io_qos->latency != match->latency) + return false; + + if (io_qos->sdu != match->sdu) + return false; + + if (io_qos->phy != match->phy) + return false; + + if (io_qos->rtn != match->rtn) + return false; + + return true; +} + +static bool match_bcast_qos(const struct bt_bap_bcast_qos *qos, + const struct bt_bap_bcast_qos *match) +{ + if (qos->sync_factor != match->sync_factor) + return false; + + if (qos->packing != match->packing) + return false; + + if (qos->framing != match->framing) + return false; + + if (qos->encryption != match->encryption) + return false; + + if (qos->encryption && util_iov_memcmp(qos->bcode, match->bcode)) + return false; + + if (qos->options != match->options) + return false; + + if (qos->skip != match->skip) + return false; + + if (qos->sync_timeout != match->sync_timeout) + return false; + + if (qos->sync_cte_type != match->sync_cte_type) + return false; + + if (qos->mse != match->mse) + return false; + + if (qos->timeout != match->timeout) + return false; + + if (qos->pa_sync != match->pa_sync) + return false; + + return match_io_qos(&qos->io_qos, &match->io_qos); +} + +static bool setup_mismatch_qos(const void *data, const void *user_data) +{ + const struct bap_setup *setup = data; + const struct bap_setup *match = user_data; + + /* Match setups that are part of the same BIG */ + if (setup == match || + setup->qos.bcast.big == BT_ISO_QOS_BIG_UNSET || + setup->qos.bcast.big != match->qos.bcast.big) + return false; + + return !match_bcast_qos(&setup->qos.bcast, &match->qos.bcast); +} + static DBusMessage *set_configuration(DBusConnection *conn, DBusMessage *msg, void *data) { @@ -937,6 +1014,15 @@ static DBusMessage *set_configuration(DBusConnection *conn, DBusMessage *msg, return btd_error_invalid_args(msg); } + if (bt_bap_pac_get_type(ep->lpac) == BT_BAP_BCAST_SOURCE) + /* All streams in a BIG should have the same QoS. + * Check that the new configuration matches previous ones. + */ + if (queue_find(setup->ep->setups, setup_mismatch_qos, setup)) { + setup_free(setup); + return btd_error_invalid_args(msg); + } + setup->stream = bt_bap_stream_new(ep->data->bap, ep->lpac, ep->rpac, &setup->qos, setup->caps); bt_bap_stream_set_user_data(setup->stream, ep->path); -- 2.47.3