/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- * * Copyright (C) 2011 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #include "config.h" #include #include #include "cd-inhibit.h" static void cd_inhibit_finalize (GObject *object); #define GET_PRIVATE(o) (cd_inhibit_get_instance_private (o)) typedef struct { GPtrArray *array; } CdInhibitPrivate; typedef struct { gchar *sender; guint watcher_id; } CdInhibitItem; enum { SIGNAL_CHANGED, SIGNAL_LAST }; static guint signals [SIGNAL_LAST] = { 0 }; G_DEFINE_TYPE_WITH_PRIVATE (CdInhibit, cd_inhibit, G_TYPE_OBJECT) gboolean cd_inhibit_valid (CdInhibit *inhibit) { CdInhibitPrivate *priv = GET_PRIVATE (inhibit); g_return_val_if_fail (CD_IS_INHIBIT (inhibit), FALSE); return priv->array->len == 0; } gchar ** cd_inhibit_get_bus_names (CdInhibit *inhibit) { CdInhibitItem *item_tmp; CdInhibitPrivate *priv = GET_PRIVATE (inhibit); gchar **bus_names; guint i; /* just copy senders */ bus_names = g_new0 (gchar *, priv->array->len + 1); for (i = 0; i < priv->array->len; i++) { item_tmp = g_ptr_array_index (priv->array, i); bus_names[i] = g_strdup (item_tmp->sender); } return bus_names; } static void cd_inhibit_item_free (CdInhibitItem *item) { if (item->watcher_id > 0) g_bus_unwatch_name (item->watcher_id); g_free (item->sender); g_free (item); } static CdInhibitItem * cd_inhibit_get_by_sender (CdInhibit *inhibit, const gchar *sender) { CdInhibitItem *item_tmp; CdInhibitPrivate *priv = GET_PRIVATE (inhibit); guint i; /* find sender */ for (i = 0; i < priv->array->len; i++) { item_tmp = g_ptr_array_index (priv->array, i); if (g_strcmp0 (item_tmp->sender, sender) == 0) return item_tmp; } return NULL; } gboolean cd_inhibit_remove (CdInhibit *inhibit, const gchar *sender, GError **error) { CdInhibitPrivate *priv = GET_PRIVATE (inhibit); CdInhibitItem *item; g_return_val_if_fail (CD_IS_INHIBIT (inhibit), FALSE); g_return_val_if_fail (sender != NULL, FALSE); /* do we already exist */ item = cd_inhibit_get_by_sender (inhibit, sender); if (item == NULL) { g_set_error (error, 1, 0, "not set inhibitor for %s", sender); return FALSE; } /* remove */ if (!g_ptr_array_remove (priv->array, item)) { g_set_error (error, 1, 0, "cannot remove inhibit item for %s", sender); return FALSE; } /* emit signal */ g_debug ("CdInhibit: emit changed"); g_signal_emit (inhibit, signals[SIGNAL_CHANGED], 0); return TRUE; } static void cd_inhibit_name_vanished_cb (GDBusConnection *connection, const gchar *name, gpointer user_data) { CdInhibit *inhibit = CD_INHIBIT (user_data); g_autoptr(GError) error = NULL; /* just remove */ if (!cd_inhibit_remove (inhibit, name, &error)) { g_warning ("CdInhibit: failed to remove when %s vanished: %s", name, error->message); } else { g_debug ("CdInhibit: remove inhibit as %s vanished", name); } } gboolean cd_inhibit_add (CdInhibit *inhibit, const gchar *sender, GError **error) { CdInhibitPrivate *priv = GET_PRIVATE (inhibit); CdInhibitItem *item; g_return_val_if_fail (CD_IS_INHIBIT (inhibit), FALSE); g_return_val_if_fail (sender != NULL, FALSE); /* do we already exist */ item = cd_inhibit_get_by_sender (inhibit, sender); if (item != NULL) { g_set_error (error, 1, 0, "already set inhibitor for %s", sender); return FALSE; } /* add */ item = g_new0 (CdInhibitItem, 1); item->sender = g_strdup (sender); item->watcher_id = g_bus_watch_name (G_BUS_TYPE_SYSTEM, sender, G_BUS_NAME_WATCHER_FLAGS_NONE, NULL, cd_inhibit_name_vanished_cb, g_object_ref (inhibit), g_object_unref); g_ptr_array_add (priv->array, item); /* emit signal */ g_debug ("CdInhibit: emit changed"); g_signal_emit (inhibit, signals[SIGNAL_CHANGED], 0); return TRUE; } static void cd_inhibit_class_init (CdInhibitClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->finalize = cd_inhibit_finalize; signals [SIGNAL_CHANGED] = g_signal_new ("changed", G_TYPE_FROM_CLASS (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (CdInhibitClass, changed), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); } static void cd_inhibit_init (CdInhibit *inhibit) { CdInhibitPrivate *priv = GET_PRIVATE (inhibit); priv->array = g_ptr_array_new_with_free_func ((GDestroyNotify) cd_inhibit_item_free); } static void cd_inhibit_finalize (GObject *object) { CdInhibit *inhibit = CD_INHIBIT (object); CdInhibitPrivate *priv = GET_PRIVATE (inhibit); g_ptr_array_unref (priv->array); G_OBJECT_CLASS (cd_inhibit_parent_class)->finalize (object); } CdInhibit * cd_inhibit_new (void) { CdInhibit *inhibit; inhibit = g_object_new (CD_TYPE_INHIBIT, NULL); return CD_INHIBIT (inhibit); }