Blame SOURCES/582b2d345abaa0e313cf16c902e602084ea59551.patch

bdd22e
From 582b2d345abaa0e313cf16c902e602084ea59551 Mon Sep 17 00:00:00 2001
bdd22e
From: Erik Kurzinger <ekurzinger@nvidia.com>
bdd22e
Date: Tue, 23 Nov 2021 14:15:14 -0500
bdd22e
Subject: [PATCH] egl-wayland: retrieve DRM device name before acquiring API
bdd22e
 lock
bdd22e
bdd22e
wlEglBindDisplaysHook acquires the external API lock before calling
bdd22e
wl_eglstream_display_bind, which in turn calls wl_drm_display_bind. That
bdd22e
function calls back into EGL to query the DRM device associated with the
bdd22e
given EGLDisplay.
bdd22e
bdd22e
Normally this is not a problem since the EGLDisplay passed to
bdd22e
eglBindWaylandDisplayWL will be an internal EGL_PLATFORM_DEVICE handle.
bdd22e
However, some applications, notably anything WebKit-based, will instead
bdd22e
pass in an external EGL_PLATFORM_WAYLAND handle. This means that the
bdd22e
eglQueryDisplayAttrib call by wl_drm_display_bind will require EGL to
bdd22e
call back into the egl-wayland library to look up the internal handle.
bdd22e
This is done by wlEglGetInternalHandleExport, which will attempt to
bdd22e
acquire the external API lock a second time, which will fail.
bdd22e
bdd22e
To avoid this, add a new function to wayland-drm.c which will retrieve
bdd22e
the DRM device name for the given EGLDisplay. wlEglBindDisplaysHook will
bdd22e
call this *before* acquiring the external API lock, and then pass it to
bdd22e
wl_drm_display_bind via wl_eglstream_display_bind so it can be saved in
bdd22e
the wl_eglstream_display struct.
bdd22e
---
bdd22e
 include/wayland-drm.h              |  7 ++++++-
bdd22e
 include/wayland-eglstream-server.h |  3 ++-
bdd22e
 src/wayland-drm.c                  | 33 +++++++++++++++---------------
bdd22e
 src/wayland-egldisplay.c           |  8 +++++---
bdd22e
 src/wayland-eglstream-server.c     |  5 +++--
bdd22e
 5 files changed, 32 insertions(+), 24 deletions(-)
bdd22e
bdd22e
diff --git a/include/wayland-drm.h b/include/wayland-drm.h
bdd22e
index be363c6..84d0f11 100644
bdd22e
--- a/include/wayland-drm.h
bdd22e
+++ b/include/wayland-drm.h
bdd22e
@@ -23,9 +23,14 @@
bdd22e
 #ifndef WAYLAND_DRM_H
bdd22e
 #define WAYLAND_DRM_H
bdd22e
 
bdd22e
+extern const char *
bdd22e
+wl_drm_get_dev_name(const WlEglPlatformData *data,
bdd22e
+                    EGLDisplay dpy);
bdd22e
+
bdd22e
 extern EGLBoolean
bdd22e
 wl_drm_display_bind(struct wl_display *display,
bdd22e
-                    struct wl_eglstream_display *wlStreamDpy);
bdd22e
+                    struct wl_eglstream_display *wlStreamDpy,
bdd22e
+                    const char *dev_name);
bdd22e
 extern void
bdd22e
 wl_drm_display_unbind(struct wl_eglstream_display *wlStreamDpy);
bdd22e
 
bdd22e
diff --git a/include/wayland-eglstream-server.h b/include/wayland-eglstream-server.h
bdd22e
index 76e772c..0f7d477 100644
bdd22e
--- a/include/wayland-eglstream-server.h
bdd22e
+++ b/include/wayland-eglstream-server.h
bdd22e
@@ -49,7 +49,8 @@ EGLBoolean
bdd22e
 wl_eglstream_display_bind(WlEglPlatformData *data,
bdd22e
                           struct wl_display *wlDisplay,
bdd22e
                           EGLDisplay eglDisplay,
bdd22e
-                          const char *exts);
bdd22e
+                          const char *exts,
bdd22e
+                          const char *dev_name);
bdd22e
 
