mutter/src/tests/meta-wayland-test-driver.c
Robert Mader 11fc5b6c6c tests/wayland: Add more requests to test driver protocol
Add `sync_effects_completed()` and `verify_view()` in
order to allow Wayland test clients to trigger verifications
and add convenience functions to use them to client-utils.

Notes:
- `sync_effects_completed()` works in two stages in order
  to ensure it doesn't race with window effects. By the time
  `sync_effects_completed()` is processed, an effect could
  already have ended or not yet been scheduled. Thus we
  defer a check for pending effects to the next paint cycle,
  assuming that by then they should have been scheduled.
- `meta_ref_test_verify_view()` internally triggers the
  `paint` signal for the stage which is why it can not be run
  in the after-paint signal handler.

Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1055>
2022-06-03 09:03:10 +00:00

321 lines
9.4 KiB
C

/*
* 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 "tests/meta-wayland-test-driver.h"
#include <wayland-server.h>
#include "compositor/meta-window-actor-private.h"
#include "tests/meta-ref-test.h"
#include "wayland/meta-wayland-actor-surface.h"
#include "wayland/meta-wayland-private.h"
#include "wayland/meta-wayland-surface.h"
#include "test-driver-server-protocol.h"
enum
{
SYNC_POINT,
N_SIGNALS
};
static int signals[N_SIGNALS];
struct _MetaWaylandTestDriver
{
GObject parent;
struct wl_global *test_driver;
GList *resources;
GHashTable *properties;
};
G_DEFINE_TYPE (MetaWaylandTestDriver, meta_wayland_test_driver,
G_TYPE_OBJECT)
typedef struct _PendingEffectsData
{
MetaWaylandSurface *surface;
struct wl_resource *callback;
} PendingEffectsData;
static void
on_actor_destroyed (ClutterActor *actor,
struct wl_resource *callback)
{
wl_callback_send_done (callback, 0);
wl_resource_destroy (callback);
}
static void
sync_actor_destroy (struct wl_client *client,
struct wl_resource *resource,
uint32_t id,
struct wl_resource *surface_resource)
{
MetaWaylandSurface *surface = wl_resource_get_user_data (surface_resource);
MetaWaylandActorSurface *actor_surface;
MetaSurfaceActor *actor;
struct wl_resource *callback;
g_assert_nonnull (surface);
actor_surface = (MetaWaylandActorSurface *) surface->role;
g_assert_nonnull (actor_surface);
actor = meta_wayland_actor_surface_get_actor (actor_surface);
g_assert_nonnull (actor);
callback = wl_resource_create (client, &wl_callback_interface, 1, id);
g_signal_connect (actor, "destroy", G_CALLBACK (on_actor_destroyed),
callback);
}
static void
on_effects_completed (ClutterActor *actor,
struct wl_resource *callback)
{
g_signal_handlers_disconnect_by_data (actor, callback);
wl_callback_send_done (callback, 0);
wl_resource_destroy (callback);
}
static void
check_for_pending_effects (ClutterStage *stage,
ClutterStageView *view,
PendingEffectsData *data)
{
MetaWindow *window;
MetaWindowActor *window_actor;
g_signal_handlers_disconnect_by_data (stage, data);
window = meta_wayland_surface_get_window (data->surface);
g_assert_nonnull (window);
window_actor = meta_window_actor_from_window (window);
g_assert_nonnull (window_actor);
if (meta_window_actor_effect_in_progress (window_actor))
{
g_signal_connect (window_actor, "effects-completed",
G_CALLBACK (on_effects_completed), data->callback);
}
else
{
on_effects_completed (CLUTTER_ACTOR (window_actor), data->callback);
}
g_free (data);
}
static void
sync_effects_completed (struct wl_client *client,
struct wl_resource *resource,
uint32_t id,
struct wl_resource *surface_resource)
{
MetaBackend *backend = meta_get_backend ();
ClutterActor *stage = meta_backend_get_stage (backend);
MetaWaylandSurface *surface = wl_resource_get_user_data (surface_resource);
PendingEffectsData *data;
GList *stage_views;
g_assert_nonnull (surface);
data = g_new0 (PendingEffectsData, 1);
data->surface = surface;
data->callback = wl_resource_create (client, &wl_callback_interface, 1, id);
stage_views = clutter_stage_peek_stage_views (CLUTTER_STAGE (stage));
g_assert (g_list_length (stage_views) > 0);
g_signal_connect (CLUTTER_STAGE (stage), "after-update",
G_CALLBACK (check_for_pending_effects), data);
clutter_stage_schedule_update (CLUTTER_STAGE (stage));
}
static void
sync_point (struct wl_client *client,
struct wl_resource *resource,
uint32_t sequence,
struct wl_resource *surface_resource)
{
MetaWaylandTestDriver *test_driver = wl_resource_get_user_data (resource);
g_signal_emit (test_driver, signals[SYNC_POINT], 0,
sequence,
surface_resource,
client);
}
static void
on_after_paint (ClutterStage *stage,
ClutterStageView *view,
struct wl_resource *callback)
{
g_signal_handlers_disconnect_by_data (stage, callback);
wl_callback_send_done (callback, 0);
wl_resource_destroy (callback);
}
static void
verify_view (struct wl_client *client,
struct wl_resource *resource,
uint32_t id,
uint32_t sequence)
{
MetaBackend *backend = meta_get_backend ();
ClutterActor *stage = meta_backend_get_stage (backend);
GList *stage_views;
struct wl_resource *callback;
stage_views = clutter_stage_peek_stage_views (CLUTTER_STAGE (stage));
g_assert (g_list_length (stage_views) > 0);
callback = wl_resource_create (client, &wl_callback_interface, 1, id);
g_signal_connect_after (CLUTTER_STAGE (stage), "after-paint",
G_CALLBACK (on_after_paint), callback);
meta_ref_test_verify_view (CLUTTER_STAGE_VIEW (stage_views->data),
g_test_get_path (),
sequence,
meta_ref_test_determine_ref_test_flag ());
}
static const struct test_driver_interface meta_test_driver_interface = {
sync_actor_destroy,
sync_effects_completed,
sync_point,
verify_view,
};
static void
test_driver_destructor (struct wl_resource *resource)
{
MetaWaylandTestDriver *test_driver = wl_resource_get_user_data (resource);
test_driver->resources = g_list_remove (test_driver->resources, resource);
}
static void
bind_test_driver (struct wl_client *client,
void *user_data,
uint32_t version,
uint32_t id)
{
MetaWaylandTestDriver *test_driver = user_data;
struct wl_resource *resource;
GHashTableIter iter;
gpointer key, value;
resource = wl_resource_create (client, &test_driver_interface,
version, id);
wl_resource_set_implementation (resource, &meta_test_driver_interface,
test_driver, test_driver_destructor);
test_driver->resources = g_list_prepend (test_driver->resources, resource);
g_hash_table_iter_init (&iter, test_driver->properties);
while (g_hash_table_iter_next (&iter, &key, &value))
test_driver_send_property (resource, key, value);
}
static void
meta_wayland_test_driver_finalize (GObject *object)
{
MetaWaylandTestDriver *test_driver = META_WAYLAND_TEST_DRIVER (object);
g_clear_pointer (&test_driver->test_driver, wl_global_destroy);
g_clear_pointer (&test_driver->properties, g_hash_table_unref);
G_OBJECT_CLASS (meta_wayland_test_driver_parent_class)->finalize (object);
}
static void
meta_wayland_test_driver_class_init (MetaWaylandTestDriverClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
object_class->finalize = meta_wayland_test_driver_finalize;
signals[SYNC_POINT] =
g_signal_new ("sync-point",
G_TYPE_FROM_CLASS (klass),
G_SIGNAL_RUN_LAST,
0,
NULL, NULL, NULL,
G_TYPE_NONE, 3,
G_TYPE_UINT,
G_TYPE_POINTER,
G_TYPE_POINTER);
}
static void
meta_wayland_test_driver_init (MetaWaylandTestDriver *test_driver)
{
test_driver->properties = g_hash_table_new_full (g_str_hash, g_str_equal,
g_free, g_free);
}
MetaWaylandTestDriver *
meta_wayland_test_driver_new (MetaWaylandCompositor *compositor)
{
MetaWaylandTestDriver *test_driver;
test_driver = g_object_new (META_TYPE_WAYLAND_TEST_DRIVER, NULL);
test_driver->test_driver = wl_global_create (compositor->wayland_display,
&test_driver_interface,
1,
test_driver, bind_test_driver);
if (!test_driver->test_driver)
g_error ("Failed to register a global wl-subcompositor object");
return test_driver;
}
void
meta_wayland_test_driver_emit_sync_event (MetaWaylandTestDriver *test_driver,
uint32_t serial)
{
GList *l;
for (l = test_driver->resources; l; l = l->next)
{
struct wl_resource *resource = l->data;
test_driver_send_sync_event (resource, serial);
}
}
void
meta_wayland_test_driver_set_property (MetaWaylandTestDriver *test_driver,
const char *name,
const char *value)
{
g_hash_table_replace (test_driver->properties,
g_strdup (name),
g_strdup (value));
}