mirror of
https://github.com/brl/mutter.git
synced 2024-11-27 10:30:47 -05:00
f8376a43d3
The "later" API is used to queue actions in relation to compositing, thus is owned by the MetaCompositor instance. Make users of this functionality get MetaLaters instance from the compositor, and stop using the global meta_later() API. display: Use non-singleton MetaLater API tests: Use non-singleton MetaLater API meta/common: Make docs refer to context aware MetaLater API Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2718>
852 lines
24 KiB
C
852 lines
24 KiB
C
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
/*
|
|
* Copyright (C) 2014-2017 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-test-utils-private.h"
|
|
|
|
#include <gio/gio.h>
|
|
#include <string.h>
|
|
#include <X11/Xlib-xcb.h>
|
|
|
|
#include "backends/meta-monitor-config-store.h"
|
|
#include "backends/meta-virtual-monitor.h"
|
|
#include "backends/native/meta-backend-native.h"
|
|
#include "backends/native/meta-input-thread.h"
|
|
#include "backends/native/meta-seat-native.h"
|
|
#include "core/display-private.h"
|
|
#include "core/window-private.h"
|
|
#include "meta-test/meta-context-test.h"
|
|
#include "wayland/meta-wayland.h"
|
|
#include "wayland/meta-xwayland.h"
|
|
#include "x11/meta-x11-display-private.h"
|
|
|
|
struct _MetaTestClient
|
|
{
|
|
MetaContext *context;
|
|
|
|
char *id;
|
|
MetaWindowClientType type;
|
|
GSubprocess *subprocess;
|
|
GCancellable *cancellable;
|
|
GMainLoop *loop;
|
|
GDataOutputStream *in;
|
|
GDataInputStream *out;
|
|
|
|
char *line;
|
|
GError **error;
|
|
|
|
MetaAsyncWaiter *waiter;
|
|
MetaX11AlarmFilter *alarm_filter;
|
|
};
|
|
|
|
struct _MetaAsyncWaiter
|
|
{
|
|
MetaX11Display *x11_display;
|
|
|
|
XSyncCounter counter;
|
|
int counter_value;
|
|
XSyncAlarm alarm;
|
|
|
|
GMainLoop *loop;
|
|
int counter_wait_value;
|
|
};
|
|
|
|
typedef struct
|
|
{
|
|
GList *subprocesses;
|
|
} ClientProcessHandler;
|
|
|
|
G_DEFINE_QUARK (meta-test-client-error-quark, meta_test_client_error)
|
|
|
|
static char *test_client_path;
|
|
|
|
void
|
|
meta_ensure_test_client_path (int argc,
|
|
char **argv)
|
|
{
|
|
test_client_path = g_test_build_filename (G_TEST_BUILT,
|
|
"src",
|
|
"tests",
|
|
"mutter-test-client",
|
|
NULL);
|
|
if (!g_file_test (test_client_path,
|
|
G_FILE_TEST_EXISTS | G_FILE_TEST_IS_EXECUTABLE))
|
|
{
|
|
g_autofree char *basename = NULL;
|
|
g_autofree char *dirname = NULL;
|
|
|
|
basename = g_path_get_basename (argv[0]);
|
|
|
|
dirname = g_path_get_dirname (argv[0]);
|
|
test_client_path = g_build_filename (dirname,
|
|
"mutter-test-client", NULL);
|
|
}
|
|
|
|
if (!g_file_test (test_client_path,
|
|
G_FILE_TEST_EXISTS | G_FILE_TEST_IS_EXECUTABLE))
|
|
g_error ("mutter-test-client executable not found");
|
|
}
|
|
|
|
MetaAsyncWaiter *
|
|
meta_async_waiter_new (MetaX11Display *x11_display)
|
|
{
|
|
Display *xdisplay = meta_x11_display_get_xdisplay (x11_display);
|
|
MetaAsyncWaiter *waiter;
|
|
XSyncValue value;
|
|
XSyncAlarmAttributes attr;
|
|
|
|
waiter = g_new0 (MetaAsyncWaiter, 1);
|
|
|
|
waiter->x11_display = x11_display;
|
|
g_object_add_weak_pointer (G_OBJECT (x11_display),
|
|
(gpointer *) &waiter->x11_display);
|
|
waiter->counter_value = 0;
|
|
XSyncIntToValue (&value, waiter->counter_value);
|
|
|
|
waiter->counter = XSyncCreateCounter (xdisplay, value);
|
|
|
|
attr.trigger.counter = waiter->counter;
|
|
attr.trigger.test_type = XSyncPositiveComparison;
|
|
|
|
/* Initialize to one greater than the current value */
|
|
attr.trigger.value_type = XSyncRelative;
|
|
XSyncIntToValue (&attr.trigger.wait_value, 1);
|
|
|
|
/* After triggering, increment test_value by this until
|
|
* until the test condition is false */
|
|
XSyncIntToValue (&attr.delta, 1);
|
|
|
|
/* we want events (on by default anyway) */
|
|
attr.events = True;
|
|
|
|
waiter->alarm = XSyncCreateAlarm (xdisplay,
|
|
XSyncCACounter |
|
|
XSyncCAValueType |
|
|
XSyncCAValue |
|
|
XSyncCATestType |
|
|
XSyncCADelta |
|
|
XSyncCAEvents,
|
|
&attr);
|
|
|
|
waiter->loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
return waiter;
|
|
}
|
|
|
|
void
|
|
meta_async_waiter_destroy (MetaAsyncWaiter *waiter)
|
|
{
|
|
MetaX11Display *x11_display;
|
|
|
|
x11_display = waiter->x11_display;
|
|
if (x11_display)
|
|
{
|
|
Display *xdisplay = meta_x11_display_get_xdisplay (x11_display);
|
|
|
|
XSyncDestroyAlarm (xdisplay, waiter->alarm);
|
|
XSyncDestroyCounter (xdisplay, waiter->counter);
|
|
|
|
g_object_remove_weak_pointer (G_OBJECT (x11_display),
|
|
(gpointer *) &waiter->x11_display);
|
|
}
|
|
g_main_loop_unref (waiter->loop);
|
|
}
|
|
|
|
static int
|
|
meta_async_waiter_next_value (MetaAsyncWaiter *waiter)
|
|
{
|
|
return waiter->counter_value + 1;
|
|
}
|
|
|
|
static void
|
|
meta_async_waiter_wait (MetaAsyncWaiter *waiter,
|
|
int wait_value)
|
|
{
|
|
if (waiter->counter_value < wait_value)
|
|
{
|
|
waiter->counter_wait_value = wait_value;
|
|
g_main_loop_run (waiter->loop);
|
|
waiter->counter_wait_value = 0;
|
|
}
|
|
}
|
|
|
|
void
|
|
meta_async_waiter_set_and_wait (MetaAsyncWaiter *waiter)
|
|
{
|
|
Display *xdisplay;
|
|
int wait_value;
|
|
|
|
g_return_if_fail (waiter->x11_display);
|
|
|
|
wait_value = meta_async_waiter_next_value (waiter);
|
|
|
|
XSyncValue sync_value;
|
|
XSyncIntToValue (&sync_value, wait_value);
|
|
|
|
xdisplay = meta_x11_display_get_xdisplay (waiter->x11_display);
|
|
XSyncSetCounter (xdisplay, waiter->counter, sync_value);
|
|
meta_async_waiter_wait (waiter, wait_value);
|
|
}
|
|
|
|
gboolean
|
|
meta_async_waiter_process_x11_event (MetaAsyncWaiter *waiter,
|
|
MetaX11Display *x11_display,
|
|
XSyncAlarmNotifyEvent *event)
|
|
{
|
|
g_assert (x11_display == waiter->x11_display);
|
|
|
|
if (event->alarm != waiter->alarm)
|
|
return FALSE;
|
|
|
|
waiter->counter_value = XSyncValueLow32 (event->counter_value);
|
|
|
|
if (waiter->counter_wait_value != 0 &&
|
|
waiter->counter_value >= waiter->counter_wait_value)
|
|
g_main_loop_quit (waiter->loop);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
char *
|
|
meta_test_client_get_id (MetaTestClient *client)
|
|
{
|
|
return client->id;
|
|
}
|
|
|
|
static void
|
|
test_client_line_read (GObject *source,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
MetaTestClient *client = user_data;
|
|
|
|
client->line = g_data_input_stream_read_line_finish_utf8 (client->out,
|
|
result,
|
|
NULL,
|
|
client->error);
|
|
g_main_loop_quit (client->loop);
|
|
}
|
|
|
|
gboolean
|
|
meta_test_client_dov (MetaTestClient *client,
|
|
GError **error,
|
|
va_list vap)
|
|
{
|
|
GString *command = g_string_new (NULL);
|
|
char *line = NULL;
|
|
GError *local_error = NULL;
|
|
|
|
while (TRUE)
|
|
{
|
|
char *word = va_arg (vap, char *);
|
|
char *quoted;
|
|
|
|
if (word == NULL)
|
|
break;
|
|
|
|
if (command->len > 0)
|
|
g_string_append_c (command, ' ');
|
|
|
|
quoted = g_shell_quote (word);
|
|
g_string_append (command, quoted);
|
|
g_free (quoted);
|
|
}
|
|
|
|
g_string_append_c (command, '\n');
|
|
|
|
if (!g_data_output_stream_put_string (client->in, command->str,
|
|
client->cancellable, &local_error))
|
|
goto out;
|
|
|
|
g_data_input_stream_read_line_async (client->out,
|
|
G_PRIORITY_DEFAULT,
|
|
client->cancellable,
|
|
test_client_line_read,
|
|
client);
|
|
|
|
client->error = &local_error;
|
|
g_main_loop_run (client->loop);
|
|
line = client->line;
|
|
client->line = NULL;
|
|
client->error = NULL;
|
|
|
|
if (!line)
|
|
{
|
|
if (!local_error)
|
|
{
|
|
g_set_error (&local_error,
|
|
META_TEST_CLIENT_ERROR,
|
|
META_TEST_CLIENT_ERROR_RUNTIME_ERROR,
|
|
"test client exited");
|
|
}
|
|
goto out;
|
|
}
|
|
|
|
if (strcmp (line, "OK") != 0)
|
|
{
|
|
g_set_error (&local_error,
|
|
META_TEST_CLIENT_ERROR,
|
|
META_TEST_CLIENT_ERROR_RUNTIME_ERROR,
|
|
"%s", line);
|
|
goto out;
|
|
}
|
|
|
|
out:
|
|
g_string_free (command, TRUE);
|
|
g_free (line);
|
|
|
|
if (local_error)
|
|
{
|
|
g_propagate_error (error, local_error);
|
|
return FALSE;
|
|
}
|
|
else
|
|
{
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
gboolean
|
|
meta_test_client_do (MetaTestClient *client,
|
|
GError **error,
|
|
...)
|
|
{
|
|
va_list vap;
|
|
gboolean retval;
|
|
|
|
va_start (vap, error);
|
|
retval = meta_test_client_dov (client, error, vap);
|
|
va_end (vap);
|
|
|
|
return retval;
|
|
}
|
|
|
|
gboolean
|
|
meta_test_client_wait (MetaTestClient *client,
|
|
GError **error)
|
|
{
|
|
if (client->type == META_WINDOW_CLIENT_TYPE_WAYLAND)
|
|
{
|
|
return meta_test_client_do (client, error, "sync", NULL);
|
|
}
|
|
else
|
|
{
|
|
int wait_value = meta_async_waiter_next_value (client->waiter);
|
|
char *counter_str = g_strdup_printf ("%lu", client->waiter->counter);
|
|
char *wait_value_str = g_strdup_printf ("%d", wait_value);
|
|
gboolean success;
|
|
|
|
success = meta_test_client_do (client, error,
|
|
"set_counter", counter_str, wait_value_str,
|
|
NULL);
|
|
g_free (counter_str);
|
|
g_free (wait_value_str);
|
|
if (!success)
|
|
return FALSE;
|
|
|
|
meta_async_waiter_wait (client->waiter, wait_value);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
MetaWindow *
|
|
meta_find_window_from_title (MetaContext *context,
|
|
const char *title)
|
|
{
|
|
g_autoptr (GList) windows = NULL;
|
|
GList *l;
|
|
|
|
windows = meta_display_list_all_windows (meta_context_get_display (context));
|
|
for (l = windows; l; l = l->next)
|
|
{
|
|
MetaWindow *window = l->data;
|
|
|
|
if (g_strcmp0 (window->title, title) == 0)
|
|
return window;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
MetaWindow *
|
|
meta_test_client_find_window (MetaTestClient *client,
|
|
const char *window_id,
|
|
GError **error)
|
|
{
|
|
MetaDisplay *display = meta_context_get_display (client->context);
|
|
g_autofree char *expected_title = NULL;
|
|
MetaWindow *window;
|
|
|
|
expected_title = g_strdup_printf ("test/%s/%s", client->id, window_id);
|
|
window = meta_find_window_from_title (meta_display_get_context (display),
|
|
expected_title);
|
|
|
|
if (!window)
|
|
{
|
|
g_set_error (error,
|
|
META_TEST_CLIENT_ERROR,
|
|
META_TEST_CLIENT_ERROR_RUNTIME_ERROR,
|
|
"window %s/%s isn't known to Mutter", client->id, window_id);
|
|
return NULL;
|
|
}
|
|
|
|
return window;
|
|
}
|
|
|
|
typedef struct _WaitForShownData
|
|
{
|
|
GMainLoop *loop;
|
|
MetaWindow *window;
|
|
gulong shown_handler_id;
|
|
} WaitForShownData;
|
|
|
|
static void
|
|
on_window_shown (MetaWindow *window,
|
|
WaitForShownData *data)
|
|
{
|
|
g_main_loop_quit (data->loop);
|
|
}
|
|
|
|
static gboolean
|
|
wait_for_showing_before_redraw (gpointer user_data)
|
|
{
|
|
WaitForShownData *data = user_data;
|
|
|
|
if (meta_window_is_hidden (data->window))
|
|
{
|
|
data->shown_handler_id = g_signal_connect (data->window, "shown",
|
|
G_CALLBACK (on_window_shown),
|
|
data);
|
|
}
|
|
else
|
|
{
|
|
g_main_loop_quit (data->loop);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
void
|
|
meta_test_client_wait_for_window_shown (MetaTestClient *client,
|
|
MetaWindow *window)
|
|
{
|
|
MetaDisplay *display = meta_window_get_display (window);
|
|
MetaCompositor *compositor = meta_display_get_compositor (display);
|
|
MetaLaters *laters = meta_compositor_get_laters (compositor);
|
|
|
|
WaitForShownData data = {
|
|
.loop = g_main_loop_new (NULL, FALSE),
|
|
.window = window,
|
|
};
|
|
meta_laters_add (laters, META_LATER_BEFORE_REDRAW,
|
|
wait_for_showing_before_redraw,
|
|
&data,
|
|
NULL);
|
|
g_main_loop_run (data.loop);
|
|
g_clear_signal_handler (&data.shown_handler_id, window);
|
|
g_main_loop_unref (data.loop);
|
|
}
|
|
|
|
static gboolean
|
|
meta_test_client_process_x11_event (MetaTestClient *client,
|
|
MetaX11Display *x11_display,
|
|
XSyncAlarmNotifyEvent *event)
|
|
{
|
|
if (client->waiter)
|
|
{
|
|
return meta_async_waiter_process_x11_event (client->waiter,
|
|
x11_display,
|
|
event);
|
|
}
|
|
else
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
static gpointer
|
|
spawn_xwayland (gpointer user_data)
|
|
{
|
|
xcb_connection_t *connection;
|
|
|
|
connection = xcb_connect (NULL, NULL);
|
|
g_assert_nonnull (connection);
|
|
xcb_disconnect (connection);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
on_prepare_shutdown (MetaBackend *backend,
|
|
ClientProcessHandler *process_handler)
|
|
{
|
|
g_debug ("Waiting for test clients to exit before shutting down");
|
|
while (process_handler->subprocesses)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
}
|
|
|
|
static void
|
|
wait_check_cb (GObject *source_object,
|
|
GAsyncResult *result,
|
|
gpointer user_data)
|
|
{
|
|
GSubprocess *subprocess = G_SUBPROCESS (source_object);
|
|
ClientProcessHandler *process_handler = user_data;
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
if (!g_subprocess_wait_check_finish (subprocess, result, &error))
|
|
{
|
|
if (g_error_matches (error, G_SPAWN_EXIT_ERROR, 1))
|
|
{
|
|
g_debug ("Test client process %s exited with exit status 1",
|
|
g_subprocess_get_identifier (subprocess));
|
|
}
|
|
else
|
|
{
|
|
g_warning ("Test client process %s crashed with status %d",
|
|
g_subprocess_get_identifier (subprocess),
|
|
error->code);
|
|
}
|
|
}
|
|
|
|
process_handler->subprocesses = g_list_remove (process_handler->subprocesses,
|
|
subprocess);
|
|
}
|
|
|
|
static ClientProcessHandler *
|
|
ensure_process_handler (MetaContext *context)
|
|
{
|
|
ClientProcessHandler *process_handler;
|
|
const char data_key[] = "test-client-subprocess-handler";
|
|
MetaBackend *backend;
|
|
|
|
process_handler = g_object_get_data (G_OBJECT (context), data_key);
|
|
if (process_handler)
|
|
return process_handler;
|
|
|
|
process_handler = g_new0 (ClientProcessHandler, 1);
|
|
g_object_set_data_full (G_OBJECT (context), data_key,
|
|
process_handler, g_free);
|
|
|
|
backend = meta_context_get_backend (context);
|
|
g_signal_connect (backend, "prepare-shutdown",
|
|
G_CALLBACK (on_prepare_shutdown),
|
|
process_handler);
|
|
|
|
return process_handler;
|
|
}
|
|
|
|
static gboolean
|
|
alarm_filter (MetaX11Display *x11_display,
|
|
XSyncAlarmNotifyEvent *event,
|
|
gpointer user_data)
|
|
{
|
|
MetaTestClient *client = user_data;
|
|
|
|
return meta_test_client_process_x11_event (client, x11_display, event);
|
|
}
|
|
|
|
MetaTestClient *
|
|
meta_test_client_new (MetaContext *context,
|
|
const char *id,
|
|
MetaWindowClientType type,
|
|
GError **error)
|
|
{
|
|
MetaTestClient *client;
|
|
GSubprocessLauncher *launcher;
|
|
GSubprocess *subprocess;
|
|
ClientProcessHandler *process_handler;
|
|
MetaWaylandCompositor *compositor;
|
|
const char *wayland_display_name;
|
|
const char *x11_display_name;
|
|
|
|
launcher = g_subprocess_launcher_new ((G_SUBPROCESS_FLAGS_STDIN_PIPE |
|
|
G_SUBPROCESS_FLAGS_STDOUT_PIPE));
|
|
|
|
g_assert (meta_is_wayland_compositor ());
|
|
compositor = meta_context_get_wayland_compositor (context);
|
|
wayland_display_name = meta_wayland_get_wayland_display_name (compositor);
|
|
x11_display_name = meta_wayland_get_public_xwayland_display_name (compositor);
|
|
|
|
if (wayland_display_name)
|
|
{
|
|
g_subprocess_launcher_setenv (launcher,
|
|
"WAYLAND_DISPLAY", wayland_display_name,
|
|
TRUE);
|
|
}
|
|
|
|
if (x11_display_name)
|
|
{
|
|
g_subprocess_launcher_setenv (launcher,
|
|
"DISPLAY", x11_display_name,
|
|
TRUE);
|
|
}
|
|
|
|
subprocess = g_subprocess_launcher_spawn (launcher,
|
|
error,
|
|
test_client_path,
|
|
"--client-id",
|
|
id,
|
|
(type == META_WINDOW_CLIENT_TYPE_WAYLAND ?
|
|
"--wayland" : NULL),
|
|
NULL);
|
|
g_object_unref (launcher);
|
|
|
|
if (!subprocess)
|
|
return NULL;
|
|
|
|
process_handler = ensure_process_handler (context);
|
|
process_handler->subprocesses = g_list_prepend (process_handler->subprocesses,
|
|
subprocess);
|
|
g_subprocess_wait_check_async (subprocess, NULL,
|
|
wait_check_cb,
|
|
process_handler);
|
|
|
|
client = g_new0 (MetaTestClient, 1);
|
|
client->context = context;
|
|
client->type = type;
|
|
client->id = g_strdup (id);
|
|
client->cancellable = g_cancellable_new ();
|
|
client->subprocess = subprocess;
|
|
client->in =
|
|
g_data_output_stream_new (g_subprocess_get_stdin_pipe (subprocess));
|
|
client->out =
|
|
g_data_input_stream_new (g_subprocess_get_stdout_pipe (subprocess));
|
|
client->loop = g_main_loop_new (NULL, FALSE);
|
|
|
|
if (client->type == META_WINDOW_CLIENT_TYPE_X11)
|
|
{
|
|
MetaDisplay *display = meta_context_get_display (context);
|
|
MetaX11Display *x11_display;
|
|
|
|
x11_display = meta_display_get_x11_display (display);
|
|
if (!x11_display)
|
|
{
|
|
GThread *thread;
|
|
|
|
thread = g_thread_new ("Mutter Spawn Xwayland Thread",
|
|
spawn_xwayland,
|
|
NULL);
|
|
meta_context_test_wait_for_x11_display (META_CONTEXT_TEST (context));
|
|
g_thread_join (thread);
|
|
}
|
|
x11_display = meta_display_get_x11_display (display);
|
|
g_assert_nonnull (x11_display);
|
|
|
|
client->alarm_filter = meta_x11_display_add_alarm_filter (x11_display,
|
|
alarm_filter,
|
|
client);
|
|
|
|
client->waiter = meta_async_waiter_new (x11_display);
|
|
}
|
|
|
|
return client;
|
|
}
|
|
|
|
gboolean
|
|
meta_test_client_quit (MetaTestClient *client,
|
|
GError **error)
|
|
{
|
|
if (!meta_test_client_do (client, error, "destroy_all", NULL))
|
|
return FALSE;
|
|
|
|
if (!meta_test_client_wait (client, error))
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
meta_test_client_destroy (MetaTestClient *client)
|
|
{
|
|
MetaDisplay *display = meta_context_get_display (client->context);
|
|
MetaX11Display *x11_display;
|
|
GError *error = NULL;
|
|
|
|
x11_display = meta_display_get_x11_display (display);
|
|
if (client->alarm_filter && x11_display)
|
|
{
|
|
meta_x11_display_remove_alarm_filter (x11_display,
|
|
client->alarm_filter);
|
|
}
|
|
|
|
if (client->waiter)
|
|
meta_async_waiter_destroy (client->waiter);
|
|
|
|
g_output_stream_close (G_OUTPUT_STREAM (client->in), NULL, &error);
|
|
if (error)
|
|
{
|
|
g_warning ("Error closing client stdin: %s", error->message);
|
|
g_clear_error (&error);
|
|
}
|
|
g_object_unref (client->in);
|
|
|
|
g_input_stream_close (G_INPUT_STREAM (client->out), NULL, &error);
|
|
if (error)
|
|
{
|
|
g_warning ("Error closing client stdout: %s", error->message);
|
|
g_clear_error (&error);
|
|
}
|
|
g_object_unref (client->out);
|
|
|
|
g_object_unref (client->cancellable);
|
|
g_object_unref (client->subprocess);
|
|
g_main_loop_unref (client->loop);
|
|
g_free (client->id);
|
|
g_free (client);
|
|
}
|
|
|
|
const char *
|
|
meta_test_get_plugin_name (void)
|
|
{
|
|
const char *name;
|
|
|
|
name = g_getenv ("MUTTER_TEST_PLUGIN_PATH");
|
|
if (name)
|
|
return name;
|
|
else
|
|
return "libdefault";
|
|
}
|
|
|
|
void
|
|
meta_set_custom_monitor_config_full (MetaBackend *backend,
|
|
const char *filename,
|
|
MetaMonitorsConfigFlag configs_flags)
|
|
{
|
|
MetaMonitorManager *monitor_manager =
|
|
meta_backend_get_monitor_manager (backend);
|
|
MetaMonitorConfigManager *config_manager = monitor_manager->config_manager;
|
|
MetaMonitorConfigStore *config_store;
|
|
GError *error = NULL;
|
|
const char *path;
|
|
|
|
g_assert_nonnull (config_manager);
|
|
|
|
config_store = meta_monitor_config_manager_get_store (config_manager);
|
|
|
|
path = g_test_get_filename (G_TEST_DIST, "tests", "monitor-configs",
|
|
filename, NULL);
|
|
if (!meta_monitor_config_store_set_custom (config_store, path, NULL,
|
|
configs_flags,
|
|
&error))
|
|
g_warning ("Failed to set custom config: %s", error->message);
|
|
}
|
|
|
|
static void
|
|
on_view_presented (ClutterStage *stage,
|
|
ClutterStageView *view,
|
|
ClutterFrameInfo *frame_info,
|
|
GList **presented_views)
|
|
{
|
|
*presented_views = g_list_remove (*presented_views, view);
|
|
}
|
|
|
|
void
|
|
meta_wait_for_paint (MetaContext *context)
|
|
{
|
|
MetaBackend *backend = meta_context_get_backend (context);
|
|
ClutterActor *stage = meta_backend_get_stage (backend);
|
|
MetaRenderer *renderer = meta_backend_get_renderer (backend);
|
|
GList *views;
|
|
gulong handler_id;
|
|
|
|
clutter_actor_queue_redraw (stage);
|
|
|
|
views = g_list_copy (meta_renderer_get_views (renderer));
|
|
handler_id = g_signal_connect (stage, "presented",
|
|
G_CALLBACK (on_view_presented), &views);
|
|
while (views)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
g_signal_handler_disconnect (stage, handler_id);
|
|
}
|
|
|
|
MetaVirtualMonitor *
|
|
meta_create_test_monitor (MetaContext *context,
|
|
int width,
|
|
int height,
|
|
float refresh_rate)
|
|
{
|
|
MetaBackend *backend = meta_context_get_backend (context);
|
|
MetaMonitorManager *monitor_manager = meta_backend_get_monitor_manager (backend);
|
|
g_autoptr (MetaVirtualMonitorInfo) monitor_info = NULL;
|
|
g_autoptr (GError) error = NULL;
|
|
static int serial_count = 0x10000;
|
|
g_autofree char *serial = NULL;
|
|
MetaVirtualMonitor *virtual_monitor;
|
|
|
|
serial = g_strdup_printf ("0x%x", serial_count++);
|
|
monitor_info = meta_virtual_monitor_info_new (width, height, refresh_rate,
|
|
"MetaTestVendor",
|
|
"MetaVirtualMonitor",
|
|
serial);
|
|
virtual_monitor = meta_monitor_manager_create_virtual_monitor (monitor_manager,
|
|
monitor_info,
|
|
&error);
|
|
if (!virtual_monitor)
|
|
g_error ("Failed to create virtual monitor: %s", error->message);
|
|
meta_monitor_manager_reload (monitor_manager);
|
|
|
|
return virtual_monitor;
|
|
}
|
|
|
|
#ifdef HAVE_NATIVE_BACKEND
|
|
static gboolean
|
|
callback_idle (gpointer user_data)
|
|
{
|
|
GMainLoop *loop = user_data;
|
|
|
|
g_main_loop_quit (loop);
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
static gboolean
|
|
queue_callback (GTask *task)
|
|
{
|
|
g_idle_add (callback_idle, g_task_get_task_data (task));
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
#endif
|
|
|
|
void
|
|
meta_flush_input (MetaContext *context)
|
|
{
|
|
#ifdef HAVE_NATIVE_BACKEND
|
|
MetaBackend *backend = meta_context_get_backend (context);
|
|
ClutterSeat *seat;
|
|
MetaSeatNative *seat_native;
|
|
g_autoptr (GTask) task = NULL;
|
|
g_autoptr (GMainLoop) loop = NULL;
|
|
|
|
g_assert_true (META_IS_BACKEND_NATIVE (backend));
|
|
|
|
seat = meta_backend_get_default_seat (backend);
|
|
seat_native = META_SEAT_NATIVE (seat);
|
|
|
|
task = g_task_new (backend, NULL, NULL, NULL);
|
|
loop = g_main_loop_new (NULL, FALSE);
|
|
g_task_set_task_data (task, loop, NULL);
|
|
|
|
meta_seat_impl_run_input_task (seat_native->impl, task,
|
|
(GSourceFunc) queue_callback);
|
|
|
|
g_main_loop_run (loop);
|
|
#endif
|
|
}
|