4a600e1031
g_thread_join() frees the thread object, so don't use g_autoptr (Gthread) when they will all be joined. Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/3234>
184 lines
4.8 KiB
C
184 lines
4.8 KiB
C
/*
|
|
* Copyright (C) 2023 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 <wayland-client.h>
|
|
|
|
#include "meta-test/meta-context-test.h"
|
|
#include "wayland/meta-wayland-client-private.h"
|
|
|
|
static MetaContext *test_context;
|
|
|
|
static gpointer
|
|
test_client_destroyed_thread_func (gpointer user_data)
|
|
{
|
|
int fd = GPOINTER_TO_INT (user_data);
|
|
struct wl_display *wl_display;
|
|
|
|
wl_display = wl_display_connect_to_fd (fd);
|
|
g_assert_nonnull (wl_display);
|
|
|
|
wl_display_roundtrip (wl_display);
|
|
wl_display_disconnect (wl_display);
|
|
|
|
return GINT_TO_POINTER (TRUE);
|
|
}
|
|
|
|
static void
|
|
on_client_destroyed (MetaWaylandClient *client,
|
|
gboolean *client_destroyed)
|
|
{
|
|
*client_destroyed = TRUE;
|
|
}
|
|
|
|
static void
|
|
meta_test_wayland_client_indirect_self_terminate (void)
|
|
{
|
|
g_autoptr (MetaWaylandClient) client = NULL;
|
|
g_autoptr (GError) error = NULL;
|
|
GThread *thread;
|
|
int fd;
|
|
gboolean client_destroyed = FALSE;
|
|
|
|
client = meta_wayland_client_new_indirect (test_context, &error);
|
|
g_assert_nonnull (client);
|
|
g_assert_null (error);
|
|
|
|
g_signal_connect (client, "client-destroyed",
|
|
G_CALLBACK (on_client_destroyed), &client_destroyed);
|
|
|
|
fd = meta_wayland_client_setup_fd (client, &error);
|
|
g_assert_cmpint (fd, >=, 0);
|
|
g_assert_null (error);
|
|
|
|
thread = g_thread_new ("test client thread (self-terminated)",
|
|
test_client_destroyed_thread_func,
|
|
GINT_TO_POINTER (fd));
|
|
|
|
g_debug ("Waiting for client to disconnect itself");
|
|
|
|
while (!client_destroyed)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
g_debug ("Waiting for thread to terminate");
|
|
g_thread_join (thread);
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
int fd;
|
|
volatile gboolean round_tripped;
|
|
} DestroyTestData;
|
|
|
|
static gpointer
|
|
test_client_indefinite_thread_func (gpointer user_data)
|
|
{
|
|
DestroyTestData *data = user_data;
|
|
int fd = data->fd;
|
|
struct wl_display *wl_display;
|
|
|
|
wl_display = wl_display_connect_to_fd (fd);
|
|
g_assert_nonnull (wl_display);
|
|
|
|
wl_display_roundtrip (wl_display);
|
|
g_atomic_int_set (&data->round_tripped, TRUE);
|
|
|
|
while (TRUE)
|
|
{
|
|
if (wl_display_dispatch (wl_display) == -1)
|
|
break;
|
|
}
|
|
|
|
wl_display_disconnect (wl_display);
|
|
|
|
return GINT_TO_POINTER (TRUE);
|
|
}
|
|
|
|
static void
|
|
meta_test_wayland_client_indirect_destroy (void)
|
|
{
|
|
DestroyTestData data;
|
|
g_autoptr (MetaWaylandClient) client = NULL;
|
|
g_autoptr (GError) error = NULL;
|
|
GThread *thread;
|
|
int fd;
|
|
gboolean client_destroyed = FALSE;
|
|
|
|
client = meta_wayland_client_new_indirect (test_context, &error);
|
|
g_assert_nonnull (client);
|
|
g_assert_null (error);
|
|
|
|
g_signal_connect (client, "client-destroyed",
|
|
G_CALLBACK (on_client_destroyed), &client_destroyed);
|
|
|
|
fd = meta_wayland_client_setup_fd (client, &error);
|
|
g_assert_cmpint (fd, >=, 0);
|
|
g_assert_null (error);
|
|
|
|
data = (DestroyTestData) {
|
|
.fd = fd,
|
|
.round_tripped = FALSE,
|
|
};
|
|
thread = g_thread_new ("test client thread (indefinite)",
|
|
test_client_indefinite_thread_func,
|
|
&data);
|
|
|
|
g_debug ("Waiting for client to round-trip");
|
|
while (!g_atomic_int_get (&data.round_tripped))
|
|
g_main_context_iteration (NULL, FALSE);
|
|
|
|
g_debug ("Destroying client");
|
|
g_clear_object (&client);
|
|
|
|
g_debug ("Waiting for client to terminate");
|
|
while (!client_destroyed)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
g_debug ("Waiting for thread to terminate");
|
|
g_thread_join (thread);
|
|
}
|
|
|
|
static void
|
|
init_tests (void)
|
|
{
|
|
g_test_add_func ("/wayland/client/indirect/self-terminate",
|
|
meta_test_wayland_client_indirect_self_terminate);
|
|
g_test_add_func ("/wayland/client/indirect/destroy",
|
|
meta_test_wayland_client_indirect_destroy);
|
|
}
|
|
|
|
int
|
|
main (int argc,
|
|
char **argv)
|
|
{
|
|
g_autoptr (MetaContext) context = NULL;
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
context = meta_create_test_context (META_CONTEXT_TEST_TYPE_HEADLESS,
|
|
META_CONTEXT_TEST_FLAG_NO_X11);
|
|
g_assert (meta_context_configure (context, &argc, &argv, NULL));
|
|
|
|
test_context = context;
|
|
|
|
init_tests ();
|
|
|
|
return meta_context_test_run_tests (META_CONTEXT_TEST (context),
|
|
META_TEST_RUN_FLAG_CAN_SKIP);
|
|
}
|