Re: [PATCH spice-server v2 6/6] stream-channel: Send the full frame in a single message

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Wed, Feb 28, 2018 at 08:30:33AM +0000, Frediano Ziglio wrote:
> The current implementation of server and client assume that a single
> data message contains an encoded frame.
> This is not a problem for most encoding but for MJPEG this causes
> the client to fail decoding.

This commit really is missing a part about "how" it is achieving this.

Christophe

> 
> Signed-off-by: Frediano Ziglio <fziglio@xxxxxxxxxx>
> ---
>  server/stream-channel.c | 99 ++++++++++++++++++++++++++++++++++++++++++-------
>  server/stream-channel.h | 12 ++++++
>  server/stream-device.c  |  7 ++--
>  3 files changed, 101 insertions(+), 17 deletions(-)
> 
> diff --git a/server/stream-channel.c b/server/stream-channel.c
> index 88f859f6..3a3b733f 100644
> --- a/server/stream-channel.c
> +++ b/server/stream-channel.c
> @@ -44,6 +44,7 @@
>  
>  typedef struct StreamChannelClient StreamChannelClient;
>  typedef struct StreamChannelClientClass StreamChannelClientClass;
> +typedef struct StreamDataItem StreamDataItem;
>  
>  /* we need to inherit from CommonGraphicsChannelClient
>   * to get buffer handling */
> @@ -74,6 +75,10 @@ struct StreamChannel {
>  
>      StreamQueueStat queue_stat;
>  
> +    /* pending partial data item */
> +    StreamDataItem *data_item;
> +    uint32_t data_item_pos;
> +
>      /* callback to notify when a stream should be started or stopped */
>      stream_channel_start_proc start_cb;
>      void *start_opaque;
> @@ -104,12 +109,12 @@ typedef struct StreamCreateItem {
>      SpiceMsgDisplayStreamCreate stream_create;
>  } StreamCreateItem;
>  
> -typedef struct StreamDataItem {
> +struct StreamDataItem {
>      RedPipeItem base;
>      StreamChannel *channel;
>      // NOTE: this must be the last field in the structure
>      SpiceMsgDisplayStreamData data;
> -} StreamDataItem;
> +};
>  
>  #define PRIMARY_SURFACE_ID 0
>  
> @@ -129,6 +134,16 @@ stream_channel_client_init(StreamChannelClient *client)
>      client->stream_id = -1;
>  }
>  
> +static void
> +stream_channel_unref_data_item(StreamChannel *channel)
> +{
> +    if (channel->data_item) {
> +        red_pipe_item_unref(&channel->data_item->base);
> +        channel->data_item = NULL;
> +        channel->data_item_pos = 0;
> +    }
> +}
> +
>  static void
>  request_new_stream(StreamChannel *channel, StreamMsgStartStop *start)
>  {
> @@ -152,6 +167,7 @@ stream_channel_client_on_disconnect(RedChannelClient *rcc)
>      channel->stream_id = -1;
>      channel->width = 0;
>      channel->height = 0;
> +    stream_channel_unref_data_item(channel);
>  
>      // send stream stop to device
>      StreamMsgStartStop stop = { 0, };
> @@ -424,6 +440,16 @@ stream_channel_constructed(GObject *object)
>      reds_register_channel(reds, red_channel);
>  }
>  
> +static void
> +stream_channel_finalize(GObject *object)
> +{
> +    StreamChannel *channel = STREAM_CHANNEL(object);
> +
> +    stream_channel_unref_data_item(channel);
> +
> +    G_OBJECT_CLASS(stream_channel_parent_class)->finalize(object);
> +}
> +
>  static void
>  stream_channel_class_init(StreamChannelClass *klass)
>  {
> @@ -431,6 +457,7 @@ stream_channel_class_init(StreamChannelClass *klass)
>      RedChannelClass *channel_class = RED_CHANNEL_CLASS(klass);
>  
>      object_class->constructed = stream_channel_constructed;
> +    object_class->finalize = stream_channel_finalize;
>  
>      channel_class->parser = spice_get_client_channel_parser(SPICE_CHANNEL_DISPLAY, NULL);
>      channel_class->handle_message = handle_message;
> @@ -503,11 +530,46 @@ data_item_free(RedPipeItem *base)
>  {
>      StreamDataItem *pipe_item = SPICE_UPCAST(StreamDataItem, base);
>  
> -    stream_channel_update_queue_stat(pipe_item->channel, -1, -pipe_item->data.data_size);
> +    if (pipe_item->channel->data_item != pipe_item) {
> +        stream_channel_update_queue_stat(pipe_item->channel, -1, -pipe_item->data.data_size);
> +    }
>  
>      g_free(pipe_item);
>  }
>  
> +static StreamDataItem*
> +stream_channel_new_data_item(StreamChannel *channel, size_t size, uint32_t mm_time)
> +{
> +    stream_channel_unref_data_item(channel);
> +
> +    StreamDataItem *item = g_malloc(sizeof(*item) + size);
> +    red_pipe_item_init_full(&item->base, RED_PIPE_ITEM_TYPE_STREAM_DATA,
> +                            data_item_free);
> +    item->data.base.id = channel->stream_id;
> +    item->data.base.multi_media_time = mm_time;
> +    item->data.data_size = size;
> +    item->channel = channel;
> +
> +    channel->data_item = item;
> +    channel->data_item_pos = 0;
> +
> +    return item;
> +}
> +
> +void
> +stream_channel_start_data(StreamChannel *channel, size_t size, uint32_t mm_time)
> +{
> +    // see stream_channel_send_data comment
> +    if (channel->stream_id < 0) {
> +        return;
> +    }
> +
> +    // TODO this collects all chunks in a single message
> +    // up: we send a single frame together (more compatible)
> +    // down: guest can cause a crash due to DoS. As a safe measure we limit the maximum message
> +    stream_channel_new_data_item(channel, MIN(size, 32*1024*1024), mm_time);
> +}
> +
>  void
>  stream_channel_send_data(StreamChannel *channel, const void *data, size_t size, uint32_t mm_time)
>  {
> @@ -520,17 +582,25 @@ stream_channel_send_data(StreamChannel *channel, const void *data, size_t size,
>  
>      RedChannel *red_channel = RED_CHANNEL(channel);
>  
> -    StreamDataItem *item = g_malloc(sizeof(*item) + size);
> -    red_pipe_item_init_full(&item->base, RED_PIPE_ITEM_TYPE_STREAM_DATA,
> -                            data_item_free);
> -    item->data.base.id = channel->stream_id;
> -    item->data.base.multi_media_time = mm_time;
> -    item->data.data_size = size;
> -    item->channel = channel;
> -    stream_channel_update_queue_stat(channel, 1, size);
> -    // TODO try to optimize avoiding the copy
> -    memcpy(item->data.data, data, size);
> -    red_channel_pipes_add(red_channel, &item->base);
> +    while (size) {
> +        StreamDataItem *item = channel->data_item;
> +
> +        if (!item) {
> +            item = stream_channel_new_data_item(channel, size, mm_time);
> +        }
> +
> +        size_t copy_size = item->data.data_size - channel->data_item_pos;
> +        copy_size = MIN(copy_size, size);
> +        // TODO try to optimize avoiding the copy
> +        memcpy(item->data.data + channel->data_item_pos, data, copy_size);
> +        size -= copy_size;
> +        channel->data_item_pos += copy_size;
> +        if (channel->data_item_pos == item->data.data_size) {
> +            channel->data_item = NULL;
> +            stream_channel_update_queue_stat(channel, 1, item->data.data_size);
> +            red_channel_pipes_add(red_channel, &item->base);
> +        }
> +    }
>  }
>  
>  void
> @@ -570,6 +640,7 @@ stream_channel_reset(StreamChannel *channel)
>      channel->stream_id = -1;
>      channel->width = 0;
>      channel->height = 0;
> +    stream_channel_unref_data_item(channel);
>  
>      if (!red_channel_is_connected(red_channel)) {
>          return;
> diff --git a/server/stream-channel.h b/server/stream-channel.h
> index e8bec80b..18a1bdea 100644
> --- a/server/stream-channel.h
> +++ b/server/stream-channel.h
> @@ -60,6 +60,18 @@ struct StreamMsgStartStop;
>  
>  void stream_channel_change_format(StreamChannel *channel,
>                                    const struct StreamMsgFormat *fmt);
> +
> +/**
> + * Tell the channel that a new data packet is starting.
> + * This can be used to group all chunks together.
> + */
> +void stream_channel_start_data(StreamChannel *channel,
> +                               size_t size,
> +                               uint32_t mm_time);
> +
> +/**
> + * Send to channel a chunk of data.
> + */
>  void stream_channel_send_data(StreamChannel *channel,
>                                const void *data, size_t size,
>                                uint32_t mm_time);
> diff --git a/server/stream-device.c b/server/stream-device.c
> index abd198e4..bb7c9eff 100644
> --- a/server/stream-device.c
> +++ b/server/stream-device.c
> @@ -285,11 +285,12 @@ handle_msg_data(StreamDevice *dev, SpiceCharDeviceInstance *sin)
>          if (n <= 0) {
>              break;
>          }
> -        // TODO collect all message ??
> -        // up: we send a single frame together
> -        // down: guest can cause a crash
> +        if (dev->msg_pos == 0) {
> +            stream_channel_start_data(dev->stream_channel, dev->hdr.size, reds_get_mm_time());
> +        }
>          stream_channel_send_data(dev->stream_channel, buf, n, reds_get_mm_time());
>          dev->hdr.size -= n;
> +        dev->msg_pos += n;
>      }
>  
>      return dev->hdr.size == 0;
> -- 
> 2.14.3
> 
> _______________________________________________
> Spice-devel mailing list
> Spice-devel@xxxxxxxxxxxxxxxxxxxxx
> https://lists.freedesktop.org/mailman/listinfo/spice-devel

Attachment: signature.asc
Description: PGP signature

_______________________________________________
Spice-devel mailing list
Spice-devel@xxxxxxxxxxxxxxxxxxxxx
https://lists.freedesktop.org/mailman/listinfo/spice-devel

[Index of Archives]     [Linux ARM Kernel]     [Linux ARM]     [Linux Omap]     [Fedora ARM]     [IETF Annouce]     [Security]     [Bugtraq]     [Linux]     [Linux OMAP]     [Linux MIPS]     [ECOS]     [Asterisk Internet PBX]     [Linux API]     [Monitors]