bdd22e
 /*
bdd22e
  * wl_eglstream_display_unbind()
bdd22e
diff --git a/src/wayland-drm.c b/src/wayland-drm.c
bdd22e
index aa6de23..a08d82f 100644
bdd22e
--- a/src/wayland-drm.c
bdd22e
+++ b/src/wayland-drm.c
bdd22e
@@ -152,37 +152,36 @@ bind(struct wl_client *client, void *data, uint32_t version, uint32_t id)
bdd22e
         wl_resource_post_event(resource, WL_DRM_CAPABILITIES, 0);
bdd22e
 }
bdd22e
 
bdd22e
-EGLBoolean
bdd22e
-wl_drm_display_bind(struct wl_display *display,
bdd22e
-                    struct wl_eglstream_display *wlStreamDpy)
bdd22e
+const char *
bdd22e
+wl_drm_get_dev_name(const WlEglPlatformData *data,
bdd22e
+                    EGLDisplay dpy)
bdd22e
 {
bdd22e
-    EGLDisplay dpy = wlStreamDpy->eglDisplay;
bdd22e
     EGLDeviceEXT egl_dev;
bdd22e
     const char *dev_exts;
bdd22e
-    const char *dev_name;
bdd22e
 
bdd22e
-    if (!wlStreamDpy->data->egl.queryDisplayAttrib(dpy,
bdd22e
-                                                   EGL_DEVICE_EXT,
bdd22e
-                                                   (EGLAttribKHR*)&egl_dev)) {
bdd22e
-        return EGL_FALSE;
bdd22e
+    if (!data->egl.queryDisplayAttrib(dpy, EGL_DEVICE_EXT,
bdd22e
+                                      (EGLAttribKHR*)&egl_dev)) {
bdd22e
+        return NULL;
bdd22e
     }
bdd22e
 
bdd22e
-
bdd22e
-    dev_exts = wlStreamDpy->data->egl.queryDeviceString(egl_dev,
bdd22e
-                                                        EGL_EXTENSIONS);
bdd22e
+    dev_exts = data->egl.queryDeviceString(egl_dev, EGL_EXTENSIONS);
bdd22e
 
bdd22e
     if (!dev_exts) {
bdd22e
-        return EGL_FALSE;
bdd22e
+        return NULL;
bdd22e
     }
bdd22e
 
bdd22e
     if (!wlEglFindExtension("EGL_EXT_device_drm_render_node", dev_exts)) {
bdd22e
-        return EGL_FALSE;
bdd22e
+        return NULL;
bdd22e
     }
bdd22e
 
bdd22e
-    dev_name =
bdd22e
-        wlStreamDpy->data->egl.queryDeviceString(egl_dev,
bdd22e
-                                                 EGL_DRM_RENDER_NODE_FILE_EXT);
bdd22e
+    return data->egl.queryDeviceString(egl_dev, EGL_DRM_RENDER_NODE_FILE_EXT);
bdd22e
+}
bdd22e
 
bdd22e
+EGLBoolean
bdd22e
+wl_drm_display_bind(struct wl_display *display,
bdd22e
+                    struct wl_eglstream_display *wlStreamDpy,
bdd22e
+                    const char *dev_name)
bdd22e
+{
bdd22e
     if (!dev_name) {
bdd22e
         return EGL_FALSE;
bdd22e
     }
bdd22e
diff --git a/src/wayland-egldisplay.c b/src/wayland-egldisplay.c
bdd22e
index 8b7394a..d285bf7 100644
bdd22e
--- a/src/wayland-egldisplay.c
bdd22e
+++ b/src/wayland-egldisplay.c
bdd22e
@@ -30,6 +30,7 @@
bdd22e
 #include "wayland-eglhandle.h"
bdd22e
 #include "wayland-eglutils.h"
bdd22e
 #include "wayland-drm-client-protocol.h"
bdd22e
+#include "wayland-drm.h"
bdd22e
 #include <string.h>
bdd22e
 #include <stdlib.h>
bdd22e
 #include <assert.h>
bdd22e
@@ -70,15 +71,16 @@ EGLBoolean wlEglIsValidNativeDisplayExport(void *data, void *nativeDpy)
bdd22e
 
bdd22e
 EGLBoolean wlEglBindDisplaysHook(void *data, EGLDisplay dpy, void *nativeDpy)
bdd22e
 {
bdd22e
-    /* Retrieve extension string before taking external API lock */
bdd22e
-    const char *exts = ((WlEglPlatformData *)data)->egl.queryString(dpy, EGL_EXTENSIONS);
bdd22e
+    /* Retrieve extension string and device name before taking external API lock */
bdd22e
+    const char *exts = ((WlEglPlatformData *)data)->egl.queryString(dpy, EGL_EXTENSIONS),
bdd22e
+               *dev_name = wl_drm_get_dev_name(data, dpy);
bdd22e
     EGLBoolean res = EGL_FALSE;
bdd22e
 
bdd22e
     wlExternalApiLock();
bdd22e
 
bdd22e
     res = wl_eglstream_display_bind((WlEglPlatformData *)data,
bdd22e
                                     (struct wl_display *)nativeDpy,
bdd22e
-                                    dpy, exts);
bdd22e
+                                    dpy, exts, dev_name);
bdd22e
 
bdd22e
     wlExternalApiUnlock();
bdd22e
 
bdd22e
diff --git a/src/wayland-eglstream-server.c b/src/wayland-eglstream-server.c
bdd22e
index b1baa08..1dfd7ce 100644
bdd22e
--- a/src/wayland-eglstream-server.c
bdd22e
+++ b/src/wayland-eglstream-server.c
bdd22e
@@ -289,7 +289,8 @@ EGLBoolean
bdd22e
 wl_eglstream_display_bind(WlEglPlatformData *data,
bdd22e
                           struct wl_display *wlDisplay,
bdd22e
                           EGLDisplay eglDisplay,
bdd22e
-                          const char *exts)
bdd22e
+                          const char *exts,
bdd22e
+                          const char *dev_name)
bdd22e
 {
bdd22e
     struct wl_eglstream_display *wlStreamDpy = NULL;
bdd22e
     char                        *env         = NULL;
bdd22e
@@ -355,7 +356,7 @@ wl_eglstream_display_bind(WlEglPlatformData *data,
bdd22e
                                            wl_eglstream_display_global_bind);
bdd22e
 
bdd22e
     /* Failure is not fatal */
bdd22e
-    wl_drm_display_bind(wlDisplay, wlStreamDpy);
bdd22e
+    wl_drm_display_bind(wlDisplay, wlStreamDpy, dev_name);
bdd22e
 
bdd22e
     wl_list_insert(&wlStreamDpyList, &wlStreamDpy->link);
bdd22e