mirror of
https://github.com/brl/mutter.git
synced 2024-11-28 11:00:54 -05:00
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;
|
||
|
}
|