kwin/src/wayland/xdgforeign_v2_interface.cpp
Marco Martin 9feb359676 Wayland foreign protocol
Summary:
Implement the "foreign" wayland protocol.
A client can export a surface with an unique string as handle,
then another client can refer to that surface and set an own surface as
child of that surface.
Potential use cases are out-of-process dialogs, such as file dialogs,
meant to be used by sandboxed processes that may not have the access
it needs to implement such dialogs.
The handle needs to be shared between the processes with other means,
such as dbus or command line paramenters.

The public api of the server side only tracks parent/child relationships as this is the only data kwin would need it for, the rest of the api is not exported so should be safer from eventual protocol changes

Test Plan:
the autotest works, but has a lot of random crashes when deleting surfaces,
unfortunately backtraces don't tell much and the crashes never occur when running into valgrind
behavior may still be wrong, depending on how the protocol is supposed
to work if more clients try to set the same exported surface as parent

Reviewers: #plasma, #kwin, davidedmundson, graesslin

Reviewed By: #plasma, #kwin, graesslin

Subscribers: davidedmundson, graesslin, plasma-devel, #frameworks

Tags: #frameworks, #plasma_on_wayland

Differential Revision: https://phabricator.kde.org/D7369
2017-10-13 11:30:13 +02:00

459 lines
15 KiB
C++

