aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSimon Ser <contact@emersion.fr>2020-02-03 16:39:27 +0100
committerDrew DeVault <sir@cmpwn.com>2020-02-17 20:52:37 +0100
commit29726e6816e47adb199f2c84fbb4365e98bbeb6c (patch)
tree860dc59252fe2b4551fc95e345c0c6592972ba8f
parent2bad34e02423f19257b6f0473aebc46dcab74098 (diff)
util: add wlr_output_destroy_safe
This adds a generic helper to destory transient globals. See [1]. This patch depends on [2] and [3]. [1]: https://gitlab.freedesktop.org/wayland/wayland/issues/10 [2]: https://gitlab.freedesktop.org/wayland/wayland/merge_requests/28 [3]: https://gitlab.freedesktop.org/wayland/wayland/merge_requests/30
-rw-r--r--include/util/global.h15
-rw-r--r--util/global.c43
-rw-r--r--util/meson.build1
-rw-r--r--wlroots.syms1
4 files changed, 60 insertions, 0 deletions
diff --git a/include/util/global.h b/include/util/global.h
new file mode 100644
index 00000000..ea0fda83
--- /dev/null
+++ b/include/util/global.h
@@ -0,0 +1,15 @@
+#ifndef UTIL_GLOBAL_H
+#define UTIL_GLOBAL_H
+
+#include <wayland-server-core.h>
+
+/**
+ * Destroy a transient global.
+ *
+ * Globals that are created and destroyed on the fly need special handling to
+ * prevent race conditions with wl_registry. Use this function to destroy them.
+ */
+void wlr_global_destroy_safe(struct wl_global *global,
+ struct wl_display *display);
+
+#endif
diff --git a/util/global.c b/util/global.c
new file mode 100644
index 00000000..a0d84ed3
--- /dev/null
+++ b/util/global.c
@@ -0,0 +1,43 @@
+#include <stdlib.h>
+#include "util/global.h"
+
+struct destroy_global_data {
+ struct wl_global *global;
+ struct wl_event_source *event_source;
+};
+
+static int destroy_global(void *_data) {
+ struct destroy_global_data *data = _data;
+ wl_global_destroy(data->global);
+ wl_event_source_remove(data->event_source);
+ free(data);
+ return 0;
+}
+
+void wlr_global_destroy_safe(struct wl_global *global,
+ struct wl_display *display) {
+ // Don't destroy the global immediately. If the global has been created
+ // recently, clients might try to bind to it after we've destroyed it.
+ // Instead, remove the global so that clients stop seeing it and wait an
+ // arbitrary amount of time before destroying the global as a workaround.
+ // See: https://gitlab.freedesktop.org/wayland/wayland/issues/10
+
+ wl_global_remove(global);
+ wl_global_set_user_data(global, NULL); // safety net
+
+ struct wl_event_loop *event_loop = wl_display_get_event_loop(display);
+ struct destroy_global_data *data = calloc(1, sizeof(*data));
+ if (data == NULL) {
+ wl_global_destroy(global);
+ return;
+ }
+ data->global = global;
+ data->event_source =
+ wl_event_loop_add_timer(event_loop, destroy_global, data);
+ if (data->event_source == NULL) {
+ free(data);
+ wl_global_destroy(global);
+ return;
+ }
+ wl_event_source_timer_update(data->event_source, 5000);
+}
diff --git a/util/meson.build b/util/meson.build
index c6614275..8bdd7a74 100644
--- a/util/meson.build
+++ b/util/meson.build
@@ -1,5 +1,6 @@
wlr_files += files(
'array.c',
+ 'global.c',
'log.c',
'region.c',
'shm.c',
diff --git a/wlroots.syms b/wlroots.syms
index 3176f874..167046ed 100644
--- a/wlroots.syms
+++ b/wlroots.syms
@@ -6,5 +6,6 @@
_wlr_strip_path;
local:
wlr_signal_emit_safe;
+ wlr_global_destroy_safe;
*;
};