这是本文档旧的修订版!
SPICE 视频重定向的传输路径选择
1 借鉴
1.1 spice-stream-agent
- 数据传输通过 StreamDev,StreamDev 是 char_device 的一种,虚拟化上是 virtio spiceport类型设备;
- guest 与 server 之间的协议(stream-device.h)是额外自定添加的;不影响 server 与 client 之间的协议
- 在 server 与 client 之间没有增加新类型通道,stream-channel 实现的通道类型是 DisplayChannel
StreamChannel* stream_channel_new(RedsState *server, uint32_t id) { return g_object_new(TYPE_STREAM_CHANNEL, "spice-server", server, "core-interface", reds_get_core_interface(server), "channel-type", SPICE_CHANNEL_DISPLAY, // TODO this id should be after all qxl devices "id", id, "migration-flags", 0, "handle-acks", TRUE, // TODO sure ?? NULL); }
- 在 client 端,收到通道列表时就会额外创建一个屏幕来对应这个 DisplayChannel, 表现为有两个屏幕显示
- 发送创建 SPICE_MSG_DISPLAY_SURFACE_CREATE 消息时,标识 SPICE_SURFACE_FLAGS_STREAMING_MODE 来表示全屏流显示
// give an hint to client that we are sending just streaming // see spice.proto for capability check here if (red_channel_client_test_remote_cap(rcc, SPICE_DISPLAY_CAP_MULTI_CODEC)) { surface_create.flags |= SPICE_SURFACE_FLAGS_STREAMING_MODE; }
- server 与 client 之间传输视频流, SPICE_MSG_DISPLAY_STREAM_DATA
- 视频重定向如果直接借用这种方式来传输流,需要修改客户端对新增 DisplayChannel 的操作,魔改协议,得不偿失
- 问题:是否可以直接把数据插入到主 DisplayChannel 的 视频流里? 有点复杂
1.2 webdav
- guest 与 server 之间的传输方式是相同的,都是 spiceport virtio 设备
- server 与 client 之间传输新增了一个通道类型 WebDAVChannel,但 spice-protocol 里 只新增了通道类型标识 SPICE_CHANNEL_WEBDAV,没有新增其它消息 ;再看 WebDAVChannel 的声明如下:
channel PortChannel : SpicevmcChannel { client: message { uint8 event; } @declare event = 201; server: message { uint32 name_size; uint8 *name[name_size] @zero_terminated @marshall @nonnull; uint8 opened; } @declare init = 201; message { uint8 event; } @declare event; }; channel WebDAVChannel : PortChannel { };
可以看到 WebDAVChannel 协议是继承 PortChannel 通道协议,而且完全没有新增元素
- 查看 server 代码,是能自动识别创建任意名称的 PortChannel 通道:
else if (strcmp(char_device->subtype, SUBTYPE_PORT) == 0) { if (strcmp(char_device->portname, "org.spice-space.webdav.0") == 0) { dev_state = spicevmc_device_connect(reds, char_device, SPICE_CHANNEL_WEBDAV); } else if (strcmp(char_device->portname, "org.spice-space.stream.0") == 0) { dev_state = RED_CHAR_DEVICE(stream_device_connect(reds, char_device)); } else { dev_state = spicevmc_device_connect(reds, char_device, SPICE_CHANNEL_PORT); } }
- 那么传输路径与其复用WebDAVChannel 不如直接创建新名称的 PortChannel