1 From 85a85537b17742a839092bcc2a4684ab0bdb65d5 Mon Sep 17 00:00:00 2001
2 From: Naoto Yamaguchi <naoto.yamaguchi@aisin.co.jp>
3 Date: Tue, 4 Jan 2022 23:52:31 +0900
4 Subject: [PATCH] Add drm-lease client support to eglfs kms backend
6 This patch is adding drm-lease-manager client support to Qt eglfs backend.
7 When this patch is enable, Qt wayland compositor and Qt eglfs without
8 compositor possible to use leased drm device.
10 List of environment variable:
12 QT_QPA_EGLFS_DRMLEASE :
13 When this variable is set, drm-lease-manager client support is enable.
14 Shall set drm lease device such as `card0-LVDS-1`, `card0-HDMI-A-1 ` and etc.
17 It use drm-lease-manager client, typically it need to set own.
19 Signed-off-by: Naoto Yamaguchi <naoto.yamaguchi@aisin.co.jp>
21 cmake/3rdparty/kwin/FindLibdlmclient.cmake | 127 ++++++++++++++++++
22 src/gui/configure.cmake | 1 +
23 .../eglfs_kms/CMakeLists.txt | 2 +
24 .../eglfs_kms/qeglfskmsgbmdevice.cpp | 43 ++++--
25 .../eglfs_kms/qeglfskmsgbmdevice_p.h | 4 +
26 5 files changed, 167 insertions(+), 10 deletions(-)
27 create mode 100644 cmake/3rdparty/kwin/FindLibdlmclient.cmake
29 diff --git a/cmake/3rdparty/kwin/FindLibdlmclient.cmake b/cmake/3rdparty/kwin/FindLibdlmclient.cmake
31 index 0000000000..8dfac920ca
33 +++ b/cmake/3rdparty/kwin/FindLibdlmclient.cmake
38 +# FindLibdlmclient.cmake is created based on FindLibinput.cmake
40 +# Try to find libdlmclient on a Unix system.
42 +# This will define the following variables:
44 +# ``Libdlmclient_FOUND``
45 +# True if (the requested version of) libdlmclient is available
46 +# ``Libdlmclient_VERSION``
47 +# The version of libdlmclient
48 +# ``Libdlmclient_LIBRARIES``
49 +# This can be passed to target_link_libraries() instead of the ``Libdlmclient::Libdlmclient``
51 +# ``Libdlmclient_INCLUDE_DIRS``
52 +# This should be passed to target_include_directories() if the target is not
54 +# ``Libdlmclient_DEFINITIONS``
55 +# This should be passed to target_compile_options() if the target is not
58 +# If ``Libdlmclient_FOUND`` is TRUE, it will also define the following imported target:
60 +# ``Libdlmclient::Libdlmclient``
61 +# The libdlmclient library
63 +# In general we recommend using the imported target, as it is easier to use.
64 +# Bear in mind, however, that if the target is in the link interface of an
65 +# exported library, it must be made available by the package config file.
67 +#=============================================================================
68 +# Copyright 2014 Alex Merry <alex.merry@kde.org>
69 +# Copyright 2014 Martin Gräßlin <mgraesslin@kde.org>
70 +# Copyright 2022 Naoto Yamaguchi <naoto.yamaguchi@aisin.co.jp>
72 +# Redistribution and use in source and binary forms, with or without
73 +# modification, are permitted provided that the following conditions
76 +# 1. Redistributions of source code must retain the copyright
77 +# notice, this list of conditions and the following disclaimer.
78 +# 2. Redistributions in binary form must reproduce the copyright
79 +# notice, this list of conditions and the following disclaimer in the
80 +# documentation and/or other materials provided with the distribution.
81 +# 3. The name of the author may not be used to endorse or promote products
82 +# derived from this software without specific prior written permission.
84 +# THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
85 +# IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
86 +# OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
87 +# IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
88 +# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
89 +# NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
90 +# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
91 +# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
92 +# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
93 +# THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
94 +#=============================================================================
96 +if(CMAKE_VERSION VERSION_LESS 2.8.12)
97 + message(FATAL_ERROR "CMake 2.8.12 is required by FindLibdlmclient.cmake")
99 +if(CMAKE_MINIMUM_REQUIRED_VERSION VERSION_LESS 2.8.12)
100 + message(AUTHOR_WARNING "Your project should require at least CMake 2.8.12 to use FindLibdlmclient.cmake")
104 + # Use pkg-config to get the directories and then use these values
105 + # in the FIND_PATH() and FIND_LIBRARY() calls
106 + find_package(PkgConfig QUIET)
107 + pkg_check_modules(PKG_Libdlmclient QUIET libdlmclient)
109 + set(Libdlmclient_DEFINITIONS ${PKG_Libdlmclient_CFLAGS_OTHER})
110 + set(Libdlmclient_VERSION ${PKG_Libdlmclient_VERSION})
112 + find_path(Libdlmclient_INCLUDE_DIR
116 + ${PKG_Libdlmclient_INCLUDE_DIRS}
118 + find_library(Libdlmclient_LIBRARY
122 + ${PKG_Libdlmclient_LIBRARY_DIRS}
125 + include(FindPackageHandleStandardArgs)
126 + find_package_handle_standard_args(Libdlmclient
130 + Libdlmclient_LIBRARY
131 + Libdlmclient_INCLUDE_DIR
133 + Libdlmclient_VERSION
136 + if(Libdlmclient_FOUND AND NOT TARGET Libdlmclient::Libdlmclient)
137 + add_library(Libdlmclient::Libdlmclient UNKNOWN IMPORTED)
138 + set_target_properties(Libdlmclient::Libdlmclient PROPERTIES
139 + IMPORTED_LOCATION "${Libdlmclient_LIBRARY}"
140 + INTERFACE_COMPILE_OPTIONS "${Libdlmclient_DEFINITIONS}"
141 + INTERFACE_INCLUDE_DIRECTORIES "${Libdlmclient_INCLUDE_DIR}"
145 + mark_as_advanced(Libdlmclient_LIBRARY Libdlmclient_INCLUDE_DIR)
147 + # compatibility variables
148 + set(Libdlmclient_LIBRARIES ${Libdlmclient_LIBRARY})
149 + set(Libdlmclient_INCLUDE_DIRS ${Libdlmclient_INCLUDE_DIR})
150 + set(Libdlmclient_VERSION_STRING ${Libdlmclient_VERSION})
153 + message(STATUS "FindLibdlmclient.cmake cannot find libdlmclient on Windows systems.")
154 + set(Libdlmclient_FOUND FALSE)
157 +include(FeatureSummary)
158 +set_package_properties(Libdlmclient PROPERTIES
159 + URL "https://git.automotivelinux.org/src/drm-lease-manager/"
160 + DESCRIPTION "The DRM Lease Manager uses the DRM Lease feature, introduced in the Linux kernel version 4.15, to partition display controller output resources between multiple processes."
162 diff --git a/src/gui/configure.cmake b/src/gui/configure.cmake
163 index 0bf250eab6..b092b746c0 100644
164 --- a/src/gui/configure.cmake
165 +++ b/src/gui/configure.cmake
166 @@ -38,6 +38,7 @@ endif()
167 qt_find_package(Fontconfig PROVIDED_TARGETS Fontconfig::Fontconfig MODULE_NAME gui QMAKE_LIB fontconfig)
168 qt_add_qmake_lib_dependency(fontconfig freetype)
169 qt_find_package(gbm PROVIDED_TARGETS gbm::gbm MODULE_NAME gui QMAKE_LIB gbm)
170 +qt_find_package(Libdlmclient PROVIDED_TARGETS Libdlmclient::Libdlmclient MODULE_NAME gui QMAKE_LIB libdlmclient)
171 qt_find_package(WrapSystemHarfbuzz 2.6.0 PROVIDED_TARGETS WrapSystemHarfbuzz::WrapSystemHarfbuzz MODULE_NAME gui QMAKE_LIB harfbuzz)
172 qt_find_package(Libinput PROVIDED_TARGETS Libinput::Libinput MODULE_NAME gui QMAKE_LIB libinput)
173 qt_find_package(JPEG PROVIDED_TARGETS JPEG::JPEG MODULE_NAME gui QMAKE_LIB libjpeg)
174 diff --git a/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/CMakeLists.txt b/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/CMakeLists.txt
175 index a20a4a084d..2af7904eb6 100644
176 --- a/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/CMakeLists.txt
177 +++ b/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/CMakeLists.txt
178 @@ -23,6 +23,7 @@ qt_internal_add_module(EglFsKmsGbmSupportPrivate
180 Qt::KmsSupportPrivate
182 + Libdlmclient::Libdlmclient
184 #####################################################################
185 ## QEglFSKmsGbmIntegrationPlugin Plugin:
186 @@ -44,6 +45,7 @@ qt_internal_add_plugin(QEglFSKmsGbmIntegrationPlugin
188 Qt::KmsSupportPrivate
190 + Libdlmclient::Libdlmclient
193 #### Keys ignored in scope 3:.:.:eglfs_kms-plugin.pro:<TRUE>:
194 diff --git a/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/qeglfskmsgbmdevice.cpp b/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/qeglfskmsgbmdevice.cpp
195 index b5b7811b79..1e77c72ee5 100644
196 --- a/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/qeglfskmsgbmdevice.cpp
197 +++ b/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/qeglfskmsgbmdevice.cpp
198 @@ -56,32 +56,51 @@ Q_DECLARE_LOGGING_CATEGORY(qLcEglfsKmsDebug)
199 QEglFSKmsGbmDevice::QEglFSKmsGbmDevice(QKmsScreenConfig *screenConfig, const QString &path)
200 : QEglFSKmsDevice(screenConfig, path)
201 , m_gbm_device(nullptr)
203 , m_globalCursor(nullptr)
207 bool QEglFSKmsGbmDevice::open()
209 + QByteArray lease = qgetenv("QT_QPA_EGLFS_DRMLEASE");
211 Q_ASSERT(fd() == -1);
212 Q_ASSERT(m_gbm_device == nullptr);
214 - int fd = qt_safe_open(devicePath().toLocal8Bit().constData(), O_RDWR | O_CLOEXEC);
216 - qErrnoWarning("Could not open DRM device %s", qPrintable(devicePath()));
218 + if (lease.isEmpty()) {
219 + drmfd = qt_safe_open(devicePath().toLocal8Bit().constData(), O_RDWR | O_CLOEXEC);
221 + qErrnoWarning("Could not open DRM device %s", qPrintable(devicePath()));
225 + QString lease_path(QString::fromUtf8(lease));
227 + m_lease = ::dlm_get_lease(lease_path.toLocal8Bit().constData());
228 + if (m_lease == nullptr) {
229 + qErrnoWarning("Could not open DRM lease %s", qPrintable(lease_path));
232 + drmfd = dlm_lease_fd(m_lease);
235 - qCDebug(qLcEglfsKmsDebug) << "Creating GBM device for file descriptor" << fd
236 + qCDebug(qLcEglfsKmsDebug) << "Creating GBM device for file descriptor" << drmfd
237 << "obtained from" << devicePath();
238 - m_gbm_device = gbm_create_device(fd);
239 + m_gbm_device = gbm_create_device(drmfd);
241 qErrnoWarning("Could not create GBM device");
245 + ::dlm_release_lease(m_lease);
248 + qt_safe_close(drmfd);
257 m_eventReader.create(this);
259 @@ -99,7 +118,11 @@ void QEglFSKmsGbmDevice::close()
260 m_gbm_device = nullptr;
266 + ::dlm_release_lease(m_lease);
268 + } else if (fd() != -1) {
272 diff --git a/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/qeglfskmsgbmdevice_p.h b/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/qeglfskmsgbmdevice_p.h
273 index bb67d800ee..b2d705800e 100644
274 --- a/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/qeglfskmsgbmdevice_p.h
275 +++ b/src/plugins/platforms/eglfs/deviceintegration/eglfs_kms/qeglfskmsgbmdevice_p.h
277 #include <private/qeglfskmsdevice_p.h>
280 +#include <dlmclient.h>
284 @@ -92,6 +93,9 @@ private:
286 gbm_device *m_gbm_device;
288 + // only used for DRM lease
289 + struct dlm_lease *m_lease;
291 QEglFSKmsGbmCursor *m_globalCursor;