2014-04-21 23:13:04 +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-x11.h"
|
|
|
|
|
2014-04-21 23:47:04 +00:00
|
|
|
#include <clutter/x11/clutter-x11.h>
|
|
|
|
|
2014-04-22 14:14:59 +00:00
|
|
|
#include <X11/extensions/sync.h>
|
|
|
|
|
2014-04-21 23:13:04 +00:00
|
|
|
#include "meta-idle-monitor-xsync.h"
|
2014-04-21 23:41:11 +00:00
|
|
|
#include "meta-monitor-manager-xrandr.h"
|
|
|
|
#include "backends/meta-monitor-manager-dummy.h"
|
2014-04-21 23:13:04 +00:00
|
|
|
|
2014-04-23 14:50:07 +00:00
|
|
|
#include "meta-cursor-tracker-private.h"
|
|
|
|
#include "meta-cursor.h"
|
|
|
|
#include <meta/util.h>
|
2014-04-23 18:50:27 +00:00
|
|
|
#include "display-private.h"
|
|
|
|
#include "compositor/compositor-private.h"
|
2014-04-23 14:50:07 +00:00
|
|
|
|
2014-04-22 14:14:59 +00:00
|
|
|
struct _MetaBackendX11Private
|
|
|
|
{
|
|
|
|
/* The host X11 display */
|
|
|
|
Display *xdisplay;
|
|
|
|
GSource *source;
|
|
|
|
|
|
|
|
int xsync_event_base;
|
|
|
|
int xsync_error_base;
|
2014-04-23 17:59:58 +00:00
|
|
|
|
|
|
|
int xinput_opcode;
|
|
|
|
int xinput_event_base;
|
|
|
|
int xinput_error_base;
|
2014-04-22 14:14:59 +00:00
|
|
|
};
|
|
|
|
typedef struct _MetaBackendX11Private MetaBackendX11Private;
|
|
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (MetaBackendX11, meta_backend_x11, META_TYPE_BACKEND);
|
|
|
|
|
|
|
|
static void
|
|
|
|
handle_alarm_notify (MetaBackend *backend,
|
2014-04-23 19:19:08 +00:00
|
|
|
XEvent *event)
|
2014-04-22 14:14:59 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i <= backend->device_id_max; i++)
|
|
|
|
if (backend->device_monitors[i])
|
2014-04-23 19:19:08 +00:00
|
|
|
meta_idle_monitor_xsync_handle_xevent (backend->device_monitors[i], (XSyncAlarmNotifyEvent*) event);
|
2014-04-22 14:14:59 +00:00
|
|
|
}
|
|
|
|
|
2014-04-28 17:22:17 +00:00
|
|
|
static Window
|
|
|
|
get_stage_window (MetaBackendX11 *x11)
|
2014-04-24 16:06:32 +00:00
|
|
|
{
|
|
|
|
MetaDisplay *display = meta_get_display ();
|
|
|
|
MetaCompositor *compositor = display->compositor;
|
|
|
|
ClutterStage *stage = CLUTTER_STAGE (compositor->stage);
|
|
|
|
|
2014-04-28 17:22:17 +00:00
|
|
|
return clutter_x11_get_stage_window (stage);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
translate_device_event (MetaBackendX11 *x11,
|
|
|
|
XIDeviceEvent *device_event)
|
|
|
|
{
|
|
|
|
device_event->event = get_stage_window (x11);
|
2014-04-24 16:06:32 +00:00
|
|
|
device_event->event_x = device_event->root_x;
|
|
|
|
device_event->event_y = device_event->root_y;
|
|
|
|
}
|
|
|
|
|
2014-04-23 17:59:58 +00:00
|
|
|
/* Clutter makes the assumption that there is only one X window
|
|
|
|
* per stage, which is a valid assumption to make for a generic
|
|
|
|
* application toolkit. As such, it will ignore any events sent
|
|
|
|
* to the a stage that isn't its X window.
|
|
|
|
*
|
|
|
|
* When running as an X window manager, we need to respond to
|
|
|
|
* events from lots of windows. Trick Clutter into translating
|
|
|
|
* these events by pretending we got an event on the stage window.
|
|
|
|
*/
|
|
|
|
static void
|
|
|
|
maybe_spoof_event_as_stage_event (MetaBackendX11 *x11,
|
|
|
|
XEvent *event)
|
|
|
|
{
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
if (event->type == GenericEvent &&
|
|
|
|
event->xcookie.extension == priv->xinput_opcode)
|
|
|
|
{
|
|
|
|
XIEvent *input_event = (XIEvent *) event->xcookie.data;
|
|
|
|
|
|
|
|
switch (input_event->evtype)
|
|
|
|
{
|
|
|
|
case XI_Motion:
|
|
|
|
case XI_ButtonPress:
|
|
|
|
case XI_ButtonRelease:
|
|
|
|
case XI_KeyPress:
|
|
|
|
case XI_KeyRelease:
|
2014-04-24 16:06:32 +00:00
|
|
|
translate_device_event (x11, (XIDeviceEvent *) input_event);
|
|
|
|
break;
|
2014-04-23 17:59:58 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-22 14:14:59 +00:00
|
|
|
static void
|
|
|
|
handle_host_xevent (MetaBackend *backend,
|
2014-04-23 19:19:08 +00:00
|
|
|
XEvent *event)
|
2014-04-22 14:14:59 +00:00
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
2014-04-23 16:05:14 +00:00
|
|
|
gboolean bypass_clutter = FALSE;
|
2014-04-22 14:14:59 +00:00
|
|
|
|
2014-04-23 19:17:21 +00:00
|
|
|
XGetEventData (priv->xdisplay, &event->xcookie);
|
|
|
|
|
2014-04-23 19:19:08 +00:00
|
|
|
if (event->type == (priv->xsync_event_base + XSyncAlarmNotify))
|
|
|
|
handle_alarm_notify (backend, event);
|
2014-04-22 14:14:59 +00:00
|
|
|
|
2014-04-23 16:05:14 +00:00
|
|
|
{
|
|
|
|
MetaMonitorManager *manager = meta_backend_get_monitor_manager (backend);
|
2014-04-23 16:07:38 +00:00
|
|
|
if (META_IS_MONITOR_MANAGER_XRANDR (manager) &&
|
2014-04-23 19:19:08 +00:00
|
|
|
meta_monitor_manager_xrandr_handle_xevent (META_MONITOR_MANAGER_XRANDR (manager), event))
|
2014-04-23 16:05:14 +00:00
|
|
|
{
|
|
|
|
bypass_clutter = TRUE;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-23 19:19:08 +00:00
|
|
|
maybe_spoof_event_as_stage_event (x11, event);
|
2014-04-23 17:59:58 +00:00
|
|
|
|
2014-04-23 16:05:14 +00:00
|
|
|
out:
|
|
|
|
if (!bypass_clutter)
|
2014-04-23 19:19:08 +00:00
|
|
|
clutter_x11_handle_event (event);
|
2014-04-23 19:17:21 +00:00
|
|
|
|
|
|
|
XFreeEventData (priv->xdisplay, &event->xcookie);
|
2014-04-22 14:14:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
GSource base;
|
|
|
|
GPollFD event_poll_fd;
|
|
|
|
MetaBackend *backend;
|
|
|
|
} XEventSource;
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
x_event_source_prepare (GSource *source,
|
|
|
|
int *timeout)
|
|
|
|
{
|
|
|
|
XEventSource *x_source = (XEventSource *) source;
|
|
|
|
MetaBackend *backend = x_source->backend;
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
*timeout = -1;
|
|
|
|
|
|
|
|
return XPending (priv->xdisplay);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
x_event_source_check (GSource *source)
|
|
|
|
{
|
|
|
|
XEventSource *x_source = (XEventSource *) source;
|
|
|
|
MetaBackend *backend = x_source->backend;
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
return XPending (priv->xdisplay);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
x_event_source_dispatch (GSource *source,
|
|
|
|
GSourceFunc callback,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
XEventSource *x_source = (XEventSource *) source;
|
|
|
|
MetaBackend *backend = x_source->backend;
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
while (XPending (priv->xdisplay))
|
|
|
|
{
|
2014-04-23 19:19:08 +00:00
|
|
|
XEvent event;
|
2014-04-22 14:14:59 +00:00
|
|
|
|
2014-04-23 19:19:08 +00:00
|
|
|
XNextEvent (priv->xdisplay, &event);
|
2014-04-22 14:14:59 +00:00
|
|
|
|
2014-04-23 19:19:08 +00:00
|
|
|
handle_host_xevent (backend, &event);
|
2014-04-22 14:14:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GSourceFuncs x_event_funcs = {
|
|
|
|
x_event_source_prepare,
|
|
|
|
x_event_source_check,
|
|
|
|
x_event_source_dispatch,
|
|
|
|
};
|
|
|
|
|
|
|
|
static GSource *
|
|
|
|
x_event_source_new (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
GSource *source;
|
|
|
|
XEventSource *x_source;
|
|
|
|
|
|
|
|
source = g_source_new (&x_event_funcs, sizeof (XEventSource));
|
|
|
|
x_source = (XEventSource *) source;
|
|
|
|
x_source->backend = backend;
|
|
|
|
x_source->event_poll_fd.fd = ConnectionNumber (priv->xdisplay);
|
|
|
|
x_source->event_poll_fd.events = G_IO_IN;
|
|
|
|
g_source_add_poll (source, &x_source->event_poll_fd);
|
|
|
|
|
|
|
|
g_source_attach (source, NULL);
|
|
|
|
return source;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_backend_x11_post_init (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
int major, minor;
|
|
|
|
|
|
|
|
priv->xdisplay = clutter_x11_get_default_display ();
|
|
|
|
|
|
|
|
priv->source = x_event_source_new (backend);
|
|
|
|
|
2014-04-22 16:45:13 +00:00
|
|
|
if (!XSyncQueryExtension (priv->xdisplay, &priv->xsync_event_base, &priv->xsync_error_base) ||
|
|
|
|
!XSyncInitialize (priv->xdisplay, &major, &minor))
|
2014-04-22 14:14:59 +00:00
|
|
|
meta_fatal ("Could not initialize XSync");
|
|
|
|
|
2014-04-23 17:59:58 +00:00
|
|
|
{
|
|
|
|
int major = 2, minor = 3;
|
|
|
|
gboolean has_xi = FALSE;
|
|
|
|
|
|
|
|
if (XQueryExtension (priv->xdisplay,
|
|
|
|
"XInputExtension",
|
|
|
|
&priv->xinput_opcode,
|
|
|
|
&priv->xinput_error_base,
|
|
|
|
&priv->xinput_event_base))
|
|
|
|
{
|
|
|
|
if (XIQueryVersion (priv->xdisplay, &major, &minor) == Success)
|
|
|
|
{
|
|
|
|
int version = (major * 10) + minor;
|
|
|
|
if (version >= 22)
|
|
|
|
has_xi = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!has_xi)
|
|
|
|
meta_fatal ("X server doesn't have the XInput extension, version 2.2 or newer\n");
|
|
|
|
}
|
|
|
|
|
2014-04-22 14:14:59 +00:00
|
|
|
META_BACKEND_CLASS (meta_backend_x11_parent_class)->post_init (backend);
|
|
|
|
}
|
2014-04-21 23:13:04 +00:00
|
|
|
|
|
|
|
static MetaIdleMonitor *
|
|
|
|
meta_backend_x11_create_idle_monitor (MetaBackend *backend,
|
|
|
|
int device_id)
|
|
|
|
{
|
|
|
|
return g_object_new (META_TYPE_IDLE_MONITOR_XSYNC,
|
|
|
|
"device-id", device_id,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
2014-04-21 23:41:11 +00:00
|
|
|
static MetaMonitorManager *
|
|
|
|
meta_backend_x11_create_monitor_manager (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
/* If we're a Wayland compositor using the X11 backend,
|
|
|
|
* we're a nested configuration, so return the dummy
|
|
|
|
* monitor setup. */
|
|
|
|
if (meta_is_wayland_compositor ())
|
|
|
|
return g_object_new (META_TYPE_MONITOR_MANAGER_DUMMY, NULL);
|
|
|
|
|
|
|
|
return g_object_new (META_TYPE_MONITOR_MANAGER_XRANDR, NULL);
|
|
|
|
}
|
|
|
|
|
2014-04-23 14:50:07 +00:00
|
|
|
static gboolean
|
|
|
|
meta_backend_x11_grab_device (MetaBackend *backend,
|
|
|
|
int device_id,
|
|
|
|
uint32_t timestamp)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
unsigned char mask_bits[XIMaskLen (XI_LASTEVENT)] = { 0 };
|
|
|
|
XIEventMask mask = { XIAllMasterDevices, sizeof (mask_bits), mask_bits };
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
XISetMask (mask.mask, XI_ButtonPress);
|
|
|
|
XISetMask (mask.mask, XI_ButtonRelease);
|
|
|
|
XISetMask (mask.mask, XI_Enter);
|
|
|
|
XISetMask (mask.mask, XI_Leave);
|
|
|
|
XISetMask (mask.mask, XI_Motion);
|
2014-04-23 17:36:11 +00:00
|
|
|
XISetMask (mask.mask, XI_KeyPress);
|
|
|
|
XISetMask (mask.mask, XI_KeyRelease);
|
2014-04-23 14:50:07 +00:00
|
|
|
|
|
|
|
MetaCursorTracker *tracker = meta_cursor_tracker_get_for_screen (NULL);
|
|
|
|
MetaCursorReference *cursor_ref = meta_cursor_tracker_get_displayed_cursor (tracker);
|
|
|
|
MetaCursor cursor = meta_cursor_reference_get_meta_cursor (cursor_ref);
|
|
|
|
|
|
|
|
ret = XIGrabDevice (priv->xdisplay, device_id,
|
2014-04-28 17:22:17 +00:00
|
|
|
get_stage_window (x11),
|
2014-04-23 14:50:07 +00:00
|
|
|
timestamp,
|
|
|
|
meta_cursor_create_x_cursor (priv->xdisplay, cursor),
|
|
|
|
XIGrabModeAsync, XIGrabModeAsync,
|
|
|
|
False, /* owner_events */
|
|
|
|
&mask);
|
|
|
|
|
|
|
|
return (ret == Success);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
meta_backend_x11_ungrab_device (MetaBackend *backend,
|
|
|
|
int device_id,
|
|
|
|
uint32_t timestamp)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = XIUngrabDevice (priv->xdisplay, device_id, timestamp);
|
|
|
|
|
|
|
|
return (ret == Success);
|
|
|
|
}
|
|
|
|
|
2014-04-21 23:13:04 +00:00
|
|
|
static void
|
|
|
|
meta_backend_x11_class_init (MetaBackendX11Class *klass)
|
|
|
|
{
|
|
|
|
MetaBackendClass *backend_class = META_BACKEND_CLASS (klass);
|
|
|
|
|
2014-04-22 14:14:59 +00:00
|
|
|
backend_class->post_init = meta_backend_x11_post_init;
|
2014-04-21 23:13:04 +00:00
|
|
|
backend_class->create_idle_monitor = meta_backend_x11_create_idle_monitor;
|
2014-04-21 23:41:11 +00:00
|
|
|
backend_class->create_monitor_manager = meta_backend_x11_create_monitor_manager;
|
2014-04-23 14:50:07 +00:00
|
|
|
|
|
|
|
backend_class->grab_device = meta_backend_x11_grab_device;
|
|
|
|
backend_class->ungrab_device = meta_backend_x11_ungrab_device;
|
2014-04-21 23:13:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_backend_x11_init (MetaBackendX11 *x11)
|
|
|
|
{
|
2014-04-22 14:14:59 +00:00
|
|
|
/* We do X11 event retrieval ourselves */
|
|
|
|
clutter_x11_disable_event_retrieval ();
|
2014-04-21 23:13:04 +00:00
|
|
|
}
|
2014-04-23 13:57:16 +00:00
|
|
|
|
|
|
|
Display *
|
|
|
|
meta_backend_x11_get_xdisplay (MetaBackendX11 *x11)
|
|
|
|
{
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
return priv->xdisplay;
|
|
|
|
}
|
|
|
|
|