2020-08-02 22:22:19 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
2015-05-06 15:47:07 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-FileCopyrightText: 2015 Martin Gräßlin <mgraesslin@kde.org>
|
2015-05-06 15:47:07 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
*/
|
2015-05-06 15:47:07 +00:00
|
|
|
#include "egl_hwcomposer_backend.h"
|
|
|
|
#include "hwcomposer_backend.h"
|
|
|
|
#include "logging.h"
|
|
|
|
#include "screens_hwcomposer.h"
|
2015-05-11 12:43:51 +00:00
|
|
|
#include "composite.h"
|
2016-04-07 06:28:35 +00:00
|
|
|
#include "main.h"
|
2015-05-08 19:38:12 +00:00
|
|
|
#include "wayland_server.h"
|
|
|
|
// KWayland
|
2020-04-29 15:18:41 +00:00
|
|
|
#include <KWaylandServer/output_interface.h>
|
[platforms/hwcomposer] Add scaling support
Summary:
Despite plasma frameworks doing it's own scaling with fonts, it's been
requested to use kwin/wayland scaling.
Like DRM, when kscreen is not used, scale value is loaded from a config
file.
Config format is
[HWComposerOutputs][0]
Scale=N
The 0 is to map similarly to DRM and support multi-screen, but with a
screen index
rather than a UUID based on EDID.
Because we don't support multi screen this is always 0 for now.
Test Plan: Ran with the config value unset and with the config value at
Scale=3.
Reviewers: #kwin, romangg
Reviewed By: #kwin, romangg
Subscribers: bshah, romangg, nicolasfella, zzag, kwin
Tags: #kwin
Differential Revision: https://phabricator.kde.org/D18810
2019-02-22 00:44:12 +00:00
|
|
|
// KDE
|
|
|
|
#include <KConfigGroup>
|
2016-02-16 09:40:20 +00:00
|
|
|
// Qt
|
|
|
|
#include <QKeyEvent>
|
[platforms/hwcomposer] Reset old brightness when turning screen back on
Summary:
Now powerdevil can adjust brightness using leds subsystem, however, kwin
as well sets brightness to 0 for turning off screen and 0xff when
turning screen back on. This resets the brightness set by the powerdevil
to 100%.
As a solution now kwin listens to brightnessChanged dbus signal of
brightnesscontrol and book-keeps the changed brightness, if screen is
turned off it sets brightness to 0 and when turning screen on, it resets
to old brightness.
If powermanagement service doesn't appear on dbus by default it restores
100% brightness.
Test Plan:
Appearantly this still doesn't work as-it-is on phone, because powerdevil
doesn't emit brightnessChanged dbus signal because of bug in the driver of
backlight control, driver doesn't seem to trigger uevents for changes in
backlight. But with hack in powerdevil to emit brightnessChanged when setting
brightness, this works
Reviewers: broulik, #plasma_on_wayland, graesslin
Reviewed By: #plasma_on_wayland, graesslin
Subscribers: plasma-devel, kwin
Tags: #plasma_on_wayland, #kwin
Differential Revision: https://phabricator.kde.org/D2468
2016-08-17 09:09:58 +00:00
|
|
|
#include <QDBusConnection>
|
2015-05-08 19:38:12 +00:00
|
|
|
// hybris/android
|
2015-05-06 15:47:07 +00:00
|
|
|
#include <hardware/hardware.h>
|
2015-11-16 08:01:48 +00:00
|
|
|
#include <hardware/lights.h>
|
2015-05-11 12:06:38 +00:00
|
|
|
// linux
|
|
|
|
#include <linux/input.h>
|
2015-05-06 15:47:07 +00:00
|
|
|
|
|
|
|
// based on test_hwcomposer.c from libhybris project (Apache 2 licensed)
|
|
|
|
|
2020-04-29 15:18:41 +00:00
|
|
|
using namespace KWaylandServer;
|
2019-02-20 13:19:51 +00:00
|
|
|
|
2015-05-06 15:47:07 +00:00
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
2016-02-16 09:40:20 +00:00
|
|
|
BacklightInputEventFilter::BacklightInputEventFilter(HwcomposerBackend *backend)
|
|
|
|
: InputEventFilter()
|
|
|
|
, m_backend(backend)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
BacklightInputEventFilter::~BacklightInputEventFilter() = default;
|
|
|
|
|
|
|
|
bool BacklightInputEventFilter::pointerEvent(QMouseEvent *event, quint32 nativeButton)
|
|
|
|
{
|
|
|
|
Q_UNUSED(event)
|
|
|
|
Q_UNUSED(nativeButton)
|
|
|
|
if (!m_backend->isBacklightOff()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
toggleBacklight();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool BacklightInputEventFilter::wheelEvent(QWheelEvent *event)
|
|
|
|
{
|
|
|
|
Q_UNUSED(event)
|
|
|
|
if (!m_backend->isBacklightOff()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
toggleBacklight();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool BacklightInputEventFilter::keyEvent(QKeyEvent *event)
|
|
|
|
{
|
|
|
|
if (event->key() == Qt::Key_PowerOff && event->type() == QEvent::KeyRelease) {
|
|
|
|
toggleBacklight();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return m_backend->isBacklightOff();
|
|
|
|
}
|
|
|
|
|
2019-08-17 10:54:09 +00:00
|
|
|
bool BacklightInputEventFilter::touchDown(qint32 id, const QPointF &pos, quint32 time)
|
2016-02-16 09:40:20 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(pos)
|
|
|
|
Q_UNUSED(time)
|
|
|
|
if (!m_backend->isBacklightOff()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (m_touchPoints.isEmpty()) {
|
|
|
|
if (!m_doubleTapTimer.isValid()) {
|
|
|
|
// this is the first tap
|
|
|
|
m_doubleTapTimer.start();
|
|
|
|
} else {
|
|
|
|
if (m_doubleTapTimer.elapsed() < qApp->doubleClickInterval()) {
|
|
|
|
m_secondTap = true;
|
|
|
|
} else {
|
|
|
|
// took too long. Let's consider it a new click
|
|
|
|
m_doubleTapTimer.restart();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// not a double tap
|
|
|
|
m_doubleTapTimer.invalidate();
|
|
|
|
m_secondTap = false;
|
|
|
|
}
|
|
|
|
m_touchPoints << id;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-08-17 10:54:09 +00:00
|
|
|
bool BacklightInputEventFilter::touchUp(qint32 id, quint32 time)
|
2016-02-16 09:40:20 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(time)
|
|
|
|
m_touchPoints.removeAll(id);
|
|
|
|
if (!m_backend->isBacklightOff()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (m_touchPoints.isEmpty() && m_doubleTapTimer.isValid() && m_secondTap) {
|
|
|
|
if (m_doubleTapTimer.elapsed() < qApp->doubleClickInterval()) {
|
|
|
|
toggleBacklight();
|
|
|
|
}
|
|
|
|
m_doubleTapTimer.invalidate();
|
|
|
|
m_secondTap = false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-08-17 10:54:09 +00:00
|
|
|
bool BacklightInputEventFilter::touchMotion(qint32 id, const QPointF &pos, quint32 time)
|
2016-02-16 09:40:20 +00:00
|
|
|
{
|
|
|
|
Q_UNUSED(id)
|
|
|
|
Q_UNUSED(pos)
|
|
|
|
Q_UNUSED(time)
|
|
|
|
return m_backend->isBacklightOff();
|
|
|
|
}
|
|
|
|
|
|
|
|
void BacklightInputEventFilter::toggleBacklight()
|
|
|
|
{
|
|
|
|
// queued to not modify the list of event filters while filtering
|
|
|
|
QMetaObject::invokeMethod(m_backend, "toggleBlankOutput", Qt::QueuedConnection);
|
|
|
|
}
|
|
|
|
|
2015-05-06 15:47:07 +00:00
|
|
|
HwcomposerBackend::HwcomposerBackend(QObject *parent)
|
2016-04-07 07:18:10 +00:00
|
|
|
: Platform(parent)
|
2015-05-06 15:47:07 +00:00
|
|
|
{
|
[platforms/hwcomposer] Reset old brightness when turning screen back on
Summary:
Now powerdevil can adjust brightness using leds subsystem, however, kwin
as well sets brightness to 0 for turning off screen and 0xff when
turning screen back on. This resets the brightness set by the powerdevil
to 100%.
As a solution now kwin listens to brightnessChanged dbus signal of
brightnesscontrol and book-keeps the changed brightness, if screen is
turned off it sets brightness to 0 and when turning screen on, it resets
to old brightness.
If powermanagement service doesn't appear on dbus by default it restores
100% brightness.
Test Plan:
Appearantly this still doesn't work as-it-is on phone, because powerdevil
doesn't emit brightnessChanged dbus signal because of bug in the driver of
backlight control, driver doesn't seem to trigger uevents for changes in
backlight. But with hack in powerdevil to emit brightnessChanged when setting
brightness, this works
Reviewers: broulik, #plasma_on_wayland, graesslin
Reviewed By: #plasma_on_wayland, graesslin
Subscribers: plasma-devel, kwin
Tags: #plasma_on_wayland, #kwin
Differential Revision: https://phabricator.kde.org/D2468
2016-08-17 09:09:58 +00:00
|
|
|
if (!QDBusConnection::sessionBus().connect(QStringLiteral("org.kde.Solid.PowerManagement"),
|
|
|
|
QStringLiteral("/org/kde/Solid/PowerManagement/Actions/BrightnessControl"),
|
|
|
|
QStringLiteral("org.kde.Solid.PowerManagement.Actions.BrightnessControl"),
|
|
|
|
QStringLiteral("brightnessChanged"), this,
|
|
|
|
SLOT(screenBrightnessChanged(int)))) {
|
|
|
|
qCWarning(KWIN_HWCOMPOSER) << "Failed to connect to brightness control";
|
|
|
|
}
|
2015-05-06 15:47:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
HwcomposerBackend::~HwcomposerBackend()
|
|
|
|
{
|
2015-11-13 16:39:16 +00:00
|
|
|
if (!m_outputBlank) {
|
|
|
|
toggleBlankOutput();
|
|
|
|
}
|
2015-05-06 15:47:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void HwcomposerBackend::init()
|
|
|
|
{
|
|
|
|
hw_module_t *hwcModule = nullptr;
|
|
|
|
if (hw_get_module(HWC_HARDWARE_MODULE_ID, (const hw_module_t **)&hwcModule) != 0) {
|
|
|
|
qCWarning(KWIN_HWCOMPOSER) << "Failed to get hwcomposer module";
|
|
|
|
emit initFailed();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
hwc_composer_device_1_t *hwcDevice = nullptr;
|
|
|
|
if (hwc_open_1(hwcModule, &hwcDevice) != 0) {
|
|
|
|
qCWarning(KWIN_HWCOMPOSER) << "Failed to open hwcomposer device";
|
|
|
|
emit initFailed();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// unblank, setPowerMode?
|
2015-05-11 12:43:51 +00:00
|
|
|
m_device = hwcDevice;
|
2015-06-26 08:09:16 +00:00
|
|
|
|
2016-12-16 04:48:05 +00:00
|
|
|
m_hwcVersion = m_device->common.version;
|
|
|
|
if ((m_hwcVersion & 0xffff0000) == 0) {
|
|
|
|
// Assume header version is always 1
|
|
|
|
uint32_t header_version = 1;
|
|
|
|
// Legacy version encoding
|
|
|
|
m_hwcVersion = (m_hwcVersion << 16) | header_version;
|
|
|
|
}
|
2016-12-15 14:02:49 +00:00
|
|
|
|
2015-06-26 08:09:16 +00:00
|
|
|
// register callbacks
|
|
|
|
hwc_procs_t *procs = new hwc_procs_t;
|
|
|
|
procs->invalidate = [] (const struct hwc_procs* procs) {
|
|
|
|
Q_UNUSED(procs)
|
|
|
|
};
|
|
|
|
procs->vsync = [] (const struct hwc_procs* procs, int disp, int64_t timestamp) {
|
|
|
|
Q_UNUSED(procs)
|
|
|
|
if (disp != 0) {
|
|
|
|
return;
|
|
|
|
}
|
2016-04-07 06:28:35 +00:00
|
|
|
dynamic_cast<HwcomposerBackend*>(kwinApp()->platform())->wakeVSync();
|
2015-06-26 08:09:16 +00:00
|
|
|
};
|
|
|
|
procs->hotplug = [] (const struct hwc_procs* procs, int disp, int connected) {
|
|
|
|
Q_UNUSED(procs)
|
|
|
|
Q_UNUSED(disp)
|
|
|
|
Q_UNUSED(connected)
|
|
|
|
};
|
|
|
|
m_device->registerProcs(m_device, procs);
|
|
|
|
|
2019-02-20 13:19:51 +00:00
|
|
|
//move to HwcomposerOutput + signal
|
|
|
|
|
2015-11-16 08:01:48 +00:00
|
|
|
initLights();
|
2015-05-11 12:43:51 +00:00
|
|
|
toggleBlankOutput();
|
2016-02-16 09:40:20 +00:00
|
|
|
m_filter.reset(new BacklightInputEventFilter(this));
|
2017-01-02 19:13:30 +00:00
|
|
|
input()->prependInputEventFilter(m_filter.data());
|
2015-05-06 15:47:07 +00:00
|
|
|
|
|
|
|
// get display configuration
|
2019-02-20 13:19:51 +00:00
|
|
|
m_output.reset(new HwcomposerOutput(hwcDevice));
|
|
|
|
if (!m_output->isValid()) {
|
2015-05-06 15:47:07 +00:00
|
|
|
emit initFailed();
|
|
|
|
return;
|
|
|
|
}
|
2019-02-20 13:19:51 +00:00
|
|
|
|
|
|
|
if (m_output->refreshRate() != 0) {
|
|
|
|
m_vsyncInterval = 1000000/m_output->refreshRate();
|
2015-10-20 11:16:05 +00:00
|
|
|
}
|
2019-02-20 13:19:51 +00:00
|
|
|
|
2015-11-17 13:46:54 +00:00
|
|
|
if (m_lights) {
|
2020-04-29 15:18:41 +00:00
|
|
|
using namespace KWaylandServer;
|
2019-02-20 13:19:51 +00:00
|
|
|
|
|
|
|
auto updateDpms = [this] {
|
|
|
|
if (!m_output || !m_output->waylandOutput()) {
|
|
|
|
m_output->waylandOutput()->setDpmsMode(m_outputBlank ? OutputInterface::DpmsMode::Off : OutputInterface::DpmsMode::On);
|
|
|
|
}
|
2015-11-17 13:46:54 +00:00
|
|
|
};
|
|
|
|
connect(this, &HwcomposerBackend::outputBlankChanged, this, updateDpms);
|
2019-02-20 13:19:51 +00:00
|
|
|
|
|
|
|
connect(m_output.data(), &HwcomposerOutput::dpmsModeRequested, this,
|
2020-04-29 15:18:41 +00:00
|
|
|
[this] (KWaylandServer::OutputInterface::DpmsMode mode) {
|
2015-11-17 13:46:54 +00:00
|
|
|
if (mode == OutputInterface::DpmsMode::On) {
|
|
|
|
if (m_outputBlank) {
|
|
|
|
toggleBlankOutput();
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!m_outputBlank) {
|
|
|
|
toggleBlankOutput();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2015-05-06 15:47:07 +00:00
|
|
|
|
|
|
|
emit screensQueried();
|
|
|
|
setReady(true);
|
|
|
|
}
|
|
|
|
|
[platforms/hwcomposer] Add scaling support
Summary:
Despite plasma frameworks doing it's own scaling with fonts, it's been
requested to use kwin/wayland scaling.
Like DRM, when kscreen is not used, scale value is loaded from a config
file.
Config format is
[HWComposerOutputs][0]
Scale=N
The 0 is to map similarly to DRM and support multi-screen, but with a
screen index
rather than a UUID based on EDID.
Because we don't support multi screen this is always 0 for now.
Test Plan: Ran with the config value unset and with the config value at
Scale=3.
Reviewers: #kwin, romangg
Reviewed By: #kwin, romangg
Subscribers: bshah, romangg, nicolasfella, zzag, kwin
Tags: #kwin
Differential Revision: https://phabricator.kde.org/D18810
2019-02-22 00:44:12 +00:00
|
|
|
QSize HwcomposerBackend::size() const
|
2019-02-20 13:19:51 +00:00
|
|
|
{
|
|
|
|
if (m_output) {
|
|
|
|
return m_output->pixelSize();
|
|
|
|
}
|
|
|
|
return QSize();
|
|
|
|
}
|
|
|
|
|
[platforms/hwcomposer] Add scaling support
Summary:
Despite plasma frameworks doing it's own scaling with fonts, it's been
requested to use kwin/wayland scaling.
Like DRM, when kscreen is not used, scale value is loaded from a config
file.
Config format is
[HWComposerOutputs][0]
Scale=N
The 0 is to map similarly to DRM and support multi-screen, but with a
screen index
rather than a UUID based on EDID.
Because we don't support multi screen this is always 0 for now.
Test Plan: Ran with the config value unset and with the config value at
Scale=3.
Reviewers: #kwin, romangg
Reviewed By: #kwin, romangg
Subscribers: bshah, romangg, nicolasfella, zzag, kwin
Tags: #kwin
Differential Revision: https://phabricator.kde.org/D18810
2019-02-22 00:44:12 +00:00
|
|
|
QSize HwcomposerBackend::screenSize() const
|
|
|
|
{
|
|
|
|
if (m_output) {
|
|
|
|
return m_output->pixelSize() / m_output->scale();
|
|
|
|
}
|
|
|
|
return QSize();
|
|
|
|
}
|
|
|
|
|
|
|
|
int HwcomposerBackend::scale() const
|
|
|
|
{
|
|
|
|
if (m_output) {
|
|
|
|
return m_output->scale();
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2015-11-16 08:01:48 +00:00
|
|
|
void HwcomposerBackend::initLights()
|
|
|
|
{
|
|
|
|
hw_module_t *lightsModule = nullptr;
|
|
|
|
if (hw_get_module(LIGHTS_HARDWARE_MODULE_ID, (const hw_module_t **)&lightsModule) != 0) {
|
|
|
|
qCWarning(KWIN_HWCOMPOSER) << "Failed to get lights module";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
light_device_t *lightsDevice = nullptr;
|
|
|
|
if (lightsModule->methods->open(lightsModule, LIGHT_ID_BACKLIGHT, (hw_device_t **)&lightsDevice) != 0) {
|
|
|
|
qCWarning(KWIN_HWCOMPOSER) << "Failed to create lights device";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_lights = lightsDevice;
|
|
|
|
}
|
|
|
|
|
2015-05-11 12:43:51 +00:00
|
|
|
void HwcomposerBackend::toggleBlankOutput()
|
|
|
|
{
|
|
|
|
if (!m_device) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
m_outputBlank = !m_outputBlank;
|
2015-11-16 08:01:48 +00:00
|
|
|
toggleScreenBrightness();
|
2016-12-15 14:02:49 +00:00
|
|
|
|
|
|
|
#if defined(HWC_DEVICE_API_VERSION_1_4) || defined(HWC_DEVICE_API_VERSION_1_5)
|
|
|
|
if (m_hwcVersion > HWC_DEVICE_API_VERSION_1_3)
|
2016-12-16 09:06:43 +00:00
|
|
|
m_device->setPowerMode(m_device, 0, m_outputBlank ? HWC_POWER_MODE_OFF : HWC_POWER_MODE_NORMAL);
|
2016-12-15 14:02:49 +00:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
m_device->blank(m_device, 0, m_outputBlank ? 1 : 0);
|
|
|
|
|
|
|
|
// only disable Vsync, enable happens after next frame rendered
|
2015-10-19 08:13:04 +00:00
|
|
|
if (m_outputBlank) {
|
|
|
|
enableVSync(false);
|
|
|
|
}
|
2015-05-11 12:43:51 +00:00
|
|
|
// enable/disable compositor repainting when blanked
|
2015-10-20 07:39:03 +00:00
|
|
|
setOutputsEnabled(!m_outputBlank);
|
2015-05-11 12:43:51 +00:00
|
|
|
if (Compositor *compositor = Compositor::self()) {
|
2015-10-20 07:39:03 +00:00
|
|
|
if (!m_outputBlank) {
|
2015-05-11 12:43:51 +00:00
|
|
|
compositor->addRepaintFull();
|
|
|
|
}
|
|
|
|
}
|
2015-11-17 13:46:54 +00:00
|
|
|
emit outputBlankChanged();
|
2015-05-11 12:43:51 +00:00
|
|
|
}
|
|
|
|
|
2015-11-16 08:01:48 +00:00
|
|
|
void HwcomposerBackend::toggleScreenBrightness()
|
|
|
|
{
|
|
|
|
if (!m_lights) {
|
|
|
|
return;
|
|
|
|
}
|
[platforms/hwcomposer] Reset old brightness when turning screen back on
Summary:
Now powerdevil can adjust brightness using leds subsystem, however, kwin
as well sets brightness to 0 for turning off screen and 0xff when
turning screen back on. This resets the brightness set by the powerdevil
to 100%.
As a solution now kwin listens to brightnessChanged dbus signal of
brightnesscontrol and book-keeps the changed brightness, if screen is
turned off it sets brightness to 0 and when turning screen on, it resets
to old brightness.
If powermanagement service doesn't appear on dbus by default it restores
100% brightness.
Test Plan:
Appearantly this still doesn't work as-it-is on phone, because powerdevil
doesn't emit brightnessChanged dbus signal because of bug in the driver of
backlight control, driver doesn't seem to trigger uevents for changes in
backlight. But with hack in powerdevil to emit brightnessChanged when setting
brightness, this works
Reviewers: broulik, #plasma_on_wayland, graesslin
Reviewed By: #plasma_on_wayland, graesslin
Subscribers: plasma-devel, kwin
Tags: #plasma_on_wayland, #kwin
Differential Revision: https://phabricator.kde.org/D2468
2016-08-17 09:09:58 +00:00
|
|
|
const int brightness = m_outputBlank ? 0 : m_oldScreenBrightness;
|
2015-11-16 08:01:48 +00:00
|
|
|
struct light_state_t state;
|
|
|
|
state.flashMode = LIGHT_FLASH_NONE;
|
|
|
|
state.brightnessMode = BRIGHTNESS_MODE_USER;
|
|
|
|
|
|
|
|
state.color = (int)((0xffU << 24) | (brightness << 16) |
|
|
|
|
(brightness << 8) | brightness);
|
|
|
|
m_lights->set_light(m_lights, &state);
|
|
|
|
}
|
|
|
|
|
2015-10-19 08:13:04 +00:00
|
|
|
void HwcomposerBackend::enableVSync(bool enable)
|
|
|
|
{
|
|
|
|
if (m_hasVsync == enable) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const int result = m_device->eventControl(m_device, 0, HWC_EVENT_VSYNC, enable ? 1: 0);
|
|
|
|
m_hasVsync = enable && (result == 0);
|
|
|
|
}
|
|
|
|
|
2015-05-06 15:47:07 +00:00
|
|
|
HwcomposerWindow *HwcomposerBackend::createSurface()
|
|
|
|
{
|
|
|
|
return new HwcomposerWindow(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
Screens *HwcomposerBackend::createScreens(QObject *parent)
|
|
|
|
{
|
|
|
|
return new HwcomposerScreens(this, parent);
|
|
|
|
}
|
|
|
|
|
2019-02-20 13:19:51 +00:00
|
|
|
Outputs HwcomposerBackend::outputs() const
|
|
|
|
{
|
|
|
|
if (!m_output.isNull()) {
|
|
|
|
return QVector<HwcomposerOutput*>({m_output.data()});
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
Outputs HwcomposerBackend::enabledOutputs() const
|
|
|
|
{
|
|
|
|
return outputs();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-05-06 15:47:07 +00:00
|
|
|
OpenGLBackend *HwcomposerBackend::createOpenGLBackend()
|
|
|
|
{
|
|
|
|
return new EglHwcomposerBackend(this);
|
|
|
|
}
|
|
|
|
|
2015-10-20 11:16:05 +00:00
|
|
|
void HwcomposerBackend::waitVSync()
|
2015-06-26 08:09:16 +00:00
|
|
|
{
|
2015-11-16 07:56:28 +00:00
|
|
|
if (!m_hasVsync) {
|
|
|
|
return;
|
|
|
|
}
|
2015-10-20 11:16:05 +00:00
|
|
|
m_vsyncMutex.lock();
|
|
|
|
m_vsyncWaitCondition.wait(&m_vsyncMutex, m_vsyncInterval);
|
|
|
|
m_vsyncMutex.unlock();
|
2015-06-26 08:09:16 +00:00
|
|
|
}
|
|
|
|
|
2015-10-20 11:16:05 +00:00
|
|
|
void HwcomposerBackend::wakeVSync()
|
2015-06-26 08:09:16 +00:00
|
|
|
{
|
2015-10-20 11:16:05 +00:00
|
|
|
m_vsyncMutex.lock();
|
|
|
|
m_vsyncWaitCondition.wakeAll();
|
|
|
|
m_vsyncMutex.unlock();
|
2015-06-26 08:09:16 +00:00
|
|
|
}
|
|
|
|
|
2017-11-01 06:48:56 +00:00
|
|
|
static void initLayer(hwc_layer_1_t *layer, const hwc_rect_t &rect, int layerCompositionType)
|
2015-05-06 15:47:07 +00:00
|
|
|
{
|
|
|
|
memset(layer, 0, sizeof(hwc_layer_1_t));
|
2017-11-01 06:48:56 +00:00
|
|
|
layer->compositionType = layerCompositionType;
|
2015-05-06 15:47:07 +00:00
|
|
|
layer->hints = 0;
|
|
|
|
layer->flags = 0;
|
|
|
|
layer->handle = 0;
|
|
|
|
layer->transform = 0;
|
|
|
|
layer->blending = HWC_BLENDING_NONE;
|
2017-11-01 06:48:56 +00:00
|
|
|
#ifdef HWC_DEVICE_API_VERSION_1_3
|
|
|
|
layer->sourceCropf.top = 0.0f;
|
|
|
|
layer->sourceCropf.left = 0.0f;
|
|
|
|
layer->sourceCropf.bottom = (float) rect.bottom;
|
|
|
|
layer->sourceCropf.right = (float) rect.right;
|
|
|
|
#else
|
2015-05-06 15:47:07 +00:00
|
|
|
layer->sourceCrop = rect;
|
2017-11-01 06:48:56 +00:00
|
|
|
#endif
|
2015-05-06 15:47:07 +00:00
|
|
|
layer->displayFrame = rect;
|
|
|
|
layer->visibleRegionScreen.numRects = 1;
|
|
|
|
layer->visibleRegionScreen.rects = &layer->displayFrame;
|
|
|
|
layer->acquireFenceFd = -1;
|
|
|
|
layer->releaseFenceFd = -1;
|
2015-10-09 11:12:40 +00:00
|
|
|
layer->planeAlpha = 0xFF;
|
2017-11-02 04:38:52 +00:00
|
|
|
#ifdef HWC_DEVICE_API_VERSION_1_5
|
|
|
|
layer->surfaceDamage.numRects = 0;
|
|
|
|
#endif
|
2015-05-06 15:47:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
HwcomposerWindow::HwcomposerWindow(HwcomposerBackend *backend)
|
2016-07-19 06:16:54 +00:00
|
|
|
: HWComposerNativeWindow(backend->size().width(), backend->size().height(), HAL_PIXEL_FORMAT_RGBA_8888)
|
2015-05-06 15:47:07 +00:00
|
|
|
, m_backend(backend)
|
|
|
|
{
|
2015-10-20 09:39:42 +00:00
|
|
|
setBufferCount(3);
|
2015-10-09 11:12:40 +00:00
|
|
|
|
2015-05-06 15:47:07 +00:00
|
|
|
size_t size = sizeof(hwc_display_contents_1_t) + 2 * sizeof(hwc_layer_1_t);
|
|
|
|
hwc_display_contents_1_t *list = (hwc_display_contents_1_t*)malloc(size);
|
|
|
|
m_list = (hwc_display_contents_1_t**)malloc(HWC_NUM_DISPLAY_TYPES * sizeof(hwc_display_contents_1_t *));
|
|
|
|
for (int i = 0; i < HWC_NUM_DISPLAY_TYPES; ++i) {
|
2015-10-09 11:12:40 +00:00
|
|
|
m_list[i] = nullptr;
|
2015-05-06 15:47:07 +00:00
|
|
|
}
|
2015-10-09 11:12:40 +00:00
|
|
|
// Assign buffer only to the first item, otherwise you get tearing
|
|
|
|
// if passed the same to multiple places
|
|
|
|
// see https://github.com/mer-hybris/qt5-qpa-hwcomposer-plugin/commit/f1d802151e8a4f5d10d60eb8de8e07552b93a34a
|
|
|
|
m_list[0] = list;
|
2015-05-06 15:47:07 +00:00
|
|
|
const hwc_rect_t rect = {
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
m_backend->size().width(),
|
|
|
|
m_backend->size().height()
|
|
|
|
};
|
2017-11-01 06:48:56 +00:00
|
|
|
initLayer(&list->hwLayers[0], rect, HWC_FRAMEBUFFER);
|
|
|
|
initLayer(&list->hwLayers[1], rect, HWC_FRAMEBUFFER_TARGET);
|
2015-05-06 15:47:07 +00:00
|
|
|
|
|
|
|
list->retireFenceFd = -1;
|
|
|
|
list->flags = HWC_GEOMETRY_CHANGED;
|
|
|
|
list->numHwLayers = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
HwcomposerWindow::~HwcomposerWindow()
|
|
|
|
{
|
|
|
|
// TODO: cleanup
|
|
|
|
}
|
|
|
|
|
2015-10-09 11:12:40 +00:00
|
|
|
void HwcomposerWindow::present(HWComposerNativeWindowBuffer *buffer)
|
2015-05-06 15:47:07 +00:00
|
|
|
{
|
2015-10-20 11:16:05 +00:00
|
|
|
m_backend->waitVSync();
|
2015-10-09 11:12:40 +00:00
|
|
|
hwc_composer_device_1_t *device = m_backend->device();
|
2015-05-06 15:47:07 +00:00
|
|
|
|
2015-10-09 11:12:40 +00:00
|
|
|
auto fblayer = &m_list[0]->hwLayers[1];
|
|
|
|
fblayer->handle = buffer->handle;
|
|
|
|
fblayer->acquireFenceFd = getFenceBufferFd(buffer);
|
|
|
|
fblayer->releaseFenceFd = -1;
|
2015-05-06 15:47:07 +00:00
|
|
|
|
2015-10-09 11:12:40 +00:00
|
|
|
int err = device->prepare(device, 1, m_list);
|
2019-08-31 14:28:37 +00:00
|
|
|
Q_ASSERT(err == 0);
|
2015-05-06 15:47:07 +00:00
|
|
|
|
2015-10-09 11:12:40 +00:00
|
|
|
err = device->set(device, 1, m_list);
|
2019-08-31 14:28:37 +00:00
|
|
|
Q_ASSERT(err == 0);
|
2015-10-19 08:13:04 +00:00
|
|
|
m_backend->enableVSync(true);
|
2015-10-09 11:12:40 +00:00
|
|
|
setFenceBufferFd(buffer, fblayer->releaseFenceFd);
|
2015-05-06 15:47:07 +00:00
|
|
|
|
2015-10-09 11:12:40 +00:00
|
|
|
if (m_list[0]->retireFenceFd != -1) {
|
|
|
|
close(m_list[0]->retireFenceFd);
|
|
|
|
m_list[0]->retireFenceFd = -1;
|
2015-05-06 15:47:07 +00:00
|
|
|
}
|
2015-06-26 08:09:16 +00:00
|
|
|
m_list[0]->flags = 0;
|
2015-05-06 15:47:07 +00:00
|
|
|
}
|
|
|
|
|
2019-02-20 13:19:51 +00:00
|
|
|
HwcomposerOutput::HwcomposerOutput(hwc_composer_device_1_t *device)
|
2019-06-13 09:36:07 +00:00
|
|
|
: AbstractWaylandOutput()
|
2019-02-20 13:19:51 +00:00
|
|
|
, m_device(device)
|
|
|
|
{
|
|
|
|
uint32_t configs[5];
|
|
|
|
size_t numConfigs = 5;
|
|
|
|
if (device->getDisplayConfigs(device, 0, configs, &numConfigs) != 0) {
|
|
|
|
qCWarning(KWIN_HWCOMPOSER) << "Failed to get hwcomposer display configurations";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t attr_values[5];
|
|
|
|
uint32_t attributes[] = {
|
|
|
|
HWC_DISPLAY_WIDTH,
|
|
|
|
HWC_DISPLAY_HEIGHT,
|
|
|
|
HWC_DISPLAY_DPI_X,
|
|
|
|
HWC_DISPLAY_DPI_Y,
|
|
|
|
HWC_DISPLAY_VSYNC_PERIOD ,
|
|
|
|
HWC_DISPLAY_NO_ATTRIBUTE
|
|
|
|
};
|
|
|
|
device->getDisplayAttributes(device, 0, configs[0], attributes, attr_values);
|
2019-08-27 14:52:39 +00:00
|
|
|
QSize pixelSize(attr_values[0], attr_values[1]);
|
|
|
|
if (pixelSize.isEmpty()) {
|
2019-02-20 13:19:51 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-08-27 14:52:39 +00:00
|
|
|
QSizeF physicalSize;
|
2019-02-20 13:19:51 +00:00
|
|
|
if (attr_values[2] != 0 && attr_values[3] != 0) {
|
|
|
|
static const qreal factor = 25.4;
|
2019-08-27 14:52:39 +00:00
|
|
|
physicalSize = QSizeF(qreal(pixelSize.width() * 1000) / qreal(attr_values[2]) * factor,
|
|
|
|
qreal(pixelSize.height() * 1000) / qreal(attr_values[3]) * factor);
|
2019-02-20 13:19:51 +00:00
|
|
|
} else {
|
|
|
|
// couldn't read physical size, assume 96 dpi
|
2019-08-27 14:52:39 +00:00
|
|
|
physicalSize = pixelSize / 3.8;
|
2019-02-20 13:19:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
OutputDeviceInterface::Mode mode;
|
|
|
|
mode.id = 0;
|
2019-08-27 14:52:39 +00:00
|
|
|
mode.size = pixelSize;
|
2019-02-20 13:19:51 +00:00
|
|
|
mode.flags = OutputDeviceInterface::ModeFlag::Current | OutputDeviceInterface::ModeFlag::Preferred;
|
|
|
|
mode.refreshRate = (attr_values[4] == 0) ? 60000 : 10E11/attr_values[4];
|
|
|
|
|
2019-08-28 18:54:37 +00:00
|
|
|
initInterfaces(QString(), QString(), QByteArray(), physicalSize.toSize(), {mode});
|
2019-02-20 13:19:51 +00:00
|
|
|
setInternal(true);
|
|
|
|
setDpmsSupported(true);
|
[platforms/hwcomposer] Add scaling support
Summary:
Despite plasma frameworks doing it's own scaling with fonts, it's been
requested to use kwin/wayland scaling.
Like DRM, when kscreen is not used, scale value is loaded from a config
file.
Config format is
[HWComposerOutputs][0]
Scale=N
The 0 is to map similarly to DRM and support multi-screen, but with a
screen index
rather than a UUID based on EDID.
Because we don't support multi screen this is always 0 for now.
Test Plan: Ran with the config value unset and with the config value at
Scale=3.
Reviewers: #kwin, romangg
Reviewed By: #kwin, romangg
Subscribers: bshah, romangg, nicolasfella, zzag, kwin
Tags: #kwin
Differential Revision: https://phabricator.kde.org/D18810
2019-02-22 00:44:12 +00:00
|
|
|
|
|
|
|
const auto outputGroup = kwinApp()->config()->group("HWComposerOutputs").group("0");
|
|
|
|
setScale(outputGroup.readEntry("Scale", 1));
|
2019-08-27 14:52:39 +00:00
|
|
|
setWaylandMode(pixelSize, mode.refreshRate);
|
2019-02-20 13:19:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
HwcomposerOutput::~HwcomposerOutput()
|
|
|
|
{
|
|
|
|
hwc_close_1(m_device);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool HwcomposerOutput::isValid() const
|
|
|
|
{
|
2019-08-27 14:19:47 +00:00
|
|
|
return isEnabled();
|
2019-02-20 13:19:51 +00:00
|
|
|
}
|
|
|
|
|
2020-04-29 15:18:41 +00:00
|
|
|
void HwcomposerOutput::updateDpms(KWaylandServer::OutputInterface::DpmsMode mode)
|
2019-02-20 13:19:51 +00:00
|
|
|
{
|
|
|
|
emit dpmsModeRequested(mode);
|
|
|
|
}
|
|
|
|
|
2015-05-06 15:47:07 +00:00
|
|
|
}
|