[Spice-devel] [PATCH v2 6/6] sound: don't store client in SndChannel
Frediano Ziglio
fziglio at redhat.com
Wed Apr 26 10:33:02 UTC 2017
>
> The base RedChannel already keeps a list of channel clients, so there's
> no need for the SndChannel to also keep track of the client itself.
>
> Since the SndChannel only supports a single client (whereas other
> channels may have some partial support for multiple clients), I've
> provided a convenience function for getting the client and warning if
> there is more than one.
>
> Signed-off-by: Jonathon Jongsma <jjongsma at redhat.com>
> ---
> server/sound.c | 74
> +++++++++++++++++++++++++++++++++-------------------------
> 1 file changed, 42 insertions(+), 32 deletions(-)
>
> diff --git a/server/sound.c b/server/sound.c
> index 543449a..ca83eae 100644
> --- a/server/sound.c
> +++ b/server/sound.c
> @@ -166,8 +166,6 @@ GType snd_channel_get_type(void) G_GNUC_CONST;
> struct SndChannel {
> RedChannel parent;
>
> - SndChannelClient *connection; /* Only one client is supported */
> -
> gboolean active;
> SpiceVolumeState volume;
> uint32_t frequency;
> @@ -240,6 +238,20 @@ static GList *snd_channels;
>
> static void snd_send(SndChannelClient * client);
>
> +static SndChannelClient *snd_channel_get_client(SndChannel *channel)
> +{
> + /* sound channels only support a single client */
> + GList *clients = red_channel_get_clients(RED_CHANNEL(channel));
> + if (clients == NULL)
> + return NULL;
> +
brackets
> + if (clients->next != NULL) {
> + g_warning("Only one sound client is supported");
> + }
> +
why we need to enforce this? I think for play should be easy to think
about multiple clients, is currently just an implementation limit, and
this looks like a regression to me.
For recording I have no idea, I see 2 options for multiple clients:
- only one can send data;
- mix together inputs.
> + return clients->data;
> +}
> +
> static RedsState* snd_channel_get_server(SndChannelClient *client)
> {
> g_return_val_if_fail(client != NULL, NULL);
> @@ -782,10 +794,6 @@ static bool
> snd_channel_client_config_socket(RedChannelClient *rcc)
>
> static void snd_channel_on_disconnect(RedChannelClient *rcc)
> {
> - SndChannel *channel = SND_CHANNEL(red_channel_client_get_channel(rcc));
> - if (channel->connection && rcc ==
> RED_CHANNEL_CLIENT(channel->connection)) {
> - channel->connection = NULL;
> - }
> }
>
> static uint8_t*
> @@ -821,7 +829,7 @@ static void snd_channel_set_volume(SndChannel *channel,
> uint8_t nchannels, uint16_t *volume)
> {
> SpiceVolumeState *st = &channel->volume;
> - SndChannelClient *client = channel->connection;
> + SndChannelClient *client = snd_channel_get_client(channel);
>
> st->volume_nchannels = nchannels;
> free(st->volume);
> @@ -844,7 +852,7 @@ SPICE_GNUC_VISIBLE void
> spice_server_playback_set_volume(SpicePlaybackInstance *
> static void snd_channel_set_mute(SndChannel *channel, uint8_t mute)
> {
> SpiceVolumeState *st = &channel->volume;
> - SndChannelClient *client = channel->connection;
> + SndChannelClient *client = snd_channel_get_client(channel);
>
> st->mute = mute;
>
> @@ -879,12 +887,12 @@ SPICE_GNUC_VISIBLE void
> spice_server_playback_start(SpicePlaybackInstance *sin)
> {
> SndChannel *channel = &sin->st->channel;
> channel->active = TRUE;
> - return snd_playback_start(channel->connection);
> + return snd_playback_start(snd_channel_get_client(channel));
> }
>
> SPICE_GNUC_VISIBLE void spice_server_playback_stop(SpicePlaybackInstance
> *sin)
> {
> - SndChannelClient *client = sin->st->channel.connection;
> + SndChannelClient *client = snd_channel_get_client(&sin->st->channel);
>
> sin->st->channel.active = FALSE;
> if (!client)
> @@ -912,7 +920,7 @@ SPICE_GNUC_VISIBLE void
> spice_server_playback_stop(SpicePlaybackInstance *sin)
> SPICE_GNUC_VISIBLE void
> spice_server_playback_get_buffer(SpicePlaybackInstance *sin,
> uint32_t **frame,
> uint32_t
> *num_samples)
> {
> - SndChannelClient *client = sin->st->channel.connection;
> + SndChannelClient *client = snd_channel_get_client(&sin->st->channel);
>
> *frame = NULL;
> *num_samples = 0;
> @@ -948,7 +956,7 @@ SPICE_GNUC_VISIBLE void
> spice_server_playback_put_samples(SpicePlaybackInstance
> }
> }
> playback_client = frame->client;
> - if (!playback_client || sin->st->channel.connection !=
> SND_CHANNEL_CLIENT(playback_client)) {
> + if (!playback_client || snd_channel_get_client(&sin->st->channel) !=
> SND_CHANNEL_CLIENT(playback_client)) {
> /* lost last reference, client has been destroyed previously */
> spice_debug("audio samples belong to a disconnected client");
> return;
> @@ -970,18 +978,19 @@ void snd_set_playback_latency(RedClient *client,
> uint32_t latency)
>
> for (l = snd_channels; l != NULL; l = l->next) {
> SndChannel *now = l->data;
> + SndChannelClient *scc = snd_channel_get_client(now);
> uint32_t type;
> g_object_get(RED_CHANNEL(now), "channel-type", &type, NULL);
> - if (type == SPICE_CHANNEL_PLAYBACK && now->connection &&
> -
> red_channel_client_get_client(RED_CHANNEL_CLIENT(now->connection))
> == client) {
> + if (type == SPICE_CHANNEL_PLAYBACK && scc &&
> + red_channel_client_get_client(RED_CHANNEL_CLIENT(scc)) ==
> client) {
>
> - if
> (red_channel_client_test_remote_cap(RED_CHANNEL_CLIENT(now->connection),
> + if (red_channel_client_test_remote_cap(RED_CHANNEL_CLIENT(scc),
> SPICE_PLAYBACK_CAP_LATENCY)) {
> - PlaybackChannelClient* playback =
> (PlaybackChannelClient*)now->connection;
> + PlaybackChannelClient* playback =
> (PlaybackChannelClient*)scc;
>
> playback->latency = latency;
> - snd_set_command(now->connection, SND_PLAYBACK_LATENCY_MASK);
> - snd_send(now->connection);
> + snd_set_command(scc, SND_PLAYBACK_LATENCY_MASK);
> + snd_send(scc);
> } else {
> spice_debug("client doesn't not support
> SPICE_PLAYBACK_CAP_LATENCY");
> }
> @@ -1062,7 +1071,6 @@ playback_channel_client_constructed(GObject *object)
> spice_debug("playback client %p using mode %s", playback_client,
> spice_audio_data_mode_to_string(playback_client->mode));
>
> - channel->connection = scc;
When is now this value initialized? I suppose after the constructor.
Does this cause no regressions?
> if (!red_client_during_migrate_at_target(red_client)) {
> snd_set_command(scc, SND_PLAYBACK_MODE_MASK);
> if (channel->volume.volume_nchannels) {
> @@ -1080,11 +1088,11 @@ static void snd_set_peer(RedChannel *red_channel,
> RedClient *client, RedsStream
> RedChannelCapabilities *caps, GType type)
> {
> SndChannel *channel = SND_CHANNEL(red_channel);
> - SndChannelClient *snd_client;
> + SndChannelClient *snd_client = snd_channel_get_client(channel);;
>
> - if (channel->connection) {
> -
> red_channel_client_disconnect(RED_CHANNEL_CLIENT(channel->connection));
> - channel->connection = NULL;
> + /* sound channels currently only support a single client */
> + if (snd_client) {
> + red_channel_client_disconnect(RED_CHANNEL_CLIENT(snd_client));
> }
>
> snd_client = g_initable_new(type,
> @@ -1145,12 +1153,12 @@ SPICE_GNUC_VISIBLE void
> spice_server_record_start(SpiceRecordInstance *sin)
> {
> SndChannel *channel = &sin->st->channel;
> channel->active = TRUE;
> - snd_record_start(channel->connection);
> + snd_record_start(snd_channel_get_client(channel));
> }
>
> SPICE_GNUC_VISIBLE void spice_server_record_stop(SpiceRecordInstance *sin)
> {
> - SndChannelClient *client = sin->st->channel.connection;
> + SndChannelClient *client = snd_channel_get_client(&sin->st->channel);
>
> sin->st->channel.active = FALSE;
> if (!client)
> @@ -1168,7 +1176,7 @@ SPICE_GNUC_VISIBLE void
> spice_server_record_stop(SpiceRecordInstance *sin)
> SPICE_GNUC_VISIBLE uint32_t
> spice_server_record_get_samples(SpiceRecordInstance *sin,
> uint32_t
> *samples,
> uint32_t
> bufsize)
> {
> - SndChannelClient *client = sin->st->channel.connection;
> + SndChannelClient *client = snd_channel_get_client(&sin->st->channel);
> uint32_t read_pos;
> uint32_t now;
> uint32_t len;
> @@ -1218,7 +1226,8 @@ static void snd_set_rate(SndChannel *channel, uint32_t
> frequency, uint32_t cap_o
>
> SPICE_GNUC_VISIBLE uint32_t
> spice_server_get_best_playback_rate(SpicePlaybackInstance *sin)
> {
> - return snd_get_best_rate(sin ? sin->st->channel.connection : NULL,
> SPICE_PLAYBACK_CAP_OPUS);
> + SndChannelClient *client = sin ?
> snd_channel_get_client(&sin->st->channel) : NULL;
> + return snd_get_best_rate(client, SPICE_PLAYBACK_CAP_OPUS);
> }
>
> SPICE_GNUC_VISIBLE void spice_server_set_playback_rate(SpicePlaybackInstance
> *sin, uint32_t frequency)
> @@ -1228,7 +1237,8 @@ SPICE_GNUC_VISIBLE void
> spice_server_set_playback_rate(SpicePlaybackInstance *si
>
> SPICE_GNUC_VISIBLE uint32_t
> spice_server_get_best_record_rate(SpiceRecordInstance *sin)
> {
> - return snd_get_best_rate(sin ? sin->st->channel.connection : NULL,
> SPICE_RECORD_CAP_OPUS);
> + SndChannelClient *client = sin ?
> snd_channel_get_client(&sin->st->channel) : NULL;
> + return snd_get_best_rate(client, SPICE_RECORD_CAP_OPUS);
> }
>
> SPICE_GNUC_VISIBLE void spice_server_set_record_rate(SpiceRecordInstance
> *sin, uint32_t frequency)
> @@ -1256,7 +1266,6 @@ record_channel_client_constructed(GObject *object)
>
> G_OBJECT_CLASS(record_channel_client_parent_class)->constructed(object);
>
> - channel->connection = scc;
same here, can cause a regression
> if (channel->volume.volume_nchannels) {
> snd_set_command(scc, SND_VOLUME_MUTE_MASK);
> }
> @@ -1443,10 +1452,11 @@ void snd_set_playback_compression(bool on)
>
> for (l = snd_channels; l != NULL; l = l->next) {
> SndChannel *now = l->data;
> + SndChannelClient *client = snd_channel_get_client(now);
> uint32_t type;
> g_object_get(RED_CHANNEL(now), "channel-type", &type, NULL);
> - if (type == SPICE_CHANNEL_PLAYBACK && now->connection) {
> - PlaybackChannelClient* playback =
> (PlaybackChannelClient*)now->connection;
> + if (type == SPICE_CHANNEL_PLAYBACK && client) {
> + PlaybackChannelClient* playback =
> PLAYBACK_CHANNEL_CLIENT(client);
> RedChannelClient *rcc = RED_CHANNEL_CLIENT(playback);
> bool client_can_celt = red_channel_client_test_remote_cap(rcc,
> SPICE_PLAYBACK_CAP_CELT_0_5_1);
> @@ -1456,7 +1466,7 @@ void snd_set_playback_compression(bool on)
> client_can_opus,
> client_can_celt);
> if (playback->mode != desired_mode) {
> playback->mode = desired_mode;
> - snd_set_command(now->connection, SND_PLAYBACK_MODE_MASK);
> + snd_set_command(client, SND_PLAYBACK_MODE_MASK);
> spice_debug("playback client %p using mode %s", playback,
> spice_audio_data_mode_to_string(playback->mode));
> }
Frediano
More information about the Spice-devel
mailing list