/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*- * * Copyright (C) 2007 Novell, Inc. * Copyright (C) 2008 Red Hat, Inc. * Copyright (C) 2012-2021 MATE Developers * * 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 * Lesser 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., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301, USA. */ #include "config.h" #include "eggdesktopfile.h" #include "gsm-marshal.h" #include "gsm-client.h" #include "gsm-client-glue.h" static guint32 client_serial = 1; typedef struct { GObject parent; char *id; char *startup_id; char *app_id; guint status; DBusGConnection *connection; } GsmClientPrivate; enum { PROP_0, PROP_ID, PROP_STARTUP_ID, PROP_APP_ID, PROP_STATUS }; enum { DISCONNECTED, END_SESSION_RESPONSE, LAST_SIGNAL }; static guint signals[LAST_SIGNAL] = { 0 }; G_DEFINE_ABSTRACT_TYPE_WITH_PRIVATE (GsmClient, gsm_client, G_TYPE_OBJECT) GQuark gsm_client_error_quark (void) { static GQuark ret = 0; if (ret == 0) { ret = g_quark_from_static_string ("gsm_client_error"); } return ret; } #define ENUM_ENTRY(NAME, DESC) { NAME, "" #NAME "", DESC } GType gsm_client_error_get_type (void) { static GType etype = 0; if (etype == 0) { static const GEnumValue values[] = { ENUM_ENTRY (GSM_CLIENT_ERROR_GENERAL, "GeneralError"), ENUM_ENTRY (GSM_CLIENT_ERROR_NOT_REGISTERED, "NotRegistered"), { 0, 0, 0 } }; g_assert (GSM_CLIENT_NUM_ERRORS == G_N_ELEMENTS (values) - 1); etype = g_enum_register_static ("GsmClientError", values); } return etype; } static guint32 get_next_client_serial (void) { guint32 serial; serial = client_serial++; if ((gint32)client_serial < 0) { client_serial = 1; } return serial; } static gboolean register_client (GsmClient *client) { GError *error; GsmClientPrivate *priv; error = NULL; priv = gsm_client_get_instance_private (client); priv->connection = dbus_g_bus_get (DBUS_BUS_SESSION, &error); if (priv->connection == NULL) { if (error != NULL) { g_critical ("error getting session bus: %s", error->message); g_error_free (error); } return FALSE; } dbus_g_connection_register_g_object (priv->connection, priv->id, G_OBJECT (client)); return TRUE; } static GObject * gsm_client_constructor (GType type, guint n_construct_properties, GObjectConstructParam *construct_properties) { GsmClient *client; gboolean res; GsmClientPrivate *priv; client = GSM_CLIENT (G_OBJECT_CLASS (gsm_client_parent_class)->constructor (type, n_construct_properties, construct_properties)); priv = gsm_client_get_instance_private (client); g_free (priv->id); priv->id = g_strdup_printf ("/org/gnome/SessionManager/Client%u", get_next_client_serial ()); res = register_client (client); if (! res) { g_warning ("Unable to register client with session bus"); } return G_OBJECT (client); } static void gsm_client_init (GsmClient *client) { } static void gsm_client_finalize (GObject *object) { GsmClient *client; GsmClientPrivate *priv; g_return_if_fail (object != NULL); g_return_if_fail (GSM_IS_CLIENT (object)); client = GSM_CLIENT (object); priv = gsm_client_get_instance_private (client); g_return_if_fail (priv != NULL); g_free (priv->id); g_free (priv->startup_id); g_free (priv->app_id); G_OBJECT_CLASS (gsm_client_parent_class)->finalize (object); } void gsm_client_set_status (GsmClient *client, guint status) { GsmClientPrivate *priv; g_return_if_fail (GSM_IS_CLIENT (client)); priv = gsm_client_get_instance_private (client); if (priv->status != status) { priv->status = status; g_object_notify (G_OBJECT (client), "status"); } } static void gsm_client_set_startup_id (GsmClient *client, const char *startup_id) { GsmClientPrivate *priv; g_return_if_fail (GSM_IS_CLIENT (client)); priv = gsm_client_get_instance_private (client); g_free (priv->startup_id); if (startup_id != NULL) { priv->startup_id = g_strdup (startup_id); } else { priv->startup_id = g_strdup (""); } g_object_notify (G_OBJECT (client), "startup-id"); } void gsm_client_set_app_id (GsmClient *client, const char *app_id) { GsmClientPrivate *priv; g_return_if_fail (GSM_IS_CLIENT (client)); priv = gsm_client_get_instance_private (client); g_free (priv->app_id); if (app_id != NULL) { priv->app_id = g_strdup (app_id); } else { priv->app_id = g_strdup (""); } g_object_notify (G_OBJECT (client), "app-id"); } static void gsm_client_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GsmClient *self; self = GSM_CLIENT (object); switch (prop_id) { case PROP_STARTUP_ID: gsm_client_set_startup_id (self, g_value_get_string (value)); break; case PROP_APP_ID: gsm_client_set_app_id (self, g_value_get_string (value)); break; case PROP_STATUS: gsm_client_set_status (self, g_value_get_uint (value)); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gsm_client_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GsmClient *self; GsmClientPrivate *priv; self = GSM_CLIENT (object); priv = gsm_client_get_instance_private (self); switch (prop_id) { case PROP_STARTUP_ID: g_value_set_string (value, priv->startup_id); break; case PROP_APP_ID: g_value_set_string (value, priv->app_id); break; case PROP_STATUS: g_value_set_uint (value, priv->status); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static gboolean default_stop (GsmClient *client, GError **error) { g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); g_warning ("Stop not implemented"); return TRUE; } static void gsm_client_dispose (GObject *object) { GsmClient *client; GsmClientPrivate *priv; g_return_if_fail (object != NULL); g_return_if_fail (GSM_IS_CLIENT (object)); client = GSM_CLIENT (object); priv = gsm_client_get_instance_private (client); g_debug ("GsmClient: disposing %s", priv->id); G_OBJECT_CLASS (gsm_client_parent_class)->dispose (object); } static void gsm_client_class_init (GsmClientClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->get_property = gsm_client_get_property; object_class->set_property = gsm_client_set_property; object_class->constructor = gsm_client_constructor; object_class->finalize = gsm_client_finalize; object_class->dispose = gsm_client_dispose; klass->impl_stop = default_stop; signals[DISCONNECTED] = g_signal_new ("disconnected", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GsmClientClass, disconnected), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[END_SESSION_RESPONSE] = g_signal_new ("end-session-response", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GsmClientClass, end_session_response), NULL, NULL, gsm_marshal_VOID__BOOLEAN_BOOLEAN_BOOLEAN_STRING, G_TYPE_NONE, 4, G_TYPE_BOOLEAN, G_TYPE_BOOLEAN, G_TYPE_BOOLEAN, G_TYPE_STRING); g_object_class_install_property (object_class, PROP_STARTUP_ID, g_param_spec_string ("startup-id", "startup-id", "startup-id", "", G_PARAM_READWRITE | G_PARAM_CONSTRUCT)); g_object_class_install_property (object_class, PROP_APP_ID, g_param_spec_string ("app-id", "app-id", "app-id", "", G_PARAM_READWRITE | G_PARAM_CONSTRUCT)); g_object_class_install_property (object_class, PROP_STATUS, g_param_spec_uint ("status", "status", "status", 0, G_MAXINT, GSM_CLIENT_UNREGISTERED, G_PARAM_READWRITE | G_PARAM_CONSTRUCT)); dbus_g_object_type_install_info (GSM_TYPE_CLIENT, &dbus_glib_gsm_client_object_info); } const char * gsm_client_peek_id (GsmClient *client) { GsmClientPrivate *priv; g_return_val_if_fail (GSM_IS_CLIENT (client), NULL); priv = gsm_client_get_instance_private (client); return priv->id; } /** * gsm_client_peek_app_id: * @client: a #GsmClient. * * Note that the application ID might not be known; this happens when for XSMP * clients that we did not start ourselves, for instance. * * Returns: the application ID of the client, or %NULL if no such ID is * known. The string is owned by @client. **/ const char * gsm_client_peek_app_id (GsmClient *client) { GsmClientPrivate *priv; g_return_val_if_fail (GSM_IS_CLIENT (client), NULL); priv = gsm_client_get_instance_private (client); return priv->app_id; } const char * gsm_client_peek_startup_id (GsmClient *client) { GsmClientPrivate *priv; g_return_val_if_fail (GSM_IS_CLIENT (client), NULL); priv = gsm_client_get_instance_private (client); return priv->startup_id; } guint gsm_client_peek_status (GsmClient *client) { GsmClientPrivate *priv; g_return_val_if_fail (GSM_IS_CLIENT (client), GSM_CLIENT_UNREGISTERED); priv = gsm_client_get_instance_private (client); return priv->status; } guint gsm_client_peek_restart_style_hint (GsmClient *client) { g_return_val_if_fail (GSM_IS_CLIENT (client), GSM_CLIENT_RESTART_NEVER); return GSM_CLIENT_GET_CLASS (client)->impl_get_restart_style_hint (client); } gboolean gsm_client_get_startup_id (GsmClient *client, char **id, GError **error) { GsmClientPrivate *priv; g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); priv = gsm_client_get_instance_private (client); *id = g_strdup (priv->startup_id); return TRUE; } gboolean gsm_client_get_app_id (GsmClient *client, char **id, GError **error) { GsmClientPrivate *priv; g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); priv = gsm_client_get_instance_private (client); *id = g_strdup (priv->app_id); return TRUE; } gboolean gsm_client_get_restart_style_hint (GsmClient *client, guint *hint, GError **error) { g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); *hint = GSM_CLIENT_GET_CLASS (client)->impl_get_restart_style_hint (client); return TRUE; } gboolean gsm_client_get_status (GsmClient *client, guint *status, GError **error) { GsmClientPrivate *priv; g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); priv = gsm_client_get_instance_private (client); *status = priv->status; return TRUE; } gboolean gsm_client_get_unix_process_id (GsmClient *client, guint *pid, GError **error) { g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); *pid = GSM_CLIENT_GET_CLASS (client)->impl_get_unix_process_id (client); return TRUE; } /** * gsm_client_get_app_name: * @client: a #GsmClient. * * Returns: a copy of the application name of the client, or %NULL if no such * name is known. **/ char * gsm_client_get_app_name (GsmClient *client) { g_return_val_if_fail (GSM_IS_CLIENT (client), NULL); return GSM_CLIENT_GET_CLASS (client)->impl_get_app_name (client); } gboolean gsm_client_cancel_end_session (GsmClient *client, GError **error) { g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); return GSM_CLIENT_GET_CLASS (client)->impl_cancel_end_session (client, error); } gboolean gsm_client_query_end_session (GsmClient *client, guint flags, GError **error) { g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); return GSM_CLIENT_GET_CLASS (client)->impl_query_end_session (client, flags, error); } gboolean gsm_client_end_session (GsmClient *client, guint flags, GError **error) { g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); return GSM_CLIENT_GET_CLASS (client)->impl_end_session (client, flags, error); } gboolean gsm_client_stop (GsmClient *client, GError **error) { g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); return GSM_CLIENT_GET_CLASS (client)->impl_stop (client, error); } void gsm_client_disconnected (GsmClient *client) { g_signal_emit (client, signals[DISCONNECTED], 0); } GKeyFile * gsm_client_save (GsmClient *client, GError **error) { g_return_val_if_fail (GSM_IS_CLIENT (client), FALSE); return GSM_CLIENT_GET_CLASS (client)->impl_save (client, error); } void gsm_client_end_session_response (GsmClient *client, gboolean is_ok, gboolean do_last, gboolean cancel, const char *reason) { g_signal_emit (client, signals[END_SESSION_RESPONSE], 0, is_ok, do_last, cancel, reason); }