2021-07-20 17:02:13 +00:00
|
|
|
/*
|
|
|
|
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>
|
2021-11-23 13:15:41 +00:00
|
|
|
SPDX-FileCopyrightText: 2021 Xaver Hugl <xaver.hugl@gmail.com>
|
2021-07-20 17:02:13 +00:00
|
|
|
|
|
|
|
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"
|
2021-11-23 10:14:37 +00:00
|
|
|
#include "linuxdmabufv1clientbuffer_p.h"
|
2021-11-23 13:15:41 +00:00
|
|
|
#include "logging.h"
|
|
|
|
#include "surface_interface_p.h"
|
|
|
|
|
|
|
|
#include <QTemporaryFile>
|
|
|
|
#include <fcntl.h>
|
2021-07-20 17:02:13 +00:00
|
|
|
|
|
|
|
namespace KWaylandServer
|
|
|
|
{
|
2021-11-23 13:15:41 +00:00
|
|
|
static const int s_version = 4;
|
2021-07-20 17:02:13 +00:00
|
|
|
|
|
|
|
LinuxDmaBufV1ClientBufferIntegrationPrivate::LinuxDmaBufV1ClientBufferIntegrationPrivate(LinuxDmaBufV1ClientBufferIntegration *q, Display *display)
|
|
|
|
: QtWaylandServer::zwp_linux_dmabuf_v1(*display, s_version)
|
|
|
|
, q(q)
|
2021-11-23 13:15:41 +00:00
|
|
|
, defaultFeedback(new LinuxDmaBufV1Feedback(q))
|
2021-07-20 17:02:13 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void LinuxDmaBufV1ClientBufferIntegrationPrivate::zwp_linux_dmabuf_v1_bind_resource(Resource *resource)
|
|
|
|
{
|
2021-11-23 13:15:41 +00:00
|
|
|
if (resource->version() < ZWP_LINUX_DMABUF_V1_GET_DEFAULT_FEEDBACK_SINCE_VERSION) {
|
|
|
|
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);
|
|
|
|
}
|
2021-07-20 17:02:13 +00:00
|
|
|
|
2021-11-23 13:15:41 +00:00
|
|
|
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);
|
|
|
|
}
|
2021-07-20 17:02:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-23 13:15:41 +00:00
|
|
|
void LinuxDmaBufV1ClientBufferIntegrationPrivate::zwp_linux_dmabuf_v1_get_default_feedback(Resource *resource, uint32_t id)
|
|
|
|
{
|
|
|
|
LinuxDmaBufV1FeedbackPrivate::get(defaultFeedback.data())->add(resource->client(), id, resource->version());
|
|
|
|
}
|
|
|
|
|
|
|
|
void LinuxDmaBufV1ClientBufferIntegrationPrivate::zwp_linux_dmabuf_v1_get_surface_feedback(Resource *resource, uint32_t id, wl_resource *surfaceResource)
|
|
|
|
{
|
|
|
|
auto surface = SurfaceInterface::get(surfaceResource);
|
|
|
|
if (!surface) {
|
|
|
|
qCWarning(KWAYLAND_SERVER) << "requested surface feedback for nonexistant surface!";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto surfacePrivate = SurfaceInterfacePrivate::get(surface);
|
|
|
|
if (!surfacePrivate->dmabufFeedbackV1) {
|
|
|
|
surfacePrivate->dmabufFeedbackV1.reset(new LinuxDmaBufV1Feedback(q));
|
|
|
|
}
|
|
|
|
LinuxDmaBufV1FeedbackPrivate::get(surfacePrivate->dmabufFeedbackV1.data())->add(resource->client(), id, resource->version());
|
|
|
|
}
|
|
|
|
|
2021-07-20 17:02:13 +00:00
|
|
|
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)
|
|
|
|
{
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource *paramsResource = wl_resource_create(resource->client(), &zwp_linux_buffer_params_v1_interface, resource->version(), params_id);
|
2021-07-20 17:02:13 +00:00
|
|
|
if (!paramsResource) {
|
|
|
|
wl_resource_post_no_memory(resource->handle);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
new LinuxDmaBufParamsV1(q, paramsResource);
|
|
|
|
}
|
|
|
|
|
2021-11-23 13:15:41 +00:00
|
|
|
LinuxDmaBufV1ClientBufferIntegrationPrivate *LinuxDmaBufV1ClientBufferIntegrationPrivate::get(LinuxDmaBufV1ClientBufferIntegration *integration)
|
|
|
|
{
|
|
|
|
return integration->d.data();
|
|
|
|
}
|
|
|
|
|
2021-07-20 17:02:13 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2021-08-29 05:11:06 +00:00
|
|
|
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)
|
2021-07-20 17:02:13 +00:00
|
|
|
{
|
|
|
|
if (Q_UNLIKELY(m_isUsed)) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_already_used, "the params object has already been used to create a wl_buffer");
|
2021-07-20 17:02:13 +00:00
|
|
|
close(fd);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Q_UNLIKELY(plane_idx >= uint(m_planes.size()))) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_plane_idx, "plane index %d is out of bounds", plane_idx);
|
2021-07-20 17:02:13 +00:00
|
|
|
close(fd);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
LinuxDmaBufV1Plane &plane = m_planes[plane_idx];
|
|
|
|
|
|
|
|
if (Q_UNLIKELY(plane.fd != -1)) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_plane_set, "the plane index %d was already set", plane_idx);
|
2021-07-20 17:02:13 +00:00
|
|
|
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)) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_already_used, "the params object has already been used to create a wl_buffer");
|
2021-07-20 17:02:13 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Q_UNLIKELY(!test(resource, width, height))) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_isUsed = true;
|
|
|
|
m_planes.resize(m_planeCount);
|
|
|
|
|
2021-08-29 05:11:06 +00:00
|
|
|
LinuxDmaBufV1ClientBuffer *clientBuffer = m_integration->rendererInterface()->importBuffer(m_planes, format, QSize(width, height), flags);
|
2021-07-20 17:02:13 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2021-08-29 05:11:06 +00:00
|
|
|
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)
|
2021-07-20 17:02:13 +00:00
|
|
|
{
|
|
|
|
if (Q_UNLIKELY(m_isUsed)) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_already_used, "the params object has already been used to create a wl_buffer");
|
2021-07-20 17:02:13 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Q_UNLIKELY(!test(resource, width, height))) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
m_isUsed = true;
|
|
|
|
m_planes.resize(m_planeCount);
|
|
|
|
|
2021-08-29 05:11:06 +00:00
|
|
|
LinuxDmaBufV1ClientBuffer *clientBuffer = m_integration->rendererInterface()->importBuffer(m_planes, format, QSize(width, height), flags);
|
2021-07-20 17:02:13 +00:00
|
|
|
if (!clientBuffer) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_invalid_wl_buffer, "importing the supplied dmabufs failed");
|
2021-07-20 17:02:13 +00:00
|
|
|
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) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_incomplete, "no dmabuf has been added for plane %d", i);
|
2021-07-20 17:02:13 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Q_UNLIKELY(width == 0 || height == 0)) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_invalid_dimensions, "invalid width %d or height %d", width, height);
|
2021-07-20 17:02:13 +00:00
|
|
|
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)) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_out_of_bounds, "size overflow for plane %d", i);
|
2021-07-20 17:02:13 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-29 05:11:06 +00:00
|
|
|
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);
|
2021-07-20 17:02:13 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-29 05:11:06 +00:00
|
|
|
// Don't report an error as it might be caused by the kernel not supporting
|
2021-07-20 17:02:13 +00:00
|
|
|
// seeking on dmabuf.
|
|
|
|
const off_t size = lseek(plane.fd, 0, SEEK_END);
|
|
|
|
if (size == -1) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Q_UNLIKELY(plane.offset >= size)) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_out_of_bounds, "invalid offset %i for plane %d", plane.offset, i);
|
2021-07-20 17:02:13 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Q_UNLIKELY(plane.offset + plane.stride > size)) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_out_of_bounds, "invalid stride %i for plane %d", plane.stride, i);
|
2021-07-20 17:02:13 +00:00
|
|
|
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)) {
|
2021-08-29 05:11:06 +00:00
|
|
|
wl_resource_post_error(resource->handle, error_out_of_bounds, "invalid buffer stride of height for plane %d", i);
|
2021-07-20 17:02:13 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2021-11-23 13:15:41 +00:00
|
|
|
void LinuxDmaBufV1ClientBufferIntegration::setSupportedFormatsWithModifiers(dev_t mainDevice, const QHash<uint32_t, QSet<uint64_t>> &set)
|
2021-07-20 17:02:13 +00:00
|
|
|
{
|
2021-11-23 13:15:41 +00:00
|
|
|
if (d->supportedModifiers != set || d->mainDevice != mainDevice) {
|
|
|
|
d->supportedModifiers = set;
|
|
|
|
d->mainDevice = mainDevice;
|
|
|
|
d->table.reset(new LinuxDmaBufV1FormatTable(set));
|
|
|
|
LinuxDmaBufV1Feedback::Tranche tranche;
|
|
|
|
tranche.device = mainDevice;
|
|
|
|
tranche.flags = {};
|
|
|
|
tranche.formatTable = set;
|
|
|
|
d->defaultFeedback->setTranches({tranche});
|
|
|
|
}
|
2021-07-20 17:02:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2021-08-29 05:11:06 +00:00
|
|
|
LinuxDmaBufV1ClientBuffer::LinuxDmaBufV1ClientBuffer(const QSize &size, quint32 format, quint32 flags, const QVector<LinuxDmaBufV1Plane> &planes)
|
2021-07-20 17:02:13 +00:00
|
|
|
: 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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-23 13:15:41 +00:00
|
|
|
LinuxDmaBufV1Feedback::LinuxDmaBufV1Feedback(LinuxDmaBufV1ClientBufferIntegration *integration)
|
|
|
|
: d(new LinuxDmaBufV1FeedbackPrivate(LinuxDmaBufV1ClientBufferIntegrationPrivate::get(integration)))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
LinuxDmaBufV1Feedback::~LinuxDmaBufV1Feedback() = default;
|
|
|
|
|
|
|
|
void LinuxDmaBufV1Feedback::setTranches(const QVector<Tranche> &tranches)
|
|
|
|
{
|
|
|
|
if (d->m_tranches != tranches) {
|
|
|
|
d->m_tranches = tranches;
|
|
|
|
const auto &map = d->resourceMap();
|
|
|
|
for (const auto &resource : map) {
|
|
|
|
d->send(resource);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
LinuxDmaBufV1FeedbackPrivate *LinuxDmaBufV1FeedbackPrivate::get(LinuxDmaBufV1Feedback *q)
|
|
|
|
{
|
|
|
|
return q->d.data();
|
|
|
|
}
|
|
|
|
|
|
|
|
LinuxDmaBufV1FeedbackPrivate::LinuxDmaBufV1FeedbackPrivate(LinuxDmaBufV1ClientBufferIntegrationPrivate *bufferintegration)
|
|
|
|
: m_bufferintegration(bufferintegration)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator==(const LinuxDmaBufV1Feedback::Tranche &t1, const LinuxDmaBufV1Feedback::Tranche &t2)
|
|
|
|
{
|
|
|
|
return t1.device == t2.device && t1.flags == t2.flags && t1.formatTable == t2.formatTable;
|
|
|
|
}
|
|
|
|
|
|
|
|
void LinuxDmaBufV1FeedbackPrivate::send(Resource *resource)
|
|
|
|
{
|
|
|
|
send_format_table(resource->handle, m_bufferintegration->table->fd, m_bufferintegration->table->size);
|
|
|
|
QByteArray bytes;
|
|
|
|
bytes.append(reinterpret_cast<const char *>(&m_bufferintegration->mainDevice), sizeof(dev_t));
|
|
|
|
send_main_device(resource->handle, bytes);
|
|
|
|
const auto &sendTranche = [this, resource](const LinuxDmaBufV1Feedback::Tranche &tranche) {
|
|
|
|
QByteArray targetDevice;
|
|
|
|
targetDevice.append(reinterpret_cast<const char *>(&tranche.device), sizeof(dev_t));
|
|
|
|
QByteArray indices;
|
|
|
|
for (auto it = tranche.formatTable.begin(); it != tranche.formatTable.end(); it++) {
|
|
|
|
const uint32_t format = it.key();
|
|
|
|
for (const auto &mod : qAsConst(it.value())) {
|
|
|
|
uint16_t index = m_bufferintegration->table->indices[std::pair<uint32_t, uint64_t>(format, mod)];
|
|
|
|
indices.append(reinterpret_cast<const char *>(&index), 2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
send_tranche_target_device(resource->handle, targetDevice);
|
|
|
|
send_tranche_formats(resource->handle, indices);
|
|
|
|
send_tranche_flags(resource->handle, static_cast<uint32_t>(tranche.flags));
|
|
|
|
send_tranche_done(resource->handle);
|
|
|
|
};
|
|
|
|
for (const auto &tranche : qAsConst(m_tranches)) {
|
|
|
|
sendTranche(tranche);
|
|
|
|
}
|
|
|
|
// send default hints as the last fallback tranche
|
|
|
|
if (this != get(m_bufferintegration->defaultFeedback.data())) {
|
|
|
|
sendTranche(get(m_bufferintegration->defaultFeedback.data())->m_tranches[0]);
|
|
|
|
}
|
|
|
|
send_done(resource->handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
void LinuxDmaBufV1FeedbackPrivate::zwp_linux_dmabuf_feedback_v1_bind_resource(Resource *resource)
|
|
|
|
{
|
|
|
|
send(resource);
|
|
|
|
}
|
|
|
|
|
|
|
|
void LinuxDmaBufV1FeedbackPrivate::zwp_linux_dmabuf_feedback_v1_destroy(Resource *resource)
|
|
|
|
{
|
|
|
|
wl_resource_destroy(resource->handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct linux_dmabuf_feedback_v1_table_entry {
|
|
|
|
uint32_t format;
|
|
|
|
uint32_t pad; // unused
|
|
|
|
uint64_t modifier;
|
|
|
|
};
|
|
|
|
|
|
|
|
LinuxDmaBufV1FormatTable::LinuxDmaBufV1FormatTable(const QHash<uint32_t, QSet<uint64_t>> &supportedModifiers)
|
|
|
|
{
|
|
|
|
QVector<linux_dmabuf_feedback_v1_table_entry> data;
|
|
|
|
for (auto it = supportedModifiers.begin(); it != supportedModifiers.end(); it++) {
|
|
|
|
const uint32_t format = it.key();
|
|
|
|
for (const uint64_t &mod : *it) {
|
|
|
|
indices.insert({format, mod}, data.size());
|
|
|
|
data.append({format, 0, mod});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
size = data.size() * sizeof(linux_dmabuf_feedback_v1_table_entry);
|
|
|
|
QScopedPointer<QTemporaryFile> tmp(new QTemporaryFile());
|
|
|
|
if (!tmp->open()) {
|
|
|
|
qCWarning(KWAYLAND_SERVER) << "Failed to create keymap file:" << tmp->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
fd = open(tmp->fileName().toUtf8().constData(), O_RDONLY | O_CLOEXEC);
|
|
|
|
if (fd < 0) {
|
|
|
|
qCWarning(KWAYLAND_SERVER) << "Could not create readonly shm fd!" << strerror(errno);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
unlink(tmp->fileName().toUtf8().constData());
|
|
|
|
if (!tmp->resize(size)) {
|
|
|
|
qCWarning(KWAYLAND_SERVER) << "Failed to resize keymap file:" << tmp->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
uchar *address = tmp->map(0, size);
|
|
|
|
if (!address) {
|
|
|
|
qCWarning(KWAYLAND_SERVER) << "Failed to map keymap file:" << tmp->errorString();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
memcpy(address, data.data(), size);
|
|
|
|
}
|
|
|
|
|
|
|
|
LinuxDmaBufV1FormatTable::~LinuxDmaBufV1FormatTable()
|
|
|
|
{
|
|
|
|
if (fd != -1) {
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-20 17:02:13 +00:00
|
|
|
} // namespace KWaylandServer
|