public inbox for linux-bluetooth@vger.kernel.org
 help / color / mirror / Atom feed
From: Pauli Virtanen <pav@iki.fi>
To: "Frédéric Danis" <frederic.danis@collabora.com>,
	linux-bluetooth@vger.kernel.org
Subject: Re: [PATCH BlueZ 1/3] audio: Add ability to force CIS transport Links property
Date: Mon, 20 Apr 2026 19:32:35 +0300	[thread overview]
Message-ID: <af80f4c5bdab56eb525102ffcf3dbcd2f92ca06e.camel@iki.fi> (raw)
In-Reply-To: <20260420161345.145089-1-frederic.danis@collabora.com>

Hi,

ma, 2026-04-20 kello 18:13 +0200, Frédéric Danis kirjoitti:
> If bluetoothd is started in testing mode the Links property for CIS
> is readwrite and can be used to force transport objects Links.
> This can used to unlink transport objects by sending an empty array.
> 
> For unlinked transport objects, each transports needs to be acquired
> separately.
> 
> This allows to pass PTS tests BAP/UCL/STR/BV-543-C and BV-546-C.
> ---
>  profiles/audio/bap.c       | 44 +++++++++++++++++++++++++++++
>  profiles/audio/transport.c | 57 +++++++++++++++++++++++++++++++-------
>  2 files changed, 91 insertions(+), 10 deletions(-)
> 
> diff --git a/profiles/audio/bap.c b/profiles/audio/bap.c
> index 375026440..8f1f75240 100644
> --- a/profiles/audio/bap.c
> +++ b/profiles/audio/bap.c
> @@ -2651,13 +2651,57 @@ static void bap_connect_bcast_io_cb(GIOChannel *chan, GError *err,
>  	iso_connect_bcast_cb(chan, err, setup->stream);
>  }
>  
> +struct connect_io_data {
> +	GIOChannel *chan;
> +	GError *err;
> +	uint8_t cig_id;
> +};
> +
> +static bool find_enabling_stream(const void *data, const void *match_data)
> +{
> +	const struct bap_setup *setup = data;
> +	const struct connect_io_data *d = match_data;
> +	uint8_t state;
> +
> +	state = bt_bap_stream_get_state(setup->stream);
> +	if (state == BT_BAP_STREAM_STATE_ENABLING &&
> +			setup->qos.ucast.cig_id == d->cig_id)
> +		iso_connect_cb(d->chan, d->err, setup->stream);
> +
> +	return false;
> +}
> +
> +static bool find_enabling_ep(const void *data, const void *match_data)
> +{
> +	const struct bap_ep *ep = data;
> +
> +	if (ep->setups)
> +		queue_find(ep->setups, find_enabling_stream, match_data);
> +
> +	return false;
> +}
> +
>  static void bap_connect_io_cb(GIOChannel *chan, GError *err, gpointer user_data)
>  {
>  	struct bap_setup *setup = user_data;
> +	struct connect_io_data data;
>  
>  	if (!setup->stream)
>  		return;
>  
> +	if (bt_bap_stream_get_state(setup->stream) !=
> +			 BT_BAP_STREAM_STATE_ENABLING) {

The order of transition to STATE_STREAMING and iso_connect_cb() has
been reported to be nondeterministic 

https://github.com/bluez/bluez/issues/1506

and it appears in normal usage sometimes the state is STATE_STREAMING
here.

Probably the order in which the Glib mainloop handles the fd wakeups is
unspecified if the events arrive too close to each other, regardless of
the order in which kernel sees the events.

It's probably not right to assume here != ENABLING means the testing
feature is used, but it probably should check some extra unlink flag.

> +		/* The stream may have manually been unliked for tests and the
> +		 * connect event refers to another stream of the CIG.
> +		 */
> +		data.chan = chan;
> +		data.err = err;
> +		data.cig_id = setup->qos.ucast.cig_id;
> +		queue_find(setup->ep->data->snks, find_enabling_ep, &data);
> +		queue_find(setup->ep->data->srcs, find_enabling_ep, &data);
> +		return;
> +	}
> +
>  	iso_connect_cb(chan, err, setup->stream);
>  }
>  
> diff --git a/profiles/audio/transport.c b/profiles/audio/transport.c
> index d9feef768..794c492af 100644
> --- a/profiles/audio/transport.c
> +++ b/profiles/audio/transport.c
> @@ -37,6 +37,7 @@
>  #include "src/shared/bap.h"
>  #include "src/shared/bass.h"
>  #include "src/shared/io.h"
> +#include "src/btd.h"
>  
>  #ifdef HAVE_A2DP
>  #include "avdtp.h"
> @@ -114,6 +115,7 @@ struct bap_transport {
>  struct media_transport_ops {
>  	const char *uuid;
>  	const GDBusPropertyTable *properties;
> +	const GDBusPropertyTable *test_properties;
>  	void (*set_owner)(struct media_transport *transport,
>  				struct media_owner *owner);
>  	void (*remove_owner)(struct media_transport *transport,
> @@ -1417,6 +1419,9 @@ static struct media_transport *find_transport_by_path(const char *path)
>  	return NULL;
>  }
>  
> +static void bap_update_links(const struct media_transport *transport);
> +static void transport_unlink(void *data, void *user_data);
> +
>  static void set_links(const GDBusPropertyTable *property,
>  				DBusMessageIter *iter,
>  				GDBusPendingPropertySet id, void *user_data)
> @@ -1434,6 +1439,16 @@ static void set_links(const GDBusPropertyTable *property,
>  
>  	dbus_message_iter_recurse(iter, &array);
>  
> +	if (dbus_message_iter_get_arg_type(&array) == DBUS_TYPE_INVALID) {
> +		struct queue *links = bt_bap_stream_io_get_links(bap->stream);
> +
> +		/* Unlink stream from all its links */
> +		queue_foreach(links, transport_unlink, bap->stream);
> +
> +		bt_bap_stream_io_unlink(bap->stream, NULL);
> +		bap_update_links(transport);
> +	}
> +
>  	while (dbus_message_iter_get_arg_type(&array) ==
>  						DBUS_TYPE_OBJECT_PATH) {
>  		struct media_transport *link;
> @@ -1484,6 +1499,21 @@ static const GDBusPropertyTable transport_bap_uc_properties[] = {
>  	{ }
>  };
>  
> +static const GDBusPropertyTable transport_bap_uc_test_properties[] = {
> +	{ "Device", "o", get_device },
> +	{ "UUID", "s", get_uuid },
> +	{ "Codec", "y", get_codec },
> +	{ "Configuration", "ay", get_configuration },
> +	{ "State", "s", get_state },
> +	{ "QoS", "a{sv}", get_ucast_qos, NULL, qos_ucast_exists },
> +	{ "Endpoint", "o", get_endpoint, NULL, endpoint_exists },
> +	{ "Location", "u", get_location },
> +	{ "Metadata", "ay", get_metadata, set_metadata },
> +	{ "Links", "ao", get_links, set_links, links_exists },
> +	{ "Volume", "q", get_volume, set_volume, volume_exists },
> +	{ }
> +};
> +
>  static gboolean get_bcast_qos(const GDBusPropertyTable *property,
>  					DBusMessageIter *iter, void *data)
>  {
> @@ -1879,8 +1909,6 @@ static void bap_resume_complete(struct media_transport *transport)
>  	transport_set_state(transport, TRANSPORT_STATE_ACTIVE);
>  }
>  
> -static void bap_update_links(const struct media_transport *transport);
> -
>  static bool match_link_transport(const void *data, const void *user_data)
>  {
>  	const struct bt_bap_stream *stream = data;
> @@ -2535,10 +2563,11 @@ static void *transport_asha_init(struct media_transport *transport, void *data)
>  #define TRANSPORT_OPS(_uuid, _props, _set_owner, _remove_owner, _init, \
>  		      _resume, _suspend, _cancel, _set_state, _get_stream, \
>  		      _get_volume, _set_volume, _set_delay, _update_links, \
> -		      _destroy) \
> +		      _destroy, _test_props) \
>  { \
>  	.uuid = _uuid, \
>  	.properties = _props, \
> +	.test_properties = _test_props, \
>  	.set_owner = _set_owner, \
>  	.remove_owner = _remove_owner, \
>  	.init = _init, \
> @@ -2560,26 +2589,28 @@ static void *transport_asha_init(struct media_transport *transport, void *data)
>  			transport_a2dp_resume, transport_a2dp_suspend, \
>  			transport_a2dp_cancel, NULL, \
>  			transport_a2dp_get_stream, transport_a2dp_get_volume, \
> -			_set_volume, _set_delay, NULL, _destroy)
> +			_set_volume, _set_delay, NULL, _destroy, NULL)
>  
>  #define BAP_OPS(_uuid, _props, _set_owner, _remove_owner, _update_links, \
> -		_set_state) \
> +		_set_state, _test_props) \
>  	TRANSPORT_OPS(_uuid, _props, _set_owner, _remove_owner,\
>  			transport_bap_init, \
>  			transport_bap_resume, transport_bap_suspend, \
>  			transport_bap_cancel, _set_state, \
>  			transport_bap_get_stream, transport_bap_get_volume, \
>  			transport_bap_set_volume, NULL, \
> -			_update_links, transport_bap_destroy)
> +			_update_links, transport_bap_destroy, _test_props)
>  
>  #define BAP_UC_OPS(_uuid) \
>  	BAP_OPS(_uuid, transport_bap_uc_properties, \
>  			transport_bap_set_owner, transport_bap_remove_owner, \
> -			transport_bap_update_links_uc, transport_bap_set_state)
> +			transport_bap_update_links_uc, \
> +			transport_bap_set_state, \
> +			transport_bap_uc_test_properties)
>  
>  #define BAP_BC_OPS(_uuid) \
>  	BAP_OPS(_uuid, transport_bap_bc_properties, NULL, NULL, \
> -			transport_bap_update_links_bc, NULL)
> +			transport_bap_update_links_bc, NULL, NULL)
>  
>  #define ASHA_OPS(_uuid) \
>  	TRANSPORT_OPS(_uuid, transport_asha_properties, NULL, NULL, \
> @@ -2587,7 +2618,7 @@ static void *transport_asha_init(struct media_transport *transport, void *data)
>  			transport_asha_resume, transport_asha_suspend, \
>  			transport_asha_cancel, NULL, NULL, \
>  			transport_asha_get_volume, transport_asha_set_volume, \
> -			NULL, NULL, NULL)
> +			NULL, NULL, NULL, NULL)
>  
>  static const struct media_transport_ops transport_ops[] = {
>  #ifdef HAVE_A2DP
> @@ -2642,6 +2673,7 @@ struct media_transport *media_transport_create(struct btd_device *device,
>  	struct media_transport *transport;
>  	const struct media_transport_ops *ops;
>  	int fd;
> +	const GDBusPropertyTable *properties;
>  
>  	transport = g_new0(struct media_transport, 1);
>  	if (device)
> @@ -2696,9 +2728,14 @@ struct media_transport *media_transport_create(struct btd_device *device,
>  			goto fail;
>  	}
>  
> +	if (btd_opts.testing && ops->test_properties)
> +		properties = ops->test_properties;
> +	else
> +		properties = ops->properties;
> +
>  	if (g_dbus_register_interface(btd_get_dbus_connection(),
>  				transport->path, MEDIA_TRANSPORT_INTERFACE,
> -				transport_methods, NULL, ops->properties,
> +				transport_methods, NULL, properties,
>  				transport, media_transport_free) == FALSE) {
>  		error("Could not register transport %s", transport->path);
>  		goto fail;

-- 
Pauli Virtanen

  parent reply	other threads:[~2026-04-20 16:33 UTC|newest]

Thread overview: 7+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2026-04-20 16:13 [PATCH BlueZ 1/3] audio: Add ability to force CIS transport Links property Frédéric Danis
2026-04-20 16:13 ` [PATCH BlueZ 2/3] doc: Add documentation for readwrite " Frédéric Danis
2026-04-20 16:13 ` [PATCH BlueZ 3/3] client/player: Add support to unlink transports Frédéric Danis
2026-04-20 16:32 ` Pauli Virtanen [this message]
2026-04-20 17:44 ` [BlueZ,1/3] audio: Add ability to force CIS transport Links property bluez.test.bot
2026-04-20 19:52 ` [PATCH BlueZ 1/3] " Luiz Augusto von Dentz
2026-04-21  8:22   ` Frédéric Danis

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=af80f4c5bdab56eb525102ffcf3dbcd2f92ca06e.camel@iki.fi \
    --to=pav@iki.fi \
    --cc=frederic.danis@collabora.com \
    --cc=linux-bluetooth@vger.kernel.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox