blob: 37542f75c6f2aa91e0f0f44cf23185ec989b9f0d [file] [log] [blame]
/* GIO - GLib Input, Output and Streaming Library
*
* Copyright 2019 Red Hat, Inc.
*
* SPDX-License-Identifier: LGPL-2.1-or-later
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library 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 Lesser General
* Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
*/
#include "config.h"
#include "gmemorymonitor.h"
#include "gmemorymonitordbus.h"
#include "gioerror.h"
#include "ginitable.h"
#include "giomodule-priv.h"
#include "glibintl.h"
#include "glib/gstdio.h"
#include "gcancellable.h"
#include "gdbusproxy.h"
#include "gdbusnamewatching.h"
#define G_MEMORY_MONITOR_DBUS_GET_INITABLE_IFACE(o) (G_TYPE_INSTANCE_GET_INTERFACE ((o), G_TYPE_INITABLE, GInitable))
static void g_memory_monitor_dbus_iface_init (GMemoryMonitorInterface *iface);
static void g_memory_monitor_dbus_initable_iface_init (GInitableIface *iface);
struct _GMemoryMonitorDBus
{
GObject parent_instance;
guint watch_id;
GCancellable *cancellable;
GDBusProxy *proxy;
gulong signal_id;
};
G_DEFINE_TYPE_WITH_CODE (GMemoryMonitorDBus, g_memory_monitor_dbus, G_TYPE_OBJECT,
G_IMPLEMENT_INTERFACE (G_TYPE_INITABLE,
g_memory_monitor_dbus_initable_iface_init)
G_IMPLEMENT_INTERFACE (G_TYPE_MEMORY_MONITOR,
g_memory_monitor_dbus_iface_init)
_g_io_modules_ensure_extension_points_registered ();
g_io_extension_point_implement (G_MEMORY_MONITOR_EXTENSION_POINT_NAME,
g_define_type_id,
"dbus",
30))
static void
g_memory_monitor_dbus_init (GMemoryMonitorDBus *dbus)
{
}
static void
proxy_signal_cb (GDBusProxy *proxy,
const gchar *sender_name,
const gchar *signal_name,
GVariant *parameters,
GMemoryMonitorDBus *dbus)
{
guint8 level;
if (g_strcmp0 (signal_name, "LowMemoryWarning") != 0)
return;
if (parameters == NULL)
return;
g_variant_get (parameters, "(y)", &level);
g_signal_emit_by_name (dbus, "low-memory-warning", level);
}
static void
lmm_proxy_cb (GObject *source_object,
GAsyncResult *res,
gpointer user_data)
{
GMemoryMonitorDBus *dbus = user_data;
GDBusProxy *proxy;
GError *error = NULL;
proxy = g_dbus_proxy_new_finish (res, &error);
if (!proxy)
{
g_debug ("Failed to create LowMemoryMonitor D-Bus proxy: %s",
error->message);
g_error_free (error);
return;
}
dbus->signal_id = g_signal_connect (G_OBJECT (proxy), "g-signal",
G_CALLBACK (proxy_signal_cb), dbus);
dbus->proxy = proxy;
}
static void
lmm_appeared_cb (GDBusConnection *connection,
const gchar *name,
const gchar *name_owner,
gpointer user_data)
{
GMemoryMonitorDBus *dbus = user_data;
g_dbus_proxy_new (connection,
G_DBUS_PROXY_FLAGS_DO_NOT_AUTO_START,
NULL,
"org.freedesktop.LowMemoryMonitor",
"/org/freedesktop/LowMemoryMonitor",
"org.freedesktop.LowMemoryMonitor",
dbus->cancellable,
lmm_proxy_cb,
dbus);
}
static void
lmm_vanished_cb (GDBusConnection *connection,
const gchar *name,
gpointer user_data)
{
GMemoryMonitorDBus *dbus = user_data;
g_clear_signal_handler (&dbus->signal_id, dbus->proxy);
g_clear_object (&dbus->proxy);
}
static gboolean
g_memory_monitor_dbus_initable_init (GInitable *initable,
GCancellable *cancellable,
GError **error)
{
GMemoryMonitorDBus *dbus = G_MEMORY_MONITOR_DBUS (initable);
dbus->cancellable = g_cancellable_new ();
dbus->watch_id = g_bus_watch_name (G_BUS_TYPE_SYSTEM,
"org.freedesktop.LowMemoryMonitor",
G_BUS_NAME_WATCHER_FLAGS_AUTO_START,
lmm_appeared_cb,
lmm_vanished_cb,
dbus,
NULL);
return TRUE;
}
static void
g_memory_monitor_dbus_finalize (GObject *object)
{
GMemoryMonitorDBus *dbus = G_MEMORY_MONITOR_DBUS (object);
g_cancellable_cancel (dbus->cancellable);
g_clear_object (&dbus->cancellable);
g_clear_signal_handler (&dbus->signal_id, dbus->proxy);
g_clear_object (&dbus->proxy);
g_clear_handle_id (&dbus->watch_id, g_bus_unwatch_name);
G_OBJECT_CLASS (g_memory_monitor_dbus_parent_class)->finalize (object);
}
static void
g_memory_monitor_dbus_class_init (GMemoryMonitorDBusClass *nl_class)
{
GObjectClass *gobject_class = G_OBJECT_CLASS (nl_class);
gobject_class->finalize = g_memory_monitor_dbus_finalize;
}
static void
g_memory_monitor_dbus_iface_init (GMemoryMonitorInterface *monitor_iface)
{
}
static void
g_memory_monitor_dbus_initable_iface_init (GInitableIface *iface)
{
iface->init = g_memory_monitor_dbus_initable_init;
}