diff mbox series

[v2,3/4] contrib/vhost-user-gpu: implement get_edid feature

Message ID 20230531160631.612893-4-ernunes@redhat.com
State New
Headers show
Series vhost-user-gpu get_edid feature | expand

Commit Message

Erico Nunes May 31, 2023, 4:06 p.m. UTC
Implement the virtio-gpu feature in contrib/vhost-user-gpu, which was
unsupported until now.
In this implementation, the feature is enabled inconditionally to avoid
creating another optional config argument.
Similarly to get_display_info, vhost-user-gpu sends a message back to
the frontend to have access to all the display information. In the
case of get_edid, it also needs to pass which scanout we should
retrieve the edid for.

The VHOST_USER_GPU_PROTOCOL_F_EDID protocol feature is required if the
frontend sets the VIRTIO_GPU_F_EDID virtio-gpu feature. If the frontend
sets the virtio-gpu feature but does not support the protocol feature,
the backend will abort with an error.

Signed-off-by: Erico Nunes <ernunes@redhat.com>
---
 contrib/vhost-user-gpu/vhost-user-gpu.c | 67 +++++++++++++++++++++++--
 contrib/vhost-user-gpu/virgl.c          |  3 ++
 contrib/vhost-user-gpu/vugpu.h          | 11 ++++
 3 files changed, 77 insertions(+), 4 deletions(-)

Comments

Marc-André Lureau June 2, 2023, 6:30 a.m. UTC | #1
Hi

On Wed, May 31, 2023 at 8:07 PM Erico Nunes <ernunes@redhat.com> wrote:

