Message ID | 20230531160631.612893-4-ernunes@redhat.com (mailing list archive) |
---|---|
State | New, archived |
Headers | show |
Series | vhost-user-gpu get_edid feature | expand |
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 --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);
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(-)