Dynamically detect libwayland-client

Following the strategy already adopted with XCB, dynamically detect
whether libwayland-client is available in the system.

Bug: chromium:1327041
Change-Id: Ib7d31641b832453932e09c8a8020793c6a529fbf
Reviewed-on: https://swiftshader-review.googlesource.com/c/SwiftShader/+/66088
Reviewed-by: Alexis Hétu <sugoi@google.com>
Tested-by: Nicolas Capens <nicolascapens@google.com>
Kokoro-Result: kokoro <noreply+kokoro@google.com>
Reviewed-by: Nicolas Capens <nicolascapens@google.com>
diff --git a/CMakeLists.txt b/CMakeLists.txt
index 7c9b701..4350821 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -132,16 +132,20 @@
     check_symbol_exists(mallinfo2 malloc.h HAVE_MALLINFO2)
 
     include(CheckIncludeFiles)
+
     CHECK_INCLUDE_FILES("xcb/xcb.h;xcb/shm.h" HAVE_XCB_H)
     if(NOT HAVE_XCB_H)
         message(WARNING "xcb/xcb.h or xcb/shm.h was not found. Install the libx11-xcb-dev and "
                         "libxcb-shm0-dev packages to build with WSI support for XCB surfaces.")
     endif()
+
+    CHECK_INCLUDE_FILES("wayland-client.h" HAVE_WAYLAND_CLIENT_H)
+    if(NOT HAVE_WAYLAND_CLIENT_H)
+        message(WARNING "wayland-client.h was not found. Install the libwayland-dev "
+                        "package to build with WSI support for Wayland surfaces.")
+    endif()
 endif()
 
-if(SWIFTSHADER_BUILD_WSI_WAYLAND)
-    find_library(WAYLAND wayland-client)
-endif(SWIFTSHADER_BUILD_WSI_WAYLAND)
 if(SWIFTSHADER_BUILD_WSI_DIRECTFB)
     find_library(DIRECTFB directfb)
     find_path(DIRECTFB_INCLUDE_DIR directfb/directfb.h)
@@ -821,11 +825,9 @@
     if(HAVE_XCB_H)
         target_compile_definitions(vk_base INTERFACE "VK_USE_PLATFORM_XCB_KHR")
     endif()
-    if(SWIFTSHADER_BUILD_WSI_WAYLAND)
-        if(WAYLAND)
-            target_compile_definitions(vk_base INTERFACE "VK_USE_PLATFORM_WAYLAND_KHR")
-        endif()
-    endif(SWIFTSHADER_BUILD_WSI_WAYLAND)
+    if(HAVE_WAYLAND_CLIENT_H)
+        target_compile_definitions(vk_base INTERFACE "VK_USE_PLATFORM_WAYLAND_KHR")
+    endif()
     if(SWIFTSHADER_BUILD_WSI_DIRECTFB)
         if(DIRECTFB AND DIRECTFB_INCLUDE_DIR)
             target_compile_definitions(vk_base INTERFACE "VK_USE_PLATFORM_DIRECTFB_EXT")
diff --git a/src/Vulkan/libVulkan.cpp b/src/Vulkan/libVulkan.cpp
index cb13d64..b6b5962 100644
--- a/src/Vulkan/libVulkan.cpp
+++ b/src/Vulkan/libVulkan.cpp
@@ -291,7 +291,7 @@
 	{ { VK_KHR_XCB_SURFACE_EXTENSION_NAME, VK_KHR_XCB_SURFACE_SPEC_VERSION }, [] { return vk::XcbSurfaceKHR::isSupported(); } },
 #endif
 #ifdef VK_USE_PLATFORM_WAYLAND_KHR
