/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */ /* * Copyright (C) 2016 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: * Jonas Ã…dahl */ #include "config.h" #include #include "backends/meta-backend-private.h" #include "backends/meta-renderer.h" #include "backends/x11/meta-clutter-backend-x11.h" #include "backends/x11/meta-keymap-x11.h" #include "backends/x11/meta-seat-x11.h" #include "backends/x11/meta-xkb-a11y-x11.h" #include "backends/x11/nested/meta-stage-x11-nested.h" #include "clutter/clutter-mutter.h" #include "clutter/clutter.h" #include "cogl/cogl-xlib.h" #include "core/bell.h" #include "meta/meta-backend.h" typedef struct _MetaX11EventFilter MetaX11EventFilter; struct _MetaX11EventFilter { MetaX11FilterFunc func; gpointer data; }; typedef struct _MetaClutterBackendX11Private { MetaBackend *backend; } MetaClutterBackendX11Private; G_DEFINE_TYPE_WITH_PRIVATE (MetaClutterBackendX11, meta_clutter_backend_x11, CLUTTER_TYPE_BACKEND) /* atoms; remember to add the code that assigns the atom value to * the member of the MetaClutterBackendX11 structure if you add an * atom name here. do not change the order! */ static const gchar *atom_names[] = { "_NET_WM_PID", "_NET_WM_PING", "_NET_WM_STATE", "_NET_WM_USER_TIME", "WM_PROTOCOLS", "WM_DELETE_WINDOW", "_XEMBED", "_XEMBED_INFO", "_NET_WM_NAME", "UTF8_STRING", }; #define N_ATOM_NAMES G_N_ELEMENTS (atom_names) /* various flags corresponding to pre init setup calls */ static gboolean clutter_enable_stereo = FALSE; static Display *_foreign_dpy = NULL; /* options */ static gboolean clutter_synchronise = FALSE; /* X error trap */ static int TrappedErrorCode = 0; static int (* old_error_handler) (Display *, XErrorEvent *); static MetaX11FilterReturn cogl_xlib_filter (XEvent *xevent, ClutterEvent *event, gpointer data) { ClutterBackend *clutter_backend = data; MetaX11FilterReturn retval; CoglFilterReturn ret; ret = cogl_xlib_renderer_handle_event (clutter_backend->cogl_renderer, xevent); switch (ret) { case COGL_FILTER_REMOVE: retval = META_X11_FILTER_REMOVE; break; case COGL_FILTER_CONTINUE: default: retval = META_X11_FILTER_CONTINUE; break; } return retval; } static gboolean meta_clutter_backend_x11_finish_init (ClutterBackend *clutter_backend, GError **error) { MetaClutterBackendX11 *clutter_backend_x11 = META_CLUTTER_BACKEND_X11 (clutter_backend); Atom atoms[N_ATOM_NAMES]; Screen *xscreen; if (_foreign_dpy) backend_x11->xdisplay = _foreign_dpy; /* Only open connection if not already set by prior call to * clutter_x11_set_display() */ if (clutter_backend_x11->xdisplay == NULL) { const char *display_name; display_name = g_getenv ("DISPLAY"); if (display_name && *display_name != '\0') { g_debug ("XOpenDisplay on '%s'", display_name); clutter_backend_x11->xdisplay = XOpenDisplay (display_name); if (clutter_backend_x11->xdisplay == NULL) { g_set_error (error, CLUTTER_INIT_ERROR, CLUTTER_INIT_ERROR_BACKEND, "Unable to open display '%s'", display_name); return FALSE; } } else { g_set_error_literal (error, CLUTTER_INIT_ERROR, CLUTTER_INIT_ERROR_BACKEND, "Unable to open display. You have to set the " "DISPLAY environment variable, or use the " "--display command line argument"); return FALSE; } } g_debug ("Getting the X screen"); /* add event filter for Cogl events */ meta_clutter_backend_x11_add_filter (clutter_backend_x11, cogl_xlib_filter, clutter_backend); xscreen = DefaultScreenOfDisplay (clutter_backend_x11->xdisplay); clutter_backend_x11->xscreen_num = XScreenNumberOfScreen (xscreen); clutter_backend_x11->xwin_root = RootWindow (clutter_backend_x11->xdisplay, clutter_backend_x11->xscreen_num); if (clutter_synchronise) XSynchronize (clutter_backend_x11->xdisplay, True); XInternAtoms (clutter_backend_x11->xdisplay, (char **) atom_names, N_ATOM_NAMES, False, atoms); clutter_backend_x11->atom_NET_WM_PID = atoms[0]; clutter_backend_x11->atom_NET_WM_PING = atoms[1]; clutter_backend_x11->atom_NET_WM_STATE = atoms[2]; clutter_backend_x11->atom_NET_WM_USER_TIME = atoms[3]; clutter_backend_x11->atom_WM_PROTOCOLS = atoms[4]; clutter_backend_x11->atom_WM_DELETE_WINDOW = atoms[5]; clutter_backend_x11->atom_XEMBED = atoms[6]; clutter_backend_x11->atom_XEMBED_INFO = atoms[7]; clutter_backend_x11->atom_NET_WM_NAME = atoms[8]; clutter_backend_x11->atom_UTF8_STRING = atoms[9]; g_debug ("X Display '%s'[%p] opened (screen:%d, root:%u, dpi:%f)", g_getenv ("DISPLAY"), clutter_backend_x11->xdisplay, clutter_backend_x11->xscreen_num, (unsigned int) clutter_backend_x11->xwin_root, clutter_backend_get_resolution (clutter_backend)); return TRUE; } static void meta_clutter_backend_x11_finalize (GObject *gobject) { MetaClutterBackendX11 *clutter_backend_x11 = META_CLUTTER_BACKEND_X11 (gobject); meta_clutter_backend_x11_remove_filter (clutter_backend_x11, cogl_xlib_filter, clutter_backend_x11); XCloseDisplay (clutter_backend_x11->xdisplay); G_OBJECT_CLASS (meta_clutter_backend_x11_parent_class)->finalize (gobject); } static ClutterFeatureFlags meta_clutter_backend_x11_get_features (ClutterBackend *clutter_backend) { ClutterBackendClass *parent_class = CLUTTER_BACKEND_CLASS (meta_clutter_backend_x11_parent_class); ClutterFeatureFlags flags = CLUTTER_FEATURE_STAGE_CURSOR; flags |= parent_class->get_features (clutter_backend); return flags; } static void update_last_event_time (MetaClutterBackendX11 *clutter_backend_x11, XEvent *xevent) { Time current_time = CurrentTime; Time last_time = clutter_backend_x11->last_event_time; switch (xevent->type) { case KeyPress: case KeyRelease: current_time = xevent->xkey.time; break; case ButtonPress: case ButtonRelease: current_time = xevent->xbutton.time; break; case MotionNotify: current_time = xevent->xmotion.time; break; case EnterNotify: case LeaveNotify: current_time = xevent->xcrossing.time; break; case PropertyNotify: current_time = xevent->xproperty.time; break; default: break; } /* only change the current event time if it's after the previous event * time, or if it is at least 30 seconds earlier - in case the system * clock was changed */ if ((current_time != CurrentTime) && (current_time > last_time || (last_time - current_time > (30 * 1000)))) clutter_backend_x11->last_event_time = current_time; } static gboolean check_onscreen_template (CoglRenderer *renderer, CoglOnscreenTemplate *onscreen_template, gboolean enable_stereo, GError **error) { GError *internal_error = NULL; cogl_onscreen_template_set_stereo_enabled (onscreen_template, clutter_enable_stereo); /* cogl_renderer_check_onscreen_template() is actually just a * shorthand for creating a CoglDisplay, and calling * cogl_display_setup() on it, then throwing the display away. If we * could just return that display, then it would be more efficient * not to use cogl_renderer_check_onscreen_template(). However, the * backend API requires that we return an CoglDisplay that has not * yet been setup, so one way or the other we'll have to discard the * first display and make a new fresh one. */ if (cogl_renderer_check_onscreen_template (renderer, onscreen_template, &internal_error)) { clutter_enable_stereo = enable_stereo; return TRUE; } else { g_set_error_literal (error, CLUTTER_INIT_ERROR, CLUTTER_INIT_ERROR_BACKEND, internal_error != NULL ? internal_error->message : "Creation of a CoglDisplay failed"); g_clear_error (&internal_error); return FALSE; } } static CoglDisplay * meta_clutter_backend_x11_get_display (ClutterBackend *clutter_backend, CoglRenderer *renderer, CoglSwapChain *swap_chain, GError **error) { CoglOnscreenTemplate *onscreen_template; CoglDisplay *display = NULL; gboolean res = FALSE; onscreen_template = cogl_onscreen_template_new (swap_chain); /* It's possible that the current renderer doesn't support transparency * or doesn't support stereo, so we try the different combinations. */ if (clutter_enable_stereo) res = check_onscreen_template (renderer, onscreen_template, TRUE, error); if (!res) res = check_onscreen_template (renderer, onscreen_template, FALSE, error); if (res) display = cogl_display_new (renderer, onscreen_template); cogl_object_unref (onscreen_template); return display; } static CoglRenderer * meta_clutter_backend_x11_get_renderer (ClutterBackend *clutter_backend, GError **error) { MetaClutterBackendX11 *clutter_backend_x11 = META_CLUTTER_BACKEND_X11 (clutter_backend); MetaClutterBackendX11Private *priv = meta_clutter_backend_x11_get_instance_private (clutter_backend_x11); MetaRenderer *renderer = meta_backend_get_renderer (priv->backend); return meta_renderer_create_cogl_renderer (renderer); } static ClutterStageWindow * meta_clutter_backend_x11_create_stage (ClutterBackend *clutter_backend, ClutterStage *wrapper, GError **error) { ClutterStageWindow *stage; GType stage_type; if (meta_is_wayland_compositor ()) stage_type = META_TYPE_STAGE_X11_NESTED; else stage_type = META_TYPE_STAGE_X11; stage = g_object_new (stage_type, "backend", clutter_backend, "wrapper", wrapper, NULL); return stage; } static gboolean meta_clutter_backend_x11_process_event_filters (MetaClutterBackendX11 *clutter_backend_x11, gpointer native, ClutterEvent *event) { XEvent *xevent = native; /* X11 filter functions have a higher priority */ if (clutter_backend_x11->event_filters != NULL) { GSList *node = clutter_backend_x11->event_filters; while (node != NULL) { MetaX11EventFilter *filter = node->data; switch (filter->func (xevent, event, filter->data)) { case META_X11_FILTER_CONTINUE: break; case META_X11_FILTER_TRANSLATE: return TRUE; case META_X11_FILTER_REMOVE: return FALSE; default: break; } node = node->next; } } return FALSE; } static gboolean meta_clutter_backend_x11_translate_event (ClutterBackend *clutter_backend, gpointer native, ClutterEvent *event) { MetaClutterBackendX11 *clutter_backend_x11 = META_CLUTTER_BACKEND_X11 (clutter_backend); MetaClutterBackendX11Private *priv = meta_clutter_backend_x11_get_instance_private (clutter_backend_x11); MetaStageX11 *stage_x11; ClutterSeat *seat; if (meta_clutter_backend_x11_process_event_filters (clutter_backend_x11, native, event)) return TRUE; /* we update the event time only for events that can * actually reach Clutter's event queue */ update_last_event_time (clutter_backend_x11, native); stage_x11 = META_STAGE_X11 (clutter_backend_get_stage_window (clutter_backend)); if (meta_stage_x11_translate_event (stage_x11, native, event)) return TRUE; seat = meta_backend_get_default_seat (priv->backend); if (meta_seat_x11_translate_event (META_SEAT_X11 (seat), native, event)) return TRUE; return FALSE; } static ClutterSeat * meta_clutter_backend_x11_get_default_seat (ClutterBackend *clutter_backend) { MetaClutterBackendX11 *clutter_backend_x11 = META_CLUTTER_BACKEND_X11 (clutter_backend); MetaClutterBackendX11Private *priv = meta_clutter_backend_x11_get_instance_private (clutter_backend_x11); return meta_backend_get_default_seat (priv->backend); } static gboolean meta_clutter_backend_x11_is_display_server (ClutterBackend *clutter_backend) { return meta_is_wayland_compositor (); } static void meta_clutter_backend_x11_init (MetaClutterBackendX11 *clutter_backend_x11) { clutter_backend_x11->last_event_time = CurrentTime; } static void meta_clutter_backend_x11_class_init (MetaClutterBackendX11Class *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); ClutterBackendClass *clutter_backend_class = CLUTTER_BACKEND_CLASS (klass); gobject_class->finalize = meta_clutter_backend_x11_finalize; clutter_backend_class->finish_init = meta_clutter_backend_x11_finish_init; clutter_backend_class->get_features = meta_clutter_backend_x11_get_features; clutter_backend_class->get_display = meta_clutter_backend_x11_get_display; clutter_backend_class->get_renderer = meta_clutter_backend_x11_get_renderer; clutter_backend_class->create_stage = meta_clutter_backend_x11_create_stage; clutter_backend_class->translate_event = meta_clutter_backend_x11_translate_event; clutter_backend_class->get_default_seat = meta_clutter_backend_x11_get_default_seat; clutter_backend_class->is_display_server = meta_clutter_backend_x11_is_display_server; } MetaClutterBackendX11 * meta_clutter_backend_x11_new (MetaBackend *backend) { MetaClutterBackendX11 *clutter_backend_x11; MetaClutterBackendX11Private *priv; clutter_backend_x11 = g_object_new (META_TYPE_CLUTTER_BACKEND_X11, NULL); priv = meta_clutter_backend_x11_get_instance_private (clutter_backend_x11); priv->backend = backend; return clutter_backend_x11; } static int error_handler (Display *xdisplay, XErrorEvent *error) { TrappedErrorCode = error->error_code; return 0; } void meta_clutter_x11_trap_x_errors (void) { TrappedErrorCode = 0; old_error_handler = XSetErrorHandler (error_handler); } gint meta_clutter_x11_untrap_x_errors (void) { XSetErrorHandler (old_error_handler); return TrappedErrorCode; } Display * meta_clutter_x11_get_default_display (void) { ClutterBackend *clutter_backend = clutter_get_default_backend (); if (clutter_backend == NULL) { g_critical ("The Clutter backend has not been initialised"); return NULL; } if (!META_IS_CLUTTER_BACKEND_X11 (clutter_backend)) { g_critical ("The Clutter backend is not a X11 backend"); return NULL; } return META_CLUTTER_BACKEND_X11 (clutter_backend)->xdisplay; } void meta_clutter_x11_set_display (Display *xdisplay) { if (_clutter_context_is_initialized ()) { g_warning ("%s() can only be used before calling clutter_init()", G_STRFUNC); return; } _foreign_dpy= xdisplay; } int meta_clutter_x11_get_default_screen (void) { ClutterBackend *clutter_backend = clutter_get_default_backend (); if (clutter_backend == NULL) { g_critical ("The Clutter backend has not been initialised"); return 0; } if (!META_IS_CLUTTER_BACKEND_X11 (clutter_backend)) { g_critical ("The Clutter backend is not a X11 backend"); return 0; } return META_CLUTTER_BACKEND_X11 (clutter_backend)->xscreen_num; } Window meta_clutter_x11_get_root_window (void) { ClutterBackend *clutter_backend = clutter_get_default_backend (); if (clutter_backend == NULL) { g_critical ("The Clutter backend has not been initialised"); return None; } if (!META_IS_CLUTTER_BACKEND_X11 (clutter_backend)) { g_critical ("The Clutter backend is not a X11 backend"); return None; } return META_CLUTTER_BACKEND_X11 (clutter_backend)->xwin_root; } void meta_clutter_backend_x11_add_filter (MetaClutterBackendX11 *clutter_backend_x11, MetaX11FilterFunc func, gpointer data) { MetaX11EventFilter *filter; g_return_if_fail (func != NULL); filter = g_new0 (MetaX11EventFilter, 1); filter->func = func; filter->data = data; clutter_backend_x11->event_filters = g_slist_append (clutter_backend_x11->event_filters, filter); return; } void meta_clutter_backend_x11_remove_filter (MetaClutterBackendX11 *clutter_backend_x11, MetaX11FilterFunc func, gpointer data) { GSList *tmp_list, *this; MetaX11EventFilter *filter; g_return_if_fail (func != NULL); tmp_list = clutter_backend_x11->event_filters; while (tmp_list) { filter = tmp_list->data; this = tmp_list; tmp_list = tmp_list->next; if (filter->func == func && filter->data == data) { clutter_backend_x11->event_filters = g_slist_remove_link (clutter_backend_x11->event_filters, this); g_slist_free_1 (this); g_free (filter); return; } } } void meta_clutter_x11_set_use_stereo_stage (gboolean use_stereo) { if (_clutter_context_is_initialized ()) { g_warning ("%s() can only be used before calling clutter_init()", G_STRFUNC); return; } g_debug ("STEREO stages are %s", use_stereo ? "enabled" : "disabled"); clutter_enable_stereo = use_stereo; } gboolean meta_clutter_x11_get_use_stereo_stage (void) { return clutter_enable_stereo; }