kwin/src/wayland/linuxdmabufv1clientbuffer.cpp
Vlad Zahorodnii 0d2879c62d Split BufferInterface
Currently, the BufferInterface encapsulates all the kinds of client
buffers. This has become a somewhat annoying issue as we want to
reference the shm pool if a shm buffer is destroyed, or have custom
buffer readiness logic for linux dma-buf client buffers.

Implementing all of that with the current abstractions will be
challenging as there's no good separation between different client
buffer types.

This change splits the BufferInterface class in three sub-classes -
DrmClientBuffer, LinuxDmaBufV1ClientBuffer, and ShmClientBuffer.

In addition to that, this change fixes the broken buffer ref'ing api.
2021-07-22 11:41:06 +03:00

470 lines
15 KiB
C++

/*
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