-	{ { VK_KHR_WAYLAND_SURFACE_EXTENSION_NAME, VK_KHR_WAYLAND_SURFACE_SPEC_VERSION } },
+	{ { VK_KHR_WAYLAND_SURFACE_EXTENSION_NAME, VK_KHR_WAYLAND_SURFACE_SPEC_VERSION }, [] { return vk::WaylandSurfaceKHR::isSupported(); } },
 #endif
 #ifdef VK_USE_PLATFORM_DIRECTFB_EXT
 	{ { VK_EXT_DIRECTFB_SURFACE_EXTENSION_NAME, VK_EXT_DIRECTFB_SURFACE_SPEC_VERSION } },
diff --git a/src/WSI/CMakeLists.txt b/src/WSI/CMakeLists.txt
index f324582..3429ed6 100644
--- a/src/WSI/CMakeLists.txt
+++ b/src/WSI/CMakeLists.txt
@@ -41,10 +41,12 @@
         )
     endif()
 
-    if(WAYLAND)
+    if(HAVE_WAYLAND_CLIENT_H)
         list(APPEND WSI_SRC_FILES
             WaylandSurfaceKHR.cpp
             WaylandSurfaceKHR.hpp
+            libWaylandClient.cpp
+            libWaylandClient.hpp
         )
     endif()
 
diff --git a/src/WSI/WaylandSurfaceKHR.cpp b/src/WSI/WaylandSurfaceKHR.cpp
index e678d05..9ecf2dc 100644
--- a/src/WSI/WaylandSurfaceKHR.cpp
+++ b/src/WSI/WaylandSurfaceKHR.cpp
@@ -14,6 +14,7 @@
 
 #include "WaylandSurfaceKHR.hpp"
 
+#include "libWaylandClient.hpp"
 #include "Vulkan/VkDeviceMemory.hpp"
 #include "Vulkan/VkImage.hpp"
 
@@ -23,12 +24,17 @@
 
 namespace vk {
 
+bool WaylandSurfaceKHR::isSupported()
+{
+	return libWaylandClient.isPresent();
+}
+
 static void wl_registry_handle_global(void *data, struct wl_registry *registry, unsigned int name, const char *interface, unsigned int version)
 {
 	struct wl_shm **pshm = (struct wl_shm **)data;
 	if(!strcmp(interface, "wl_shm"))
 	{
-		*pshm = static_cast<struct wl_shm *>(wl_registry_bind(registry, name, &wl_shm_interface, 1));
+		*pshm = static_cast<struct wl_shm *>(libWaylandClient->wl_registry_bind(registry, name, libWaylandClient->wl_shm_interface, 1));
 	}
 }
 
@@ -42,9 +48,9 @@
     : display(pCreateInfo->display)
     , surface(pCreateInfo->surface)
 {
-	struct wl_registry *registry = wl_display_get_registry(display);
-	wl_registry_add_listener(registry, &wl_registry_listener, &shm);
-	wl_display_dispatch(display);
+	struct wl_registry *registry = libWaylandClient->wl_display_get_registry(display);
+	libWaylandClient->wl_registry_add_listener(registry, &wl_registry_listener, &shm);
+	libWaylandClient->wl_display_dispatch(display);
 }
 
 void WaylandSurfaceKHR::destroySurface(const VkAllocationCallbacks *pAllocator)
@@ -73,11 +79,11 @@
 	int fd = mkstemp(path);
 	const VkExtent3D &extent = image->getImage()->getExtent();
 	int stride = image->getImage()->rowPitchBytes(VK_IMAGE_ASPECT_COLOR_BIT, 0);
-	ftruncate(fd, extent.height * stride);
-	struct wl_shm_pool *pool = wl_shm_create_pool(shm, fd, extent.height * stride);
-	wlImage->buffer = wl_shm_pool_create_buffer(pool, 0, extent.width, extent.height, stride, WL_SHM_FORMAT_XRGB8888);
+	assert(ftruncate(fd, extent.height * stride) == 0);
+	struct wl_shm_pool *pool = libWaylandClient->wl_shm_create_pool(shm, fd, extent.height * stride);
+	wlImage->buffer = libWaylandClient->wl_shm_pool_create_buffer(pool, 0, extent.width, extent.height, stride, WL_SHM_FORMAT_XRGB8888);
 	wlImage->data = static_cast<uint8_t *>(mmap(NULL, extent.height * stride, PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0));
-	wl_shm_pool_destroy(pool);
+	libWaylandClient->wl_shm_pool_destroy(pool);
 	close(fd);
 	imageMap[image] = wlImage;
 }
