[autotests] Sub-surface resize test
Summary: Adds an autotest to show that KWin fails an assertion when a client tries to resize a sub-surface. Since it is the first autotest dealing with sub-surfaces explicitly additional autotest helpers are introduced to allow that. We also add a new signal in Compositor to spy on to know when the buffer swap has been completed. Test Plan: Test fails as expected: ``` QFATAL : KWin::BufferSizeChangeTest::testShmBufferSizeChangeOnSubSurface() ASSERT: "image.size() == m_size" in file /home/roman/dev/kde/src/kde/workspace/kwin/platformsupport/scenes/opengl/abstract_egl_backend.cpp, line 394 FAIL! : KWin::BufferSizeChangeTest::testShmBufferSizeChangeOnSubSurface() Received a fatal error. Loc: [Unknown file(0)] Totals: 4 passed, 1 failed, 0 skipped, 0 blacklisted, 367ms ********* Finished testing of KWin::BufferSizeChangeTest ********* ``` Reviewers: #kwin, zzag Subscribers: zzag, graesslin, kwin Tags: #kwin Differential Revision: https://phabricator.kde.org/D18452
This commit is contained in:
parent
1e6f6700f8
commit
06f64d5e56
6 changed files with 164 additions and 0 deletions
|
@ -59,6 +59,7 @@ integrationTest(WAYLAND_ONLY NAME testColorCorrectNightColor SRCS colorcorrect_n
|
||||||
integrationTest(WAYLAND_ONLY NAME testDontCrashCursorPhysicalSizeEmpty SRCS dont_crash_cursor_physical_size_empty.cpp)
|
integrationTest(WAYLAND_ONLY NAME testDontCrashCursorPhysicalSizeEmpty SRCS dont_crash_cursor_physical_size_empty.cpp)
|
||||||
integrationTest(WAYLAND_ONLY NAME testDontCrashReinitializeCompositor SRCS dont_crash_reinitialize_compositor.cpp)
|
integrationTest(WAYLAND_ONLY NAME testDontCrashReinitializeCompositor SRCS dont_crash_reinitialize_compositor.cpp)
|
||||||
integrationTest(WAYLAND_ONLY NAME testNoGlobalShortcuts SRCS no_global_shortcuts_test.cpp)
|
integrationTest(WAYLAND_ONLY NAME testNoGlobalShortcuts SRCS no_global_shortcuts_test.cpp)
|
||||||
|
integrationTest(WAYLAND_ONLY NAME testBufferSizeChange SRCS buffer_size_change_test.cpp generic_scene_opengl_test.cpp)
|
||||||
|
|
||||||
if (XCB_ICCCM_FOUND)
|
if (XCB_ICCCM_FOUND)
|
||||||
integrationTest(NAME testMoveResize SRCS move_resize_window_test.cpp LIBS XCB::ICCCM)
|
integrationTest(NAME testMoveResize SRCS move_resize_window_test.cpp LIBS XCB::ICCCM)
|
||||||
|
|
128
autotests/integration/buffer_size_change_test.cpp
Normal file
128
autotests/integration/buffer_size_change_test.cpp
Normal file
|
@ -0,0 +1,128 @@
|
||||||
|
/********************************************************************
|
||||||
|
KWin - the KDE window manager
|
||||||
|
This file is part of the KDE project.
|
||||||
|
|
||||||
|
Copyright 2019 Roman Gilg <subdiff@gmail.com>
|
||||||
|
|
||||||
|
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 "generic_scene_opengl_test.h"
|
||||||
|
|
||||||
|
#include "composite.h"
|
||||||
|
#include "shell_client.h"
|
||||||
|
#include "wayland_server.h"
|
||||||
|
|
||||||
|
#include <KWayland/Client/xdgshell.h>
|
||||||
|
#include <KWayland/Client/subsurface.h>
|
||||||
|
#include <KWayland/Client/surface.h>
|
||||||
|
|
||||||
|
namespace KWin
|
||||||
|
{
|
||||||
|
|
||||||
|
static const QString s_socketName = QStringLiteral("wayland_test_buffer_size_change-0");
|
||||||
|
|
||||||
|
class BufferSizeChangeTest : public GenericSceneOpenGLTest
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
BufferSizeChangeTest() : GenericSceneOpenGLTest(QByteArrayLiteral("O2")) {}
|
||||||
|
private Q_SLOTS:
|
||||||
|
void init();
|
||||||
|
void testShmBufferSizeChange();
|
||||||
|
void testShmBufferSizeChangeOnSubSurface();
|
||||||
|
};
|
||||||
|
|
||||||
|
void BufferSizeChangeTest::init()
|
||||||
|
{
|
||||||
|
QVERIFY(Test::setupWaylandConnection());
|
||||||
|
}
|
||||||
|
|
||||||
|
void BufferSizeChangeTest::testShmBufferSizeChange()
|
||||||
|
{
|
||||||
|
// This test verifies that an SHM buffer size change is handled correctly
|
||||||
|
|
||||||
|
using namespace KWayland::Client;
|
||||||
|
|
||||||
|
QScopedPointer<Surface> surface(Test::createSurface());
|
||||||
|
QVERIFY(!surface.isNull());
|
||||||
|
|
||||||
|
QScopedPointer<XdgShellSurface> shellSurface(Test::createXdgShellStableSurface(surface.data()));
|
||||||
|
QVERIFY(!shellSurface.isNull());
|
||||||
|
Test::flushWaylandConnection();
|
||||||
|
|
||||||
|
// set buffer size
|
||||||
|
ShellClient *client = Test::renderAndWaitForShown(surface.data(), QSize(100, 50), Qt::blue);
|
||||||
|
QVERIFY(client);
|
||||||
|
|
||||||
|
// add a first repaint
|
||||||
|
QSignalSpy swapSpy(Compositor::self(), &Compositor::bufferSwapCompleted);
|
||||||
|
QVERIFY(swapSpy.isValid());
|
||||||
|
Compositor::self()->addRepaintFull();
|
||||||
|
QVERIFY(swapSpy.wait());
|
||||||
|
|
||||||
|
// now change buffer size
|
||||||
|
Test::render(surface.data(), QSize(30, 10), Qt::red);
|
||||||
|
Test::flushWaylandConnection();
|
||||||
|
|
||||||
|
QSignalSpy damagedSpy(client, &ShellClient::damaged);
|
||||||
|
QVERIFY(damagedSpy.isValid());
|
||||||
|
QVERIFY(damagedSpy.wait());
|
||||||
|
KWin::Compositor::self()->addRepaintFull();
|
||||||
|
QVERIFY(swapSpy.wait());
|
||||||
|
}
|
||||||
|
|
||||||
|
void BufferSizeChangeTest::testShmBufferSizeChangeOnSubSurface()
|
||||||
|
{
|
||||||
|
using namespace KWayland::Client;
|
||||||
|
|
||||||
|
// setup parent surface
|
||||||
|
QScopedPointer<Surface> parentSurface(Test::createSurface());
|
||||||
|
QVERIFY(!parentSurface.isNull());
|
||||||
|
QScopedPointer<XdgShellSurface> shellSurface(Test::createXdgShellStableSurface(parentSurface.data()));
|
||||||
|
QVERIFY(!shellSurface.isNull());
|
||||||
|
|
||||||
|
// setup sub surface
|
||||||
|
QScopedPointer<Surface> surface(Test::createSurface());
|
||||||
|
QVERIFY(!surface.isNull());
|
||||||
|
QScopedPointer<SubSurface> subSurface(Test::createSubSurface(surface.data(), parentSurface.data()));
|
||||||
|
QVERIFY(!subSurface.isNull());
|
||||||
|
|
||||||
|
// set buffer sizes
|
||||||
|
Test::render(surface.data(), QSize(30, 10), Qt::red);
|
||||||
|
ShellClient *parent = Test::renderAndWaitForShown(parentSurface.data(), QSize(100, 50), Qt::blue);
|
||||||
|
QVERIFY(parent);
|
||||||
|
|
||||||
|
// add a first repaint
|
||||||
|
QSignalSpy swapSpy(Compositor::self(), &Compositor::bufferSwapCompleted);
|
||||||
|
QVERIFY(swapSpy.isValid());
|
||||||
|
Compositor::self()->addRepaintFull();
|
||||||
|
QVERIFY(swapSpy.wait());
|
||||||
|
|
||||||
|
// change buffer size of sub surface
|
||||||
|
QSignalSpy damagedParentSpy(parent, &ShellClient::damaged);
|
||||||
|
QVERIFY(damagedParentSpy.isValid());
|
||||||
|
Test::render(surface.data(), QSize(20, 10), Qt::red);
|
||||||
|
parentSurface->commit(Surface::CommitFlag::None);
|
||||||
|
|
||||||
|
QVERIFY(damagedParentSpy.wait());
|
||||||
|
|
||||||
|
// add a second repaint
|
||||||
|
KWin::Compositor::self()->addRepaintFull();
|
||||||
|
QVERIFY(swapSpy.wait());
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
WAYLANDTEST_MAIN(KWin::BufferSizeChangeTest)
|
||||||
|
#include "buffer_size_change_test.moc"
|
|
@ -45,6 +45,8 @@ class ShadowManager;
|
||||||
class Shell;
|
class Shell;
|
||||||
class ShellSurface;
|
class ShellSurface;
|
||||||
class ShmPool;
|
class ShmPool;
|
||||||
|
class SubCompositor;
|
||||||
|
class SubSurface;
|
||||||
class Surface;
|
class Surface;
|
||||||
class XdgDecorationManager;
|
class XdgDecorationManager;
|
||||||
}
|
}
|
||||||
|
@ -112,6 +114,7 @@ void destroyWaylandConnection();
|
||||||
|
|
||||||
KWayland::Client::ConnectionThread *waylandConnection();
|
KWayland::Client::ConnectionThread *waylandConnection();
|
||||||
KWayland::Client::Compositor *waylandCompositor();
|
KWayland::Client::Compositor *waylandCompositor();
|
||||||
|
KWayland::Client::SubCompositor *waylandSubCompositor();
|
||||||
KWayland::Client::ShadowManager *waylandShadowManager();
|
KWayland::Client::ShadowManager *waylandShadowManager();
|
||||||
KWayland::Client::Shell *waylandShell();
|
KWayland::Client::Shell *waylandShell();
|
||||||
KWayland::Client::ShmPool *waylandShmPool();
|
KWayland::Client::ShmPool *waylandShmPool();
|
||||||
|
@ -131,6 +134,8 @@ bool waitForWaylandKeyboard();
|
||||||
void flushWaylandConnection();
|
void flushWaylandConnection();
|
||||||
|
|
||||||
KWayland::Client::Surface *createSurface(QObject *parent = nullptr);
|
KWayland::Client::Surface *createSurface(QObject *parent = nullptr);
|
||||||
|
KWayland::Client::SubSurface *createSubSurface(KWayland::Client::Surface *surface,
|
||||||
|
KWayland::Client::Surface *parentSurface, QObject *parent = nullptr);
|
||||||
enum class ShellSurfaceType {
|
enum class ShellSurfaceType {
|
||||||
WlShell,
|
WlShell,
|
||||||
XdgShellV5,
|
XdgShellV5,
|
||||||
|
|
|
@ -36,6 +36,8 @@ along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
#include <KWayland/Client/shell.h>
|
#include <KWayland/Client/shell.h>
|
||||||
#include <KWayland/Client/shm_pool.h>
|
#include <KWayland/Client/shm_pool.h>
|
||||||
#include <KWayland/Client/output.h>
|
#include <KWayland/Client/output.h>
|
||||||
|
#include <KWayland/Client/subcompositor.h>
|
||||||
|
#include <KWayland/Client/subsurface.h>
|
||||||
#include <KWayland/Client/surface.h>
|
#include <KWayland/Client/surface.h>
|
||||||
#include <KWayland/Client/appmenu.h>
|
#include <KWayland/Client/appmenu.h>
|
||||||
#include <KWayland/Client/xdgshell.h>
|
#include <KWayland/Client/xdgshell.h>
|
||||||
|
@ -63,6 +65,7 @@ static struct {
|
||||||
ConnectionThread *connection = nullptr;
|
ConnectionThread *connection = nullptr;
|
||||||
EventQueue *queue = nullptr;
|
EventQueue *queue = nullptr;
|
||||||
Compositor *compositor = nullptr;
|
Compositor *compositor = nullptr;
|
||||||
|
SubCompositor *subCompositor = nullptr;
|
||||||
ServerSideDecorationManager *decoration = nullptr;
|
ServerSideDecorationManager *decoration = nullptr;
|
||||||
ShadowManager *shadowManager = nullptr;
|
ShadowManager *shadowManager = nullptr;
|
||||||
Shell *shell = nullptr;
|
Shell *shell = nullptr;
|
||||||
|
@ -146,6 +149,10 @@ bool setupWaylandConnection(AdditionalWaylandInterfaces flags)
|
||||||
if (!s_waylandConnection.compositor->isValid()) {
|
if (!s_waylandConnection.compositor->isValid()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
s_waylandConnection.subCompositor = registry->createSubCompositor(registry->interface(Registry::Interface::SubCompositor).name, registry->interface(Registry::Interface::SubCompositor).version);
|
||||||
|
if (!s_waylandConnection.subCompositor->isValid()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
s_waylandConnection.shm = registry->createShmPool(registry->interface(Registry::Interface::Shm).name, registry->interface(Registry::Interface::Shm).version);
|
s_waylandConnection.shm = registry->createShmPool(registry->interface(Registry::Interface::Shm).name, registry->interface(Registry::Interface::Shm).version);
|
||||||
if (!s_waylandConnection.shm->isValid()) {
|
if (!s_waylandConnection.shm->isValid()) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -234,6 +241,8 @@ void destroyWaylandConnection()
|
||||||
{
|
{
|
||||||
delete s_waylandConnection.compositor;
|
delete s_waylandConnection.compositor;
|
||||||
s_waylandConnection.compositor = nullptr;
|
s_waylandConnection.compositor = nullptr;
|
||||||
|
delete s_waylandConnection.subCompositor;
|
||||||
|
s_waylandConnection.subCompositor = nullptr;
|
||||||
delete s_waylandConnection.windowManagement;
|
delete s_waylandConnection.windowManagement;
|
||||||
s_waylandConnection.windowManagement = nullptr;
|
s_waylandConnection.windowManagement = nullptr;
|
||||||
delete s_waylandConnection.plasmaShell;
|
delete s_waylandConnection.plasmaShell;
|
||||||
|
@ -292,6 +301,11 @@ Compositor *waylandCompositor()
|
||||||
return s_waylandConnection.compositor;
|
return s_waylandConnection.compositor;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SubCompositor *waylandSubCompositor()
|
||||||
|
{
|
||||||
|
return s_waylandConnection.subCompositor;
|
||||||
|
}
|
||||||
|
|
||||||
ShadowManager *waylandShadowManager()
|
ShadowManager *waylandShadowManager()
|
||||||
{
|
{
|
||||||
return s_waylandConnection.shadowManager;
|
return s_waylandConnection.shadowManager;
|
||||||
|
@ -444,6 +458,19 @@ Surface *createSurface(QObject *parent)
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SubSurface *createSubSurface(Surface *surface, Surface *parentSurface, QObject *parent)
|
||||||
|
{
|
||||||
|
if (!s_waylandConnection.subCompositor) {
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
auto s = s_waylandConnection.subCompositor->createSubSurface(surface, parentSurface, parent);
|
||||||
|
if (!s->isValid()) {
|
||||||
|
delete s;
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
return s;
|
||||||
|
}
|
||||||
|
|
||||||
ShellSurface *createShellSurface(Surface *surface, QObject *parent)
|
ShellSurface *createShellSurface(Surface *surface, QObject *parent)
|
||||||
{
|
{
|
||||||
if (!s_waylandConnection.shell) {
|
if (!s_waylandConnection.shell) {
|
||||||
|
|
|
@ -655,6 +655,8 @@ void Compositor::bufferSwapComplete()
|
||||||
assert(m_bufferSwapPending);
|
assert(m_bufferSwapPending);
|
||||||
m_bufferSwapPending = false;
|
m_bufferSwapPending = false;
|
||||||
|
|
||||||
|
emit bufferSwapCompleted();
|
||||||
|
|
||||||
if (m_composeAtSwapCompletion) {
|
if (m_composeAtSwapCompletion) {
|
||||||
m_composeAtSwapCompletion = false;
|
m_composeAtSwapCompletion = false;
|
||||||
performCompositing();
|
performCompositing();
|
||||||
|
|
|
@ -178,6 +178,7 @@ Q_SIGNALS:
|
||||||
void aboutToDestroy();
|
void aboutToDestroy();
|
||||||
void aboutToToggleCompositing();
|
void aboutToToggleCompositing();
|
||||||
void sceneCreated();
|
void sceneCreated();
|
||||||
|
void bufferSwapCompleted();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void timerEvent(QTimerEvent *te);
|
void timerEvent(QTimerEvent *te);
|
||||||
|
|
Loading…
Reference in a new issue