/* * Copyright (c) 2019 Red Hat, Inc. * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice (including the * next paragraph) shall be included in all copies or substantial * portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ #include #include #include "wayland-server.h" #include "wayland-client.h" #include "test-runner.h" static struct { struct wl_display *display; struct wl_event_loop *loop; int sync_count; } server; static struct { struct wl_display *display; struct wl_callback *callback_a; struct wl_callback *callback_b; int callback_count; } client; static const char *tag_a = "tag"; static const char *tag_b = "tag"; static void callback_done(void *data, struct wl_callback *cb, uint32_t time) { const char * const *expected_tag; const char * const *tag; if (cb == client.callback_a) expected_tag = &tag_a; else if (cb == client.callback_b) expected_tag = &tag_b; else assert(!"unexpected callback"); tag = wl_proxy_get_tag((struct wl_proxy *) cb); assert(tag == expected_tag); assert(strcmp(*tag, "tag") == 0); wl_callback_destroy(cb); client.callback_count++; } static const struct wl_callback_listener callback_listener = { callback_done, }; static void logger_func(void *user_data, enum wl_protocol_logger_type type, const struct wl_protocol_logger_message *message) { if (type != WL_PROTOCOL_LOGGER_REQUEST) return; assert(strcmp(wl_resource_get_class(message->resource), "wl_display") == 0); assert(strcmp(message->message->name, "sync") == 0); server.sync_count++; } TEST(proxy_tag) { const char *socket; struct wl_protocol_logger *logger; assert(&tag_a != &tag_b); server.display = wl_display_create(); assert(server.display); server.loop = wl_display_get_event_loop(server.display); assert(server.loop); socket = wl_display_add_socket_auto(server.display); assert(socket); logger = wl_display_add_protocol_logger(server.display, logger_func, NULL); assert(logger); client.display = wl_display_connect(socket); assert(client.display); client.callback_a = wl_display_sync(client.display); wl_callback_add_listener(client.callback_a, &callback_listener, NULL); wl_proxy_set_tag((struct wl_proxy *) client.callback_a, &tag_a); client.callback_b = wl_display_sync(client.display); wl_callback_add_listener(client.callback_b, &callback_listener, NULL); wl_proxy_set_tag((struct wl_proxy *) client.callback_b, &tag_b); wl_display_flush(client.display); while (server.sync_count < 2) { wl_event_loop_dispatch(server.loop, -1); wl_display_flush_clients(server.display); } wl_display_dispatch(client.display); assert(client.callback_count == 2); wl_display_disconnect(client.display); wl_event_loop_dispatch(server.loop, 100); wl_display_destroy(server.display); }