@@ -91,7 +97,7 @@
 		const VkExtent3D &extent = image->getImage()->getExtent();
 		int stride = image->getImage()->rowPitchBytes(VK_IMAGE_ASPECT_COLOR_BIT, 0);
 		munmap(wlImage->data, extent.height * stride);
-		wl_buffer_destroy(wlImage->buffer);
+		libWaylandClient->wl_buffer_destroy(wlImage->buffer);
 		delete wlImage;
 		imageMap.erase(it);
 	}
@@ -106,11 +112,11 @@
 		const VkExtent3D &extent = image->getImage()->getExtent();
 		int bufferRowPitch = image->getImage()->rowPitchBytes(VK_IMAGE_ASPECT_COLOR_BIT, 0);
 		image->getImage()->copyTo(reinterpret_cast<uint8_t *>(wlImage->data), bufferRowPitch);
-		wl_surface_attach(surface, wlImage->buffer, 0, 0);
-		wl_surface_damage(surface, 0, 0, extent.width, extent.height);
-		wl_surface_commit(surface);
-		wl_display_roundtrip(display);
-		wl_display_sync(display);
+		libWaylandClient->wl_surface_attach(surface, wlImage->buffer, 0, 0);
+		libWaylandClient->wl_surface_damage(surface, 0, 0, extent.width, extent.height);
+		libWaylandClient->wl_surface_commit(surface);
+		libWaylandClient->wl_display_roundtrip(display);
+		libWaylandClient->wl_display_sync(display);
 	}
 
 	return VK_SUCCESS;