/****************************************************************************
Copyright 2017 Marco Martin <notmart@gmail.com>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) version 3, or any
later version accepted by the membership of KDE e.V. (or its
successor approved by the membership of KDE e.V.), which shall
act as a proxy defined in Section 6 of version 3 of the license.
This library 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
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library. If not, see <http://www.gnu.org/licenses/>.
****************************************************************************/
#include "xdgforeign_interface.h"
#include "xdgforeign_v2_interface_p.h"
#include "display.h"
#include "global_p.h"
#include "resource_p.h"
#include "surface_interface_p.h"
#include "wayland-xdg-foreign-unstable-v2-server-protocol.h"
#include <QUuid>
#include <QDebug>
namespace KWayland
{
namespace Server
{
class Q_DECL_HIDDEN XdgExporterUnstableV2Interface::Private : public Global::Private
{
public:
Private(XdgExporterUnstableV2Interface *q, Display *d, XdgForeignInterface *foreignInterface);
XdgForeignInterface *foreignInterface;
QHash<QString, XdgExportedUnstableV2Interface *> exportedSurfaces;
private:
void bind(wl_client *client, uint32_t version, uint32_t id) override;
static void unbind(wl_resource *resource);
static Private *cast(wl_resource *r) {
return reinterpret_cast<Private*>(wl_resource_get_user_data(r));
}
static void destroyCallback(wl_client *client, wl_resource *resource);
static void exportCallback(wl_client *client, wl_resource *resource, uint32_t id, wl_resource * surface);
XdgExporterUnstableV2Interface *q;
static const struct zxdg_exporter_v2_interface s_interface;
static const quint32 s_version;
};
const quint32 XdgExporterUnstableV2Interface::Private::s_version = 1;
#ifndef DOXYGEN_SHOULD_SKIP_THIS
const struct zxdg_exporter_v2_interface XdgExporterUnstableV2Interface::Private::s_interface = {
destroyCallback,
exportCallback
};
#endif
XdgExporterUnstableV2Interface::XdgExporterUnstableV2Interface(Display *display, XdgForeignInterface *parent)
: Global(new Private(this, display, parent), parent)
{
}
XdgExporterUnstableV2Interface::~XdgExporterUnstableV2Interface()
{}
XdgExporterUnstableV2Interface::Private *XdgExporterUnstableV2Interface::d_func() const
{
return reinterpret_cast<Private*>(d.data());
}
XdgExportedUnstableV2Interface *XdgExporterUnstableV2Interface::exportedSurface(const QString &handle)
{
Q_D();
auto it = d->exportedSurfaces.constFind(handle);
if (it != d->exportedSurfaces.constEnd()) {
return it.value();
}
return nullptr;
}
void XdgExporterUnstableV2Interface::Private::destroyCallback(wl_client *client, wl_resource *resource)
{
Q_UNUSED(client)
}
void XdgExporterUnstableV2Interface::Private::exportCallback(wl_client *client, wl_resource *resource, uint32_t id, wl_resource * surface)
{
auto s = cast(resource);
QPointer <XdgExportedUnstableV2Interface> e = new XdgExportedUnstableV2Interface(s->q, surface);
e->create(s->display->getConnection(client), wl_resource_get_version(resource), id);
if (!e->resource()) {
wl_resource_post_no_memory(resource);
delete e;
return;
}
const QString handle = QUuid::createUuid().toString();
//a surface not exported anymore
connect(e.data(), &XdgExportedUnstableV2Interface::unbound,
s->q, [s, handle]() {
s->exportedSurfaces.remove(handle);
emit s->q->surfaceUnexported(handle);
});
//if the surface dies before this, this dies too
connect(SurfaceInterface::get(surface), &Resource::unbound,
s->q, [s, e, handle]() {
if (e) {
e->deleteLater();
}
s->exportedSurfaces.remove(handle);
emit s->q->surfaceUnexported(handle);
});
s->exportedSurfaces[handle] = e;
zxdg_exported_v2_send_handle(e->resource(), handle.toUtf8().constData());
emit s->q->surfaceExported(handle, e);
}
XdgExporterUnstableV2Interface::Private::Private(XdgExporterUnstableV2Interface *q, Display *d,XdgForeignInterface *foreignInterface)
: Global::Private(d, &zxdg_exporter_v2_interface, s_version)
, foreignInterface(foreignInterface)
, q(q)
{
}
void XdgExporterUnstableV2Interface::Private::bind(wl_client *client, uint32_t version, uint32_t id)
{
auto c = display->getConnection(client);
wl_resource *resource = c->createResource(&zxdg_exporter_v2_interface, qMin(version, s_version), id);
if (!resource) {
wl_client_post_no_memory(client);
return;
}
wl_resource_set_implementation(resource, &s_interface, this, unbind);
// TODO: should we track?
}
void XdgExporterUnstableV2Interface::Private::unbind(wl_resource *resource)
{
Q_UNUSED(resource)
// TODO: implement?
}
class Q_DECL_HIDDEN XdgImporterUnstableV2Interface::Private : public Global::Private
{
public:
Private(XdgImporterUnstableV2Interface *q, Display *d, XdgForeignInterface *foreignInterface);
XdgForeignInterface *foreignInterface;
QHash<QString, XdgImportedUnstableV2Interface *> importedSurfaces;
//child->parent hash
QHash<SurfaceInterface *, XdgImportedUnstableV2Interface *> parents;
//parent->child hash
QHash<XdgImportedUnstableV2Interface *, SurfaceInterface *> children;
private:
void bind(wl_client *client, uint32_t version, uint32_t id) override;
static void unbind(wl_resource *resource);
static Private *cast(wl_resource *r) {
return reinterpret_cast<Private*>(wl_resource_get_user_data(r));
}
static void destroyCallback(wl_client *client, wl_resource *resource);
static void importCallback(wl_client *client, wl_resource *resource, uint32_t id, const char * handle);
XdgImporterUnstableV2Interface *q;
static const struct zxdg_importer_v2_interface s_interface;
static const quint32 s_version;
};
const quint32 XdgImporterUnstableV2Interface::Private::s_version = 1;
#ifndef DOXYGEN_SHOULD_SKIP_THIS
const struct zxdg_importer_v2_interface XdgImporterUnstableV2Interface::Private::s_interface = {
destroyCallback,
importCallback
};
#endif
XdgImporterUnstableV2Interface::XdgImporterUnstableV2Interface(Display *display, XdgForeignInterface *parent)
: Global(new Private(this, display, parent), parent)
{
}
XdgImporterUnstableV2Interface::~XdgImporterUnstableV2Interface()
{
}
XdgImportedUnstableV2Interface *XdgImporterUnstableV2Interface::importedSurface(const QString &handle)
{
Q_D();
auto it = d->importedSurfaces.constFind(handle);
if (it != d->importedSurfaces.constEnd()) {
return it.value();
}
return nullptr;
}
SurfaceInterface *XdgImporterUnstableV2Interface::transientFor(SurfaceInterface *surface)
{
Q_D();
auto it = d->parents.constFind(surface);
if (it == d->parents.constEnd()) {
return nullptr;
}
return SurfaceInterface::get((*it)->parentResource());
}
XdgImporterUnstableV2Interface::Private *XdgImporterUnstableV2Interface::d_func() const
{
return reinterpret_cast<Private*>(d.data());
}
void XdgImporterUnstableV2Interface::Private::destroyCallback(wl_client *client, wl_resource *resource)
{
Q_UNUSED(client)
}
void XdgImporterUnstableV2Interface::Private::importCallback(wl_client *client, wl_resource *resource, uint32_t id, const char * handle)
{
auto s = cast(resource);
Q_ASSERT(s->foreignInterface);
XdgExportedUnstableV2Interface *exp = s->foreignInterface->d->exporter->exportedSurface(QString::fromUtf8(handle));
if (!exp) {
zxdg_imported_v2_send_destroyed(resource);
return;
}
wl_resource *surface = exp->parentResource();
if (!surface) {
zxdg_imported_v2_send_destroyed(resource);
return;
}
QPointer<XdgImportedUnstableV2Interface> imp = new XdgImportedUnstableV2Interface(s->q, surface);
imp->create(s->display->getConnection(client), wl_resource_get_version(resource), id);
//surface no longer exported
connect(exp, &XdgExportedUnstableV2Interface::unbound,
s->q, [s, imp, handle]() {
//imp valid when the exported is deleted before the imported
if (imp) {
zxdg_imported_v2_send_destroyed(imp->resource());
imp->deleteLater();
}
s->importedSurfaces.remove(QString::fromUtf8(handle));
emit s->q->surfaceUnimported(QString::fromUtf8(handle));
});
connect(imp.data(), &XdgImportedUnstableV2Interface::childChanged,
s->q, [s, imp](SurfaceInterface *child) {
//remove any previous association
auto it = s->children.find(imp);
if (it != s->children.end()) {
s->parents.remove(*it);
s->children.erase(it);
}
s->parents[child] = imp;
s->children[imp] = child;
SurfaceInterface *parent = SurfaceInterface::get(imp->parentResource());
emit s->q->transientChanged(child, parent);
//child surface destroyed
connect(child, &Resource::unbound,
s->q, [s, child]() {
auto it = s->parents.find(child);
if (it != s->parents.end()) {
s->children.remove(*it);
s->parents.erase(it);
emit s->q->transientChanged(nullptr, SurfaceInterface::get((*it)->parentResource()));
}
});
});
//surface no longer imported
connect(imp.data(), &XdgImportedUnstableV2Interface::unbound,
s->q, [s, handle, imp]() {
s->importedSurfaces.remove(QString::fromUtf8(handle));
emit s->q->surfaceUnimported(QString::fromUtf8(handle));
auto it = s->children.find(imp);
if (it != s->children.end()) {
s->parents.remove(*it);
s->children.erase(it);
emit s->q->transientChanged(*it, nullptr);
}
});
if (!imp->resource()) {
wl_resource_post_no_memory(resource);
delete imp;
return;
}
s->importedSurfaces[QString::fromUtf8(handle)] = imp;
emit s->q->surfaceImported(QString::fromUtf8(handle), imp);
}
XdgImporterUnstableV2Interface::Private::Private(XdgImporterUnstableV2Interface *q, Display *d, XdgForeignInterface *foreignInterface)
: Global::Private(d, &zxdg_importer_v2_interface, s_version)
, foreignInterface(foreignInterface)
, q(q)
{
}
void XdgImporterUnstableV2Interface::Private::bind(wl_client *client, uint32_t version, uint32_t id)
{
auto c = display->getConnection(client);
wl_resource *resource = c->createResource(&zxdg_importer_v2_interface, qMin(version, s_version), id);
if (!resource) {
wl_client_post_no_memory(client);
return;
}
wl_resource_set_implementation(resource, &s_interface, this, unbind);
// TODO: should we track?
}
void XdgImporterUnstableV2Interface::Private::unbind(wl_resource *resource)
{
Q_UNUSED(resource)
// TODO: implement?
}
class Q_DECL_HIDDEN XdgExportedUnstableV2Interface::Private : public Resource::Private
{
public:
Private(XdgExportedUnstableV2Interface *q, XdgExporterUnstableV2Interface *c, wl_resource *parentResource);
~Private();
private:
XdgExportedUnstableV2Interface *q_func() {
return reinterpret_cast<XdgExportedUnstableV2Interface *>(q);
}
static const struct zxdg_exported_v2_interface s_interface;
};
#ifndef DOXYGEN_SHOULD_SKIP_THIS
const struct zxdg_exported_v2_interface XdgExportedUnstableV2Interface::Private::s_interface = {
resourceDestroyedCallback
};
#endif
XdgExportedUnstableV2Interface::XdgExportedUnstableV2Interface(XdgExporterUnstableV2Interface *parent, wl_resource *parentResource)
: Resource(new Private(this, parent, parentResource))
{
}
XdgExportedUnstableV2Interface::~XdgExportedUnstableV2Interface()
{}
XdgExportedUnstableV2Interface::Private *XdgExportedUnstableV2Interface::d_func() const
{
return reinterpret_cast<Private*>(d.data());
}
XdgExportedUnstableV2Interface::Private::Private(XdgExportedUnstableV2Interface *q, XdgExporterUnstableV2Interface *c, wl_resource *parentResource)
: Resource::Private(q, c, parentResource, &zxdg_exported_v2_interface, &s_interface)
{
}
XdgExportedUnstableV2Interface::Private::~Private()
{}
class Q_DECL_HIDDEN XdgImportedUnstableV2Interface::Private : public Resource::Private
{
public:
Private(XdgImportedUnstableV2Interface *q, XdgImporterUnstableV2Interface *c, wl_resource *parentResource);
~Private();
QPointer<SurfaceInterface> parentOf;
private:
static void setParentOfCallback(wl_client *client, wl_resource *resource, wl_resource * surface);
XdgImportedUnstableV2Interface *q_func() {
return reinterpret_cast<XdgImportedUnstableV2Interface *>(q);
}
static const struct zxdg_imported_v2_interface s_interface;
};
#ifndef DOXYGEN_SHOULD_SKIP_THIS
const struct zxdg_imported_v2_interface XdgImportedUnstableV2Interface::Private::s_interface = {
resourceDestroyedCallback,
setParentOfCallback
};
#endif
XdgImportedUnstableV2Interface::XdgImportedUnstableV2Interface(XdgImporterUnstableV2Interface *parent, wl_resource *parentResource)
: Resource(new Private(this, parent, parentResource))
{
}
XdgImportedUnstableV2Interface::~XdgImportedUnstableV2Interface()
{}
XdgImportedUnstableV2Interface::Private *XdgImportedUnstableV2Interface::d_func() const
{
return reinterpret_cast<Private*>(d.data());
}
SurfaceInterface *XdgImportedUnstableV2Interface::child() const
{
Q_D();
return d->parentOf.data();
}
void XdgImportedUnstableV2Interface::Private::setParentOfCallback(wl_client *client, wl_resource *resource, wl_resource * surface)
{
auto s = cast<Private>(resource);
SurfaceInterface *surf = SurfaceInterface::get(surface);
if (!surf) {
return;
}
s->parentOf = surf;
emit s->q_func()->childChanged(surf);
}
XdgImportedUnstableV2Interface::Private::Private(XdgImportedUnstableV2Interface *q, XdgImporterUnstableV2Interface *c, wl_resource *parentResource)
: Resource::Private(q, c, parentResource, &zxdg_imported_v2_interface, &s_interface)
{
}
XdgImportedUnstableV2Interface::Private::~Private()
{}
}
}