diff options
author | Robert Foss <[email protected]> | 2016-08-29 23:13:45 +0000 |
---|---|---|
committer | Robert Foss <[email protected]> | 2018-11-16 14:41:57 +0100 |
commit | d1a1c21e7621b5177febf191fcd3d3b8ef69dc96 (patch) | |
tree | f6d05d201f9f4125a9c5d0bb6039fec8ba495058 /src/gallium/winsys | |
parent | 0db898cef2f5a455138e5845689c075aadba1c1f (diff) |
virgl: native fence fd support
Following the support for fences on the virtio driver add support
for native fence on virgl. This was somewhat based on the freedeno one.
Signed-off-by: Gustavo Padovan <[email protected]>
Signed-off-by: Robert Foss <[email protected]>
Reviewed-by: Emil Velikov <[email protected]>
Diffstat (limited to 'src/gallium/winsys')
-rw-r--r-- | src/gallium/winsys/virgl/drm/virgl_drm_winsys.c | 84 | ||||
-rw-r--r-- | src/gallium/winsys/virgl/drm/virgl_drm_winsys.h | 2 | ||||
-rw-r--r-- | src/gallium/winsys/virgl/drm/virtgpu_drm.h | 14 | ||||
-rw-r--r-- | src/gallium/winsys/virgl/vtest/virgl_vtest_winsys.c | 10 |
4 files changed, 104 insertions, 6 deletions
diff --git a/src/gallium/winsys/virgl/drm/virgl_drm_winsys.c b/src/gallium/winsys/virgl/drm/virgl_drm_winsys.c index 98e0e99f661..5b604044598 100644 --- a/src/gallium/winsys/virgl/drm/virgl_drm_winsys.c +++ b/src/gallium/winsys/virgl/drm/virgl_drm_winsys.c @@ -38,11 +38,17 @@ #include "virgl/virgl_public.h" #include <xf86drm.h> +#include <libsync.h> #include "virtgpu_drm.h" #include "virgl_drm_winsys.h" #include "virgl_drm_public.h" + +#define VIRGL_DRM_VERSION(major, minor) ((major) << 16 | (minor)) +#define VIRGL_DRM_VERSION_FENCE_FD VIRGL_DRM_VERSION(1, 0) + + static inline boolean can_cache_resource(struct virgl_hw_res *res) { return res->cacheable == TRUE; @@ -70,6 +76,9 @@ static void virgl_hw_res_destroy(struct virgl_drm_winsys *qdws, if (res->ptr) os_munmap(res->ptr, res->size); + if (res->fence_fd != -1) + close(res->fence_fd); + memset(&args, 0, sizeof(args)); args.handle = res->bo_handle; drmIoctl(qdws->fd, DRM_IOCTL_GEM_CLOSE, &args); @@ -222,6 +231,7 @@ virgl_drm_winsys_resource_create(struct virgl_winsys *qws, res->stride = stride; pipe_reference_init(&res->reference, 1); res->num_cs_references = 0; + res->fence_fd = -1; return res; } @@ -457,6 +467,7 @@ virgl_drm_winsys_resource_create_handle(struct virgl_winsys *qws, res->stride = info_arg.stride; pipe_reference_init(&res->reference, 1); res->num_cs_references = 0; + res->fence_fd = -1; util_hash_table_set(qdws->bo_handles, (void *)(uintptr_t)handle, res); @@ -577,6 +588,7 @@ static struct virgl_cmd_buf *virgl_drm_cmd_buf_create(struct virgl_winsys *qws) } cbuf->base.buf = cbuf->buf; + cbuf->base.in_fence_fd = -1; return &cbuf->base; } @@ -687,7 +699,8 @@ static boolean virgl_drm_res_is_ref(struct virgl_winsys *qws, } static int virgl_drm_winsys_submit_cmd(struct virgl_winsys *qws, - struct virgl_cmd_buf *_cbuf) + struct virgl_cmd_buf *_cbuf, + int in_fence_fd, int *out_fence_fd) { struct virgl_drm_winsys *qdws = virgl_drm_winsys(qws); struct virgl_drm_cmd_buf *cbuf = virgl_drm_cmd_buf(_cbuf); @@ -702,12 +715,24 @@ static int virgl_drm_winsys_submit_cmd(struct virgl_winsys *qws, eb.size = cbuf->base.cdw * 4; eb.num_bo_handles = cbuf->cres; eb.bo_handles = (unsigned long)(void *)cbuf->res_hlist; + eb.fence_fd = -1; + + if (in_fence_fd != -1) { + eb.flags |= VIRTGPU_EXECBUF_FENCE_FD_IN; + eb.fence_fd = in_fence_fd; + } + + if (out_fence_fd != NULL) + eb.flags |= VIRTGPU_EXECBUF_FENCE_FD_OUT; ret = drmIoctl(qdws->fd, DRM_IOCTL_VIRTGPU_EXECBUFFER, &eb); if (ret == -1) fprintf(stderr,"got error from kernel - expect bad rendering %d\n", errno); cbuf->base.cdw = 0; + if (out_fence_fd != NULL) + *out_fence_fd = eb.fence_fd; + virgl_drm_release_all_res(qdws, cbuf); memset(cbuf->is_handle_added, 0, sizeof(cbuf->is_handle_added)); @@ -759,7 +784,7 @@ static int handle_compare(void *key1, void *key2) } static struct pipe_fence_handle * -virgl_cs_create_fence(struct virgl_winsys *vws) +virgl_cs_create_fence(struct virgl_winsys *vws, int fd) { struct virgl_hw_res *res; @@ -769,6 +794,7 @@ virgl_cs_create_fence(struct virgl_winsys *vws) VIRGL_BIND_CUSTOM, 8, 1, 1, 0, 0, 0, 8); + res->fence_fd = fd; return (struct pipe_fence_handle *)res; } @@ -793,6 +819,12 @@ static bool virgl_fence_wait(struct virgl_winsys *vws, return TRUE; } virgl_drm_resource_wait(vws, res); + + if (res->fence_fd != -1) { + int ret = sync_wait(res->fence_fd, timeout / 1000000); + return ret == 0; + } + return TRUE; } @@ -805,11 +837,51 @@ static void virgl_fence_reference(struct virgl_winsys *vws, virgl_hw_res(src)); } +static void virgl_fence_server_sync(struct virgl_winsys *vws, + struct virgl_cmd_buf *cbuf, + struct pipe_fence_handle *fence) +{ + struct virgl_hw_res *hw_res = virgl_hw_res(fence); + + /* if not an external fence, then nothing more to do without preemption: */ + if (hw_res->fence_fd == -1) + return; + + sync_accumulate("virgl", &cbuf->in_fence_fd, hw_res->fence_fd); +} + +static int virgl_fence_get_fd(struct virgl_winsys *vws, + struct pipe_fence_handle *fence) +{ + struct virgl_hw_res *hw_res = virgl_hw_res(fence); + + return dup(hw_res->fence_fd); +} + +static int virgl_drm_get_version(int fd) +{ + int ret; + drmVersionPtr version; + + version = drmGetVersion(fd); + + if (!version) + ret = -EFAULT; + else if (version->version_major != 0) + ret = -EINVAL; + else + ret = version->version_minor; + + drmFreeVersion(version); + + return ret; +} static struct virgl_winsys * virgl_drm_winsys_create(int drmFD) { struct virgl_drm_winsys *qdws; + int drm_version; int ret; int gl = 0; struct drm_virtgpu_getparam getparam = {0}; @@ -820,6 +892,10 @@ virgl_drm_winsys_create(int drmFD) if (ret < 0 || !gl) return NULL; + drm_version = virgl_drm_get_version(qdws->fd); + if (drm_version < 0) + return NULL; + qdws = CALLOC_STRUCT(virgl_drm_winsys); if (!qdws) return NULL; @@ -851,9 +927,13 @@ virgl_drm_winsys_create(int drmFD) qdws->base.cs_create_fence = virgl_cs_create_fence; qdws->base.fence_wait = virgl_fence_wait; qdws->base.fence_reference = virgl_fence_reference; + qdws->base.fence_server_sync = virgl_fence_server_sync; + qdws->base.fence_get_fd = virgl_fence_get_fd; + qdws->base.supports_fences = drm_version >= VIRGL_DRM_VERSION_FENCE_FD; qdws->base.get_caps = virgl_drm_get_caps; + uint32_t value = 0; getparam.param = VIRTGPU_PARAM_CAPSET_QUERY_FIX; getparam.value = (uint64_t)(uintptr_t)&value; diff --git a/src/gallium/winsys/virgl/drm/virgl_drm_winsys.h b/src/gallium/winsys/virgl/drm/virgl_drm_winsys.h index b28e7127ca0..4316a74977d 100644 --- a/src/gallium/winsys/virgl/drm/virgl_drm_winsys.h +++ b/src/gallium/winsys/virgl/drm/virgl_drm_winsys.h @@ -50,12 +50,14 @@ struct virgl_hw_res { int64_t start, end; boolean flinked; uint32_t flink; + int fence_fd; }; struct virgl_drm_winsys { struct virgl_winsys base; int fd; + int drm_version; struct list_head delayed; int num_delayed; unsigned usecs; diff --git a/src/gallium/winsys/virgl/drm/virtgpu_drm.h b/src/gallium/winsys/virgl/drm/virtgpu_drm.h index 8596febe9fd..4be22aa077c 100644 --- a/src/gallium/winsys/virgl/drm/virtgpu_drm.h +++ b/src/gallium/winsys/virgl/drm/virtgpu_drm.h @@ -44,6 +44,16 @@ #define DRM_VIRTGPU_WAIT 0x08 #define DRM_VIRTGPU_GET_CAPS 0x09 +/* + * virtgpu execbuffer flags + */ +#define VIRTGPU_EXECBUF_FENCE_FD_IN 0x01 +#define VIRTGPU_EXECBUF_FENCE_FD_OUT 0x02 +#define VIRTGPU_EXECBUF_FLAGS (\ + VIRTGPU_EXECBUF_FENCE_FD_IN |\ + VIRTGPU_EXECBUF_FENCE_FD_OUT |\ + 0) + struct drm_virtgpu_map { uint64_t offset; /* use for mmap system call */ uint32_t handle; @@ -56,7 +66,7 @@ struct drm_virtgpu_execbuffer { uint64_t command; /* void* */ uint64_t bo_handles; uint32_t num_bo_handles; - uint32_t pad; + int32_t fence_fd; }; #define VIRTGPU_PARAM_3D_FEATURES 1 /* do we have 3D features in the hw */ @@ -130,7 +140,7 @@ struct drm_virtgpu_get_caps { DRM_IOWR(DRM_COMMAND_BASE + DRM_VIRTGPU_MAP, struct drm_virtgpu_map) #define DRM_IOCTL_VIRTGPU_EXECBUFFER \ - DRM_IOW(DRM_COMMAND_BASE + DRM_VIRTGPU_EXECBUFFER,\ + DRM_IOWR(DRM_COMMAND_BASE + DRM_VIRTGPU_EXECBUFFER,\ struct drm_virtgpu_execbuffer) #define DRM_IOCTL_VIRTGPU_GETPARAM \ diff --git a/src/gallium/winsys/virgl/vtest/virgl_vtest_winsys.c b/src/gallium/winsys/virgl/vtest/virgl_vtest_winsys.c index f44d4d74ff1..c1c88151ad3 100644 --- a/src/gallium/winsys/virgl/vtest/virgl_vtest_winsys.c +++ b/src/gallium/winsys/virgl/vtest/virgl_vtest_winsys.c @@ -501,7 +501,8 @@ static void virgl_vtest_add_res(struct virgl_vtest_winsys *vtws, } static int virgl_vtest_winsys_submit_cmd(struct virgl_winsys *vws, - struct virgl_cmd_buf *_cbuf) + struct virgl_cmd_buf *_cbuf, + int in_fence_fd, int *out_fence_fd) { struct virgl_vtest_winsys *vtws = virgl_vtest_winsys(vws); struct virgl_vtest_cmd_buf *cbuf = virgl_vtest_cmd_buf(_cbuf); @@ -510,6 +511,9 @@ static int virgl_vtest_winsys_submit_cmd(struct virgl_winsys *vws, if (cbuf->base.cdw == 0) return 0; + assert(in_fence_fd == -1); + assert(out_fence_fd == NULL); + ret = virgl_vtest_submit_cmd(vtws, cbuf); virgl_vtest_release_all_res(vtws, cbuf); @@ -552,7 +556,7 @@ static int virgl_vtest_get_caps(struct virgl_winsys *vws, } static struct pipe_fence_handle * -virgl_cs_create_fence(struct virgl_winsys *vws) +virgl_cs_create_fence(struct virgl_winsys *vws, int fd) { struct virgl_hw_res *res; @@ -694,8 +698,10 @@ virgl_vtest_winsys_wrap(struct sw_winsys *sws) vtws->base.cs_create_fence = virgl_cs_create_fence; vtws->base.fence_wait = virgl_fence_wait; vtws->base.fence_reference = virgl_fence_reference; + vtws->base.supports_fences = 0; vtws->base.flush_frontbuffer = virgl_vtest_flush_frontbuffer; + return &vtws->base; } |