diff --git a/src/WSI/WaylandSurfaceKHR.hpp b/src/WSI/WaylandSurfaceKHR.hpp
index 17abe89..8c7c064 100644
--- a/src/WSI/WaylandSurfaceKHR.hpp
+++ b/src/WSI/WaylandSurfaceKHR.hpp
@@ -34,6 +34,7 @@
 class WaylandSurfaceKHR : public SurfaceKHR, public ObjectBase<WaylandSurfaceKHR, VkSurfaceKHR>
 {
 public:
+	static bool isSupported();
 	WaylandSurfaceKHR(const VkWaylandSurfaceCreateInfoKHR *pCreateInfo, void *mem);
 
 	void destroySurface(const VkAllocationCallbacks *pAllocator) override;
diff --git a/src/WSI/libWaylandClient.cpp b/src/WSI/libWaylandClient.cpp
new file mode 100644
index 0000000..b0dc1af
--- /dev/null
+++ b/src/WSI/libWaylandClient.cpp
@@ -0,0 +1,68 @@
+// Copyright 2021 The SwiftShader Authors. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//    http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "libWaylandClient.hpp"
+
+#include "System/SharedLibrary.hpp"
+
+#include <memory>
+
+LibWaylandClientExports::LibWaylandClientExports(void *libwl)
+{
+	getFuncAddress(libwl, "wl_display_dispatch", &wl_display_dispatch);
+	getFuncAddress(libwl, "wl_display_get_registry", &wl_display_get_registry);
+	getFuncAddress(libwl, "wl_display_roundtrip", &wl_display_roundtrip);
+	getFuncAddress(libwl, "wl_display_sync", &wl_display_sync);
+
+	getFuncAddress(libwl, "wl_registry_add_listener", &wl_registry_add_listener);
+	getFuncAddress(libwl, "wl_registry_bind", &wl_registry_bind);
+
+	getFuncAddress(libwl, "wl_buffer_destroy", &wl_buffer_destroy);
+	getFuncAddress(libwl, "wl_shm_create_pool", &wl_shm_create_pool);
+	getFuncAddress(libwl, "wl_shm_pool_create_buffer", &wl_shm_pool_create_buffer);
+	getFuncAddress(libwl, "wl_shm_pool_destroy", &wl_shm_pool_destroy);
+
+	getFuncAddress(libwl, "wl_surface_attach", &wl_surface_attach);
+	getFuncAddress(libwl, "wl_surface_damage", &wl_surface_damage);
+	getFuncAddress(libwl, "wl_surface_commit", &wl_surface_commit);
+
+	wl_shm_interface = reinterpret_cast<const wl_interface *>(getProcAddress(libwl, "wl_shm_interface"));
+}
+
+LibWaylandClientExports *LibWaylandClient::operator->()
+{
+	return loadExports();
+}
+
+LibWaylandClientExports *LibWaylandClient::loadExports()
+{
+	static LibWaylandClientExports exports = [] {
+		void *libwl = nullptr;
+
+		if(getProcAddress(RTLD_DEFAULT, "wl_display_dispatch"))  // Search the global scope for pre-loaded Wayland client library.
+		{
+			libwl = RTLD_DEFAULT;
+		}
+		else
+		{
+			libwl = loadLibrary("libwayland-client.so.0");
+		}
+
+		return LibWaylandClientExports(libwl);
+	}();
+
+	return exports.wl_display_dispatch ? &exports : nullptr;
+}
+
+LibWaylandClient libWaylandClient;
diff --git a/src/WSI/libWaylandClient.hpp b/src/WSI/libWaylandClient.hpp
new file mode 100644
index 0000000..d0a31a2
--- /dev/null
+++ b/src/WSI/libWaylandClient.hpp
@@ -0,0 +1,62 @@
+// Copyright 2022 The SwiftShader Authors. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//    http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef libWaylandClient_hpp
+#define libWaylandClient_hpp
+
+#include <wayland-client.h>
+
+struct LibWaylandClientExports
+{
+	LibWaylandClientExports() {}
+	LibWaylandClientExports(void *libwl);
+
+	int (*wl_display_dispatch)(wl_display *d) = nullptr;
+	wl_registry *(*wl_display_get_registry)(wl_display *d) = nullptr;
+	int (*wl_display_roundtrip)(wl_display *d) = nullptr;
+	wl_callback *(*wl_display_sync)(wl_display *d) = nullptr;
+
+	int (*wl_registry_add_listener)(wl_registry *r,
+	                                const wl_registry_listener *l, void *data) = nullptr;
+	void *(*wl_registry_bind)(wl_registry *r, uint32_t name, const wl_interface *i, uint32_t version) = nullptr;
+
+	void (*wl_buffer_destroy)(wl_buffer *b) = nullptr;
+	wl_shm_pool *(*wl_shm_create_pool)(wl_shm *shm, int32_t fd, int32_t size) = nullptr;
+	wl_buffer *(*wl_shm_pool_create_buffer)(wl_shm_pool *p, int32_t offset, int32_t width, int32_t height, int32_t stride, uint32_t format) = nullptr;
+	void (*wl_shm_pool_destroy)(wl_shm_pool *p) = nullptr;
+
+	void (*wl_surface_attach)(wl_surface *s, wl_buffer *b, int32_t x, int32_t y) = nullptr;
+	void (*wl_surface_damage)(wl_surface *s, int32_t x, int32_t y, int32_t width, int32_t height) = nullptr;
+	void (*wl_surface_commit)(wl_surface *s) = nullptr;
+
+	const wl_interface *wl_shm_interface = nullptr;
+};
+
+class LibWaylandClient
+{
+public:
+	bool isPresent()
+	{
+		return loadExports() != nullptr;
+	}
+
+	LibWaylandClientExports *operator->();
+
+private:
+	LibWaylandClientExports *loadExports();
+};
+
+extern LibWaylandClient libWaylandClient;
+
+#endif  // libWaylandClient_hpp