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
459 lines
15 KiB
C++
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()
|
|
{}
|
|
|
|
}
|
|
}
|
|
|