/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- * * Copyright (C) 2008 Richard Hughes * * Licensed under the GNU General Public License Version 2 * * 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 #include #include #include #include "egg-debug.h" #include "dkp-client.h" #include "dkp-client-device.h" static void dkp_client_class_init (DkpClientClass *klass); static void dkp_client_init (DkpClient *client); static void dkp_client_finalize (GObject *object); #define DKP_CLIENT_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), DKP_TYPE_CLIENT, DkpClientPrivate)) struct DkpClientPrivate { DBusGConnection *bus; DBusGProxy *proxy; DBusGProxy *prop_proxy; GHashTable *hash; GPtrArray *array; gboolean have_properties; char *daemon_version; gboolean can_suspend; gboolean can_hibernate; gboolean on_battery; gboolean on_low_battery; }; enum { DKP_CLIENT_DEVICE_ADDED, DKP_CLIENT_DEVICE_CHANGED, DKP_CLIENT_DEVICE_REMOVED, DKP_CLIENT_CHANGED, DKP_CLIENT_LAST_SIGNAL }; static guint signals [DKP_CLIENT_LAST_SIGNAL] = { 0 }; G_DEFINE_TYPE (DkpClient, dkp_client, G_TYPE_OBJECT) /** * dkp_client_get_device: **/ static DkpClientDevice * dkp_client_get_device (DkpClient *client, const gchar *object_path) { DkpClientDevice *device; device = g_hash_table_lookup (client->priv->hash, object_path); return device; } /** * dkp_client_enumerate_devices: **/ GPtrArray * dkp_client_enumerate_devices (DkpClient *client, GError **error) { gboolean ret; GError *error_local = NULL; GPtrArray *devices = NULL; GType g_type_array; g_type_array = dbus_g_type_get_collection ("GPtrArray", DBUS_TYPE_G_OBJECT_PATH); ret = dbus_g_proxy_call (client->priv->proxy, "EnumerateDevices", &error_local, G_TYPE_INVALID, g_type_array, &devices, G_TYPE_INVALID); if (!ret) { egg_warning ("Couldn't enumerate devices: %s", error_local->message); if (error != NULL) *error = g_error_new (1, 0, "%s", error_local->message); g_error_free (error_local); } return devices; } /** * dkp_client_ensure_properties: **/ static void dkp_client_ensure_properties (DkpClient *client) { gboolean ret; GError *error; GHashTable *props; GValue *value; props = NULL; if (client->priv->have_properties) goto out; error = NULL; ret = dbus_g_proxy_call (client->priv->prop_proxy, "GetAll", &error, G_TYPE_STRING, "org.freedesktop.DeviceKit.Power", G_TYPE_INVALID, dbus_g_type_get_map ("GHashTable", G_TYPE_STRING, G_TYPE_VALUE), &props, G_TYPE_INVALID); if (!ret) { g_warning ("Error invoking GetAll() to get properties: %s", error->message); g_error_free (error); goto out; } value = g_hash_table_lookup (props, "daemon-version"); if (value == NULL) { g_warning ("No 'daemon-version' property"); goto out; } client->priv->daemon_version = g_strdup (g_value_get_string (value)); value = g_hash_table_lookup (props, "can-suspend"); if (value == NULL) { g_warning ("No 'can-suspend' property"); goto out; } client->priv->can_suspend = g_value_get_boolean (value); value = g_hash_table_lookup (props, "can-hibernate"); if (value == NULL) { g_warning ("No 'can-hibernate' property"); goto out; } client->priv->can_hibernate = g_value_get_boolean (value); value = g_hash_table_lookup (props, "on-battery"); if (value == NULL) { g_warning ("No 'on-battery' property"); goto out; } client->priv->on_battery = g_value_get_boolean (value); value = g_hash_table_lookup (props, "on-low-battery"); if (value == NULL) { g_warning ("No 'on-low-battery' property"); goto out; } client->priv->on_low_battery = g_value_get_boolean (value); /* cached */ client->priv->have_properties = TRUE; out: if (props != NULL) g_hash_table_unref (props); } /** * dkp_client_get_daemon_version: **/ const gchar * dkp_client_get_daemon_version (DkpClient *client) { g_return_val_if_fail (DKP_IS_CLIENT (client), NULL); dkp_client_ensure_properties (client); return client->priv->daemon_version; } /** * dkp_client_can_hibernate: **/ gboolean dkp_client_can_hibernate (DkpClient *client) { g_return_val_if_fail (DKP_IS_CLIENT (client), FALSE); dkp_client_ensure_properties (client); return client->priv->can_hibernate; } /** * dkp_client_can_suspend: **/ gboolean dkp_client_can_suspend (DkpClient *client) { g_return_val_if_fail (DKP_IS_CLIENT (client), FALSE); dkp_client_ensure_properties (client); return client->priv->can_suspend; } /** * dkp_client_on_battery: **/ gboolean dkp_client_on_battery (DkpClient *client) { g_return_val_if_fail (DKP_IS_CLIENT (client), FALSE); dkp_client_ensure_properties (client); return client->priv->on_battery; } /** * dkp_client_on_low_battery: **/ gboolean dkp_client_on_low_battery (DkpClient *client) { g_return_val_if_fail (DKP_IS_CLIENT (client), FALSE); dkp_client_ensure_properties (client); return client->priv->on_low_battery; } /** * dkp_client_add: **/ static DkpClientDevice * dkp_client_add (DkpClient *client, const gchar *object_path) { DkpClientDevice *device; /* create new device */ device = dkp_client_device_new (); dkp_client_device_set_object_path (device, object_path); g_ptr_array_add (client->priv->array, device); g_hash_table_insert (client->priv->hash, g_strdup (object_path), device); return device; } /** * dkp_client_remove: **/ static gboolean dkp_client_remove (DkpClient *client, DkpClientDevice *device) { /* deallocate it */ g_object_unref (device); g_ptr_array_remove (client->priv->array, device); g_hash_table_remove (client->priv->hash, device); return TRUE; } /** * dkp_client_added_cb: **/ static void dkp_client_device_added_cb (DBusGProxy *proxy, const gchar *object_path, DkpClient *client) { DkpClientDevice *device; /* create new device */ device = dkp_client_add (client, object_path); g_signal_emit (client, signals [DKP_CLIENT_DEVICE_ADDED], 0, device); } /** * dkp_client_changed_cb: **/ static void dkp_client_device_changed_cb (DBusGProxy *proxy, const gchar *object_path, DkpClient *client) { DkpClientDevice *device; device = dkp_client_get_device (client, object_path); if (device != NULL) g_signal_emit (client, signals [DKP_CLIENT_DEVICE_CHANGED], 0, device); } /** * dkp_client_removed_cb: **/ static void dkp_client_device_removed_cb (DBusGProxy *proxy, const gchar *object_path, DkpClient *client) { DkpClientDevice *device; device = dkp_client_get_device (client, object_path); if (device != NULL) g_signal_emit (client, signals [DKP_CLIENT_DEVICE_REMOVED], 0, device); dkp_client_remove (client, device); } /** * dkp_client_changed_cb: **/ static void dkp_client_changed_cb (DBusGProxy *proxy, DkpClient *client) { client->priv->have_properties = FALSE; g_signal_emit (client, signals [DKP_CLIENT_CHANGED], 0); } /** * dkp_client_class_init: * @klass: The DkpClientClass **/ static void dkp_client_class_init (DkpClientClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = dkp_client_finalize; signals [DKP_CLIENT_DEVICE_ADDED] = g_signal_new ("device-added", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (DkpClientClass, device_added), NULL, NULL, g_cclosure_marshal_VOID__POINTER, G_TYPE_NONE, 1, G_TYPE_POINTER); signals [DKP_CLIENT_DEVICE_REMOVED] = g_signal_new ("device-removed", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (DkpClientClass, device_removed), NULL, NULL, g_cclosure_marshal_VOID__POINTER, G_TYPE_NONE, 1, G_TYPE_POINTER); signals [DKP_CLIENT_DEVICE_CHANGED] = g_signal_new ("device-changed", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (DkpClientClass, device_changed), NULL, NULL, g_cclosure_marshal_VOID__POINTER, G_TYPE_NONE, 1, G_TYPE_POINTER); signals [DKP_CLIENT_CHANGED] = g_signal_new ("changed", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (DkpClientClass, changed), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); g_type_class_add_private (klass, sizeof (DkpClientPrivate)); } /** * dkp_client_init: * @client: This class instance **/ static void dkp_client_init (DkpClient *client) { GError *error = NULL; const gchar *object_path; GPtrArray *devices; guint i; client->priv = DKP_CLIENT_GET_PRIVATE (client); client->priv->hash = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL); client->priv->array = g_ptr_array_new (); /* get on the bus */ client->priv->bus = dbus_g_bus_get (DBUS_BUS_SYSTEM, &error); if (client->priv->bus == NULL) { egg_warning ("Couldn't connect to system bus: %s", error->message); g_error_free (error); goto out; } /* connect to main interface */ client->priv->proxy = dbus_g_proxy_new_for_name (client->priv->bus, "org.freedesktop.DeviceKit.Power", "/org/freedesktop/DeviceKit/Power", "org.freedesktop.DeviceKit.Power"); if (client->priv->proxy == NULL) { egg_warning ("Couldn't connect to proxy"); goto out; } client->priv->prop_proxy = dbus_g_proxy_new_for_name (client->priv->bus, "org.freedesktop.DeviceKit.Power", "/org/freedesktop/DeviceKit/Power", "org.freedesktop.DBus.Properties"); if (client->priv->prop_proxy == NULL) { egg_warning ("Couldn't connect to proxy"); goto out; } dbus_g_proxy_add_signal (client->priv->proxy, "DeviceAdded", G_TYPE_STRING, G_TYPE_INVALID); dbus_g_proxy_add_signal (client->priv->proxy, "DeviceRemoved", G_TYPE_STRING, G_TYPE_INVALID); dbus_g_proxy_add_signal (client->priv->proxy, "DeviceChanged", G_TYPE_STRING, G_TYPE_INVALID); dbus_g_proxy_add_signal (client->priv->proxy, "Changed", G_TYPE_INVALID); /* all callbacks */ dbus_g_proxy_connect_signal (client->priv->proxy, "DeviceAdded", G_CALLBACK (dkp_client_device_added_cb), client, NULL); dbus_g_proxy_connect_signal (client->priv->proxy, "DeviceRemoved", G_CALLBACK (dkp_client_device_removed_cb), client, NULL); dbus_g_proxy_connect_signal (client->priv->proxy, "DeviceChanged", G_CALLBACK (dkp_client_device_changed_cb), client, NULL); dbus_g_proxy_connect_signal (client->priv->proxy, "Changed", G_CALLBACK (dkp_client_changed_cb), client, NULL); /* coldplug */ devices = dkp_client_enumerate_devices (client, NULL); if (devices == NULL) goto out; for (i=0; ilen; i++) { object_path = (const gchar *) g_ptr_array_index (devices, i); dkp_client_add (client, object_path); } out: return; } /** * dkp_client_finalize: * @object: The object to finalize **/ static void dkp_client_finalize (GObject *object) { DkpClient *client; DkpClientDevice *device; guint i; g_return_if_fail (DKP_IS_CLIENT (object)); client = DKP_CLIENT (object); /* free any devices */ for (i=0; ipriv->array->len; i++) { device = (DkpClientDevice *) g_ptr_array_index (client->priv->array, i); dkp_client_remove (client, device); } g_ptr_array_free (client->priv->array, TRUE); g_hash_table_unref (client->priv->hash); dbus_g_connection_unref (client->priv->bus); if (client->priv->proxy != NULL) g_object_unref (client->priv->proxy); if (client->priv->prop_proxy != NULL) g_object_unref (client->priv->prop_proxy); g_free (client->priv->daemon_version); G_OBJECT_CLASS (dkp_client_parent_class)->finalize (object); } /** * dkp_client_new: * * Return value: a new DkpClient object. **/ DkpClient * dkp_client_new (void) { DkpClient *client; client = g_object_new (DKP_TYPE_CLIENT, NULL); return DKP_CLIENT (client); }