这是本文档旧的修订版!


SPICE 视频重定向的传输路径选择

  • 数据传输通过 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 的 视频流里?  有点复杂

 

  • 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 
  • public/it/spice/codec-agent-trans.1646015962.txt.gz
  • 最后更改: 2022/02/28 10:39
  • oakfire