2020-08-02 22:22:19 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
2015-10-02 08:44:29 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-FileCopyrightText: 2015 Martin Gräßlin <mgraesslin@kde.org>
|
2015-10-02 08:44:29 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
2015-10-02 08:44:29 +00:00
|
|
|
#include "virtual_backend.h"
|
2018-03-22 13:42:06 +00:00
|
|
|
#include "virtual_output.h"
|
2015-10-02 08:44:29 +00:00
|
|
|
#include "scene_qpainter_virtual_backend.h"
|
|
|
|
#include "wayland_server.h"
|
2015-10-08 14:12:09 +00:00
|
|
|
#include "egl_gbm_backend.h"
|
2015-10-08 13:53:03 +00:00
|
|
|
// Qt
|
|
|
|
#include <QTemporaryDir>
|
2015-10-02 08:44:29 +00:00
|
|
|
// KWayland
|
2020-04-29 15:18:41 +00:00
|
|
|
#include <KWaylandServer/seat_interface.h>
|
2016-08-19 13:40:39 +00:00
|
|
|
// system
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
2017-09-08 20:30:18 +00:00
|
|
|
#include <config-kwin.h>
|
2015-10-02 08:44:29 +00:00
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
|
|
|
VirtualBackend::VirtualBackend(QObject *parent)
|
2016-04-07 07:18:10 +00:00
|
|
|
: Platform(parent)
|
2015-10-02 08:44:29 +00:00
|
|
|
{
|
2015-10-08 13:53:03 +00:00
|
|
|
if (qEnvironmentVariableIsSet("KWIN_WAYLAND_VIRTUAL_SCREENSHOTS")) {
|
|
|
|
m_screenshotDir.reset(new QTemporaryDir);
|
|
|
|
if (!m_screenshotDir->isValid()) {
|
|
|
|
m_screenshotDir.reset();
|
|
|
|
}
|
|
|
|
if (!m_screenshotDir.isNull()) {
|
|
|
|
qDebug() << "Screenshots saved to: " << m_screenshotDir->path();
|
|
|
|
}
|
|
|
|
}
|
2021-01-14 08:21:59 +00:00
|
|
|
|
|
|
|
supportsOutputChanges();
|
2015-10-02 08:44:29 +00:00
|
|
|
setSupportsPointerWarping(true);
|
[colorcorrection] Night Color - blue light filter at nighttime
With Wayland KWin needs to provide certain services, which were provided
before that by the Xserver. One of these is gamma correction, which includes
the - by many people beloved - functionality to reduce the blue light at
nighttime. This patch provides the KWin part of that. It is self contained,
but in the end will work in tandem with a lib in Plasma Workspace and a KCM
in Plasma Desktop, which can be used to configure Night Color.
* Three modi:
** Automatic: The location and sun timings are determined automatically
(location data updates will be provided by the workspace)
** Location: The sun timings are determined by fixed location data
** Timings: The sun timings are set manually by the user
* Color temperature value changes are smoothly applied:
** Configuration changes, which lead to other current values are changed
in a quick way over a few seconds
** Changes on sunrise and sunset are applied slowly over the course of few
minutes till several hours depending on the configuration
* The current color value is set immediately at startup or after suspend
phases and VT switches. There is no flickering.
* All configuration is done via a DBus interface, changed values are tested
on correctness and applied atomically
* Self contained mechanism, speaks directly to the hardware by setting the
gamma ramps on the CRTC
* Currently working on DRM backend, extensible to other platform backends in
the future
* The code is written in a way to make the classes later easily extendable to
also provide normal color correction, as it's currently done by KGamma on X
Test Plan:
Manually with the workspace parts and added integration tests in KWin using
the virtual backend.
BUG:371494
Reviewers: #kwin, graesslin
Subscribers: kwin, plasma-devel, #kwin
Tags: #kwin
Differential Revision: https://phabricator.kde.org/D5928
2017-12-11 09:43:12 +00:00
|
|
|
setSupportsGammaControl(true);
|
2020-11-09 12:19:15 +00:00
|
|
|
setPerScreenRenderingEnabled(true);
|
2015-10-02 08:44:29 +00:00
|
|
|
}
|
|
|
|
|
2016-08-19 13:40:39 +00:00
|
|
|
VirtualBackend::~VirtualBackend()
|
|
|
|
{
|
2020-12-02 12:12:56 +00:00
|
|
|
if (sceneEglDisplay() != EGL_NO_DISPLAY) {
|
|
|
|
eglTerminate(sceneEglDisplay());
|
|
|
|
}
|
2016-08-19 13:40:39 +00:00
|
|
|
}
|
2015-10-02 08:44:29 +00:00
|
|
|
|
|
|
|
void VirtualBackend::init()
|
|
|
|
{
|
2018-03-19 11:05:57 +00:00
|
|
|
/*
|
|
|
|
* Some tests currently expect one output present at start,
|
|
|
|
* others set them explicitly.
|
|
|
|
*
|
|
|
|
* TODO: rewrite all tests to explicitly set the outputs.
|
|
|
|
*/
|
2021-01-18 16:49:29 +00:00
|
|
|
if (m_outputs.isEmpty()) {
|
2018-03-22 13:42:06 +00:00
|
|
|
VirtualOutput *dummyOutput = new VirtualOutput(this);
|
2019-08-26 21:16:53 +00:00
|
|
|
dummyOutput->init(QPoint(0, 0), initialWindowSize());
|
2018-03-29 16:38:27 +00:00
|
|
|
m_outputs << dummyOutput ;
|
2021-01-18 16:49:29 +00:00
|
|
|
m_outputsEnabled << dummyOutput;
|
2020-12-03 20:03:02 +00:00
|
|
|
emit outputAdded(dummyOutput);
|
2020-12-09 14:01:29 +00:00
|
|
|
emit outputEnabled(dummyOutput);
|
2018-03-19 11:05:57 +00:00
|
|
|
}
|
|
|
|
|
2020-10-28 07:03:09 +00:00
|
|
|
setSoftwareCursorForced(true);
|
2015-10-02 08:44:29 +00:00
|
|
|
setReady(true);
|
|
|
|
waylandServer()->seat()->setHasPointer(true);
|
|
|
|
waylandServer()->seat()->setHasKeyboard(true);
|
|
|
|
waylandServer()->seat()->setHasTouch(true);
|
2018-03-19 11:05:57 +00:00
|
|
|
|
2015-10-02 08:44:29 +00:00
|
|
|
emit screensQueried();
|
|
|
|
}
|
|
|
|
|
2015-10-08 13:53:03 +00:00
|
|
|
QString VirtualBackend::screenshotDirPath() const
|
|
|
|
{
|
|
|
|
if (m_screenshotDir.isNull()) {
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
return m_screenshotDir->path();
|
|
|
|
}
|
|
|
|
|
2015-10-02 08:44:29 +00:00
|
|
|
QPainterBackend *VirtualBackend::createQPainterBackend()
|
|
|
|
{
|
|
|
|
return new VirtualQPainterBackend(this);
|
|
|
|
}
|
|
|
|
|
2015-10-08 14:12:09 +00:00
|
|
|
OpenGLBackend *VirtualBackend::createOpenGLBackend()
|
|
|
|
{
|
|
|
|
return new EglGbmBackend(this);
|
|
|
|
}
|
|
|
|
|
2018-03-29 16:38:27 +00:00
|
|
|
Outputs VirtualBackend::outputs() const
|
|
|
|
{
|
|
|
|
return m_outputs;
|
|
|
|
}
|
|
|
|
|
|
|
|
Outputs VirtualBackend::enabledOutputs() const
|
|
|
|
{
|
2021-01-14 08:21:59 +00:00
|
|
|
return m_outputsEnabled;
|
2018-03-29 16:38:27 +00:00
|
|
|
}
|
|
|
|
|
2019-01-08 15:51:33 +00:00
|
|
|
void VirtualBackend::setVirtualOutputs(int count, QVector<QRect> geometries, QVector<int> scales)
|
2018-03-19 11:05:57 +00:00
|
|
|
{
|
|
|
|
Q_ASSERT(geometries.size() == 0 || geometries.size() == count);
|
2019-01-08 15:51:33 +00:00
|
|
|
Q_ASSERT(scales.size() == 0 || scales.size() == count);
|
2018-03-19 11:05:57 +00:00
|
|
|
|
2021-01-14 08:21:59 +00:00
|
|
|
while (!m_outputsEnabled.isEmpty()) {
|
|
|
|
VirtualOutput *output = m_outputsEnabled.takeLast();
|
|
|
|
emit outputDisabled(output);
|
|
|
|
}
|
|
|
|
|
2020-12-02 18:53:50 +00:00
|
|
|
while (!m_outputs.isEmpty()) {
|
|
|
|
VirtualOutput *output = m_outputs.takeLast();
|
|
|
|
emit outputRemoved(output);
|
|
|
|
delete output;
|
|
|
|
}
|
2018-03-19 11:05:57 +00:00
|
|
|
|
|
|
|
int sumWidth = 0;
|
|
|
|
for (int i = 0; i < count; i++) {
|
2018-03-22 13:42:06 +00:00
|
|
|
VirtualOutput *vo = new VirtualOutput(this);
|
2018-03-19 11:05:57 +00:00
|
|
|
if (geometries.size()) {
|
2019-08-26 21:16:53 +00:00
|
|
|
const QRect geo = geometries.at(i);
|
|
|
|
vo->init(geo.topLeft(), geo.size());
|
|
|
|
} else {
|
|
|
|
vo->init(QPoint(sumWidth, 0), initialWindowSize());
|
2018-03-29 16:38:27 +00:00
|
|
|
sumWidth += initialWindowSize().width();
|
2018-03-19 11:05:57 +00:00
|
|
|
}
|
2019-08-26 21:16:53 +00:00
|
|
|
if (scales.size()) {
|
|
|
|
vo->setScale(scales.at(i));
|
|
|
|
}
|
2020-12-02 18:53:50 +00:00
|
|
|
m_outputs.append(vo);
|
2021-01-14 08:21:59 +00:00
|
|
|
m_outputsEnabled.append(vo);
|
2020-12-02 18:53:50 +00:00
|
|
|
emit outputAdded(vo);
|
2020-12-09 14:01:29 +00:00
|
|
|
emit outputEnabled(vo);
|
2018-03-19 11:05:57 +00:00
|
|
|
}
|
|
|
|
|
2020-12-20 13:15:57 +00:00
|
|
|
emit screensQueried();
|
2018-03-19 11:05:57 +00:00
|
|
|
}
|
|
|
|
|
2021-01-14 08:21:59 +00:00
|
|
|
void VirtualBackend::enableOutput(VirtualOutput *output, bool enable)
|
|
|
|
{
|
|
|
|
if (enable) {
|
|
|
|
Q_ASSERT(!m_outputsEnabled.contains(output));
|
|
|
|
m_outputsEnabled << output;
|
|
|
|
emit outputEnabled(output);
|
|
|
|
} else {
|
|
|
|
Q_ASSERT(m_outputsEnabled.contains(output));
|
|
|
|
m_outputsEnabled.removeOne(output);
|
|
|
|
emit outputDisabled(output);
|
|
|
|
}
|
|
|
|
|
|
|
|
emit screensQueried();
|
|
|
|
}
|
|
|
|
|
|
|
|
void VirtualBackend::removeOutput(AbstractOutput *output)
|
|
|
|
{
|
2021-01-14 09:37:15 +00:00
|
|
|
VirtualOutput *virtualOutput = static_cast<VirtualOutput *>(output);
|
2021-01-14 08:21:59 +00:00
|
|
|
if (m_outputsEnabled.removeOne(virtualOutput)) {
|
|
|
|
emit outputDisabled(virtualOutput);
|
|
|
|
}
|
|
|
|
|
|
|
|
emit outputRemoved(virtualOutput);
|
|
|
|
m_outputsEnabled.removeOne(virtualOutput);
|
|
|
|
|
|
|
|
delete virtualOutput;
|
|
|
|
|
|
|
|
emit screensQueried();
|
|
|
|
}
|
|
|
|
|
2015-10-02 08:44:29 +00:00
|
|
|
}
|