Merge branch 'xwayland' into 'master'

xwayland: support create multi wlr_xwayland

See merge request wlroots/wlroots!4346
This commit is contained in:
JiDe Zhang 2023-10-02 21:02:53 +00:00
commit c23b18d5d0
6 changed files with 102 additions and 28 deletions

View file

@ -1,2 +1,3 @@
#include <wlr/xwayland/server.h> #include <wlr/xwayland/server.h>
#include <wlr/xwayland/xwayland.h> #include <wlr/xwayland/xwayland.h>
#include <wlr/xwayland/shell.h>

View file

@ -26,11 +26,10 @@ struct wlr_xwayland_shell_v1 {
// private state // private state
struct wl_client *client; struct wl_list clients; // wlr_xwayland_shell_client.link
struct wl_list surfaces; // wlr_xwayland_surface_v1.link struct wl_list surfaces; // wlr_xwayland_surface_v1.link
struct wl_listener display_destroy; struct wl_listener display_destroy;
struct wl_listener client_destroy;
}; };
/** /**
@ -65,9 +64,22 @@ void wlr_xwayland_shell_v1_destroy(struct wlr_xwayland_shell_v1 *shell);
/** /**
* Allow a client to bind to the xwayland_shell_v1 global. * Allow a client to bind to the xwayland_shell_v1 global.
*/ */
void wlr_xwayland_shell_v1_set_client(struct wlr_xwayland_shell_v1 *shell, void wlr_xwayland_shell_v1_add_client(struct wlr_xwayland_shell_v1 *shell,
struct wl_client *client); struct wl_client *client);
/**
* Cancel allow a client to bind to the xwayland_shell_v1 global.
*/
void wlr_xwayland_shell_v1_remove_client(struct wlr_xwayland_shell_v1 *shell,
struct wl_client *client);
/**
* Return true if allow this client bind to the xwayland_shell_v1 global,
* Return false otherwise.
*/
bool wlr_xwayland_shell_has_client(struct wlr_xwayland_shell_v1 *shell,
const struct wl_client *client);
/** /**
* Get a Wayland surface from an xwayland_shell_v1 serial. * Get a Wayland surface from an xwayland_shell_v1 serial.
* *

View file

@ -19,6 +19,7 @@
struct wlr_xwm; struct wlr_xwm;
struct wlr_data_source; struct wlr_data_source;
struct wlr_drag; struct wlr_drag;
struct wlr_xwayland_shell_v1;
struct wlr_xwayland { struct wlr_xwayland {
struct wlr_xwayland_server *server; struct wlr_xwayland_server *server;
@ -197,7 +198,8 @@ struct wlr_xwayland_minimize_event {
* client tries to connect. * client tries to connect.
*/ */
struct wlr_xwayland *wlr_xwayland_create(struct wl_display *wl_display, struct wlr_xwayland *wlr_xwayland_create(struct wl_display *wl_display,
struct wlr_compositor *compositor, bool lazy); struct wlr_compositor *compositor, struct wlr_xwayland_shell_v1 *shell,
bool lazy);
void wlr_xwayland_destroy(struct wlr_xwayland *wlr_xwayland); void wlr_xwayland_destroy(struct wlr_xwayland *wlr_xwayland);

View file

@ -23,6 +23,7 @@
#include <wlr/types/wlr_subcompositor.h> #include <wlr/types/wlr_subcompositor.h>
#include <wlr/types/wlr_xcursor_manager.h> #include <wlr/types/wlr_xcursor_manager.h>
#include <wlr/types/wlr_xdg_shell.h> #include <wlr/types/wlr_xdg_shell.h>
#include <wlr/xwayland.h>
#include <wlr/util/log.h> #include <wlr/util/log.h>
#include <xkbcommon/xkbcommon.h> #include <xkbcommon/xkbcommon.h>

View file

