2015-04-09 12:49:32 +00:00
|
|
|
/********************************************************************
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
Copyright (C) 2015 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*********************************************************************/
|
|
|
|
#ifndef KWIN_DRM_BACKEND_H
|
|
|
|
#define KWIN_DRM_BACKEND_H
|
|
|
|
#include "abstract_backend.h"
|
2016-02-15 14:40:25 +00:00
|
|
|
#include "input.h"
|
2015-04-09 12:49:32 +00:00
|
|
|
|
2016-02-15 16:35:36 +00:00
|
|
|
#include <QElapsedTimer>
|
2015-04-09 12:49:32 +00:00
|
|
|
#include <QImage>
|
2015-06-29 10:35:38 +00:00
|
|
|
#include <QPointer>
|
2015-04-09 12:49:32 +00:00
|
|
|
#include <QSize>
|
|
|
|
#include <xf86drmMode.h>
|
|
|
|
|
2015-04-10 08:44:07 +00:00
|
|
|
struct gbm_bo;
|
|
|
|
struct gbm_surface;
|
|
|
|
|
2015-04-24 10:01:39 +00:00
|
|
|
namespace KWayland
|
|
|
|
{
|
|
|
|
namespace Server
|
|
|
|
{
|
|
|
|
class OutputInterface;
|
2016-03-10 18:57:07 +00:00
|
|
|
class OutputDeviceInterface;
|
|
|
|
class OutputChangeSet;
|
|
|
|
class OutputManagementInterface;
|
2015-04-24 10:01:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-09 12:49:32 +00:00
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
|
|
|
|
class Udev;
|
2015-04-24 06:15:55 +00:00
|
|
|
class UdevMonitor;
|
2015-04-09 12:49:32 +00:00
|
|
|
|
|
|
|
class DrmBuffer;
|
2015-04-14 09:25:16 +00:00
|
|
|
class DrmOutput;
|
2016-02-15 14:40:25 +00:00
|
|
|
class DpmsInputEventFilter;
|
2015-04-09 12:49:32 +00:00
|
|
|
|
2015-08-27 14:15:04 +00:00
|
|
|
template <typename Pointer, void (*cleanupFunc)(Pointer*)>
|
|
|
|
struct DrmCleanup
|
|
|
|
{
|
|
|
|
static inline void cleanup(Pointer *ptr)
|
|
|
|
{
|
|
|
|
cleanupFunc(ptr);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
template <typename T, void (*cleanupFunc)(T*)> using ScopedDrmPointer = QScopedPointer<T, DrmCleanup<T, cleanupFunc>>;
|
|
|
|
|
2015-04-09 12:49:32 +00:00
|
|
|
class KWIN_EXPORT DrmBackend : public AbstractBackend
|
|
|
|
{
|
|
|
|
Q_OBJECT
|
2015-05-05 15:58:09 +00:00
|
|
|
Q_INTERFACES(KWin::AbstractBackend)
|
|
|
|
Q_PLUGIN_METADATA(IID "org.kde.kwin.AbstractBackend" FILE "drm.json")
|
2015-04-09 12:49:32 +00:00
|
|
|
public:
|
|
|
|
explicit DrmBackend(QObject *parent = nullptr);
|
|
|
|
virtual ~DrmBackend();
|
|
|
|
|
2016-03-10 18:57:07 +00:00
|
|
|
void configurationChangeRequested(KWayland::Server::OutputConfigurationInterface *config) override;
|
2015-04-09 12:49:32 +00:00
|
|
|
Screens *createScreens(QObject *parent = nullptr) override;
|
|
|
|
QPainterBackend *createQPainterBackend() override;
|
2015-04-10 08:44:07 +00:00
|
|
|
OpenGLBackend* createOpenGLBackend() override;
|
2015-04-09 12:49:32 +00:00
|
|
|
|
2015-05-06 06:32:39 +00:00
|
|
|
void init() override;
|
2015-04-09 12:49:32 +00:00
|
|
|
DrmBuffer *createBuffer(const QSize &size);
|
2015-04-10 08:44:07 +00:00
|
|
|
DrmBuffer *createBuffer(gbm_surface *surface);
|
2015-04-17 09:16:00 +00:00
|
|
|
void present(DrmBuffer *buffer, DrmOutput *output);
|
2015-04-09 12:49:32 +00:00
|
|
|
|
2015-04-14 09:25:16 +00:00
|
|
|
QSize size() const;
|
2015-04-09 12:49:32 +00:00
|
|
|
int fd() const {
|
|
|
|
return m_fd;
|
|
|
|
}
|
2015-04-17 09:13:31 +00:00
|
|
|
QVector<DrmOutput*> outputs() const {
|
|
|
|
return m_outputs;
|
|
|
|
}
|
2015-04-24 08:23:06 +00:00
|
|
|
QVector<DrmBuffer*> buffers() const {
|
|
|
|
return m_buffers;
|
|
|
|
}
|
|
|
|
void bufferDestroyed(DrmBuffer *b);
|
2015-04-09 12:49:32 +00:00
|
|
|
|
2016-02-15 14:40:25 +00:00
|
|
|
void outputWentOff();
|
|
|
|
void checkOutputsAreOn();
|
|
|
|
|
|
|
|
public Q_SLOTS:
|
2016-02-15 15:30:22 +00:00
|
|
|
void turnOutputsOn();
|
2016-02-15 14:40:25 +00:00
|
|
|
|
|
|
|
|
2015-04-09 12:49:32 +00:00
|
|
|
Q_SIGNALS:
|
2015-04-24 06:15:55 +00:00
|
|
|
void outputRemoved(KWin::DrmOutput *output);
|
|
|
|
void outputAdded(KWin::DrmOutput *output);
|
2015-04-09 12:49:32 +00:00
|
|
|
|
|
|
|
private:
|
2015-04-13 07:23:03 +00:00
|
|
|
static void pageFlipHandler(int fd, unsigned int frame, unsigned int sec, unsigned int usec, void *data);
|
2015-04-09 12:49:32 +00:00
|
|
|
void openDrm();
|
2015-04-15 16:26:25 +00:00
|
|
|
void activate(bool active);
|
|
|
|
void reactivate();
|
|
|
|
void deactivate();
|
2015-04-09 12:49:32 +00:00
|
|
|
void queryResources();
|
2015-04-13 14:25:34 +00:00
|
|
|
void setCursor();
|
|
|
|
void updateCursor();
|
|
|
|
void hideCursor();
|
|
|
|
void moveCursor();
|
|
|
|
void initCursor();
|
2015-04-17 14:41:18 +00:00
|
|
|
quint32 findCrtc(drmModeRes *res, drmModeConnector *connector, bool *ok = nullptr);
|
|
|
|
bool crtcIsUsed(quint32 crtc);
|
2015-08-31 09:41:19 +00:00
|
|
|
void outputDpmsChanged();
|
2015-11-02 08:34:32 +00:00
|
|
|
void readOutputsConfiguration();
|
|
|
|
QByteArray generateOutputConfigurationUuid() const;
|
2015-04-24 06:15:55 +00:00
|
|
|
DrmOutput *findOutput(quint32 connector);
|
2016-03-10 18:57:07 +00:00
|
|
|
DrmOutput *findOutput(const QByteArray &uuid);
|
2015-04-09 12:49:32 +00:00
|
|
|
QScopedPointer<Udev> m_udev;
|
2015-04-24 06:15:55 +00:00
|
|
|
QScopedPointer<UdevMonitor> m_udevMonitor;
|
2015-04-09 12:49:32 +00:00
|
|
|
int m_fd = -1;
|
|
|
|
int m_drmId = 0;
|
2015-04-14 09:25:16 +00:00
|
|
|
QVector<DrmOutput*> m_outputs;
|
|
|
|
DrmBuffer *m_cursor[2];
|
|
|
|
int m_cursorIndex = 0;
|
|
|
|
int m_pageFlipsPending = 0;
|
2015-04-15 16:26:25 +00:00
|
|
|
bool m_active = false;
|
2015-04-24 08:23:06 +00:00
|
|
|
QVector<DrmBuffer*> m_buffers;
|
2016-02-15 14:40:25 +00:00
|
|
|
QScopedPointer<DpmsInputEventFilter> m_dpmsFilter;
|
2016-03-10 18:57:07 +00:00
|
|
|
KWayland::Server::OutputManagementInterface *m_outputManagement = nullptr;
|
2015-04-14 09:25:16 +00:00
|
|
|
};
|
|
|
|
|
2015-08-31 07:02:44 +00:00
|
|
|
class DrmOutput : public QObject
|
2015-04-14 09:25:16 +00:00
|
|
|
{
|
2015-08-31 07:02:44 +00:00
|
|
|
Q_OBJECT
|
2015-04-14 09:25:16 +00:00
|
|
|
public:
|
2015-04-27 09:06:04 +00:00
|
|
|
struct Edid {
|
|
|
|
QByteArray eisaId;
|
|
|
|
QByteArray monitorName;
|
|
|
|
QByteArray serialNumber;
|
|
|
|
QSize physicalSize;
|
|
|
|
};
|
2015-04-14 09:25:16 +00:00
|
|
|
virtual ~DrmOutput();
|
|
|
|
void showCursor(DrmBuffer *buffer);
|
|
|
|
void hideCursor();
|
|
|
|
void moveCursor(const QPoint &globalPos);
|
|
|
|
bool present(DrmBuffer *buffer);
|
|
|
|
void pageFlipped();
|
2015-04-27 09:06:04 +00:00
|
|
|
void init(drmModeConnector *connector);
|
2015-04-15 16:26:25 +00:00
|
|
|
void restoreSaved();
|
|
|
|
void blank();
|
2015-04-14 09:25:16 +00:00
|
|
|
|
2016-03-10 18:57:07 +00:00
|
|
|
/**
|
|
|
|
* This sets the changes and tests them against the DRM output
|
|
|
|
*/
|
|
|
|
void setChanges(KWayland::Server::OutputChangeSet *changeset);
|
|
|
|
bool commitChanges();
|
|
|
|
|
2015-04-14 09:25:16 +00:00
|
|
|
QSize size() const;
|
2015-04-17 09:13:31 +00:00
|
|
|
QRect geometry() const;
|
2015-05-19 07:35:57 +00:00
|
|
|
QString name() const;
|
2015-05-19 07:49:37 +00:00
|
|
|
int currentRefreshRate() const;
|
2015-08-27 14:15:04 +00:00
|
|
|
enum class DpmsMode {
|
|
|
|
On = DRM_MODE_DPMS_ON,
|
|
|
|
Standby = DRM_MODE_DPMS_STANDBY,
|
|
|
|
Suspend = DRM_MODE_DPMS_SUSPEND,
|
|
|
|
Off = DRM_MODE_DPMS_OFF
|
|
|
|
};
|
|
|
|
void setDpms(DpmsMode mode);
|
2015-08-31 09:41:19 +00:00
|
|
|
bool isDpmsEnabled() const {
|
|
|
|
return m_dpmsMode == DpmsMode::On;
|
|
|
|
}
|
|
|
|
|
2015-11-02 08:34:32 +00:00
|
|
|
QByteArray uuid() const {
|
|
|
|
return m_uuid;
|
|
|
|
}
|
|
|
|
|
2015-08-31 09:41:19 +00:00
|
|
|
Q_SIGNALS:
|
|
|
|
void dpmsChanged();
|
2015-04-14 09:25:16 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
friend class DrmBackend;
|
|
|
|
DrmOutput(DrmBackend *backend);
|
|
|
|
void cleanupBlackBuffer();
|
2015-04-16 11:23:33 +00:00
|
|
|
bool setMode(DrmBuffer *buffer);
|
2015-04-27 09:06:04 +00:00
|
|
|
void initEdid(drmModeConnector *connector);
|
2015-08-27 14:15:04 +00:00
|
|
|
void initDpms(drmModeConnector *connector);
|
2015-04-28 06:16:07 +00:00
|
|
|
bool isCurrentMode(const drmModeModeInfo *mode) const;
|
2015-11-02 08:34:32 +00:00
|
|
|
void initUuid();
|
|
|
|
void setGlobalPos(const QPoint &pos);
|
2015-04-14 09:25:16 +00:00
|
|
|
|
|
|
|
DrmBackend *m_backend;
|
|
|
|
QPoint m_globalPos;
|
2015-04-09 12:49:32 +00:00
|
|
|
quint32 m_crtcId = 0;
|
|
|
|
quint32 m_connector = 0;
|
2015-04-16 11:23:33 +00:00
|
|
|
quint32 m_lastStride = 0;
|
2016-01-11 11:53:14 +00:00
|
|
|
bool m_lastGbm = false;
|
2015-04-09 12:49:32 +00:00
|
|
|
drmModeModeInfo m_mode;
|
2015-04-14 09:25:16 +00:00
|
|
|
DrmBuffer *m_currentBuffer = nullptr;
|
|
|
|
DrmBuffer *m_blackBuffer = nullptr;
|
|
|
|
struct CrtcCleanup {
|
|
|
|
static void inline cleanup(_drmModeCrtc *ptr) {
|
|
|
|
drmModeFreeCrtc(ptr);
|
|
|
|
}
|
|
|
|
};
|
2015-04-27 09:06:04 +00:00
|
|
|
Edid m_edid;
|
2015-04-14 09:25:16 +00:00
|
|
|
QScopedPointer<_drmModeCrtc, CrtcCleanup> m_savedCrtc;
|
2015-06-29 10:35:38 +00:00
|
|
|
QPointer<KWayland::Server::OutputInterface> m_waylandOutput;
|
2016-03-10 18:57:07 +00:00
|
|
|
QPointer<KWayland::Server::OutputDeviceInterface> m_waylandOutputDevice;
|
|
|
|
QPointer<KWayland::Server::OutputChangeSet> m_changeset;
|
2015-08-27 14:15:04 +00:00
|
|
|
ScopedDrmPointer<_drmModeProperty, &drmModeFreeProperty> m_dpms;
|
|
|
|
DpmsMode m_dpmsMode = DpmsMode::On;
|
2015-11-02 08:34:32 +00:00
|
|
|
QByteArray m_uuid;
|
2015-04-09 12:49:32 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class DrmBuffer
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
~DrmBuffer();
|
|
|
|
|
2015-04-13 14:25:34 +00:00
|
|
|
bool map(QImage::Format format = QImage::Format_RGB32);
|
2015-04-09 12:49:32 +00:00
|
|
|
QImage *image() const {
|
|
|
|
return m_image;
|
|
|
|
}
|
2015-04-14 09:25:16 +00:00
|
|
|
quint32 handle() const {
|
|
|
|
return m_handle;
|
|
|
|
}
|
|
|
|
const QSize &size() const {
|
|
|
|
return m_size;
|
|
|
|
}
|
|
|
|
quint32 bufferId() const {
|
|
|
|
return m_bufferId;
|
|
|
|
}
|
2015-04-16 11:23:33 +00:00
|
|
|
quint32 stride() const {
|
|
|
|
return m_stride;
|
|
|
|
}
|
2015-04-24 08:23:06 +00:00
|
|
|
gbm_bo *gbm() const {
|
|
|
|
return m_bo;
|
|
|
|
}
|
2016-01-11 11:53:14 +00:00
|
|
|
bool isGbm() const {
|
|
|
|
return m_bo != nullptr;
|
|
|
|
}
|
2015-04-14 09:25:16 +00:00
|
|
|
void releaseGbm();
|
2015-04-09 12:49:32 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
friend class DrmBackend;
|
|
|
|
DrmBuffer(DrmBackend *backend, const QSize &size);
|
2015-04-10 08:44:07 +00:00
|
|
|
DrmBuffer(DrmBackend *backend, gbm_surface *surface);
|
2015-04-09 12:49:32 +00:00
|
|
|
DrmBackend *m_backend;
|
2015-04-10 08:44:07 +00:00
|
|
|
gbm_surface *m_surface = nullptr;
|
|
|
|
gbm_bo *m_bo = nullptr;
|
2015-04-09 12:49:32 +00:00
|
|
|
QSize m_size;
|
|
|
|
quint32 m_handle = 0;
|
|
|
|
quint32 m_bufferId = 0;
|
|
|
|
quint32 m_stride = 0;
|
|
|
|
quint64 m_bufferSize = 0;
|
|
|
|
void *m_memory = nullptr;
|
|
|
|
QImage *m_image = nullptr;
|
|
|
|
};
|
|
|
|
|
2016-02-15 14:40:25 +00:00
|
|
|
class DpmsInputEventFilter : public InputEventFilter
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
DpmsInputEventFilter(DrmBackend *backend);
|
|
|
|
~DpmsInputEventFilter();
|
|
|
|
|
|
|
|
bool pointerEvent(QMouseEvent *event, quint32 nativeButton) override;
|
|
|
|
bool wheelEvent(QWheelEvent *event) override;
|
|
|
|
bool keyEvent(QKeyEvent *event) override;
|
2016-02-15 16:35:36 +00:00
|
|
|
bool touchDown(quint32 id, const QPointF &pos, quint32 time) override;
|
|
|
|
bool touchMotion(quint32 id, const QPointF &pos, quint32 time) override;
|
|
|
|
bool touchUp(quint32 id, quint32 time) override;
|
2016-02-15 14:40:25 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
void notify();
|
|
|
|
DrmBackend *m_backend;
|
2016-02-15 16:35:36 +00:00
|
|
|
QElapsedTimer m_doubleTapTimer;
|
|
|
|
QVector<qint32> m_touchPoints;
|
|
|
|
bool m_secondTap = false;
|
2016-02-15 14:40:25 +00:00
|
|
|
};
|
|
|
|
|
2015-04-09 12:49:32 +00:00
|
|
|
}
|
|
|
|
|
2015-04-24 06:15:55 +00:00
|
|
|
Q_DECLARE_METATYPE(KWin::DrmOutput*)
|
|
|
|
|
2015-04-09 12:49:32 +00:00
|
|
|
#endif
|
|
|
|
|