/* KWin - the KDE window manager This file is part of the KDE project. SPDX-FileCopyrightText: 2017 Roman Gilg SPDX-FileCopyrightText: 2015 Martin Gräßlin SPDX-License-Identifier: GPL-2.0-or-later */ #include "drm_buffer_gbm.h" #include "gbm_surface.h" #include "logging.h" #include "drm_gpu.h" // system #include // c++ #include // drm #include #include #include // KWaylandServer #include "KWaylandServer/buffer_interface.h" #include namespace KWin { // DrmSurfaceBuffer DrmSurfaceBuffer::DrmSurfaceBuffer(DrmGpu *gpu, const QSharedPointer &surface) : DrmBuffer(gpu) , m_surface(surface) { m_bo = m_surface->lockFrontBuffer(); if (!m_bo) { qCWarning(KWIN_DRM) << "Locking front buffer failed"; return; } initialize(); } DrmSurfaceBuffer::DrmSurfaceBuffer(DrmGpu *gpu, gbm_bo *buffer, KWaylandServer::BufferInterface *bufferInterface) : DrmBuffer(gpu) , m_bo(buffer) , m_bufferInterface(bufferInterface) { if (m_bufferInterface) { m_bufferInterface->ref(); connect(m_bufferInterface, &KWaylandServer::BufferInterface::aboutToBeDestroyed, this, &DrmSurfaceBuffer::clearBufferInterface); } initialize(); } DrmSurfaceBuffer::~DrmSurfaceBuffer() { if (m_bufferId) { drmModeRmFB(m_gpu->fd(), m_bufferId); } releaseGbm(); if (m_bufferInterface) { clearBufferInterface(); } } void DrmSurfaceBuffer::releaseGbm() { if (m_surface) { m_surface->releaseBuffer(m_bo); } m_bo = nullptr; } void DrmSurfaceBuffer::clearBufferInterface() { disconnect(m_bufferInterface, &KWaylandServer::BufferInterface::aboutToBeDestroyed, this, &DrmSurfaceBuffer::clearBufferInterface); m_bufferInterface->unref(); m_bufferInterface = nullptr; } void DrmSurfaceBuffer::initialize() { m_size = QSize(gbm_bo_get_width(m_bo), gbm_bo_get_height(m_bo)); uint32_t handles[4] = { }; uint32_t strides[4] = { }; uint32_t offsets[4] = { }; uint64_t modifiers[4] = { }; if (gbm_bo_get_handle_for_plane(m_bo, 0).s32 != -1) { for (int i = 0; i < gbm_bo_get_plane_count(m_bo); i++) { handles[i] = gbm_bo_get_handle_for_plane(m_bo, i).u32; strides[i] = gbm_bo_get_stride_for_plane(m_bo, i); offsets[i] = gbm_bo_get_offset(m_bo, i); modifiers[i] = gbm_bo_get_modifier(m_bo); } } else { handles[0] = gbm_bo_get_format(m_bo); strides[0] = gbm_bo_get_stride(m_bo); modifiers[0] = DRM_FORMAT_MOD_INVALID; } if (modifiers[0] != DRM_FORMAT_MOD_INVALID && m_gpu->addFB2ModifiersSupported()) { if (drmModeAddFB2WithModifiers(m_gpu->fd(), m_size.width(), m_size.height(), gbm_bo_get_format(m_bo), handles, strides, offsets, modifiers, &m_bufferId, DRM_MODE_FB_MODIFIERS)) { qCWarning(KWIN_DRM) << "drmModeAddFB2WithModifiers failed!" << strerror(errno); } } else { if (drmModeAddFB2(m_gpu->fd(), m_size.width(), m_size.height(), gbm_bo_get_format(m_bo), handles, strides, offsets, &m_bufferId, 0)) { // fallback if (drmModeAddFB(m_gpu->fd(), m_size.width(), m_size.height(), 24, 32, strides[0], handles[0], &m_bufferId) != 0) { qCWarning(KWIN_DRM) << "drmModeAddFB2 and drmModeAddFB both failed!" << strerror(errno); } } } gbm_bo_set_user_data(m_bo, this, nullptr); } }