/**************************************************************************** ** ** Copyright (C) 2021 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the plugins of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 3 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL3 included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 3 requirements ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 2.0 or (at your option) the GNU General ** Public license version 3 or any later version approved by the KDE Free ** Qt Foundation. The licenses are as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3 ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-2.0.html and ** https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #include "qvkkhrdisplayintegration.h" #include "qvkkhrdisplayvulkaninstance.h" #include #include #include #include #include #include #include #include #include #include #include #if QT_CONFIG(libinput) #include #endif #if QT_CONFIG(evdev) #include #include #include #endif #if QT_CONFIG(tslib) #include #endif QT_BEGIN_NAMESPACE class QVkKhrDisplayScreen : public QPlatformScreen { public: QRect geometry() const override { return m_geometry; } int depth() const override { return m_depth; } QImage::Format format() const override { return m_format; } void setVk(QVkKhrDisplayVulkanInstance *inst); private: QVkKhrDisplayVulkanInstance *m_vk = nullptr; QRect m_geometry; int m_depth = 32; QImage::Format m_format = QImage::Format_ARGB32_Premultiplied; friend class QVkKhrDisplayIntegration; }; void QVkKhrDisplayScreen::setVk(QVkKhrDisplayVulkanInstance *inst) { m_vk = inst; m_geometry = QRect(QPoint(0, 0), m_vk->displaySize()); QWindowSystemInterface::handleScreenGeometryChange(screen(), m_geometry, m_geometry); qDebug() << "Screen will report geometry" << m_geometry; // Thanks to this deferred screen setup, a QWindow with a size based on the // dummy screen size may already exist. Try to resize it. QScreen *thisScreen = screen(); for (QWindow *window : QGuiApplication::allWindows()) { if (window->isTopLevel() && window->screen() == thisScreen) window->handle()->setGeometry(QRect()); // set fullscreen geometry } } class QVkKhrDisplayWindow : public QPlatformWindow { public: QVkKhrDisplayWindow(QWindow *window) : QPlatformWindow(window) { } ~QVkKhrDisplayWindow(); void *vulkanSurfacePtr(); void setGeometry(const QRect &r) override; private: VkSurfaceKHR m_surface = VK_NULL_HANDLE; }; QVkKhrDisplayWindow::~QVkKhrDisplayWindow() { if (m_surface) { QVulkanInstance *inst = window()->vulkanInstance(); if (inst) static_cast(inst->handle())->destroySurface(m_surface); } } void *QVkKhrDisplayWindow::vulkanSurfacePtr() { if (m_surface) return &m_surface; QVulkanInstance *inst = window()->vulkanInstance(); if (!inst) { qWarning("Attempted to create Vulkan surface without an instance; was QWindow::setVulkanInstance() called?"); return nullptr; } QVkKhrDisplayVulkanInstance *vkdinst = static_cast(inst->handle()); m_surface = vkdinst->createSurface(window()); return &m_surface; } void QVkKhrDisplayWindow::setGeometry(const QRect &) { // We only support full-screen windows QRect rect(screen()->availableGeometry()); QWindowSystemInterface::handleGeometryChange(window(), rect); QPlatformWindow::setGeometry(rect); const QRect lastReportedGeometry = qt_window_private(window())->geometry; if (rect != lastReportedGeometry) QWindowSystemInterface::handleExposeEvent(window(), QRect(QPoint(0, 0), rect.size())); } // does not actually support raster content, just paint into a QImage and that's it for now class QVkKhrDisplayBackingStore : public QPlatformBackingStore { public: QVkKhrDisplayBackingStore(QWindow *window) : QPlatformBackingStore(window) { } QPaintDevice *paintDevice() override { return &m_image; } void flush(QWindow *window, const QRegion ®ion, const QPoint &offset) override { Q_UNUSED(window); Q_UNUSED(region); Q_UNUSED(offset); } void resize(const QSize &size, const QRegion &staticContents) override { Q_UNUSED(staticContents); QImage::Format format = QGuiApplication::primaryScreen()->handle()->format(); if (m_image.size() != size) m_image = QImage(size, format); } private: QImage m_image; }; QVkKhrDisplayIntegration::QVkKhrDisplayIntegration(const QStringList ¶meters) { Q_UNUSED(parameters); } QVkKhrDisplayIntegration::~QVkKhrDisplayIntegration() { QWindowSystemInterface::handleScreenRemoved(m_primaryScreen); delete m_services; delete m_fontDatabase; delete m_vtHandler; } bool QVkKhrDisplayIntegration::hasCapability(QPlatformIntegration::Capability cap) const { switch (cap) { case ThreadedPixmaps: return true; case WindowManagement: return false; default: return QPlatformIntegration::hasCapability(cap); } } void QVkKhrDisplayIntegration::initialize() { m_primaryScreen = new QVkKhrDisplayScreen; // The real values are only known when the QVulkanInstance initializes, use // dummy values until then. m_primaryScreen->m_geometry = QRect(0, 0, 1920, 1080); m_primaryScreen->m_depth = 32; m_primaryScreen->m_format = QImage::Format_ARGB32_Premultiplied; QWindowSystemInterface::handleScreenAdded(m_primaryScreen); m_inputContext = QPlatformInputContextFactory::create(); m_vtHandler = new QFbVtHandler; if (!qEnvironmentVariableIntValue("QT_QPA_DISABLE_INPUT")) createInputHandlers(); } QPlatformFontDatabase *QVkKhrDisplayIntegration::fontDatabase() const { if (!m_fontDatabase) m_fontDatabase = new QGenericUnixFontDatabase; return m_fontDatabase; } QPlatformServices *QVkKhrDisplayIntegration::services() const { if (!m_services) m_services = new QGenericUnixServices; return m_services; } QPlatformInputContext *QVkKhrDisplayIntegration::inputContext() const { return m_inputContext; } QPlatformTheme *QVkKhrDisplayIntegration::createPlatformTheme(const QString &name) const { return QGenericUnixTheme::createUnixTheme(name); } QPlatformNativeInterface *QVkKhrDisplayIntegration::nativeInterface() const { return const_cast(this); } QPlatformWindow *QVkKhrDisplayIntegration::createPlatformWindow(QWindow *window) const { if (window->surfaceType() != QSurface::VulkanSurface) { qWarning("vkkhrdisplay platform plugin only supports QWindow with surfaceType == VulkanSurface"); return nullptr; } QVkKhrDisplayWindow *w = new QVkKhrDisplayWindow(window); w->setGeometry(QRect()); // set fullscreen geometry w->requestActivateWindow(); return w; } QPlatformBackingStore *QVkKhrDisplayIntegration::createPlatformBackingStore(QWindow *window) const { return new QVkKhrDisplayBackingStore(window); } QAbstractEventDispatcher *QVkKhrDisplayIntegration::createEventDispatcher() const { return createUnixEventDispatcher(); } void QVkKhrDisplayIntegration::handleInstanceCreated(QVkKhrDisplayVulkanInstance *inst, void *userData) { QVkKhrDisplayIntegration *self = static_cast(userData); self->m_primaryScreen->setVk(inst); } QPlatformVulkanInstance *QVkKhrDisplayIntegration::createPlatformVulkanInstance(QVulkanInstance *instance) const { QVkKhrDisplayVulkanInstance *inst = new QVkKhrDisplayVulkanInstance(instance); inst->setCreatedCallback(handleInstanceCreated, const_cast(this)); return inst; } enum ResourceType { VkSurface }; static int resourceType(const QByteArray &key) { static const QByteArray names[] = { // match ResourceType QByteArrayLiteral("vksurface") }; const QByteArray *end = names + sizeof(names) / sizeof(names[0]); const QByteArray *result = std::find(names, end, key); if (result == end) result = std::find(names, end, key.toLower()); return int(result - names); } void *QVkKhrDisplayIntegration::nativeResourceForWindow(const QByteArray &resource, QWindow *window) { void *result = nullptr; switch (resourceType(resource)) { case VkSurface: if (window && window->handle() && window->surfaceType() == QSurface::VulkanSurface) result = static_cast(window->handle())->vulkanSurfacePtr(); break; default: break; } return result; } void QVkKhrDisplayIntegration::createInputHandlers() { #if QT_CONFIG(libinput) if (!qEnvironmentVariableIntValue("QT_QPA_NO_LIBINPUT")) { new QLibInputHandler(QLatin1String("libinput"), QString()); return; } #endif #if QT_CONFIG(tslib) bool useTslib = qEnvironmentVariableIntValue("QT_QPA_TSLIB"); if (useTslib) new QTsLibMouseHandler(QLatin1String("TsLib"), QString()); #endif #if QT_CONFIG(evdev) new QEvdevKeyboardManager(QLatin1String("EvdevKeyboard"), QString(), this); new QEvdevMouseManager(QLatin1String("EvdevMouse"), QString(), this); #if QT_CONFIG(tslib) if (!useTslib) #endif new QEvdevTouchManager(QLatin1String("EvdevTouch"), QString() /* spec */, this); #endif } QT_END_NAMESPACE