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. 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