2022-03-30 15:27:51 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2022 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, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
|
|
* 02111-1307, USA.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <glib.h>
|
2022-04-01 13:22:17 +00:00
|
|
|
#include <gio/gunixinputstream.h>
|
2022-04-27 08:50:43 +00:00
|
|
|
#include <gio/gunixfdlist.h>
|
|
|
|
#include <libei.h>
|
|
|
|
#include <linux/input.h>
|
2022-03-30 15:27:51 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
|
2022-04-27 08:50:43 +00:00
|
|
|
#include "backends/meta-fd-source.h"
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
#include "meta-dbus-input-capture.h"
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
unsigned int width;
|
|
|
|
unsigned int height;
|
|
|
|
int x;
|
|
|
|
int y;
|
|
|
|
} Zone;
|
|
|
|
|
|
|
|
G_DEFINE_AUTOPTR_CLEANUP_FUNC (Zone, g_free)
|
|
|
|
|
|
|
|
typedef enum _Capabilities
|
|
|
|
{
|
|
|
|
CAPABILITY_NONE = 0,
|
|
|
|
CAPABILITY_KEYBOARD = 1,
|
|
|
|
CAPABILITY_POINTER = 2,
|
|
|
|
CAPABILITY_TOUCH = 4,
|
|
|
|
} Capabilities;
|
|
|
|
|
|
|
|
typedef struct _InputCapture
|
|
|
|
{
|
|
|
|
MetaDBusInputCapture *proxy;
|
|
|
|
} InputCapture;
|
|
|
|
|
2022-04-27 08:50:43 +00:00
|
|
|
typedef struct _Event
|
|
|
|
{
|
|
|
|
enum ei_event_type type;
|
|
|
|
struct {
|
|
|
|
double dx;
|
|
|
|
double dy;
|
|
|
|
} motion;
|
|
|
|
struct {
|
|
|
|
uint32_t button;
|
|
|
|
gboolean is_press;
|
|
|
|
} button;
|
|
|
|
struct {
|
|
|
|
uint32_t key;
|
|
|
|
gboolean is_press;
|
|
|
|
} key;
|
|
|
|
} Event;
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
typedef struct _InputCaptureSession
|
|
|
|
{
|
|
|
|
MetaDBusInputCaptureSession *proxy;
|
|
|
|
unsigned int serial;
|
2022-04-27 08:50:43 +00:00
|
|
|
|
|
|
|
struct ei *ei;
|
|
|
|
GSource *ei_source;
|
|
|
|
|
|
|
|
Event *expected_events;
|
|
|
|
int n_expected_events;
|
|
|
|
int next_event;
|
|
|
|
|
|
|
|
gboolean has_pointer;
|
|
|
|
gboolean has_keyboard;
|
2022-03-30 15:27:51 +00:00
|
|
|
} InputCaptureSession;
|
|
|
|
|
2022-04-01 13:22:17 +00:00
|
|
|
static GDataInputStream *stdin_reader;
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
static void
|
|
|
|
ping_mutter (InputCaptureSession *session)
|
|
|
|
{
|
|
|
|
GDBusProxy *proxy = G_DBUS_PROXY (session->proxy);
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
if (!g_dbus_connection_call_sync (g_dbus_proxy_get_connection (proxy),
|
|
|
|
"org.gnome.Mutter.InputCapture",
|
|
|
|
g_dbus_proxy_get_object_path (proxy),
|
|
|
|
"org.freedesktop.DBus.Peer",
|
|
|
|
"Ping",
|
|
|
|
NULL,
|
|
|
|
NULL, G_DBUS_CALL_FLAGS_NO_AUTO_START, -1,
|
|
|
|
NULL, &error))
|
|
|
|
g_error ("Failed to ping D-Bus peer: %s", error->message);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
write_state (InputCaptureSession *session,
|
|
|
|
const char *state)
|
|
|
|
{
|
|
|
|
ping_mutter (session);
|
|
|
|
fprintf (stdout, "%s\n", state);
|
|
|
|
fflush (stdout);
|
|
|
|
}
|
|
|
|
|
2022-04-01 13:22:17 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
GMainLoop *loop;
|
|
|
|
const char *expected_state;
|
|
|
|
} WaitData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_line_read (GObject *source_object,
|
|
|
|
GAsyncResult *res,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
WaitData *data = user_data;
|
|
|
|
g_autofree char *line = NULL;
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
|
|
|
|
line =
|
|
|
|
g_data_input_stream_read_line_finish (G_DATA_INPUT_STREAM (source_object),
|
|
|
|
res, NULL, &error);
|
|
|
|
if (error)
|
|
|
|
g_error ("Failed to read line from test client: %s", error->message);
|
|
|
|
if (!line)
|
|
|
|
g_error ("Unexpected EOF");
|
|
|
|
|
|
|
|
g_assert_cmpstr (data->expected_state, ==, line);
|
|
|
|
|
|
|
|
g_main_loop_quit (data->loop);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
wait_for_state (InputCaptureSession *session,
|
|
|
|
const char *expected_state)
|
|
|
|
{
|
|
|
|
WaitData data;
|
|
|
|
|
|
|
|
data.loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
data.expected_state = expected_state;
|
|
|
|
|
|
|
|
g_data_input_stream_read_line_async (stdin_reader,
|
|
|
|
G_PRIORITY_DEFAULT,
|
|
|
|
NULL,
|
|
|
|
on_line_read,
|
|
|
|
&data);
|
|
|
|
|
|
|
|
g_main_loop_run (data.loop);
|
|
|
|
g_main_loop_unref (data.loop);
|
|
|
|
ping_mutter (session);
|
|
|
|
}
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
static InputCapture *
|
|
|
|
input_capture_new (void)
|
|
|
|
{
|
|
|
|
InputCapture *input_capture;
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
input_capture = g_new0 (InputCapture, 1);
|
|
|
|
input_capture->proxy = meta_dbus_input_capture_proxy_new_for_bus_sync (
|
|
|
|
G_BUS_TYPE_SESSION,
|
|
|
|
G_DBUS_PROXY_FLAGS_DO_NOT_AUTO_START,
|
|
|
|
"org.gnome.Mutter.InputCapture",
|
|
|
|
"/org/gnome/Mutter/InputCapture",
|
|
|
|
NULL,
|
|
|
|
&error);
|
|
|
|
if (!input_capture->proxy)
|
|
|
|
g_error ("Failed to acquire proxy: %s", error->message);
|
|
|
|
|
|
|
|
return input_capture;
|
|
|
|
}
|
|
|
|
|
|
|
|
static InputCaptureSession *
|
|
|
|
input_capture_create_session (InputCapture *input_capture)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
|
|
|
InputCaptureSession *session;
|
|
|
|
g_autofree char *session_path = NULL;
|
|
|
|
|
|
|
|
if (!meta_dbus_input_capture_call_create_session_sync (input_capture->proxy,
|
|
|
|
CAPABILITY_POINTER,
|
|
|
|
&session_path,
|
|
|
|
NULL,
|
|
|
|
&error))
|
|
|
|
g_error ("Failed to create input capture session: %s", error->message);
|
|
|
|
|
|
|
|
session = g_new0 (InputCaptureSession, 1);
|
|
|
|
session->proxy = meta_dbus_input_capture_session_proxy_new_for_bus_sync (
|
|
|
|
G_BUS_TYPE_SESSION,
|
|
|
|
G_DBUS_PROXY_FLAGS_DO_NOT_AUTO_START,
|
|
|
|
"org.gnome.Mutter.InputCapture",
|
|
|
|
session_path,
|
|
|
|
NULL, &error);
|
|
|
|
if (!session->proxy)
|
|
|
|
g_error ("Failed to acquire proxy: %s", error->message);
|
|
|
|
|
|
|
|
return session;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
input_capture_session_close (InputCaptureSession *session)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
|
|
|
|
2022-04-27 08:50:43 +00:00
|
|
|
g_clear_pointer (&session->ei, ei_unref);
|
|
|
|
g_clear_pointer (&session->ei_source, g_source_destroy);
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
if (!meta_dbus_input_capture_session_call_close_sync (session->proxy,
|
|
|
|
NULL, &error))
|
|
|
|
g_error ("Failed to close session: %s", error->message);
|
|
|
|
|
|
|
|
g_object_unref (session->proxy);
|
|
|
|
g_free (session);
|
|
|
|
}
|
|
|
|
|
2022-04-27 08:50:43 +00:00
|
|
|
static void
|
|
|
|
record_event (InputCaptureSession *session,
|
|
|
|
const Event *event)
|
|
|
|
{
|
|
|
|
const Event *expected_event;
|
|
|
|
|
|
|
|
g_debug ("Record event #%d, with type %s",
|
|
|
|
session->next_event + 1, ei_event_type_to_string (event->type));
|
|
|
|
g_assert_nonnull (session->expected_events);
|
|
|
|
g_assert_cmpint (session->next_event, <, session->n_expected_events);
|
|
|
|
|
|
|
|
expected_event = &session->expected_events[session->next_event++];
|
|
|
|
|
|
|
|
g_assert_cmpint (expected_event->type, ==, event->type);
|
|
|
|
|
|
|
|
switch (event->type)
|
|
|
|
{
|
|
|
|
case EI_EVENT_POINTER_MOTION:
|
|
|
|
g_assert_cmpfloat_with_epsilon (event->motion.dx,
|
|
|
|
expected_event->motion.dx,
|
|
|
|
DBL_EPSILON);
|
|
|
|
g_assert_cmpfloat_with_epsilon (event->motion.dy,
|
|
|
|
expected_event->motion.dy,
|
|
|
|
DBL_EPSILON);
|
|
|
|
break;
|
|
|
|
case EI_EVENT_BUTTON_BUTTON:
|
|
|
|
g_assert_cmpint (event->button.button, ==, expected_event->button.button);
|
|
|
|
break;
|
|
|
|
case EI_EVENT_KEYBOARD_KEY:
|
|
|
|
g_assert_cmpint (event->key.key, ==, expected_event->key.key);
|
|
|
|
break;
|
|
|
|
case EI_EVENT_FRAME:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
process_ei_event (InputCaptureSession *session,
|
|
|
|
struct ei_event *ei_event)
|
|
|
|
{
|
|
|
|
g_debug ("Processing event %s", ei_event_type_to_string (ei_event_get_type (ei_event)));
|
|
|
|
|
|
|
|
switch (ei_event_get_type (ei_event))
|
|
|
|
{
|
|
|
|
case EI_EVENT_SEAT_ADDED:
|
|
|
|
{
|
|
|
|
struct ei_seat *ei_seat = ei_event_get_seat (ei_event);
|
|
|
|
|
|
|
|
g_assert_true (ei_seat_has_capability (ei_seat, EI_DEVICE_CAP_POINTER));
|
|
|
|
g_assert_true (ei_seat_has_capability (ei_seat, EI_DEVICE_CAP_KEYBOARD));
|
|
|
|
g_assert_true (ei_seat_has_capability (ei_seat, EI_DEVICE_CAP_BUTTON));
|
|
|
|
g_assert_true (ei_seat_has_capability (ei_seat, EI_DEVICE_CAP_SCROLL));
|
|
|
|
ei_seat_bind_capabilities (ei_seat,
|
|
|
|
EI_DEVICE_CAP_POINTER,
|
|
|
|
EI_DEVICE_CAP_BUTTON,
|
|
|
|
EI_DEVICE_CAP_SCROLL,
|
|
|
|
EI_DEVICE_CAP_KEYBOARD,
|
|
|
|
NULL);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case EI_EVENT_DEVICE_ADDED:
|
|
|
|
{
|
|
|
|
struct ei_device *ei_device = ei_event_get_device (ei_event);
|
|
|
|
|
|
|
|
if (ei_device_has_capability (ei_device, EI_DEVICE_CAP_POINTER) &&
|
|
|
|
ei_device_has_capability (ei_device, EI_DEVICE_CAP_BUTTON) &&
|
|
|
|
ei_device_has_capability (ei_device, EI_DEVICE_CAP_SCROLL))
|
|
|
|
session->has_pointer = TRUE;
|
|
|
|
if (ei_device_has_capability (ei_device, EI_DEVICE_CAP_KEYBOARD))
|
|
|
|
session->has_keyboard = TRUE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case EI_EVENT_DEVICE_REMOVED:
|
|
|
|
{
|
|
|
|
struct ei_device *ei_device = ei_event_get_device (ei_event);
|
|
|
|
|
|
|
|
if (ei_device_has_capability (ei_device, EI_DEVICE_CAP_POINTER) &&
|
|
|
|
ei_device_has_capability (ei_device, EI_DEVICE_CAP_BUTTON) &&
|
|
|
|
ei_device_has_capability (ei_device, EI_DEVICE_CAP_SCROLL))
|
|
|
|
session->has_pointer = FALSE;
|
|
|
|
if (ei_device_has_capability (ei_device, EI_DEVICE_CAP_KEYBOARD))
|
|
|
|
session->has_keyboard = FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case EI_EVENT_POINTER_MOTION:
|
|
|
|
record_event (session,
|
|
|
|
&(Event) {
|
|
|
|
.type = EI_EVENT_POINTER_MOTION,
|
|
|
|
.motion.dx = ei_event_pointer_get_dx (ei_event),
|
|
|
|
.motion.dy = ei_event_pointer_get_dy (ei_event),
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case EI_EVENT_BUTTON_BUTTON:
|
|
|
|
record_event (session,
|
|
|
|
&(Event) {
|
|
|
|
.type = EI_EVENT_BUTTON_BUTTON,
|
|
|
|
.button.button = ei_event_button_get_button (ei_event),
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case EI_EVENT_KEYBOARD_KEY:
|
|
|
|
record_event (session,
|
|
|
|
&(Event) {
|
|
|
|
.type = EI_EVENT_KEYBOARD_KEY,
|
|
|
|
.key.key = ei_event_keyboard_get_key (ei_event),
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case EI_EVENT_FRAME:
|
|
|
|
record_event (session, &(Event) { .type = EI_EVENT_FRAME });
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
ei_source_prepare (gpointer user_data)
|
|
|
|
{
|
|
|
|
InputCaptureSession *session = user_data;
|
|
|
|
struct ei_event *ei_event;
|
|
|
|
gboolean retval;
|
|
|
|
|
|
|
|
ei_event = ei_peek_event (session->ei);
|
|
|
|
retval = !!ei_event;
|
|
|
|
ei_event_unref (ei_event);
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
ei_source_dispatch (gpointer user_data)
|
|
|
|
{
|
|
|
|
InputCaptureSession *session = user_data;
|
|
|
|
|
|
|
|
ei_dispatch (session->ei);
|
|
|
|
|
|
|
|
while (TRUE)
|
|
|
|
{
|
|
|
|
struct ei_event *ei_event;
|
|
|
|
|
|
|
|
ei_event = ei_get_event (session->ei);
|
|
|
|
if (!ei_event)
|
|
|
|
break;
|
|
|
|
|
|
|
|
process_ei_event (session, ei_event);
|
|
|
|
ei_event_unref (ei_event);
|
|
|
|
}
|
|
|
|
|
|
|
|
return G_SOURCE_CONTINUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_expected_events (InputCaptureSession *session,
|
|
|
|
Event *expected_events,
|
|
|
|
int n_expected_events)
|
|
|
|
{
|
|
|
|
session->expected_events = expected_events;
|
|
|
|
session->n_expected_events = n_expected_events;
|
|
|
|
session->next_event = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
log_handler (struct ei *ei,
|
|
|
|
enum ei_log_priority priority,
|
|
|
|
const char *message,
|
|
|
|
struct ei_log_context *ctx)
|
|
|
|
{
|
|
|
|
int message_length = strlen (message);
|
|
|
|
|
|
|
|
if (priority >= EI_LOG_PRIORITY_ERROR)
|
|
|
|
g_critical ("libei: %.*s", message_length, message);
|
|
|
|
else if (priority >= EI_LOG_PRIORITY_WARNING)
|
|
|
|
g_warning ("libei: %.*s", message_length, message);
|
|
|
|
else if (priority >= EI_LOG_PRIORITY_INFO)
|
|
|
|
g_info ("libei: %.*s", message_length, message);
|
|
|
|
else
|
|
|
|
g_debug ("libei: %.*s", message_length, message);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
input_capture_session_connect_to_eis (InputCaptureSession *session)
|
|
|
|
{
|
|
|
|
g_autoptr (GVariant) fd_variant = NULL;
|
|
|
|
g_autoptr (GUnixFDList) fd_list = NULL;
|
|
|
|
GError *error = NULL;
|
|
|
|
int fd;
|
|
|
|
struct ei *ei;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!meta_dbus_input_capture_session_call_connect_to_eis_sync (session->proxy,
|
|
|
|
NULL,
|
|
|
|
&fd_variant,
|
|
|
|
&fd_list,
|
|
|
|
NULL, &error))
|
|
|
|
g_error ("Failed to connect to EIS: %s", error->message);
|
|
|
|
|
|
|
|
fd = g_unix_fd_list_get (fd_list, g_variant_get_handle (fd_variant), &error);
|
|
|
|
if (fd == -1)
|
|
|
|
g_error ("Failed to get EIS file descriptor: %s", error->message);
|
|
|
|
|
|
|
|
ei = ei_new_receiver (session);
|
|
|
|
ei_log_set_handler (ei, log_handler);
|
|
|
|
ei_log_set_priority (ei, EI_LOG_PRIORITY_DEBUG);
|
|
|
|
|
|
|
|
ret = ei_setup_backend_fd (ei, fd);
|
|
|
|
if (ret < 0)
|
|
|
|
g_error ("Failed to setup libei backend: %s", g_strerror (errno));
|
|
|
|
|
|
|
|
session->ei = ei;
|
|
|
|
session->ei_source = meta_create_fd_source (ei_get_fd (ei),
|
|
|
|
"libei",
|
|
|
|
ei_source_prepare,
|
|
|
|
ei_source_dispatch,
|
|
|
|
session,
|
|
|
|
NULL);
|
|
|
|
g_source_attach (session->ei_source, NULL);
|
|
|
|
g_source_unref (session->ei_source);
|
|
|
|
}
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
static GList *
|
|
|
|
input_capture_session_get_zones (InputCaptureSession *session)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
|
|
|
g_autoptr (GVariant) zones_variant = NULL;
|
|
|
|
GVariantIter iter;
|
|
|
|
GList *zones = NULL;
|
|
|
|
unsigned int width, height;
|
|
|
|
int x, y;
|
|
|
|
|
|
|
|
if (!meta_dbus_input_capture_session_call_get_zones_sync (session->proxy,
|
|
|
|
&session->serial,
|
|
|
|
&zones_variant,
|
|
|
|
NULL, &error))
|
|
|
|
g_error ("Failed to get zones: %s", error->message);
|
|
|
|
|
|
|
|
g_variant_iter_init (&iter, zones_variant);
|
|
|
|
while (g_variant_iter_next (&iter, "(uuii)", &width, &height, &x, &y))
|
|
|
|
{
|
|
|
|
Zone *zone;
|
|
|
|
|
|
|
|
zone = g_new0 (Zone, 1);
|
|
|
|
*zone = (Zone) {
|
|
|
|
.width = width,
|
|
|
|
.height = height,
|
|
|
|
.x = x,
|
|
|
|
.y = y,
|
|
|
|
};
|
|
|
|
zones = g_list_append (zones, zone);
|
|
|
|
}
|
|
|
|
|
|
|
|
return zones;
|
|
|
|
}
|
|
|
|
|
2022-04-01 13:22:17 +00:00
|
|
|
static unsigned int
|
|
|
|
input_capture_session_add_barrier (InputCaptureSession *session,
|
|
|
|
int x1,
|
|
|
|
int y1,
|
|
|
|
int x2,
|
|
|
|
int y2)
|
|
|
|
{
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
unsigned int barrier_id;
|
|
|
|
|
|
|
|
if (!meta_dbus_input_capture_session_call_add_barrier_sync (
|
|
|
|
session->proxy,
|
|
|
|
session->serial,
|
|
|
|
g_variant_new ("(iiii)", x1, y1, x2, y2),
|
|
|
|
&barrier_id,
|
|
|
|
NULL,
|
|
|
|
&error))
|
|
|
|
{
|
|
|
|
g_warning ("Failed to add barrier: %s", error->message);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return barrier_id;
|
|
|
|
}
|
|
|
|
|
2022-04-01 17:14:48 +00:00
|
|
|
static void
|
|
|
|
input_capture_session_clear_barriers (InputCaptureSession *session)
|
|
|
|
{
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
|
|
|
|
if (!meta_dbus_input_capture_session_call_clear_barriers_sync (
|
|
|
|
session->proxy, NULL, &error))
|
|
|
|
g_warning ("Failed to clear barriers: %s", error->message);
|
|
|
|
}
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
static void
|
|
|
|
input_capture_session_enable (InputCaptureSession *session)
|
|
|
|
{
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
|
|
|
|
if (!meta_dbus_input_capture_session_call_enable_sync (session->proxy,
|
|
|
|
NULL, &error))
|
|
|
|
g_warning ("Failed to enable session: %s", error->message);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
input_capture_session_disable (InputCaptureSession *session)
|
|
|
|
{
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
|
|
|
|
if (!meta_dbus_input_capture_session_call_disable_sync (session->proxy,
|
|
|
|
NULL, &error))
|
|
|
|
g_warning ("Failed to disable session: %s", error->message);
|
|
|
|
}
|
|
|
|
|
2022-04-01 13:22:17 +00:00
|
|
|
static void
|
|
|
|
input_capture_session_release (InputCaptureSession *session,
|
|
|
|
double x,
|
|
|
|
double y)
|
|
|
|
{
|
|
|
|
g_autoptr (GError) error = NULL;
|
2022-04-27 08:50:43 +00:00
|
|
|
GVariantBuilder options_builder;
|
|
|
|
|
|
|
|
g_variant_builder_init (&options_builder, G_VARIANT_TYPE_VARDICT);
|
|
|
|
g_variant_builder_add (&options_builder, "{sv}",
|
|
|
|
"cursor_position",
|
|
|
|
g_variant_new ("(dd)", x, y));
|
2022-04-01 13:22:17 +00:00
|
|
|
|
|
|
|
if (!meta_dbus_input_capture_session_call_release_sync (session->proxy,
|
2022-04-27 08:50:43 +00:00
|
|
|
g_variant_builder_end (&options_builder),
|
2022-04-01 13:22:17 +00:00
|
|
|
NULL, &error))
|
|
|
|
g_warning ("Failed to release pointer: %s", error->message);
|
|
|
|
}
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
static void
|
|
|
|
test_sanity (void)
|
|
|
|
{
|
|
|
|
InputCapture *input_capture;
|
|
|
|
InputCaptureSession *session;
|
|
|
|
|
|
|
|
input_capture = input_capture_new ();
|
|
|
|
session = input_capture_create_session (input_capture);
|
|
|
|
|
|
|
|
g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING,
|
|
|
|
"*org.freedesktop.DBus.Error.Failed: Session not enabled*");
|
|
|
|
input_capture_session_disable (session);
|
|
|
|
g_test_assert_expected_messages ();
|
|
|
|
|
|
|
|
input_capture_session_enable (session);
|
|
|
|
|
|
|
|
g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING,
|
|
|
|
"*org.freedesktop.DBus.Error.Failed: Already enabled*");
|
|
|
|
input_capture_session_enable (session);
|
|
|
|
g_test_assert_expected_messages ();
|
|
|
|
|
|
|
|
input_capture_session_disable (session);
|
|
|
|
|
|
|
|
g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING,
|
|
|
|
"*org.freedesktop.DBus.Error.Failed: Session not enabled*");
|
|
|
|
input_capture_session_disable (session);
|
|
|
|
g_test_assert_expected_messages ();
|
|
|
|
|
|
|
|
input_capture_session_close (session);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_zones_changed (MetaDBusInputCaptureSession *proxy,
|
|
|
|
int *zones_changed_count)
|
|
|
|
{
|
|
|
|
*zones_changed_count += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
assert_zones (GList *zones,
|
|
|
|
const Zone *expected_zones,
|
|
|
|
int n_expected_zones)
|
|
|
|
{
|
|
|
|
GList *l;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
g_assert_cmpuint (g_list_length (zones), ==, n_expected_zones);
|
|
|
|
|
|
|
|
for (l = zones, i = 0; l; l = l->next, i++)
|
|
|
|
{
|
|
|
|
Zone *zone = l->data;
|
|
|
|
|
|
|
|
g_assert_cmpint (zone->width, ==, expected_zones[i].width);
|
|
|
|
g_assert_cmpint (zone->height, ==, expected_zones[i].height);
|
|
|
|
g_assert_cmpint (zone->x, ==, expected_zones[i].x);
|
|
|
|
g_assert_cmpint (zone->y, ==, expected_zones[i].y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
test_zones (void)
|
|
|
|
{
|
|
|
|
InputCapture *input_capture;
|
|
|
|
InputCaptureSession *session;
|
|
|
|
static const Zone expected_zones1[] = {
|
|
|
|
{ .width = 800, .height = 600, .x = 0, .y = 0 },
|
|
|
|
{ .width = 1024, .height = 768, .x = 800, .y = 0 },
|
|
|
|
};
|
|
|
|
static const Zone expected_zones2[] = {
|
|
|
|
{ .width = 1024, .height = 768, .x = 0, .y = 0 },
|
|
|
|
};
|
|
|
|
GList *zones;
|
|
|
|
int zones_changed_count = 0;
|
|
|
|
unsigned int serial;
|
|
|
|
|
|
|
|
input_capture = input_capture_new ();
|
|
|
|
session = input_capture_create_session (input_capture);
|
|
|
|
|
|
|
|
g_signal_connect (session->proxy, "zones-changed",
|
|
|
|
G_CALLBACK (on_zones_changed),
|
|
|
|
&zones_changed_count);
|
|
|
|
|
|
|
|
zones = input_capture_session_get_zones (session);
|
|
|
|
assert_zones (zones, expected_zones1, G_N_ELEMENTS (expected_zones1));
|
|
|
|
g_clear_list (&zones, g_free);
|
|
|
|
|
|
|
|
write_state (session, "1");
|
|
|
|
|
|
|
|
while (zones_changed_count == 0)
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
serial = session->serial;
|
|
|
|
g_clear_list (&zones, g_free);
|
|
|
|
|
|
|
|
zones = input_capture_session_get_zones (session);
|
|
|
|
g_assert_cmpuint (session->serial, >, serial);
|
|
|
|
assert_zones (zones, expected_zones2, G_N_ELEMENTS (expected_zones2));
|
|
|
|
g_clear_list (&zones, g_free);
|
|
|
|
|
|
|
|
input_capture_session_close (session);
|
|
|
|
}
|
|
|
|
|
2022-04-01 13:22:17 +00:00
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
unsigned int activated_barrier_id;
|
|
|
|
double activated_x;
|
|
|
|
double activated_y;
|
|
|
|
unsigned int activated_serial;
|
|
|
|
} BarriersTestData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_activated (MetaDBusInputCaptureSession *proxy,
|
|
|
|
unsigned int barrier_id,
|
|
|
|
unsigned int serial,
|
|
|
|
GVariant *cursor_position,
|
|
|
|
BarriersTestData *data)
|
|
|
|
{
|
|
|
|
g_assert_cmpuint (data->activated_barrier_id, ==, 0);
|
|
|
|
|
|
|
|
data->activated_barrier_id = barrier_id;
|
|
|
|
data->activated_serial = serial;
|
|
|
|
g_variant_get (cursor_position, "(dd)",
|
|
|
|
&data->activated_x, &data->activated_y);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
test_barriers (void)
|
|
|
|
{
|
|
|
|
InputCapture *input_capture;
|
|
|
|
InputCaptureSession *session;
|
|
|
|
g_autolist (Zone) zones = NULL;
|
|
|
|
unsigned int barrier1, barrier2;
|
|
|
|
BarriersTestData data = {};
|
|
|
|
unsigned int prev_activated_serial;
|
|
|
|
|
|
|
|
input_capture = input_capture_new ();
|
|
|
|
session = input_capture_create_session (input_capture);
|
|
|
|
|
|
|
|
zones = input_capture_session_get_zones (session);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* +-------------+--------------+
|
|
|
|
* || | |
|
|
|
|
* ||<--B#1 | |
|
|
|
|
* || | B#2 |
|
|
|
|
* +-------------+ | |
|
|
|
|
* | V |
|
|
|
|
* +==============+
|
|
|
|
*/
|
|
|
|
barrier1 = input_capture_session_add_barrier (session, 0, 0, 0, 600);
|
|
|
|
barrier2 = input_capture_session_add_barrier (session, 800, 768, 1824, 768);
|
|
|
|
|
|
|
|
g_assert_cmpuint (barrier1, !=, 0);
|
|
|
|
g_assert_cmpuint (barrier2, !=, 0);
|
|
|
|
g_assert_cmpuint (barrier1, !=, barrier2);
|
|
|
|
|
|
|
|
g_signal_connect (session->proxy, "activated",
|
|
|
|
G_CALLBACK (on_activated), &data);
|
|
|
|
|
|
|
|
input_capture_session_enable (session);
|
|
|
|
|
|
|
|
write_state (session, "1");
|
|
|
|
|
|
|
|
while (data.activated_barrier_id == 0)
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
g_assert_cmpuint (data.activated_serial, !=, 0);
|
|
|
|
g_assert_cmpuint (data.activated_barrier_id, ==, barrier1);
|
|
|
|
g_assert_cmpfloat_with_epsilon (data.activated_x, 0.0, DBL_EPSILON);
|
|
|
|
g_assert_cmpfloat_with_epsilon (data.activated_y, 15.0, DBL_EPSILON);
|
|
|
|
|
|
|
|
wait_for_state (session, "1");
|
|
|
|
|
|
|
|
input_capture_session_release (session, 200, 150);
|
|
|
|
|
|
|
|
write_state (session, "2");
|
|
|
|
|
|
|
|
prev_activated_serial = data.activated_serial;
|
|
|
|
|
|
|
|
data = (BarriersTestData) {};
|
|
|
|
while (data.activated_barrier_id == 0)
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
g_assert_cmpuint (data.activated_serial, !=, 0);
|
|
|
|
g_assert_cmpuint (data.activated_serial, !=, prev_activated_serial);
|
|
|
|
g_assert_cmpuint (data.activated_barrier_id, ==, barrier2);
|
|
|
|
g_assert_cmpfloat_with_epsilon (data.activated_x, 1000.0, DBL_EPSILON);
|
|
|
|
g_assert_cmpfloat_with_epsilon (data.activated_y, 768.0, DBL_EPSILON);
|
|
|
|
|
|
|
|
input_capture_session_release (session, 1200, 700);
|
|
|
|
write_state (session, "3");
|
|
|
|
|
|
|
|
input_capture_session_close (session);
|
|
|
|
}
|
|
|
|
|
2022-04-01 17:14:48 +00:00
|
|
|
static void
|
|
|
|
test_clear_barriers (void)
|
|
|
|
{
|
|
|
|
InputCapture *input_capture;
|
|
|
|
InputCaptureSession *session;
|
|
|
|
g_autolist (Zone) zones = NULL;
|
|
|
|
BarriersTestData data = {};
|
|
|
|
|
|
|
|
input_capture = input_capture_new ();
|
|
|
|
session = input_capture_create_session (input_capture);
|
|
|
|
|
|
|
|
zones = input_capture_session_get_zones (session);
|
|
|
|
|
|
|
|
input_capture_session_add_barrier (session, 0, 0, 0, 600);
|
|
|
|
|
|
|
|
g_signal_connect (session->proxy, "activated",
|
|
|
|
G_CALLBACK (on_activated), &data);
|
|
|
|
|
|
|
|
input_capture_session_enable (session);
|
|
|
|
|
|
|
|
write_state (session, "1");
|
|
|
|
|
|
|
|
while (data.activated_barrier_id == 0)
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
input_capture_session_clear_barriers (session);
|
|
|
|
write_state (session, "2");
|
|
|
|
wait_for_state (session, "1");
|
|
|
|
|
|
|
|
input_capture_session_close (session);
|
|
|
|
}
|
|
|
|
|
2022-04-27 08:50:43 +00:00
|
|
|
static void
|
|
|
|
test_cancel_keybinding (void)
|
|
|
|
{
|
|
|
|
InputCapture *input_capture;
|
|
|
|
InputCaptureSession *session;
|
|
|
|
g_autolist (Zone) zones = NULL;
|
|
|
|
|
|
|
|
input_capture = input_capture_new ();
|
|
|
|
session = input_capture_create_session (input_capture);
|
|
|
|
|
|
|
|
zones = input_capture_session_get_zones (session);
|
|
|
|
input_capture_session_add_barrier (session, 0, 0, 0, 600);
|
|
|
|
input_capture_session_enable (session);
|
|
|
|
|
|
|
|
write_state (session, "1");
|
|
|
|
wait_for_state (session, "1");
|
|
|
|
|
|
|
|
input_capture_session_close (session);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
test_events (void)
|
|
|
|
{
|
|
|
|
InputCapture *input_capture;
|
|
|
|
InputCaptureSession *session;
|
|
|
|
g_autolist (Zone) zones = NULL;
|
|
|
|
Event expected_events[] = {
|
|
|
|
/* Move the pointer with deltas (10, 15) and (2, -5), then click */
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_POINTER_MOTION,
|
|
|
|
.motion = { .dx = -10.0, .dy = -10.0 },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_FRAME,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_POINTER_MOTION,
|
|
|
|
.motion = { .dx = 2.0, .dy = -5.0 },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_FRAME,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_BUTTON_BUTTON,
|
|
|
|
.button = { .button = BTN_LEFT, .is_press = TRUE },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_FRAME,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_BUTTON_BUTTON,
|
|
|
|
.button = { .button = BTN_LEFT, .is_press = FALSE },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_FRAME,
|
|
|
|
},
|
|
|
|
|
|
|
|
/* Press, then release, KEY_A */
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_KEYBOARD_KEY,
|
|
|
|
.key = { .key = KEY_A, .is_press = TRUE },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_FRAME,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_KEYBOARD_KEY,
|
|
|
|
.key = { .key = KEY_A, .is_press = FALSE },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_FRAME,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
input_capture = input_capture_new ();
|
|
|
|
session = input_capture_create_session (input_capture);
|
|
|
|
|
|
|
|
input_capture_session_connect_to_eis (session);
|
|
|
|
zones = input_capture_session_get_zones (session);
|
|
|
|
input_capture_session_add_barrier (session, 0, 0, 0, 600);
|
|
|
|
|
|
|
|
input_capture_session_enable (session);
|
|
|
|
|
|
|
|
while (!session->has_pointer ||
|
|
|
|
!session->has_keyboard)
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
write_state (session, "1");
|
|
|
|
|
|
|
|
set_expected_events (session,
|
|
|
|
expected_events,
|
|
|
|
G_N_ELEMENTS (expected_events));
|
|
|
|
|
|
|
|
while (session->next_event < session->n_expected_events)
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
input_capture_session_close (session);
|
|
|
|
}
|
|
|
|
|
2022-04-28 15:46:34 +00:00
|
|
|
static void
|
|
|
|
test_a11y (void)
|
|
|
|
{
|
|
|
|
InputCapture *input_capture;
|
|
|
|
InputCaptureSession *session;
|
|
|
|
g_autolist (Zone) zones = NULL;
|
|
|
|
Event expected_events[] = {
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_POINTER_MOTION,
|
|
|
|
.motion = { .dx = -10.0, .dy = 0.0 },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_FRAME,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_BUTTON_BUTTON,
|
|
|
|
.button = { .button = BTN_LEFT, .is_press = TRUE },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_FRAME,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_BUTTON_BUTTON,
|
|
|
|
.button = { .button = BTN_LEFT, .is_press = FALSE },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
.type = EI_EVENT_FRAME,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
input_capture = input_capture_new ();
|
|
|
|
session = input_capture_create_session (input_capture);
|
|
|
|
|
|
|
|
input_capture_session_connect_to_eis (session);
|
|
|
|
zones = input_capture_session_get_zones (session);
|
|
|
|
input_capture_session_add_barrier (session, 0, 0, 0, 600);
|
|
|
|
input_capture_session_enable (session);
|
|
|
|
|
|
|
|
set_expected_events (session,
|
|
|
|
expected_events,
|
|
|
|
G_N_ELEMENTS (expected_events));
|
|
|
|
write_state (session, "1");
|
|
|
|
|
|
|
|
while (session->next_event < session->n_expected_events)
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
wait_for_state (session, "1");
|
|
|
|
|
|
|
|
input_capture_session_close (session);
|
|
|
|
}
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
static const struct
|
|
|
|
{
|
|
|
|
const char *name;
|
|
|
|
void (* func) (void);
|
|
|
|
} test_cases[] = {
|
|
|
|
{ "sanity", test_sanity, },
|
|
|
|
{ "zones", test_zones, },
|
2022-04-01 13:22:17 +00:00
|
|
|
{ "barriers", test_barriers, },
|
2022-04-01 17:14:48 +00:00
|
|
|
{ "clear-barriers", test_clear_barriers, },
|
2022-04-27 08:50:43 +00:00
|
|
|
{ "cancel-keybinding", test_cancel_keybinding, },
|
|
|
|
{ "events", test_events, },
|
2022-04-28 15:46:34 +00:00
|
|
|
{ "a11y", test_a11y, },
|
2022-03-30 15:27:51 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
print_to_stderr (const char *text)
|
|
|
|
{
|
|
|
|
fputs (text, stderr);
|
|
|
|
fflush (stderr);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
main (int argc,
|
|
|
|
char **argv)
|
|
|
|
{
|
|
|
|
const char *test_case;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
g_assert_cmpint (argc, ==, 2);
|
|
|
|
|
|
|
|
test_case = argv[1];
|
|
|
|
|
|
|
|
g_set_print_handler (print_to_stderr);
|
|
|
|
g_test_init (&argc, &argv, NULL);
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (test_cases); i++)
|
|
|
|
{
|
|
|
|
if (g_strcmp0 (test_cases[i].name, test_case) == 0)
|
|
|
|
{
|
2022-04-01 13:22:17 +00:00
|
|
|
g_autoptr (GInputStream) stdin_stream = NULL;
|
|
|
|
|
|
|
|
stdin_stream = g_unix_input_stream_new (fileno (stdin), FALSE);
|
|
|
|
stdin_reader = g_data_input_stream_new (stdin_stream);
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
test_cases[i].func ();
|
2022-04-01 13:22:17 +00:00
|
|
|
|
|
|
|
g_clear_object (&stdin_reader);
|
|
|
|
g_clear_object (&stdin_stream);
|
|
|
|
|
2022-03-30 15:27:51 +00:00
|
|
|
return EXIT_SUCCESS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_warning ("Invalid test case '%s'", test_case);
|
|
|
|
return EXIT_FAILURE;
|
|
|
|
}
|