mutter/src/backends/meta-eis-client.c
Bilal Elmoussaoui 918ae0f4d4 mutter: Replace cairo_rectangle_int_t with MtkRectangle
Note: the various helpers in boxes.c were not ported yet, see the next
commit. This was intentionally done to simplify the porting process

Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3128>
2023-08-30 16:46:13 +02:00

745 lines
24 KiB
C

/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
/*
* Copyright (C) 2021 Red Hat Inc.
*
* 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, see <http://www.gnu.org/licenses/>.
*
*/
#include "config.h"
#include "backends/meta-backend-private.h"
#include "backends/meta-eis-client.h"
#include "backends/meta-monitor-manager-private.h"
#include "backends/meta-viewport-info.h"
#include "clutter/clutter-mutter.h"
#include "core/meta-anonymous-file.h"
#define MAX_BUTTON 128
#define MAX_KEY 0x2ff /* KEY_MAX as of 5.13 */
typedef struct _MetaEisDevice MetaEisDevice;
struct _MetaEisDevice
{
struct eis_device *eis_device;
ClutterVirtualInputDevice *device;
guchar button_state[(MAX_BUTTON + 7) / 8];
guchar key_state[(MAX_KEY + 7) / 8];
};
struct _MetaEisClient
{
GObject parent_instance;
MetaEis *meta_eis;
MetaViewportInfo *viewports;
struct eis_client *eis_client;
struct eis_seat *eis_seat;
GHashTable *eis_devices; /* eis_device => MetaEisDevice*/
};
G_DEFINE_TYPE (MetaEisClient, meta_eis_client, G_TYPE_OBJECT)
typedef void (* MetaEisDeviceConfigFunc) (MetaEisClient *meta_eis_client,
struct eis_device *device);
static bool
bit_is_set (const guchar *array,
int bit)
{
return !!(array[bit / 8] & (1 << (bit % 8)));
}
static void
bit_set (guchar *array,
int bit)
{
array[bit / 8] |= (1 << (bit % 8));
}
static void
bit_clear (guchar *array,
int bit)
{
array[bit / 8] &= ~(1 << (bit % 8));
}
static void
notify_key (MetaEisDevice *meta_eis_device,
uint32_t key,
gboolean is_press)
{
ClutterKeyState state;
state = is_press ? CLUTTER_KEY_STATE_PRESSED : CLUTTER_KEY_STATE_RELEASED;
clutter_virtual_input_device_notify_key (meta_eis_device->device,
g_get_monotonic_time (),
key,
state);
}
static void
notify_button (MetaEisDevice *meta_eis_device,
uint32_t button,
gboolean is_press)
{
ClutterButtonState state;
state = is_press ? CLUTTER_BUTTON_STATE_PRESSED : CLUTTER_BUTTON_STATE_RELEASED;
clutter_virtual_input_device_notify_button (meta_eis_device->device,
g_get_monotonic_time (),
button,
state);
}
static void
remove_device (MetaEisClient *meta_eis_client,
struct eis_device *eis_device,
gboolean remove_from_hashtable)
{
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
struct eis_keymap *eis_keymap = eis_device_keyboard_get_keymap (eis_device);
if (eis_keymap)
{
MetaAnonymousFile *f = eis_keymap_get_user_data (eis_keymap);
if (f)
meta_anonymous_file_free (f);
}
eis_device_pause (eis_device);
eis_device_remove (eis_device);
g_clear_pointer (&meta_eis_device->eis_device, eis_device_unref);
g_clear_object (&meta_eis_device->device);
if (remove_from_hashtable)
g_hash_table_remove (meta_eis_client->eis_devices, eis_device);
}
static gboolean
drop_device (gpointer htkey,
gpointer value,
gpointer data)
{
MetaEisClient *meta_eis_client = data;
struct eis_device *eis_device = htkey;
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
uint32_t key, button;
for (key = 0; key < MAX_KEY; key++)
{
if (bit_is_set (meta_eis_device->key_state, key))
notify_key (meta_eis_device, key, FALSE);
}
for (button = 0; button < MAX_BUTTON; button++)
{
if (bit_is_set (meta_eis_device->button_state, key))
notify_button (meta_eis_device, button, FALSE);
}
remove_device (meta_eis_client, eis_device, FALSE);
return TRUE;
}
static void
meta_eis_device_free (MetaEisDevice *device)
{
eis_device_unref (device->eis_device);
free (device);
}
static void
configure_rel (MetaEisClient *meta_eis_client,
struct eis_device *eis_device)
{
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_POINTER);
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_BUTTON);
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_SCROLL);
}
static void
configure_keyboard (MetaEisClient *meta_eis_client,
struct eis_device *eis_device)
{
size_t len;
MetaAnonymousFile *f;
int fd = -1;
char *data;
struct xkb_keymap *xkb_keymap;
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_KEYBOARD);
xkb_keymap =
meta_backend_get_keymap (meta_eis_get_backend (meta_eis_client->meta_eis));
if (!xkb_keymap)
return;
data = xkb_keymap_get_as_string (xkb_keymap, XKB_KEYMAP_FORMAT_TEXT_V1);
if (!data)
return;
len = strlen (data);
f = meta_anonymous_file_new (len, (uint8_t*)data);
if (f)
fd = meta_anonymous_file_open_fd (f, META_ANONYMOUS_FILE_MAPMODE_SHARED);
g_free (data);
if (fd != -1)
{
struct eis_keymap *eis_keymap;
eis_keymap = eis_device_new_keymap (eis_device, EIS_KEYMAP_TYPE_XKB,
fd, len);
/* libeis dup()s the fd */
meta_anonymous_file_close_fd (fd);
/* The memfile must be kept alive while the device is alive */
eis_keymap_set_user_data (eis_keymap, f);
eis_keymap_add (eis_keymap);
eis_keymap_unref (eis_keymap);
}
}
static void
configure_abs (MetaEisClient *meta_eis_client,
struct eis_device *eis_device)
{
int idx = 0;
MtkRectangle rect;
float scale;
if (!meta_eis_client->viewports)
return; /* FIXME: should be an error */
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_POINTER_ABSOLUTE);
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_BUTTON);
eis_device_configure_capability (eis_device, EIS_DEVICE_CAP_SCROLL);
while (meta_viewport_info_get_view_info (meta_eis_client->viewports, idx++, &rect, &scale))
{
struct eis_region *r = eis_device_new_region (eis_device);
eis_region_set_offset (r, rect.x, rect.y);
eis_region_set_size (r, rect.width, rect.height);
eis_region_set_physical_scale (r, scale);
eis_region_add (r);
eis_region_unref (r);
}
}
static void
add_device (MetaEisClient *meta_eis_client,
struct eis_seat *eis_seat,
ClutterInputDeviceType type,
const char *name_suffix,
MetaEisDeviceConfigFunc extra_config_func)
{
MetaBackend *backend = meta_eis_get_backend (meta_eis_client->meta_eis);
MetaEisDevice *meta_eis_device;
ClutterSeat *seat = meta_backend_get_default_seat (backend);
ClutterVirtualInputDevice *device;
struct eis_device *eis_device;
gchar *name;
device = clutter_seat_create_virtual_device (seat, type);
eis_device = eis_seat_new_device (eis_seat);
name = g_strdup_printf ("%s %s", eis_client_get_name (meta_eis_client->eis_client),
name_suffix);
eis_device_configure_name (eis_device, name);
if (extra_config_func)
extra_config_func (meta_eis_client, eis_device);
meta_eis_device = g_new0 (MetaEisDevice, 1);
meta_eis_device->eis_device = eis_device_ref (eis_device);
meta_eis_device->device = device;
eis_device_set_user_data (eis_device, meta_eis_device);
g_hash_table_insert (meta_eis_client->eis_devices,
eis_device, /* owns the initial ref now */
meta_eis_device);
eis_device_add (eis_device);
eis_device_resume (eis_device);
g_free (name);
}
static void
handle_motion_relative (MetaEisClient *meta_eis_client,
struct eis_event *event)
{
struct eis_device *eis_device = eis_event_get_device (event);
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
double dx, dy;
dx = eis_event_pointer_get_dx (event);
dy = eis_event_pointer_get_dy (event);
clutter_virtual_input_device_notify_relative_motion (meta_eis_device->device,
g_get_monotonic_time (),
dx, dy);
}
static void
handle_motion_absolute (MetaEisClient *meta_eis_client,
struct eis_event *event)
{
struct eis_device *eis_device = eis_event_get_device (event);
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
double x, y;
x = eis_event_pointer_get_absolute_x (event);
y = eis_event_pointer_get_absolute_y (event);
clutter_virtual_input_device_notify_absolute_motion (meta_eis_device->device,
g_get_monotonic_time (),
x, y);
}
static void
handle_scroll (MetaEisClient *meta_eis_client,
struct eis_event *event)
{
struct eis_device *eis_device = eis_event_get_device (event);
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
double dx, dy;
dx = eis_event_scroll_get_dx (event);
dy = eis_event_scroll_get_dy (event);
clutter_virtual_input_device_notify_scroll_continuous (meta_eis_device->device,
g_get_monotonic_time (),
dx, dy,
CLUTTER_SCROLL_SOURCE_UNKNOWN,
CLUTTER_SCROLL_FINISHED_NONE);
}
static void
handle_scroll_stop (MetaEisClient *meta_eis_client,
struct eis_event *event)
{
struct eis_device *eis_device = eis_event_get_device (event);
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
ClutterScrollFinishFlags finish_flags = CLUTTER_SCROLL_FINISHED_NONE;
if (eis_event_scroll_get_stop_x (event))
finish_flags |= CLUTTER_SCROLL_FINISHED_HORIZONTAL;
if (eis_event_scroll_get_stop_y (event))
finish_flags |= CLUTTER_SCROLL_FINISHED_VERTICAL;
if (finish_flags != CLUTTER_SCROLL_FINISHED_NONE)
clutter_virtual_input_device_notify_scroll_continuous (meta_eis_device->device,
g_get_monotonic_time (),
0.0, 0.0,
CLUTTER_SCROLL_SOURCE_UNKNOWN,
finish_flags);
}
static void
handle_scroll_cancel (MetaEisClient *meta_eis_client,
struct eis_event *event)
{
struct eis_device *eis_device = eis_event_get_device (event);
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
double dx = 0.0, dy = 0.0;
/* There's no real match for the EIS scroll cancel event, so let's just send a
* really small scroll event that hopefully resets the scroll speed to
* something where kinetic scrolling is not viable.
*/
if (eis_event_scroll_get_stop_x (event))
dx = 0.01;
if (eis_event_scroll_get_stop_y (event))
dy = 0.01;
if (dx != 0.0 || dy != 0.0)
clutter_virtual_input_device_notify_scroll_continuous (meta_eis_device->device,
g_get_monotonic_time (),
dx, dy,
CLUTTER_SCROLL_SOURCE_UNKNOWN,
CLUTTER_SCROLL_FINISHED_NONE);
}
static void
handle_scroll_discrete (MetaEisClient *meta_eis_client,
struct eis_event *event)
{
struct eis_device *eis_device = eis_event_get_device (event);
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
int dx, dy;
/* FIXME: need to handle the remainders here for high-resolution scrolling */
dx = eis_event_scroll_get_discrete_dx (event) / 120;
dy = eis_event_scroll_get_discrete_dy (event) / 120;
/* Intentionally interleaved */
while (dx || dy)
{
if (dx > 0)
{
clutter_virtual_input_device_notify_discrete_scroll (meta_eis_device->device,
g_get_monotonic_time (),
CLUTTER_SCROLL_RIGHT,
CLUTTER_SCROLL_SOURCE_UNKNOWN);
dx--;
}
else if (dx < 0)
{
clutter_virtual_input_device_notify_discrete_scroll (meta_eis_device->device,
g_get_monotonic_time (),
CLUTTER_SCROLL_LEFT,
CLUTTER_SCROLL_SOURCE_UNKNOWN);
dx++;
}
if (dy > 0)
{
clutter_virtual_input_device_notify_discrete_scroll (meta_eis_device->device,
g_get_monotonic_time (),
CLUTTER_SCROLL_DOWN,
CLUTTER_SCROLL_SOURCE_UNKNOWN);
dy--;
}
else if (dy < 0)
{
clutter_virtual_input_device_notify_discrete_scroll (meta_eis_device->device,
g_get_monotonic_time (),
CLUTTER_SCROLL_UP,
CLUTTER_SCROLL_SOURCE_UNKNOWN);
dy++;
}
}
}
static void
handle_button (MetaEisClient *meta_eis_client,
struct eis_event *event)
{
struct eis_device *eis_device = eis_event_get_device (event);
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
uint32_t button;
gboolean is_press = eis_event_button_get_is_press (event);
button = eis_event_button_get_button (event);
switch (button)
{
case 0x110: /* BTN_LEFT */
button = CLUTTER_BUTTON_PRIMARY;
break;
case 0x111: /* BTN_RIGHT */
button = CLUTTER_BUTTON_SECONDARY;
break;
case 0x112: /* BTN_MIDDLE */
button = CLUTTER_BUTTON_MIDDLE;
break;
default:
if (button > 0x110)
button -= 0x110;
break;
}
if (button > MAX_BUTTON)
return;
if (is_press && !bit_is_set (meta_eis_device->button_state, button))
bit_set (meta_eis_device->button_state, button);
else if (!is_press && bit_is_set (meta_eis_device->button_state, button))
bit_clear (meta_eis_device->button_state, button);
else
return; /* Duplicate press/release, should've been filtered by libeis */
notify_button (meta_eis_device,
button,
eis_event_button_get_is_press (event));
}
static void
handle_key (MetaEisClient *meta_eis_client,
struct eis_event *event)
{
struct eis_device *eis_device = eis_event_get_device (event);
MetaEisDevice *meta_eis_device = eis_device_get_user_data (eis_device);
uint32_t key;
gboolean is_press = eis_event_keyboard_get_key_is_press (event);
key = eis_event_keyboard_get_key (event);
if (key > MAX_KEY)
return;
if (is_press && !bit_is_set (meta_eis_device->key_state, key))
bit_set (meta_eis_device->key_state, key);
else if (!is_press && bit_is_set (meta_eis_device->key_state, key))
bit_clear (meta_eis_device->key_state, key);
else
return; /* Duplicate press/release, should've been filtered by libeis */
notify_key (meta_eis_device, key, is_press);
}
static gboolean
drop_kbd_devices (gpointer htkey,
gpointer value,
gpointer data)
{
struct eis_device *eis_device = htkey;
if (!eis_device_has_capability (eis_device, EIS_DEVICE_CAP_KEYBOARD))
return FALSE;
return drop_device (htkey, value, data);
}
static void
on_keymap_changed (MetaBackend *backend,
gpointer data)
{
MetaEisClient *meta_eis_client = data;
/* Changing the keymap means we have to remove our device and recreate it
* with the new keymap.
*/
g_hash_table_foreach_remove (meta_eis_client->eis_devices,
drop_kbd_devices,
meta_eis_client);
add_device (meta_eis_client,
meta_eis_client->eis_seat,
CLUTTER_KEYBOARD_DEVICE,
"virtual keyboard",
configure_keyboard);
}
gboolean
meta_eis_client_process_event (MetaEisClient *meta_eis_client,
struct eis_event *event)
{
enum eis_event_type type = eis_event_get_type (event);
struct eis_seat *eis_seat;
switch (type)
{
case EIS_EVENT_SEAT_BIND:
eis_seat = eis_event_get_seat (event);
if (eis_event_seat_has_capability (event, EIS_DEVICE_CAP_POINTER))
add_device (meta_eis_client,
eis_seat,
CLUTTER_POINTER_DEVICE,
"virtual pointer",
configure_rel);
if (eis_event_seat_has_capability (event, EIS_DEVICE_CAP_KEYBOARD))
{
add_device (meta_eis_client,
eis_seat,
CLUTTER_KEYBOARD_DEVICE,
"virtual keyboard",
configure_keyboard);
g_signal_connect (meta_eis_get_backend (meta_eis_client->meta_eis),
"keymap-changed",
G_CALLBACK (on_keymap_changed),
meta_eis_client);
}
if (eis_event_seat_has_capability (event, EIS_DEVICE_CAP_POINTER_ABSOLUTE))
add_device (meta_eis_client,
eis_seat,
CLUTTER_POINTER_DEVICE,
"virtual absolute pointer",
configure_abs);
break;
/* We only have one seat, so if the client unbinds from that
* just disconnect it, no point keeping it alive */
case EIS_EVENT_DEVICE_CLOSED:
remove_device (meta_eis_client, eis_event_get_device (event), TRUE);
break;
case EIS_EVENT_POINTER_MOTION:
handle_motion_relative (meta_eis_client, event);
break;
case EIS_EVENT_POINTER_MOTION_ABSOLUTE:
handle_motion_absolute (meta_eis_client, event);
break;
case EIS_EVENT_BUTTON_BUTTON:
handle_button (meta_eis_client, event);
break;
case EIS_EVENT_SCROLL_DELTA:
handle_scroll (meta_eis_client, event);
break;
case EIS_EVENT_SCROLL_STOP:
handle_scroll_stop (meta_eis_client, event);
break;
case EIS_EVENT_SCROLL_CANCEL:
handle_scroll_cancel (meta_eis_client, event);
break;
case EIS_EVENT_SCROLL_DISCRETE:
handle_scroll_discrete (meta_eis_client, event);
break;
case EIS_EVENT_KEYBOARD_KEY:
handle_key (meta_eis_client, event);
break;
case EIS_EVENT_FRAME:
/* FIXME: we should be accumulating the above events */
break;
case EIS_EVENT_DEVICE_START_EMULATING:
break;
case EIS_EVENT_DEVICE_STOP_EMULATING:
break;
default:
g_warning ("Unhandled EIS event type %d", type);
return FALSE;
}
return TRUE;
}
static gboolean
drop_abs_devices (gpointer key,
gpointer value,
gpointer data)
{
struct eis_device *eis_device = key;
if (!eis_device_has_capability (eis_device, EIS_DEVICE_CAP_POINTER_ABSOLUTE))
return FALSE;
return drop_device (key, value, data);
}
static void
meta_eis_client_set_viewports (MetaEisClient *meta_eis_client,
MetaViewportInfo *viewports)
{
/* Updating viewports means we have to recreate our absolute pointer
* devices. */
g_hash_table_foreach_remove (meta_eis_client->eis_devices,
drop_abs_devices,
meta_eis_client);
g_clear_object (&meta_eis_client->viewports);
meta_eis_client->viewports = g_object_ref (viewports);
add_device (meta_eis_client,
meta_eis_client->eis_seat,
CLUTTER_POINTER_DEVICE,
"virtual absolute pointer",
configure_abs);
}
static void
on_monitors_changed (MetaMonitorManager *monitor_manager,
MetaEisClient *meta_eis_client)
{
MetaViewportInfo *viewports;
viewports = meta_monitor_manager_get_viewports (monitor_manager);
meta_eis_client_set_viewports (meta_eis_client, viewports);
}
static void
meta_eis_client_disconnect (MetaEisClient *meta_eis_client)
{
MetaBackend *backend = meta_eis_get_backend (meta_eis_client->meta_eis);
MetaMonitorManager *monitor_manager = meta_backend_get_monitor_manager (backend);
g_signal_handlers_disconnect_by_func (monitor_manager,
on_monitors_changed,
meta_eis_client);
g_hash_table_foreach_remove (meta_eis_client->eis_devices, drop_device, meta_eis_client);
g_clear_pointer (&meta_eis_client->eis_seat, eis_seat_unref);
if (meta_eis_client->eis_client)
eis_client_disconnect (meta_eis_client->eis_client);
g_clear_pointer (&meta_eis_client->eis_client, eis_client_unref);
g_clear_object (&meta_eis_client->viewports);
}
MetaEisClient *
meta_eis_client_new (MetaEis *meta_eis,
struct eis_client *eis_client)
{
MetaEisClient *meta_eis_client;
MetaBackend *backend;
MetaMonitorManager *monitor_manager;
MetaViewportInfo *viewports;
struct eis_seat *eis_seat;
meta_eis_client = g_object_new (META_TYPE_EIS_CLIENT, NULL);
meta_eis_client->meta_eis = meta_eis;
meta_eis_client->eis_client = eis_client_ref (eis_client);
eis_client_set_user_data (meta_eis_client->eis_client, meta_eis_client);
/* We're relying on some third party to filter clients for us */
eis_client_connect (eis_client);
/* We only support one seat for now. libeis keeps the ref for the
* seat and we don't need to care about it.
* The capabilities we define are the maximum capabilities, the client
* may only bind to a subset of those, reducing the capabilities
* of the seat in the process.
*/
eis_seat = eis_client_new_seat (eis_client, "mutter default seat");
eis_seat_configure_capability (eis_seat, EIS_DEVICE_CAP_KEYBOARD);
eis_seat_configure_capability (eis_seat, EIS_DEVICE_CAP_POINTER);
eis_seat_configure_capability (eis_seat, EIS_DEVICE_CAP_POINTER_ABSOLUTE);
eis_seat_configure_capability (eis_seat, EIS_DEVICE_CAP_BUTTON);
eis_seat_configure_capability (eis_seat, EIS_DEVICE_CAP_SCROLL);
eis_seat_add (eis_seat);
eis_seat_unref (eis_seat);
meta_eis_client->eis_seat = eis_seat_ref (eis_seat);
meta_eis_client->eis_devices = g_hash_table_new_full (g_direct_hash, g_direct_equal,
(GDestroyNotify) eis_device_unref,
(GDestroyNotify) meta_eis_device_free);
backend = meta_eis_get_backend (meta_eis);
monitor_manager = meta_backend_get_monitor_manager (backend);
viewports = meta_monitor_manager_get_viewports (monitor_manager);
meta_eis_client_set_viewports (meta_eis_client, viewports);
g_signal_connect (monitor_manager, "monitors-changed",
G_CALLBACK (on_monitors_changed),
meta_eis_client);
return meta_eis_client;
}
static void
meta_eis_client_init (MetaEisClient *meta_eis_client)
{
}
static void
meta_eis_client_finalize (GObject *object)
{
MetaEisClient *meta_eis_client = META_EIS_CLIENT (object);
g_signal_handlers_disconnect_by_func (meta_eis_get_backend (meta_eis_client->meta_eis),
on_keymap_changed,
meta_eis_client);
meta_eis_client_disconnect (meta_eis_client);
G_OBJECT_CLASS (meta_eis_client_parent_class)->finalize (object);
}
static void
meta_eis_client_class_init (MetaEisClientClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
object_class->finalize = meta_eis_client_finalize;
}