2014-04-01 03:00:07 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 Red Hat
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
|
|
|
* Written by:
|
|
|
|
* Jasper St. Pierre <jstpierre@mecheye.net>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include "meta-backend.h"
|
2014-04-21 23:03:22 +00:00
|
|
|
#include "meta-backend-private.h"
|
2014-04-01 03:19:13 +00:00
|
|
|
#include <meta/main.h>
|
2014-04-01 03:00:07 +00:00
|
|
|
|
|
|
|
#include <gdk/gdkx.h>
|
|
|
|
#include <clutter/clutter.h>
|
|
|
|
#include <clutter/x11/clutter-x11.h>
|
|
|
|
|
2014-04-01 03:25:37 +00:00
|
|
|
#include "backends/native/meta-weston-launch.h"
|
2014-04-01 18:04:53 +00:00
|
|
|
#include <meta/util.h>
|
2014-04-01 03:19:13 +00:00
|
|
|
|
2014-04-21 23:03:22 +00:00
|
|
|
#include "backends/x11/meta-idle-monitor-xsync.h"
|
|
|
|
#include "backends/native/meta-idle-monitor-native.h"
|
|
|
|
|
|
|
|
static MetaBackend *_backend;
|
|
|
|
|
|
|
|
MetaBackend *
|
|
|
|
meta_get_backend (void)
|
|
|
|
{
|
|
|
|
return _backend;
|
|
|
|
}
|
|
|
|
|
|
|
|
G_DEFINE_TYPE (MetaBackend, meta_backend, G_TYPE_OBJECT);
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_backend_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = META_BACKEND (object);
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i <= backend->device_id_max; i++)
|
|
|
|
{
|
|
|
|
if (backend->device_monitors[i])
|
|
|
|
g_object_unref (backend->device_monitors[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (meta_backend_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_backend_class_init (MetaBackendClass *klass)
|
|
|
|
{
|
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->finalize = meta_backend_finalize;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_backend_init (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
_backend = backend;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GType
|
|
|
|
get_idle_monitor_type (void)
|
|
|
|
{
|
|
|
|
#if defined(CLUTTER_WINDOWING_X11)
|
|
|
|
if (clutter_check_windowing_backend (CLUTTER_WINDOWING_X11))
|
|
|
|
return META_TYPE_IDLE_MONITOR_XSYNC;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return META_TYPE_IDLE_MONITOR_NATIVE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* FIXME -- destroy device monitors at some point */
|
|
|
|
G_GNUC_UNUSED static void
|
|
|
|
destroy_device_monitor (MetaBackend *backend,
|
|
|
|
int device_id)
|
|
|
|
{
|
|
|
|
g_clear_object (&backend->device_monitors[device_id]);
|
|
|
|
if (device_id == backend->device_id_max)
|
|
|
|
backend->device_id_max--;
|
|
|
|
}
|
|
|
|
|
|
|
|
MetaIdleMonitor *
|
|
|
|
meta_backend_get_idle_monitor (MetaBackend *backend,
|
|
|
|
int device_id)
|
|
|
|
{
|
|
|
|
g_return_val_if_fail (device_id >= 0 && device_id < 256, NULL);
|
|
|
|
|
|
|
|
if (!backend->device_monitors[device_id])
|
|
|
|
{
|
|
|
|
backend->device_monitors[device_id] = g_object_new (get_idle_monitor_type (),
|
|
|
|
"device-id", device_id,
|
|
|
|
NULL);
|
|
|
|
backend->device_id_max = MAX (backend->device_id_max, device_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
return backend->device_monitors[device_id];
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_backend_x11_handle_alarm_notify (MetaBackend *backend,
|
|
|
|
XEvent *xevent)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i <= backend->device_id_max; i++)
|
|
|
|
{
|
|
|
|
if (backend->device_monitors[i])
|
|
|
|
{
|
|
|
|
if (!META_IS_IDLE_MONITOR_XSYNC (backend->device_monitors[i]))
|
|
|
|
return;
|
|
|
|
|
|
|
|
meta_idle_monitor_xsync_handle_xevent (backend->device_monitors[i], (XSyncAlarmNotifyEvent*)xevent);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static GType
|
|
|
|
get_backend_type (void)
|
|
|
|
{
|
|
|
|
return META_TYPE_BACKEND;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_create_backend (void)
|
|
|
|
{
|
|
|
|
/* The initializer above installs it in _backend so meta_get_backend
|
|
|
|
* is valid during initialization. */
|
|
|
|
g_object_new (get_backend_type (), NULL);
|
|
|
|
}
|
|
|
|
|
2014-04-01 03:00:07 +00:00
|
|
|
/* Mutter is responsible for pulling events off the X queue, so Clutter
|
|
|
|
* doesn't need (and shouldn't) run its normal event source which polls
|
|
|
|
* the X fd, but we do have to deal with dispatching events that accumulate
|
|
|
|
* in the clutter queue. This happens, for example, when clutter generate
|
|
|
|
* enter/leave events on mouse motion - several events are queued in the
|
|
|
|
* clutter queue but only one dispatched. It could also happen because of
|
|
|
|
* explicit calls to clutter_event_put(). We add a very simple custom
|
|
|
|
* event loop source which is simply responsible for pulling events off
|
|
|
|
* of the queue and dispatching them before we block for new events.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
event_prepare (GSource *source,
|
|
|
|
gint *timeout_)
|
|
|
|
{
|
|
|
|
*timeout_ = -1;
|
|
|
|
|
|
|
|
return clutter_events_pending ();
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
event_check (GSource *source)
|
|
|
|
{
|
|
|
|
return clutter_events_pending ();
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
event_dispatch (GSource *source,
|
|
|
|
GSourceFunc callback,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
ClutterEvent *event = clutter_event_get ();
|
|
|
|
|
|
|
|
if (event)
|
|
|
|
{
|
|
|
|
clutter_do_event (event);
|
|
|
|
clutter_event_free (event);
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GSourceFuncs event_funcs = {
|
|
|
|
event_prepare,
|
|
|
|
event_check,
|
|
|
|
event_dispatch
|
|
|
|
};
|
|
|
|
|
2014-04-01 03:19:13 +00:00
|
|
|
static MetaLauncher *launcher;
|
|
|
|
|
2014-04-01 03:00:07 +00:00
|
|
|
void
|
|
|
|
meta_clutter_init (void)
|
|
|
|
{
|
|
|
|
GSource *source;
|
|
|
|
|
2014-04-21 23:03:22 +00:00
|
|
|
meta_create_backend ();
|
|
|
|
|
2014-04-01 17:56:40 +00:00
|
|
|
/* When running as an X11 compositor, we install our own event filter and
|
|
|
|
* pass events to Clutter explicitly, so we need to prevent Clutter from
|
|
|
|
* handling our events.
|
|
|
|
*
|
|
|
|
* However, when running as a Wayland compostior under X11 nested, Clutter
|
|
|
|
* Clutter needs to see events related to its own window. We need to
|
|
|
|
* eventually replace this with a proper frontend / backend split: Clutter
|
|
|
|
* under nested is connecting to the "host X server" to get its events it
|
|
|
|
* needs to put up a window, and GTK+ is connecting to the "inner X server".
|
|
|
|
* The two would the same in the X11 compositor case, but not when running
|
|
|
|
* XWayland as a Wayland compositor.
|
|
|
|
*/
|
|
|
|
if (!meta_is_wayland_compositor ())
|
|
|
|
{
|
|
|
|
clutter_x11_set_display (GDK_DISPLAY_XDISPLAY (gdk_display_get_default ()));
|
|
|
|
clutter_x11_disable_event_retrieval ();
|
|
|
|
}
|
2014-04-01 03:00:07 +00:00
|
|
|
|
2014-04-01 03:19:13 +00:00
|
|
|
/* If we're running on bare metal, we're a display server,
|
|
|
|
* so start talking to weston-launch. */
|
|
|
|
#if defined(CLUTTER_WINDOWING_EGL)
|
|
|
|
if (clutter_check_windowing_backend (CLUTTER_WINDOWING_EGL))
|
|
|
|
launcher = meta_launcher_new ();
|
|
|
|
#endif
|
|
|
|
|
2014-04-01 03:00:07 +00:00
|
|
|
if (clutter_init (NULL, NULL) != CLUTTER_INIT_SUCCESS)
|
|
|
|
g_error ("Unable to initialize Clutter.\n");
|
|
|
|
|
|
|
|
source = g_source_new (&event_funcs, sizeof (GSource));
|
|
|
|
g_source_attach (source, NULL);
|
|
|
|
g_source_unref (source);
|
|
|
|
}
|
2014-04-01 03:19:13 +00:00
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_activate_vt (int vt, GError **error)
|
|
|
|
{
|
|
|
|
if (launcher)
|
|
|
|
return meta_launcher_activate_vt (launcher, vt, error);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_debug ("Ignoring VT switch keybinding, not running as display server");
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* meta_activate_session:
|
|
|
|
*
|
|
|
|
* Tells mutter to activate the session. When mutter is a
|
|
|
|
* Wayland compositor, this tells logind to switch over to
|
|
|
|
* the new session.
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
meta_activate_session (void)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
if (!meta_launcher_activate_vt (launcher, -1, &error))
|
|
|
|
{
|
|
|
|
g_warning ("Could not activate session: %s\n", error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|