summaryrefslogtreecommitdiffstats
path: root/src/hardwareintegration/compositor/linux-dmabuf-unstable-v1/linuxdmabuf.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'src/hardwareintegration/compositor/linux-dmabuf-unstable-v1/linuxdmabuf.cpp')
-rw-r--r--src/hardwareintegration/compositor/linux-dmabuf-unstable-v1/linuxdmabuf.cpp332
1 files changed, 332 insertions, 0 deletions
diff --git a/src/hardwareintegration/compositor/linux-dmabuf-unstable-v1/linuxdmabuf.cpp b/src/hardwareintegration/compositor/linux-dmabuf-unstable-v1/linuxdmabuf.cpp
new file mode 100644
index 000000000..2ba0462c7
--- /dev/null
+++ b/src/hardwareintegration/compositor/linux-dmabuf-unstable-v1/linuxdmabuf.cpp
@@ -0,0 +1,332 @@
+/****************************************************************************
+**
+** Copyright (C) 2018 The Qt Company Ltd.
+** Contact: https://www.qt.io/licensing/
+**
+** This file is part of the QtWaylandCompositor module 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 "linuxdmabuf.h"
+#include "linuxdmabufclientbufferintegration.h"
+
+#include <QtWaylandCompositor/QWaylandCompositor>
+
+#include <drm_fourcc.h>
+#include <drm_mode.h>
+#include <unistd.h>
+
+QT_BEGIN_NAMESPACE
+
+LinuxDmabuf::LinuxDmabuf(wl_display *display, LinuxDmabufClientBufferIntegration *clientBufferIntegration)
+ : zwp_linux_dmabuf_v1(display, 3 /*version*/)
+ , m_clientBufferIntegration(clientBufferIntegration)
+{
+}
+
+void LinuxDmabuf::setSupportedModifiers(const QHash<uint32_t, QVector<uint64_t>> &modifiers)
+{
+ Q_ASSERT(resourceMap().isEmpty());
+ m_modifiers = modifiers;
+}
+
+void LinuxDmabuf::zwp_linux_dmabuf_v1_bind_resource(Resource *resource)
+{
+ for (auto it = m_modifiers.constBegin(); it != m_modifiers.constEnd(); ++it) {
+ auto format = it.key();
+ auto modifiers = it.value();
+ // send DRM_FORMAT_MOD_INVALID when no modifiers are supported for a format
+ if (modifiers.isEmpty())
+ modifiers << DRM_FORMAT_MOD_INVALID;
+ for (const auto &modifier : qAsConst(modifiers)) {
+ if (resource->version() >= ZWP_LINUX_DMABUF_V1_MODIFIER_SINCE_VERSION) {
+ const uint32_t modifier_lo = modifier & 0xFFFFFFFF;
+ const uint32_t modifier_hi = modifier >> 32;
+ send_modifier(resource->handle, format, modifier_hi, modifier_lo);
+ } else if (modifier == DRM_FORMAT_MOD_LINEAR || modifier == DRM_FORMAT_MOD_INVALID) {
+ send_format(resource->handle, format);
+ }
+ }
+ }
+}
+
+void LinuxDmabuf::zwp_linux_dmabuf_v1_create_params(Resource *resource, uint32_t params_id)
+{
+ wl_resource *r = wl_resource_create(resource->client(), &zwp_linux_buffer_params_v1_interface,
+ wl_resource_get_version(resource->handle), params_id);
+ new LinuxDmabufParams(m_clientBufferIntegration, r); // deleted by the client, or when it disconnects
+}
+
+LinuxDmabufParams::LinuxDmabufParams(LinuxDmabufClientBufferIntegration *clientBufferIntegration, wl_resource *resource)
+ : zwp_linux_buffer_params_v1(resource)
+ , m_clientBufferIntegration(clientBufferIntegration)
+{
+}
+
+LinuxDmabufParams::~LinuxDmabufParams()
+{
+ for (auto it = m_planes.begin(); it != m_planes.end(); ++it) {
+ if (it.value().fd != -1)
+ close(it.value().fd);
+ it.value().fd = -1;
+ }
+}
+
+bool LinuxDmabufParams::handleCreateParams(Resource *resource, int width, int height, uint format, uint flags)
+{
+ if (m_used) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_ALREADY_USED,
+ "Params already used");
+ return false;
+ }
+
+ if (width <= 0 || height <= 0) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INVALID_DIMENSIONS,
+ "Invalid dimensions in create request");
+ return false;
+ }
+
+ if (m_planes.isEmpty()) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INCOMPLETE,
+ "Cannot create a buffer with no planes");
+ return false;
+ }
+
+ // check for holes in plane sequence
+ auto planeIds = m_planes.keys();
+ std::sort(planeIds.begin(), planeIds.end());
+ for (int i = 0; i < planeIds.count(); ++i) {
+ if (uint(i) != planeIds[i]) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INCOMPLETE,
+ "No dmabuf parameters provided for plane %i", i);
+ return false;
+ }
+ }
+
+ // check for overflows
+ for (auto it = m_planes.constBegin(); it != m_planes.constEnd(); ++it) {
+ const auto planeId = it.key();
+ const auto plane = it.value();
+ if (static_cast<int64_t>(plane.offset) + plane.stride > UINT32_MAX) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS,
+ "Size overflow for plane %i",
+ planeId);
+ return false;
+ }
+ if (planeId == 0 && static_cast<int64_t>(plane.offset) + plane.stride * static_cast<int64_t>(height) > UINT32_MAX) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS,
+ "Size overflow for plane %i",
+ planeId);
+ return false;
+ }
+
+ // do not report an error as it might be caused by the kernel not supporting seeking on dmabuf
+ off_t size = lseek(plane.fd, 0, SEEK_END);
+ if (size == -1) {
+ qCDebug(qLcWaylandCompositorHardwareIntegration) << "Seeking is not supported";
+ continue;
+ }
+
+ if (static_cast<int64_t>(plane.offset) >= size) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS,
+ "Invalid offset %i for plane %i",
+ plane.offset, planeId);
+ return false;
+ }
+
+ if (static_cast<int64_t>(plane.offset) + static_cast<int64_t>(plane.stride) > size) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS,
+ "Invalid stride %i for plane %i",
+ plane.stride, planeId);
+ return false;
+ }
+
+ // only valid for first plane as other planes might be sub-sampled
+ if (planeId == 0 && plane.offset + static_cast<int64_t>(plane.stride) * height > size) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS,
+ "Invalid buffer stride or height for plane %i", planeId);
+ return false;
+ }
+ }
+
+ m_size = QSize(width, height);
+ m_drmFormat = format;
+ m_flags = flags;
+ m_used = true;
+
+ return true;
+}
+
+void LinuxDmabufParams::zwp_linux_buffer_params_v1_destroy(Resource *resource)
+{
+ wl_resource_destroy(resource->handle);
+}
+
+void LinuxDmabufParams::zwp_linux_buffer_params_v1_destroy_resource(Resource *resource)
+{
+ Q_UNUSED(resource);
+ delete this;
+}
+
+void LinuxDmabufParams::zwp_linux_buffer_params_v1_add(Resource *resource, int32_t fd, uint32_t plane_idx, uint32_t offset, uint32_t stride, uint32_t modifier_hi, uint32_t modifier_lo)
+{
+ const uint64_t modifiers = (static_cast<uint64_t>(modifier_hi) << 32) | modifier_lo;
+ if (plane_idx >= LinuxDmabufWlBuffer::MaxDmabufPlanes) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_PLANE_IDX,
+ "Plane index %i is out of bounds", plane_idx);
+ }
+
+ if (m_planes.contains(plane_idx)) {
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_PLANE_SET,
+ "Plane already set");
+ }
+
+ Plane plane;
+ plane.fd = fd;
+ plane.modifiers = modifiers;
+ plane.offset = offset;
+ plane.stride = stride;
+ m_planes.insert(plane_idx, plane);
+}
+
+void LinuxDmabufParams::zwp_linux_buffer_params_v1_create(Resource *resource, int32_t width, int32_t height, uint32_t format, uint32_t flags)
+{
+ if (!handleCreateParams(resource, width, height, format, flags))
+ return;
+
+ auto *buffer = new LinuxDmabufWlBuffer(resource->client(), m_clientBufferIntegration);
+ buffer->m_size = m_size;
+ buffer->m_flags = m_flags;
+ buffer->m_drmFormat = m_drmFormat;
+ buffer->m_planesNumber = m_planes.size(); // it is checked before that planes are in consecutive sequence
+ for (auto it = m_planes.begin(); it != m_planes.end(); ++it) {
+ buffer->m_planes[it.key()] = it.value();
+ it.value().fd = -1; // ownership is moved
+ }
+
+ if (!m_clientBufferIntegration->importBuffer(buffer->resource()->handle, buffer)) {
+ send_failed(resource->handle);
+ } else {
+ send_created(resource->handle, buffer->resource()->handle);
+ }
+}
+
+void LinuxDmabufParams::zwp_linux_buffer_params_v1_create_immed(Resource *resource, uint32_t buffer_id, int32_t width, int32_t height, uint32_t format, uint32_t flags)
+{
+ if (!handleCreateParams(resource, width, height, format, flags))
+ return;
+
+ auto *buffer = new LinuxDmabufWlBuffer(resource->client(), m_clientBufferIntegration, buffer_id);
+ buffer->m_size = m_size;
+ buffer->m_flags = m_flags;
+ buffer->m_drmFormat = m_drmFormat;
+ buffer->m_planesNumber = m_planes.size(); // it is checked before that planes are in consecutive sequence
+ for (auto it = m_planes.begin(); it != m_planes.end(); ++it) {
+ buffer->m_planes[it.key()] = it.value();
+ it.value().fd = -1; // ownership is moved
+ }
+
+ if (!m_clientBufferIntegration->importBuffer(buffer->resource()->handle, buffer)) {
+ // for the 'create_immed' request, the implementation can decide
+ // how to handle the failure by an unknown cause; we decide
+ // to raise a fatal error at the client
+ wl_resource_post_error(resource->handle,
+ ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INVALID_WL_BUFFER,
+ "Import of the provided DMA buffer failed");
+ }
+ // note: create signal shall not be sent for the 'create_immed' request
+}
+
+LinuxDmabufWlBuffer::LinuxDmabufWlBuffer(::wl_client *client, LinuxDmabufClientBufferIntegration *clientBufferIntegration, uint id)
+ : wl_buffer(client, id, 1 /*version*/)
+ , m_clientBufferIntegration(clientBufferIntegration)
+{
+}
+
+LinuxDmabufWlBuffer::~LinuxDmabufWlBuffer()
+{
+ m_clientBufferIntegration->removeBuffer(resource()->handle);
+ buffer_destroy(resource());
+}
+
+void LinuxDmabufWlBuffer::buffer_destroy(Resource *resource)
+{
+ Q_UNUSED(resource);
+ for (uint32_t i = 0; i < m_planesNumber; ++i) {
+ if (m_textures[i] != nullptr) {
+ m_clientBufferIntegration->deleteGLTextureWhenPossible(m_textures[i]);
+ m_textures[i] = nullptr;
+ }
+ if (m_eglImages[i] != EGL_NO_IMAGE_KHR) {
+ m_clientBufferIntegration->deleteImage(m_eglImages[i]);
+ m_eglImages[i] = EGL_NO_IMAGE_KHR;
+ }
+ if (m_planes[i].fd != -1)
+ close(m_planes[i].fd);
+ m_planes[i].fd = -1;
+ }
+ m_planesNumber = 0;
+}
+
+void LinuxDmabufWlBuffer::initImage(uint32_t plane, EGLImageKHR image)
+{
+ Q_ASSERT(plane < m_planesNumber);
+ Q_ASSERT(m_eglImages.at(plane) == EGL_NO_IMAGE_KHR);
+ m_eglImages[plane] = image;
+}
+
+void LinuxDmabufWlBuffer::initTexture(uint32_t plane, QOpenGLTexture *texture)
+{
+ Q_ASSERT(plane < m_planesNumber);
+ Q_ASSERT(m_textures.at(plane) == nullptr);
+ m_textures[plane] = texture;
+}
+
+void LinuxDmabufWlBuffer::buffer_destroy_resource(Resource *resource)
+{
+ Q_UNUSED(resource);
+ delete this;
+}
+
+QT_END_NAMESPACE