[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:
parent
c09f4039d1
commit
1998d5ac1a
8 changed files with 16 additions and 11 deletions
|
@ -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()
|
||||||
|
|
4
main.cpp
4
main.cpp
|
@ -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();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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()
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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)
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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();
|
||||||
|
|
Loading…
Reference in a new issue