> Implement the virtio-gpu feature in contrib/vhost-user-gpu, which was
> unsupported until now.
> In this implementation, the feature is enabled inconditionally to avoid
> creating another optional config argument.
> Similarly to get_display_info, vhost-user-gpu sends a message back to
> the frontend to have access to all the display information. In the
> case of get_edid, it also needs to pass which scanout we should
> retrieve the edid for.
>
> The VHOST_USER_GPU_PROTOCOL_F_EDID protocol feature is required if the
> frontend sets the VIRTIO_GPU_F_EDID virtio-gpu feature. If the frontend
> sets the virtio-gpu feature but does not support the protocol feature,
> the backend will abort with an error.
>
> Signed-off-by: Erico Nunes <ernunes@redhat.com>
> ---
>  contrib/vhost-user-gpu/vhost-user-gpu.c | 67 +++++++++++++++++++++++--
>  contrib/vhost-user-gpu/virgl.c          |  3 ++
>  contrib/vhost-user-gpu/vugpu.h          | 11 ++++
>  3 files changed, 77 insertions(+), 4 deletions(-)
>
> diff --git a/contrib/vhost-user-gpu/vhost-user-gpu.c
> b/contrib/vhost-user-gpu/vhost-user-gpu.c
> index bfb8d93cf8..8035dc71d7 100644
> --- a/contrib/vhost-user-gpu/vhost-user-gpu.c
> +++ b/contrib/vhost-user-gpu/vhost-user-gpu.c
> @@ -303,6 +303,53 @@ vg_get_display_info(VuGpu *vg, struct
> virtio_gpu_ctrl_command *cmd)
>      cmd->state = VG_CMD_STATE_PENDING;
>  }
>
> +static gboolean
> +get_edid_cb(gint fd, GIOCondition condition, gpointer user_data)
> +{
> +    struct virtio_gpu_resp_edid resp_edid;
> +    VuGpu *vg = user_data;
> +    struct virtio_gpu_ctrl_command *cmd = QTAILQ_LAST(&vg->fenceq);
> +
> +    g_debug("get edid cb");
> +    assert(cmd->cmd_hdr.type == VIRTIO_GPU_CMD_GET_EDID);
> +    if (!vg_recv_msg(vg, VHOST_USER_GPU_GET_EDID,
> +                     sizeof(resp_edid), &resp_edid)) {
> +        return G_SOURCE_CONTINUE;
> +    }
> +
> +    QTAILQ_REMOVE(&vg->fenceq, cmd, next);
> +    vg_ctrl_response(vg, cmd, &resp_edid.hdr, sizeof(resp_edid));
> +
> +    vg->wait_in = 0;
> +    vg_handle_ctrl(&vg->dev.parent, 0);
> +
> +    return G_SOURCE_REMOVE;
> +}
> +
> +void
> +vg_get_edid(VuGpu *vg, struct virtio_gpu_ctrl_command *cmd)
> +{
> +    struct virtio_gpu_cmd_get_edid get_edid;
> +
> +    VUGPU_FILL_CMD(get_edid);
> +    virtio_gpu_bswap_32(&get_edid, sizeof(get_edid));
> +
> +    VhostUserGpuMsg msg = {
> +        .request = VHOST_USER_GPU_GET_EDID,
> +        .size = sizeof(VhostUserGpuEdidRequest),
> +        .payload.edid_req = {
> +                .scanout_id = get_edid.scanout,
> +        },
> +    };
> +
> +    assert(vg->wait_in == 0);
> +
> +    vg_send_msg(vg, &msg, -1);
> +    vg->wait_in = g_unix_fd_add(vg->sock_fd, G_IO_IN | G_IO_HUP,
> +                               get_edid_cb, vg);
> +    cmd->state = VG_CMD_STATE_PENDING;
> +}
> +
>  static void
>  vg_resource_create_2d(VuGpu *g,
>                        struct virtio_gpu_ctrl_command *cmd)
> @@ -837,8 +884,9 @@ vg_process_cmd(VuGpu *vg, struct
> virtio_gpu_ctrl_command *cmd)
>      case VIRTIO_GPU_CMD_RESOURCE_DETACH_BACKING:
>          vg_resource_detach_backing(vg, cmd);
>          break;
> -    /* case VIRTIO_GPU_CMD_GET_EDID: */
> -    /*     break */
> +    case VIRTIO_GPU_CMD_GET_EDID:
> +        vg_get_edid(vg, cmd);
> +        break;
>      default:
>          g_warning("TODO handle ctrl %x\n", cmd->cmd_hdr.type);
>          cmd->error = VIRTIO_GPU_RESP_ERR_UNSPEC;
> @@ -1022,6 +1070,7 @@ vg_queue_set_started(VuDev *dev, int qidx, bool
> started)
>  static gboolean
>  protocol_features_cb(gint fd, GIOCondition condition, gpointer user_data)
>  {
> +    const uint64_t protocol_edid = (1 << VHOST_USER_GPU_PROTOCOL_F_EDID);
>      VuGpu *g = user_data;
>      uint64_t u64;
>      VhostUserGpuMsg msg = {
> @@ -1035,13 +1084,19 @@ protocol_features_cb(gint fd, GIOCondition
> condition, gpointer user_data)
>      msg = (VhostUserGpuMsg) {
>          .request = VHOST_USER_GPU_SET_PROTOCOL_FEATURES,
>          .size = sizeof(uint64_t),
> -        .payload.u64 = 0
> +        .payload.u64 = protocol_edid,
>


You should not set flags unsupported by the frontend (qemu). Use
"protocol_edid & u64". It might be worth renaming the "u64" variable too,
to "protocol_features" for ex. You could reuse that variable and udate with
"protocol_features &= (1 << VHOST_USER_GPU_PROTOCOL_F_EDID);"


>      };
>      vg_send_msg(g, &msg, -1);
>
>      g->wait_in = 0;
>      vg_handle_ctrl(&g->dev.parent, 0);
>
> +    if (g->edid_inited && !(u64 & protocol_edid)) {
> +        g_printerr("EDID feature set by the frontend but it does not
> support "
> +                   "the EDID vhost-user-gpu protocol.\n");
> +        exit(EXIT_FAILURE);
> +    }
> +
>      return G_SOURCE_REMOVE;
>  }
>
> @@ -1049,7 +1104,8 @@ static void
>  set_gpu_protocol_features(VuGpu *g)
>  {
>      VhostUserGpuMsg msg = {
> -        .request = VHOST_USER_GPU_GET_PROTOCOL_FEATURES
> +        .request = VHOST_USER_GPU_GET_PROTOCOL_FEATURES,
> +        .size = sizeof(uint64_t),
>

Why? there is no payload


>      };
>
>      vg_send_msg(g, &msg, -1);
> @@ -1086,6 +1142,7 @@ vg_get_features(VuDev *dev)
>      if (opt_virgl) {
>          features |= 1 << VIRTIO_GPU_F_VIRGL;
>      }
> +    features |= 1 << VIRTIO_GPU_F_EDID;
>

>      return features;
>  }
> @@ -1103,6 +1160,8 @@ vg_set_features(VuDev *dev, uint64_t features)
>          g->virgl_inited = true;
>      }
>
> +    g->edid_inited = !!(features & (1 << VIRTIO_GPU_F_EDID));
> +
>      g->virgl = virgl;
>  }
>
> diff --git a/contrib/vhost-user-gpu/virgl.c
> b/contrib/vhost-user-gpu/virgl.c
> index 3e45e1bd33..211aa110a9 100644
> --- a/contrib/vhost-user-gpu/virgl.c
> +++ b/contrib/vhost-user-gpu/virgl.c
> @@ -495,6 +495,9 @@ void vg_virgl_process_cmd(VuGpu *g, struct
> virtio_gpu_ctrl_command *cmd)
>      case VIRTIO_GPU_CMD_GET_DISPLAY_INFO:
>          vg_get_display_info(g, cmd);
>          break;
> +    case VIRTIO_GPU_CMD_GET_EDID:
> +        vg_get_edid(g, cmd);
> +        break;
>      default:
>          g_debug("TODO handle ctrl %x\n", cmd->cmd_hdr.type);
>          cmd->error = VIRTIO_GPU_RESP_ERR_UNSPEC;
> diff --git a/contrib/vhost-user-gpu/vugpu.h
> b/contrib/vhost-user-gpu/vugpu.h
> index e2864bba68..f0f2069c47 100644
> --- a/contrib/vhost-user-gpu/vugpu.h
> +++ b/contrib/vhost-user-gpu/vugpu.h
> @@ -36,6 +36,7 @@ typedef enum VhostUserGpuRequest {
>      VHOST_USER_GPU_UPDATE,
>      VHOST_USER_GPU_DMABUF_SCANOUT,
>      VHOST_USER_GPU_DMABUF_UPDATE,
> +    VHOST_USER_GPU_GET_EDID,
>  } VhostUserGpuRequest;
>
>  typedef struct VhostUserGpuDisplayInfoReply {
> @@ -83,6 +84,10 @@ typedef struct VhostUserGpuDMABUFScanout {
>      int fd_drm_fourcc;
>  } QEMU_PACKED VhostUserGpuDMABUFScanout;
>
> +typedef struct VhostUserGpuEdidRequest {
> +    uint32_t scanout_id;
> +} QEMU_PACKED VhostUserGpuEdidRequest;
> +
>  typedef struct VhostUserGpuMsg {
>      uint32_t request; /* VhostUserGpuRequest */
>      uint32_t flags;
> @@ -93,6 +98,8 @@ typedef struct VhostUserGpuMsg {
>          VhostUserGpuScanout scanout;
>          VhostUserGpuUpdate update;
>          VhostUserGpuDMABUFScanout dmabuf_scanout;
> +        VhostUserGpuEdidRequest edid_req;
> +        struct virtio_gpu_resp_edid resp_edid;
>          struct virtio_gpu_resp_display_info display_info;
>          uint64_t u64;
>      } payload;
> @@ -104,6 +111,8 @@ static VhostUserGpuMsg m __attribute__ ((unused));
>
>  #define VHOST_USER_GPU_MSG_FLAG_REPLY 0x4
>
> +#define VHOST_USER_GPU_PROTOCOL_F_EDID 0
> +
>  struct virtio_gpu_scanout {
>      uint32_t width, height;
>      int x, y;
> @@ -122,6 +131,7 @@ typedef struct VuGpu {
>
>      bool virgl;
>      bool virgl_inited;
> +    bool edid_inited;
>      uint32_t inflight;
>
>      struct virtio_gpu_scanout scanout[VIRTIO_GPU_MAX_SCANOUTS];
> @@ -171,6 +181,7 @@ int     vg_create_mapping_iov(VuGpu *g,
>                                struct iovec **iov);
>  void    vg_cleanup_mapping_iov(VuGpu *g, struct iovec *iov, uint32_t
> count);
>  void    vg_get_display_info(VuGpu *vg, struct virtio_gpu_ctrl_command
> *cmd);
> +void    vg_get_edid(VuGpu *vg, struct virtio_gpu_ctrl_command *cmd);
>
>  void    vg_wait_ok(VuGpu *g);
>
> --
> 2.40.1
>
>
>
diff mbox series

Patch

diff --git a/contrib/vhost-user-gpu/vhost-user-gpu.c b/contrib/vhost-user-gpu/vhost-user-gpu.c
index bfb8d93cf8..8035dc71d7 100644
--- a/contrib/vhost-user-gpu/vhost-user-gpu.c
+++ b/contrib/vhost-user-gpu/vhost-user-gpu.c
@@ -303,6 +303,53 @@  vg_get_display_info(VuGpu *vg, struct virtio_gpu_ctrl_command *cmd)
     cmd->state = VG_CMD_STATE_PENDING;
 }
 
+static gboolean
+get_edid_cb(gint fd, GIOCondition condition, gpointer user_data)
+{
+    struct virtio_gpu_resp_edid resp_edid;
+    VuGpu *vg = user_data;
+    struct virtio_gpu_ctrl_command *cmd = QTAILQ_LAST(&vg->fenceq);
+
+    g_debug("get edid cb");
+    assert(cmd->cmd_hdr.type == VIRTIO_GPU_CMD_GET_EDID);
+    if (!vg_recv_msg(vg, VHOST_USER_GPU_GET_EDID,
+                     sizeof(resp_edid), &resp_edid)) {
+        return G_SOURCE_CONTINUE;
+    }
+
+    QTAILQ_REMOVE(&vg->fenceq, cmd, next);
+    vg_ctrl_response(vg, cmd, &resp_edid.hdr, sizeof(resp_edid));
+
+    vg->wait_in = 0;
+    vg_handle_ctrl(&vg->dev.parent, 0);
+
+    return G_SOURCE_REMOVE;
+}
+
+void
+vg_get_edid(VuGpu *vg, struct virtio_gpu_ctrl_command *cmd)
+{
+    struct virtio_gpu_cmd_get_edid get_edid;
+
+    VUGPU_FILL_CMD(get_edid);
+    virtio_gpu_bswap_32(&get_edid, sizeof(get_edid));
+
+    VhostUserGpuMsg msg = {
+        .request = VHOST_USER_GPU_GET_EDID,
+        .size = sizeof(VhostUserGpuEdidRequest),
+        .payload.edid_req = {
+                .scanout_id = get_edid.scanout,
+        },
+    };
+
+    assert(vg->wait_in == 0);
+
+    vg_send_msg(vg, &msg, -1);
+    vg->wait_in = g_unix_fd_add(vg->sock_fd, G_IO_IN | G_IO_HUP,
+                               get_edid_cb, vg);
+    cmd->state = VG_CMD_STATE_PENDING;
+}
+
 static void
 vg_resource_create_2d(VuGpu *g,
                       struct virtio_gpu_ctrl_command *cmd)
@@ -837,8 +884,9 @@  vg_process_cmd(VuGpu *vg, struct virtio_gpu_ctrl_command *cmd)
     case VIRTIO_GPU_CMD_RESOURCE_DETACH_BACKING:
         vg_resource_detach_backing(vg, cmd);
         break;
-    /* case VIRTIO_GPU_CMD_GET_EDID: */
-    /*     break */
+    case VIRTIO_GPU_CMD_GET_EDID:
+        vg_get_edid(vg, cmd);
+        break;
     default:
         g_warning("TODO handle ctrl %x\n", cmd->cmd_hdr.type);
         cmd->error = VIRTIO_GPU_RESP_ERR_UNSPEC;
@@ -1022,6 +1070,7 @@  vg_queue_set_started(VuDev *dev, int qidx, bool started)
 static gboolean
 protocol_features_cb(gint fd, GIOCondition condition, gpointer user_data)
 {
+    const uint64_t protocol_edid = (1 << VHOST_USER_GPU_PROTOCOL_F_EDID);
     VuGpu *g = user_data;
     uint64_t u64;
     VhostUserGpuMsg msg = {
@@ -1035,13 +1084,19 @@  protocol_features_cb(gint fd, GIOCondition condition, gpointer user_data)
     msg = (VhostUserGpuMsg) {
         .request = VHOST_USER_GPU_SET_PROTOCOL_FEATURES,
         .size = sizeof(uint64_t),
-        .payload.u64 = 0
+        .payload.u64 = protocol_edid,
     };
     vg_send_msg(g, &msg, -1);
 
     g->wait_in = 0;
     vg_handle_ctrl(&g->dev.parent, 0);
 
+    if (g->edid_inited && !(u64 & protocol_edid)) {
+        g_printerr("EDID feature set by the frontend but it does not support "
+                   "the EDID vhost-user-gpu protocol.\n");
+        exit(EXIT_FAILURE);
+    }
+
     return G_SOURCE_REMOVE;
 }
 
@@ -1049,7 +1104,8 @@  static void
 set_gpu_protocol_features(VuGpu *g)
 {
     VhostUserGpuMsg msg = {
-        .request = VHOST_USER_GPU_GET_PROTOCOL_FEATURES
+        .request = VHOST_USER_GPU_GET_PROTOCOL_FEATURES,
+        .size = sizeof(uint64_t),
     };
 
     vg_send_msg(g, &msg, -1);
@@ -1086,6 +1142,7 @@  vg_get_features(VuDev *dev)
     if (opt_virgl) {
         features |= 1 << VIRTIO_GPU_F_VIRGL;
     }
+    features |= 1 << VIRTIO_GPU_F_EDID;
 
     return features;
 }
@@ -1103,6 +1160,8 @@  vg_set_features(VuDev *dev, uint64_t features)
         g->virgl_inited = true;
     }
 
+    g->edid_inited = !!(features & (1 << VIRTIO_GPU_F_EDID));
+
     g->virgl = virgl;
 }
 
diff --git a/contrib/vhost-user-gpu/virgl.c b/contrib/vhost-user-gpu/virgl.c
index 3e45e1bd33..211aa110a9 100644
--- a/contrib/vhost-user-gpu/virgl.c
+++ b/contrib/vhost-user-gpu/virgl.c
@@ -495,6 +495,9 @@  void vg_virgl_process_cmd(VuGpu *g, struct virtio_gpu_ctrl_command *cmd)
     case VIRTIO_GPU_CMD_GET_DISPLAY_INFO:
         vg_get_display_info(g, cmd);
         break;
+    case VIRTIO_GPU_CMD_GET_EDID:
+        vg_get_edid(g, cmd);
+        break;
     default:
         g_debug("TODO handle ctrl %x\n", cmd->cmd_hdr.type);
         cmd->error = VIRTIO_GPU_RESP_ERR_UNSPEC;
diff --git a/contrib/vhost-user-gpu/vugpu.h b/contrib/vhost-user-gpu/vugpu.h
index e2864bba68..f0f2069c47 100644
--- a/contrib/vhost-user-gpu/vugpu.h
+++ b/contrib/vhost-user-gpu/vugpu.h
@@ -36,6 +36,7 @@  typedef enum VhostUserGpuRequest {
     VHOST_USER_GPU_UPDATE,
     VHOST_USER_GPU_DMABUF_SCANOUT,
     VHOST_USER_GPU_DMABUF_UPDATE,
+    VHOST_USER_GPU_GET_EDID,
 } VhostUserGpuRequest;
 
 typedef struct VhostUserGpuDisplayInfoReply {
@@ -83,6 +84,10 @@  typedef struct VhostUserGpuDMABUFScanout {
     int fd_drm_fourcc;
 } QEMU_PACKED VhostUserGpuDMABUFScanout;
 
+typedef struct VhostUserGpuEdidRequest {
+    uint32_t scanout_id;
+} QEMU_PACKED VhostUserGpuEdidRequest;
+
 typedef struct VhostUserGpuMsg {
     uint32_t request; /* VhostUserGpuRequest */
     uint32_t flags;
@@ -93,6 +98,8 @@  typedef struct VhostUserGpuMsg {
         VhostUserGpuScanout scanout;
         VhostUserGpuUpdate update;
         VhostUserGpuDMABUFScanout dmabuf_scanout;
+        VhostUserGpuEdidRequest edid_req;
+        struct virtio_gpu_resp_edid resp_edid;
         struct virtio_gpu_resp_display_info display_info;
         uint64_t u64;
     } payload;
@@ -104,6 +111,8 @@  static VhostUserGpuMsg m __attribute__ ((unused));
 
 #define VHOST_USER_GPU_MSG_FLAG_REPLY 0x4
 
+#define VHOST_USER_GPU_PROTOCOL_F_EDID 0
+
 struct virtio_gpu_scanout {
     uint32_t width, height;
     int x, y;
@@ -122,6 +131,7 @@  typedef struct VuGpu {
 
     bool virgl;
     bool virgl_inited;
+    bool edid_inited;
     uint32_t inflight;
 
     struct virtio_gpu_scanout scanout[VIRTIO_GPU_MAX_SCANOUTS];
@@ -171,6 +181,7 @@  int     vg_create_mapping_iov(VuGpu *g,
                               struct iovec **iov);
 void    vg_cleanup_mapping_iov(VuGpu *g, struct iovec *iov, uint32_t count);
 void    vg_get_display_info(VuGpu *vg, struct virtio_gpu_ctrl_command *cmd);
+void    vg_get_edid(VuGpu *vg, struct virtio_gpu_ctrl_command *cmd);
 
 void    vg_wait_ok(VuGpu *g);