[wayland] Improve tear-down to not crash if X11 applications are still around

We need to destroy the compositor after Xwayland terminated and after
the internal Wayland connection is destroyed. This means when destroying
the Workspace we may no longer destroy the Compositor at the same time.
Also we need to ensure that other tear down functionality doesn't call
into the no longer existing internal client connection.

With this change kwin doesn't crash when exiting with Wayland and/or
X11 windows still open.
This commit is contained in:
Martin Gräßlin 2015-11-10 08:52:40 +01:00
parent c09f4039d1
commit 1998d5ac1a
8 changed files with 16 additions and 11 deletions

View file

@ -50,8 +50,8 @@ WaylandTestApplication::WaylandTestApplication(int &argc, char **argv)
WaylandTestApplication::~WaylandTestApplication() WaylandTestApplication::~WaylandTestApplication()
{ {
destroyCompositor();
destroyWorkspace(); destroyWorkspace();
waylandServer()->dispatch();
if (x11Connection()) { if (x11Connection()) {
Xcb::setInputFocus(XCB_INPUT_FOCUS_POINTER_ROOT); Xcb::setInputFocus(XCB_INPUT_FOCUS_POINTER_ROOT);
destroyAtoms(); destroyAtoms();
@ -61,6 +61,8 @@ WaylandTestApplication::~WaylandTestApplication()
m_xwaylandProcess->terminate(); m_xwaylandProcess->terminate();
m_xwaylandProcess->waitForFinished(); m_xwaylandProcess->waitForFinished();
} }
waylandServer()->destroyInternalConnection();
destroyCompositor();
} }
void WaylandTestApplication::performStartup() void WaylandTestApplication::performStartup()

View file

@ -418,10 +418,6 @@ void Application::destroyWorkspace()
void Application::destroyCompositor() void Application::destroyCompositor()
{ {
if (Workspace::self()) {
// compositor is destroyed together with Workspace
return;
}
delete Compositor::self(); delete Compositor::self();
} }

View file

@ -74,8 +74,8 @@ ApplicationWayland::ApplicationWayland(int &argc, char **argv)
ApplicationWayland::~ApplicationWayland() ApplicationWayland::~ApplicationWayland()
{ {
destroyCompositor();
destroyWorkspace(); destroyWorkspace();
waylandServer()->dispatch();
if (x11Connection()) { if (x11Connection()) {
Xcb::setInputFocus(XCB_INPUT_FOCUS_POINTER_ROOT); Xcb::setInputFocus(XCB_INPUT_FOCUS_POINTER_ROOT);
destroyAtoms(); destroyAtoms();
@ -85,6 +85,8 @@ ApplicationWayland::~ApplicationWayland()
m_xwaylandProcess->terminate(); m_xwaylandProcess->terminate();
m_xwaylandProcess->waitForFinished(); m_xwaylandProcess->waitForFinished();
} }
waylandServer()->destroyInternalConnection();
destroyCompositor();
} }
void ApplicationWayland::performStartup() void ApplicationWayland::performStartup()

View file

@ -121,6 +121,7 @@ ApplicationX11::ApplicationX11(int &argc, char **argv)
ApplicationX11::~ApplicationX11() ApplicationX11::~ApplicationX11()
{ {
destroyCompositor();
destroyWorkspace(); destroyWorkspace();
if (!owner.isNull() && owner->ownerWindow() != XCB_WINDOW_NONE) // If there was no --replace (no new WM) if (!owner.isNull() && owner->ownerWindow() != XCB_WINDOW_NONE) // If there was no --replace (no new WM)
Xcb::setInputFocus(XCB_INPUT_FOCUS_POINTER_ROOT); Xcb::setInputFocus(XCB_INPUT_FOCUS_POINTER_ROOT);
@ -134,6 +135,7 @@ void ApplicationX11::setReplace(bool replace)
void ApplicationX11::lostSelection() void ApplicationX11::lostSelection()
{ {
sendPostedEvents(); sendPostedEvents();
destroyCompositor();
destroyWorkspace(); destroyWorkspace();
// Remove windowmanager privileges // Remove windowmanager privileges
Xcb::selectInput(rootWindow(), XCB_EVENT_MASK_PROPERTY_CHANGE); Xcb::selectInput(rootWindow(), XCB_EVENT_MASK_PROPERTY_CHANGE);

View file

@ -57,7 +57,6 @@ Window::~Window()
} }
delete m_shellSurface; delete m_shellSurface;
delete m_surface; delete m_surface;
waylandServer()->internalClientConection()->flush();
} }
WId Window::winId() const WId Window::winId() const
@ -106,7 +105,9 @@ void Window::unmap()
} }
m_surface->attachBuffer(KWayland::Client::Buffer::Ptr()); m_surface->attachBuffer(KWayland::Client::Buffer::Ptr());
m_surface->commit(KWayland::Client::Surface::CommitFlag::None); m_surface->commit(KWayland::Client::Surface::CommitFlag::None);
waylandServer()->internalClientConection()->flush(); if (waylandServer()->internalClientConection()) {
waylandServer()->internalClientConection()->flush();
}
} }
void Window::createEglSurface(EGLDisplay dpy, EGLConfig config) void Window::createEglSurface(EGLDisplay dpy, EGLConfig config)

View file

@ -70,13 +70,16 @@ WaylandServer::WaylandServer(QObject *parent)
qRegisterMetaType<KWayland::Server::OutputInterface::DpmsMode>(); qRegisterMetaType<KWayland::Server::OutputInterface::DpmsMode>();
} }
WaylandServer::~WaylandServer() WaylandServer::~WaylandServer() = default;
void WaylandServer::destroyInternalConnection()
{ {
if (m_internalConnection.client) { if (m_internalConnection.client) {
dispatch(); dispatch();
m_internalConnection.client->deleteLater(); m_internalConnection.client->deleteLater();
m_internalConnection.clientThread->quit(); m_internalConnection.clientThread->quit();
m_internalConnection.clientThread->wait(); m_internalConnection.clientThread->wait();
m_internalConnection.client = nullptr;
} }
} }

View file

@ -116,6 +116,7 @@ public:
int createInputMethodConnection(); int createInputMethodConnection();
void createInternalConnection(); void createInternalConnection();
void destroyInternalConnection();
void initWorkspace(); void initWorkspace();
KWayland::Server::ClientConnection *xWaylandConnection() const { KWayland::Server::ClientConnection *xWaylandConnection() const {

View file

@ -425,8 +425,6 @@ void Workspace::init()
Workspace::~Workspace() Workspace::~Workspace()
{ {
delete m_compositor;
m_compositor = NULL;
blockStackingUpdates(true); blockStackingUpdates(true);
// TODO: grabXServer(); // TODO: grabXServer();