@ -7,6 +7,55 @@
#define SHELL_VERSION 1 #define SHELL_VERSION 1
struct wlr_xwayland_shell_client {
struct wl_client *client;
struct wl_listener client_destroy;
struct wl_list link;
};
static void xwl_shell_client_destroy(struct wlr_xwayland_shell_client *xwl_client) {
wl_list_remove(&xwl_client->link);
wl_list_remove(&xwl_client->client_destroy.link);
free(xwl_client);
}
static void shell_client_handle_client_destroy(struct wl_listener *listener, void *data) {
struct wlr_xwayland_shell_client *shell_client =
wl_container_of(listener, shell_client, client_destroy);
xwl_shell_client_destroy(shell_client);
}
static struct wlr_xwayland_shell_client *xwl_shell_client_create(struct wl_client *client) {
struct wlr_xwayland_shell_client *xwl_client = calloc(1, sizeof(*xwl_client));
if (xwl_client == NULL) {
wl_client_post_no_memory(client);
return NULL;
}
xwl_client->client = client;
wl_list_init(&xwl_client->link);
wl_list_init(&xwl_client->client_destroy.link);
xwl_client->client_destroy.notify = shell_client_handle_client_destroy;
wl_client_add_destroy_listener(client, &xwl_client->client_destroy);
return xwl_client;
}
static struct wlr_xwayland_shell_client *get_shell_client(
struct wlr_xwayland_shell_v1 *shell,
const struct wl_client *client) {
struct wlr_xwayland_shell_client *xwl_client, *tmp;
wl_list_for_each_safe(xwl_client, tmp, &shell->clients, link) {
if (xwl_client->client == client) {
return xwl_client;
}
}
return NULL;
}
static void destroy_resource(struct wl_client *client, static void destroy_resource(struct wl_client *client,
struct wl_resource *resource) { struct wl_resource *resource) {
wl_resource_destroy(resource); wl_resource_destroy(resource);
@ -140,7 +189,7 @@ static void shell_bind(struct wl_client *client, void *data, uint32_t version,
uint32_t id) { uint32_t id) {
struct wlr_xwayland_shell_v1 *shell = data; struct wlr_xwayland_shell_v1 *shell = data;
if (client != shell->client) { if (!get_shell_client(shell, client)) {
wl_client_post_implementation_error(client, wl_client_post_implementation_error(client,
"Permission denied to bind to %s", xwayland_shell_v1_interface.name); "Permission denied to bind to %s", xwayland_shell_v1_interface.name);
return; return;
@ -183,7 +232,7 @@ struct wlr_xwayland_shell_v1 *wlr_xwayland_shell_v1_create(
shell->display_destroy.notify = handle_display_destroy; shell->display_destroy.notify = handle_display_destroy;
wl_display_add_destroy_listener(display, &shell->display_destroy); wl_display_add_destroy_listener(display, &shell->display_destroy);
wl_list_init(&shell->client_destroy.link); wl_list_init(&shell->clients);
return shell; return shell;
} }
@ -198,30 +247,44 @@ void wlr_xwayland_shell_v1_destroy(struct wlr_xwayland_shell_v1 *shell) {
xwl_surface_destroy(xwl_surface); xwl_surface_destroy(xwl_surface);
} }
{
struct wlr_xwayland_shell_client *xwl_client, *tmp;
wl_list_for_each_safe(xwl_client, tmp, &shell->clients, link) {
xwl_shell_client_destroy(xwl_client);
}
}
wl_list_remove(&shell->display_destroy.link); wl_list_remove(&shell->display_destroy.link);
wl_list_remove(&shell->client_destroy.link);
wl_global_destroy(shell->global); wl_global_destroy(shell->global);
free(shell); free(shell);
} }
static void shell_handle_client_destroy(struct wl_listener *listener, void *data) { void wlr_xwayland_shell_v1_add_client(struct wlr_xwayland_shell_v1 *shell,
struct wlr_xwayland_shell_v1 *shell = struct wl_client *client) {
wl_container_of(listener, shell, client_destroy); if (get_shell_client(shell, client))
wlr_xwayland_shell_v1_set_client(shell, NULL); return; // already added
struct wlr_xwayland_shell_client *xwl_client = xwl_shell_client_create(client);
if (!xwl_client) {
return;
}
wl_list_insert(&shell->clients, &xwl_client->link);
} }
void wlr_xwayland_shell_v1_set_client(struct wlr_xwayland_shell_v1 *shell, void wlr_xwayland_shell_v1_remove_client(struct wlr_xwayland_shell_v1 *shell,
struct wl_client *client) { struct wl_client *client) {
wl_list_remove(&shell->client_destroy.link); struct wlr_xwayland_shell_client *xwl_client = get_shell_client(shell, client);
shell->client = client; if (xwl_client) {
if (client != NULL) { xwl_shell_client_destroy(xwl_client);
shell->client_destroy.notify = shell_handle_client_destroy;
wl_client_add_destroy_listener(client, &shell->client_destroy);
} else {
wl_list_init(&shell->client_destroy.link);
} }
} }
bool wlr_xwayland_shell_has_client(struct wlr_xwayland_shell_v1 *shell,
const struct wl_client *client) {
return get_shell_client(shell, client);
}
struct wlr_surface *wlr_xwayland_shell_v1_surface_from_serial( struct wlr_surface *wlr_xwayland_shell_v1_surface_from_serial(
struct wlr_xwayland_shell_v1 *shell, uint64_t serial) { struct wlr_xwayland_shell_v1 *shell, uint64_t serial) {
struct wlr_xwayland_surface_v1 *xwl_surface; struct wlr_xwayland_surface_v1 *xwl_surface;

View file

@ -37,7 +37,7 @@ static void handle_server_destroy(struct wl_listener *listener, void *data) {
static void handle_server_start(struct wl_listener *listener, void *data) { static void handle_server_start(struct wl_listener *listener, void *data) {
struct wlr_xwayland *xwayland = struct wlr_xwayland *xwayland =
wl_container_of(listener, xwayland, server_start); wl_container_of(listener, xwayland, server_start);
wlr_xwayland_shell_v1_set_client(xwayland->shell_v1, xwayland->server->client); wlr_xwayland_shell_v1_add_client(xwayland->shell_v1, xwayland->server->client);
} }
static void handle_server_ready(struct wl_listener *listener, void *data) { static void handle_server_ready(struct wl_listener *listener, void *data) {
@ -76,12 +76,12 @@ void wlr_xwayland_destroy(struct wlr_xwayland *xwayland) {
wlr_xwayland_set_seat(xwayland, NULL); wlr_xwayland_set_seat(xwayland, NULL);
wlr_xwayland_server_destroy(xwayland->server); wlr_xwayland_server_destroy(xwayland->server);
xwayland->server = NULL; xwayland->server = NULL;
wlr_xwayland_shell_v1_destroy(xwayland->shell_v1);
free(xwayland); free(xwayland);
} }
struct wlr_xwayland *wlr_xwayland_create(struct wl_display *wl_display, struct wlr_xwayland *wlr_xwayland_create(struct wl_display *wl_display,
struct wlr_compositor *compositor, bool lazy) { struct wlr_compositor *compositor, struct wlr_xwayland_shell_v1 *shell,
bool lazy) {
struct wlr_xwayland *xwayland = calloc(1, sizeof(struct wlr_xwayland)); struct wlr_xwayland *xwayland = calloc(1, sizeof(struct wlr_xwayland));
if (!xwayland) { if (!xwayland) {
return NULL; return NULL;
@ -94,11 +94,7 @@ struct wlr_xwayland *wlr_xwayland_create(struct wl_display *wl_display,
wl_signal_init(&xwayland->events.ready); wl_signal_init(&xwayland->events.ready);
wl_signal_init(&xwayland->events.remove_startup_info); wl_signal_init(&xwayland->events.remove_startup_info);
xwayland->shell_v1 = wlr_xwayland_shell_v1_create(wl_display, 1); xwayland->shell_v1 = shell;
if (xwayland->shell_v1 == NULL) {
free(xwayland);
return NULL;
}
struct wlr_xwayland_server_options options = { struct wlr_xwayland_server_options options = {
.lazy = lazy, .lazy = lazy,
@ -109,7 +105,6 @@ struct wlr_xwayland *wlr_xwayland_create(struct wl_display *wl_display,
}; };
xwayland->server = wlr_xwayland_server_create(wl_display, &options); xwayland->server = wlr_xwayland_server_create(wl_display, &options);
if (xwayland->server == NULL) { if (xwayland->server == NULL) {
wlr_xwayland_shell_v1_destroy(xwayland->shell_v1);
free(xwayland); free(xwayland);
return NULL; return NULL;
} }