| /**************************************************************************** |
| ** |
| ** 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:GPL$ |
| ** 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 General Public License Usage |
| ** Alternatively, this file may be used under the terms of the GNU |
| ** General Public License version 3 or (at your option) 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.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-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 |