/* * 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 . * */ #include "config.h" #include #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); }