2013-05-03 18:51:22 +01:00
|
|
|
/*
|
|
|
|
* Wayland Support
|
|
|
|
*
|
|
|
|
* Copyright (C) 2013 Intel Corporation
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright © 2010-2011 Intel Corporation
|
|
|
|
* Copyright © 2008-2011 Kristian Høgsberg
|
|
|
|
* Copyright © 2012 Collabora, Ltd.
|
|
|
|
*
|
|
|
|
* Permission to use, copy, modify, distribute, and sell this software and
|
|
|
|
* its documentation for any purpose is hereby granted without fee, provided
|
|
|
|
* that the above copyright notice appear in all copies and that both that
|
|
|
|
* copyright notice and this permission notice appear in supporting
|
|
|
|
* documentation, and that the name of the copyright holders not be used in
|
|
|
|
* advertising or publicity pertaining to distribution of the software
|
|
|
|
* without specific, written prior permission. The copyright holders make
|
|
|
|
* no representations about the suitability of this software for any
|
|
|
|
* purpose. It is provided "as is" without express or implied warranty.
|
|
|
|
*
|
|
|
|
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS
|
|
|
|
* SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
|
|
* FITNESS, IN NO EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
|
|
* SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER
|
|
|
|
* RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF
|
|
|
|
* CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* The file is based on src/input.c from Weston */
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <errno.h>
|
2018-07-10 10:36:24 +02:00
|
|
|
#include <glib.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2014-08-05 14:11:59 +02:00
|
|
|
#include "backends/meta-backend-private.h"
|
2018-07-10 10:36:24 +02:00
|
|
|
#include "core/display-private.h"
|
2020-01-17 23:54:31 +01:00
|
|
|
#include "core/meta-anonymous-file.h"
|
2018-07-10 10:36:24 +02:00
|
|
|
#include "wayland/meta-wayland-private.h"
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2016-09-12 23:17:28 +08:00
|
|
|
G_DEFINE_TYPE (MetaWaylandKeyboard, meta_wayland_keyboard,
|
|
|
|
META_TYPE_WAYLAND_INPUT_DEVICE)
|
2016-04-01 16:39:30 +08:00
|
|
|
|
2014-07-22 00:54:04 +02:00
|
|
|
static void meta_wayland_keyboard_update_xkb_state (MetaWaylandKeyboard *keyboard);
|
2014-09-10 19:23:55 +02:00
|
|
|
static void notify_modifiers (MetaWaylandKeyboard *keyboard);
|
2014-07-22 00:54:04 +02:00
|
|
|
|
2022-05-30 23:48:44 +02:00
|
|
|
static MetaBackend *
|
|
|
|
backend_from_keyboard (MetaWaylandKeyboard *keyboard)
|
|
|
|
{
|
|
|
|
MetaWaylandInputDevice *input_device = META_WAYLAND_INPUT_DEVICE (keyboard);
|
|
|
|
MetaWaylandSeat *seat = meta_wayland_input_device_get_seat (input_device);
|
|
|
|
MetaWaylandCompositor *compositor = meta_wayland_seat_get_compositor (seat);
|
|
|
|
MetaContext *context = meta_wayland_compositor_get_context (compositor);
|
|
|
|
|
|
|
|
return meta_context_get_backend (context);
|
|
|
|
}
|
|
|
|
|
2014-04-17 18:22:26 -04:00
|
|
|
static void
|
|
|
|
unbind_resource (struct wl_resource *resource)
|
|
|
|
{
|
|
|
|
wl_list_remove (wl_resource_get_link (resource));
|
|
|
|
}
|
|
|
|
|
2013-09-04 11:11:39 +02:00
|
|
|
static void
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
send_keymap (MetaWaylandKeyboard *keyboard,
|
|
|
|
struct wl_resource *resource)
|
2013-05-03 18:51:22 +01:00
|
|
|
{
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
MetaWaylandXkbInfo *xkb_info = &keyboard->xkb_info;
|
|
|
|
int fd;
|
2020-01-17 23:54:31 +01:00
|
|
|
size_t size;
|
|
|
|
MetaAnonymousFileMapmode mapmode;
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
|
2020-01-17 23:54:31 +01:00
|
|
|
if (wl_resource_get_version (resource) < 7)
|
|
|
|
mapmode = META_ANONYMOUS_FILE_MAPMODE_SHARED;
|
|
|
|
else
|
|
|
|
mapmode = META_ANONYMOUS_FILE_MAPMODE_PRIVATE;
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
|
2020-01-17 23:54:31 +01:00
|
|
|
fd = meta_anonymous_file_open_fd (xkb_info->keymap_rofile, mapmode);
|
|
|
|
size = meta_anonymous_file_size (xkb_info->keymap_rofile);
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
|
2020-01-17 23:54:31 +01:00
|
|
|
if (fd == -1)
|
2014-07-07 13:53:11 +02:00
|
|
|
{
|
2020-01-17 23:54:31 +01:00
|
|
|
g_warning ("Creating a keymap file failed: %s", strerror (errno));
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
return;
|
2014-07-07 13:53:11 +02:00
|
|
|
}
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
|
|
|
|
wl_keyboard_send_keymap (resource,
|
|
|
|
WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1,
|
2020-01-17 23:54:31 +01:00
|
|
|
fd, size);
|
|
|
|
|
|
|
|
meta_anonymous_file_close_fd (fd);
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
inform_clients_of_new_keymap (MetaWaylandKeyboard *keyboard)
|
|
|
|
{
|
|
|
|
struct wl_resource *keyboard_resource;
|
|
|
|
|
|
|
|
wl_resource_for_each (keyboard_resource, &keyboard->resource_list)
|
|
|
|
send_keymap (keyboard, keyboard_resource);
|
|
|
|
wl_resource_for_each (keyboard_resource, &keyboard->focus_resource_list)
|
|
|
|
send_keymap (keyboard, keyboard_resource);
|
2013-09-04 11:11:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_wayland_keyboard_take_keymap (MetaWaylandKeyboard *keyboard,
|
2014-08-04 16:50:04 +02:00
|
|
|
struct xkb_keymap *keymap)
|
2013-09-04 11:11:39 +02:00
|
|
|
{
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
MetaWaylandXkbInfo *xkb_info = &keyboard->xkb_info;
|
2020-01-17 23:54:31 +01:00
|
|
|
char *keymap_string;
|
|
|
|
size_t keymap_size;
|
2013-09-04 11:11:39 +02:00
|
|
|
|
|
|
|
if (keymap == NULL)
|
|
|
|
{
|
|
|
|
g_warning ("Attempting to set null keymap (compilation probably failed)");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-03-06 14:41:45 +01:00
|
|
|
xkb_keymap_unref (xkb_info->keymap);
|
2014-08-04 16:50:04 +02:00
|
|
|
xkb_info->keymap = xkb_keymap_ref (keymap);
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2014-07-22 00:54:04 +02:00
|
|
|
meta_wayland_keyboard_update_xkb_state (keyboard);
|
2014-03-06 14:41:45 +01:00
|
|
|
|
2020-01-17 23:54:31 +01:00
|
|
|
keymap_string =
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
xkb_keymap_get_as_string (xkb_info->keymap, XKB_KEYMAP_FORMAT_TEXT_V1);
|
2020-01-17 23:54:31 +01:00
|
|
|
if (!keymap_string)
|
2013-05-03 18:51:22 +01:00
|
|
|
{
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
g_warning ("Failed to get string version of keymap");
|
2013-09-04 11:11:39 +02:00
|
|
|
return;
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
2020-01-17 23:54:31 +01:00
|
|
|
keymap_size = strlen (keymap_string) + 1;
|
|
|
|
|
2020-06-08 10:02:49 +02:00
|
|
|
g_clear_pointer (&xkb_info->keymap_rofile, meta_anonymous_file_free);
|
2020-01-17 23:54:31 +01:00
|
|
|
xkb_info->keymap_rofile =
|
|
|
|
meta_anonymous_file_new (keymap_size, (const uint8_t *) keymap_string);
|
|
|
|
|
|
|
|
free (keymap_string);
|
|
|
|
|
|
|
|
if (!xkb_info->keymap_rofile)
|
|
|
|
{
|
|
|
|
g_warning ("Failed to create anonymous file for keymap");
|
|
|
|
return;
|
|
|
|
}
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2014-08-04 16:50:04 +02:00
|
|
|
inform_clients_of_new_keymap (keyboard);
|
2013-09-04 11:11:39 +02:00
|
|
|
|
2014-09-10 19:23:55 +02:00
|
|
|
notify_modifiers (keyboard);
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
2017-10-19 08:15:08 +02:00
|
|
|
static xkb_mod_mask_t
|
|
|
|
kbd_a11y_apply_mask (MetaWaylandKeyboard *keyboard)
|
|
|
|
{
|
|
|
|
xkb_mod_mask_t latched, locked, depressed, group;
|
|
|
|
xkb_mod_mask_t update_mask = 0;
|
|
|
|
|
|
|
|
depressed = xkb_state_serialize_mods(keyboard->xkb_info.state, XKB_STATE_DEPRESSED);
|
|
|
|
latched = xkb_state_serialize_mods (keyboard->xkb_info.state, XKB_STATE_MODS_LATCHED);
|
|
|
|
locked = xkb_state_serialize_mods (keyboard->xkb_info.state, XKB_STATE_MODS_LOCKED);
|
|
|
|
group = xkb_state_serialize_layout (keyboard->xkb_info.state, XKB_STATE_LAYOUT_EFFECTIVE);
|
|
|
|
|
|
|
|
if ((latched & keyboard->kbd_a11y_latched_mods) != keyboard->kbd_a11y_latched_mods)
|
|
|
|
update_mask |= XKB_STATE_MODS_LATCHED;
|
|
|
|
|
|
|
|
if ((locked & keyboard->kbd_a11y_locked_mods) != keyboard->kbd_a11y_locked_mods)
|
|
|
|
update_mask |= XKB_STATE_MODS_LOCKED;
|
|
|
|
|
|
|
|
if (update_mask)
|
|
|
|
{
|
|
|
|
latched |= keyboard->kbd_a11y_latched_mods;
|
|
|
|
locked |= keyboard->kbd_a11y_locked_mods;
|
|
|
|
xkb_state_update_mask (keyboard->xkb_info.state, depressed, latched, locked, 0, 0, group);
|
|
|
|
}
|
|
|
|
|
|
|
|
return update_mask;
|
|
|
|
}
|
|
|
|
|
2014-09-10 19:13:46 +02:00
|
|
|
static void
|
|
|
|
on_keymap_changed (MetaBackend *backend,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
MetaWaylandKeyboard *keyboard = data;
|
|
|
|
|
|
|
|
meta_wayland_keyboard_take_keymap (keyboard, meta_backend_get_keymap (backend));
|
|
|
|
}
|
|
|
|
|
2014-09-10 19:23:55 +02:00
|
|
|
static void
|
|
|
|
on_keymap_layout_group_changed (MetaBackend *backend,
|
|
|
|
guint idx,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
MetaWaylandKeyboard *keyboard = data;
|
|
|
|
xkb_mod_mask_t depressed_mods;
|
|
|
|
xkb_mod_mask_t latched_mods;
|
|
|
|
xkb_mod_mask_t locked_mods;
|
|
|
|
struct xkb_state *state;
|
|
|
|
|
|
|
|
state = keyboard->xkb_info.state;
|
|
|
|
|
|
|
|
depressed_mods = xkb_state_serialize_mods (state, XKB_STATE_MODS_DEPRESSED);
|
|
|
|
latched_mods = xkb_state_serialize_mods (state, XKB_STATE_MODS_LATCHED);
|
|
|
|
locked_mods = xkb_state_serialize_mods (state, XKB_STATE_MODS_LOCKED);
|
|
|
|
|
|
|
|
xkb_state_update_mask (state, depressed_mods, latched_mods, locked_mods, 0, 0, idx);
|
2017-10-19 08:15:08 +02:00
|
|
|
kbd_a11y_apply_mask (keyboard);
|
2014-09-10 19:23:55 +02:00
|
|
|
|
|
|
|
notify_modifiers (keyboard);
|
|
|
|
}
|
|
|
|
|
2013-12-03 10:46:57 -05:00
|
|
|
static void
|
2014-02-17 19:13:16 -05:00
|
|
|
keyboard_handle_focus_surface_destroy (struct wl_listener *listener, void *data)
|
2013-12-03 10:46:57 -05:00
|
|
|
{
|
2023-02-09 16:17:34 +01:00
|
|
|
MetaWaylandKeyboard *keyboard =
|
|
|
|
wl_container_of (listener, keyboard, focus_surface_listener);
|
2014-02-20 11:29:28 -05:00
|
|
|
|
wayland: Unset keyboard focus if surface is destroyed
Depending on the ordering of the surface-associated resources
being destroyed, we may fall into the following situation:
- wl_surface is destroyed
- destruction notifications for the surface runs
- The MetaWaylandKeyboard attempts to synchronize the window
focus
- The MetaWindow is not destroyed yet, so the focused window
remains the same, and the MetaWaylandKeyboard keeps the same
focus MetaWaylandSurface.
- wl_surface finalizes destruction, MetaWaylandSurface now has
a NULL resource
- xdg_toplevel destructor kicks in, it unmanages the window
- The current focus window is again looked up, forced to look
a different window
- The MetaWaylandKeyboard focus now changes, tries to leave the
old surface, but it has a NULL resource already, and raises
a protocol error.
If the order is inverted, the window being unmanaged triggers a
focus change into a different window, the MetaWaylandKeyboard
triggers a focus change while the MetaWaylandSurface is still
intact, it succeeds, and the window gets properly destroyed.
In order to make this independent of the order, it makes sense
to make MetaWaylandKeyboard do like the other objects tracking
focus surfaces, and have it care of its own little parcel. The
surface destructor changed to simply unsetting the keyboard focus
to NULL (guaranteeing that the old focus is left while the surface
resource is still up), and leaving potential focus changes to
the xdg_toplevel_destructor->unmanage->update_focus paths.
Doing that alone is basically a revert of commit 228d681b, thus
is still subject to keyboard focus being lost after a popup is
destroyed. Change the approach to trigger the focus sync (and
new focus surface lookup) so it happens from xdg_popup_destructor
specifically to popups and alike xdg_toplevel.
Fixes: 228d681b ("wayland: Trigger full focus sync after keyboard focus surface is destroyed")
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2853
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3077>
2023-06-14 21:15:36 +02:00
|
|
|
meta_wayland_keyboard_set_focus (keyboard, NULL);
|
2013-12-03 10:46:57 -05:00
|
|
|
}
|
|
|
|
|
2013-09-04 15:01:11 +02:00
|
|
|
static gboolean
|
2015-04-07 15:54:41 +02:00
|
|
|
meta_wayland_keyboard_broadcast_key (MetaWaylandKeyboard *keyboard,
|
|
|
|
uint32_t time,
|
|
|
|
uint32_t key,
|
|
|
|
uint32_t state)
|
2013-05-03 18:51:22 +01:00
|
|
|
{
|
|
|
|
struct wl_resource *resource;
|
|
|
|
|
2016-09-20 16:46:16 +08:00
|
|
|
if (!wl_list_empty (&keyboard->focus_resource_list))
|
2013-05-03 18:51:22 +01:00
|
|
|
{
|
2016-09-20 16:44:46 +08:00
|
|
|
MetaWaylandInputDevice *input_device =
|
|
|
|
META_WAYLAND_INPUT_DEVICE (keyboard);
|
2018-07-25 16:18:36 +02:00
|
|
|
uint32_t serial;
|
2015-10-09 16:40:45 +02:00
|
|
|
|
2018-07-25 16:18:36 +02:00
|
|
|
serial = meta_wayland_input_device_next_serial (input_device);
|
2014-04-16 15:11:10 -04:00
|
|
|
|
2018-07-25 16:18:36 +02:00
|
|
|
if (state)
|
|
|
|
{
|
|
|
|
keyboard->key_down_serial = serial;
|
|
|
|
keyboard->key_down_keycode = key;
|
|
|
|
}
|
|
|
|
else
|
2014-04-16 15:11:10 -04:00
|
|
|
{
|
2018-07-25 16:18:36 +02:00
|
|
|
keyboard->key_up_serial = serial;
|
|
|
|
keyboard->key_up_keycode = key;
|
2014-04-16 15:11:10 -04:00
|
|
|
}
|
2018-07-25 16:18:36 +02:00
|
|
|
|
|
|
|
wl_resource_for_each (resource, &keyboard->focus_resource_list)
|
|
|
|
wl_keyboard_send_key (resource, serial, time, key, state);
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
2013-09-04 15:01:11 +02:00
|
|
|
|
2014-04-17 16:33:34 -04:00
|
|
|
/* Eat the key events if we have a focused surface. */
|
|
|
|
return (keyboard->focus_surface != NULL);
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
2015-04-07 15:54:41 +02:00
|
|
|
static gboolean
|
|
|
|
notify_key (MetaWaylandKeyboard *keyboard,
|
|
|
|
const ClutterEvent *event)
|
|
|
|
{
|
|
|
|
return keyboard->grab->interface->key (keyboard->grab, event);
|
|
|
|
}
|
|
|
|
|
2015-11-03 17:54:45 +01:00
|
|
|
static xkb_mod_mask_t
|
|
|
|
add_vmod (xkb_mod_mask_t mask,
|
|
|
|
xkb_mod_mask_t mod,
|
|
|
|
xkb_mod_mask_t vmod,
|
|
|
|
xkb_mod_mask_t *added)
|
|
|
|
{
|
|
|
|
if ((mask & mod) && !(mod & *added))
|
|
|
|
{
|
|
|
|
mask |= vmod;
|
|
|
|
*added |= mod;
|
|
|
|
}
|
|
|
|
return mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
static xkb_mod_mask_t
|
2022-05-30 23:48:44 +02:00
|
|
|
add_virtual_mods (MetaDisplay *display,
|
|
|
|
xkb_mod_mask_t mask)
|
2015-11-03 17:54:45 +01:00
|
|
|
{
|
2022-05-30 23:48:44 +02:00
|
|
|
MetaKeyBindingManager *keys = &display->key_binding_manager;
|
2015-11-03 17:54:45 +01:00
|
|
|
xkb_mod_mask_t added;
|
|
|
|
guint i;
|
|
|
|
/* Order is important here: if multiple vmods share the same real
|
|
|
|
modifier we only want to add the first. */
|
|
|
|
struct {
|
|
|
|
xkb_mod_mask_t mod;
|
|
|
|
xkb_mod_mask_t vmod;
|
|
|
|
} mods[] = {
|
|
|
|
{ keys->super_mask, keys->virtual_super_mask },
|
|
|
|
{ keys->hyper_mask, keys->virtual_hyper_mask },
|
|
|
|
{ keys->meta_mask, keys->virtual_meta_mask },
|
|
|
|
};
|
|
|
|
|
|
|
|
added = 0;
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (mods); ++i)
|
|
|
|
mask = add_vmod (mask, mods[i].mod, mods[i].vmod, &added);
|
|
|
|
|
|
|
|
return mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
keyboard_send_modifiers (MetaWaylandKeyboard *keyboard,
|
|
|
|
struct wl_resource *resource,
|
|
|
|
uint32_t serial)
|
|
|
|
{
|
2022-05-30 23:48:44 +02:00
|
|
|
MetaWaylandInputDevice *input_device = META_WAYLAND_INPUT_DEVICE (keyboard);
|
|
|
|
MetaWaylandSeat *seat = meta_wayland_input_device_get_seat (input_device);
|
|
|
|
MetaWaylandCompositor *compositor = meta_wayland_seat_get_compositor (seat);
|
|
|
|
MetaContext *context = meta_wayland_compositor_get_context (compositor);
|
|
|
|
MetaDisplay *display = meta_context_get_display (context);
|
2015-11-03 17:54:45 +01:00
|
|
|
struct xkb_state *state = keyboard->xkb_info.state;
|
|
|
|
xkb_mod_mask_t depressed, latched, locked;
|
|
|
|
|
2022-05-30 23:48:44 +02:00
|
|
|
depressed = add_virtual_mods (display,
|
|
|
|
xkb_state_serialize_mods (state, XKB_STATE_MODS_DEPRESSED));
|
|
|
|
latched = add_virtual_mods (display,
|
|
|
|
xkb_state_serialize_mods (state, XKB_STATE_MODS_LATCHED));
|
|
|
|
locked = add_virtual_mods (display,
|
|
|
|
xkb_state_serialize_mods (state, XKB_STATE_MODS_LOCKED));
|
2015-11-03 17:54:45 +01:00
|
|
|
|
|
|
|
wl_keyboard_send_modifiers (resource, serial, depressed, latched, locked,
|
|
|
|
xkb_state_serialize_layout (state, XKB_STATE_LAYOUT_EFFECTIVE));
|
|
|
|
}
|
|
|
|
|
2013-05-03 18:51:22 +01:00
|
|
|
static void
|
2015-04-07 15:54:41 +02:00
|
|
|
meta_wayland_keyboard_broadcast_modifiers (MetaWaylandKeyboard *keyboard)
|
2013-05-03 18:51:22 +01:00
|
|
|
{
|
2014-04-16 15:11:10 -04:00
|
|
|
struct wl_resource *resource;
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2016-09-20 16:46:16 +08:00
|
|
|
if (!wl_list_empty (&keyboard->focus_resource_list))
|
2013-05-03 18:51:22 +01:00
|
|
|
{
|
2016-09-12 23:20:36 +08:00
|
|
|
MetaWaylandInputDevice *input_device =
|
|
|
|
META_WAYLAND_INPUT_DEVICE (keyboard);
|
|
|
|
uint32_t serial;
|
|
|
|
|
2016-09-20 16:44:46 +08:00
|
|
|
serial = meta_wayland_input_device_next_serial (input_device);
|
2014-09-17 17:51:27 -06:00
|
|
|
|
2016-09-20 16:46:16 +08:00
|
|
|
wl_resource_for_each (resource, &keyboard->focus_resource_list)
|
2015-11-03 17:54:45 +01:00
|
|
|
keyboard_send_modifiers (keyboard, resource, serial);
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-07 15:54:41 +02:00
|
|
|
static void
|
|
|
|
notify_modifiers (MetaWaylandKeyboard *keyboard)
|
|
|
|
{
|
|
|
|
struct xkb_state *state;
|
|
|
|
|
|
|
|
state = keyboard->xkb_info.state;
|
|
|
|
keyboard->grab->interface->modifiers (keyboard->grab,
|
|
|
|
xkb_state_serialize_mods (state, XKB_STATE_MODS_EFFECTIVE));
|
|
|
|
}
|
|
|
|
|
2014-07-22 00:54:04 +02:00
|
|
|
static void
|
|
|
|
meta_wayland_keyboard_update_xkb_state (MetaWaylandKeyboard *keyboard)
|
|
|
|
{
|
|
|
|
MetaWaylandXkbInfo *xkb_info = &keyboard->xkb_info;
|
2019-10-07 19:14:20 +02:00
|
|
|
xkb_mod_mask_t latched, locked, numlock;
|
2022-05-30 23:48:44 +02:00
|
|
|
MetaBackend *backend = backend_from_keyboard (keyboard);
|
2017-12-19 16:32:32 +01:00
|
|
|
xkb_layout_index_t layout_idx;
|
2019-10-07 19:14:20 +02:00
|
|
|
ClutterKeymap *keymap;
|
2019-10-01 17:27:23 +02:00
|
|
|
ClutterSeat *seat;
|
2014-07-22 00:54:04 +02:00
|
|
|
|
|
|
|
/* Preserve latched/locked modifiers state */
|
|
|
|
if (xkb_info->state)
|
|
|
|
{
|
|
|
|
latched = xkb_state_serialize_mods (xkb_info->state, XKB_STATE_MODS_LATCHED);
|
|
|
|
locked = xkb_state_serialize_mods (xkb_info->state, XKB_STATE_MODS_LOCKED);
|
|
|
|
xkb_state_unref (xkb_info->state);
|
|
|
|
}
|
|
|
|
else
|
2017-11-01 11:03:18 +08:00
|
|
|
{
|
|
|
|
latched = locked = 0;
|
|
|
|
}
|
2014-07-22 00:54:04 +02:00
|
|
|
|
2019-10-01 17:27:23 +02:00
|
|
|
seat = clutter_backend_get_default_seat (clutter_get_default_backend ());
|
|
|
|
keymap = clutter_seat_get_keymap (seat);
|
2019-10-07 19:14:20 +02:00
|
|
|
numlock = (1 << xkb_keymap_mod_get_index (xkb_info->keymap, "Mod2"));
|
|
|
|
|
|
|
|
if (clutter_keymap_get_num_lock_state (keymap))
|
|
|
|
locked |= numlock;
|
|
|
|
else
|
|
|
|
locked &= ~numlock;
|
|
|
|
|
2014-07-22 00:54:04 +02:00
|
|
|
xkb_info->state = xkb_state_new (xkb_info->keymap);
|
|
|
|
|
2017-12-19 16:32:32 +01:00
|
|
|
layout_idx = meta_backend_get_keymap_layout_group (backend);
|
|
|
|
xkb_state_update_mask (xkb_info->state, 0, latched, locked, 0, 0, layout_idx);
|
2017-10-19 08:15:08 +02:00
|
|
|
|
|
|
|
kbd_a11y_apply_mask (keyboard);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2019-10-01 18:45:43 +02:00
|
|
|
on_kbd_a11y_mask_changed (ClutterSeat *seat,
|
|
|
|
xkb_mod_mask_t new_latched_mods,
|
|
|
|
xkb_mod_mask_t new_locked_mods,
|
|
|
|
MetaWaylandKeyboard *keyboard)
|
2017-10-19 08:15:08 +02:00
|
|
|
{
|
|
|
|
xkb_mod_mask_t latched, locked, depressed, group;
|
|
|
|
|
|
|
|
if (keyboard->xkb_info.state == NULL)
|
|
|
|
return;
|
|
|
|
|
|
|
|
depressed = xkb_state_serialize_mods(keyboard->xkb_info.state, XKB_STATE_DEPRESSED);
|
|
|
|
latched = xkb_state_serialize_mods (keyboard->xkb_info.state, XKB_STATE_MODS_LATCHED);
|
|
|
|
locked = xkb_state_serialize_mods (keyboard->xkb_info.state, XKB_STATE_MODS_LOCKED);
|
|
|
|
group = xkb_state_serialize_layout (keyboard->xkb_info.state, XKB_STATE_LAYOUT_EFFECTIVE);
|
|
|
|
|
|
|
|
/* Clear previous masks */
|
|
|
|
latched &= ~keyboard->kbd_a11y_latched_mods;
|
|
|
|
locked &= ~keyboard->kbd_a11y_locked_mods;
|
|
|
|
xkb_state_update_mask (keyboard->xkb_info.state, depressed, latched, locked, 0, 0, group);
|
|
|
|
|
|
|
|
/* Apply new masks */
|
|
|
|
keyboard->kbd_a11y_latched_mods = new_latched_mods;
|
|
|
|
keyboard->kbd_a11y_locked_mods = new_locked_mods;
|
|
|
|
kbd_a11y_apply_mask (keyboard);
|
|
|
|
|
|
|
|
notify_modifiers (keyboard);
|
2014-07-22 00:54:04 +02:00
|
|
|
}
|
|
|
|
|
2014-07-25 09:20:47 -04:00
|
|
|
static void
|
|
|
|
notify_key_repeat_for_resource (MetaWaylandKeyboard *keyboard,
|
|
|
|
struct wl_resource *keyboard_resource)
|
|
|
|
{
|
|
|
|
if (wl_resource_get_version (keyboard_resource) >= WL_KEYBOARD_REPEAT_INFO_SINCE_VERSION)
|
|
|
|
{
|
|
|
|
gboolean repeat;
|
|
|
|
unsigned int delay, rate;
|
|
|
|
|
|
|
|
repeat = g_settings_get_boolean (keyboard->settings, "repeat");
|
|
|
|
|
|
|
|
if (repeat)
|
|
|
|
{
|
|
|
|
unsigned int interval;
|
|
|
|
interval = g_settings_get_uint (keyboard->settings, "repeat-interval");
|
|
|
|
/* Our setting is in the milliseconds between keys. "rate" is the number
|
|
|
|
* of keys per second. */
|
2017-01-09 19:58:23 +01:00
|
|
|
if (interval > 0)
|
|
|
|
rate = (1000 / interval);
|
|
|
|
else
|
|
|
|
rate = 0;
|
|
|
|
|
2014-07-25 09:20:47 -04:00
|
|
|
delay = g_settings_get_uint (keyboard->settings, "delay");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
rate = 0;
|
|
|
|
delay = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
wl_keyboard_send_repeat_info (keyboard_resource, rate, delay);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
notify_key_repeat (MetaWaylandKeyboard *keyboard)
|
|
|
|
{
|
|
|
|
struct wl_resource *keyboard_resource;
|
|
|
|
|
|
|
|
wl_resource_for_each (keyboard_resource, &keyboard->resource_list)
|
|
|
|
{
|
|
|
|
notify_key_repeat_for_resource (keyboard, keyboard_resource);
|
|
|
|
}
|
|
|
|
|
|
|
|
wl_resource_for_each (keyboard_resource, &keyboard->focus_resource_list)
|
|
|
|
{
|
|
|
|
notify_key_repeat_for_resource (keyboard, keyboard_resource);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
settings_changed (GSettings *settings,
|
|
|
|
const char *key,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
MetaWaylandKeyboard *keyboard = data;
|
|
|
|
|
|
|
|
notify_key_repeat (keyboard);
|
|
|
|
}
|
|
|
|
|
2015-04-07 15:54:41 +02:00
|
|
|
static gboolean
|
|
|
|
default_grab_key (MetaWaylandKeyboardGrab *grab,
|
|
|
|
const ClutterEvent *event)
|
|
|
|
{
|
|
|
|
MetaWaylandKeyboard *keyboard = grab->keyboard;
|
2023-08-04 12:33:33 +02:00
|
|
|
gboolean is_press = clutter_event_type (event) == CLUTTER_KEY_PRESS;
|
2018-09-27 21:05:30 +02:00
|
|
|
guint32 code = 0;
|
2015-04-07 15:54:41 +02:00
|
|
|
|
2018-06-25 15:21:16 +00:00
|
|
|
/* Ignore autorepeat events, as autorepeat in Wayland is done on the client
|
|
|
|
* side. */
|
2023-08-04 12:33:33 +02:00
|
|
|
if (clutter_event_get_flags (event) & CLUTTER_EVENT_FLAG_REPEATED)
|
2015-04-07 15:54:41 +02:00
|
|
|
return FALSE;
|
|
|
|
|
2020-12-08 12:28:01 +01:00
|
|
|
code = clutter_event_get_event_code (event);
|
2015-04-07 15:54:41 +02:00
|
|
|
|
2023-08-04 12:33:33 +02:00
|
|
|
return meta_wayland_keyboard_broadcast_key (keyboard,
|
|
|
|
clutter_event_get_time (event),
|
2015-04-07 15:54:41 +02:00
|
|
|
code, is_press);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
default_grab_modifiers (MetaWaylandKeyboardGrab *grab,
|
|
|
|
ClutterModifierType modifiers)
|
|
|
|
{
|
|
|
|
meta_wayland_keyboard_broadcast_modifiers (grab->keyboard);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const MetaWaylandKeyboardGrabInterface default_keyboard_grab_interface = {
|
|
|
|
default_grab_key,
|
|
|
|
default_grab_modifiers
|
|
|
|
};
|
|
|
|
|
2014-04-17 19:11:47 -04:00
|
|
|
void
|
2016-09-12 23:20:36 +08:00
|
|
|
meta_wayland_keyboard_enable (MetaWaylandKeyboard *keyboard)
|
2013-05-03 18:51:22 +01:00
|
|
|
{
|
2022-05-30 23:48:44 +02:00
|
|
|
MetaBackend *backend = backend_from_keyboard (keyboard);
|
2019-10-01 18:45:43 +02:00
|
|
|
ClutterBackend *clutter_backend = clutter_get_default_backend ();
|
2014-09-10 19:13:46 +02:00
|
|
|
|
2014-12-09 20:58:22 +01:00
|
|
|
keyboard->settings = g_settings_new ("org.gnome.desktop.peripherals.keyboard");
|
Revert "wayland-keyboard: Don't send pressed keys on enter"
Back in 2014 sending pressed keys to Wayland clients caused issues,
because at least Xwayland didn't handle that gracefully, causing issues
like ghost-pressed keys. A way it was reproduced was quickly alt-tab:ing
to and from a Firefox window, which would cause the File menu bar
incorrectly appearing.
While this was reported to the Xwayland component back then, it was,
probably by mistake, assumed to be an issue in mutter, and mutter
stopped sending pressed key events on enter.
The following year, Xwayland was eventually fixed, but the work around
in mutter has been kept around until it was again noticed as an
inconsistency between compositor implementations.
Lets remove the work around, and follow the spec, again.
This reverts commit c39f18c2d438efe3a866767c9546a6140dd5aaad.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2457
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2657>
2022-10-13 11:51:00 +02:00
|
|
|
|
|
|
|
wl_array_init (&keyboard->pressed_keys);
|
|
|
|
|
2014-07-25 09:20:47 -04:00
|
|
|
g_signal_connect (keyboard->settings, "changed",
|
|
|
|
G_CALLBACK (settings_changed), keyboard);
|
2014-08-04 16:50:04 +02:00
|
|
|
|
2014-09-10 19:13:46 +02:00
|
|
|
g_signal_connect (backend, "keymap-changed",
|
|
|
|
G_CALLBACK (on_keymap_changed), keyboard);
|
2014-09-10 19:23:55 +02:00
|
|
|
g_signal_connect (backend, "keymap-layout-group-changed",
|
|
|
|
G_CALLBACK (on_keymap_layout_group_changed), keyboard);
|
2017-10-19 08:15:08 +02:00
|
|
|
|
2019-10-01 18:45:43 +02:00
|
|
|
g_signal_connect (clutter_backend_get_default_seat (clutter_backend),
|
|
|
|
"kbd-a11y-mods-state-changed",
|
2017-10-19 08:15:08 +02:00
|
|
|
G_CALLBACK (on_kbd_a11y_mask_changed), keyboard);
|
|
|
|
|
2014-09-10 19:13:46 +02:00
|
|
|
meta_wayland_keyboard_take_keymap (keyboard, meta_backend_get_keymap (backend));
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_wayland_xkb_info_destroy (MetaWaylandXkbInfo *xkb_info)
|
|
|
|
{
|
2016-04-01 16:39:30 +08:00
|
|
|
g_clear_pointer (&xkb_info->keymap, xkb_keymap_unref);
|
|
|
|
g_clear_pointer (&xkb_info->state, xkb_state_unref);
|
2021-04-14 15:13:08 +02:00
|
|
|
g_clear_pointer (&xkb_info->keymap_rofile, meta_anonymous_file_free);
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
2014-04-17 18:50:16 -04:00
|
|
|
void
|
2016-04-01 16:39:30 +08:00
|
|
|
meta_wayland_keyboard_disable (MetaWaylandKeyboard *keyboard)
|
2014-04-17 18:50:16 -04:00
|
|
|
{
|
2022-05-30 23:48:44 +02:00
|
|
|
MetaBackend *backend = backend_from_keyboard (keyboard);
|
2015-04-02 18:57:16 +02:00
|
|
|
|
|
|
|
g_signal_handlers_disconnect_by_func (backend, on_keymap_changed, keyboard);
|
|
|
|
g_signal_handlers_disconnect_by_func (backend, on_keymap_layout_group_changed, keyboard);
|
|
|
|
|
2016-09-21 11:27:06 +08:00
|
|
|
meta_wayland_keyboard_end_grab (keyboard);
|
2014-07-22 00:30:21 +02:00
|
|
|
meta_wayland_keyboard_set_focus (keyboard, NULL);
|
2014-04-17 18:50:16 -04:00
|
|
|
|
2016-09-21 11:30:02 +08:00
|
|
|
wl_list_remove (&keyboard->resource_list);
|
|
|
|
wl_list_init (&keyboard->resource_list);
|
|
|
|
wl_list_remove (&keyboard->focus_resource_list);
|
|
|
|
wl_list_init (&keyboard->focus_resource_list);
|
2014-07-25 09:20:47 -04:00
|
|
|
|
Revert "wayland-keyboard: Don't send pressed keys on enter"
Back in 2014 sending pressed keys to Wayland clients caused issues,
because at least Xwayland didn't handle that gracefully, causing issues
like ghost-pressed keys. A way it was reproduced was quickly alt-tab:ing
to and from a Firefox window, which would cause the File menu bar
incorrectly appearing.
While this was reported to the Xwayland component back then, it was,
probably by mistake, assumed to be an issue in mutter, and mutter
stopped sending pressed key events on enter.
The following year, Xwayland was eventually fixed, but the work around
in mutter has been kept around until it was again noticed as an
inconsistency between compositor implementations.
Lets remove the work around, and follow the spec, again.
This reverts commit c39f18c2d438efe3a866767c9546a6140dd5aaad.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2457
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2657>
2022-10-13 11:51:00 +02:00
|
|
|
wl_array_release (&keyboard->pressed_keys);
|
|
|
|
|
2016-04-01 16:39:30 +08:00
|
|
|
g_clear_object (&keyboard->settings);
|
2014-04-17 18:50:16 -04:00
|
|
|
}
|
|
|
|
|
2023-06-07 12:52:46 +02:00
|
|
|
static gboolean
|
Revert "wayland-keyboard: Don't send pressed keys on enter"
Back in 2014 sending pressed keys to Wayland clients caused issues,
because at least Xwayland didn't handle that gracefully, causing issues
like ghost-pressed keys. A way it was reproduced was quickly alt-tab:ing
to and from a Firefox window, which would cause the File menu bar
incorrectly appearing.
While this was reported to the Xwayland component back then, it was,
probably by mistake, assumed to be an issue in mutter, and mutter
stopped sending pressed key events on enter.
The following year, Xwayland was eventually fixed, but the work around
in mutter has been kept around until it was again noticed as an
inconsistency between compositor implementations.
Lets remove the work around, and follow the spec, again.
This reverts commit c39f18c2d438efe3a866767c9546a6140dd5aaad.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2457
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2657>
2022-10-13 11:51:00 +02:00
|
|
|
update_pressed_keys (struct wl_array *keys,
|
|
|
|
uint32_t evdev_code,
|
|
|
|
gboolean is_press)
|
|
|
|
{
|
|
|
|
uint32_t *end = (void *) ((char *) keys->data + keys->size);
|
|
|
|
uint32_t *k;
|
|
|
|
|
|
|
|
if (is_press)
|
|
|
|
{
|
|
|
|
/* Make sure we don't already have this key. */
|
|
|
|
for (k = keys->data; k < end; k++)
|
|
|
|
{
|
|
|
|
if (*k == evdev_code)
|
2023-06-07 12:52:46 +02:00
|
|
|
return FALSE;
|
Revert "wayland-keyboard: Don't send pressed keys on enter"
Back in 2014 sending pressed keys to Wayland clients caused issues,
because at least Xwayland didn't handle that gracefully, causing issues
like ghost-pressed keys. A way it was reproduced was quickly alt-tab:ing
to and from a Firefox window, which would cause the File menu bar
incorrectly appearing.
While this was reported to the Xwayland component back then, it was,
probably by mistake, assumed to be an issue in mutter, and mutter
stopped sending pressed key events on enter.
The following year, Xwayland was eventually fixed, but the work around
in mutter has been kept around until it was again noticed as an
inconsistency between compositor implementations.
Lets remove the work around, and follow the spec, again.
This reverts commit c39f18c2d438efe3a866767c9546a6140dd5aaad.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2457
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2657>
2022-10-13 11:51:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Otherwise add the key to the list of pressed keys */
|
|
|
|
k = wl_array_add (keys, sizeof (*k));
|
|
|
|
*k = evdev_code;
|
2023-06-07 12:52:46 +02:00
|
|
|
return TRUE;
|
Revert "wayland-keyboard: Don't send pressed keys on enter"
Back in 2014 sending pressed keys to Wayland clients caused issues,
because at least Xwayland didn't handle that gracefully, causing issues
like ghost-pressed keys. A way it was reproduced was quickly alt-tab:ing
to and from a Firefox window, which would cause the File menu bar
incorrectly appearing.
While this was reported to the Xwayland component back then, it was,
probably by mistake, assumed to be an issue in mutter, and mutter
stopped sending pressed key events on enter.
The following year, Xwayland was eventually fixed, but the work around
in mutter has been kept around until it was again noticed as an
inconsistency between compositor implementations.
Lets remove the work around, and follow the spec, again.
This reverts commit c39f18c2d438efe3a866767c9546a6140dd5aaad.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2457
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2657>
2022-10-13 11:51:00 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Remove the key from the array */
|
|
|
|
for (k = keys->data; k < end; k++)
|
|
|
|
{
|
|
|
|
if (*k == evdev_code)
|
|
|
|
{
|
|
|
|
*k = *(end - 1);
|
|
|
|
keys->size -= sizeof (*k);
|
2023-06-07 12:52:46 +02:00
|
|
|
return TRUE;
|
Revert "wayland-keyboard: Don't send pressed keys on enter"
Back in 2014 sending pressed keys to Wayland clients caused issues,
because at least Xwayland didn't handle that gracefully, causing issues
like ghost-pressed keys. A way it was reproduced was quickly alt-tab:ing
to and from a Firefox window, which would cause the File menu bar
incorrectly appearing.
While this was reported to the Xwayland component back then, it was,
probably by mistake, assumed to be an issue in mutter, and mutter
stopped sending pressed key events on enter.
The following year, Xwayland was eventually fixed, but the work around
in mutter has been kept around until it was again noticed as an
inconsistency between compositor implementations.
Lets remove the work around, and follow the spec, again.
This reverts commit c39f18c2d438efe3a866767c9546a6140dd5aaad.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2457
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2657>
2022-10-13 11:51:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-07 12:52:46 +02:00
|
|
|
return FALSE;
|
Revert "wayland-keyboard: Don't send pressed keys on enter"
Back in 2014 sending pressed keys to Wayland clients caused issues,
because at least Xwayland didn't handle that gracefully, causing issues
like ghost-pressed keys. A way it was reproduced was quickly alt-tab:ing
to and from a Firefox window, which would cause the File menu bar
incorrectly appearing.
While this was reported to the Xwayland component back then, it was,
probably by mistake, assumed to be an issue in mutter, and mutter
stopped sending pressed key events on enter.
The following year, Xwayland was eventually fixed, but the work around
in mutter has been kept around until it was again noticed as an
inconsistency between compositor implementations.
Lets remove the work around, and follow the spec, again.
This reverts commit c39f18c2d438efe3a866767c9546a6140dd5aaad.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2457
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2657>
2022-10-13 11:51:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-18 18:00:48 +01:00
|
|
|
void
|
|
|
|
meta_wayland_keyboard_update (MetaWaylandKeyboard *keyboard,
|
|
|
|
const ClutterKeyEvent *event)
|
|
|
|
{
|
2023-08-04 12:33:33 +02:00
|
|
|
gboolean is_press = clutter_event_type ((ClutterEvent *) event) == CLUTTER_KEY_PRESS;
|
|
|
|
uint32_t evdev_code, hardware_keycode;
|
2014-03-18 18:00:48 +01:00
|
|
|
|
2023-08-04 12:33:33 +02:00
|
|
|
evdev_code = clutter_event_get_event_code ((ClutterEvent *) event);
|
|
|
|
hardware_keycode = clutter_event_get_key_code ((ClutterEvent *) event);
|
|
|
|
|
|
|
|
if (!update_pressed_keys (&keyboard->pressed_keys, evdev_code, is_press))
|
2018-03-15 18:33:05 +01:00
|
|
|
return;
|
|
|
|
|
2015-11-03 17:58:16 +01:00
|
|
|
/* If we get a key event but still have pending modifier state
|
|
|
|
* changes from a previous event that didn't get cleared, we need to
|
|
|
|
* send that state right away so that the new key event can be
|
|
|
|
* interpreted by clients correctly modified. */
|
|
|
|
if (keyboard->mods_changed)
|
|
|
|
notify_modifiers (keyboard);
|
|
|
|
|
2014-07-22 11:24:56 +02:00
|
|
|
keyboard->mods_changed = xkb_state_update_key (keyboard->xkb_info.state,
|
2023-08-04 12:33:33 +02:00
|
|
|
hardware_keycode,
|
2014-07-22 11:24:56 +02:00
|
|
|
is_press ? XKB_KEY_DOWN : XKB_KEY_UP);
|
2017-10-19 08:15:08 +02:00
|
|
|
keyboard->mods_changed |= kbd_a11y_apply_mask (keyboard);
|
2014-03-18 18:00:48 +01:00
|
|
|
}
|
|
|
|
|
2013-09-04 15:01:11 +02:00
|
|
|
gboolean
|
2023-08-04 12:33:33 +02:00
|
|
|
meta_wayland_keyboard_handle_event (MetaWaylandKeyboard *keyboard,
|
2013-09-04 15:01:11 +02:00
|
|
|
const ClutterKeyEvent *event)
|
|
|
|
{
|
2018-07-12 17:08:52 +02:00
|
|
|
#ifdef WITH_VERBOSE_MODE
|
2023-08-04 12:33:33 +02:00
|
|
|
gboolean is_press =
|
|
|
|
clutter_event_type ((ClutterEvent *) event) == CLUTTER_KEY_PRESS;
|
2018-07-12 17:08:52 +02:00
|
|
|
#endif
|
2013-09-04 15:01:11 +02:00
|
|
|
gboolean handled;
|
2023-08-04 12:33:33 +02:00
|
|
|
ClutterEventFlags flags;
|
|
|
|
uint32_t hardware_keycode;
|
|
|
|
|
|
|
|
flags = clutter_event_get_flags ((ClutterEvent *) event);
|
|
|
|
hardware_keycode = clutter_event_get_key_code ((ClutterEvent *) event);
|
2013-09-04 15:01:11 +02:00
|
|
|
|
2014-03-18 13:24:52 -04:00
|
|
|
/* Synthetic key events are for autorepeat. Ignore those, as
|
|
|
|
* autorepeat in Wayland is done on the client side. */
|
2023-08-04 12:33:33 +02:00
|
|
|
if ((flags & CLUTTER_EVENT_FLAG_SYNTHETIC) &&
|
|
|
|
!(flags & CLUTTER_EVENT_FLAG_INPUT_METHOD))
|
2014-03-18 13:24:52 -04:00
|
|
|
return FALSE;
|
2013-09-04 15:01:11 +02:00
|
|
|
|
2020-10-02 17:47:22 +02:00
|
|
|
meta_verbose ("Handling key %s event code %d",
|
2013-09-04 15:01:11 +02:00
|
|
|
is_press ? "press" : "release",
|
2023-08-04 12:33:33 +02:00
|
|
|
hardware_keycode);
|
2013-09-04 15:01:11 +02:00
|
|
|
|
2015-04-07 15:54:41 +02:00
|
|
|
handled = notify_key (keyboard, (const ClutterEvent *) event);
|
2013-09-04 15:01:11 +02:00
|
|
|
|
|
|
|
if (handled)
|
2020-10-02 17:47:22 +02:00
|
|
|
meta_verbose ("Sent event to wayland client");
|
2013-09-04 15:01:11 +02:00
|
|
|
else
|
2020-10-02 17:47:22 +02:00
|
|
|
meta_verbose ("No wayland surface is focused, continuing normal operation");
|
2013-09-04 15:01:11 +02:00
|
|
|
|
2014-07-22 11:24:56 +02:00
|
|
|
if (keyboard->mods_changed != 0)
|
|
|
|
{
|
|
|
|
notify_modifiers (keyboard);
|
|
|
|
keyboard->mods_changed = 0;
|
|
|
|
}
|
|
|
|
|
2013-09-04 15:01:11 +02:00
|
|
|
return handled;
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
2015-08-21 16:25:53 -04:00
|
|
|
void
|
|
|
|
meta_wayland_keyboard_update_key_state (MetaWaylandKeyboard *keyboard,
|
|
|
|
char *key_vector,
|
|
|
|
int key_vector_len,
|
|
|
|
int offset)
|
|
|
|
{
|
|
|
|
gboolean mods_changed = FALSE;
|
|
|
|
|
2015-09-03 16:12:06 -04:00
|
|
|
int i;
|
|
|
|
for (i = offset; i < key_vector_len * 8; i++)
|
2015-08-21 16:25:53 -04:00
|
|
|
{
|
|
|
|
gboolean set = (key_vector[i/8] & (1 << (i % 8))) != 0;
|
|
|
|
|
|
|
|
/* The 'offset' parameter allows the caller to have the indices
|
|
|
|
* into key_vector to either be X-style (base 8) or evdev (base 0), or
|
|
|
|
* something else (unlikely). We subtract 'offset' to convert to evdev
|
|
|
|
* style, then add 8 to convert the "evdev" style keycode back to
|
|
|
|
* the X-style that xkbcommon expects.
|
|
|
|
*/
|
|
|
|
mods_changed |= xkb_state_update_key (keyboard->xkb_info.state,
|
|
|
|
i - offset + 8,
|
|
|
|
set ? XKB_KEY_DOWN : XKB_KEY_UP);
|
|
|
|
}
|
|
|
|
|
2017-10-19 08:15:08 +02:00
|
|
|
mods_changed |= kbd_a11y_apply_mask (keyboard);
|
2015-08-21 16:25:53 -04:00
|
|
|
if (mods_changed)
|
|
|
|
notify_modifiers (keyboard);
|
|
|
|
}
|
|
|
|
|
2014-04-17 18:50:16 -04:00
|
|
|
static void
|
|
|
|
move_resources (struct wl_list *destination, struct wl_list *source)
|
|
|
|
{
|
|
|
|
wl_list_insert_list (destination, source);
|
|
|
|
wl_list_init (source);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
move_resources_for_client (struct wl_list *destination,
|
|
|
|
struct wl_list *source,
|
|
|
|
struct wl_client *client)
|
|
|
|
{
|
|
|
|
struct wl_resource *resource, *tmp;
|
|
|
|
wl_resource_for_each_safe (resource, tmp, source)
|
|
|
|
{
|
|
|
|
if (wl_resource_get_client (resource) == client)
|
|
|
|
{
|
|
|
|
wl_list_remove (wl_resource_get_link (resource));
|
|
|
|
wl_list_insert (destination, wl_resource_get_link (resource));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-05 16:29:24 -07:00
|
|
|
static void
|
|
|
|
broadcast_focus (MetaWaylandKeyboard *keyboard,
|
|
|
|
struct wl_resource *resource)
|
|
|
|
{
|
2015-11-03 17:54:45 +01:00
|
|
|
keyboard_send_modifiers (keyboard, resource, keyboard->focus_serial);
|
2014-09-05 16:29:24 -07:00
|
|
|
wl_keyboard_send_enter (resource, keyboard->focus_serial,
|
|
|
|
keyboard->focus_surface->resource,
|
Revert "wayland-keyboard: Don't send pressed keys on enter"
Back in 2014 sending pressed keys to Wayland clients caused issues,
because at least Xwayland didn't handle that gracefully, causing issues
like ghost-pressed keys. A way it was reproduced was quickly alt-tab:ing
to and from a Firefox window, which would cause the File menu bar
incorrectly appearing.
While this was reported to the Xwayland component back then, it was,
probably by mistake, assumed to be an issue in mutter, and mutter
stopped sending pressed key events on enter.
The following year, Xwayland was eventually fixed, but the work around
in mutter has been kept around until it was again noticed as an
inconsistency between compositor implementations.
Lets remove the work around, and follow the spec, again.
This reverts commit c39f18c2d438efe3a866767c9546a6140dd5aaad.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/2457
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2657>
2022-10-13 11:51:00 +02:00
|
|
|
&keyboard->pressed_keys);
|
2014-09-05 16:29:24 -07:00
|
|
|
}
|
|
|
|
|
2013-05-03 18:51:22 +01:00
|
|
|
void
|
|
|
|
meta_wayland_keyboard_set_focus (MetaWaylandKeyboard *keyboard,
|
|
|
|
MetaWaylandSurface *surface)
|
|
|
|
{
|
2016-09-12 23:20:36 +08:00
|
|
|
MetaWaylandInputDevice *input_device = META_WAYLAND_INPUT_DEVICE (keyboard);
|
2014-09-05 18:00:09 -07:00
|
|
|
|
2014-09-05 16:29:24 -07:00
|
|
|
if (keyboard->focus_surface == surface)
|
2013-11-19 20:25:52 -05:00
|
|
|
return;
|
2013-05-03 18:51:22 +01:00
|
|
|
|
2014-02-18 23:16:54 -05:00
|
|
|
if (keyboard->focus_surface != NULL)
|
2013-11-19 20:25:52 -05:00
|
|
|
{
|
2016-09-20 16:46:16 +08:00
|
|
|
if (!wl_list_empty (&keyboard->focus_resource_list))
|
2014-02-18 23:16:54 -05:00
|
|
|
{
|
2016-09-20 16:52:29 +08:00
|
|
|
struct wl_resource *resource;
|
2016-09-20 16:44:46 +08:00
|
|
|
uint32_t serial;
|
|
|
|
|
|
|
|
serial = meta_wayland_input_device_next_serial (input_device);
|
2014-02-18 23:16:54 -05:00
|
|
|
|
2016-09-20 16:46:16 +08:00
|
|
|
wl_resource_for_each (resource, &keyboard->focus_resource_list)
|
2014-04-16 15:11:10 -04:00
|
|
|
{
|
2016-09-20 16:53:36 +08:00
|
|
|
wl_keyboard_send_leave (resource, serial,
|
|
|
|
keyboard->focus_surface->resource);
|
2014-04-16 15:11:10 -04:00
|
|
|
}
|
|
|
|
|
2016-09-20 16:53:36 +08:00
|
|
|
move_resources (&keyboard->resource_list,
|
|
|
|
&keyboard->focus_resource_list);
|
2014-02-18 23:16:54 -05:00
|
|
|
}
|
2014-02-19 13:56:56 +01:00
|
|
|
|
|
|
|
wl_list_remove (&keyboard->focus_surface_listener.link);
|
|
|
|
keyboard->focus_surface = NULL;
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
2014-02-18 23:16:54 -05:00
|
|
|
if (surface != NULL)
|
2013-05-03 18:51:22 +01:00
|
|
|
{
|
2016-09-20 16:52:29 +08:00
|
|
|
struct wl_resource *focus_surface_resource;
|
2014-04-16 15:11:10 -04:00
|
|
|
|
2014-02-18 18:00:26 -05:00
|
|
|
keyboard->focus_surface = surface;
|
2016-09-20 16:53:36 +08:00
|
|
|
focus_surface_resource = keyboard->focus_surface->resource;
|
|
|
|
wl_resource_add_destroy_listener (focus_surface_resource,
|
|
|
|
&keyboard->focus_surface_listener);
|
2014-02-18 18:00:26 -05:00
|
|
|
|
2014-04-16 15:11:10 -04:00
|
|
|
move_resources_for_client (&keyboard->focus_resource_list,
|
|
|
|
&keyboard->resource_list,
|
2016-09-20 16:53:36 +08:00
|
|
|
wl_resource_get_client (focus_surface_resource));
|
2014-04-16 15:11:10 -04:00
|
|
|
|
2017-10-19 08:15:08 +02:00
|
|
|
/* Make sure a11y masks are applied before braodcasting modifiers */
|
|
|
|
kbd_a11y_apply_mask (keyboard);
|
|
|
|
|
2016-09-20 16:46:16 +08:00
|
|
|
if (!wl_list_empty (&keyboard->focus_resource_list))
|
2014-02-18 23:16:54 -05:00
|
|
|
{
|
2016-09-20 16:52:29 +08:00
|
|
|
struct wl_resource *resource;
|
|
|
|
|
2016-09-20 16:44:46 +08:00
|
|
|
keyboard->focus_serial =
|
|
|
|
meta_wayland_input_device_next_serial (input_device);
|
2014-02-18 23:16:54 -05:00
|
|
|
|
2016-09-20 16:46:16 +08:00
|
|
|
wl_resource_for_each (resource, &keyboard->focus_resource_list)
|
2014-04-16 15:11:10 -04:00
|
|
|
{
|
2014-09-05 16:29:24 -07:00
|
|
|
broadcast_focus (keyboard, resource);
|
2014-04-16 15:11:10 -04:00
|
|
|
}
|
2014-02-18 23:16:54 -05:00
|
|
|
}
|
2014-02-18 18:00:26 -05:00
|
|
|
}
|
2013-05-03 18:51:22 +01:00
|
|
|
}
|
|
|
|
|
2014-04-16 15:20:07 -04:00
|
|
|
struct wl_client *
|
|
|
|
meta_wayland_keyboard_get_focus_client (MetaWaylandKeyboard *keyboard)
|
|
|
|
{
|
|
|
|
if (keyboard->focus_surface)
|
|
|
|
return wl_resource_get_client (keyboard->focus_surface->resource);
|
|
|
|
else
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-04-17 18:22:26 -04:00
|
|
|
|
|
|
|
static void
|
|
|
|
keyboard_release (struct wl_client *client,
|
|
|
|
struct wl_resource *resource)
|
|
|
|
{
|
|
|
|
wl_resource_destroy (resource);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wl_keyboard_interface keyboard_interface = {
|
|
|
|
keyboard_release,
|
|
|
|
};
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_wayland_keyboard_create_new_resource (MetaWaylandKeyboard *keyboard,
|
|
|
|
struct wl_client *client,
|
|
|
|
struct wl_resource *seat_resource,
|
|
|
|
uint32_t id)
|
|
|
|
{
|
2016-09-20 16:53:36 +08:00
|
|
|
struct wl_resource *resource;
|
2014-04-17 18:22:26 -04:00
|
|
|
|
2016-09-20 16:53:36 +08:00
|
|
|
resource = wl_resource_create (client, &wl_keyboard_interface,
|
|
|
|
wl_resource_get_version (seat_resource), id);
|
|
|
|
wl_resource_set_implementation (resource, &keyboard_interface,
|
|
|
|
keyboard, unbind_resource);
|
2014-04-17 18:22:26 -04:00
|
|
|
|
wayland/keyboard: Create a separate keymap shm file per resource
By using the shm file when sending the keymap to all clients, we
effectively allows any client to change the keymap, as any client has
the ability to change the content of the file. Sending a read-only file
descriptor, or making the file itself read-only before unlinking, can
be worked around by the client by using chmod(2) and open(2) on
/proc/<pid>/<fd>.
Using memfd could potentially solve this issue, but as the usage of
mmap with MAP_SHARED is wide spread among clients, such a change can
not be introduced without causing wide spread compatibility issues.
So, to avoid allowing clients to interfere with each other, create a
separate shm file for each wl_keyboard resource when sending the
keymap. We could eventually do this per client, but in most cases,
there will only be one wl_keyboard resource per client anyway.
https://bugzilla.gnome.org/show_bug.cgi?id=784206
2017-06-29 12:22:47 +08:00
|
|
|
send_keymap (keyboard, resource);
|
2014-04-17 18:22:26 -04:00
|
|
|
|
2016-09-20 16:53:36 +08:00
|
|
|
notify_key_repeat_for_resource (keyboard, resource);
|
2014-07-25 09:20:47 -04:00
|
|
|
|
2016-09-20 16:53:36 +08:00
|
|
|
if (keyboard->focus_surface &&
|
|
|
|
wl_resource_get_client (keyboard->focus_surface->resource) == client)
|
2014-09-16 21:24:28 -06:00
|
|
|
{
|
2016-09-20 16:53:36 +08:00
|
|
|
wl_list_insert (&keyboard->focus_resource_list,
|
|
|
|
wl_resource_get_link (resource));
|
|
|
|
broadcast_focus (keyboard, resource);
|
2014-09-16 21:24:28 -06:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-09-20 16:53:36 +08:00
|
|
|
wl_list_insert (&keyboard->resource_list,
|
|
|
|
wl_resource_get_link (resource));
|
2014-09-16 21:24:28 -06:00
|
|
|
}
|
2014-04-17 18:22:26 -04:00
|
|
|
}
|
2015-10-09 16:42:06 +02:00
|
|
|
|
2021-12-09 12:47:04 +01:00
|
|
|
gboolean
|
|
|
|
meta_wayland_keyboard_can_grab_surface (MetaWaylandKeyboard *keyboard,
|
|
|
|
MetaWaylandSurface *surface,
|
|
|
|
uint32_t serial)
|
|
|
|
{
|
|
|
|
if (keyboard->focus_surface != surface)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return (keyboard->focus_serial == serial ||
|
|
|
|
meta_wayland_keyboard_can_popup (keyboard, serial));
|
|
|
|
}
|
|
|
|
|
2015-10-09 16:42:06 +02:00
|
|
|
gboolean
|
|
|
|
meta_wayland_keyboard_can_popup (MetaWaylandKeyboard *keyboard,
|
|
|
|
uint32_t serial)
|
|
|
|
{
|
2018-07-25 16:18:36 +02:00
|
|
|
return (keyboard->key_down_serial == serial ||
|
|
|
|
((keyboard->key_down_keycode == keyboard->key_up_keycode) &&
|
|
|
|
keyboard->key_up_serial == serial));
|
2015-10-09 16:42:06 +02:00
|
|
|
}
|
2015-04-07 15:54:41 +02:00
|
|
|
|
|
|
|
void
|
|
|
|
meta_wayland_keyboard_start_grab (MetaWaylandKeyboard *keyboard,
|
|
|
|
MetaWaylandKeyboardGrab *grab)
|
|
|
|
{
|
|
|
|
meta_wayland_keyboard_set_focus (keyboard, NULL);
|
|
|
|
keyboard->grab = grab;
|
|
|
|
grab->keyboard = keyboard;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_wayland_keyboard_end_grab (MetaWaylandKeyboard *keyboard)
|
|
|
|
{
|
|
|
|
keyboard->grab = &keyboard->default_grab;
|
|
|
|
}
|
2016-04-01 16:39:30 +08:00
|
|
|
|
|
|
|
static void
|
|
|
|
meta_wayland_keyboard_init (MetaWaylandKeyboard *keyboard)
|
|
|
|
{
|
2016-09-21 11:30:02 +08:00
|
|
|
wl_list_init (&keyboard->resource_list);
|
|
|
|
wl_list_init (&keyboard->focus_resource_list);
|
|
|
|
|
2016-09-20 16:59:52 +08:00
|
|
|
keyboard->default_grab.interface = &default_keyboard_grab_interface;
|
|
|
|
keyboard->default_grab.keyboard = keyboard;
|
2016-09-21 11:27:06 +08:00
|
|
|
keyboard->grab = &keyboard->default_grab;
|
2016-09-20 16:59:52 +08:00
|
|
|
|
|
|
|
keyboard->focus_surface_listener.notify =
|
|
|
|
keyboard_handle_focus_surface_destroy;
|
2016-04-01 16:39:30 +08:00
|
|
|
}
|
|
|
|
|
2020-04-08 11:43:23 +02:00
|
|
|
static void
|
|
|
|
meta_wayland_keyboard_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
MetaWaylandKeyboard *keyboard = META_WAYLAND_KEYBOARD (object);
|
|
|
|
|
|
|
|
meta_wayland_xkb_info_destroy (&keyboard->xkb_info);
|
2020-05-07 09:13:58 +02:00
|
|
|
|
|
|
|
G_OBJECT_CLASS (meta_wayland_keyboard_parent_class)->finalize (object);
|
2020-04-08 11:43:23 +02:00
|
|
|
}
|
|
|
|
|
2016-04-01 16:39:30 +08:00
|
|
|
static void
|
|
|
|
meta_wayland_keyboard_class_init (MetaWaylandKeyboardClass *klass)
|
|
|
|
{
|
2020-04-08 11:43:23 +02:00
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
|
|
|
|
|
|
|
object_class->finalize = meta_wayland_keyboard_finalize;
|
2016-04-01 16:39:30 +08:00
|
|
|
}
|
2022-09-29 15:05:10 +02:00
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_wayland_keyboard_is_grabbed (MetaWaylandKeyboard *keyboard)
|
|
|
|
{
|
|
|
|
return keyboard->grab != &keyboard->default_grab;
|
|
|
|
}
|