2014-09-11 17:43:32 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2014 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/>.
|
|
|
|
*/
|
|
|
|
|
2018-07-10 08:36:24 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
#include <gio/gio.h>
|
|
|
|
#include <stdarg.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2018-07-10 08:36:24 +00:00
|
|
|
#include "compositor/meta-plugin-manager.h"
|
|
|
|
#include "core/window-private.h"
|
|
|
|
#include "meta/main.h"
|
|
|
|
#include "meta/util.h"
|
|
|
|
#include "meta/window.h"
|
2017-06-26 08:31:48 +00:00
|
|
|
#include "tests/test-utils.h"
|
2018-07-10 08:36:24 +00:00
|
|
|
#include "ui/ui.h"
|
|
|
|
#include "wayland/meta-wayland.h"
|
2017-08-26 16:26:30 +00:00
|
|
|
#include "x11/meta-x11-display-private.h"
|
2014-09-11 17:43:32 +00:00
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
GHashTable *clients;
|
|
|
|
AsyncWaiter *waiter;
|
2014-09-15 18:50:21 +00:00
|
|
|
guint log_handler_id;
|
|
|
|
GString *warning_messages;
|
2014-09-24 19:52:11 +00:00
|
|
|
GMainLoop *loop;
|
2014-09-11 17:43:32 +00:00
|
|
|
} TestCase;
|
|
|
|
|
|
|
|
static gboolean
|
2017-08-26 16:56:44 +00:00
|
|
|
test_case_alarm_filter (MetaX11Display *x11_display,
|
2014-09-11 17:43:32 +00:00
|
|
|
XSyncAlarmNotifyEvent *event,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
TestCase *test = data;
|
|
|
|
GHashTableIter iter;
|
|
|
|
gpointer key, value;
|
|
|
|
|
2017-08-26 16:56:44 +00:00
|
|
|
if (async_waiter_alarm_filter (test->waiter, x11_display, event))
|
2014-09-11 17:43:32 +00:00
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
g_hash_table_iter_init (&iter, test->clients);
|
|
|
|
while (g_hash_table_iter_next (&iter, &key, &value))
|
2017-08-26 16:56:44 +00:00
|
|
|
if (test_client_alarm_filter (value, x11_display, event))
|
2014-09-11 17:43:32 +00:00
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2014-09-15 18:50:21 +00:00
|
|
|
static gboolean
|
|
|
|
test_case_check_warnings (TestCase *test,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
if (test->warning_messages != NULL)
|
|
|
|
{
|
|
|
|
g_set_error (error, TEST_RUNNER_ERROR, TEST_RUNNER_ERROR_RUNTIME_ERROR,
|
|
|
|
"Warning messages:\n %s", test->warning_messages->str);
|
|
|
|
g_string_free (test->warning_messages, TRUE);
|
|
|
|
test->warning_messages = NULL;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
test_case_log_func (const gchar *log_domain,
|
|
|
|
GLogLevelFlags log_level,
|
|
|
|
const gchar *message,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
TestCase *test = user_data;
|
|
|
|
|
|
|
|
if (test->warning_messages == NULL)
|
|
|
|
test->warning_messages = g_string_new (message);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_string_append (test->warning_messages, "\n ");
|
|
|
|
g_string_append (test->warning_messages, message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
static TestCase *
|
|
|
|
test_case_new (void)
|
|
|
|
{
|
|
|
|
TestCase *test = g_new0 (TestCase, 1);
|
|
|
|
|
2014-09-15 18:50:21 +00:00
|
|
|
test->log_handler_id = g_log_set_handler ("mutter",
|
|
|
|
G_LOG_LEVEL_CRITICAL | G_LOG_LEVEL_WARNING,
|
|
|
|
test_case_log_func,
|
|
|
|
test);
|
|
|
|
|
2017-08-26 16:56:44 +00:00
|
|
|
meta_x11_display_set_alarm_filter (meta_get_display ()->x11_display,
|
|
|
|
test_case_alarm_filter, test);
|
2014-09-11 17:43:32 +00:00
|
|
|
|
|
|
|
test->clients = g_hash_table_new (g_str_hash, g_str_equal);
|
|
|
|
test->waiter = async_waiter_new ();
|
2014-09-24 19:52:11 +00:00
|
|
|
test->loop = g_main_loop_new (NULL, FALSE);
|
2014-09-11 17:43:32 +00:00
|
|
|
|
|
|
|
return test;
|
|
|
|
}
|
|
|
|
|
2014-09-24 19:52:11 +00:00
|
|
|
static gboolean
|
|
|
|
test_case_before_redraw (gpointer data)
|
|
|
|
{
|
|
|
|
TestCase *test = data;
|
|
|
|
|
|
|
|
g_main_loop_quit (test->loop);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
static gboolean
|
|
|
|
test_case_wait (TestCase *test,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
GHashTableIter iter;
|
|
|
|
gpointer key, value;
|
|
|
|
|
2014-09-24 19:52:11 +00:00
|
|
|
/* First have each client set a XSync counter, and wait until
|
|
|
|
* we receive the resulting event - so we know we've received
|
|
|
|
* everything that the client have sent us.
|
|
|
|
*/
|
2014-09-11 17:43:32 +00:00
|
|
|
g_hash_table_iter_init (&iter, test->clients);
|
|
|
|
while (g_hash_table_iter_next (&iter, &key, &value))
|
|
|
|
if (!test_client_wait (value, error))
|
|
|
|
return FALSE;
|
|
|
|
|
2014-09-24 19:52:11 +00:00
|
|
|
/* Then wait until we've done any outstanding queued up work.
|
|
|
|
* Though we add this as BEFORE_REDRAW, the iteration that runs the
|
|
|
|
* BEFORE_REDRAW idles will proceed on and do the redraw, so we're
|
|
|
|
* waiting until after *all* frame processing.
|
|
|
|
*/
|
|
|
|
meta_later_add (META_LATER_BEFORE_REDRAW,
|
|
|
|
test_case_before_redraw,
|
|
|
|
test,
|
|
|
|
NULL);
|
|
|
|
g_main_loop_run (test->loop);
|
|
|
|
|
|
|
|
/* Then set an XSync counter ourselves and and wait until
|
|
|
|
* we receive the resulting event - this makes sure that we've
|
|
|
|
* received back any X events we generated.
|
|
|
|
*/
|
2014-09-11 17:43:32 +00:00
|
|
|
async_waiter_set_and_wait (test->waiter);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define BAD_COMMAND(...) \
|
|
|
|
G_STMT_START { \
|
|
|
|
g_set_error (error, \
|
|
|
|
TEST_RUNNER_ERROR, TEST_RUNNER_ERROR_BAD_COMMAND, \
|
|
|
|
__VA_ARGS__); \
|
|
|
|
return FALSE; \
|
|
|
|
} G_STMT_END
|
|
|
|
|
|
|
|
static TestClient *
|
|
|
|
test_case_lookup_client (TestCase *test,
|
|
|
|
char *client_id,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
TestClient *client = g_hash_table_lookup (test->clients, client_id);
|
|
|
|
if (!client)
|
|
|
|
g_set_error (error, TEST_RUNNER_ERROR, TEST_RUNNER_ERROR_BAD_COMMAND,
|
|
|
|
"No such client %s", client_id);
|
|
|
|
|
|
|
|
return client;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
test_case_parse_window_id (TestCase *test,
|
|
|
|
const char *client_and_window_id,
|
|
|
|
TestClient **client,
|
|
|
|
const char **window_id,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
const char *slash = strchr (client_and_window_id, '/');
|
|
|
|
char *tmp;
|
|
|
|
if (slash == NULL)
|
|
|
|
BAD_COMMAND ("client/window ID %s doesnt' contain a /", client_and_window_id);
|
|
|
|
|
|
|
|
*window_id = slash + 1;
|
|
|
|
|
|
|
|
tmp = g_strndup (client_and_window_id, slash - client_and_window_id);
|
|
|
|
*client = test_case_lookup_client (test, tmp, error);
|
|
|
|
g_free (tmp);
|
|
|
|
|
|
|
|
return client != NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
test_case_assert_stacking (TestCase *test,
|
|
|
|
char **expected_windows,
|
|
|
|
int n_expected_windows,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaDisplay *display = meta_get_display ();
|
2014-09-09 01:20:14 +00:00
|
|
|
guint64 *windows;
|
2014-09-11 17:43:32 +00:00
|
|
|
int n_windows;
|
|
|
|
GString *stack_string = g_string_new (NULL);
|
|
|
|
GString *expected_string = g_string_new (NULL);
|
|
|
|
int i;
|
|
|
|
|
2017-08-26 17:03:51 +00:00
|
|
|
meta_stack_tracker_get_stack (display->stack_tracker, &windows, &n_windows);
|
2014-09-11 17:43:32 +00:00
|
|
|
for (i = 0; i < n_windows; i++)
|
|
|
|
{
|
2014-09-09 01:20:14 +00:00
|
|
|
MetaWindow *window = meta_display_lookup_stack_id (display, windows[i]);
|
2014-09-11 17:43:32 +00:00
|
|
|
if (window != NULL && window->title)
|
|
|
|
{
|
|
|
|
/* See comment in meta_ui_new() about why the dummy window for GTK+ theming
|
|
|
|
* is managed as a MetaWindow.
|
|
|
|
*/
|
2014-09-09 01:20:14 +00:00
|
|
|
if (META_STACK_ID_IS_X11 (windows[i]) &&
|
2017-08-26 19:38:59 +00:00
|
|
|
meta_ui_window_is_dummy (display->x11_display->ui, windows[i]))
|
2014-09-11 17:43:32 +00:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if (stack_string->len > 0)
|
|
|
|
g_string_append_c (stack_string, ' ');
|
|
|
|
|
|
|
|
if (g_str_has_prefix (window->title, "test/"))
|
|
|
|
g_string_append (stack_string, window->title + 5);
|
|
|
|
else
|
|
|
|
g_string_append_printf (stack_string, "(%s)", window->title);
|
|
|
|
}
|
2017-08-26 17:03:51 +00:00
|
|
|
else if (windows[i] == display->x11_display->guard_window)
|
2014-09-24 20:07:11 +00:00
|
|
|
{
|
|
|
|
if (stack_string->len > 0)
|
|
|
|
g_string_append_c (stack_string, ' ');
|
|
|
|
|
|
|
|
g_string_append_c (stack_string, '|');
|
|
|
|
}
|
2014-09-11 17:43:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < n_expected_windows; i++)
|
|
|
|
{
|
|
|
|
if (expected_string->len > 0)
|
|
|
|
g_string_append_c (expected_string, ' ');
|
|
|
|
|
|
|
|
g_string_append (expected_string, expected_windows[i]);
|
|
|
|
}
|
|
|
|
|
2014-09-24 20:07:11 +00:00
|
|
|
/* Don't require '| ' as a prefix if there are no hidden windows - we
|
|
|
|
* remove the prefix from the actual string instead of adding it to the
|
|
|
|
* expected string for clarity of the error message
|
|
|
|
*/
|
|
|
|
if (index (expected_string->str, '|') == NULL && stack_string->str[0] == '|')
|
|
|
|
{
|
|
|
|
g_string_erase (stack_string,
|
|
|
|
0, stack_string->str[1] == ' ' ? 2 : 1);
|
|
|
|
}
|
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
if (strcmp (expected_string->str, stack_string->str) != 0)
|
|
|
|
{
|
|
|
|
g_set_error (error, TEST_RUNNER_ERROR, TEST_RUNNER_ERROR_ASSERTION_FAILED,
|
|
|
|
"stacking: expected='%s', actual='%s'",
|
|
|
|
expected_string->str, stack_string->str);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_string_free (stack_string, TRUE);
|
|
|
|
g_string_free (expected_string, TRUE);
|
|
|
|
|
|
|
|
return *error == NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
test_case_check_xserver_stacking (TestCase *test,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaDisplay *display = meta_get_display ();
|
|
|
|
GString *local_string = g_string_new (NULL);
|
|
|
|
GString *x11_string = g_string_new (NULL);
|
|
|
|
int i;
|
|
|
|
|
2014-09-09 01:20:14 +00:00
|
|
|
guint64 *windows;
|
2014-09-11 17:43:32 +00:00
|
|
|
int n_windows;
|
2017-08-26 17:03:51 +00:00
|
|
|
meta_stack_tracker_get_stack (display->stack_tracker, &windows, &n_windows);
|
2014-09-11 17:43:32 +00:00
|
|
|
|
|
|
|
for (i = 0; i < n_windows; i++)
|
|
|
|
{
|
2014-09-09 01:20:14 +00:00
|
|
|
if (META_STACK_ID_IS_X11 (windows[i]))
|
2014-09-11 17:43:32 +00:00
|
|
|
{
|
|
|
|
if (local_string->len > 0)
|
|
|
|
g_string_append_c (local_string, ' ');
|
|
|
|
|
2014-09-09 01:20:14 +00:00
|
|
|
g_string_append_printf (local_string, "%#lx", (Window)windows[i]);
|
2014-09-11 17:43:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Window root;
|
|
|
|
Window parent;
|
|
|
|
Window *children;
|
|
|
|
unsigned int n_children;
|
2017-08-26 16:26:30 +00:00
|
|
|
XQueryTree (display->x11_display->xdisplay,
|
|
|
|
display->x11_display->xroot,
|
2014-09-11 17:43:32 +00:00
|
|
|
&root, &parent, &children, &n_children);
|
|
|
|
|
|
|
|
for (i = 0; i < (int)n_children; i++)
|
|
|
|
{
|
|
|
|
if (x11_string->len > 0)
|
|
|
|
g_string_append_c (x11_string, ' ');
|
|
|
|
|
|
|
|
g_string_append_printf (x11_string, "%#lx", (Window)children[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp (x11_string->str, local_string->str) != 0)
|
|
|
|
g_set_error (error, TEST_RUNNER_ERROR, TEST_RUNNER_ERROR_ASSERTION_FAILED,
|
|
|
|
"xserver stacking: x11='%s', local='%s'",
|
|
|
|
x11_string->str, local_string->str);
|
|
|
|
|
|
|
|
XFree (children);
|
|
|
|
|
|
|
|
g_string_free (local_string, TRUE);
|
|
|
|
g_string_free (x11_string, TRUE);
|
|
|
|
|
|
|
|
return *error == NULL;
|
|
|
|
}
|
|
|
|
|
2018-07-12 08:25:16 +00:00
|
|
|
typedef struct _WaitForShownData
|
|
|
|
{
|
|
|
|
GMainLoop *loop;
|
|
|
|
MetaWindow *window;
|
|
|
|
guint shown_handler_id;
|
|
|
|
} WaitForShownData;
|
|
|
|
|
|
|
|
static void
|
|
|
|
on_window_shown (MetaWindow *window,
|
|
|
|
WaitForShownData *data)
|
|
|
|
{
|
|
|
|
g_main_loop_quit (data->loop);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
test_case_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;
|
|
|
|
}
|
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
static gboolean
|
|
|
|
test_case_do (TestCase *test,
|
|
|
|
int argc,
|
|
|
|
char **argv,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
if (strcmp (argv[0], "new_client") == 0)
|
|
|
|
{
|
|
|
|
MetaWindowClientType type;
|
2017-06-26 08:31:48 +00:00
|
|
|
TestClient *client;
|
2014-09-11 17:43:32 +00:00
|
|
|
|
|
|
|
if (argc != 3)
|
|
|
|
BAD_COMMAND("usage: new_client <client-id> [wayland|x11]");
|
|
|
|
|
|
|
|
if (strcmp (argv[2], "x11") == 0)
|
|
|
|
type = META_WINDOW_CLIENT_TYPE_X11;
|
|
|
|
else if (strcmp (argv[2], "wayland") == 0)
|
|
|
|
type = META_WINDOW_CLIENT_TYPE_WAYLAND;
|
|
|
|
else
|
|
|
|
BAD_COMMAND("usage: new_client <client-id> [wayland|x11]");
|
|
|
|
|
|
|
|
if (g_hash_table_lookup (test->clients, argv[1]))
|
|
|
|
BAD_COMMAND("client %s already exists", argv[1]);
|
|
|
|
|
2017-06-26 08:31:48 +00:00
|
|
|
client = test_client_new (argv[1], type, error);
|
2014-09-11 17:43:32 +00:00
|
|
|
if (!client)
|
|
|
|
return FALSE;
|
|
|
|
|
2017-06-26 08:31:48 +00:00
|
|
|
g_hash_table_insert (test->clients, test_client_get_id (client), client);
|
2014-09-11 17:43:32 +00:00
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "quit_client") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
BAD_COMMAND("usage: quit_client <client-id>");
|
|
|
|
|
|
|
|
TestClient *client = test_case_lookup_client (test, argv[1], error);
|
|
|
|
if (!client)
|
|
|
|
return FALSE;
|
|
|
|
|
2017-08-18 06:23:42 +00:00
|
|
|
if (!test_client_quit (client, error))
|
2014-09-11 17:43:32 +00:00
|
|
|
return FALSE;
|
|
|
|
|
2017-06-26 08:31:48 +00:00
|
|
|
g_hash_table_remove (test->clients, test_client_get_id (client));
|
2014-09-11 17:43:32 +00:00
|
|
|
test_client_destroy (client);
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "create") == 0)
|
|
|
|
{
|
|
|
|
if (!(argc == 2 ||
|
2014-09-15 17:24:05 +00:00
|
|
|
(argc == 3 && strcmp (argv[2], "override") == 0) ||
|
|
|
|
(argc == 3 && strcmp (argv[2], "csd") == 0)))
|
|
|
|
BAD_COMMAND("usage: %s <client-id>/<window-id > [override|csd]", argv[0]);
|
2014-09-11 17:43:32 +00:00
|
|
|
|
|
|
|
TestClient *client;
|
|
|
|
const char *window_id;
|
|
|
|
if (!test_case_parse_window_id (test, argv[1], &client, &window_id, error))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!test_client_do (client, error,
|
|
|
|
"create", window_id,
|
|
|
|
argc == 3 ? argv[2] : NULL,
|
|
|
|
NULL))
|
|
|
|
return FALSE;
|
|
|
|
}
|
2018-07-13 14:40:31 +00:00
|
|
|
else if (strcmp (argv[0], "set_parent") == 0 ||
|
|
|
|
strcmp (argv[0], "set_parent_exported") == 0)
|
2015-09-24 05:39:58 +00:00
|
|
|
{
|
|
|
|
if (argc != 3)
|
|
|
|
BAD_COMMAND("usage: %s <client-id>/<window-id> <parent-window-id>",
|
|
|
|
argv[0]);
|
|
|
|
|
|
|
|
TestClient *client;
|
|
|
|
const char *window_id;
|
|
|
|
if (!test_case_parse_window_id (test, argv[1], &client, &window_id, error))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!test_client_do (client, error,
|
2018-07-13 14:40:31 +00:00
|
|
|
argv[0], window_id,
|
2015-09-24 05:39:58 +00:00
|
|
|
argv[2],
|
|
|
|
NULL))
|
|
|
|
return FALSE;
|
|
|
|
}
|
2018-07-12 08:25:16 +00:00
|
|
|
else if (strcmp (argv[0], "show") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
BAD_COMMAND("usage: %s <client-id>/<window-id>", argv[0]);
|
|
|
|
|
|
|
|
TestClient *client;
|
|
|
|
const char *window_id;
|
|
|
|
if (!test_case_parse_window_id (test, argv[1], &client, &window_id, error))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!test_client_do (client, error, argv[0], window_id, NULL))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
MetaWindow *window = test_client_find_window (client, window_id, error);
|
|
|
|
if (!window)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
WaitForShownData data = {
|
|
|
|
.loop = g_main_loop_new (NULL, FALSE),
|
|
|
|
.window = window,
|
|
|
|
};
|
|
|
|
meta_later_add (META_LATER_BEFORE_REDRAW,
|
|
|
|
test_case_wait_for_showing_before_redraw,
|
|
|
|
&data,
|
|
|
|
NULL);
|
|
|
|
g_main_loop_run (data.loop);
|
|
|
|
if (data.shown_handler_id)
|
|
|
|
g_signal_handler_disconnect (window, data.shown_handler_id);
|
|
|
|
g_main_loop_unref (data.loop);
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "hide") == 0 ||
|
2014-09-11 17:43:32 +00:00
|
|
|
strcmp (argv[0], "activate") == 0 ||
|
|
|
|
strcmp (argv[0], "raise") == 0 ||
|
|
|
|
strcmp (argv[0], "lower") == 0 ||
|
2014-09-24 20:09:20 +00:00
|
|
|
strcmp (argv[0], "minimize") == 0 ||
|
|
|
|
strcmp (argv[0], "unminimize") == 0 ||
|
2014-09-11 17:43:32 +00:00
|
|
|
strcmp (argv[0], "destroy") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
BAD_COMMAND("usage: %s <client-id>/<window-id>", argv[0]);
|
|
|
|
|
|
|
|
TestClient *client;
|
|
|
|
const char *window_id;
|
|
|
|
if (!test_case_parse_window_id (test, argv[1], &client, &window_id, error))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!test_client_do (client, error, argv[0], window_id, NULL))
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "local_activate") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 2)
|
|
|
|
BAD_COMMAND("usage: %s <client-id>/<window-id>", argv[0]);
|
|
|
|
|
|
|
|
TestClient *client;
|
|
|
|
const char *window_id;
|
|
|
|
if (!test_case_parse_window_id (test, argv[1], &client, &window_id, error))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
MetaWindow *window = test_client_find_window (client, window_id, error);
|
|
|
|
if (!window)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
meta_window_activate (window, 0);
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "wait") == 0)
|
|
|
|
{
|
|
|
|
if (argc != 1)
|
|
|
|
BAD_COMMAND("usage: %s", argv[0]);
|
|
|
|
|
|
|
|
if (!test_case_wait (test, error))
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
else if (strcmp (argv[0], "assert_stacking") == 0)
|
|
|
|
{
|
|
|
|
if (!test_case_assert_stacking (test, argv + 1, argc - 1, error))
|
|
|
|
return FALSE;
|
2014-09-24 20:09:20 +00:00
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
if (!test_case_check_xserver_stacking (test, error))
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
BAD_COMMAND("Unknown command %s", argv[0]);
|
|
|
|
}
|
|
|
|
|
2014-09-15 18:50:21 +00:00
|
|
|
return test_case_check_warnings (test, error);
|
2014-09-11 17:43:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
test_case_destroy (TestCase *test,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
/* Failures when cleaning up the test case aren't recoverable, since we'll
|
|
|
|
* pollute the subsequent test cases, so we just return the error, and
|
|
|
|
* skip the rest of the cleanup.
|
|
|
|
*/
|
|
|
|
GHashTableIter iter;
|
|
|
|
gpointer key, value;
|
|
|
|
|
|
|
|
g_hash_table_iter_init (&iter, test->clients);
|
|
|
|
while (g_hash_table_iter_next (&iter, &key, &value))
|
|
|
|
{
|
|
|
|
if (!test_client_do (value, error, "destroy_all", NULL))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!test_case_wait (test, error))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!test_case_assert_stacking (test, NULL, 0, error))
|
|
|
|
return FALSE;
|
|
|
|
|
2014-09-15 18:50:21 +00:00
|
|
|
if (!test_case_check_warnings (test, error))
|
|
|
|
return FALSE;
|
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
g_hash_table_iter_init (&iter, test->clients);
|
|
|
|
while (g_hash_table_iter_next (&iter, &key, &value))
|
|
|
|
test_client_destroy (value);
|
|
|
|
|
|
|
|
async_waiter_destroy (test->waiter);
|
|
|
|
|
2017-08-26 16:56:44 +00:00
|
|
|
meta_x11_display_set_alarm_filter (meta_get_display ()->x11_display,
|
|
|
|
NULL, NULL);
|
2014-09-11 17:43:32 +00:00
|
|
|
|
|
|
|
g_hash_table_destroy (test->clients);
|
|
|
|
g_free (test);
|
|
|
|
|
2014-09-15 18:50:21 +00:00
|
|
|
g_log_remove_handler ("mutter", test->log_handler_id);
|
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**********************************************************************/
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
run_test (const char *filename,
|
|
|
|
int index)
|
|
|
|
{
|
|
|
|
TestCase *test = test_case_new ();
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
GFile *file = g_file_new_for_path (filename);
|
|
|
|
|
|
|
|
GDataInputStream *in = NULL;
|
|
|
|
|
|
|
|
GFileInputStream *in_raw = g_file_read (file, NULL, &error);
|
|
|
|
g_object_unref (file);
|
|
|
|
if (in_raw == NULL)
|
|
|
|
goto out;
|
|
|
|
|
|
|
|
in = g_data_input_stream_new (G_INPUT_STREAM (in_raw));
|
|
|
|
g_object_unref (in_raw);
|
|
|
|
|
|
|
|
int line_no = 0;
|
|
|
|
while (error == NULL)
|
|
|
|
{
|
|
|
|
char *line = g_data_input_stream_read_line_utf8 (in, NULL, NULL, &error);
|
|
|
|
if (line == NULL)
|
|
|
|
break;
|
|
|
|
|
|
|
|
line_no++;
|
|
|
|
|
|
|
|
int argc;
|
|
|
|
char **argv = NULL;
|
|
|
|
if (!g_shell_parse_argv (line, &argc, &argv, &error))
|
|
|
|
{
|
|
|
|
if (g_error_matches (error, G_SHELL_ERROR, G_SHELL_ERROR_EMPTY_STRING))
|
|
|
|
{
|
|
|
|
g_clear_error (&error);
|
|
|
|
goto next;
|
|
|
|
}
|
|
|
|
|
|
|
|
goto next;
|
|
|
|
}
|
|
|
|
|
|
|
|
test_case_do (test, argc, argv, &error);
|
|
|
|
|
|
|
|
next:
|
|
|
|
if (error)
|
|
|
|
g_prefix_error (&error, "%d: ", line_no);
|
|
|
|
|
|
|
|
g_free (line);
|
|
|
|
g_strfreev (argv);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
GError *tmp_error = NULL;
|
|
|
|
if (!g_input_stream_close (G_INPUT_STREAM (in), NULL, &tmp_error))
|
|
|
|
{
|
|
|
|
if (error != NULL)
|
|
|
|
g_clear_error (&tmp_error);
|
|
|
|
else
|
|
|
|
g_propagate_error (&error, tmp_error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
out:
|
|
|
|
if (in != NULL)
|
|
|
|
g_object_unref (in);
|
|
|
|
|
|
|
|
GError *cleanup_error = NULL;
|
|
|
|
test_case_destroy (test, &cleanup_error);
|
|
|
|
|
|
|
|
const char *testspos = strstr (filename, "tests/");
|
|
|
|
char *pretty_name;
|
|
|
|
if (testspos)
|
|
|
|
pretty_name = g_strdup (testspos + strlen("tests/"));
|
|
|
|
else
|
|
|
|
pretty_name = g_strdup (filename);
|
|
|
|
|
|
|
|
if (error || cleanup_error)
|
|
|
|
{
|
|
|
|
g_print ("not ok %d %s\n", index, pretty_name);
|
|
|
|
|
|
|
|
if (error)
|
|
|
|
g_print (" %s\n", error->message);
|
|
|
|
|
|
|
|
if (cleanup_error)
|
|
|
|
{
|
|
|
|
g_print (" Fatal Error During Cleanup\n");
|
|
|
|
g_print (" %s\n", cleanup_error->message);
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
g_print ("ok %d %s\n", index, pretty_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (pretty_name);
|
|
|
|
|
|
|
|
gboolean success = error == NULL;
|
|
|
|
|
|
|
|
g_clear_error (&error);
|
|
|
|
g_clear_error (&cleanup_error);
|
|
|
|
|
|
|
|
return success;
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
int n_tests;
|
|
|
|
char **tests;
|
|
|
|
} RunTestsInfo;
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
run_tests (gpointer data)
|
|
|
|
{
|
|
|
|
RunTestsInfo *info = data;
|
|
|
|
int i;
|
|
|
|
gboolean success = TRUE;
|
|
|
|
|
|
|
|
g_print ("1..%d\n", info->n_tests);
|
|
|
|
|
|
|
|
for (i = 0; i < info->n_tests; i++)
|
|
|
|
if (!run_test (info->tests[i], i + 1))
|
|
|
|
success = FALSE;
|
|
|
|
|
|
|
|
meta_quit (success ? 0 : 1);
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**********************************************************************/
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
find_metatests_in_directory (GFile *directory,
|
|
|
|
GPtrArray *results,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
GFileEnumerator *enumerator = g_file_enumerate_children (directory,
|
|
|
|
"standard::name,standard::type",
|
|
|
|
G_FILE_QUERY_INFO_NONE,
|
|
|
|
NULL, error);
|
|
|
|
if (!enumerator)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
while (*error == NULL)
|
|
|
|
{
|
|
|
|
GFileInfo *info = g_file_enumerator_next_file (enumerator, NULL, error);
|
|
|
|
if (info == NULL)
|
|
|
|
break;
|
|
|
|
|
|
|
|
GFile *child = g_file_enumerator_get_child (enumerator, info);
|
|
|
|
switch (g_file_info_get_file_type (info))
|
|
|
|
{
|
|
|
|
case G_FILE_TYPE_REGULAR:
|
|
|
|
{
|
|
|
|
const char *name = g_file_info_get_name (info);
|
|
|
|
if (g_str_has_suffix (name, ".metatest"))
|
|
|
|
g_ptr_array_add (results, g_file_get_path (child));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case G_FILE_TYPE_DIRECTORY:
|
|
|
|
find_metatests_in_directory (child, results, error);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_object_unref (child);
|
|
|
|
g_object_unref (info);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
GError *tmp_error = NULL;
|
|
|
|
if (!g_file_enumerator_close (enumerator, NULL, &tmp_error))
|
|
|
|
{
|
|
|
|
if (*error != NULL)
|
|
|
|
g_clear_error (&tmp_error);
|
|
|
|
else
|
|
|
|
g_propagate_error (error, tmp_error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_object_unref (enumerator);
|
|
|
|
return *error == NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean all_tests = FALSE;
|
|
|
|
|
|
|
|
const GOptionEntry options[] = {
|
|
|
|
{
|
|
|
|
"all", 0, 0, G_OPTION_ARG_NONE,
|
|
|
|
&all_tests,
|
|
|
|
"Run all installed tests",
|
|
|
|
NULL
|
|
|
|
},
|
|
|
|
{ NULL }
|
|
|
|
};
|
|
|
|
|
|
|
|
int
|
|
|
|
main (int argc, char **argv)
|
|
|
|
{
|
|
|
|
GOptionContext *ctx;
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
/* First parse the arguments that are passed to us */
|
|
|
|
|
|
|
|
ctx = g_option_context_new (NULL);
|
|
|
|
g_option_context_add_main_entries (ctx, options, NULL);
|
|
|
|
|
|
|
|
if (!g_option_context_parse (ctx,
|
|
|
|
&argc, &argv, &error))
|
|
|
|
{
|
|
|
|
g_printerr ("%s", error->message);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
g_option_context_free (ctx);
|
|
|
|
|
2017-06-26 08:31:48 +00:00
|
|
|
test_init (argc, argv);
|
|
|
|
|
2014-09-11 17:43:32 +00:00
|
|
|
GPtrArray *tests = g_ptr_array_new ();
|
|
|
|
|
|
|
|
if (all_tests)
|
|
|
|
{
|
|
|
|
GFile *test_dir = g_file_new_for_path (MUTTER_PKGDATADIR "/tests");
|
|
|
|
|
|
|
|
if (!find_metatests_in_directory (test_dir, tests, &error))
|
|
|
|
{
|
|
|
|
g_printerr ("Error enumerating tests: %s\n", error->message);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
char *curdir = g_get_current_dir ();
|
|
|
|
|
|
|
|
for (i = 1; i < argc; i++)
|
|
|
|
{
|
|
|
|
if (g_path_is_absolute (argv[i]))
|
|
|
|
g_ptr_array_add (tests, g_strdup (argv[i]));
|
|
|
|
else
|
|
|
|
g_ptr_array_add (tests, g_build_filename (curdir, argv[i], NULL));
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (curdir);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Then initalize mutter with a different set of arguments */
|
|
|
|
|
2016-03-05 06:17:00 +00:00
|
|
|
char *fake_args[] = { NULL, (char *)"--wayland", (char *)"--nested" };
|
2014-09-11 17:43:32 +00:00
|
|
|
fake_args[0] = argv[0];
|
|
|
|
char **fake_argv = fake_args;
|
2016-03-05 06:17:00 +00:00
|
|
|
int fake_argc = G_N_ELEMENTS (fake_args);
|
2014-09-11 17:43:32 +00:00
|
|
|
|
|
|
|
ctx = meta_get_option_context ();
|
|
|
|
if (!g_option_context_parse (ctx, &fake_argc, &fake_argv, &error))
|
|
|
|
{
|
|
|
|
g_printerr ("mutter: %s\n", error->message);
|
|
|
|
exit (1);
|
|
|
|
}
|
|
|
|
g_option_context_free (ctx);
|
|
|
|
|
2018-07-12 20:20:20 +00:00
|
|
|
meta_plugin_manager_load (test_get_plugin_name ());
|
2016-12-05 13:26:46 +00:00
|
|
|
meta_wayland_override_display_name ("mutter-test-display");
|
2014-09-11 17:43:32 +00:00
|
|
|
|
|
|
|
meta_init ();
|
|
|
|
meta_register_with_session ();
|
|
|
|
|
|
|
|
RunTestsInfo info;
|
|
|
|
info.tests = (char **)tests->pdata;
|
|
|
|
info.n_tests = tests->len;
|
|
|
|
|
|
|
|
g_idle_add (run_tests, &info);
|
|
|
|
|
|
|
|
return meta_run ();
|
|
|
|
}
|