2021-05-25 22:05:17 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
SPDX-FileCopyrightText: 2021 Xaver Hugl <xaver.hugl@gmail.com>
|
|
|
|
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "drm_pipeline.h"
|
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
#include "logging.h"
|
|
|
|
#include "drm_gpu.h"
|
|
|
|
#include "drm_object_connector.h"
|
|
|
|
#include "drm_object_crtc.h"
|
|
|
|
#include "drm_object_plane.h"
|
|
|
|
#include "drm_buffer.h"
|
|
|
|
#include "cursor.h"
|
|
|
|
#include "session.h"
|
2021-07-26 23:11:50 +00:00
|
|
|
#include "drm_output.h"
|
2021-05-25 22:05:17 +00:00
|
|
|
#include "drm_backend.h"
|
|
|
|
|
|
|
|
#if HAVE_GBM
|
|
|
|
#include <gbm.h>
|
|
|
|
|
|
|
|
#include "egl_gbm_backend.h"
|
|
|
|
#include "drm_buffer_gbm.h"
|
|
|
|
#endif
|
|
|
|
|
2021-08-26 16:17:31 +00:00
|
|
|
#include <drm_fourcc.h>
|
|
|
|
|
2021-05-25 22:05:17 +00:00
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
2021-09-28 08:29:56 +00:00
|
|
|
DrmPipeline::DrmPipeline(DrmConnector *conn)
|
2021-08-07 22:03:09 +00:00
|
|
|
: m_output(nullptr)
|
2021-05-25 22:05:17 +00:00
|
|
|
, m_connector(conn)
|
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
if (!gpu()->atomicModeSetting()) {
|
2021-09-16 10:48:38 +00:00
|
|
|
m_formats.insert(DRM_FORMAT_XRGB8888, {});
|
|
|
|
m_formats.insert(DRM_FORMAT_ARGB8888, {});
|
2021-08-07 18:06:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-28 08:29:56 +00:00
|
|
|
DrmPipeline::~DrmPipeline()
|
2021-05-25 22:05:17 +00:00
|
|
|
{
|
2021-10-08 08:52:01 +00:00
|
|
|
m_output = nullptr;
|
|
|
|
if (m_pageflipPending && m_current.crtc) {
|
|
|
|
pageFlipped({});
|
|
|
|
}
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool DrmPipeline::present(const QSharedPointer<DrmBuffer> &buffer)
|
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
Q_ASSERT(pending.crtc);
|
2021-10-08 08:52:01 +00:00
|
|
|
if (!buffer) {
|
|
|
|
if (m_output) {
|
|
|
|
m_output->presentFailed();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
2021-05-25 22:05:17 +00:00
|
|
|
m_primaryBuffer = buffer;
|
2021-09-28 08:29:56 +00:00
|
|
|
if (gpu()->useEglStreams() && gpu()->eglBackend() != nullptr && gpu() == gpu()->platform()->primaryGpu()) {
|
2021-05-25 22:05:17 +00:00
|
|
|
// EglStreamBackend queues normal page flips through EGL,
|
|
|
|
// modesets etc are performed through DRM-KMS
|
2021-09-28 08:29:56 +00:00
|
|
|
if (!m_connector->needsCommit() && !pending.crtc->needsCommit()) {
|
2021-05-25 22:05:17 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
2021-10-08 08:52:01 +00:00
|
|
|
bool directScanout = false;
|
|
|
|
#if HAVE_GBM
|
|
|
|
// with direct scanout disallow modesets, calling presentFailed() and logging warnings
|
|
|
|
if (auto buf = dynamic_cast<DrmGbmBuffer*>(buffer.data()); buf && buf->clientBuffer()) {
|
|
|
|
directScanout = true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if (gpu()->needsModeset()) {
|
|
|
|
if (directScanout) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
m_modesetPresentPending = true;
|
|
|
|
return gpu()->maybeModeset();
|
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
if (gpu()->atomicModeSetting()) {
|
|
|
|
if (!commitPipelines({this}, CommitMode::Commit)) {
|
2021-06-03 13:50:37 +00:00
|
|
|
// update properties and try again
|
|
|
|
updateProperties();
|
2021-09-28 08:29:56 +00:00
|
|
|
if (!commitPipelines({this}, CommitMode::Commit)) {
|
2021-10-08 08:52:01 +00:00
|
|
|
if (directScanout) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-06-03 13:50:37 +00:00
|
|
|
qCWarning(KWIN_DRM) << "Atomic present failed!" << strerror(errno);
|
|
|
|
printDebugInfo();
|
2021-10-08 08:52:01 +00:00
|
|
|
if (m_output) {
|
|
|
|
m_output->presentFailed();
|
|
|
|
}
|
2021-06-03 13:50:37 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!presentLegacy()) {
|
|
|
|
qCWarning(KWIN_DRM) << "Present failed!" << strerror(errno);
|
2021-10-08 08:52:01 +00:00
|
|
|
if (m_output) {
|
|
|
|
m_output->presentFailed();
|
|
|
|
}
|
2021-06-03 13:50:37 +00:00
|
|
|
return false;
|
|
|
|
}
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
2021-06-03 13:50:37 +00:00
|
|
|
return true;
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
2021-08-07 18:06:40 +00:00
|
|
|
bool DrmPipeline::commitPipelines(const QVector<DrmPipeline*> &pipelines, CommitMode mode)
|
2021-05-25 22:05:17 +00:00
|
|
|
{
|
2021-08-07 18:06:40 +00:00
|
|
|
Q_ASSERT(!pipelines.isEmpty());
|
2021-09-28 08:29:56 +00:00
|
|
|
if (pipelines[0]->gpu()->atomicModeSetting()) {
|
2021-08-07 18:06:40 +00:00
|
|
|
drmModeAtomicReq *req = drmModeAtomicAlloc();
|
|
|
|
if (!req) {
|
|
|
|
qCDebug(KWIN_DRM) << "Failed to allocate drmModeAtomicReq!" << strerror(errno);
|
|
|
|
return false;
|
|
|
|
}
|
2021-10-08 08:52:01 +00:00
|
|
|
uint32_t flags = 0;
|
|
|
|
const auto &failed = [pipelines, req, mode](){
|
2021-09-09 10:52:04 +00:00
|
|
|
drmModeAtomicFree(req);
|
|
|
|
for (const auto &pipeline : pipelines) {
|
|
|
|
pipeline->printDebugInfo();
|
|
|
|
if (pipeline->m_oldTestBuffer) {
|
|
|
|
pipeline->m_primaryBuffer = pipeline->m_oldTestBuffer;
|
|
|
|
pipeline->m_oldTestBuffer = nullptr;
|
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
pipeline->m_connector->rollbackPending();
|
|
|
|
if (pipeline->pending.crtc) {
|
|
|
|
pipeline->pending.crtc->rollbackPending();
|
|
|
|
pipeline->pending.crtc->primaryPlane()->rollbackPending();
|
2021-09-09 10:52:04 +00:00
|
|
|
}
|
2021-10-08 08:52:01 +00:00
|
|
|
if (mode != CommitMode::Test && pipeline->activePending() && pipeline->output()) {
|
|
|
|
pipeline->m_modesetPresentPending = false;
|
|
|
|
pipeline->output()->presentFailed();
|
|
|
|
}
|
2021-09-09 10:52:04 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
};
|
2021-08-07 18:06:40 +00:00
|
|
|
for (const auto &pipeline : pipelines) {
|
2021-09-09 10:52:04 +00:00
|
|
|
if (!pipeline->checkTestBuffer()) {
|
|
|
|
qCWarning(KWIN_DRM) << "Checking test buffer failed for" << mode;
|
|
|
|
return failed();
|
|
|
|
}
|
|
|
|
if (!pipeline->populateAtomicValues(req, flags)) {
|
|
|
|
qCWarning(KWIN_DRM) << "Populating atomic values failed for" << mode;
|
|
|
|
return failed();
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
}
|
2021-10-08 08:52:01 +00:00
|
|
|
bool modeset = flags & DRM_MODE_ATOMIC_ALLOW_MODESET;
|
|
|
|
Q_ASSERT(!modeset || mode != CommitMode::Commit);
|
|
|
|
if (modeset) {
|
|
|
|
// The kernel fails commits with DRM_MODE_PAGE_FLIP_EVENT when a crtc is disabled in the commit
|
|
|
|
// and already was disabled before, to work around some quirks in old userspace.
|
|
|
|
// Instead of using DRM_MODE_PAGE_FLIP_EVENT | DRM_MODE_ATOMIC_NONBLOCK, do the modeset in a blocking
|
|
|
|
// fashion without page flip events and directly call the pageFlipped method afterwards
|
|
|
|
flags = flags & (~DRM_MODE_PAGE_FLIP_EVENT);
|
|
|
|
} else {
|
|
|
|
flags |= DRM_MODE_ATOMIC_NONBLOCK;
|
|
|
|
}
|
|
|
|
if (drmModeAtomicCommit(pipelines[0]->gpu()->fd(), req, (flags & (~DRM_MODE_PAGE_FLIP_EVENT)) | DRM_MODE_ATOMIC_TEST_ONLY, nullptr) != 0) {
|
2021-09-09 10:52:04 +00:00
|
|
|
qCWarning(KWIN_DRM) << "Atomic test for" << mode << "failed!" << strerror(errno);
|
|
|
|
return failed();
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
2021-10-08 08:52:01 +00:00
|
|
|
if (mode != CommitMode::Test && drmModeAtomicCommit(pipelines[0]->gpu()->fd(), req, flags, nullptr) != 0) {
|
2021-09-09 10:52:04 +00:00
|
|
|
qCWarning(KWIN_DRM) << "Atomic commit failed! This should never happen!" << strerror(errno);
|
|
|
|
return failed();
|
|
|
|
}
|
|
|
|
for (const auto &pipeline : pipelines) {
|
|
|
|
pipeline->m_oldTestBuffer = nullptr;
|
2021-09-28 08:29:56 +00:00
|
|
|
pipeline->m_connector->commitPending();
|
|
|
|
if (pipeline->pending.crtc) {
|
|
|
|
pipeline->pending.crtc->commitPending();
|
|
|
|
pipeline->pending.crtc->primaryPlane()->commitPending();
|
2021-08-07 18:06:40 +00:00
|
|
|
}
|
2021-09-09 10:52:04 +00:00
|
|
|
if (mode != CommitMode::Test) {
|
2021-10-08 08:52:01 +00:00
|
|
|
pipeline->m_modesetPresentPending = false;
|
|
|
|
pipeline->m_pageflipPending = true;
|
2021-09-28 08:29:56 +00:00
|
|
|
pipeline->m_connector->commit();
|
|
|
|
if (pipeline->pending.crtc) {
|
|
|
|
pipeline->pending.crtc->primaryPlane()->setNext(pipeline->m_primaryBuffer);
|
|
|
|
pipeline->pending.crtc->commit();
|
|
|
|
pipeline->pending.crtc->primaryPlane()->commit();
|
2021-08-07 18:06:40 +00:00
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
pipeline->m_current = pipeline->pending;
|
2021-10-08 08:52:01 +00:00
|
|
|
if (modeset && pipeline->activePending()) {
|
|
|
|
pipeline->pageFlipped(std::chrono::steady_clock::now().time_since_epoch());
|
|
|
|
}
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
}
|
2021-08-07 18:06:40 +00:00
|
|
|
drmModeAtomicFree(req);
|
2021-09-09 10:52:04 +00:00
|
|
|
return true;
|
2021-05-25 22:05:17 +00:00
|
|
|
} else {
|
2021-09-28 08:29:56 +00:00
|
|
|
bool failure = false;
|
2021-08-07 18:06:40 +00:00
|
|
|
for (const auto &pipeline : pipelines) {
|
2021-09-28 08:29:56 +00:00
|
|
|
if (!pipeline->applyPendingChangesLegacy()) {
|
|
|
|
failure = true;
|
|
|
|
break;
|
2021-08-07 18:06:40 +00:00
|
|
|
}
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
if (failure) {
|
|
|
|
// at least try to revert the config
|
|
|
|
for (const auto &pipeline : pipelines) {
|
2021-10-08 08:52:01 +00:00
|
|
|
pipeline->revertPendingChanges();
|
2021-09-28 08:29:56 +00:00
|
|
|
pipeline->applyPendingChangesLegacy();
|
2021-10-08 08:52:01 +00:00
|
|
|
if (mode == CommitMode::CommitModeset && pipeline->output() && pipeline->activePending()) {
|
|
|
|
pipeline->output()->presentFailed();
|
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
} else {
|
2021-10-08 08:52:01 +00:00
|
|
|
for (const auto &pipeline : pipelines) {
|
|
|
|
pipeline->applyPendingChanges();
|
|
|
|
pipeline->m_current = pipeline->pending;
|
|
|
|
if (mode == CommitMode::CommitModeset && mode != CommitMode::Test && pipeline->activePending()) {
|
|
|
|
pipeline->pageFlipped(std::chrono::steady_clock::now().time_since_epoch());
|
|
|
|
}
|
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DrmPipeline::populateAtomicValues(drmModeAtomicReq *req, uint32_t &flags)
|
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
if (needsModeset()) {
|
|
|
|
prepareModeset();
|
2021-05-25 22:05:17 +00:00
|
|
|
flags |= DRM_MODE_ATOMIC_ALLOW_MODESET;
|
2021-09-28 08:29:56 +00:00
|
|
|
}
|
|
|
|
bool usesEglStreams = gpu()->useEglStreams() && gpu()->eglBackend() != nullptr && gpu() == gpu()->platform()->primaryGpu();
|
|
|
|
if (!usesEglStreams && activePending()) {
|
|
|
|
flags |= DRM_MODE_PAGE_FLIP_EVENT;
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
m_lastFlags = flags;
|
2021-09-28 08:29:56 +00:00
|
|
|
if (pending.crtc) {
|
|
|
|
auto modeSize = m_connector->modes()[pending.modeIndex].size;
|
|
|
|
pending.crtc->primaryPlane()->set(QPoint(0, 0), m_primaryBuffer ? m_primaryBuffer->size() : modeSize, QPoint(0, 0), modeSize);
|
|
|
|
pending.crtc->primaryPlane()->setBuffer(activePending() ? m_primaryBuffer.get() : nullptr);
|
|
|
|
pending.crtc->setPending(DrmCrtc::PropertyIndex::VrrEnabled, pending.syncMode == RenderLoopPrivate::SyncMode::Adaptive);
|
|
|
|
if (pending.gamma != m_current.gamma) {
|
|
|
|
if (pending.gamma) {
|
|
|
|
pending.crtc->setPendingBlob(DrmCrtc::PropertyIndex::Gamma_LUT, pending.gamma->atomicLut, pending.gamma->size * sizeof(drm_color_lut));
|
|
|
|
} else {
|
|
|
|
pending.crtc->setPendingBlob(DrmCrtc::PropertyIndex::Gamma_LUT, nullptr, 256 * sizeof(drm_color_lut));
|
|
|
|
}
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
return m_connector->atomicPopulate(req) && (!pending.crtc || (pending.crtc->atomicPopulate(req) && pending.crtc->primaryPlane()->atomicPopulate(req)));
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool DrmPipeline::presentLegacy()
|
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
if ((!pending.crtc->current() || pending.crtc->current()->needsModeChange(m_primaryBuffer.get())) && !legacyModeset()) {
|
2021-05-25 22:05:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
m_lastFlags = DRM_MODE_PAGE_FLIP_EVENT;
|
2021-09-28 08:29:56 +00:00
|
|
|
QVector<DrmPipeline*> *userData = new QVector<DrmPipeline*>();
|
|
|
|
*userData << this;
|
|
|
|
if (drmModePageFlip(gpu()->fd(), pending.crtc->id(), m_primaryBuffer ? m_primaryBuffer->bufferId() : 0, DRM_MODE_PAGE_FLIP_EVENT, userData) != 0) {
|
2021-05-25 22:05:17 +00:00
|
|
|
qCWarning(KWIN_DRM) << "Page flip failed:" << strerror(errno) << m_primaryBuffer;
|
|
|
|
return false;
|
|
|
|
}
|
2021-10-08 08:52:01 +00:00
|
|
|
m_pageflipPending = true;
|
2021-09-28 08:29:56 +00:00
|
|
|
pending.crtc->setNext(m_primaryBuffer);
|
2021-05-25 22:05:17 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DrmPipeline::checkTestBuffer()
|
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
if (!pending.crtc || (m_primaryBuffer && m_primaryBuffer->size() == sourceSize())) {
|
2021-05-25 22:05:17 +00:00
|
|
|
return true;
|
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
auto backend = gpu()->eglBackend();
|
2021-09-16 11:17:42 +00:00
|
|
|
QSharedPointer<DrmBuffer> buffer;
|
2021-09-17 09:01:18 +00:00
|
|
|
// try to re-use buffers if possible.
|
|
|
|
const auto &checkBuffer = [this, backend, &buffer](const QSharedPointer<DrmBuffer> &buf){
|
2021-09-20 12:27:05 +00:00
|
|
|
const auto &mods = supportedModifiers(buf->format());
|
2021-10-20 01:33:01 +00:00
|
|
|
if (backend && buf->format() == backend->drmFormat()
|
2021-09-20 12:27:05 +00:00
|
|
|
&& (mods.isEmpty() || mods.contains(buf->modifier()))
|
2021-09-17 09:01:18 +00:00
|
|
|
&& buf->size() == sourceSize()) {
|
|
|
|
buffer = buf;
|
|
|
|
}
|
|
|
|
};
|
2021-09-28 08:29:56 +00:00
|
|
|
if (pending.crtc->primaryPlane() && pending.crtc->primaryPlane()->next()) {
|
|
|
|
checkBuffer(pending.crtc->primaryPlane()->next());
|
|
|
|
} else if (pending.crtc->primaryPlane() && pending.crtc->primaryPlane()->current()) {
|
|
|
|
checkBuffer(pending.crtc->primaryPlane()->current());
|
|
|
|
} else if (pending.crtc->next()) {
|
|
|
|
checkBuffer(pending.crtc->next());
|
|
|
|
} else if (pending.crtc->current()) {
|
|
|
|
checkBuffer(pending.crtc->current());
|
2021-09-17 09:01:18 +00:00
|
|
|
}
|
|
|
|
// if we don't have a fitting buffer already, get or create one
|
|
|
|
if (buffer) {
|
2021-05-25 22:05:17 +00:00
|
|
|
#if HAVE_GBM
|
2021-09-17 09:01:18 +00:00
|
|
|
} else if (backend && m_output) {
|
2021-09-16 11:17:42 +00:00
|
|
|
buffer = backend->renderTestFrame(m_output);
|
2021-09-28 08:29:56 +00:00
|
|
|
} else if (backend && gpu()->gbmDevice()) {
|
|
|
|
gbm_bo *bo = gbm_bo_create(gpu()->gbmDevice(), sourceSize().width(), sourceSize().height(), GBM_FORMAT_XRGB8888, GBM_BO_USE_SCANOUT | GBM_BO_USE_RENDERING);
|
2021-05-25 22:05:17 +00:00
|
|
|
if (!bo) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
buffer = QSharedPointer<DrmGbmBuffer>::create(gpu(), bo, nullptr);
|
2021-09-17 09:01:18 +00:00
|
|
|
#endif
|
2021-05-25 22:05:17 +00:00
|
|
|
} else {
|
2021-09-28 08:29:56 +00:00
|
|
|
buffer = QSharedPointer<DrmDumbBuffer>::create(gpu(), sourceSize());
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
if (buffer && buffer->bufferId()) {
|
|
|
|
m_oldTestBuffer = m_primaryBuffer;
|
|
|
|
m_primaryBuffer = buffer;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-08-02 09:03:07 +00:00
|
|
|
bool DrmPipeline::setCursor(const QSharedPointer<DrmDumbBuffer> &buffer, const QPoint &hotspot)
|
2021-05-25 22:05:17 +00:00
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
return pending.crtc->setLegacyCursor(buffer, hotspot);
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool DrmPipeline::moveCursor(QPoint pos)
|
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
return pending.crtc->moveLegacyCursor(pos);
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
2021-09-28 08:29:56 +00:00
|
|
|
void DrmPipeline::prepareModeset()
|
2021-05-25 22:05:17 +00:00
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
if (!pending.crtc) {
|
|
|
|
m_connector->setPending(DrmConnector::PropertyIndex::CrtcId, 0);
|
|
|
|
return;
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
auto mode = m_connector->modes()[pending.modeIndex];
|
|
|
|
|
|
|
|
m_connector->setPending(DrmConnector::PropertyIndex::CrtcId, activePending() ? pending.crtc->id() : 0);
|
|
|
|
if (const auto &prop = m_connector->getProp(DrmConnector::PropertyIndex::Broadcast_RGB)) {
|
|
|
|
prop->setEnum(pending.rgbRange);
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
|
|
|
|
pending.crtc->setPending(DrmCrtc::PropertyIndex::Active, activePending());
|
|
|
|
pending.crtc->setPendingBlob(DrmCrtc::PropertyIndex::ModeId, activePending() ? &mode.mode : nullptr, sizeof(drmModeModeInfo));
|
|
|
|
|
|
|
|
pending.crtc->primaryPlane()->setPending(DrmPlane::PropertyIndex::CrtcId, activePending() ? pending.crtc->id() : 0);
|
|
|
|
pending.crtc->primaryPlane()->setTransformation(DrmPlane::Transformation::Rotate0);
|
|
|
|
pending.crtc->primaryPlane()->set(QPoint(0, 0), sourceSize(), QPoint(0, 0), mode.size);
|
|
|
|
|
|
|
|
m_formats = pending.crtc->primaryPlane()->formats();
|
|
|
|
}
|
|
|
|
|
|
|
|
void DrmPipeline::applyPendingChanges()
|
|
|
|
{
|
|
|
|
if (!pending.crtc) {
|
|
|
|
pending.active = false;
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
m_next = pending;
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
2021-09-28 08:29:56 +00:00
|
|
|
bool DrmPipeline::applyPendingChangesLegacy()
|
2021-05-25 22:05:17 +00:00
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
if (!pending.active && pending.crtc) {
|
|
|
|
drmModeSetCursor(gpu()->fd(), pending.crtc->id(), 0, 0, 0);
|
|
|
|
}
|
|
|
|
if (pending.active) {
|
|
|
|
Q_ASSERT(pending.crtc);
|
|
|
|
if (auto vrr = pending.crtc->getProp(DrmCrtc::PropertyIndex::VrrEnabled); !vrr->setPropertyLegacy(pending.syncMode == RenderLoopPrivate::SyncMode::Adaptive)) {
|
|
|
|
qCWarning(KWIN_DRM) << "Setting vrr failed!" << strerror(errno);
|
2021-05-25 22:05:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
if (const auto &rgbRange = m_connector->getProp(DrmConnector::PropertyIndex::Broadcast_RGB)) {
|
|
|
|
rgbRange->setEnumLegacy(pending.rgbRange);
|
|
|
|
}
|
|
|
|
if (needsModeset() &&!legacyModeset()) {
|
2021-05-25 22:05:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
m_connector->getProp(DrmConnector::PropertyIndex::Dpms)->setCurrent(DRM_MODE_DPMS_ON);
|
|
|
|
if (pending.gamma && drmModeCrtcSetGamma(gpu()->fd(), pending.crtc->id(), pending.gamma->size,
|
|
|
|
const_cast<uint16_t*>(pending.gamma->lut.red()),
|
|
|
|
const_cast<uint16_t*>(pending.gamma->lut.blue()),
|
|
|
|
const_cast<uint16_t*>(pending.gamma->lut.green())) != 0) {
|
|
|
|
qCWarning(KWIN_DRM) << "Setting gamma failed!" << strerror(errno);
|
2021-05-25 22:05:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-09-28 08:29:56 +00:00
|
|
|
pending.crtc->setLegacyCursor();
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
if (!m_connector->getProp(DrmConnector::PropertyIndex::Dpms)->setPropertyLegacy(pending.active ? DRM_MODE_DPMS_ON : DRM_MODE_DPMS_OFF)) {
|
|
|
|
qCWarning(KWIN_DRM) << "Setting legacy dpms failed!" << strerror(errno);
|
2021-05-25 22:05:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-09-28 08:29:56 +00:00
|
|
|
bool DrmPipeline::legacyModeset()
|
2021-05-25 22:05:17 +00:00
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
auto mode = m_connector->modes()[pending.modeIndex];
|
|
|
|
uint32_t connId = m_connector->id();
|
|
|
|
if (!checkTestBuffer() || drmModeSetCrtc(gpu()->fd(), pending.crtc->id(), m_primaryBuffer->bufferId(), 0, 0, &connId, 1, &mode.mode) != 0) {
|
|
|
|
qCWarning(KWIN_DRM) << "Modeset failed!" << strerror(errno);
|
|
|
|
pending = m_next;
|
|
|
|
m_primaryBuffer = m_oldTestBuffer;
|
2021-05-25 22:05:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
m_oldTestBuffer = nullptr;
|
|
|
|
// make sure the buffer gets kept alive, or the modeset gets reverted by the kernel
|
|
|
|
if (pending.crtc->current()) {
|
|
|
|
pending.crtc->setNext(m_primaryBuffer);
|
2021-08-31 14:59:06 +00:00
|
|
|
} else {
|
2021-09-28 08:29:56 +00:00
|
|
|
pending.crtc->setCurrent(m_primaryBuffer);
|
2021-08-31 14:59:06 +00:00
|
|
|
}
|
2021-09-28 08:29:56 +00:00
|
|
|
return true;
|
2021-08-31 14:59:06 +00:00
|
|
|
}
|
|
|
|
|
2021-05-25 22:05:17 +00:00
|
|
|
QSize DrmPipeline::sourceSize() const
|
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
auto mode = m_connector->modes()[pending.modeIndex];
|
|
|
|
if (pending.transformation & (DrmPlane::Transformation::Rotate90 | DrmPlane::Transformation::Rotate270)) {
|
2021-05-25 22:05:17 +00:00
|
|
|
return mode.size.transposed();
|
|
|
|
}
|
|
|
|
return mode.size;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DrmPipeline::isCursorVisible() const
|
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
return pending.crtc && pending.crtc->isCursorVisible(QRect(QPoint(0, 0), m_connector->currentMode().size));
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QPoint DrmPipeline::cursorPos() const
|
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
return pending.crtc->cursorPos();
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
DrmConnector *DrmPipeline::connector() const
|
|
|
|
{
|
|
|
|
return m_connector;
|
|
|
|
}
|
|
|
|
|
2021-09-28 08:29:56 +00:00
|
|
|
DrmGpu *DrmPipeline::gpu() const
|
2021-05-25 22:05:17 +00:00
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
return m_connector->gpu();
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
2021-10-08 08:52:01 +00:00
|
|
|
void DrmPipeline::pageFlipped(std::chrono::nanoseconds timestamp)
|
2021-05-25 22:05:17 +00:00
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
m_current.crtc->flipBuffer();
|
|
|
|
if (m_current.crtc->primaryPlane()) {
|
|
|
|
m_current.crtc->primaryPlane()->flipBuffer();
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
2021-10-08 08:52:01 +00:00
|
|
|
m_pageflipPending = false;
|
|
|
|
if (m_output) {
|
|
|
|
m_output->pageFlipped(timestamp);
|
|
|
|
}
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
2021-08-07 22:03:09 +00:00
|
|
|
void DrmPipeline::setOutput(DrmOutput *output)
|
2021-05-25 22:05:17 +00:00
|
|
|
{
|
2021-08-07 22:03:09 +00:00
|
|
|
m_output = output;
|
|
|
|
}
|
|
|
|
|
|
|
|
DrmOutput *DrmPipeline::output() const
|
|
|
|
{
|
|
|
|
return m_output;
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
|
2021-06-03 13:50:37 +00:00
|
|
|
void DrmPipeline::updateProperties()
|
|
|
|
{
|
2021-09-28 08:29:56 +00:00
|
|
|
m_connector->updateProperties();
|
|
|
|
if (pending.crtc) {
|
|
|
|
pending.crtc->updateProperties();
|
|
|
|
if (pending.crtc->primaryPlane()) {
|
|
|
|
pending.crtc->primaryPlane()->updateProperties();
|
|
|
|
}
|
2021-06-03 13:50:37 +00:00
|
|
|
}
|
2021-06-07 09:28:17 +00:00
|
|
|
// with legacy we don't know what happened to the cursor after VT switch
|
|
|
|
// so make sure it gets set again
|
2021-09-28 08:29:56 +00:00
|
|
|
pending.crtc->setLegacyCursor();
|
2021-06-03 13:50:37 +00:00
|
|
|
}
|
|
|
|
|
2021-07-31 13:32:50 +00:00
|
|
|
bool DrmPipeline::isFormatSupported(uint32_t drmFormat) const
|
|
|
|
{
|
2021-09-16 10:48:38 +00:00
|
|
|
return m_formats.contains(drmFormat);
|
2021-07-31 13:32:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
QVector<uint64_t> DrmPipeline::supportedModifiers(uint32_t drmFormat) const
|
|
|
|
{
|
2021-09-16 10:48:38 +00:00
|
|
|
return m_formats[drmFormat];
|
2021-07-31 13:32:50 +00:00
|
|
|
}
|
|
|
|
|
2021-09-28 08:29:56 +00:00
|
|
|
bool DrmPipeline::needsModeset() const
|
|
|
|
{
|
|
|
|
return pending.crtc != m_current.crtc
|
|
|
|
|| pending.active != m_current.active
|
|
|
|
|| pending.modeIndex != m_current.modeIndex
|
|
|
|
|| pending.rgbRange != m_current.rgbRange
|
2021-10-08 08:52:01 +00:00
|
|
|
|| pending.transformation != m_current.transformation
|
|
|
|
|| m_modesetPresentPending;
|
2021-09-28 08:29:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool DrmPipeline::activePending() const
|
|
|
|
{
|
|
|
|
return pending.crtc && pending.active;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DrmPipeline::revertPendingChanges()
|
|
|
|
{
|
|
|
|
pending = m_next;
|
|
|
|
}
|
|
|
|
|
2021-10-08 08:52:01 +00:00
|
|
|
bool DrmPipeline::pageflipPending() const
|
|
|
|
{
|
|
|
|
return m_pageflipPending;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DrmPipeline::modesetPresentPending() const
|
|
|
|
{
|
|
|
|
return m_modesetPresentPending;
|
|
|
|
}
|
|
|
|
|
|
|
|
DrmCrtc *DrmPipeline::currentCrtc() const
|
|
|
|
{
|
|
|
|
return m_current.crtc;
|
|
|
|
}
|
|
|
|
|
2021-09-28 08:29:56 +00:00
|
|
|
DrmGammaRamp::DrmGammaRamp(DrmGpu *gpu, const GammaRamp &lut)
|
|
|
|
: lut(lut)
|
|
|
|
, size(lut.size())
|
|
|
|
{
|
|
|
|
if (gpu->atomicModeSetting()) {
|
|
|
|
atomicLut = new drm_color_lut[size];
|
|
|
|
for (uint32_t i = 0; i < size; i++) {
|
|
|
|
atomicLut[i].red = lut.red()[i];
|
|
|
|
atomicLut[i].green = lut.green()[i];
|
|
|
|
atomicLut[i].blue = lut.blue()[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DrmGammaRamp::~DrmGammaRamp()
|
|
|
|
{
|
|
|
|
delete[] atomicLut;
|
|
|
|
}
|
|
|
|
|
2021-05-25 22:05:17 +00:00
|
|
|
static void printProps(DrmObject *object)
|
|
|
|
{
|
|
|
|
auto list = object->properties();
|
|
|
|
for (const auto &prop : list) {
|
|
|
|
if (prop) {
|
2021-09-16 10:37:44 +00:00
|
|
|
uint64_t current = prop->name().startsWith("SRC_") ? prop->current() >> 16 : prop->current();
|
2021-05-25 22:05:17 +00:00
|
|
|
if (prop->isImmutable() || !prop->needsCommit()) {
|
2021-09-16 10:37:44 +00:00
|
|
|
qCWarning(KWIN_DRM).nospace() << "\t" << prop->name() << ": " << current;
|
2021-05-25 22:05:17 +00:00
|
|
|
} else {
|
2021-09-16 10:37:44 +00:00
|
|
|
uint64_t pending = prop->name().startsWith("SRC_") ? prop->pending() >> 16 : prop->pending();
|
|
|
|
qCWarning(KWIN_DRM).nospace() << "\t" << prop->name() << ": " << current << "->" << pending;
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DrmPipeline::printDebugInfo() const
|
|
|
|
{
|
|
|
|
if (m_lastFlags == 0) {
|
|
|
|
qCWarning(KWIN_DRM) << "Flags: none";
|
|
|
|
} else {
|
|
|
|
qCWarning(KWIN_DRM) << "Flags:";
|
|
|
|
if (m_lastFlags & DRM_MODE_PAGE_FLIP_EVENT) {
|
|
|
|
qCWarning(KWIN_DRM) << "\t DRM_MODE_PAGE_FLIP_EVENT";
|
|
|
|
}
|
|
|
|
if (m_lastFlags & DRM_MODE_ATOMIC_ALLOW_MODESET) {
|
|
|
|
qCWarning(KWIN_DRM) << "\t DRM_MODE_ATOMIC_ALLOW_MODESET";
|
|
|
|
}
|
|
|
|
if (m_lastFlags & DRM_MODE_PAGE_FLIP_ASYNC) {
|
|
|
|
qCWarning(KWIN_DRM) << "\t DRM_MODE_PAGE_FLIP_ASYNC";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
qCWarning(KWIN_DRM) << "Drm objects:";
|
|
|
|
qCWarning(KWIN_DRM) << "connector" << m_connector->id();
|
|
|
|
printProps(m_connector);
|
2021-09-28 08:29:56 +00:00
|
|
|
if (pending.crtc) {
|
|
|
|
qCWarning(KWIN_DRM) << "crtc" << pending.crtc->id();
|
|
|
|
printProps(pending.crtc);
|
|
|
|
if (pending.crtc->primaryPlane()) {
|
|
|
|
qCWarning(KWIN_DRM) << "primary plane" << pending.crtc->primaryPlane()->id();
|
|
|
|
printProps(pending.crtc->primaryPlane());
|
|
|
|
}
|
2021-05-25 22:05:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|