9ca24efc07
Summary: No kwin changes Test Plan: Compiles More tests pass Reviewers: #kwin, graesslin Reviewed By: #kwin, graesslin Subscribers: kwin Tags: #kwin Differential Revision: https://phabricator.kde.org/D16148
577 lines
19 KiB
C++
577 lines
19 KiB
C++
/********************************************************************
|
|
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 "kwin_wayland_test.h"
|
|
#include "shell_client.h"
|
|
#include "screenlockerwatcher.h"
|
|
#include "wayland_server.h"
|
|
|
|
#include <KWayland/Client/compositor.h>
|
|
#include <KWayland/Client/connection_thread.h>
|
|
#include <KWayland/Client/event_queue.h>
|
|
#include <KWayland/Client/idleinhibit.h>
|
|
#include <KWayland/Client/registry.h>
|
|
#include <KWayland/Client/plasmashell.h>
|
|
#include <KWayland/Client/plasmawindowmanagement.h>
|
|
#include <KWayland/Client/pointerconstraints.h>
|
|
#include <KWayland/Client/seat.h>
|
|
#include <KWayland/Client/server_decoration.h>
|
|
#include <KWayland/Client/shadow.h>
|
|
#include <KWayland/Client/shell.h>
|
|
#include <KWayland/Client/shm_pool.h>
|
|
#include <KWayland/Client/output.h>
|
|
#include <KWayland/Client/surface.h>
|
|
#include <KWayland/Client/appmenu.h>
|
|
#include <KWayland/Client/xdgshell.h>
|
|
#include <KWayland/Server/display.h>
|
|
|
|
//screenlocker
|
|
#include <KScreenLocker/KsldApp>
|
|
|
|
#include <QThread>
|
|
|
|
// system
|
|
#include <sys/types.h>
|
|
#include <sys/socket.h>
|
|
#include <unistd.h>
|
|
|
|
using namespace KWayland::Client;
|
|
|
|
namespace KWin
|
|
{
|
|
namespace Test
|
|
{
|
|
|
|
static struct {
|
|
ConnectionThread *connection = nullptr;
|
|
EventQueue *queue = nullptr;
|
|
Compositor *compositor = nullptr;
|
|
ServerSideDecorationManager *decoration = nullptr;
|
|
ShadowManager *shadowManager = nullptr;
|
|
Shell *shell = nullptr;
|
|
XdgShell *xdgShellV5 = nullptr;
|
|
XdgShell *xdgShellV6 = nullptr;
|
|
XdgShell *xdgShellStable = nullptr;
|
|
ShmPool *shm = nullptr;
|
|
Seat *seat = nullptr;
|
|
PlasmaShell *plasmaShell = nullptr;
|
|
PlasmaWindowManagement *windowManagement = nullptr;
|
|
PointerConstraints *pointerConstraints = nullptr;
|
|
Registry *registry = nullptr;
|
|
QThread *thread = nullptr;
|
|
QVector<Output*> outputs;
|
|
IdleInhibitManager *idleInhibit = nullptr;
|
|
AppMenuManager *appMenu = nullptr;
|
|
} s_waylandConnection;
|
|
|
|
bool setupWaylandConnection(AdditionalWaylandInterfaces flags)
|
|
{
|
|
if (s_waylandConnection.connection) {
|
|
return false;
|
|
}
|
|
|
|
int sx[2];
|
|
if (socketpair(AF_UNIX, SOCK_STREAM | SOCK_CLOEXEC, 0, sx) < 0) {
|
|
return false;
|
|
}
|
|
KWin::waylandServer()->display()->createClient(sx[0]);
|
|
// setup connection
|
|
s_waylandConnection.connection = new ConnectionThread;
|
|
QSignalSpy connectedSpy(s_waylandConnection.connection, &ConnectionThread::connected);
|
|
if (!connectedSpy.isValid()) {
|
|
return false;
|
|
}
|
|
s_waylandConnection.connection->setSocketFd(sx[1]);
|
|
|
|
s_waylandConnection.thread = new QThread(kwinApp());
|
|
s_waylandConnection.connection->moveToThread(s_waylandConnection.thread);
|
|
s_waylandConnection.thread->start();
|
|
|
|
s_waylandConnection.connection->initConnection();
|
|
if (!connectedSpy.wait()) {
|
|
return false;
|
|
}
|
|
|
|
s_waylandConnection.queue = new EventQueue;
|
|
s_waylandConnection.queue->setup(s_waylandConnection.connection);
|
|
if (!s_waylandConnection.queue->isValid()) {
|
|
return false;
|
|
}
|
|
|
|
Registry *registry = new Registry;
|
|
s_waylandConnection.registry = registry;
|
|
registry->setEventQueue(s_waylandConnection.queue);
|
|
|
|
QObject::connect(registry, &Registry::outputAnnounced, [=](quint32 name, quint32 version) {
|
|
auto output = registry->createOutput(name, version, s_waylandConnection.registry);
|
|
s_waylandConnection.outputs << output;
|
|
QObject::connect(output, &Output::removed, [=]() {
|
|
output->deleteLater();
|
|
s_waylandConnection.outputs.removeOne(output);
|
|
});
|
|
});
|
|
|
|
QSignalSpy allAnnounced(registry, &Registry::interfacesAnnounced);
|
|
if (!allAnnounced.isValid()) {
|
|
return false;
|
|
}
|
|
registry->create(s_waylandConnection.connection);
|
|
if (!registry->isValid()) {
|
|
return false;
|
|
}
|
|
registry->setup();
|
|
if (!allAnnounced.wait()) {
|
|
return false;
|
|
}
|
|
|
|
s_waylandConnection.compositor = registry->createCompositor(registry->interface(Registry::Interface::Compositor).name, registry->interface(Registry::Interface::Compositor).version);
|
|
if (!s_waylandConnection.compositor->isValid()) {
|
|
return false;
|
|
}
|
|
s_waylandConnection.shm = registry->createShmPool(registry->interface(Registry::Interface::Shm).name, registry->interface(Registry::Interface::Shm).version);
|
|
if (!s_waylandConnection.shm->isValid()) {
|
|
return false;
|
|
}
|
|
s_waylandConnection.shell = registry->createShell(registry->interface(Registry::Interface::Shell).name, registry->interface(Registry::Interface::Shell).version);
|
|
if (!s_waylandConnection.shell->isValid()) {
|
|
return false;
|
|
}
|
|
s_waylandConnection.xdgShellV5 = registry->createXdgShell(registry->interface(Registry::Interface::XdgShellUnstableV5).name, registry->interface(Registry::Interface::XdgShellUnstableV5).version);
|
|
if (!s_waylandConnection.xdgShellV5->isValid()) {
|
|
return false;
|
|
}
|
|
s_waylandConnection.xdgShellV6 = registry->createXdgShell(registry->interface(Registry::Interface::XdgShellUnstableV6).name, registry->interface(Registry::Interface::XdgShellUnstableV6).version);
|
|
if (!s_waylandConnection.xdgShellV6->isValid()) {
|
|
return false;
|
|
}
|
|
s_waylandConnection.xdgShellStable = registry->createXdgShell(registry->interface(Registry::Interface::XdgShellStable).name, registry->interface(Registry::Interface::XdgShellStable).version);
|
|
if (!s_waylandConnection.xdgShellStable->isValid()) {
|
|
return false;
|
|
}
|
|
if (flags.testFlag(AdditionalWaylandInterface::Seat)) {
|
|
s_waylandConnection.seat = registry->createSeat(registry->interface(Registry::Interface::Seat).name, registry->interface(Registry::Interface::Seat).version);
|
|
if (!s_waylandConnection.seat->isValid()) {
|
|
return false;
|
|
}
|
|
}
|
|
if (flags.testFlag(AdditionalWaylandInterface::ShadowManager)) {
|
|
s_waylandConnection.shadowManager = registry->createShadowManager(registry->interface(Registry::Interface::Shadow).name,
|
|
registry->interface(Registry::Interface::Shadow).version);
|
|
if (!s_waylandConnection.shadowManager->isValid()) {
|
|
return false;
|
|
}
|
|
}
|
|
if (flags.testFlag(AdditionalWaylandInterface::Decoration)) {
|
|
s_waylandConnection.decoration = registry->createServerSideDecorationManager(registry->interface(Registry::Interface::ServerSideDecorationManager).name,
|
|
registry->interface(Registry::Interface::ServerSideDecorationManager).version);
|
|
if (!s_waylandConnection.decoration->isValid()) {
|
|
return false;
|
|
}
|
|
}
|
|
if (flags.testFlag(AdditionalWaylandInterface::PlasmaShell)) {
|
|
s_waylandConnection.plasmaShell = registry->createPlasmaShell(registry->interface(Registry::Interface::PlasmaShell).name,
|
|
registry->interface(Registry::Interface::PlasmaShell).version);
|
|
if (!s_waylandConnection.plasmaShell->isValid()) {
|
|
return false;
|
|
}
|
|
}
|
|
if (flags.testFlag(AdditionalWaylandInterface::WindowManagement)) {
|
|
s_waylandConnection.windowManagement = registry->createPlasmaWindowManagement(registry->interface(Registry::Interface::PlasmaWindowManagement).name,
|
|
registry->interface(Registry::Interface::PlasmaWindowManagement).version);
|
|
if (!s_waylandConnection.windowManagement->isValid()) {
|
|
return false;
|
|
}
|
|
}
|
|
if (flags.testFlag(AdditionalWaylandInterface::PointerConstraints)) {
|
|
s_waylandConnection.pointerConstraints = registry->createPointerConstraints(registry->interface(Registry::Interface::PointerConstraintsUnstableV1).name,
|
|
registry->interface(Registry::Interface::PointerConstraintsUnstableV1).version);
|
|
if (!s_waylandConnection.pointerConstraints->isValid()) {
|
|
return false;
|
|
}
|
|
}
|
|
if (flags.testFlag(AdditionalWaylandInterface::IdleInhibition)) {
|
|
s_waylandConnection.idleInhibit = registry->createIdleInhibitManager(registry->interface(Registry::Interface::IdleInhibitManagerUnstableV1).name,
|
|
registry->interface(Registry::Interface::IdleInhibitManagerUnstableV1).version);
|
|
if (!s_waylandConnection.idleInhibit->isValid()) {
|
|
return false;
|
|
}
|
|
}
|
|
if (flags.testFlag(AdditionalWaylandInterface::AppMenu)) {
|
|
s_waylandConnection.appMenu = registry->createAppMenuManager(registry->interface(Registry::Interface::AppMenu).name, registry->interface(Registry::Interface::AppMenu).version);
|
|
if (!s_waylandConnection.appMenu->isValid()) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void destroyWaylandConnection()
|
|
{
|
|
delete s_waylandConnection.compositor;
|
|
s_waylandConnection.compositor = nullptr;
|
|
delete s_waylandConnection.windowManagement;
|
|
s_waylandConnection.windowManagement = nullptr;
|
|
delete s_waylandConnection.plasmaShell;
|
|
s_waylandConnection.plasmaShell = nullptr;
|
|
delete s_waylandConnection.decoration;
|
|
s_waylandConnection.decoration = nullptr;
|
|
delete s_waylandConnection.decoration;
|
|
s_waylandConnection.decoration = nullptr;
|
|
delete s_waylandConnection.seat;
|
|
s_waylandConnection.seat = nullptr;
|
|
delete s_waylandConnection.pointerConstraints;
|
|
s_waylandConnection.pointerConstraints = nullptr;
|
|
delete s_waylandConnection.xdgShellV5;
|
|
s_waylandConnection.xdgShellV5 = nullptr;
|
|
delete s_waylandConnection.xdgShellV6;
|
|
s_waylandConnection.xdgShellV6 = nullptr;
|
|
delete s_waylandConnection.xdgShellStable;
|
|
s_waylandConnection.xdgShellStable = nullptr;
|
|
delete s_waylandConnection.shell;
|
|
s_waylandConnection.shell = nullptr;
|
|
delete s_waylandConnection.shadowManager;
|
|
s_waylandConnection.shadowManager = nullptr;
|
|
delete s_waylandConnection.idleInhibit;
|
|
s_waylandConnection.idleInhibit = nullptr;
|
|
delete s_waylandConnection.shm;
|
|
s_waylandConnection.shm = nullptr;
|
|
delete s_waylandConnection.queue;
|
|
s_waylandConnection.queue = nullptr;
|
|
delete s_waylandConnection.registry;
|
|
s_waylandConnection.registry = nullptr;
|
|
delete s_waylandConnection.appMenu;
|
|
s_waylandConnection.appMenu = nullptr;
|
|
if (s_waylandConnection.thread) {
|
|
QSignalSpy spy(s_waylandConnection.connection, &QObject::destroyed);
|
|
s_waylandConnection.connection->deleteLater();
|
|
if (spy.isEmpty()) {
|
|
QVERIFY(spy.wait());
|
|
}
|
|
s_waylandConnection.thread->quit();
|
|
s_waylandConnection.thread->wait();
|
|
delete s_waylandConnection.thread;
|
|
s_waylandConnection.thread = nullptr;
|
|
s_waylandConnection.connection = nullptr;
|
|
}
|
|
}
|
|
|
|
ConnectionThread *waylandConnection()
|
|
{
|
|
return s_waylandConnection.connection;
|
|
}
|
|
|
|
Compositor *waylandCompositor()
|
|
{
|
|
return s_waylandConnection.compositor;
|
|
}
|
|
|
|
ShadowManager *waylandShadowManager()
|
|
{
|
|
return s_waylandConnection.shadowManager;
|
|
}
|
|
|
|
Shell *waylandShell()
|
|
{
|
|
return s_waylandConnection.shell;
|
|
}
|
|
|
|
ShmPool *waylandShmPool()
|
|
{
|
|
return s_waylandConnection.shm;
|
|
}
|
|
|
|
Seat *waylandSeat()
|
|
{
|
|
return s_waylandConnection.seat;
|
|
}
|
|
|
|
ServerSideDecorationManager *waylandServerSideDecoration()
|
|
{
|
|
return s_waylandConnection.decoration;
|
|
}
|
|
|
|
PlasmaShell *waylandPlasmaShell()
|
|
{
|
|
return s_waylandConnection.plasmaShell;
|
|
}
|
|
|
|
PlasmaWindowManagement *waylandWindowManagement()
|
|
{
|
|
return s_waylandConnection.windowManagement;
|
|
}
|
|
|
|
PointerConstraints *waylandPointerConstraints()
|
|
{
|
|
return s_waylandConnection.pointerConstraints;
|
|
}
|
|
|
|
IdleInhibitManager *waylandIdleInhibitManager()
|
|
{
|
|
return s_waylandConnection.idleInhibit;
|
|
}
|
|
|
|
AppMenuManager* waylandAppMenuManager()
|
|
{
|
|
return s_waylandConnection.appMenu;
|
|
}
|
|
|
|
bool waitForWaylandPointer()
|
|
{
|
|
if (!s_waylandConnection.seat) {
|
|
return false;
|
|
}
|
|
QSignalSpy hasPointerSpy(s_waylandConnection.seat, &Seat::hasPointerChanged);
|
|
if (!hasPointerSpy.isValid()) {
|
|
return false;
|
|
}
|
|
return hasPointerSpy.wait();
|
|
}
|
|
|
|
bool waitForWaylandTouch()
|
|
{
|
|
if (!s_waylandConnection.seat) {
|
|
return false;
|
|
}
|
|
QSignalSpy hasTouchSpy(s_waylandConnection.seat, &Seat::hasTouchChanged);
|
|
if (!hasTouchSpy.isValid()) {
|
|
return false;
|
|
}
|
|
return hasTouchSpy.wait();
|
|
}
|
|
|
|
bool waitForWaylandKeyboard()
|
|
{
|
|
if (!s_waylandConnection.seat) {
|
|
return false;
|
|
}
|
|
QSignalSpy hasKeyboardSpy(s_waylandConnection.seat, &Seat::hasKeyboardChanged);
|
|
if (!hasKeyboardSpy.isValid()) {
|
|
return false;
|
|
}
|
|
return hasKeyboardSpy.wait();
|
|
}
|
|
|
|
void render(Surface *surface, const QSize &size, const QColor &color, const QImage::Format &format)
|
|
{
|
|
QImage img(size, format);
|
|
img.fill(color);
|
|
render(surface, img);
|
|
}
|
|
|
|
void render(Surface *surface, const QImage &img)
|
|
{
|
|
surface->attachBuffer(s_waylandConnection.shm->createBuffer(img));
|
|
surface->damage(QRect(QPoint(0, 0), img.size()));
|
|
surface->commit(Surface::CommitFlag::None);
|
|
}
|
|
|
|
ShellClient *waitForWaylandWindowShown(int timeout)
|
|
{
|
|
QSignalSpy clientAddedSpy(waylandServer(), &WaylandServer::shellClientAdded);
|
|
if (!clientAddedSpy.isValid()) {
|
|
return nullptr;
|
|
}
|
|
if (!clientAddedSpy.wait(timeout)) {
|
|
return nullptr;
|
|
}
|
|
return clientAddedSpy.first().first().value<ShellClient*>();
|
|
}
|
|
|
|
ShellClient *renderAndWaitForShown(Surface *surface, const QSize &size, const QColor &color, const QImage::Format &format, int timeout)
|
|
{
|
|
QSignalSpy clientAddedSpy(waylandServer(), &WaylandServer::shellClientAdded);
|
|
if (!clientAddedSpy.isValid()) {
|
|
return nullptr;
|
|
}
|
|
render(surface, size, color, format);
|
|
flushWaylandConnection();
|
|
if (!clientAddedSpy.wait(timeout)) {
|
|
return nullptr;
|
|
}
|
|
return clientAddedSpy.first().first().value<ShellClient*>();
|
|
}
|
|
|
|
void flushWaylandConnection()
|
|
{
|
|
if (s_waylandConnection.connection) {
|
|
s_waylandConnection.connection->flush();
|
|
}
|
|
}
|
|
|
|
Surface *createSurface(QObject *parent)
|
|
{
|
|
if (!s_waylandConnection.compositor) {
|
|
return nullptr;
|
|
}
|
|
auto s = s_waylandConnection.compositor->createSurface(parent);
|
|
if (!s->isValid()) {
|
|
delete s;
|
|
return nullptr;
|
|
}
|
|
return s;
|
|
}
|
|
|
|
ShellSurface *createShellSurface(Surface *surface, QObject *parent)
|
|
{
|
|
if (!s_waylandConnection.shell) {
|
|
return nullptr;
|
|
}
|
|
auto s = s_waylandConnection.shell->createSurface(surface, parent);
|
|
if (!s->isValid()) {
|
|
delete s;
|
|
return nullptr;
|
|
}
|
|
return s;
|
|
}
|
|
|
|
XdgShellSurface *createXdgShellV5Surface(Surface *surface, QObject *parent)
|
|
{
|
|
if (!s_waylandConnection.xdgShellV5) {
|
|
return nullptr;
|
|
}
|
|
auto s = s_waylandConnection.xdgShellV5->createSurface(surface, parent);
|
|
if (!s->isValid()) {
|
|
delete s;
|
|
return nullptr;
|
|
}
|
|
return s;
|
|
}
|
|
|
|
XdgShellSurface *createXdgShellV6Surface(Surface *surface, QObject *parent)
|
|
{
|
|
if (!s_waylandConnection.xdgShellV6) {
|
|
return nullptr;
|
|
}
|
|
auto s = s_waylandConnection.xdgShellV6->createSurface(surface, parent);
|
|
if (!s->isValid()) {
|
|
delete s;
|
|
return nullptr;
|
|
}
|
|
return s;
|
|
}
|
|
|
|
XdgShellSurface *createXdgShellStableSurface(Surface *surface, QObject *parent)
|
|
{
|
|
if (!s_waylandConnection.xdgShellStable) {
|
|
return nullptr;
|
|
}
|
|
auto s = s_waylandConnection.xdgShellStable->createSurface(surface, parent);
|
|
if (!s->isValid()) {
|
|
delete s;
|
|
return nullptr;
|
|
}
|
|
return s;
|
|
}
|
|
|
|
QObject *createShellSurface(ShellSurfaceType type, KWayland::Client::Surface *surface, QObject *parent)
|
|
{
|
|
switch (type) {
|
|
case ShellSurfaceType::WlShell:
|
|
return createShellSurface(surface, parent);
|
|
case ShellSurfaceType::XdgShellV5:
|
|
return createXdgShellV5Surface(surface, parent);
|
|
case ShellSurfaceType::XdgShellV6:
|
|
return createXdgShellV6Surface(surface, parent);
|
|
case ShellSurfaceType::XdgShellStable:
|
|
return createXdgShellStableSurface(surface, parent);
|
|
default:
|
|
Q_UNREACHABLE();
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
bool waitForWindowDestroyed(AbstractClient *client)
|
|
{
|
|
QSignalSpy destroyedSpy(client, &QObject::destroyed);
|
|
if (!destroyedSpy.isValid()) {
|
|
return false;
|
|
}
|
|
return destroyedSpy.wait();
|
|
}
|
|
|
|
bool lockScreen()
|
|
{
|
|
if (waylandServer()->isScreenLocked()) {
|
|
return false;
|
|
}
|
|
QSignalSpy lockStateChangedSpy(ScreenLocker::KSldApp::self(), &ScreenLocker::KSldApp::lockStateChanged);
|
|
if (!lockStateChangedSpy.isValid()) {
|
|
return false;
|
|
}
|
|
ScreenLocker::KSldApp::self()->lock(ScreenLocker::EstablishLock::Immediate);
|
|
if (lockStateChangedSpy.count() != 1) {
|
|
return false;
|
|
}
|
|
if (!waylandServer()->isScreenLocked()) {
|
|
return false;
|
|
}
|
|
if (!ScreenLockerWatcher::self()->isLocked()) {
|
|
QSignalSpy lockedSpy(ScreenLockerWatcher::self(), &ScreenLockerWatcher::locked);
|
|
if (!lockedSpy.isValid()) {
|
|
return false;
|
|
}
|
|
if (!lockedSpy.wait()) {
|
|
return false;
|
|
}
|
|
if (!ScreenLockerWatcher::self()->isLocked()) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool unlockScreen()
|
|
{
|
|
QSignalSpy lockStateChangedSpy(ScreenLocker::KSldApp::self(), &ScreenLocker::KSldApp::lockStateChanged);
|
|
if (!lockStateChangedSpy.isValid()) {
|
|
return false;
|
|
}
|
|
using namespace ScreenLocker;
|
|
const auto children = KSldApp::self()->children();
|
|
for (auto it = children.begin(); it != children.end(); ++it) {
|
|
if (qstrcmp((*it)->metaObject()->className(), "LogindIntegration") != 0) {
|
|
continue;
|
|
}
|
|
QMetaObject::invokeMethod(*it, "requestUnlock");
|
|
break;
|
|
}
|
|
if (waylandServer()->isScreenLocked()) {
|
|
lockStateChangedSpy.wait();
|
|
}
|
|
if (waylandServer()->isScreenLocked()) {
|
|
return true;
|
|
}
|
|
if (ScreenLockerWatcher::self()->isLocked()) {
|
|
QSignalSpy lockedSpy(ScreenLockerWatcher::self(), &ScreenLockerWatcher::locked);
|
|
if (!lockedSpy.isValid()) {
|
|
return false;
|
|
}
|
|
if (!lockedSpy.wait()) {
|
|
return false;
|
|
}
|
|
if (ScreenLockerWatcher::self()->isLocked()) {
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
}
|
|
}
|