[wayland] Add a basic drm/kms backend
Introduces a new (optional) dependency: libdrm. The DrmBackend currently supports finding the first connected output. It can create shared memory buffers which are used by SceneQPainter to do double buffered rendering. There is still lots to do, the following things are not yet working: * multiple outputs * page flip * OpenGL (through gbm) * restoring mode setting to start valuemaster
parent
da3f7914f7
commit
c759551340
@ -0,0 +1,126 @@
|
||||
#.rst:
|
||||
# FindLibdrm
|
||||
# -------
|
||||
#
|
||||
# Try to find libdrm on a Unix system.
|
||||
#
|
||||
# This will define the following variables:
|
||||
#
|
||||
# ``Libdrm_FOUND``
|
||||
# True if (the requested version of) libdrm is available
|
||||
# ``Libdrm_VERSION``
|
||||
# The version of libdrm
|
||||
# ``Libdrm_LIBRARIES``
|
||||
# This can be passed to target_link_libraries() instead of the ``Libdrm::Libdrm``
|
||||
# target
|
||||
# ``Libdrm_INCLUDE_DIRS``
|
||||
# This should be passed to target_include_directories() if the target is not
|
||||
# used for linking
|
||||
# ``Libdrm_DEFINITIONS``
|
||||
# This should be passed to target_compile_options() if the target is not
|
||||
# used for linking
|
||||
#
|
||||
# If ``Libdrm_FOUND`` is TRUE, it will also define the following imported target:
|
||||
#
|
||||
# ``Libdrm::Libdrm``
|
||||
# The libdrm library
|
||||
#
|
||||
# In general we recommend using the imported target, as it is easier to use.
|
||||
# Bear in mind, however, that if the target is in the link interface of an
|
||||
# exported library, it must be made available by the package config file.
|
||||
|
||||
#=============================================================================
|
||||
# Copyright 2014 Alex Merry <alex.merry@kde.org>
|
||||
# Copyright 2014 Martin Gräßlin <mgraesslin@kde.org>
|
||||
#
|
||||
# Redistribution and use in source and binary forms, with or without
|
||||
# modification, are permitted provided that the following conditions
|
||||
# are met:
|
||||
#
|
||||
# 1. Redistributions of source code must retain the copyright
|
||||
# notice, this list of conditions and the following disclaimer.
|
||||
# 2. Redistributions in binary form must reproduce the copyright
|
||||
# notice, this list of conditions and the following disclaimer in the
|
||||
# documentation and/or other materials provided with the distribution.
|
||||
# 3. The name of the author may not be used to endorse or promote products
|
||||
# derived from this software without specific prior written permission.
|
||||
#
|
||||
# THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
||||
# IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
||||
# OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
||||
# IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
||||
# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
||||
# NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
||||
# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
||||
# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||||
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
||||
# THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||
#=============================================================================
|
||||
|
||||
if(CMAKE_VERSION VERSION_LESS 2.8.12)
|
||||
message(FATAL_ERROR "CMake 2.8.12 is required by FindLibdrm.cmake")
|
||||
endif()
|
||||
if(CMAKE_MINIMUM_REQUIRED_VERSION VERSION_LESS 2.8.12)
|
||||
message(AUTHOR_WARNING "Your project should require at least CMake 2.8.12 to use FindLibdrm.cmake")
|
||||
endif()
|
||||
|
||||
if(NOT WIN32)
|
||||
# Use pkg-config to get the directories and then use these values
|
||||
# in the FIND_PATH() and FIND_LIBRARY() calls
|
||||
find_package(PkgConfig)
|
||||
pkg_check_modules(PKG_Libdrm QUIET libdrm)
|
||||
|
||||
set(Libdrm_DEFINITIONS ${PKG_Libdrm_CFLAGS_OTHER})
|
||||
set(Libdrm_VERSION ${PKG_Libdrm_VERSION})
|
||||
|
||||
find_path(Libdrm_INCLUDE_DIR
|
||||
NAMES
|
||||
xf86drm.h
|
||||
HINTS
|
||||
${PKG_Libdrm_INCLUDE_DIRS}
|
||||
)
|
||||
find_library(Libdrm_LIBRARY
|
||||
NAMES
|
||||
drm
|
||||
HINTS
|
||||
${PKG_Libdrm_LIBRARY_DIRS}
|
||||
)
|
||||
|
||||
include(FindPackageHandleStandardArgs)
|
||||
find_package_handle_standard_args(Libdrm
|
||||
FOUND_VAR
|
||||
Libdrm_FOUND
|
||||
REQUIRED_VARS
|
||||
Libdrm_LIBRARY
|
||||
Libdrm_INCLUDE_DIR
|
||||
VERSION_VAR
|
||||
Libdrm_VERSION
|
||||
)
|
||||
|
||||
if(Libdrm_FOUND AND NOT TARGET Libdrm::Libdrm)
|
||||
add_library(Libdrm::Libdrm UNKNOWN IMPORTED)
|
||||
set_target_properties(Libdrm::Libdrm PROPERTIES
|
||||
IMPORTED_LOCATION "${Libdrm_LIBRARY}"
|
||||
INTERFACE_COMPILE_OPTIONS "${Libdrm_DEFINITIONS}"
|
||||
INTERFACE_INCLUDE_DIRECTORIES "${Libdrm_INCLUDE_DIR}"
|
||||
INTERFACE_INCLUDE_DIRECTORIES "${Libdrm_INCLUDE_DIR}/libdrm"
|
||||
)
|
||||
endif()
|
||||
|
||||
mark_as_advanced(Libdrm_LIBRARY Libdrm_INCLUDE_DIR)
|
||||
|
||||
# compatibility variables
|
||||
set(Libdrm_LIBRARIES ${Libdrm_LIBRARY})
|
||||
set(Libdrm_INCLUDE_DIRS ${Libdrm_INCLUDE_DIR})
|
||||
set(Libdrm_VERSION_STRING ${Libdrm_VERSION})
|
||||
|
||||
else()
|
||||
message(STATUS "FindLibdrm.cmake cannot find libdrm on Windows systems.")
|
||||
set(Libdrm_FOUND FALSE)
|
||||
endif()
|
||||
|
||||
include(FeatureSummary)
|
||||
set_package_properties(Libdrm PROPERTIES
|
||||
URL "https://wiki.freedesktop.org/dri/"
|
||||
DESCRIPTION "Userspace interface to kernel DRM services."
|
||||
)
|
@ -0,0 +1,212 @@
|
||||
/********************************************************************
|
||||
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/>.
|
||||
*********************************************************************/
|
||||
#include "drm_backend.h"
|
||||
#include "logind.h"
|
||||
#include "scene_qpainter.h"
|
||||
#include "screens_drm.h"
|
||||
#include "udev.h"
|
||||
#include "utils.h"
|
||||
#include "virtual_terminal.h"
|
||||
// system
|
||||
#include <unistd.h>
|
||||
#include <sys/mman.h>
|
||||
// drm
|
||||
#include <xf86drm.h>
|
||||
#include <xf86drmMode.h>
|
||||
#include <libdrm/drm_mode.h>
|
||||
|
||||
#include <QDebug>
|
||||
|
||||
namespace KWin
|
||||
{
|
||||
|
||||
DrmBackend::DrmBackend(QObject *parent)
|
||||
: AbstractBackend(parent)
|
||||
, m_udev(new Udev)
|
||||
{
|
||||
}
|
||||
|
||||
DrmBackend::~DrmBackend()
|
||||
{
|
||||
if (m_fd >= 0) {
|
||||
close(m_fd);
|
||||
}
|
||||
}
|
||||
|
||||
void DrmBackend::init()
|
||||
{
|
||||
LogindIntegration *logind = LogindIntegration::self();
|
||||
auto takeControl = [logind, this]() {
|
||||
if (logind->hasSessionControl()) {
|
||||
openDrm();
|
||||
} else {
|
||||
logind->takeControl();
|
||||
connect(logind, &LogindIntegration::hasSessionControlChanged, this, &DrmBackend::openDrm);
|
||||
}
|
||||
};
|
||||
if (logind->isConnected()) {
|
||||
takeControl();
|
||||
} else {
|
||||
connect(logind, &LogindIntegration::connectedChanged, this, takeControl);
|
||||
}
|
||||
VirtualTerminal::create(this);
|
||||
}
|
||||
|
||||
void DrmBackend::openDrm()
|
||||
{
|
||||
VirtualTerminal::self()->init();
|
||||
UdevDevice::Ptr device = m_udev->primaryGpu();
|
||||
if (!device) {
|
||||
qCWarning(KWIN_CORE) << "Did not find a GPU";
|
||||
return;
|
||||
}
|
||||
int fd = LogindIntegration::self()->takeDevice(device->devNode());
|
||||
if (fd < 0) {
|
||||
qCWarning(KWIN_CORE) << "failed to open drm device at" << device->devNode();
|
||||
return;
|
||||
}
|
||||
m_fd = fd;
|
||||
m_drmId = device->sysNum();
|
||||
queryResources();
|
||||
emit screensQueried();
|
||||
}
|
||||
|
||||
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>>;
|
||||
|
||||
void DrmBackend::queryResources()
|
||||
{
|
||||
if (m_fd < 0) {
|
||||
return;
|
||||
}
|
||||
ScopedDrmPointer<_drmModeRes, &drmModeFreeResources> resources(drmModeGetResources(m_fd));
|
||||
if (!resources) {
|
||||
qCWarning(KWIN_CORE) << "drmModeGetResources failed";
|
||||
return;
|
||||
}
|
||||
for (int i = 0; i < resources->count_connectors; ++i) {
|
||||
const auto id = resources->connectors[i];
|
||||
ScopedDrmPointer<_drmModeConnector, &drmModeFreeConnector> connector(drmModeGetConnector(m_fd, id));
|
||||
if (!connector) {
|
||||
continue;
|
||||
}
|
||||
if (connector->connection != DRM_MODE_CONNECTED) {
|
||||
continue;
|
||||
}
|
||||
ScopedDrmPointer<_drmModeEncoder, &drmModeFreeEncoder> encoder(drmModeGetEncoder(m_fd, connector->encoder_id));
|
||||
if (!encoder) {
|
||||
continue;
|
||||
}
|
||||
ScopedDrmPointer<_drmModeCrtc, &drmModeFreeCrtc> crtc(drmModeGetCrtc(m_fd, encoder->crtc_id));
|
||||
if (!crtc) {
|
||||
continue;
|
||||
}
|
||||
m_resolution = QSize(crtc->mode.hdisplay, crtc->mode.vdisplay);
|
||||
m_crtcId = encoder->crtc_id;
|
||||
m_connector = connector->connector_id;
|
||||
m_mode = crtc->mode;
|
||||
// for the moment only one crtc
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void DrmBackend::present(DrmBuffer *buffer)
|
||||
{
|
||||
drmModeSetCrtc(m_fd, m_crtcId, buffer->m_bufferId, 0, 0, &m_connector, 1, &m_mode);
|
||||
}
|
||||
|
||||
Screens *DrmBackend::createScreens(QObject *parent)
|
||||
{
|
||||
return new DrmScreens(this, parent);
|
||||
}
|
||||
|
||||
QPainterBackend *DrmBackend::createQPainterBackend()
|
||||
{
|
||||
return new DrmQPainterBackend(this);
|
||||
}
|
||||
|
||||
DrmBuffer *DrmBackend::createBuffer(const QSize &size)
|
||||
{
|
||||
return new DrmBuffer(this, size);
|
||||
}
|
||||
|
||||
DrmBuffer::DrmBuffer(DrmBackend *backend, const QSize &size)
|
||||
: m_backend(backend)
|
||||
, m_size(size)
|
||||
{
|
||||
drm_mode_create_dumb createArgs;
|
||||
memset(&createArgs, 0, sizeof createArgs);
|
||||
createArgs.bpp = 32;
|
||||
createArgs.width = size.width();
|
||||
createArgs.height = size.height();
|
||||
if (drmIoctl(m_backend->fd(), DRM_IOCTL_MODE_CREATE_DUMB, &createArgs) != 0) {
|
||||
return;
|
||||
}
|
||||
m_handle = createArgs.handle;
|
||||
m_bufferSize = createArgs.size;
|
||||
m_stride = createArgs.pitch;
|
||||
drmModeAddFB(m_backend->fd(), size.width(), size.height(), 24, 32,
|
||||
m_stride, createArgs.handle, &m_bufferId);
|
||||
}
|
||||
|
||||
DrmBuffer::~DrmBuffer()
|
||||
{
|
||||
delete m_image;
|
||||
if (m_memory) {
|
||||
munmap(m_memory, m_bufferSize);
|
||||
}
|
||||
if (m_bufferId) {
|
||||
drmModeRmFB(m_backend->fd(), m_bufferId);
|
||||
}
|
||||
if (m_handle) {
|
||||
drm_mode_destroy_dumb destroyArgs;
|
||||
destroyArgs.handle = m_handle;
|
||||
drmIoctl(m_backend->fd(), DRM_IOCTL_MODE_DESTROY_DUMB, &destroyArgs);
|
||||
}
|
||||
}
|
||||
|
||||
bool DrmBuffer::map()
|
||||
{
|
||||
if (!m_handle || !m_bufferId) {
|
||||
return false;
|
||||
}
|
||||
drm_mode_map_dumb mapArgs;
|
||||
memset(&mapArgs, 0, sizeof mapArgs);
|
||||
mapArgs.handle = m_handle;
|
||||
if (drmIoctl(m_backend->fd(), DRM_IOCTL_MODE_MAP_DUMB, &mapArgs) != 0) {
|
||||
return false;
|
||||
}
|
||||
void *address = mmap(nullptr, m_bufferSize, PROT_WRITE, MAP_SHARED, m_backend->fd(), mapArgs.offset);
|
||||
if (address == MAP_FAILED) {
|
||||
return false;
|
||||
}
|
||||
m_memory = address;
|
||||
m_image = new QImage((uchar*)m_memory, m_size.width(), m_size.height(), m_stride, QImage::Format_RGB32);
|
||||
return !m_image->isNull();
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,99 @@
|
||||
/********************************************************************
|
||||
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"
|
||||
|
||||
#include <QImage>
|
||||
#include <QSize>
|
||||
#include <xf86drmMode.h>
|
||||
|
||||
namespace KWin
|
||||
{
|
||||
|
||||
class Udev;
|
||||
|
||||
class DrmBuffer;
|
||||
|
||||
class KWIN_EXPORT DrmBackend : public AbstractBackend
|
||||
{
|
||||
Q_OBJECT
|
||||
public:
|
||||
explicit DrmBackend(QObject *parent = nullptr);
|
||||
virtual ~DrmBackend();
|
||||
|
||||
Screens *createScreens(QObject *parent = nullptr) override;
|
||||
QPainterBackend *createQPainterBackend() override;
|
||||
|
||||
void init();
|
||||
DrmBuffer *createBuffer(const QSize &size);
|
||||
void present(DrmBuffer *buffer);
|
||||
|
||||
QSize size() const {
|
||||
// TODO: this is wrong
|
||||
return m_resolution;
|
||||
}
|
||||
int fd() const {
|
||||
return m_fd;
|
||||
}
|
||||
|
||||
Q_SIGNALS:
|
||||
void screensQueried();
|
||||
|
||||
private:
|
||||
void openDrm();
|
||||
void queryResources();
|
||||
QScopedPointer<Udev> m_udev;
|
||||
int m_fd = -1;
|
||||
int m_drmId = 0;
|
||||
// TODO: this is wrong
|
||||
QSize m_resolution;
|
||||
quint32 m_crtcId = 0;
|
||||
quint32 m_connector = 0;
|
||||
drmModeModeInfo m_mode;
|
||||
};
|
||||
|
||||
class DrmBuffer
|
||||
{
|
||||
public:
|
||||
~DrmBuffer();
|
||||
|
||||
bool map();
|
||||
QImage *image() const {
|
||||
return m_image;
|
||||
}
|
||||
|
||||
private:
|
||||
friend class DrmBackend;
|
||||
DrmBuffer(DrmBackend *backend, const QSize &size);
|
||||
DrmBackend *m_backend;
|
||||
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;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif
|
||||
|
@ -0,0 +1,68 @@
|
||||
/********************************************************************
|
||||
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/>.
|
||||
*********************************************************************/
|
||||
#include "screens_drm.h"
|
||||
#include "drm_backend.h"
|
||||
|
||||
namespace KWin
|
||||
{
|
||||
|
||||
DrmScreens::DrmScreens(DrmBackend *backend, QObject *parent)
|
||||
: Screens(parent)
|
||||
, m_backend(backend)
|
||||
{
|
||||
}
|
||||
|
||||
DrmScreens::~DrmScreens() = default;
|
||||
|
||||
void DrmScreens::init()
|
||||
{
|
||||
KWin::Screens::init();
|
||||
updateCount();
|
||||
emit changed();
|
||||
}
|
||||
|
||||
QRect DrmScreens::geometry(int screen) const
|
||||
{
|
||||
if (screen == 0) {
|
||||
return QRect(QPoint(0, 0), size(screen));
|
||||
}
|
||||
return QRect();
|
||||
}
|
||||
|
||||
QSize DrmScreens::size(int screen) const
|
||||
{
|
||||
if (screen == 0) {
|
||||
return m_backend->size();
|
||||
}
|
||||
return QSize();
|
||||
}
|
||||
|
||||
void DrmScreens::updateCount()
|
||||
{
|
||||
setCount(1);
|
||||
}
|
||||
|
||||
int DrmScreens::number(const QPoint &pos) const
|
||||
{
|
||||
Q_UNUSED(pos)
|
||||
return 0;
|
||||
}
|
||||
|
||||
}
|
@ -0,0 +1,46 @@
|
||||
/********************************************************************
|
||||
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_SCREENS_DRM_H
|
||||
#define KWIN_SCREENS_DRM_H
|
||||
#include "screens.h"
|
||||
|
||||
namespace KWin
|
||||
{
|
||||
class DrmBackend;
|
||||
|
||||
class DrmScreens : public Screens
|
||||
{
|
||||
Q_OBJECT
|
||||
public:
|
||||
DrmScreens(DrmBackend *backend, QObject *parent = nullptr);
|
||||
virtual ~DrmScreens();
|
||||
void init() override;
|
||||
QRect geometry(int screen) const override;
|
||||
int number(const QPoint &pos) const override;
|
||||
QSize size(int screen) const override;
|
||||
void updateCount() override;
|
||||
|
||||
private:
|
||||
DrmBackend *m_backend;
|
||||
};
|
||||
|
||||
}
|
||||
|
||||
#endif
|
Loading…
Reference in New Issue