aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSimon Ser <contact@emersion.fr>2022-12-02 14:33:02 +0100
committerSimon Zeni <simon@bl4ckb0ne.ca>2022-12-02 14:27:07 +0000
commitf36a5915da5c998d2a55b00dc62adcf9c1797d19 (patch)
tree805f084f746a23c91102e2bffa09e9cff915f8be
parentc9b378d21a06fd690223a95fe437054dff0228cd (diff)
wl-drm: don't store wlr_renderer
Query the formats at init time, then forget about the renderer. This will allow wl_drm to be created with a list of formats instead of a renderer, and will behave better after a GPU reset.
-rw-r--r--include/wlr/types/wlr_drm.h9
-rw-r--r--types/wlr_drm.c41
2 files changed, 26 insertions, 24 deletions
diff --git a/include/wlr/types/wlr_drm.h b/include/wlr/types/wlr_drm.h
index efcf139f..61780b18 100644
--- a/include/wlr/types/wlr_drm.h
+++ b/include/wlr/types/wlr_drm.h
@@ -10,6 +10,7 @@
#define WLR_TYPES_WLR_DRM_H
#include <wayland-server-protocol.h>
+#include <wlr/render/drm_format_set.h>
struct wlr_renderer;
@@ -30,15 +31,17 @@ struct wlr_drm_buffer {
*/
struct wlr_drm {
struct wl_global *global;
- struct wlr_renderer *renderer;
- char *node_name;
struct {
struct wl_signal destroy;
} events;
+ // private state
+
+ char *node_name;
+ struct wlr_drm_format_set formats;
+
struct wl_listener display_destroy;
- struct wl_listener renderer_destroy;
};
bool wlr_drm_buffer_is_resource(struct wl_resource *resource);
diff --git a/types/wlr_drm.c b/types/wlr_drm.c
index 13fe4512..ab44084b 100644
--- a/types/wlr_drm.c
+++ b/types/wlr_drm.c
@@ -11,6 +11,7 @@
#include <wlr/types/wlr_drm.h>
#include <wlr/util/log.h>
#include "drm-protocol.h"
+#include "render/drm_format_set.h"
#define WLR_DRM_VERSION 2
@@ -159,14 +160,8 @@ static void drm_bind(struct wl_client *client, void *data,
wl_drm_send_device(resource, drm->node_name);
wl_drm_send_capabilities(resource, WL_DRM_CAPABILITY_PRIME);
- const struct wlr_drm_format_set *formats =
- wlr_renderer_get_dmabuf_texture_formats(drm->renderer);
- if (formats == NULL) {
- return;
- }
-
- for (size_t i = 0; i < formats->len; i++) {
- wl_drm_send_format(resource, formats->formats[i]->format);
+ for (size_t i = 0; i < drm->formats.len; i++) {
+ wl_drm_send_format(resource, drm->formats.formats[i]->format);
}
}
@@ -185,8 +180,8 @@ static void drm_destroy(struct wlr_drm *drm) {
wl_signal_emit_mutable(&drm->events.destroy, NULL);
wl_list_remove(&drm->display_destroy.link);
- wl_list_remove(&drm->renderer_destroy.link);
+ wlr_drm_format_set_finish(&drm->formats);
free(drm->node_name);
wl_global_destroy(drm->global);
free(drm);
@@ -197,11 +192,6 @@ static void handle_display_destroy(struct wl_listener *listener, void *data) {
drm_destroy(drm);
}
-static void handle_renderer_destroy(struct wl_listener *listener, void *data) {
- struct wlr_drm *drm = wl_container_of(listener, drm, renderer_destroy);
- drm_destroy(drm);
-}
-
struct wlr_drm *wlr_drm_create(struct wl_display *display,
struct wlr_renderer *renderer) {
int drm_fd = wlr_renderer_get_drm_fd(renderer);
@@ -237,24 +227,33 @@ struct wlr_drm *wlr_drm_create(struct wl_display *display,
}
drm->node_name = node_name;
- drm->renderer = renderer;
wl_signal_init(&drm->events.destroy);
+ const struct wlr_drm_format_set *formats = wlr_renderer_get_dmabuf_texture_formats(renderer);
+ if (formats == NULL) {
+ goto error;
+ }
+
+ if (!wlr_drm_format_set_copy(&drm->formats, formats)) {
+ goto error;
+ }
+
drm->global = wl_global_create(display, &wl_drm_interface, WLR_DRM_VERSION,
drm, drm_bind);
if (drm->global == NULL) {
- free(drm->node_name);
- free(drm);
- return NULL;
+ goto error;
}
drm->display_destroy.notify = handle_display_destroy;
wl_display_add_destroy_listener(display, &drm->display_destroy);
- drm->renderer_destroy.notify = handle_renderer_destroy;
- wl_signal_add(&renderer->events.destroy, &drm->renderer_destroy);
-
wlr_buffer_register_resource_interface(&buffer_resource_interface);
return drm;
+
+error:
+ wlr_drm_format_set_finish(&drm->formats);
+ free(drm->node_name);
+ free(drm);
+ return NULL;
}