[Mesa-dev] [PATCH v2 11/11] egl/wayland: Use linux-dmabuf interface for buffers

Jason Ekstrand jason at jlekstrand.net
Thu Jul 13 16:39:33 UTC 2017


On July 13, 2017 4:13:04 AM Daniel Stone <daniels at collabora.com> wrote:

> When available, use the zwp_linux_dambuf_v1 interface to create buffers,
> which allows multiple planes and buffer modifiers to be used.
>
> Signed-off-by: Daniel Stone <daniels at collabora.com>
> ---
>  configure.ac                            |   5 +-
>  src/egl/Makefile.am                     |  23 +++-
>  src/egl/drivers/dri2/.gitignore         |   2 +
>  src/egl/drivers/dri2/egl_dri2.c         |   7 ++
>  src/egl/drivers/dri2/egl_dri2.h         |  10 ++
>  src/egl/drivers/dri2/platform_wayland.c | 195 +++++++++++++++++++++++++++++---
>  6 files changed, 221 insertions(+), 21 deletions(-)
>  create mode 100644 src/egl/drivers/dri2/.gitignore
>
> v2: Add missing include, more comments.
>
> diff --git a/configure.ac b/configure.ac
> index 61d98e28e0..33cb749a5a 100644
> --- a/configure.ac
> +++ b/configure.ac
> @@ -88,6 +88,7 @@ LIBOMXIL_BELLAGIO_REQUIRED=0.0
>  LIBVA_REQUIRED=0.38.0
>  VDPAU_REQUIRED=1.1
>  WAYLAND_REQUIRED=1.11
> +WAYLAND_PROTOCOLS_REQUIRED=1.8
>  XCB_REQUIRED=1.9.3
>  XCBDRI2_REQUIRED=1.8
>  XCBGLX_REQUIRED=1.8.1
> @@ -1683,7 +1684,9 @@ for plat in $platforms; do
>  	case "$plat" in
>  	wayland)
>
> -		PKG_CHECK_MODULES([WAYLAND], [wayland-client >= $WAYLAND_REQUIRED 
> wayland-server >= $WAYLAND_REQUIRED])
> +		PKG_CHECK_MODULES([WAYLAND], [wayland-client >= $WAYLAND_REQUIRED 
> wayland-server >= $WAYLAND_REQUIRED wayland-protocols >= 
> $WAYLAND_PROTOCOLS_REQUIRED])
> +                ac_wayland_protocols_pkgdatadir=`$PKG_CONFIG 
> --variable=pkgdatadir wayland-protocols`
> +                AC_SUBST(WAYLAND_PROTOCOLS_DATADIR, 
> $ac_wayland_protocols_pkgdatadir)
>
>  		if test "x$WAYLAND_SCANNER" = "x:"; then
>  			AC_MSG_ERROR([wayland-scanner is needed to compile the wayland platform])
> diff --git a/src/egl/Makefile.am b/src/egl/Makefile.am
> index 81090387b5..19295de3ed 100644
> --- a/src/egl/Makefile.am
> +++ b/src/egl/Makefile.am
> @@ -21,6 +21,8 @@
>
>  include Makefile.sources
>
> +BUILT_SOURCES =
> +
>  AM_CFLAGS = \
>  	-I$(top_srcdir)/include \
>  	-I$(top_srcdir)/src/egl/main \
> @@ -61,11 +63,27 @@ endif
>  endif
>
>  if HAVE_PLATFORM_WAYLAND
> +WL_DMABUF_XML = 
> $(WAYLAND_PROTOCOLS_DATADIR)/unstable/linux-dmabuf/linux-dmabuf-unstable-v1.xml
> +
> +drivers/dri2/linux-dmabuf-unstable-v1-protocol.c: $(WL_DMABUF_XML)
> +	$(MKDIR_GEN)
> +	$(AM_V_GEN)$(WAYLAND_SCANNER) code < $< > $@
> +
> +drivers/dri2/linux-dmabuf-unstable-v1-client-protocol.h: $(WL_DMABUF_XML)
> +	$(MKDIR_GEN)
> +	$(AM_V_GEN)$(WAYLAND_SCANNER) client-header < $< > $@
> +
> +BUILT_SOURCES += \
> +	drivers/dri2/linux-dmabuf-unstable-v1-protocol.c \
> +	drivers/dri2/linux-dmabuf-unstable-v1-client-protocol.h
> +
>  AM_CFLAGS += $(WAYLAND_CFLAGS)
>  libEGL_common_la_LIBADD += $(WAYLAND_LIBS)
>  libEGL_common_la_LIBADD += $(LIBDRM_LIBS)
>  libEGL_common_la_LIBADD += 
>  $(top_builddir)/src/egl/wayland/wayland-drm/libwayland-drm.la
> -dri2_backend_FILES += drivers/dri2/platform_wayland.c
> +libEGL_common_la_LIBADD += $(top_builddir)/src/util/libmesautil.la
> +dri2_backend_FILES += drivers/dri2/platform_wayland.c	\
> +	drivers/dri2/linux-dmabuf-unstable-v1-protocol.c
>  endif
>
>  if HAVE_PLATFORM_DRM
> @@ -85,6 +103,7 @@ endif
>
>  AM_CFLAGS += \
>  	-I$(top_srcdir)/src/loader \
> +	-I$(top_builddir)/src/egl/drivers/dri2 \
>  	-I$(top_srcdir)/src/egl/drivers/dri2 \
>  	-I$(top_srcdir)/src/gbm/backends/dri \
>  	-I$(top_srcdir)/src/egl/wayland/wayland-egl \
> @@ -118,7 +137,7 @@ g_egldispatchstubs.h: $(GLVND_GEN_DEPS)
>  		$(top_srcdir)/src/egl/generate/egl.xml \
>  		$(top_srcdir)/src/egl/generate/egl_other.xml > $@
>
> -BUILT_SOURCES = g_egldispatchstubs.c g_egldispatchstubs.h
> +BUILT_SOURCES += g_egldispatchstubs.c g_egldispatchstubs.h
>  CLEANFILES = $(BUILT_SOURCES)
>
>  if USE_LIBGLVND
> diff --git a/src/egl/drivers/dri2/.gitignore b/src/egl/drivers/dri2/.gitignore
> new file mode 100644
> index 0000000000..e96becbb54
> --- /dev/null
> +++ b/src/egl/drivers/dri2/.gitignore
> @@ -0,0 +1,2 @@
> +linux-dmabuf-unstable-v1-client-protocol.h
> +linux-dmabuf-unstable-v1-protocol.c
> diff --git a/src/egl/drivers/dri2/egl_dri2.c b/src/egl/drivers/dri2/egl_dri2.c
> index 2392bfbb2a..a96a42ebb4 100644
> --- a/src/egl/drivers/dri2/egl_dri2.c
> +++ b/src/egl/drivers/dri2/egl_dri2.c
> @@ -53,6 +53,7 @@
>  #ifdef HAVE_WAYLAND_PLATFORM
>  #include "wayland-drm.h"
>  #include "wayland-drm-client-protocol.h"
> +#include "linux-dmabuf-unstable-v1-client-protocol.h"
>  #endif
>
>  #ifdef HAVE_X11_PLATFORM
> @@ -62,6 +63,7 @@
>  #include "egl_dri2.h"
>  #include "loader/loader.h"
>  #include "util/u_atomic.h"
> +#include "util/u_vector.h"
>
>  /* The kernel header drm_fourcc.h defines the DRM formats below.  We duplicate
>   * some of the definitions here so that building Mesa won't bleeding-edge
> @@ -934,11 +936,16 @@ dri2_display_destroy(_EGLDisplay *disp)
>     case _EGL_PLATFORM_WAYLAND:
>        if (dri2_dpy->wl_drm)
>            wl_drm_destroy(dri2_dpy->wl_drm);
> +      if (dri2_dpy->wl_dmabuf)
> +          zwp_linux_dmabuf_v1_destroy(dri2_dpy->wl_dmabuf);
>        if (dri2_dpy->wl_shm)
>            wl_shm_destroy(dri2_dpy->wl_shm);
>        wl_registry_destroy(dri2_dpy->wl_registry);
>        wl_event_queue_destroy(dri2_dpy->wl_queue);
>        wl_proxy_wrapper_destroy(dri2_dpy->wl_dpy_wrapper);
> +      u_vector_finish(&dri2_dpy->wl_modifiers.argb8888);
> +      u_vector_finish(&dri2_dpy->wl_modifiers.xrgb8888);
> +      u_vector_finish(&dri2_dpy->wl_modifiers.rgb565);
>        if (dri2_dpy->own_device) {
>           wl_display_disconnect(dri2_dpy->wl_dpy);
>        }
> diff --git a/src/egl/drivers/dri2/egl_dri2.h b/src/egl/drivers/dri2/egl_dri2.h
> index 4a5cf8e4ef..db19bb7809 100644
> --- a/src/egl/drivers/dri2/egl_dri2.h
> +++ b/src/egl/drivers/dri2/egl_dri2.h
> @@ -45,6 +45,8 @@
>  #ifdef HAVE_WAYLAND_PLATFORM
>  #include <wayland-client.h>
>  #include "wayland-egl-priv.h"
> +/* forward declarations of protocol elements */
> +struct zwp_linux_dmabuf_v1;
>  #endif
>
>  #include <GL/gl.h>
> @@ -73,6 +75,8 @@
>  #include "eglimage.h"
>  #include "eglsync.h"
>
> +#include "util/u_vector.h"
> +
>  struct wl_buffer;
>
>  struct dri2_egl_driver
> @@ -211,6 +215,12 @@ struct dri2_egl_display
>     struct wl_drm            *wl_drm;
>     struct wl_shm            *wl_shm;
>     struct wl_event_queue    *wl_queue;
> +   struct zwp_linux_dmabuf_v1 *wl_dmabuf;
> +   struct {
> +      struct u_vector        xrgb8888;
> +      struct u_vector        argb8888;
> +      struct u_vector        rgb565;
> +   } wl_modifiers;
>     bool                      authenticated;
>     int                       formats;
>     uint32_t                  capabilities;
> diff --git a/src/egl/drivers/dri2/platform_wayland.c 
> b/src/egl/drivers/dri2/platform_wayland.c
> index 3196b4cf64..5123383bdd 100644
> --- a/src/egl/drivers/dri2/platform_wayland.c
> +++ b/src/egl/drivers/dri2/platform_wayland.c
> @@ -36,14 +36,25 @@
>  #include <unistd.h>
>  #include <fcntl.h>
>  #include <xf86drm.h>
> +#include <drm_fourcc.h>
>  #include <sys/mman.h>
>
>  #include "egl_dri2.h"
>  #include "egl_dri2_fallbacks.h"
>  #include "loader.h"
> +#include "util/u_vector.h"
>
>  #include <wayland-client.h>
>  #include "wayland-drm-client-protocol.h"
> +#include "linux-dmabuf-unstable-v1-client-protocol.h"
> +
> +#ifndef DRM_FORMAT_MOD_INVALID
> +#define DRM_FORMAT_MOD_INVALID ((1ULL << 56) - 1)
> +#endif
> +
> +#ifndef DRM_FORMAT_MOD_LINEAR
> +#define DRM_FORMAT_MOD_LINEAR 0
> +#endif
>
>  enum wl_drm_format_flags {
>     HAS_ARGB8888 = 1,
> @@ -331,6 +342,8 @@ get_back_bo(struct dri2_egl_surface *dri2_surf)
>        dri2_egl_display(dri2_surf->base.Resource.Display);
>     int use_flags;
>     unsigned int dri_image_format;
> +   uint64_t *modifiers;
> +   int num_modifiers;
>
>     /* currently supports three WL DRM formats,
>      * WL_DRM_FORMAT_ARGB8888, WL_DRM_FORMAT_XRGB8888,
> @@ -339,12 +352,18 @@ get_back_bo(struct dri2_egl_surface *dri2_surf)
>     switch (dri2_surf->format) {
>     case WL_DRM_FORMAT_ARGB8888:
>        dri_image_format = __DRI_IMAGE_FORMAT_ARGB8888;
> +      modifiers = u_vector_tail(&dri2_dpy->wl_modifiers.argb8888);
> +      num_modifiers = u_vector_length(&dri2_dpy->wl_modifiers.argb8888);
>        break;
>     case WL_DRM_FORMAT_XRGB8888:
>        dri_image_format = __DRI_IMAGE_FORMAT_XRGB8888;
> +      modifiers = u_vector_tail(&dri2_dpy->wl_modifiers.xrgb8888);
> +      num_modifiers = u_vector_length(&dri2_dpy->wl_modifiers.xrgb8888);
>        break;
>     case WL_DRM_FORMAT_RGB565:
>        dri_image_format = __DRI_IMAGE_FORMAT_RGB565;
> +      modifiers = u_vector_tail(&dri2_dpy->wl_modifiers.rgb565);
> +      num_modifiers = u_vector_length(&dri2_dpy->wl_modifiers.rgb565);
>        break;
>     default:
>        /* format is not supported */
> @@ -382,27 +401,60 @@ get_back_bo(struct dri2_egl_surface *dri2_surf)
>
>     if (dri2_dpy->is_different_gpu &&
>         dri2_surf->back->linear_copy == NULL) {
> -       dri2_surf->back->linear_copy =
> -          dri2_dpy->image->createImage(dri2_dpy->dri_screen,
> -                                      dri2_surf->base.Width,
> -                                      dri2_surf->base.Height,
> -                                      dri_image_format,
> -                                      use_flags |
> -                                      __DRI_IMAGE_USE_LINEAR,
> -                                      NULL);
> +      /* The LINEAR modifier should be a perfect alias of the LINEAR use
> +       * flag; try the new interface first before the old, then fall back. */
> +      if (dri2_dpy->image->base.version >= 15 &&
> +           dri2_dpy->image->createImageWithModifiers) {
> +         uint64_t linear_mod = DRM_FORMAT_MOD_LINEAR;
> +
> +         dri2_surf->back->linear_copy =
> +            dri2_dpy->image->createImageWithModifiers(dri2_dpy->dri_screen,
> +                                                      dri2_surf->base.Width,
> +                                                      dri2_surf->base.Height,
> +                                                      dri_image_format,
> +                                                      &linear_mod,
> +                                                      1,
> +                                                      NULL);
> +      } else {
> +         dri2_surf->back->linear_copy =
> +            dri2_dpy->image->createImage(dri2_dpy->dri_screen,
> +                                         dri2_surf->base.Width,
> +                                         dri2_surf->base.Height,
> +                                         dri_image_format,
> +                                         use_flags |
> +                                         __DRI_IMAGE_USE_LINEAR,
> +                                         NULL);
> +      }
>        if (dri2_surf->back->linear_copy == NULL)
>            return -1;
>     }
>
>     if (dri2_surf->back->dri_image == NULL) {
> -      dri2_surf->back->dri_image =
> -         dri2_dpy->image->createImage(dri2_dpy->dri_screen,
> -                                      dri2_surf->base.Width,
> -                                      dri2_surf->base.Height,
> -                                      dri_image_format,
> -                                      dri2_dpy->is_different_gpu ?
> -                                         0 : use_flags,
> -                                      NULL);
> +      /* If our DRIImage implementation does not support
> +       * createImageWithModifiers, then fall back to the old createImage,
> +       * and hope it allocates an image which is acceptable to the winsys.
> +        */
> +      if (num_modifiers && dri2_dpy->image->base.version >= 15 &&
> +          dri2_dpy->image->createImageWithModifiers) {
> +         dri2_surf->back->dri_image =
> +           dri2_dpy->image->createImageWithModifiers(dri2_dpy->dri_screen,
> +                                                     dri2_surf->base.Width,
> +                                                     dri2_surf->base.Height,
> +                                                     dri_image_format,
> +                                                     modifiers,
> +                                                     num_modifiers,
> +                                                     NULL);
> +      } else {
> +         dri2_surf->back->dri_image =
> +            dri2_dpy->image->createImage(dri2_dpy->dri_screen,
> +                                         dri2_surf->base.Width,
> +                                         dri2_surf->base.Height,
> +                                         dri_image_format,
> +                                         dri2_dpy->is_different_gpu ?
> +                                              0 : use_flags,
> +                                         NULL);
> +      }
> +
>        dri2_surf->back->age = 0;
>     }
>     if (dri2_surf->back->dri_image == NULL)
> @@ -643,17 +695,68 @@ create_wl_buffer(struct dri2_egl_display *dri2_dpy,
>                   __DRIimage *image)
>  {
>     struct wl_buffer *ret;
> -   int width, height, fourcc;
> +   int width, height, fourcc, num_planes;
>
>     dri2_dpy->image->queryImage(image, __DRI_IMAGE_ATTRIB_WIDTH, &width);
>     dri2_dpy->image->queryImage(image, __DRI_IMAGE_ATTRIB_HEIGHT, &height);
>     dri2_dpy->image->queryImage(image, __DRI_IMAGE_ATTRIB_FOURCC, &fourcc);
> +   dri2_dpy->image->queryImage(image, __DRI_IMAGE_ATTRIB_NUM_PLANES,
> +                               &num_planes);
> +
> +   if (dri2_dpy->wl_dmabuf && dri2_dpy->image->base.version >= 15) {
> +      struct zwp_linux_buffer_params_v1 *params;
> +      int mod_hi, mod_lo;
> +      int i;
> +
> +      dri2_dpy->image->queryImage(image, __DRI_IMAGE_ATTRIB_MODIFIER_UPPER,
> +                                  &mod_hi);
> +      dri2_dpy->image->queryImage(image, __DRI_IMAGE_ATTRIB_MODIFIER_LOWER,
> +                                  &mod_lo);

What if the image wasn't actually created with modifiers?  Shouldn't we 
fall back to the old interface in that case?

> +
> +      /* We don't need a wrapper for wl_dmabuf objects, because we have to
> +       * create the intermediate params object; we can set the queue on this,
> +       * and the wl_buffer inherits it race-free. */
> +      params = zwp_linux_dmabuf_v1_create_params(dri2_dpy->wl_dmabuf);
> +      if (dri2_surf)
> +         wl_proxy_set_queue((struct wl_proxy *) params, dri2_surf->wl_queue);
> +
> +      for (i = 0; i < num_planes; i++) {
> +         __DRIimage *p_image;
> +         int stride, offset, fd;
> +
> +         if (i == 0)
> +            p_image = image;
> +         else
> +            p_image = dri2_dpy->image->fromPlanar(image, i, NULL);
> +         if (!p_image) {
> +            zwp_linux_buffer_params_v1_destroy(params);
> +            return NULL;
> +         }
>
> -   if (dri2_dpy->capabilities & WL_DRM_CAPABILITY_PRIME) {
> +         dri2_dpy->image->queryImage(p_image, __DRI_IMAGE_ATTRIB_STRIDE,
> +                                     &stride);
> +         dri2_dpy->image->queryImage(p_image, __DRI_IMAGE_ATTRIB_OFFSET,
> +                                     &offset);
> +         dri2_dpy->image->queryImage(p_image, __DRI_IMAGE_ATTRIB_FD, &fd);
> +         if (image != p_image)
> +            dri2_dpy->image->destroyImage(p_image);
> +
> +         zwp_linux_buffer_params_v1_add(params, fd, i, offset, stride,
> +                                        mod_hi, mod_lo);

I like the params interface...

> +         close(fd);
> +      }
> +
> +      ret = zwp_linux_buffer_params_v1_create_immed(params, width, height,
> +                                                    fourcc, 0);
> +      zwp_linux_buffer_params_v1_destroy(params);
> +   } else if (dri2_dpy->capabilities & WL_DRM_CAPABILITY_PRIME) {
>        struct wl_drm *wl_drm =
>           dri2_surf ? dri2_surf->wl_drm_wrapper : dri2_dpy->wl_drm;
>        int stride, fd;
>
> +      if (num_planes > 1)
> +         return NULL;
> +
>        dri2_dpy->image->queryImage(image, __DRI_IMAGE_ATTRIB_STRIDE, &stride);
>        dri2_dpy->image->queryImage(image, __DRI_IMAGE_ATTRIB_FD, &fd);
>        ret = wl_drm_create_prime_buffer(wl_drm, fd, width, height, fourcc, 0,
> @@ -664,6 +767,9 @@ create_wl_buffer(struct dri2_egl_display *dri2_dpy,
>           dri2_surf ? dri2_surf->wl_drm_wrapper : dri2_dpy->wl_drm;
>        int stride, name;
>
> +      if (num_planes > 1)
> +         return NULL;
> +
>        dri2_dpy->image->queryImage(image, __DRI_IMAGE_ATTRIB_STRIDE, &stride);
>        dri2_dpy->image->queryImage(image, __DRI_IMAGE_ATTRIB_NAME, &name);
>        ret = wl_drm_create_buffer(wl_drm, name, width, height, stride, fourcc);
> @@ -930,6 +1036,47 @@ static const struct wl_drm_listener drm_listener = {
>  };
>
>  static void
> +dmabuf_ignore_format(void *data, struct zwp_linux_dmabuf_v1 *dmabuf,
> +                     uint32_t format)
> +{
> +   /* formats are implicitly advertised by the 'modifier' event, so ignore */
> +}
> +
> +static void
> +dmabuf_handle_modifier(void *data, struct zwp_linux_dmabuf_v1 *dmabuf,
> +                       uint32_t format, uint32_t modifier_hi,
> +                       uint32_t modifier_lo)
> +{
> +   struct dri2_egl_display *dri2_dpy = data;
> +   uint64_t *mod = NULL;
> +
> +   switch (format) {
> +   case WL_DRM_FORMAT_ARGB8888:
> +      mod = u_vector_add(&dri2_dpy->wl_modifiers.argb8888);
> +      break;
> +   case WL_DRM_FORMAT_XRGB8888:
> +      mod = u_vector_add(&dri2_dpy->wl_modifiers.xrgb8888);
> +      break;
> +   case WL_DRM_FORMAT_RGB565:
> +      mod = u_vector_add(&dri2_dpy->wl_modifiers.rgb565);
> +      break;
> +   default:
> +      break;
> +   }
> +
> +   if (!mod)
> +      return;
> +
> +   *mod = (uint64_t) modifier_hi << 32;
> +   *mod |= (uint64_t) (modifier_lo & 0xffffffff);
> +}
> +
> +static const struct zwp_linux_dmabuf_v1_listener dmabuf_listener = {
> +   .format = dmabuf_ignore_format,
> +   .modifier = dmabuf_handle_modifier,

Pardon my ignorance but what happens if the modifiers available change?  
That doesn't seem to be handled at all here.  That's a rather important 
case for us because scanout only supports CCS_E on two planes and doesn't 
support Y-tiling on gen8 and earlier.  If this is going to get us the perf 
improvement we're looking for, then we need the ability to renegotiate when 
a surface goes fulalscreen.

> +};
> +
> +static void
>  registry_handle_global_drm(void *data, struct wl_registry *registry,
>                             uint32_t name, const char *interface,
>                             uint32_t version)
> @@ -940,6 +1087,12 @@ registry_handle_global_drm(void *data, struct 
> wl_registry *registry,
>        dri2_dpy->wl_drm =
>           wl_registry_bind(registry, name, &wl_drm_interface, MIN2(version, 2));
>        wl_drm_add_listener(dri2_dpy->wl_drm, &drm_listener, dri2_dpy);
> +   } else if (strcmp(interface, "zwp_linux_dmabuf_v1") == 0 && version >= 3) {
> +      dri2_dpy->wl_dmabuf =
> +         wl_registry_bind(registry, name, &zwp_linux_dmabuf_v1_interface,
> +                          MIN2(version, 3));
> +      zwp_linux_dmabuf_v1_add_listener(dri2_dpy->wl_dmabuf, &dmabuf_listener,
> +                                       dri2_dpy);
>     }
>  }
>
> @@ -1108,6 +1261,12 @@ dri2_initialize_wayland_drm(_EGLDriver *drv, 
> _EGLDisplay *disp)
>        dri2_dpy->wl_dpy = disp->PlatformDisplay;
>     }
>
> +   if (!u_vector_init(&dri2_dpy->wl_modifiers.xrgb8888, sizeof(uint64_t), 
> 32) ||
> +       !u_vector_init(&dri2_dpy->wl_modifiers.argb8888, sizeof(uint64_t), 
> 32) ||
> +       !u_vector_init(&dri2_dpy->wl_modifiers.rgb565, sizeof(uint64_t), 32)) {
> +      goto cleanup;
> +   }
> +
>     dri2_dpy->wl_queue = wl_display_create_queue(dri2_dpy->wl_dpy);
>
>     dri2_dpy->wl_dpy_wrapper = wl_proxy_create_wrapper(dri2_dpy->wl_dpy);
> --
> 2.13.0
>
> _______________________________________________
> mesa-dev mailing list
> mesa-dev at lists.freedesktop.org
> https://lists.freedesktop.org/mailman/listinfo/mesa-dev




More information about the mesa-dev mailing list