/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*- * * Copyright (C) 2007 Novell, Inc. * Copyright (C) 2008 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 * 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. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #include <glib.h> #include <string.h> #include "gsm-app.h" #include "gsm-app-glue.h" #define GSM_APP_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), GSM_TYPE_APP, GsmAppPrivate)) struct _GsmAppPrivate { char *id; char *app_id; int phase; char *startup_id; DBusGConnection *connection; }; enum { EXITED, DIED, REGISTERED, LAST_SIGNAL }; static guint32 app_serial = 1; static guint signals[LAST_SIGNAL] = { 0 }; enum { PROP_0, PROP_ID, PROP_STARTUP_ID, PROP_PHASE, LAST_PROP }; G_DEFINE_TYPE (GsmApp, gsm_app, G_TYPE_OBJECT) GQuark gsm_app_error_quark (void) { static GQuark ret = 0; if (ret == 0) { ret = g_quark_from_static_string ("gsm_app_error"); } return ret; } static guint32 get_next_app_serial (void) { guint32 serial; serial = app_serial++; if ((gint32)app_serial < 0) { app_serial = 1; } return serial; } static gboolean register_app (GsmApp *app) { GError *error; error = NULL; app->priv->connection = dbus_g_bus_get (DBUS_BUS_SESSION, &error); if (app->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 (app->priv->connection, app->priv->id, G_OBJECT (app)); return TRUE; } static GObject * gsm_app_constructor (GType type, guint n_construct_properties, GObjectConstructParam *construct_properties) { GsmApp *app; gboolean res; app = GSM_APP (G_OBJECT_CLASS (gsm_app_parent_class)->constructor (type, n_construct_properties, construct_properties)); g_free (app->priv->id); app->priv->id = g_strdup_printf ("/org/mate/SessionManager/App%u", get_next_app_serial ()); res = register_app (app); if (! res) { g_warning ("Unable to register app with session bus"); } return G_OBJECT (app); } static void gsm_app_init (GsmApp *app) { app->priv = GSM_APP_GET_PRIVATE (app); } static void gsm_app_set_phase (GsmApp *app, int phase) { g_return_if_fail (GSM_IS_APP (app)); app->priv->phase = phase; } static void gsm_app_set_id (GsmApp *app, const char *id) { g_return_if_fail (GSM_IS_APP (app)); g_free (app->priv->id); app->priv->id = g_strdup (id); g_object_notify (G_OBJECT (app), "id"); } static void gsm_app_set_startup_id (GsmApp *app, const char *startup_id) { g_return_if_fail (GSM_IS_APP (app)); g_free (app->priv->startup_id); app->priv->startup_id = g_strdup (startup_id); g_object_notify (G_OBJECT (app), "startup-id"); } static void gsm_app_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GsmApp *app = GSM_APP (object); switch (prop_id) { case PROP_STARTUP_ID: gsm_app_set_startup_id (app, g_value_get_string (value)); break; case PROP_ID: gsm_app_set_id (app, g_value_get_string (value)); break; case PROP_PHASE: gsm_app_set_phase (app, g_value_get_int (value)); break; default: break; } } static void gsm_app_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GsmApp *app = GSM_APP (object); switch (prop_id) { case PROP_STARTUP_ID: g_value_set_string (value, app->priv->startup_id); break; case PROP_ID: g_value_set_string (value, app->priv->id); break; case PROP_PHASE: g_value_set_int (value, app->priv->phase); break; default: break; } } static void gsm_app_dispose (GObject *object) { GsmApp *app = GSM_APP (object); g_free (app->priv->startup_id); app->priv->startup_id = NULL; g_free (app->priv->id); app->priv->id = NULL; G_OBJECT_CLASS (gsm_app_parent_class)->dispose (object); } static void gsm_app_class_init (GsmAppClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->set_property = gsm_app_set_property; object_class->get_property = gsm_app_get_property; object_class->dispose = gsm_app_dispose; object_class->constructor = gsm_app_constructor; klass->impl_start = NULL; klass->impl_get_app_id = NULL; klass->impl_get_autorestart = NULL; klass->impl_provides = NULL; klass->impl_is_running = NULL; klass->impl_peek_autostart_delay = NULL; g_object_class_install_property (object_class, PROP_PHASE, g_param_spec_int ("phase", "Phase", "Phase", -1, G_MAXINT, -1, G_PARAM_READWRITE)); g_object_class_install_property (object_class, PROP_ID, g_param_spec_string ("id", "ID", "ID", NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT)); g_object_class_install_property (object_class, PROP_STARTUP_ID, g_param_spec_string ("startup-id", "startup ID", "Session management startup ID", NULL, G_PARAM_READWRITE | G_PARAM_CONSTRUCT)); signals[EXITED] = g_signal_new ("exited", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GsmAppClass, exited), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[DIED] = g_signal_new ("died", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GsmAppClass, died), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); signals[REGISTERED] = g_signal_new ("registered", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GsmAppClass, registered), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); g_type_class_add_private (klass, sizeof (GsmAppPrivate)); dbus_g_object_type_install_info (GSM_TYPE_APP, &dbus_glib_gsm_app_object_info); } const char * gsm_app_peek_id (GsmApp *app) { return app->priv->id; } const char * gsm_app_peek_app_id (GsmApp *app) { return GSM_APP_GET_CLASS (app)->impl_get_app_id (app); } const char * gsm_app_peek_startup_id (GsmApp *app) { return app->priv->startup_id; } /** * gsm_app_peek_phase: * @app: a %GsmApp * * Returns @app's startup phase. * * Return value: @app's startup phase **/ GsmManagerPhase gsm_app_peek_phase (GsmApp *app) { g_return_val_if_fail (GSM_IS_APP (app), GSM_MANAGER_PHASE_APPLICATION); return app->priv->phase; } gboolean gsm_app_peek_is_disabled (GsmApp *app) { g_return_val_if_fail (GSM_IS_APP (app), FALSE); if (GSM_APP_GET_CLASS (app)->impl_is_disabled) { return GSM_APP_GET_CLASS (app)->impl_is_disabled (app); } else { return FALSE; } } gboolean gsm_app_peek_is_conditionally_disabled (GsmApp *app) { g_return_val_if_fail (GSM_IS_APP (app), FALSE); if (GSM_APP_GET_CLASS (app)->impl_is_conditionally_disabled) { return GSM_APP_GET_CLASS (app)->impl_is_conditionally_disabled (app); } else { return FALSE; } } gboolean gsm_app_is_running (GsmApp *app) { g_return_val_if_fail (GSM_IS_APP (app), FALSE); if (GSM_APP_GET_CLASS (app)->impl_is_running) { return GSM_APP_GET_CLASS (app)->impl_is_running (app); } else { return FALSE; } } gboolean gsm_app_peek_autorestart (GsmApp *app) { g_return_val_if_fail (GSM_IS_APP (app), FALSE); if (GSM_APP_GET_CLASS (app)->impl_get_autorestart) { return GSM_APP_GET_CLASS (app)->impl_get_autorestart (app); } else { return FALSE; } } gboolean gsm_app_provides (GsmApp *app, const char *service) { if (GSM_APP_GET_CLASS (app)->impl_provides) { return GSM_APP_GET_CLASS (app)->impl_provides (app, service); } else { return FALSE; } } gboolean gsm_app_has_autostart_condition (GsmApp *app, const char *condition) { if (GSM_APP_GET_CLASS (app)->impl_has_autostart_condition) { return GSM_APP_GET_CLASS (app)->impl_has_autostart_condition (app, condition); } else { return FALSE; } } gboolean gsm_app_start (GsmApp *app, GError **error) { g_debug ("Starting app: %s", app->priv->id); return GSM_APP_GET_CLASS (app)->impl_start (app, error); } gboolean gsm_app_restart (GsmApp *app, GError **error) { g_debug ("Re-starting app: %s", app->priv->id); return GSM_APP_GET_CLASS (app)->impl_restart (app, error); } gboolean gsm_app_stop (GsmApp *app, GError **error) { return GSM_APP_GET_CLASS (app)->impl_stop (app, error); } void gsm_app_registered (GsmApp *app) { g_return_if_fail (GSM_IS_APP (app)); g_signal_emit (app, signals[REGISTERED], 0); } int gsm_app_peek_autostart_delay (GsmApp *app) { g_return_val_if_fail (GSM_IS_APP (app), FALSE); if (GSM_APP_GET_CLASS (app)->impl_peek_autostart_delay) { return GSM_APP_GET_CLASS (app)->impl_peek_autostart_delay (app); } else { return 0; } } void gsm_app_exited (GsmApp *app) { g_return_if_fail (GSM_IS_APP (app)); g_signal_emit (app, signals[EXITED], 0); } void gsm_app_died (GsmApp *app) { g_return_if_fail (GSM_IS_APP (app)); g_signal_emit (app, signals[DIED], 0); } gboolean gsm_app_get_app_id (GsmApp *app, char **id, GError **error) { g_return_val_if_fail (GSM_IS_APP (app), FALSE); *id = g_strdup (GSM_APP_GET_CLASS (app)->impl_get_app_id (app)); return TRUE; } gboolean gsm_app_get_startup_id (GsmApp *app, char **id, GError **error) { g_return_val_if_fail (GSM_IS_APP (app), FALSE); *id = g_strdup (app->priv->startup_id); return TRUE; } gboolean gsm_app_get_phase (GsmApp *app, guint *phase, GError **error) { g_return_val_if_fail (GSM_IS_APP (app), FALSE); *phase = app->priv->phase; return TRUE; }