2020-08-02 22:22:19 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
2017-05-09 19:00:33 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-FileCopyrightText: 2017 Roman Gilg <subdiff@gmail.com>
|
|
|
|
SPDX-FileCopyrightText: 2015 Martin Gräßlin <mgraesslin@kde.org>
|
2017-05-09 19:00:33 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
2017-05-09 19:00:33 +00:00
|
|
|
#include "drm_buffer_gbm.h"
|
2017-10-05 16:58:57 +00:00
|
|
|
#include "gbm_surface.h"
|
2017-05-09 19:00:33 +00:00
|
|
|
|
|
|
|
#include "logging.h"
|
|
|
|
|
|
|
|
// system
|
|
|
|
#include <sys/mman.h>
|
2019-07-09 19:19:26 +00:00
|
|
|
// c++
|
|
|
|
#include <cerrno>
|
2017-05-09 19:00:33 +00:00
|
|
|
// drm
|
|
|
|
#include <xf86drm.h>
|
2017-11-12 20:00:02 +00:00
|
|
|
#include <xf86drmMode.h>
|
2017-05-09 19:00:33 +00:00
|
|
|
#include <gbm.h>
|
2021-02-02 13:26:43 +00:00
|
|
|
// KWaylandServer
|
|
|
|
#include "KWaylandServer/buffer_interface.h"
|
2017-05-09 19:00:33 +00:00
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
|
|
|
// DrmSurfaceBuffer
|
2017-11-12 20:00:02 +00:00
|
|
|
DrmSurfaceBuffer::DrmSurfaceBuffer(int fd, const std::shared_ptr<GbmSurface> &surface)
|
|
|
|
: DrmBuffer(fd)
|
2017-05-09 19:00:33 +00:00
|
|
|
, m_surface(surface)
|
|
|
|
{
|
2017-10-05 16:58:57 +00:00
|
|
|
m_bo = m_surface->lockFrontBuffer();
|
2017-05-09 19:00:33 +00:00
|
|
|
if (!m_bo) {
|
|
|
|
qCWarning(KWIN_DRM) << "Locking front buffer failed";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_size = QSize(gbm_bo_get_width(m_bo), gbm_bo_get_height(m_bo));
|
2017-11-12 20:00:02 +00:00
|
|
|
if (drmModeAddFB(fd, m_size.width(), m_size.height(), 24, 32, gbm_bo_get_stride(m_bo), gbm_bo_get_handle(m_bo).u32, &m_bufferId) != 0) {
|
2017-05-09 19:00:33 +00:00
|
|
|
qCWarning(KWIN_DRM) << "drmModeAddFB failed";
|
|
|
|
}
|
|
|
|
gbm_bo_set_user_data(m_bo, this, nullptr);
|
|
|
|
}
|
|
|
|
|
2021-02-02 13:26:43 +00:00
|
|
|
DrmSurfaceBuffer::DrmSurfaceBuffer(int fd, gbm_bo *buffer, KWaylandServer::BufferInterface *bufferInterface)
|
2020-11-28 17:53:41 +00:00
|
|
|
: DrmBuffer(fd)
|
|
|
|
, m_bo(buffer)
|
2021-02-02 13:26:43 +00:00
|
|
|
, m_bufferInterface(bufferInterface)
|
2020-11-28 17:53:41 +00:00
|
|
|
{
|
2021-02-02 13:26:43 +00:00
|
|
|
if (m_bufferInterface) {
|
|
|
|
m_bufferInterface->ref();
|
|
|
|
connect(m_bufferInterface, &KWaylandServer::BufferInterface::aboutToBeDestroyed, this, &DrmSurfaceBuffer::clearBufferInterface);
|
|
|
|
}
|
2020-11-28 17:53:41 +00:00
|
|
|
m_size = QSize(gbm_bo_get_width(m_bo), gbm_bo_get_height(m_bo));
|
|
|
|
if (drmModeAddFB(fd, m_size.width(), m_size.height(), 24, 32, gbm_bo_get_stride(m_bo), gbm_bo_get_handle(m_bo).u32, &m_bufferId) != 0) {
|
|
|
|
qCWarning(KWIN_DRM) << "drmModeAddFB failed";
|
|
|
|
}
|
|
|
|
gbm_bo_set_user_data(m_bo, this, nullptr);
|
|
|
|
}
|
|
|
|
|
2017-05-09 19:00:33 +00:00
|
|
|
DrmSurfaceBuffer::~DrmSurfaceBuffer()
|
|
|
|
{
|
|
|
|
if (m_bufferId) {
|
2017-11-12 20:00:02 +00:00
|
|
|
drmModeRmFB(fd(), m_bufferId);
|
2017-05-09 19:00:33 +00:00
|
|
|
}
|
|
|
|
releaseGbm();
|
2021-02-02 13:26:43 +00:00
|
|
|
if (m_bufferInterface) {
|
|
|
|
clearBufferInterface();
|
|
|
|
}
|
2017-05-09 19:00:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DrmSurfaceBuffer::releaseGbm()
|
|
|
|
{
|
2020-11-28 17:53:41 +00:00
|
|
|
if (m_surface) {
|
|
|
|
m_surface->releaseBuffer(m_bo);
|
|
|
|
}
|
2017-10-05 16:58:57 +00:00
|
|
|
m_bo = nullptr;
|
2017-05-09 19:00:33 +00:00
|
|
|
}
|
|
|
|
|
2021-02-02 13:26:43 +00:00
|
|
|
void DrmSurfaceBuffer::clearBufferInterface()
|
|
|
|
{
|
|
|
|
disconnect(m_bufferInterface, &KWaylandServer::BufferInterface::aboutToBeDestroyed, this, &DrmSurfaceBuffer::clearBufferInterface);
|
|
|
|
m_bufferInterface->unref();
|
|
|
|
m_bufferInterface = nullptr;
|
|
|
|
}
|
|
|
|
|
2017-05-09 19:00:33 +00:00
|
|
|
}
|