kwin/src/wayland/drmclientbuffer.cpp
Vlad Zahorodnii 128570ed53 Restore correct type check for wl_drm client buffers
With the EGLStreams support gone, we can restore the correct type check
for wl_drm client buffers. The spec suggests to query the
EGL_TEXTURE_FORMAT attribute to determine whether the specified wl_buffer
is a wl_drm buffer.
2021-11-10 13:54:56 +02:00

94 lines
2.7 KiB
C++

/*
SPDX-FileCopyrightText: 2021 Vlad Zahorodnii <vlad.zahorodnii@kde.org>
SPDX-License-Identifier: LGPL-2.1-only OR LGPL-3.0-only OR LicenseRef-KDE-Accepted-LGPL
*/
#include "drmclientbuffer.h"
#include "clientbuffer_p.h"
#include "display.h"
#include <EGL/egl.h>
#include <QtGui/qopengl.h>
#ifndef EGL_WL_bind_wayland_display
#define EGL_WAYLAND_Y_INVERTED_WL 0x31DB
#endif
namespace KWaylandServer
{
typedef EGLBoolean (*eglQueryWaylandBufferWL_func)(EGLDisplay dpy, struct wl_resource *buffer, EGLint attribute, EGLint *value);
static eglQueryWaylandBufferWL_func eglQueryWaylandBufferWL = nullptr;
class DrmClientBufferPrivate : public ClientBufferPrivate
{
public:
int textureFormat = 0;
int width = 0;
int height = 0;
int yInverted = 0;
bool hasAlphaChannel = false;
};
DrmClientBuffer::DrmClientBuffer(wl_resource *resource, DrmClientBufferIntegration *integration)
: ClientBuffer(resource, *new DrmClientBufferPrivate)
{
Q_D(DrmClientBuffer);
EGLDisplay eglDisplay = integration->display()->eglDisplay();
eglQueryWaylandBufferWL(eglDisplay, resource, EGL_TEXTURE_FORMAT, &d->textureFormat);
eglQueryWaylandBufferWL(eglDisplay, resource, EGL_WIDTH, &d->width);
eglQueryWaylandBufferWL(eglDisplay, resource, EGL_HEIGHT, &d->height);
if (!eglQueryWaylandBufferWL(eglDisplay, resource, EGL_WAYLAND_Y_INVERTED_WL, &d->yInverted)) {
// If EGL_WAYLAND_Y_INVERTED_WL is unsupported, we must assume that the buffer is inverted.
d->yInverted = true;
}
}
int DrmClientBuffer::textureFormat() const
{
Q_D(const DrmClientBuffer);
return d->textureFormat;
}
QSize DrmClientBuffer::size() const
{
Q_D(const DrmClientBuffer);
return QSize(d->width, d->height);
}
bool DrmClientBuffer::hasAlphaChannel() const
{
Q_D(const DrmClientBuffer);
return d->textureFormat == EGL_TEXTURE_RGBA;
}
ClientBuffer::Origin DrmClientBuffer::origin() const
{
Q_D(const DrmClientBuffer);
return d->yInverted ? Origin::TopLeft : Origin::BottomLeft;
}
DrmClientBufferIntegration::DrmClientBufferIntegration(Display *display)
: ClientBufferIntegration(display)
{
}
ClientBuffer *DrmClientBufferIntegration::createBuffer(::wl_resource *resource)
{
EGLDisplay eglDisplay = display()->eglDisplay();
static bool resolved = false;
if (!resolved && eglDisplay != EGL_NO_DISPLAY) {
eglQueryWaylandBufferWL = (eglQueryWaylandBufferWL_func)eglGetProcAddress("eglQueryWaylandBufferWL");
resolved = true;
}
EGLint format;
if (eglQueryWaylandBufferWL(eglDisplay, resource, EGL_TEXTURE_FORMAT, &format)) {
return new DrmClientBuffer(resource, this);
}
return nullptr;
}
} // namespace KWaylandServer