2019-12-06 18:05:32 +00:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 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 <gio/gio.h>
|
|
|
|
|
2021-09-06 15:04:45 +00:00
|
|
|
#include "backends/meta-virtual-monitor.h"
|
2021-09-07 08:35:27 +00:00
|
|
|
#include "compositor/meta-window-actor-private.h"
|
2021-06-18 21:51:39 +00:00
|
|
|
#include "core/display-private.h"
|
|
|
|
#include "core/window-private.h"
|
2021-11-08 09:52:48 +00:00
|
|
|
#include "meta-test/meta-context-test.h"
|
2022-01-27 23:12:59 +00:00
|
|
|
#include "tests/meta-test-utils.h"
|
2021-12-17 23:15:56 +00:00
|
|
|
#include "meta/meta-later.h"
|
|
|
|
#include "meta/meta-workspace-manager.h"
|
2021-04-10 15:32:02 +00:00
|
|
|
#include "tests/meta-wayland-test-driver.h"
|
2022-01-27 10:51:26 +00:00
|
|
|
#include "tests/meta-wayland-test-utils.h"
|
2019-12-06 18:05:32 +00:00
|
|
|
#include "wayland/meta-wayland-surface.h"
|
|
|
|
|
2021-09-06 15:04:45 +00:00
|
|
|
static MetaContext *test_context;
|
2021-04-10 15:32:02 +00:00
|
|
|
static MetaWaylandTestDriver *test_driver;
|
2021-09-06 15:04:45 +00:00
|
|
|
static MetaVirtualMonitor *virtual_monitor;
|
2021-09-07 08:35:27 +00:00
|
|
|
static ClutterVirtualInputDevice *virtual_pointer;
|
2021-04-10 15:32:02 +00:00
|
|
|
|
2021-06-18 21:51:39 +00:00
|
|
|
static MetaWindow *
|
|
|
|
find_client_window (const char *title)
|
|
|
|
{
|
|
|
|
MetaDisplay *display = meta_get_display ();
|
|
|
|
g_autoptr (GSList) windows = NULL;
|
|
|
|
GSList *l;
|
|
|
|
|
|
|
|
windows = meta_display_list_windows (display, META_LIST_DEFAULT);
|
|
|
|
for (l = windows; l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaWindow *window = l->data;
|
|
|
|
|
|
|
|
if (g_strcmp0 (meta_window_get_title (window), title) == 0)
|
|
|
|
return window;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-12-06 18:05:32 +00:00
|
|
|
static void
|
|
|
|
subsurface_remap_toplevel (void)
|
|
|
|
{
|
2022-01-27 10:51:26 +00:00
|
|
|
MetaWaylandTestClient *wayland_test_client;
|
2019-12-06 18:05:32 +00:00
|
|
|
|
2022-01-27 10:51:26 +00:00
|
|
|
wayland_test_client =
|
|
|
|
meta_wayland_test_client_new ("subsurface-remap-toplevel");
|
|
|
|
meta_wayland_test_client_finish (wayland_test_client);
|
2019-12-06 18:05:32 +00:00
|
|
|
}
|
|
|
|
|
2021-08-06 16:17:18 +00:00
|
|
|
static void
|
|
|
|
subsurface_reparenting (void)
|
|
|
|
{
|
2022-01-27 10:51:26 +00:00
|
|
|
MetaWaylandTestClient *wayland_test_client;
|
2021-08-06 16:17:18 +00:00
|
|
|
|
2022-01-27 10:51:26 +00:00
|
|
|
wayland_test_client =
|
|
|
|
meta_wayland_test_client_new ("subsurface-reparenting");
|
|
|
|
meta_wayland_test_client_finish (wayland_test_client);
|
2021-08-06 16:17:18 +00:00
|
|
|
}
|
|
|
|
|
2020-11-05 08:56:30 +00:00
|
|
|
static void
|
|
|
|
subsurface_invalid_subsurfaces (void)
|
|
|
|
{
|
2022-01-27 10:51:26 +00:00
|
|
|
MetaWaylandTestClient *wayland_test_client;
|
2020-11-05 08:56:30 +00:00
|
|
|
|
2022-01-27 10:51:26 +00:00
|
|
|
wayland_test_client =
|
|
|
|
meta_wayland_test_client_new ("invalid-subsurfaces");
|
2020-11-05 08:56:30 +00:00
|
|
|
g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING,
|
|
|
|
"WL: error in client communication*");
|
|
|
|
g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING,
|
|
|
|
"WL: error in client communication*");
|
2022-01-27 10:51:26 +00:00
|
|
|
meta_wayland_test_client_finish (wayland_test_client);
|
2020-11-05 08:56:30 +00:00
|
|
|
g_test_assert_expected_messages ();
|
|
|
|
}
|
|
|
|
|
2020-11-11 09:49:25 +00:00
|
|
|
static void
|
|
|
|
subsurface_invalid_xdg_shell_actions (void)
|
|
|
|
{
|
2022-01-27 10:51:26 +00:00
|
|
|
MetaWaylandTestClient *wayland_test_client;
|
2020-11-11 09:49:25 +00:00
|
|
|
|
2022-01-27 10:51:26 +00:00
|
|
|
wayland_test_client =
|
|
|
|
meta_wayland_test_client_new ("invalid-xdg-shell-actions");
|
2020-11-11 09:49:25 +00:00
|
|
|
g_test_expect_message (G_LOG_DOMAIN, G_LOG_LEVEL_WARNING,
|
|
|
|
"Invalid geometry * set on xdg_surface*");
|
2022-01-27 10:51:26 +00:00
|
|
|
meta_wayland_test_client_finish (wayland_test_client);
|
2020-11-11 09:49:25 +00:00
|
|
|
g_test_assert_expected_messages ();
|
|
|
|
}
|
|
|
|
|
2021-09-07 08:35:27 +00:00
|
|
|
static void
|
|
|
|
on_after_paint (ClutterStage *stage,
|
|
|
|
ClutterStageView *view,
|
|
|
|
gboolean *was_painted)
|
|
|
|
{
|
|
|
|
*was_painted = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
wait_for_paint (ClutterActor *stage)
|
|
|
|
{
|
|
|
|
gboolean was_painted = FALSE;
|
|
|
|
gulong was_painted_id;
|
|
|
|
|
|
|
|
was_painted_id = g_signal_connect (CLUTTER_STAGE (stage),
|
|
|
|
"after-paint",
|
|
|
|
G_CALLBACK (on_after_paint),
|
|
|
|
&was_painted);
|
|
|
|
|
|
|
|
while (!was_painted)
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
|
|
|
|
g_signal_handler_disconnect (stage, was_painted_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
on_effects_completed_idle (gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaWindowActor *actor = user_data;
|
|
|
|
MetaBackend *backend = meta_context_get_backend (test_context);
|
|
|
|
ClutterActor *stage = meta_backend_get_stage (backend);
|
|
|
|
MetaWindow *window = meta_window_actor_get_meta_window (actor);
|
|
|
|
MetaRectangle buffer_rect;
|
|
|
|
|
|
|
|
/* Move the window to a known position and perform a mouse click, allowing a
|
|
|
|
* popup to be mapped. */
|
|
|
|
|
|
|
|
meta_window_move_frame (window, FALSE, 0, 0);
|
|
|
|
|
|
|
|
clutter_actor_queue_redraw (stage);
|
|
|
|
clutter_stage_schedule_update (CLUTTER_STAGE (stage));
|
|
|
|
|
|
|
|
wait_for_paint (stage);
|
|
|
|
|
|
|
|
meta_window_get_buffer_rect (window, &buffer_rect);
|
|
|
|
clutter_virtual_input_device_notify_absolute_motion (virtual_pointer,
|
|
|
|
CLUTTER_CURRENT_TIME,
|
|
|
|
buffer_rect.x + 10,
|
|
|
|
buffer_rect.y + 10);
|
|
|
|
|
|
|
|
clutter_virtual_input_device_notify_button (virtual_pointer,
|
|
|
|
CLUTTER_CURRENT_TIME,
|
|
|
|
CLUTTER_BUTTON_PRIMARY,
|
|
|
|
CLUTTER_BUTTON_STATE_PRESSED);
|
|
|
|
clutter_virtual_input_device_notify_button (virtual_pointer,
|
|
|
|
CLUTTER_CURRENT_TIME,
|
|
|
|
CLUTTER_BUTTON_PRIMARY,
|
|
|
|
CLUTTER_BUTTON_STATE_RELEASED);
|
|
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_effects_completed (MetaWindowActor *actor)
|
|
|
|
{
|
|
|
|
g_idle_add (on_effects_completed_idle, actor);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_window_added (MetaStack *stack,
|
|
|
|
MetaWindow *window)
|
|
|
|
{
|
|
|
|
MetaWindowActor *actor = meta_window_actor_from_window (window);
|
|
|
|
|
|
|
|
g_assert_nonnull (actor);
|
|
|
|
|
|
|
|
if (g_strcmp0 (meta_window_get_title (window),
|
|
|
|
"subsurface-parent-unmapped") != 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_signal_connect (actor, "effects-completed",
|
|
|
|
G_CALLBACK (on_effects_completed),
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_window_actor_destroyed (MetaWindowActor *actor,
|
|
|
|
MetaWaylandTestDriver *test_driver)
|
|
|
|
{
|
|
|
|
meta_wayland_test_driver_emit_sync_event (test_driver, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_unmap_sync_point (MetaWaylandTestDriver *test_driver,
|
|
|
|
unsigned int sequence,
|
|
|
|
struct wl_resource *surface_resource,
|
|
|
|
struct wl_client *wl_client)
|
|
|
|
{
|
|
|
|
if (sequence == 0)
|
|
|
|
{
|
|
|
|
/* Dismiss popup by clicking outside. */
|
|
|
|
|
|
|
|
clutter_virtual_input_device_notify_absolute_motion (virtual_pointer,
|
|
|
|
CLUTTER_CURRENT_TIME,
|
|
|
|
390, 390);
|
|
|
|
|
|
|
|
clutter_virtual_input_device_notify_button (virtual_pointer,
|
|
|
|
CLUTTER_CURRENT_TIME,
|
|
|
|
CLUTTER_BUTTON_PRIMARY,
|
|
|
|
CLUTTER_BUTTON_STATE_PRESSED);
|
|
|
|
clutter_virtual_input_device_notify_button (virtual_pointer,
|
|
|
|
CLUTTER_CURRENT_TIME,
|
|
|
|
CLUTTER_BUTTON_PRIMARY,
|
|
|
|
CLUTTER_BUTTON_STATE_RELEASED);
|
|
|
|
|
|
|
|
MetaWaylandSurface *surface = wl_resource_get_user_data (surface_resource);
|
|
|
|
ClutterActor *actor = CLUTTER_ACTOR (meta_wayland_surface_get_actor (surface));
|
|
|
|
MetaWindowActor *window_actor = meta_window_actor_from_actor (actor);
|
|
|
|
g_signal_connect (window_actor, "destroy",
|
|
|
|
G_CALLBACK (on_window_actor_destroyed),
|
|
|
|
test_driver);
|
|
|
|
}
|
|
|
|
else if (sequence == 1)
|
|
|
|
{
|
|
|
|
MetaWaylandSurface *surface = wl_resource_get_user_data (surface_resource);
|
|
|
|
ClutterActor *actor = CLUTTER_ACTOR (meta_wayland_surface_get_actor (surface));
|
|
|
|
MetaWindowActor *window_actor = meta_window_actor_from_actor (actor);
|
|
|
|
MetaWindow *window = meta_window_actor_get_meta_window (window_actor);
|
|
|
|
MetaRectangle buffer_rect;
|
|
|
|
|
|
|
|
/* Click inside the window to allow mapping a popup. */
|
|
|
|
|
|
|
|
meta_window_get_buffer_rect (window, &buffer_rect);
|
|
|
|
clutter_virtual_input_device_notify_absolute_motion (virtual_pointer,
|
|
|
|
CLUTTER_CURRENT_TIME,
|
|
|
|
buffer_rect.x + 10,
|
|
|
|
buffer_rect.y + 10);
|
|
|
|
|
|
|
|
clutter_virtual_input_device_notify_button (virtual_pointer,
|
|
|
|
CLUTTER_CURRENT_TIME,
|
|
|
|
CLUTTER_BUTTON_PRIMARY,
|
|
|
|
CLUTTER_BUTTON_STATE_PRESSED);
|
|
|
|
clutter_virtual_input_device_notify_button (virtual_pointer,
|
|
|
|
CLUTTER_CURRENT_TIME,
|
|
|
|
CLUTTER_BUTTON_PRIMARY,
|
|
|
|
CLUTTER_BUTTON_STATE_RELEASED);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
subsurface_parent_unmapped (void)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = meta_context_get_backend (test_context);
|
|
|
|
MetaDisplay *display = meta_context_get_display (test_context);
|
2022-01-27 10:51:26 +00:00
|
|
|
MetaWaylandTestClient *wayland_test_client;
|
2021-09-07 08:35:27 +00:00
|
|
|
ClutterSeat *seat;
|
|
|
|
gulong window_added_id;
|
|
|
|
gulong sync_point_id;
|
|
|
|
|
|
|
|
seat = meta_backend_get_default_seat (backend);
|
|
|
|
virtual_pointer = clutter_seat_create_virtual_device (seat,
|
|
|
|
CLUTTER_POINTER_DEVICE);
|
|
|
|
|
2022-01-27 10:51:26 +00:00
|
|
|
wayland_test_client =
|
|
|
|
meta_wayland_test_client_new ("subsurface-parent-unmapped");
|
2021-09-07 08:35:27 +00:00
|
|
|
|
|
|
|
window_added_id =
|
|
|
|
g_signal_connect (display->stack, "window-added",
|
|
|
|
G_CALLBACK (on_window_added),
|
|
|
|
virtual_pointer);
|
|
|
|
sync_point_id =
|
|
|
|
g_signal_connect (test_driver, "sync-point",
|
|
|
|
G_CALLBACK (on_unmap_sync_point),
|
|
|
|
NULL);
|
|
|
|
|
2022-01-27 10:51:26 +00:00
|
|
|
meta_wayland_test_client_finish (wayland_test_client);
|
2021-09-07 08:35:27 +00:00
|
|
|
|
|
|
|
g_clear_object (&virtual_pointer);
|
|
|
|
g_signal_handler_disconnect (test_driver, sync_point_id);
|
|
|
|
g_signal_handler_disconnect (display->stack, window_added_id);
|
|
|
|
}
|
|
|
|
|
2021-06-18 21:51:39 +00:00
|
|
|
typedef enum _ApplyLimitState
|
|
|
|
{
|
|
|
|
APPLY_LIMIT_STATE_INIT,
|
|
|
|
APPLY_LIMIT_STATE_RESET,
|
|
|
|
APPLY_LIMIT_STATE_FINISH,
|
|
|
|
} ApplyLimitState;
|
|
|
|
|
|
|
|
typedef struct _ApplyLimitData
|
|
|
|
{
|
|
|
|
GMainLoop *loop;
|
2022-01-27 10:51:26 +00:00
|
|
|
MetaWaylandTestClient *wayland_test_client;
|
2021-06-18 21:51:39 +00:00
|
|
|
ApplyLimitState state;
|
|
|
|
} ApplyLimitData;
|
|
|
|
|
|
|
|
static void
|
2021-12-17 23:11:33 +00:00
|
|
|
on_apply_limits_sync_point (MetaWaylandTestDriver *test_driver,
|
|
|
|
unsigned int sequence,
|
|
|
|
struct wl_resource *surface_resource,
|
|
|
|
struct wl_client *wl_client,
|
|
|
|
ApplyLimitData *data)
|
2021-06-18 21:51:39 +00:00
|
|
|
{
|
|
|
|
MetaWindow *window;
|
|
|
|
|
|
|
|
if (sequence == 0)
|
|
|
|
g_assert (data->state == APPLY_LIMIT_STATE_INIT);
|
|
|
|
else if (sequence == 0)
|
|
|
|
g_assert (data->state == APPLY_LIMIT_STATE_RESET);
|
|
|
|
|
|
|
|
window = find_client_window ("toplevel-limits-test");
|
|
|
|
|
|
|
|
if (sequence == 0)
|
|
|
|
{
|
|
|
|
g_assert_nonnull (window);
|
|
|
|
g_assert_cmpint (window->size_hints.max_width, ==, 700);
|
|
|
|
g_assert_cmpint (window->size_hints.max_height, ==, 500);
|
|
|
|
g_assert_cmpint (window->size_hints.min_width, ==, 700);
|
|
|
|
g_assert_cmpint (window->size_hints.min_height, ==, 500);
|
|
|
|
|
|
|
|
data->state = APPLY_LIMIT_STATE_RESET;
|
|
|
|
}
|
|
|
|
else if (sequence == 1)
|
|
|
|
{
|
|
|
|
g_assert_null (window);
|
|
|
|
data->state = APPLY_LIMIT_STATE_FINISH;
|
|
|
|
g_main_loop_quit (data->loop);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_assert_not_reached ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
toplevel_apply_limits (void)
|
|
|
|
{
|
|
|
|
ApplyLimitData data = {};
|
2021-09-07 08:34:04 +00:00
|
|
|
gulong handler_id;
|
2021-06-18 21:51:39 +00:00
|
|
|
|
|
|
|
data.loop = g_main_loop_new (NULL, FALSE);
|
2022-01-27 10:51:26 +00:00
|
|
|
data.wayland_test_client = meta_wayland_test_client_new ("xdg-apply-limits");
|
2021-09-07 08:34:04 +00:00
|
|
|
handler_id = g_signal_connect (test_driver, "sync-point",
|
2021-12-17 23:11:33 +00:00
|
|
|
G_CALLBACK (on_apply_limits_sync_point),
|
|
|
|
&data);
|
2021-06-18 21:51:39 +00:00
|
|
|
g_main_loop_run (data.loop);
|
|
|
|
g_assert_cmpint (data.state, ==, APPLY_LIMIT_STATE_FINISH);
|
2022-01-27 10:51:26 +00:00
|
|
|
meta_wayland_test_client_finish (data.wayland_test_client);
|
2021-06-18 21:51:39 +00:00
|
|
|
g_test_assert_expected_messages ();
|
2021-09-07 08:34:04 +00:00
|
|
|
g_signal_handler_disconnect (test_driver, handler_id);
|
2021-06-18 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
wayland/activation: Don't grab if no serial or surface was provided
A client can create a token without any seat, serial, or surface. In
this case, we'd still try to grab, which would run into some unforseen
code paths, potentially resulting in the following crash:
0) meta_wayland_tablet_seat_device_added (tablet_seat=0x55dff4271c90,
device=0x7f87b80655b0) at
../src/wayland/meta-wayland-tablet-seat.c:200
1) meta_wayland_tablet_seat_new (seat=0x0, manager=0x55dff3ec7b40) at
../src/wayland/meta-wayland-tablet-seat.c:283
2) meta_wayland_tablet_manager_ensure_seat (manager=manager@entry=0x55dff3ec7b40,
seat=seat@entry=0x0) at
../src/wayland/meta-wayland-tablet-manager.c:239
3) meta_wayland_tablet_manager_ensure_seat (seat=0x0, manager=0x55dff3ec7b40) at
../src/wayland/meta-wayland-touch.c:595
4) meta_wayland_seat_get_grab_info (seat=0x0, surface=0x55dff43ff5b0,
serial=0, require_pressed=0, x=0x0, y=0x0) at
../src/wayland/meta-wayland-seat.c:479
5) activation_activate (...) at
../src/wayland/meta-wayland-activation.c:261
Fix this by not trying to grab if not enough parameters was passed when
creating the token. Also add a test case that reproduces the above
crash.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2081>
2021-11-08 10:13:06 +00:00
|
|
|
static void
|
|
|
|
toplevel_activation (void)
|
|
|
|
{
|
|
|
|
ApplyLimitData data = {};
|
|
|
|
|
|
|
|
data.loop = g_main_loop_new (NULL, FALSE);
|
2022-01-27 10:51:26 +00:00
|
|
|
data.wayland_test_client = meta_wayland_test_client_new ("xdg-activation");
|
|
|
|
meta_wayland_test_client_finish (data.wayland_test_client);
|
wayland/activation: Don't grab if no serial or surface was provided
A client can create a token without any seat, serial, or surface. In
this case, we'd still try to grab, which would run into some unforseen
code paths, potentially resulting in the following crash:
0) meta_wayland_tablet_seat_device_added (tablet_seat=0x55dff4271c90,
device=0x7f87b80655b0) at
../src/wayland/meta-wayland-tablet-seat.c:200
1) meta_wayland_tablet_seat_new (seat=0x0, manager=0x55dff3ec7b40) at
../src/wayland/meta-wayland-tablet-seat.c:283
2) meta_wayland_tablet_manager_ensure_seat (manager=manager@entry=0x55dff3ec7b40,
seat=seat@entry=0x0) at
../src/wayland/meta-wayland-tablet-manager.c:239
3) meta_wayland_tablet_manager_ensure_seat (seat=0x0, manager=0x55dff3ec7b40) at
../src/wayland/meta-wayland-touch.c:595
4) meta_wayland_seat_get_grab_info (seat=0x0, surface=0x55dff43ff5b0,
serial=0, require_pressed=0, x=0x0, y=0x0) at
../src/wayland/meta-wayland-seat.c:479
5) activation_activate (...) at
../src/wayland/meta-wayland-activation.c:261
Fix this by not trying to grab if not enough parameters was passed when
creating the token. Also add a test case that reproduces the above
crash.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2081>
2021-11-08 10:13:06 +00:00
|
|
|
}
|
|
|
|
|
2021-12-17 23:15:56 +00:00
|
|
|
static void
|
|
|
|
on_sync_point (MetaWaylandTestDriver *test_driver,
|
|
|
|
unsigned int sequence,
|
|
|
|
struct wl_resource *surface_resource,
|
|
|
|
struct wl_client *wl_client,
|
|
|
|
unsigned int *latest_sequence)
|
|
|
|
{
|
|
|
|
*latest_sequence = sequence;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
wait_for_sync_point (unsigned int sync_point)
|
|
|
|
{
|
|
|
|
gulong handler_id;
|
|
|
|
unsigned int latest_sequence = 0;
|
|
|
|
|
|
|
|
handler_id = g_signal_connect (test_driver, "sync-point",
|
|
|
|
G_CALLBACK (on_sync_point),
|
|
|
|
&latest_sequence);
|
|
|
|
while (latest_sequence != sync_point)
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
g_signal_handler_disconnect (test_driver, handler_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
mark_later_as_done (gpointer user_data)
|
|
|
|
{
|
|
|
|
gboolean *done = user_data;
|
|
|
|
|
|
|
|
*done = TRUE;
|
|
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
wait_until_after_paint (void)
|
|
|
|
{
|
|
|
|
MetaDisplay *display = meta_context_get_display (test_context);
|
|
|
|
MetaCompositor *compositor = meta_display_get_compositor (display);
|
|
|
|
MetaLaters *laters = meta_compositor_get_laters (compositor);
|
|
|
|
gboolean done;
|
|
|
|
|
|
|
|
done = FALSE;
|
|
|
|
meta_laters_add (laters,
|
|
|
|
META_LATER_BEFORE_REDRAW,
|
|
|
|
mark_later_as_done,
|
|
|
|
&done,
|
|
|
|
NULL);
|
|
|
|
while (!done)
|
|
|
|
g_main_context_iteration (NULL, FALSE);
|
|
|
|
|
|
|
|
done = FALSE;
|
|
|
|
meta_laters_add (laters,
|
|
|
|
META_LATER_IDLE,
|
|
|
|
mark_later_as_done,
|
|
|
|
&done,
|
|
|
|
NULL);
|
|
|
|
while (!done)
|
|
|
|
g_main_context_iteration (NULL, FALSE);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
set_struts (MetaRectangle rect,
|
|
|
|
MetaSide side)
|
|
|
|
{
|
|
|
|
MetaDisplay *display = meta_context_get_display (test_context);
|
|
|
|
MetaWorkspaceManager *workspace_manager =
|
|
|
|
meta_display_get_workspace_manager (display);
|
|
|
|
GList *workspaces =
|
|
|
|
meta_workspace_manager_get_workspaces (workspace_manager);
|
|
|
|
MetaStrut strut;
|
|
|
|
g_autoptr (GSList) struts = NULL;
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
strut = (MetaStrut) { .rect = rect, .side = side };
|
|
|
|
struts = g_slist_append (NULL, &strut);
|
|
|
|
|
|
|
|
for (l = workspaces; l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaWorkspace *workspace = l->data;
|
|
|
|
|
|
|
|
meta_workspace_set_builtin_struts (workspace, struts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
clear_struts (void)
|
|
|
|
{
|
|
|
|
MetaDisplay *display = meta_context_get_display (test_context);
|
|
|
|
MetaWorkspaceManager *workspace_manager =
|
|
|
|
meta_display_get_workspace_manager (display);
|
|
|
|
GList *workspaces =
|
|
|
|
meta_workspace_manager_get_workspaces (workspace_manager);
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
for (l = workspaces; l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaWorkspace *workspace = l->data;
|
|
|
|
|
|
|
|
meta_workspace_set_builtin_struts (workspace, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaRectangle
|
|
|
|
get_primary_logical_monitor_layout (void)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = meta_context_get_backend (test_context);
|
|
|
|
MetaMonitorManager *monitor_manager =
|
|
|
|
meta_backend_get_monitor_manager (backend);
|
|
|
|
MetaLogicalMonitor *logical_monitor;
|
|
|
|
|
|
|
|
logical_monitor =
|
|
|
|
meta_monitor_manager_get_primary_logical_monitor (monitor_manager);
|
|
|
|
return meta_logical_monitor_get_layout (logical_monitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
toplevel_bounds_struts (void)
|
|
|
|
{
|
|
|
|
MetaWaylandTestClient *wayland_test_client;
|
|
|
|
MetaWindow *window;
|
|
|
|
MetaRectangle logical_monitor_layout;
|
|
|
|
MetaRectangle work_area;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This test case makes sure that setting and changing struts result in the
|
|
|
|
* right bounds are sent.
|
|
|
|
*/
|
|
|
|
|
|
|
|
logical_monitor_layout = get_primary_logical_monitor_layout ();
|
|
|
|
set_struts ((MetaRectangle) {
|
|
|
|
.x = 0,
|
|
|
|
.y = 0,
|
|
|
|
.width = logical_monitor_layout.width,
|
|
|
|
.height = 10,
|
|
|
|
},
|
|
|
|
META_SIDE_TOP);
|
|
|
|
|
|
|
|
wayland_test_client = meta_wayland_test_client_new ("xdg-toplevel-bounds");
|
|
|
|
|
|
|
|
wait_for_sync_point (1);
|
|
|
|
wait_until_after_paint ();
|
|
|
|
|
|
|
|
window = find_client_window ("toplevel-bounds-test");
|
|
|
|
|
|
|
|
g_assert_nonnull (window->monitor);
|
|
|
|
meta_window_get_work_area_current_monitor (window, &work_area);
|
|
|
|
g_assert_cmpint (work_area.width, ==, logical_monitor_layout.width);
|
|
|
|
g_assert_cmpint (work_area.height, ==, logical_monitor_layout.height - 10);
|
|
|
|
|
|
|
|
g_assert_cmpint (window->rect.width, ==, work_area.width - 10);
|
|
|
|
g_assert_cmpint (window->rect.height, ==, work_area.height - 10);
|
|
|
|
|
|
|
|
meta_wayland_test_driver_emit_sync_event (test_driver, 0);
|
|
|
|
meta_wayland_test_client_finish (wayland_test_client);
|
|
|
|
|
|
|
|
clear_struts ();
|
|
|
|
|
|
|
|
wayland_test_client = meta_wayland_test_client_new ("xdg-toplevel-bounds");
|
|
|
|
|
|
|
|
wait_for_sync_point (1);
|
|
|
|
wait_until_after_paint ();
|
|
|
|
|
|
|
|
window = find_client_window ("toplevel-bounds-test");
|
|
|
|
g_assert_nonnull (window->monitor);
|
|
|
|
meta_window_get_work_area_current_monitor (window, &work_area);
|
|
|
|
g_assert_cmpint (work_area.width, ==, logical_monitor_layout.width);
|
|
|
|
g_assert_cmpint (work_area.height, ==, logical_monitor_layout.height);
|
|
|
|
|
|
|
|
g_assert_cmpint (window->rect.width, ==, work_area.width - 10);
|
|
|
|
g_assert_cmpint (window->rect.height, ==, work_area.height - 10);
|
|
|
|
|
|
|
|
meta_wayland_test_driver_emit_sync_event (test_driver, 0);
|
|
|
|
meta_wayland_test_client_finish (wayland_test_client);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
wait_for_cursor_position (float x,
|
|
|
|
float y)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = meta_context_get_backend (test_context);
|
|
|
|
MetaCursorTracker *cursor_tracker = meta_backend_get_cursor_tracker (backend);
|
|
|
|
graphene_point_t point;
|
|
|
|
|
|
|
|
while (TRUE)
|
|
|
|
{
|
|
|
|
meta_cursor_tracker_get_pointer (cursor_tracker, &point, NULL);
|
|
|
|
if (G_APPROX_VALUE (x, point.x, FLT_EPSILON) &&
|
|
|
|
G_APPROX_VALUE (y, point.y, FLT_EPSILON))
|
|
|
|
break;
|
|
|
|
|
|
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
toplevel_bounds_monitors (void)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = meta_context_get_backend (test_context);
|
|
|
|
ClutterSeat *seat;
|
|
|
|
g_autoptr (MetaVirtualMonitor) second_virtual_monitor = NULL;
|
|
|
|
MetaWaylandTestClient *wayland_test_client;
|
|
|
|
MetaRectangle logical_monitor_layout;
|
|
|
|
MetaRectangle work_area;
|
|
|
|
MetaWindow *window;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* This test case creates two monitors, with different sizes, with a fake
|
|
|
|
* panel on top of the primary monitor. It then makes sure launching on both
|
|
|
|
* monitors results in the correct bounds.
|
|
|
|
*/
|
|
|
|
|
|
|
|
seat = meta_backend_get_default_seat (backend);
|
|
|
|
virtual_pointer = clutter_seat_create_virtual_device (seat,
|
|
|
|
CLUTTER_POINTER_DEVICE);
|
|
|
|
|
|
|
|
second_virtual_monitor = meta_create_test_monitor (test_context,
|
|
|
|
300, 200, 60.0);
|
|
|
|
|
|
|
|
logical_monitor_layout = get_primary_logical_monitor_layout ();
|
|
|
|
set_struts ((MetaRectangle) {
|
|
|
|
.x = 0,
|
|
|
|
.y = 0,
|
|
|
|
.width = logical_monitor_layout.width,
|
|
|
|
.height = 10,
|
|
|
|
},
|
|
|
|
META_SIDE_TOP);
|
|
|
|
|
|
|
|
wayland_test_client = meta_wayland_test_client_new ("xdg-toplevel-bounds");
|
|
|
|
|
|
|
|
wait_for_sync_point (1);
|
|
|
|
wait_until_after_paint ();
|
|
|
|
|
|
|
|
window = find_client_window ("toplevel-bounds-test");
|
|
|
|
|
|
|
|
g_assert_nonnull (window->monitor);
|
|
|
|
meta_window_get_work_area_current_monitor (window, &work_area);
|
|
|
|
g_assert_cmpint (work_area.width, ==, logical_monitor_layout.width);
|
|
|
|
g_assert_cmpint (work_area.height, ==, logical_monitor_layout.height - 10);
|
|
|
|
|
|
|
|
g_assert_cmpint (window->rect.width, ==, work_area.width - 10);
|
|
|
|
g_assert_cmpint (window->rect.height, ==, work_area.height - 10);
|
|
|
|
|
|
|
|
meta_wayland_test_driver_emit_sync_event (test_driver, 0);
|
|
|
|
meta_wayland_test_client_finish (wayland_test_client);
|
|
|
|
|
|
|
|
clutter_virtual_input_device_notify_absolute_motion (virtual_pointer,
|
|
|
|
CLUTTER_CURRENT_TIME,
|
|
|
|
550.0, 100.0);
|
|
|
|
wait_for_cursor_position (550.0, 100.0);
|
|
|
|
|
|
|
|
wayland_test_client = meta_wayland_test_client_new ("xdg-toplevel-bounds");
|
|
|
|
|
|
|
|
wait_for_sync_point (1);
|
|
|
|
wait_until_after_paint ();
|
|
|
|
|
|
|
|
window = find_client_window ("toplevel-bounds-test");
|
|
|
|
|
|
|
|
g_assert_nonnull (window->monitor);
|
|
|
|
meta_window_get_work_area_current_monitor (window, &work_area);
|
|
|
|
g_assert_cmpint (work_area.width, ==, 300);
|
|
|
|
g_assert_cmpint (work_area.height, ==, 200);
|
|
|
|
|
|
|
|
g_assert_cmpint (window->rect.width, ==, 300 - 10);
|
|
|
|
g_assert_cmpint (window->rect.height, ==, 200 - 10);
|
|
|
|
|
|
|
|
meta_wayland_test_driver_emit_sync_event (test_driver, 0);
|
|
|
|
meta_wayland_test_client_finish (wayland_test_client);
|
|
|
|
}
|
|
|
|
|
2021-11-08 09:52:48 +00:00
|
|
|
static void
|
2021-09-06 15:04:45 +00:00
|
|
|
on_before_tests (void)
|
2019-12-06 18:05:32 +00:00
|
|
|
{
|
2021-09-06 15:04:45 +00:00
|
|
|
MetaWaylandCompositor *compositor =
|
|
|
|
meta_context_get_wayland_compositor (test_context);
|
2019-12-06 18:05:32 +00:00
|
|
|
|
2021-04-10 15:32:02 +00:00
|
|
|
test_driver = meta_wayland_test_driver_new (compositor);
|
2022-01-27 23:12:59 +00:00
|
|
|
virtual_monitor = meta_create_test_monitor (test_context,
|
|
|
|
400, 400, 60.0);
|
2021-09-06 15:04:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_after_tests (void)
|
|
|
|
{
|
2022-01-27 23:12:59 +00:00
|
|
|
g_clear_object (&test_driver);
|
2021-09-06 15:04:45 +00:00
|
|
|
g_clear_object (&virtual_monitor);
|
2019-12-06 18:05:32 +00:00
|
|
|
}
|
|
|
|
|
2021-11-08 09:52:48 +00:00
|
|
|
static void
|
2021-11-25 09:26:44 +00:00
|
|
|
init_tests (void)
|
2019-12-06 18:05:32 +00:00
|
|
|
{
|
|
|
|
g_test_add_func ("/wayland/subsurface/remap-toplevel",
|
|
|
|
subsurface_remap_toplevel);
|
2021-08-06 16:17:18 +00:00
|
|
|
g_test_add_func ("/wayland/subsurface/reparent",
|
|
|
|
subsurface_reparenting);
|
2020-11-05 08:56:30 +00:00
|
|
|
g_test_add_func ("/wayland/subsurface/invalid-subsurfaces",
|
|
|
|
subsurface_invalid_subsurfaces);
|
2020-11-11 09:49:25 +00:00
|
|
|
g_test_add_func ("/wayland/subsurface/invalid-xdg-shell-actions",
|
|
|
|
subsurface_invalid_xdg_shell_actions);
|
2021-09-07 08:35:27 +00:00
|
|
|
g_test_add_func ("/wayland/subsurface/parent-unmapped",
|
|
|
|
subsurface_parent_unmapped);
|
2021-06-18 21:51:39 +00:00
|
|
|
g_test_add_func ("/wayland/toplevel/apply-limits",
|
|
|
|
toplevel_apply_limits);
|
wayland/activation: Don't grab if no serial or surface was provided
A client can create a token without any seat, serial, or surface. In
this case, we'd still try to grab, which would run into some unforseen
code paths, potentially resulting in the following crash:
0) meta_wayland_tablet_seat_device_added (tablet_seat=0x55dff4271c90,
device=0x7f87b80655b0) at
../src/wayland/meta-wayland-tablet-seat.c:200
1) meta_wayland_tablet_seat_new (seat=0x0, manager=0x55dff3ec7b40) at
../src/wayland/meta-wayland-tablet-seat.c:283
2) meta_wayland_tablet_manager_ensure_seat (manager=manager@entry=0x55dff3ec7b40,
seat=seat@entry=0x0) at
../src/wayland/meta-wayland-tablet-manager.c:239
3) meta_wayland_tablet_manager_ensure_seat (seat=0x0, manager=0x55dff3ec7b40) at
../src/wayland/meta-wayland-touch.c:595
4) meta_wayland_seat_get_grab_info (seat=0x0, surface=0x55dff43ff5b0,
serial=0, require_pressed=0, x=0x0, y=0x0) at
../src/wayland/meta-wayland-seat.c:479
5) activation_activate (...) at
../src/wayland/meta-wayland-activation.c:261
Fix this by not trying to grab if not enough parameters was passed when
creating the token. Also add a test case that reproduces the above
crash.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2081>
2021-11-08 10:13:06 +00:00
|
|
|
g_test_add_func ("/wayland/toplevel/activation",
|
|
|
|
toplevel_activation);
|
2021-12-17 23:15:56 +00:00
|
|
|
g_test_add_func ("/wayland/toplevel/bounds/struts",
|
|
|
|
toplevel_bounds_struts);
|
|
|
|
g_test_add_func ("/wayland/toplevel/bounds/monitors",
|
|
|
|
toplevel_bounds_monitors);
|
2019-12-06 18:05:32 +00:00
|
|
|
}
|
2021-11-08 09:52:48 +00:00
|
|
|
|
|
|
|
int
|
2021-11-25 09:26:44 +00:00
|
|
|
main (int argc,
|
|
|
|
char *argv[])
|
2021-11-08 09:52:48 +00:00
|
|
|
{
|
|
|
|
g_autoptr (MetaContext) context = NULL;
|
|
|
|
|
2021-09-06 15:04:45 +00:00
|
|
|
context = meta_create_test_context (META_CONTEXT_TEST_TYPE_HEADLESS,
|
|
|
|
META_CONTEXT_TEST_FLAG_NO_X11);
|
2021-11-08 09:52:48 +00:00
|
|
|
g_assert (meta_context_configure (context, &argc, &argv, NULL));
|
|
|
|
|
2021-09-06 15:04:45 +00:00
|
|
|
test_context = context;
|
|
|
|
|
2021-11-25 09:26:44 +00:00
|
|
|
init_tests ();
|
2021-11-08 09:52:48 +00:00
|
|
|
|
|
|
|
g_signal_connect (context, "before-tests",
|
2021-09-06 15:04:45 +00:00
|
|
|
G_CALLBACK (on_before_tests), NULL);
|
|
|
|
g_signal_connect (context, "after-tests",
|
|
|
|
G_CALLBACK (on_after_tests), NULL);
|
2021-11-08 09:52:48 +00:00
|
|
|
|
2021-05-17 10:12:49 +00:00
|
|
|
return meta_context_test_run_tests (META_CONTEXT_TEST (context),
|
|
|
|
META_TEST_RUN_FLAG_NONE);
|
2021-11-08 09:52:48 +00:00
|
|
|
}
|