mirror of
https://github.com/brl/mutter.git
synced 2024-11-28 02:50:41 -05:00
74fcdb9a62
The service channel D-Bus interface aims to be a "back door" for services that needs special casing in Mutter, e.g. have custom private protocols only meant to be used by that particular service. There are currently no special casing implemented; only the basic service channel infrastructure is added. There is a single method on the interface, that is meant to eventually be used by xdg-desktop-portal-gnome to open a Wayland connection with a private protocol needed for the portal backend's rather special window management needs. The service channel Wayland client works by allowing one instance of each "type", where each time needs to be defined to work in parallel. If a new service client connects, the old one will be disconnected. MetaWaylandClient's are used to manage the service clients, and are assigned the service client type. Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2810>
309 lines
10 KiB
C
309 lines
10 KiB
C
/*
|
|
* Copyright (C) 2023 Red Hat Inc.
|
|
*
|
|
* 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, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
* 02111-1307, USA.
|
|
*
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "core/meta-service-channel.h"
|
|
|
|
#include "wayland/meta-wayland-client-private.h"
|
|
|
|
#define META_SERVICE_CHANNEL_DBUS_SERVICE "org.gnome.Mutter.ServiceChannel"
|
|
#define META_SERVICE_CHANNEL_DBUS_PATH "/org/gnome/Mutter/ServiceChannel"
|
|
|
|
struct _MetaServiceChannel
|
|
{
|
|
MetaDBusServiceChannelSkeleton parent;
|
|
|
|
guint dbus_name_id;
|
|
|
|
MetaContext *context;
|
|
|
|
GHashTable *service_clients;
|
|
};
|
|
|
|
typedef struct _MetaServiceClient
|
|
{
|
|
MetaWaylandClient *wayland_client;
|
|
gulong destroyed_handler_id;
|
|
MetaServiceChannel *service_channel;
|
|
} MetaServiceClient;
|
|
|
|
static void meta_service_channel_init_iface (MetaDBusServiceChannelIface *iface);
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (MetaServiceChannel, meta_service_channel,
|
|
META_DBUS_TYPE_SERVICE_CHANNEL_SKELETON,
|
|
G_IMPLEMENT_INTERFACE (META_DBUS_TYPE_SERVICE_CHANNEL,
|
|
meta_service_channel_init_iface))
|
|
|
|
static void
|
|
meta_service_client_free (MetaServiceClient *service_client)
|
|
{
|
|
g_signal_handler_disconnect (service_client->wayland_client,
|
|
service_client->destroyed_handler_id);
|
|
g_object_unref (service_client->wayland_client);
|
|
g_free (service_client);
|
|
}
|
|
|
|
static void
|
|
on_service_client_destroyed (MetaWaylandClient *wayland_client,
|
|
MetaServiceClient *service_client)
|
|
{
|
|
MetaServiceClientType service_client_type;
|
|
|
|
service_client_type =
|
|
meta_wayland_client_get_service_client_type (wayland_client);
|
|
g_return_if_fail (service_client_type != META_SERVICE_CLIENT_TYPE_NONE);
|
|
|
|
g_hash_table_remove (service_client->service_channel->service_clients,
|
|
GINT_TO_POINTER (service_client_type));
|
|
}
|
|
|
|
static MetaServiceClient *
|
|
meta_service_client_new (MetaServiceChannel *service_channel,
|
|
MetaWaylandClient *wayland_client)
|
|
{
|
|
MetaServiceClient *service_client;
|
|
|
|
service_client = g_new0 (MetaServiceClient, 1);
|
|
service_client->service_channel = service_channel;
|
|
service_client->wayland_client = g_object_ref (wayland_client);
|
|
service_client->destroyed_handler_id =
|
|
g_signal_connect (wayland_client, "client-destroyed",
|
|
G_CALLBACK (on_service_client_destroyed),
|
|
service_client);
|
|
|
|
return service_client;
|
|
}
|
|
|
|
static gboolean
|
|
verify_service_client_type (uint32_t service_client_type)
|
|
{
|
|
switch ((MetaServiceClientType) service_client_type)
|
|
{
|
|
case META_SERVICE_CLIENT_TYPE_NONE:
|
|
return FALSE;
|
|
case META_SERVICE_CLIENT_TYPE_PORTAL_BACKEND:
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
handle_open_wayland_service_connection (MetaDBusServiceChannel *object,
|
|
GDBusMethodInvocation *invocation,
|
|
GUnixFDList *in_fd_list,
|
|
uint32_t service_client_type)
|
|
{
|
|
#ifdef HAVE_WAYLAND
|
|
MetaServiceChannel *service_channel = META_SERVICE_CHANNEL (object);
|
|
g_autoptr (GError) error = NULL;
|
|
g_autoptr (MetaWaylandClient) wayland_client = NULL;
|
|
g_autoptr (GUnixFDList) out_fd_list = NULL;
|
|
int fd;
|
|
int fd_id;
|
|
|
|
if (meta_context_get_compositor_type (service_channel->context) !=
|
|
META_COMPOSITOR_TYPE_WAYLAND)
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation,
|
|
G_DBUS_ERROR,
|
|
G_DBUS_ERROR_NOT_SUPPORTED,
|
|
"Not a Wayland compositor");
|
|
return G_DBUS_METHOD_INVOCATION_HANDLED;
|
|
}
|
|
|
|
if (!verify_service_client_type (service_client_type))
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation,
|
|
G_DBUS_ERROR,
|
|
G_DBUS_ERROR_INVALID_ARGS,
|
|
"Invalid service client type");
|
|
return G_DBUS_METHOD_INVOCATION_HANDLED;
|
|
}
|
|
|
|
wayland_client = meta_wayland_client_new_indirect (service_channel->context,
|
|
&error);
|
|
if (!wayland_client)
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation,
|
|
G_DBUS_ERROR,
|
|
G_DBUS_ERROR_NOT_SUPPORTED,
|
|
"Failed to create Wayland client: %s",
|
|
error->message);
|
|
return G_DBUS_METHOD_INVOCATION_HANDLED;
|
|
}
|
|
|
|
meta_wayland_client_assign_service_client_type (wayland_client,
|
|
service_client_type);
|
|
|
|
fd = meta_wayland_client_setup_fd (wayland_client, &error);
|
|
if (fd < 0)
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation,
|
|
G_DBUS_ERROR,
|
|
G_DBUS_ERROR_NOT_SUPPORTED,
|
|
"Failed to setup Wayland client socket: %s",
|
|
error->message);
|
|
return G_DBUS_METHOD_INVOCATION_HANDLED;
|
|
}
|
|
|
|
out_fd_list = g_unix_fd_list_new ();
|
|
fd_id = g_unix_fd_list_append (out_fd_list, fd, &error);
|
|
close (fd);
|
|
|
|
if (fd_id == -1)
|
|
{
|
|
g_dbus_method_invocation_return_error (invocation,
|
|
G_DBUS_ERROR,
|
|
G_DBUS_ERROR_ACCESS_DENIED,
|
|
"Failed to append fd: %s",
|
|
error->message);
|
|
return G_DBUS_METHOD_INVOCATION_HANDLED;
|
|
}
|
|
|
|
g_hash_table_replace (service_channel->service_clients,
|
|
GUINT_TO_POINTER (service_client_type),
|
|
meta_service_client_new (service_channel,
|
|
wayland_client));
|
|
|
|
meta_dbus_service_channel_complete_open_wayland_service_connection (
|
|
object, invocation, out_fd_list, g_variant_new_handle (fd_id));
|
|
return G_DBUS_METHOD_INVOCATION_HANDLED;
|
|
#else /* HAVE_WAYLAND */
|
|
g_dbus_method_invocation_return_error (invocation,
|
|
G_DBUS_ERROR,
|
|
G_DBUS_ERROR_NOT_SUPPORTED,
|
|
"Wayland not supported",
|
|
error->message);
|
|
return G_DBUS_METHOD_INVOCATION_HANDLED;
|
|
#endif /* HAVE_WAYLAND */
|
|
}
|
|
|
|
static void
|
|
meta_service_channel_init_iface (MetaDBusServiceChannelIface *iface)
|
|
{
|
|
iface->handle_open_wayland_service_connection =
|
|
handle_open_wayland_service_connection;
|
|
}
|
|
|
|
static void
|
|
on_bus_acquired (GDBusConnection *connection,
|
|
const char *name,
|
|
gpointer user_data)
|
|
{
|
|
MetaServiceChannel *service_channel = user_data;
|
|
GDBusInterfaceSkeleton *interface_skeleton =
|
|
G_DBUS_INTERFACE_SKELETON (service_channel);
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
if (!g_dbus_interface_skeleton_export (interface_skeleton,
|
|
connection,
|
|
META_SERVICE_CHANNEL_DBUS_PATH,
|
|
&error))
|
|
g_warning ("Failed to export service channel object: %s", error->message);
|
|
}
|
|
|
|
static void
|
|
on_name_acquired (GDBusConnection *connection,
|
|
const char *name,
|
|
gpointer user_data)
|
|
{
|
|
g_info ("Acquired name %s", name);
|
|
}
|
|
|
|
static void
|
|
on_name_lost (GDBusConnection *connection,
|
|
const char *name,
|
|
gpointer user_data)
|
|
{
|
|
g_warning ("Lost or failed to acquire name %s", name);
|
|
}
|
|
|
|
static void
|
|
meta_service_channel_constructed (GObject *object)
|
|
{
|
|
MetaServiceChannel *service_channel = META_SERVICE_CHANNEL (object);
|
|
|
|
service_channel->service_clients =
|
|
g_hash_table_new_full (NULL, NULL,
|
|
NULL, (GDestroyNotify) meta_service_client_free);
|
|
|
|
service_channel->dbus_name_id =
|
|
g_bus_own_name (G_BUS_TYPE_SESSION,
|
|
META_SERVICE_CHANNEL_DBUS_SERVICE,
|
|
G_BUS_NAME_OWNER_FLAGS_NONE,
|
|
on_bus_acquired,
|
|
on_name_acquired,
|
|
on_name_lost,
|
|
service_channel,
|
|
NULL);
|
|
}
|
|
|
|
static void
|
|
meta_service_channel_finalize (GObject *object)
|
|
{
|
|
MetaServiceChannel *service_channel = META_SERVICE_CHANNEL (object);
|
|
|
|
g_clear_pointer (&service_channel->service_clients, g_hash_table_unref);
|
|
g_clear_handle_id (&service_channel->dbus_name_id, g_bus_unown_name);
|
|
|
|
G_OBJECT_CLASS (meta_service_channel_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
meta_service_channel_class_init (MetaServiceChannelClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
object_class->constructed = meta_service_channel_constructed;
|
|
object_class->finalize = meta_service_channel_finalize;
|
|
}
|
|
|
|
static void
|
|
meta_service_channel_init (MetaServiceChannel *service_channel)
|
|
{
|
|
}
|
|
|
|
MetaServiceChannel *
|
|
meta_service_channel_new (MetaContext *context)
|
|
{
|
|
MetaServiceChannel *service_channel;
|
|
|
|
service_channel = g_object_new (META_TYPE_SERVICE_CHANNEL, NULL);
|
|
service_channel->context = context;
|
|
|
|
return service_channel;
|
|
}
|
|
|
|
MetaWaylandClient *
|
|
meta_service_channel_get_service_client (MetaServiceChannel *service_channel,
|
|
MetaServiceClientType service_client_type)
|
|
{
|
|
MetaServiceClient *service_client;
|
|
|
|
service_client = g_hash_table_lookup (service_channel->service_clients,
|
|
GINT_TO_POINTER (service_client_type));
|
|
if (!service_client)
|
|
return NULL;
|
|
|
|
return service_client->wayland_client;
|
|
}
|