2014-04-21 19:13:04 -04: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"
|
|
|
|
|
2014-08-04 16:47:35 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
2014-04-21 19:13:04 -04:00
|
|
|
#include "meta-backend-x11.h"
|
|
|
|
|
2014-04-21 19:47:04 -04:00
|
|
|
#include <clutter/x11/clutter-x11.h>
|
|
|
|
|
2014-04-22 10:14:59 -04:00
|
|
|
#include <X11/extensions/sync.h>
|
2014-08-04 16:47:35 +02:00
|
|
|
#include <X11/XKBlib.h>
|
|
|
|
#include <X11/extensions/XKBrules.h>
|
|
|
|
#include <X11/Xlib-xcb.h>
|
|
|
|
#include <xkbcommon/xkbcommon-x11.h>
|
2014-04-22 10:14:59 -04:00
|
|
|
|
2014-04-21 19:13:04 -04:00
|
|
|
#include "meta-idle-monitor-xsync.h"
|
2014-04-21 19:41:11 -04:00
|
|
|
#include "meta-monitor-manager-xrandr.h"
|
|
|
|
#include "backends/meta-monitor-manager-dummy.h"
|
2014-04-27 10:49:29 -04:00
|
|
|
#include "meta-cursor-renderer-x11.h"
|
2014-04-21 19:13:04 -04:00
|
|
|
|
2014-04-23 10:50:07 -04:00
|
|
|
#include <meta/util.h>
|
2014-04-23 14:50:27 -04:00
|
|
|
#include "display-private.h"
|
|
|
|
#include "compositor/compositor-private.h"
|
2014-04-23 10:50:07 -04:00
|
|
|
|
2015-06-16 08:02:13 -07:00
|
|
|
typedef enum {
|
|
|
|
/* We're a traditional CM running under the host. */
|
|
|
|
META_BACKEND_X11_MODE_COMPOSITOR,
|
|
|
|
|
|
|
|
/* We're a nested X11 client */
|
|
|
|
META_BACKEND_X11_MODE_NESTED,
|
|
|
|
} MetaBackendX11Mode;
|
|
|
|
|
2014-04-22 10:14:59 -04:00
|
|
|
struct _MetaBackendX11Private
|
|
|
|
{
|
|
|
|
/* The host X11 display */
|
|
|
|
Display *xdisplay;
|
2014-08-04 16:47:35 +02:00
|
|
|
xcb_connection_t *xcb;
|
2014-04-22 10:14:59 -04:00
|
|
|
GSource *source;
|
|
|
|
|
2015-06-16 08:02:13 -07:00
|
|
|
MetaBackendX11Mode mode;
|
|
|
|
|
2014-04-22 10:14:59 -04:00
|
|
|
int xsync_event_base;
|
|
|
|
int xsync_error_base;
|
2014-04-23 13:59:58 -04:00
|
|
|
|
|
|
|
int xinput_opcode;
|
|
|
|
int xinput_event_base;
|
|
|
|
int xinput_error_base;
|
2014-06-19 23:16:50 +02:00
|
|
|
Time latest_evtime;
|
2014-08-04 16:47:35 +02:00
|
|
|
|
|
|
|
uint8_t xkb_event_base;
|
|
|
|
uint8_t xkb_error_base;
|
2014-08-14 17:23:20 -04:00
|
|
|
|
|
|
|
struct xkb_keymap *keymap;
|
2014-10-02 19:07:21 +02:00
|
|
|
gchar *keymap_layouts;
|
|
|
|
gchar *keymap_variants;
|
|
|
|
gchar *keymap_options;
|
2014-04-22 10:14:59 -04:00
|
|
|
};
|
|
|
|
typedef struct _MetaBackendX11Private MetaBackendX11Private;
|
|
|
|
|
2014-10-02 19:07:21 +02:00
|
|
|
static void apply_keymap (MetaBackendX11 *x11);
|
|
|
|
|
2014-04-22 10:14:59 -04:00
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (MetaBackendX11, meta_backend_x11, META_TYPE_BACKEND);
|
|
|
|
|
|
|
|
static void
|
|
|
|
handle_alarm_notify (MetaBackend *backend,
|
2014-04-23 15:19:08 -04:00
|
|
|
XEvent *event)
|
2014-04-22 10:14:59 -04:00
|
|
|
{
|
2014-12-29 18:57:09 -08:00
|
|
|
GHashTableIter iter;
|
|
|
|
gpointer value;
|
2014-04-22 10:14:59 -04:00
|
|
|
|
2014-12-29 18:57:09 -08:00
|
|
|
g_hash_table_iter_init (&iter, backend->device_monitors);
|
|
|
|
while (g_hash_table_iter_next (&iter, NULL, &value))
|
|
|
|
{
|
|
|
|
MetaIdleMonitor *device_monitor = META_IDLE_MONITOR (value);
|
|
|
|
meta_idle_monitor_xsync_handle_xevent (device_monitor, (XSyncAlarmNotifyEvent*) event);
|
|
|
|
}
|
2014-04-22 10:14:59 -04:00
|
|
|
}
|
|
|
|
|
2014-04-28 13:22:17 -04:00
|
|
|
static void
|
|
|
|
translate_device_event (MetaBackendX11 *x11,
|
|
|
|
XIDeviceEvent *device_event)
|
|
|
|
{
|
2014-06-19 23:16:50 +02:00
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
2014-04-27 10:49:29 -04:00
|
|
|
Window stage_window = meta_backend_x11_get_xwindow (x11);
|
2014-04-28 13:27:02 -04:00
|
|
|
|
|
|
|
if (device_event->event != stage_window)
|
|
|
|
{
|
|
|
|
/* This codepath should only ever trigger as an X11 compositor,
|
|
|
|
* and never under nested, as under nested all backend events
|
|
|
|
* should be reported with respect to the stage window. */
|
2015-06-16 08:02:13 -07:00
|
|
|
g_assert (priv->mode == META_BACKEND_X11_MODE_COMPOSITOR);
|
2014-04-28 13:27:02 -04:00
|
|
|
|
|
|
|
device_event->event = stage_window;
|
|
|
|
|
|
|
|
/* As an X11 compositor, the stage window is always at 0,0, so
|
|
|
|
* using root coordinates will give us correct stage coordinates
|
|
|
|
* as well... */
|
|
|
|
device_event->event_x = device_event->root_x;
|
|
|
|
device_event->event_y = device_event->root_y;
|
|
|
|
}
|
2014-06-19 23:16:50 +02:00
|
|
|
|
|
|
|
if (!device_event->send_event && device_event->time != CurrentTime)
|
|
|
|
{
|
|
|
|
if (device_event->time < priv->latest_evtime)
|
|
|
|
{
|
|
|
|
/* Emulated pointer events received after XIRejectTouch is received
|
|
|
|
* on a passive touch grab will contain older timestamps, update those
|
|
|
|
* so we dont get InvalidTime at grabs.
|
|
|
|
*/
|
|
|
|
device_event->time = priv->latest_evtime;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Update the internal latest evtime, for any possible later use */
|
|
|
|
priv->latest_evtime = device_event->time;
|
|
|
|
}
|
2014-04-24 12:06:32 -04:00
|
|
|
}
|
|
|
|
|
2015-01-19 21:12:26 -08:00
|
|
|
static void
|
|
|
|
translate_crossing_event (MetaBackendX11 *x11,
|
|
|
|
XIEnterEvent *enter_event)
|
|
|
|
{
|
2015-06-24 10:30:10 -07:00
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
2015-01-19 21:12:26 -08:00
|
|
|
/* Throw out weird events generated by grabs. */
|
|
|
|
if (enter_event->mode == XINotifyGrab ||
|
|
|
|
enter_event->mode == XINotifyUngrab)
|
|
|
|
{
|
|
|
|
enter_event->event = None;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-06-24 10:30:10 -07:00
|
|
|
Window stage_window = meta_backend_x11_get_xwindow (x11);
|
2015-08-10 15:42:35 +02:00
|
|
|
if (enter_event->event != stage_window &&
|
|
|
|
priv->mode == META_BACKEND_X11_MODE_COMPOSITOR)
|
2015-06-24 10:30:10 -07:00
|
|
|
{
|
|
|
|
enter_event->event = meta_backend_x11_get_xwindow (x11);
|
|
|
|
enter_event->event_x = enter_event->root_x;
|
|
|
|
enter_event->event_y = enter_event->root_y;
|
|
|
|
}
|
2015-01-19 21:12:26 -08:00
|
|
|
}
|
|
|
|
|
2015-02-11 15:04:04 +01:00
|
|
|
static void
|
|
|
|
handle_device_change (MetaBackendX11 *x11,
|
|
|
|
XIEvent *event)
|
|
|
|
{
|
|
|
|
XIDeviceChangedEvent *device_changed;
|
|
|
|
|
|
|
|
if (event->evtype != XI_DeviceChanged)
|
|
|
|
return;
|
|
|
|
|
|
|
|
device_changed = (XIDeviceChangedEvent *) event;
|
|
|
|
|
|
|
|
if (device_changed->reason != XISlaveSwitch)
|
|
|
|
return;
|
|
|
|
|
|
|
|
meta_backend_update_last_device (META_BACKEND (x11),
|
|
|
|
device_changed->sourceid);
|
|
|
|
}
|
|
|
|
|
2014-04-23 13:59:58 -04: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,
|
2015-02-11 15:04:04 +01:00
|
|
|
XIEvent *input_event)
|
|
|
|
{
|
|
|
|
switch (input_event->evtype)
|
|
|
|
{
|
|
|
|
case XI_Motion:
|
|
|
|
case XI_ButtonPress:
|
|
|
|
case XI_ButtonRelease:
|
|
|
|
case XI_KeyPress:
|
|
|
|
case XI_KeyRelease:
|
|
|
|
case XI_TouchBegin:
|
|
|
|
case XI_TouchUpdate:
|
|
|
|
case XI_TouchEnd:
|
|
|
|
translate_device_event (x11, (XIDeviceEvent *) input_event);
|
|
|
|
break;
|
|
|
|
case XI_Enter:
|
|
|
|
case XI_Leave:
|
|
|
|
translate_crossing_event (x11, (XIEnterEvent *) input_event);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
handle_input_event (MetaBackendX11 *x11,
|
|
|
|
XEvent *event)
|
2014-04-23 13:59:58 -04:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
|
2015-02-11 15:04:04 +01:00
|
|
|
if (input_event->evtype == XI_DeviceChanged)
|
|
|
|
handle_device_change (x11, input_event);
|
|
|
|
else
|
|
|
|
maybe_spoof_event_as_stage_event (x11, input_event);
|
2014-04-23 13:59:58 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-14 17:32:41 -04:00
|
|
|
static void
|
|
|
|
keymap_changed (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
if (priv->keymap)
|
|
|
|
{
|
|
|
|
xkb_keymap_unref (priv->keymap);
|
|
|
|
priv->keymap = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_signal_emit_by_name (backend, "keymap-changed", 0);
|
|
|
|
}
|
|
|
|
|
2014-04-22 10:14:59 -04:00
|
|
|
static void
|
|
|
|
handle_host_xevent (MetaBackend *backend,
|
2014-04-23 15:19:08 -04:00
|
|
|
XEvent *event)
|
2014-04-22 10:14:59 -04:00
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
2014-04-23 12:05:14 -04:00
|
|
|
gboolean bypass_clutter = FALSE;
|
2014-04-22 10:14:59 -04:00
|
|
|
|
2014-04-23 15:17:21 -04:00
|
|
|
XGetEventData (priv->xdisplay, &event->xcookie);
|
|
|
|
|
2014-07-27 16:14:46 +02:00
|
|
|
{
|
|
|
|
MetaDisplay *display = meta_get_display ();
|
|
|
|
|
|
|
|
if (display)
|
|
|
|
{
|
|
|
|
MetaCompositor *compositor = display->compositor;
|
|
|
|
if (meta_plugin_manager_xevent_filter (compositor->plugin_mgr, event))
|
|
|
|
bypass_clutter = TRUE;
|
|
|
|
}
|
|
|
|
}
|
2014-07-19 18:45:47 +02:00
|
|
|
|
2014-04-23 15:19:08 -04:00
|
|
|
if (event->type == (priv->xsync_event_base + XSyncAlarmNotify))
|
|
|
|
handle_alarm_notify (backend, event);
|
2014-04-22 10:14:59 -04:00
|
|
|
|
2014-08-14 17:32:41 -04:00
|
|
|
if (event->type == priv->xkb_event_base)
|
|
|
|
{
|
|
|
|
XkbAnyEvent *xkb_ev = (XkbAnyEvent *) event;
|
|
|
|
|
|
|
|
if (xkb_ev->device == META_VIRTUAL_CORE_KEYBOARD_ID)
|
|
|
|
{
|
|
|
|
switch (xkb_ev->xkb_type)
|
|
|
|
{
|
|
|
|
case XkbNewKeyboardNotify:
|
|
|
|
case XkbMapNotify:
|
|
|
|
keymap_changed (backend);
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-23 12:05:14 -04:00
|
|
|
{
|
|
|
|
MetaMonitorManager *manager = meta_backend_get_monitor_manager (backend);
|
2014-04-23 12:07:38 -04:00
|
|
|
if (META_IS_MONITOR_MANAGER_XRANDR (manager) &&
|
2014-04-23 15:19:08 -04:00
|
|
|
meta_monitor_manager_xrandr_handle_xevent (META_MONITOR_MANAGER_XRANDR (manager), event))
|
2014-06-19 21:10:09 +02:00
|
|
|
bypass_clutter = TRUE;
|
2014-04-23 12:05:14 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!bypass_clutter)
|
2014-07-07 19:13:31 -04:00
|
|
|
{
|
2015-02-11 15:04:04 +01:00
|
|
|
handle_input_event (x11, event);
|
2014-07-07 19:13:31 -04:00
|
|
|
clutter_x11_handle_event (event);
|
|
|
|
}
|
2014-04-23 15:17:21 -04:00
|
|
|
|
|
|
|
XFreeEventData (priv->xdisplay, &event->xcookie);
|
2014-04-22 10:14:59 -04: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 15:19:08 -04:00
|
|
|
XEvent event;
|
2014-04-22 10:14:59 -04:00
|
|
|
|
2014-04-23 15:19:08 -04:00
|
|
|
XNextEvent (priv->xdisplay, &event);
|
2014-04-22 10:14:59 -04:00
|
|
|
|
2014-04-23 15:19:08 -04:00
|
|
|
handle_host_xevent (backend, &event);
|
2014-04-22 10:14:59 -04: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;
|
|
|
|
}
|
|
|
|
|
2014-07-14 09:41:10 -04:00
|
|
|
static void
|
|
|
|
take_touch_grab (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
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 = { META_VIRTUAL_CORE_POINTER_ID, sizeof (mask_bits), mask_bits };
|
|
|
|
XIGrabModifiers mods = { XIAnyModifier, 0 };
|
|
|
|
|
|
|
|
XISetMask (mask.mask, XI_TouchBegin);
|
|
|
|
XISetMask (mask.mask, XI_TouchUpdate);
|
|
|
|
XISetMask (mask.mask, XI_TouchEnd);
|
|
|
|
|
|
|
|
XIGrabTouchBegin (priv->xdisplay, META_VIRTUAL_CORE_POINTER_ID,
|
2014-07-17 17:18:08 +02:00
|
|
|
DefaultRootWindow (priv->xdisplay),
|
2014-07-14 09:41:10 -04:00
|
|
|
False, &mask, 1, &mods);
|
|
|
|
}
|
|
|
|
|
2014-10-02 19:07:21 +02:00
|
|
|
static void
|
|
|
|
on_device_added (ClutterDeviceManager *device_manager,
|
|
|
|
ClutterInputDevice *device,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (user_data);
|
|
|
|
|
|
|
|
if (clutter_input_device_get_device_type (device) == CLUTTER_KEYBOARD_DEVICE)
|
|
|
|
apply_keymap (x11);
|
|
|
|
}
|
|
|
|
|
2014-04-22 10:14:59 -04:00
|
|
|
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 12:45:13 -04:00
|
|
|
if (!XSyncQueryExtension (priv->xdisplay, &priv->xsync_event_base, &priv->xsync_error_base) ||
|
|
|
|
!XSyncInitialize (priv->xdisplay, &major, &minor))
|
2014-04-22 10:14:59 -04:00
|
|
|
meta_fatal ("Could not initialize XSync");
|
|
|
|
|
2014-04-23 13:59:58 -04: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");
|
|
|
|
}
|
|
|
|
|
2015-06-16 13:38:07 +02:00
|
|
|
/* We only take the passive touch grab if we are a X11 compositor */
|
|
|
|
if (priv->mode == META_BACKEND_X11_MODE_COMPOSITOR)
|
|
|
|
take_touch_grab (backend);
|
2014-07-17 17:18:08 +02:00
|
|
|
|
2014-08-04 16:47:35 +02:00
|
|
|
priv->xcb = XGetXCBConnection (priv->xdisplay);
|
|
|
|
if (!xkb_x11_setup_xkb_extension (priv->xcb,
|
|
|
|
XKB_X11_MIN_MAJOR_XKB_VERSION,
|
|
|
|
XKB_X11_MIN_MINOR_XKB_VERSION,
|
|
|
|
XKB_X11_SETUP_XKB_EXTENSION_NO_FLAGS,
|
|
|
|
NULL, NULL,
|
|
|
|
&priv->xkb_event_base,
|
|
|
|
&priv->xkb_error_base))
|
|
|
|
meta_fatal ("X server doesn't have the XKB extension, version %d.%d or newer\n",
|
|
|
|
XKB_X11_MIN_MAJOR_XKB_VERSION, XKB_X11_MIN_MINOR_XKB_VERSION);
|
|
|
|
|
2014-10-02 19:07:21 +02:00
|
|
|
g_signal_connect_object (clutter_device_manager_get_default (), "device-added",
|
|
|
|
G_CALLBACK (on_device_added), backend, 0);
|
|
|
|
|
2014-07-17 17:18:08 +02:00
|
|
|
META_BACKEND_CLASS (meta_backend_x11_parent_class)->post_init (backend);
|
2014-07-14 17:23:22 -04:00
|
|
|
}
|
|
|
|
|
2014-04-21 19:13:04 -04: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 19:41:11 -04:00
|
|
|
static MetaMonitorManager *
|
|
|
|
meta_backend_x11_create_monitor_manager (MetaBackend *backend)
|
|
|
|
{
|
2015-06-16 08:02:13 -07:00
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
2014-04-21 19:41:11 -04:00
|
|
|
|
2015-06-16 08:02:13 -07:00
|
|
|
switch (priv->mode)
|
|
|
|
{
|
|
|
|
case META_BACKEND_X11_MODE_COMPOSITOR:
|
|
|
|
return g_object_new (META_TYPE_MONITOR_MANAGER_XRANDR, NULL);
|
|
|
|
case META_BACKEND_X11_MODE_NESTED:
|
|
|
|
return g_object_new (META_TYPE_MONITOR_MANAGER_DUMMY, NULL);
|
|
|
|
default:
|
|
|
|
g_assert_not_reached ();
|
|
|
|
}
|
2014-04-21 19:41:11 -04:00
|
|
|
}
|
|
|
|
|
2014-04-27 10:49:29 -04:00
|
|
|
static MetaCursorRenderer *
|
|
|
|
meta_backend_x11_create_cursor_renderer (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
return g_object_new (META_TYPE_CURSOR_RENDERER_X11, NULL);
|
|
|
|
}
|
|
|
|
|
2014-04-23 10:50:07 -04: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;
|
|
|
|
|
2014-06-19 23:16:50 +02:00
|
|
|
if (timestamp != CurrentTime)
|
|
|
|
timestamp = MAX (timestamp, priv->latest_evtime);
|
|
|
|
|
2014-04-23 10:50:07 -04:00
|
|
|
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 13:36:11 -04:00
|
|
|
XISetMask (mask.mask, XI_KeyPress);
|
|
|
|
XISetMask (mask.mask, XI_KeyRelease);
|
2014-04-23 10:50:07 -04:00
|
|
|
|
|
|
|
ret = XIGrabDevice (priv->xdisplay, device_id,
|
2014-04-27 10:49:29 -04:00
|
|
|
meta_backend_x11_get_xwindow (x11),
|
2014-04-23 10:50:07 -04:00
|
|
|
timestamp,
|
2014-05-13 15:52:26 -04:00
|
|
|
None,
|
2014-04-23 10:50:07 -04:00
|
|
|
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-05-27 14:11:23 -04:00
|
|
|
static void
|
|
|
|
meta_backend_x11_warp_pointer (MetaBackend *backend,
|
|
|
|
int x,
|
|
|
|
int y)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
XIWarpPointer (priv->xdisplay,
|
|
|
|
META_VIRTUAL_CORE_POINTER_ID,
|
|
|
|
None,
|
|
|
|
meta_backend_x11_get_xwindow (x11),
|
|
|
|
0, 0, 0, 0,
|
|
|
|
x, y);
|
|
|
|
}
|
|
|
|
|
2014-08-04 16:47:35 +02:00
|
|
|
static void
|
|
|
|
get_xkbrf_var_defs (Display *xdisplay,
|
|
|
|
const char *layouts,
|
|
|
|
const char *variants,
|
|
|
|
const char *options,
|
|
|
|
char **rules_p,
|
|
|
|
XkbRF_VarDefsRec *var_defs)
|
|
|
|
{
|
|
|
|
char *rules = NULL;
|
|
|
|
|
|
|
|
/* Get it from the X property or fallback on defaults */
|
|
|
|
if (!XkbRF_GetNamesProp (xdisplay, &rules, var_defs) || !rules)
|
|
|
|
{
|
|
|
|
rules = strdup (DEFAULT_XKB_RULES_FILE);
|
|
|
|
var_defs->model = strdup (DEFAULT_XKB_MODEL);
|
|
|
|
var_defs->layout = NULL;
|
|
|
|
var_defs->variant = NULL;
|
|
|
|
var_defs->options = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Swap in our new options... */
|
|
|
|
free (var_defs->layout);
|
|
|
|
var_defs->layout = strdup (layouts);
|
|
|
|
free (var_defs->variant);
|
|
|
|
var_defs->variant = strdup (variants);
|
|
|
|
free (var_defs->options);
|
|
|
|
var_defs->options = strdup (options);
|
|
|
|
|
|
|
|
/* Sometimes, the property is a file path, and sometimes it's
|
|
|
|
not. Normalize it so it's always a file path. */
|
|
|
|
if (rules[0] == '/')
|
|
|
|
*rules_p = g_strdup (rules);
|
|
|
|
else
|
|
|
|
*rules_p = g_build_filename (XKB_BASE, "rules", rules, NULL);
|
|
|
|
|
|
|
|
free (rules);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_xkbrf_var_defs (XkbRF_VarDefsRec *var_defs)
|
|
|
|
{
|
|
|
|
free (var_defs->model);
|
|
|
|
free (var_defs->layout);
|
|
|
|
free (var_defs->variant);
|
|
|
|
free (var_defs->options);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_xkb_component_names (XkbComponentNamesRec *p)
|
|
|
|
{
|
|
|
|
free (p->keymap);
|
|
|
|
free (p->keycodes);
|
|
|
|
free (p->types);
|
|
|
|
free (p->compat);
|
|
|
|
free (p->symbols);
|
|
|
|
free (p->geometry);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
upload_xkb_description (Display *xdisplay,
|
|
|
|
const gchar *rules_file_path,
|
|
|
|
XkbRF_VarDefsRec *var_defs,
|
|
|
|
XkbComponentNamesRec *comp_names)
|
|
|
|
{
|
|
|
|
XkbDescRec *xkb_desc;
|
|
|
|
gchar *rules_file;
|
|
|
|
|
|
|
|
/* Upload it to the X server using the same method as setxkbmap */
|
|
|
|
xkb_desc = XkbGetKeyboardByName (xdisplay,
|
|
|
|
XkbUseCoreKbd,
|
|
|
|
comp_names,
|
|
|
|
XkbGBN_AllComponentsMask,
|
|
|
|
XkbGBN_AllComponentsMask &
|
|
|
|
(~XkbGBN_GeometryMask), True);
|
|
|
|
if (!xkb_desc)
|
|
|
|
{
|
|
|
|
g_warning ("Couldn't upload new XKB keyboard description");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
XkbFreeKeyboard (xkb_desc, 0, True);
|
|
|
|
|
|
|
|
rules_file = g_path_get_basename (rules_file_path);
|
|
|
|
|
|
|
|
if (!XkbRF_SetNamesProp (xdisplay, rules_file, var_defs))
|
|
|
|
g_warning ("Couldn't update the XKB root window property");
|
|
|
|
|
|
|
|
g_free (rules_file);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2014-10-02 19:07:21 +02:00
|
|
|
apply_keymap (MetaBackendX11 *x11)
|
2014-08-04 16:47:35 +02:00
|
|
|
{
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
XkbRF_RulesRec *xkb_rules;
|
|
|
|
XkbRF_VarDefsRec xkb_var_defs = { 0 };
|
|
|
|
gchar *rules_file_path;
|
|
|
|
|
2014-10-02 19:07:21 +02:00
|
|
|
if (!priv->keymap_layouts ||
|
|
|
|
!priv->keymap_variants ||
|
|
|
|
!priv->keymap_options)
|
|
|
|
return;
|
|
|
|
|
2014-08-04 16:47:35 +02:00
|
|
|
get_xkbrf_var_defs (priv->xdisplay,
|
2014-10-02 19:07:21 +02:00
|
|
|
priv->keymap_layouts,
|
|
|
|
priv->keymap_variants,
|
|
|
|
priv->keymap_options,
|
2014-08-04 16:47:35 +02:00
|
|
|
&rules_file_path,
|
|
|
|
&xkb_var_defs);
|
|
|
|
|
|
|
|
xkb_rules = XkbRF_Load (rules_file_path, NULL, True, True);
|
|
|
|
if (xkb_rules)
|
|
|
|
{
|
|
|
|
XkbComponentNamesRec xkb_comp_names = { 0 };
|
|
|
|
|
|
|
|
XkbRF_GetComponents (xkb_rules, &xkb_var_defs, &xkb_comp_names);
|
|
|
|
upload_xkb_description (priv->xdisplay, rules_file_path, &xkb_var_defs, &xkb_comp_names);
|
|
|
|
|
|
|
|
free_xkb_component_names (&xkb_comp_names);
|
|
|
|
XkbRF_Free (xkb_rules, True);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_warning ("Couldn't load XKB rules");
|
|
|
|
}
|
|
|
|
|
|
|
|
free_xkbrf_var_defs (&xkb_var_defs);
|
|
|
|
g_free (rules_file_path);
|
|
|
|
}
|
|
|
|
|
2014-10-02 19:07:21 +02:00
|
|
|
static void
|
|
|
|
meta_backend_x11_set_keymap (MetaBackend *backend,
|
|
|
|
const char *layouts,
|
|
|
|
const char *variants,
|
|
|
|
const char *options)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
g_free (priv->keymap_layouts);
|
|
|
|
priv->keymap_layouts = g_strdup (layouts);
|
|
|
|
g_free (priv->keymap_variants);
|
|
|
|
priv->keymap_variants = g_strdup (variants);
|
|
|
|
g_free (priv->keymap_options);
|
|
|
|
priv->keymap_options = g_strdup (options);
|
|
|
|
|
|
|
|
apply_keymap (x11);
|
|
|
|
}
|
|
|
|
|
2014-08-04 16:47:35 +02:00
|
|
|
static struct xkb_keymap *
|
|
|
|
meta_backend_x11_get_keymap (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
2014-08-14 17:23:20 -04:00
|
|
|
if (priv->keymap == NULL)
|
|
|
|
{
|
|
|
|
struct xkb_context *context = xkb_context_new (XKB_CONTEXT_NO_FLAGS);
|
|
|
|
priv->keymap = xkb_x11_keymap_new_from_device (context,
|
|
|
|
priv->xcb,
|
|
|
|
xkb_x11_get_core_keyboard_device_id (priv->xcb),
|
|
|
|
XKB_KEYMAP_COMPILE_NO_FLAGS);
|
|
|
|
xkb_context_unref (context);
|
|
|
|
}
|
2014-08-04 16:47:35 +02:00
|
|
|
|
2014-08-14 17:23:20 -04:00
|
|
|
return priv->keymap;
|
2014-08-04 16:47:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_backend_x11_lock_layout_group (MetaBackend *backend,
|
|
|
|
guint idx)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
XkbLockGroup (priv->xdisplay, XkbUseCoreKbd, idx);
|
|
|
|
}
|
|
|
|
|
2014-08-13 19:46:32 -04:00
|
|
|
static void
|
|
|
|
meta_backend_x11_update_screen_size (MetaBackend *backend,
|
|
|
|
int width, int height)
|
|
|
|
{
|
2015-06-16 08:02:13 -07:00
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
if (priv->mode == META_BACKEND_X11_MODE_NESTED)
|
2014-09-08 14:14:33 -04:00
|
|
|
{
|
|
|
|
/* For a nested wayland session, we want to go through Clutter to update the
|
|
|
|
* toplevel window size, rather than doing it directly.
|
|
|
|
*/
|
|
|
|
META_BACKEND_CLASS (meta_backend_x11_parent_class)->update_screen_size (backend, width, height);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
Window xwin = meta_backend_x11_get_xwindow (x11);
|
|
|
|
XResizeWindow (priv->xdisplay, xwin, width, height);
|
|
|
|
}
|
2014-08-13 19:46:32 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_backend_x11_select_stage_events (MetaBackend *backend)
|
|
|
|
{
|
|
|
|
MetaBackendX11 *x11 = META_BACKEND_X11 (backend);
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
Window xwin = meta_backend_x11_get_xwindow (x11);
|
|
|
|
unsigned char mask_bits[XIMaskLen (XI_LASTEVENT)] = { 0 };
|
|
|
|
XIEventMask mask = { XIAllMasterDevices, sizeof (mask_bits), mask_bits };
|
|
|
|
|
|
|
|
XISetMask (mask.mask, XI_KeyPress);
|
|
|
|
XISetMask (mask.mask, XI_KeyRelease);
|
|
|
|
XISetMask (mask.mask, XI_ButtonPress);
|
|
|
|
XISetMask (mask.mask, XI_ButtonRelease);
|
|
|
|
XISetMask (mask.mask, XI_Enter);
|
|
|
|
XISetMask (mask.mask, XI_Leave);
|
|
|
|
XISetMask (mask.mask, XI_FocusIn);
|
|
|
|
XISetMask (mask.mask, XI_FocusOut);
|
|
|
|
XISetMask (mask.mask, XI_Motion);
|
2015-06-16 13:38:07 +02:00
|
|
|
|
|
|
|
if (priv->mode == META_BACKEND_X11_MODE_NESTED)
|
|
|
|
{
|
|
|
|
/* When we're an X11 compositor, we can't take these events or else
|
|
|
|
* replaying events from our passive root window grab will cause
|
|
|
|
* them to come back to us.
|
|
|
|
*
|
|
|
|
* When we're a nested application, we want to behave like any other
|
|
|
|
* application, so select these events like normal apps do.
|
|
|
|
*/
|
|
|
|
XISetMask (mask.mask, XI_TouchBegin);
|
|
|
|
XISetMask (mask.mask, XI_TouchEnd);
|
|
|
|
XISetMask (mask.mask, XI_TouchUpdate);
|
|
|
|
}
|
|
|
|
|
2014-08-13 19:46:32 -04:00
|
|
|
XISelectEvents (priv->xdisplay, xwin, &mask, 1);
|
|
|
|
}
|
|
|
|
|
2014-04-21 19:13:04 -04:00
|
|
|
static void
|
|
|
|
meta_backend_x11_class_init (MetaBackendX11Class *klass)
|
|
|
|
{
|
|
|
|
MetaBackendClass *backend_class = META_BACKEND_CLASS (klass);
|
|
|
|
|
2014-04-22 10:14:59 -04:00
|
|
|
backend_class->post_init = meta_backend_x11_post_init;
|
2014-04-21 19:13:04 -04:00
|
|
|
backend_class->create_idle_monitor = meta_backend_x11_create_idle_monitor;
|
2014-04-21 19:41:11 -04:00
|
|
|
backend_class->create_monitor_manager = meta_backend_x11_create_monitor_manager;
|
2014-04-27 10:49:29 -04:00
|
|
|
backend_class->create_cursor_renderer = meta_backend_x11_create_cursor_renderer;
|
2014-04-23 10:50:07 -04:00
|
|
|
backend_class->grab_device = meta_backend_x11_grab_device;
|
|
|
|
backend_class->ungrab_device = meta_backend_x11_ungrab_device;
|
2014-05-27 14:11:23 -04:00
|
|
|
backend_class->warp_pointer = meta_backend_x11_warp_pointer;
|
2014-08-04 16:47:35 +02:00
|
|
|
backend_class->set_keymap = meta_backend_x11_set_keymap;
|
|
|
|
backend_class->get_keymap = meta_backend_x11_get_keymap;
|
|
|
|
backend_class->lock_layout_group = meta_backend_x11_lock_layout_group;
|
2014-08-13 19:46:32 -04:00
|
|
|
backend_class->update_screen_size = meta_backend_x11_update_screen_size;
|
|
|
|
backend_class->select_stage_events = meta_backend_x11_select_stage_events;
|
2014-04-21 19:13:04 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_backend_x11_init (MetaBackendX11 *x11)
|
|
|
|
{
|
2015-06-16 08:02:13 -07:00
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
2014-04-22 10:14:59 -04:00
|
|
|
/* We do X11 event retrieval ourselves */
|
|
|
|
clutter_x11_disable_event_retrieval ();
|
2015-06-16 08:02:13 -07:00
|
|
|
|
|
|
|
if (meta_is_wayland_compositor ())
|
|
|
|
priv->mode = META_BACKEND_X11_MODE_NESTED;
|
|
|
|
else
|
|
|
|
priv->mode = META_BACKEND_X11_MODE_COMPOSITOR;
|
2014-04-21 19:13:04 -04:00
|
|
|
}
|
2014-04-23 09:57:16 -04:00
|
|
|
|
|
|
|
Display *
|
|
|
|
meta_backend_x11_get_xdisplay (MetaBackendX11 *x11)
|
|
|
|
{
|
|
|
|
MetaBackendX11Private *priv = meta_backend_x11_get_instance_private (x11);
|
|
|
|
|
|
|
|
return priv->xdisplay;
|
|
|
|
}
|
|
|
|
|
2014-04-27 10:49:29 -04:00
|
|
|
Window
|
|
|
|
meta_backend_x11_get_xwindow (MetaBackendX11 *x11)
|
|
|
|
{
|
2014-08-13 19:46:32 -04:00
|
|
|
ClutterActor *stage = meta_backend_get_stage (META_BACKEND (x11));
|
|
|
|
return clutter_x11_get_stage_window (CLUTTER_STAGE (stage));
|
2014-04-27 10:49:29 -04:00
|
|
|
}
|