2013-08-14 10:50:48 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright 2013 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
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
2014-01-12 01:42:06 +00:00
|
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
2013-08-14 10:50:48 +00:00
|
|
|
*
|
|
|
|
* Adapted from gnome-session/gnome-session/gs-idle-monitor.c and
|
|
|
|
* from gnome-desktop/libgnome-desktop/gnome-idle-monitor.c
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* SECTION:idle-monitor
|
|
|
|
* @title: MetaIdleMonitor
|
|
|
|
* @short_description: Mutter idle counter (similar to X's IDLETIME)
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <clutter/clutter.h>
|
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/extensions/sync.h>
|
|
|
|
|
|
|
|
#include <meta/util.h>
|
|
|
|
#include <meta/main.h>
|
|
|
|
#include <meta/meta-idle-monitor.h>
|
|
|
|
#include "meta-idle-monitor-private.h"
|
2014-03-31 00:38:56 +00:00
|
|
|
#include "meta-idle-monitor-dbus.h"
|
2014-04-01 03:25:37 +00:00
|
|
|
#include "backends/x11/meta-idle-monitor-xsync.h"
|
|
|
|
#include "backends/native/meta-idle-monitor-native.h"
|
2013-08-14 10:50:48 +00:00
|
|
|
|
|
|
|
G_STATIC_ASSERT(sizeof(unsigned long) == sizeof(gpointer));
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
|
|
|
PROP_DEVICE_ID,
|
|
|
|
PROP_LAST,
|
|
|
|
};
|
|
|
|
|
|
|
|
static GParamSpec *obj_props[PROP_LAST];
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (MetaIdleMonitor, meta_idle_monitor, G_TYPE_OBJECT)
|
|
|
|
|
|
|
|
static MetaIdleMonitor *device_monitors[256];
|
|
|
|
static int device_id_max;
|
|
|
|
|
2014-03-31 00:52:25 +00:00
|
|
|
void
|
|
|
|
_meta_idle_monitor_watch_fire (MetaIdleMonitorWatch *watch)
|
2013-08-14 10:50:48 +00:00
|
|
|
{
|
|
|
|
MetaIdleMonitor *monitor;
|
2013-08-26 20:22:08 +00:00
|
|
|
guint id;
|
|
|
|
gboolean is_user_active_watch;
|
2013-08-14 10:50:48 +00:00
|
|
|
|
|
|
|
monitor = watch->monitor;
|
|
|
|
g_object_ref (monitor);
|
|
|
|
|
2013-09-16 12:27:08 +00:00
|
|
|
if (watch->idle_source_id)
|
|
|
|
{
|
|
|
|
g_source_remove (watch->idle_source_id);
|
|
|
|
watch->idle_source_id = 0;
|
|
|
|
}
|
|
|
|
|
2013-08-26 20:22:08 +00:00
|
|
|
id = watch->id;
|
|
|
|
is_user_active_watch = (watch->timeout_msec == 0);
|
|
|
|
|
2013-08-14 10:50:48 +00:00
|
|
|
if (watch->callback)
|
2013-08-26 20:22:08 +00:00
|
|
|
watch->callback (monitor, id, watch->user_data);
|
|
|
|
|
|
|
|
if (is_user_active_watch)
|
|
|
|
meta_idle_monitor_remove_watch (monitor, id);
|
2013-08-14 10:50:48 +00:00
|
|
|
|
|
|
|
g_object_unref (monitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_idle_monitor_dispose (GObject *object)
|
|
|
|
{
|
2014-03-31 00:52:25 +00:00
|
|
|
MetaIdleMonitor *monitor = META_IDLE_MONITOR (object);
|
2013-08-14 10:50:48 +00:00
|
|
|
|
|
|
|
g_clear_pointer (&monitor->watches, g_hash_table_destroy);
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (meta_idle_monitor_parent_class)->dispose (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_idle_monitor_get_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
MetaIdleMonitor *monitor = META_IDLE_MONITOR (object);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_DEVICE_ID:
|
|
|
|
g_value_set_int (value, monitor->device_id);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_idle_monitor_set_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
MetaIdleMonitor *monitor = META_IDLE_MONITOR (object);
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_DEVICE_ID:
|
|
|
|
monitor->device_id = g_value_get_int (value);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_idle_monitor_class_init (MetaIdleMonitorClass *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->dispose = meta_idle_monitor_dispose;
|
|
|
|
object_class->get_property = meta_idle_monitor_get_property;
|
|
|
|
object_class->set_property = meta_idle_monitor_set_property;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* MetaIdleMonitor:device_id:
|
|
|
|
*
|
|
|
|
* The device to listen to idletime on.
|
|
|
|
*/
|
|
|
|
obj_props[PROP_DEVICE_ID] =
|
|
|
|
g_param_spec_int ("device-id",
|
|
|
|
"Device ID",
|
|
|
|
"The device to listen to idletime on",
|
|
|
|
0, 255, 0,
|
|
|
|
G_PARAM_STATIC_STRINGS | G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY);
|
|
|
|
g_object_class_install_property (object_class, PROP_DEVICE_ID, obj_props[PROP_DEVICE_ID]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_idle_monitor_init (MetaIdleMonitor *monitor)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2014-03-31 00:52:25 +00:00
|
|
|
static GType
|
|
|
|
get_idle_monitor_type (void)
|
|
|
|
{
|
|
|
|
if (meta_is_wayland_compositor ())
|
|
|
|
return META_TYPE_IDLE_MONITOR_NATIVE;
|
|
|
|
else
|
|
|
|
return META_TYPE_IDLE_MONITOR_XSYNC;
|
|
|
|
}
|
|
|
|
|
2013-08-14 10:50:48 +00:00
|
|
|
static void
|
|
|
|
ensure_device_monitor (int device_id)
|
|
|
|
{
|
|
|
|
if (device_monitors[device_id])
|
|
|
|
return;
|
|
|
|
|
2014-03-31 00:52:25 +00:00
|
|
|
device_monitors[device_id] = g_object_new (get_idle_monitor_type (),
|
|
|
|
"device-id", device_id,
|
|
|
|
NULL);
|
2013-08-14 10:50:48 +00:00
|
|
|
device_id_max = MAX (device_id_max, device_id);
|
|
|
|
}
|
|
|
|
|
2014-03-31 00:38:56 +00:00
|
|
|
/* FIXME -- destroy device monitors at some point */
|
|
|
|
G_GNUC_UNUSED static void
|
|
|
|
destroy_device_monitor (int device_id)
|
|
|
|
{
|
|
|
|
g_clear_object (&device_monitors[device_id]);
|
|
|
|
if (device_id == device_id_max)
|
|
|
|
device_id_max--;
|
|
|
|
}
|
|
|
|
|
2013-08-14 10:50:48 +00:00
|
|
|
/**
|
|
|
|
* meta_idle_monitor_get_core:
|
|
|
|
*
|
|
|
|
* Returns: (transfer none): the #MetaIdleMonitor that tracks the server-global
|
|
|
|
* idletime for all devices. To track device-specific idletime,
|
|
|
|
* use meta_idle_monitor_get_for_device().
|
|
|
|
*/
|
|
|
|
MetaIdleMonitor *
|
|
|
|
meta_idle_monitor_get_core (void)
|
|
|
|
{
|
|
|
|
ensure_device_monitor (0);
|
|
|
|
return device_monitors[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* meta_idle_monitor_get_for_device:
|
|
|
|
* @device_id: the device to get the idle time for.
|
|
|
|
*
|
|
|
|
* Returns: (transfer none): a new #MetaIdleMonitor that tracks the
|
|
|
|
* device-specific idletime for @device. To track server-global idletime
|
|
|
|
* for all devices, use meta_idle_monitor_get_core().
|
|
|
|
*/
|
|
|
|
MetaIdleMonitor *
|
|
|
|
meta_idle_monitor_get_for_device (int device_id)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (device_id > 0 && device_id < 256, NULL);
|
|
|
|
|
|
|
|
ensure_device_monitor (device_id);
|
|
|
|
return device_monitors[device_id];
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaIdleMonitorWatch *
|
|
|
|
make_watch (MetaIdleMonitor *monitor,
|
|
|
|
guint64 timeout_msec,
|
2014-03-31 00:52:25 +00:00
|
|
|
MetaIdleMonitorWatchFunc callback,
|
|
|
|
gpointer user_data,
|
|
|
|
GDestroyNotify notify)
|
2013-08-14 10:50:48 +00:00
|
|
|
{
|
|
|
|
MetaIdleMonitorWatch *watch;
|
|
|
|
|
2014-03-31 00:52:25 +00:00
|
|
|
watch = META_IDLE_MONITOR_GET_CLASS (monitor)->make_watch (monitor,
|
|
|
|
timeout_msec,
|
|
|
|
callback,
|
|
|
|
user_data,
|
|
|
|
notify);
|
2013-08-14 10:50:48 +00:00
|
|
|
|
|
|
|
g_hash_table_insert (monitor->watches,
|
|
|
|
GUINT_TO_POINTER (watch->id),
|
|
|
|
watch);
|
|
|
|
return watch;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* meta_idle_monitor_add_idle_watch:
|
|
|
|
* @monitor: A #MetaIdleMonitor
|
|
|
|
* @interval_msec: The idletime interval, in milliseconds
|
|
|
|
* @callback: (allow-none): The callback to call when the user has
|
|
|
|
* accumulated @interval_msec milliseconds of idle time.
|
|
|
|
* @user_data: (allow-none): The user data to pass to the callback
|
|
|
|
* @notify: A #GDestroyNotify
|
|
|
|
*
|
|
|
|
* Returns: a watch id
|
|
|
|
*
|
|
|
|
* Adds a watch for a specific idle time. The callback will be called
|
|
|
|
* when the user has accumulated @interval_msec milliseconds of idle time.
|
|
|
|
* This function will return an ID that can either be passed to
|
|
|
|
* meta_idle_monitor_remove_watch(), or can be used to tell idle time
|
|
|
|
* watches apart if you have more than one.
|
|
|
|
*
|
|
|
|
* Also note that this function will only care about positive transitions
|
|
|
|
* (user's idle time exceeding a certain time). If you want to know about
|
|
|
|
* when the user has become active, use
|
|
|
|
* meta_idle_monitor_add_user_active_watch().
|
|
|
|
*/
|
|
|
|
guint
|
|
|
|
meta_idle_monitor_add_idle_watch (MetaIdleMonitor *monitor,
|
|
|
|
guint64 interval_msec,
|
|
|
|
MetaIdleMonitorWatchFunc callback,
|
|
|
|
gpointer user_data,
|
|
|
|
GDestroyNotify notify)
|
|
|
|
{
|
|
|
|
MetaIdleMonitorWatch *watch;
|
|
|
|
|
|
|
|
g_return_val_if_fail (META_IS_IDLE_MONITOR (monitor), 0);
|
|
|
|
g_return_val_if_fail (interval_msec > 0, 0);
|
|
|
|
|
|
|
|
watch = make_watch (monitor,
|
|
|
|
interval_msec,
|
|
|
|
callback,
|
|
|
|
user_data,
|
|
|
|
notify);
|
|
|
|
|
|
|
|
return watch->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* meta_idle_monitor_add_user_active_watch:
|
|
|
|
* @monitor: A #MetaIdleMonitor
|
|
|
|
* @callback: (allow-none): The callback to call when the user is
|
|
|
|
* active again.
|
|
|
|
* @user_data: (allow-none): The user data to pass to the callback
|
|
|
|
* @notify: A #GDestroyNotify
|
|
|
|
*
|
|
|
|
* Returns: a watch id
|
|
|
|
*
|
|
|
|
* Add a one-time watch to know when the user is active again.
|
|
|
|
* Note that this watch is one-time and will de-activate after the
|
|
|
|
* function is called, for efficiency purposes. It's most convenient
|
|
|
|
* to call this when an idle watch, as added by
|
|
|
|
* meta_idle_monitor_add_idle_watch(), has triggered.
|
|
|
|
*/
|
|
|
|
guint
|
|
|
|
meta_idle_monitor_add_user_active_watch (MetaIdleMonitor *monitor,
|
|
|
|
MetaIdleMonitorWatchFunc callback,
|
|
|
|
gpointer user_data,
|
|
|
|
GDestroyNotify notify)
|
|
|
|
{
|
|
|
|
MetaIdleMonitorWatch *watch;
|
|
|
|
|
|
|
|
g_return_val_if_fail (META_IS_IDLE_MONITOR (monitor), 0);
|
|
|
|
|
|
|
|
watch = make_watch (monitor,
|
|
|
|
0,
|
|
|
|
callback,
|
|
|
|
user_data,
|
|
|
|
notify);
|
|
|
|
|
|
|
|
return watch->id;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* meta_idle_monitor_remove_watch:
|
|
|
|
* @monitor: A #MetaIdleMonitor
|
|
|
|
* @id: A watch ID
|
|
|
|
*
|
|
|
|
* Removes an idle time watcher, previously added by
|
|
|
|
* meta_idle_monitor_add_idle_watch() or
|
|
|
|
* meta_idle_monitor_add_user_active_watch().
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
meta_idle_monitor_remove_watch (MetaIdleMonitor *monitor,
|
|
|
|
guint id)
|
|
|
|
{
|
|
|
|
g_return_if_fail (META_IS_IDLE_MONITOR (monitor));
|
|
|
|
|
2014-02-22 19:16:28 +00:00
|
|
|
g_object_ref (monitor);
|
2013-08-14 10:50:48 +00:00
|
|
|
g_hash_table_remove (monitor->watches,
|
|
|
|
GUINT_TO_POINTER (id));
|
2014-02-22 19:16:28 +00:00
|
|
|
g_object_unref (monitor);
|
2013-08-14 10:50:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* meta_idle_monitor_get_idletime:
|
|
|
|
* @monitor: A #MetaIdleMonitor
|
|
|
|
*
|
|
|
|
* Returns: The current idle time, in milliseconds, or -1 for not supported
|
|
|
|
*/
|
|
|
|
gint64
|
|
|
|
meta_idle_monitor_get_idletime (MetaIdleMonitor *monitor)
|
|
|
|
{
|
2014-03-31 00:52:25 +00:00
|
|
|
return META_IDLE_MONITOR_GET_CLASS (monitor)->get_idletime (monitor);
|
2013-08-23 13:07:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2014-03-31 00:52:25 +00:00
|
|
|
meta_idle_monitor_xsync_handle_xevent_all (XEvent *xevent)
|
2013-08-23 13:07:57 +00:00
|
|
|
{
|
2014-03-31 00:52:25 +00:00
|
|
|
int i;
|
2013-08-23 13:07:57 +00:00
|
|
|
|
2014-04-01 20:05:20 +00:00
|
|
|
if (meta_is_wayland_compositor ())
|
2014-03-31 22:26:18 +00:00
|
|
|
return;
|
2013-08-14 10:50:48 +00:00
|
|
|
|
2014-03-31 00:52:25 +00:00
|
|
|
for (i = 0; i <= device_id_max; i++)
|
|
|
|
if (device_monitors[i])
|
|
|
|
meta_idle_monitor_xsync_handle_xevent (device_monitors[i], (XSyncAlarmNotifyEvent*)xevent);
|
2013-08-14 10:50:48 +00:00
|
|
|
}
|