/* SPDX-FileCopyrightText: 2018 Fredrik Höglund <fredrik@kde.org> SPDX-FileCopyrightText: 2019 Roman Gilg <subdiff@gmail.com> SPDX-FileCopyrightText: 2021 Vlad Zahorodnii <vlad.zahorodnii@kde.org> Based on the libweston implementation, SPDX-FileCopyrightText: 2014, 2015 Collabora, Ltd. SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL */ #include "linuxdmabufv1clientbuffer.h" #include "clientbuffer_p.h" #include "display.h" #include "display_p.h" #include "drm_fourcc.h" #include "qwayland-server-linux-dmabuf-unstable-v1.h" #include "qwayland-server-wayland.h" #include <QDebug> #include <QVector> #include <unistd.h> namespace KWaylandServer { static const int s_version = 3; class LinuxDmaBufV1ClientBufferIntegrationPrivate : public QtWaylandServer::zwp_linux_dmabuf_v1 { public: LinuxDmaBufV1ClientBufferIntegrationPrivate(LinuxDmaBufV1ClientBufferIntegration *q, Display *display); LinuxDmaBufV1ClientBufferIntegration *q; LinuxDmaBufV1ClientBufferIntegration::RendererInterface *rendererInterface = nullptr; QHash<uint32_t, QSet<uint64_t>> supportedModifiers; protected: void zwp_linux_dmabuf_v1_bind_resource(Resource *resource) override; void zwp_linux_dmabuf_v1_destroy(Resource *resource) override; void zwp_linux_dmabuf_v1_create_params(Resource *resource, uint32_t params_id) override; }; class LinuxDmaBufV1ClientBufferPrivate : public ClientBufferPrivate, public QtWaylandServer::wl_buffer { public: QSize size; quint32 format; quint32 flags; QVector<LinuxDmaBufV1Plane> planes; bool hasAlphaChannel = false; protected: void buffer_destroy(Resource *resource) override; }; class LinuxDmaBufParamsV1 : public QtWaylandServer::zwp_linux_buffer_params_v1 { public: LinuxDmaBufParamsV1(LinuxDmaBufV1ClientBufferIntegration *integration, ::wl_resource *resource); ~LinuxDmaBufParamsV1() override; protected: void zwp_linux_buffer_params_v1_destroy_resource(Resource *resource) override; void zwp_linux_buffer_params_v1_destroy(Resource *resource) override; void 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) override; void zwp_linux_buffer_params_v1_create(Resource *resource, int32_t width, int32_t height, uint32_t format, uint32_t flags) override; void 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) override; private: bool test(Resource *resource, uint32_t width, uint32_t height); LinuxDmaBufV1ClientBufferIntegration *m_integration; QVector<LinuxDmaBufV1Plane> m_planes; int m_planeCount = 0; bool m_isUsed = false; }; LinuxDmaBufV1ClientBufferIntegrationPrivate::LinuxDmaBufV1ClientBufferIntegrationPrivate(LinuxDmaBufV1ClientBufferIntegration *q, Display *display) : QtWaylandServer::zwp_linux_dmabuf_v1(*display, s_version) , q(q) { } void LinuxDmaBufV1ClientBufferIntegrationPrivate::zwp_linux_dmabuf_v1_bind_resource(Resource *resource) { for (auto it = supportedModifiers.constBegin(); it != supportedModifiers.constEnd(); ++it) { const uint32_t format = it.key(); QSet<uint64_t> modifiers = it.value(); if (modifiers.isEmpty()) { modifiers.insert(DRM_FORMAT_MOD_INVALID); } for (const uint64_t &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 LinuxDmaBufV1ClientBufferIntegrationPrivate::zwp_linux_dmabuf_v1_destroy(Resource *resource) { wl_resource_destroy(resource->handle); } void LinuxDmaBufV1ClientBufferIntegrationPrivate::zwp_linux_dmabuf_v1_create_params(Resource *resource, uint32_t params_id) { wl_resource *paramsResource = wl_resource_create(resource->client(), &zwp_linux_buffer_params_v1_interface, resource->version(), params_id); if (!paramsResource) { wl_resource_post_no_memory(resource->handle); return; } new LinuxDmaBufParamsV1(q, paramsResource); } LinuxDmaBufParamsV1::LinuxDmaBufParamsV1(LinuxDmaBufV1ClientBufferIntegration *integration, ::wl_resource *resource) : QtWaylandServer::zwp_linux_buffer_params_v1(resource) , m_integration(integration) , m_planes(4) { } LinuxDmaBufParamsV1::~LinuxDmaBufParamsV1() { for (const LinuxDmaBufV1Plane &plane : m_planes) { if (plane.fd != -1) { close(plane.fd); } } } void LinuxDmaBufParamsV1::zwp_linux_buffer_params_v1_destroy_resource(Resource *resource) { Q_UNUSED(resource) delete this; } void LinuxDmaBufParamsV1::zwp_linux_buffer_params_v1_destroy(Resource *resource) { wl_resource_destroy(resource->handle); } void LinuxDmaBufParamsV1::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) { if (Q_UNLIKELY(m_isUsed)) { wl_resource_post_error(resource->handle, error_already_used, "the params object has already been used to create a wl_buffer"); close(fd); return; } if (Q_UNLIKELY(plane_idx >= uint(m_planes.size()))) { wl_resource_post_error(resource->handle, error_plane_idx, "plane index %d is out of bounds", plane_idx); close(fd); return; } LinuxDmaBufV1Plane &plane = m_planes[plane_idx]; if (Q_UNLIKELY(plane.fd != -1)) { wl_resource_post_error(resource->handle, error_plane_set, "the plane index %d was already set", plane_idx); close(fd); return; } plane.fd = fd; plane.modifier = (quint64(modifier_hi) << 32) | modifier_lo; plane.offset = offset; plane.stride = stride; m_planeCount++; } void LinuxDmaBufParamsV1::zwp_linux_buffer_params_v1_create(Resource *resource, int32_t width, int32_t height, uint32_t format, uint32_t flags) { if (Q_UNLIKELY(m_isUsed)) { wl_resource_post_error(resource->handle, error_already_used, "the params object has already been used to create a wl_buffer"); return; } if (Q_UNLIKELY(!test(resource, width, height))) { return; } m_isUsed = true; m_planes.resize(m_planeCount); LinuxDmaBufV1ClientBuffer *clientBuffer = m_integration->rendererInterface()->importBuffer( m_planes, format, QSize(width, height), flags); if (!clientBuffer) { send_failed(resource->handle); return; } m_planes.clear(); // the ownership of file descriptors has been moved to the buffer wl_resource *bufferResource = wl_resource_create(resource->client(), &wl_buffer_interface, 1, 0); if (!bufferResource) { delete clientBuffer; wl_resource_post_no_memory(resource->handle); return; } clientBuffer->initialize(bufferResource); send_created(resource->handle, bufferResource); DisplayPrivate *displayPrivate = DisplayPrivate::get(m_integration->display()); displayPrivate->registerClientBuffer(clientBuffer); } void LinuxDmaBufParamsV1::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 (Q_UNLIKELY(m_isUsed)) { wl_resource_post_error(resource->handle, error_already_used, "the params object has already been used to create a wl_buffer"); return; } if (Q_UNLIKELY(!test(resource, width, height))) { return; } m_isUsed = true; m_planes.resize(m_planeCount); LinuxDmaBufV1ClientBuffer *clientBuffer = m_integration->rendererInterface()->importBuffer( m_planes, format, QSize(width, height), flags); if (!clientBuffer) { wl_resource_post_error(resource->handle, error_invalid_wl_buffer, "importing the supplied dmabufs failed"); return; } m_planes.clear(); // the ownership of file descriptors has been moved to the buffer wl_resource *bufferResource = wl_resource_create(resource->client(), &wl_buffer_interface, 1, buffer_id); if (!bufferResource) { delete clientBuffer; wl_resource_post_no_memory(resource->handle); return; } clientBuffer->initialize(bufferResource); DisplayPrivate *displayPrivate = DisplayPrivate::get(m_integration->display()); displayPrivate->registerClientBuffer(clientBuffer); } bool LinuxDmaBufParamsV1::test(Resource *resource, uint32_t width, uint32_t height) { if (Q_UNLIKELY(!m_planeCount)) { wl_resource_post_error(resource->handle, error_incomplete, "no planes have been specified"); return false; } // Check for holes in the dmabuf set (e.g. [0, 1, 3]). for (int i = 0; i < m_planeCount; ++i) { if (m_planes[i].fd == -1) { wl_resource_post_error(resource->handle, error_incomplete, "no dmabuf has been added for plane %d", i); return false; } } if (Q_UNLIKELY(width == 0 || height == 0)) { wl_resource_post_error(resource->handle, error_invalid_dimensions, "invalid width %d or height %d", width, height); return false; } for (int i = 0; i < m_planeCount; ++i) { const LinuxDmaBufV1Plane &plane = m_planes.at(i); // Check for overflows. if (Q_UNLIKELY(uint64_t(plane.offset) + plane.stride > UINT32_MAX)) { wl_resource_post_error(resource->handle, error_out_of_bounds, "size overflow for plane %d", i); return false; } if (Q_UNLIKELY(i == 0 && uint64_t(plane.offset) + uint64_t(plane.stride) * height > UINT32_MAX)) { wl_resource_post_error(resource->handle, error_out_of_bounds, "size overflow for plane %d", i); return false; } // Don't report an error as it might be caused by the kernel not supporting // seeking on dmabuf. const off_t size = lseek(plane.fd, 0, SEEK_END); if (size == -1) { continue; } if (Q_UNLIKELY(plane.offset >= size)) { wl_resource_post_error(resource->handle, error_out_of_bounds, "invalid offset %i for plane %d", plane.offset, i); return false; } if (Q_UNLIKELY(plane.offset + plane.stride > size)) { wl_resource_post_error(resource->handle, error_out_of_bounds, "invalid stride %i for plane %d", plane.stride, i); return false; } // Only valid for first plane as other planes might be sub-sampled according to // fourcc format. if (Q_UNLIKELY(i == 0 && plane.offset + plane.stride * height > size)) { wl_resource_post_error(resource->handle, error_out_of_bounds, "invalid buffer stride of height for plane %d", i); return false; } } return true; } LinuxDmaBufV1ClientBufferIntegration::LinuxDmaBufV1ClientBufferIntegration(Display *display) : ClientBufferIntegration(display) , d(new LinuxDmaBufV1ClientBufferIntegrationPrivate(this, display)) { } LinuxDmaBufV1ClientBufferIntegration::~LinuxDmaBufV1ClientBufferIntegration() { } LinuxDmaBufV1ClientBufferIntegration::RendererInterface *LinuxDmaBufV1ClientBufferIntegration::rendererInterface() const { return d->rendererInterface; } void LinuxDmaBufV1ClientBufferIntegration::setRendererInterface(RendererInterface *rendererInterface) { d->rendererInterface = rendererInterface; } void LinuxDmaBufV1ClientBufferIntegration::setSupportedFormatsWithModifiers(const QHash<uint32_t, QSet<uint64_t>> &set) { d->supportedModifiers = set; } static bool testAlphaChannel(uint32_t drmFormat) { switch (drmFormat) { case DRM_FORMAT_ARGB4444: case DRM_FORMAT_ABGR4444: case DRM_FORMAT_RGBA4444: case DRM_FORMAT_BGRA4444: case DRM_FORMAT_ARGB1555: case DRM_FORMAT_ABGR1555: case DRM_FORMAT_RGBA5551: case DRM_FORMAT_BGRA5551: case DRM_FORMAT_ARGB8888: case DRM_FORMAT_ABGR8888: case DRM_FORMAT_RGBA8888: case DRM_FORMAT_BGRA8888: case DRM_FORMAT_ARGB2101010: case DRM_FORMAT_ABGR2101010: case DRM_FORMAT_RGBA1010102: case DRM_FORMAT_BGRA1010102: case DRM_FORMAT_XRGB8888_A8: case DRM_FORMAT_XBGR8888_A8: case DRM_FORMAT_RGBX8888_A8: case DRM_FORMAT_BGRX8888_A8: case DRM_FORMAT_RGB888_A8: case DRM_FORMAT_BGR888_A8: case DRM_FORMAT_RGB565_A8: case DRM_FORMAT_BGR565_A8: return true; default: return false; } } void LinuxDmaBufV1ClientBufferPrivate::buffer_destroy(Resource *resource) { wl_resource_destroy(resource->handle); } LinuxDmaBufV1ClientBuffer::LinuxDmaBufV1ClientBuffer(const QSize &size, quint32 format, quint32 flags, const QVector<LinuxDmaBufV1Plane> &planes) : ClientBuffer(*new LinuxDmaBufV1ClientBufferPrivate) { Q_D(LinuxDmaBufV1ClientBuffer); d->size = size; d->format = format; d->flags = flags; d->planes = planes; d->hasAlphaChannel = testAlphaChannel(format); } LinuxDmaBufV1ClientBuffer::~LinuxDmaBufV1ClientBuffer() { Q_D(LinuxDmaBufV1ClientBuffer); for (int i = 0; i < d->planes.count(); ++i) { if (d->planes[i].fd != -1) { close(d->planes[i].fd); d->planes[i].fd = -1; } } } void LinuxDmaBufV1ClientBuffer::initialize(wl_resource *resource) { Q_D(LinuxDmaBufV1ClientBuffer); d->init(resource); ClientBuffer::initialize(resource); } quint32 LinuxDmaBufV1ClientBuffer::format() const { Q_D(const LinuxDmaBufV1ClientBuffer); return d->format; } quint32 LinuxDmaBufV1ClientBuffer::flags() const { Q_D(const LinuxDmaBufV1ClientBuffer); return d->flags; } QVector<LinuxDmaBufV1Plane> LinuxDmaBufV1ClientBuffer::planes() const { Q_D(const LinuxDmaBufV1ClientBuffer); return d->planes; } QSize LinuxDmaBufV1ClientBuffer::size() const { Q_D(const LinuxDmaBufV1ClientBuffer); return d->size; } bool LinuxDmaBufV1ClientBuffer::hasAlphaChannel() const { Q_D(const LinuxDmaBufV1ClientBuffer); return d->hasAlphaChannel; } ClientBuffer::Origin LinuxDmaBufV1ClientBuffer::origin() const { Q_D(const LinuxDmaBufV1ClientBuffer); if (d->flags & QtWaylandServer::zwp_linux_buffer_params_v1::flags_y_invert) { return ClientBuffer::Origin::BottomLeft; } else { return ClientBuffer::Origin::TopLeft; } } } // namespace KWaylandServer