#define _POSIX_C_SOURCE 200809L #include #include #include #include #include #include #include #include #include "linux-dmabuf-unstable-v1-protocol.h" #include "render/drm_format_set.h" #include "util/signal.h" #include "util/shm.h" #define LINUX_DMABUF_VERSION 4 struct wlr_linux_buffer_params_v1 { struct wl_resource *resource; struct wlr_linux_dmabuf_v1 *linux_dmabuf; struct wlr_dmabuf_attributes attributes; bool has_modifier; }; struct wlr_linux_dmabuf_feedback_v1_compiled_tranche { dev_t target_device; uint32_t flags; // bitfield of enum zwp_linux_dmabuf_feedback_v1_tranche_flags struct wl_array indices; // uint16_t }; struct wlr_linux_dmabuf_feedback_v1_compiled { dev_t main_device; int table_fd; size_t table_size; size_t tranches_len; struct wlr_linux_dmabuf_feedback_v1_compiled_tranche tranches[]; }; struct wlr_linux_dmabuf_feedback_v1_table_entry { uint32_t format; uint32_t pad; // unused uint64_t modifier; }; // TODO: switch back to static_assert once this fix propagates in stable trees: // https://bugs.freebsd.org/bugzilla/show_bug.cgi?id=255290 _Static_assert(sizeof(struct wlr_linux_dmabuf_feedback_v1_table_entry) == 16, "Expected wlr_linux_dmabuf_feedback_v1_table_entry to be tightly packed"); static void buffer_handle_destroy(struct wl_client *client, struct wl_resource *resource) { wl_resource_destroy(resource); } static const struct wl_buffer_interface wl_buffer_impl = { .destroy = buffer_handle_destroy, }; bool wlr_dmabuf_v1_resource_is_buffer(struct wl_resource *resource) { if (!wl_resource_instance_of(resource, &wl_buffer_interface, &wl_buffer_impl)) { return false; } return wl_resource_get_user_data(resource) != NULL; } struct wlr_dmabuf_v1_buffer *wlr_dmabuf_v1_buffer_from_buffer_resource( struct wl_resource *resource) { assert(wl_resource_instance_of(resource, &wl_buffer_interface, &wl_buffer_impl)); return wl_resource_get_user_data(resource); } static const struct wlr_buffer_impl buffer_impl; static struct wlr_dmabuf_v1_buffer *dmabuf_v1_buffer_from_buffer( struct wlr_buffer *buffer) { assert(buffer->impl == &buffer_impl); return (struct wlr_dmabuf_v1_buffer *)buffer; } static void buffer_destroy(struct wlr_buffer *wlr_buffer) { struct wlr_dmabuf_v1_buffer *buffer = dmabuf_v1_buffer_from_buffer(wlr_buffer); if (buffer->resource != NULL) { wl_resource_set_user_data(buffer->resource, NULL); } wlr_dmabuf_attributes_finish(&buffer->attributes); wl_list_remove(&buffer->release.link); free(buffer); } static bool buffer_get_dmabuf(struct wlr_buffer *wlr_buffer, struct wlr_dmabuf_attributes *attribs) { struct wlr_dmabuf_v1_buffer *buffer = dmabuf_v1_buffer_from_buffer(wlr_buffer); memcpy(attribs, &buffer->attributes, sizeof(buffer->attributes)); return true; } static const struct wlr_buffer_impl buffer_impl = { .destroy = buffer_destroy, .get_dmabuf = buffer_get_dmabuf, }; static void buffer_handle_release(struct wl_listener *listener, void *data) { struct wlr_dmabuf_v1_buffer *buffer = wl_container_of(listener, buffer, release); if (buffer->resource != NULL) { wl_buffer_send_release(buffer->resource); } } static const struct zwp_linux_buffer_params_v1_interface buffer_params_impl; static struct wlr_linux_buffer_params_v1 *params_from_resource( struct wl_resource *resource) { assert(wl_resource_instance_of(resource, &zwp_linux_buffer_params_v1_interface, &buffer_params_impl)); return wl_resource_get_user_data(resource); } static void params_destroy(struct wl_client *client, struct wl_resource *resource) { wl_resource_destroy(resource); } static void params_add(struct wl_client *client, struct wl_resource *params_resource, int32_t fd, uint32_t plane_idx, uint32_t offset, uint32_t stride, uint32_t modifier_hi, uint32_t modifier_lo) { struct wlr_linux_buffer_params_v1 *params = params_from_resource(params_resource); if (!params) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_ALREADY_USED, "params was already used to create a wl_buffer"); close(fd); return; } if (plane_idx >= WLR_DMABUF_MAX_PLANES) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_PLANE_IDX, "plane index %u > %u", plane_idx, WLR_DMABUF_MAX_PLANES); close(fd); return; } if (params->attributes.fd[plane_idx] != -1) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_PLANE_SET, "a dmabuf with FD %d has already been added for plane %u", params->attributes.fd[plane_idx], plane_idx); close(fd); return; } uint64_t modifier = ((uint64_t)modifier_hi << 32) | modifier_lo; if (params->has_modifier && modifier != params->attributes.modifier) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INVALID_FORMAT, "sent modifier %" PRIu64 " for plane %u, expected" " modifier %" PRIu64 " like other planes", modifier, plane_idx, params->attributes.modifier); close(fd); return; } params->attributes.modifier = modifier; params->has_modifier = true; params->attributes.fd[plane_idx] = fd; params->attributes.offset[plane_idx] = offset; params->attributes.stride[plane_idx] = stride; params->attributes.n_planes++; } static void buffer_handle_resource_destroy(struct wl_resource *buffer_resource) { struct wlr_dmabuf_v1_buffer *buffer = wlr_dmabuf_v1_buffer_from_buffer_resource(buffer_resource); buffer->resource = NULL; wlr_buffer_drop(&buffer->base); } static bool check_import_dmabuf(struct wlr_linux_dmabuf_v1 *linux_dmabuf, struct wlr_dmabuf_attributes *attribs) { struct wlr_texture *texture = wlr_texture_from_dmabuf(linux_dmabuf->renderer, attribs); if (texture == NULL) { return false; } // We can import the image, good. No need to keep it since wlr_surface will // import it again on commit. wlr_texture_destroy(texture); return true; } static void params_create_common(struct wl_resource *params_resource, uint32_t buffer_id, int32_t width, int32_t height, uint32_t format, uint32_t flags) { struct wlr_linux_buffer_params_v1 *params = params_from_resource(params_resource); if (!params) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_ALREADY_USED, "params was already used to create a wl_buffer"); return; } struct wlr_dmabuf_attributes attribs = params->attributes; struct wlr_linux_dmabuf_v1 *linux_dmabuf = params->linux_dmabuf; // Make the params resource inert wl_resource_set_user_data(params_resource, NULL); free(params); if (!attribs.n_planes) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INCOMPLETE, "no dmabuf has been added to the params"); goto err_out; } if (attribs.fd[0] == -1) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INCOMPLETE, "no dmabuf has been added for plane 0"); goto err_out; } if ((attribs.fd[3] >= 0 || attribs.fd[2] >= 0) && (attribs.fd[2] == -1 || attribs.fd[1] == -1)) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INCOMPLETE, "gap in dmabuf planes"); goto err_out; } /* reject unknown flags */ uint32_t all_flags = ZWP_LINUX_BUFFER_PARAMS_V1_FLAGS_Y_INVERT | ZWP_LINUX_BUFFER_PARAMS_V1_FLAGS_INTERLACED | ZWP_LINUX_BUFFER_PARAMS_V1_FLAGS_BOTTOM_FIRST; if (flags & ~all_flags) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INVALID_FORMAT, "Unknown dmabuf flags %"PRIu32, flags); goto err_out; } if (flags != 0) { wlr_log(WLR_ERROR, "dmabuf flags aren't supported"); goto err_failed; } attribs.width = width; attribs.height = height; attribs.format = format; if (width < 1 || height < 1) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INVALID_DIMENSIONS, "invalid width %d or height %d", width, height); goto err_out; } for (int i = 0; i < attribs.n_planes; i++) { if ((uint64_t)attribs.offset[i] + attribs.stride[i] > UINT32_MAX) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS, "size overflow for plane %d", i); goto err_out; } if ((uint64_t)attribs.offset[i] + (uint64_t)attribs.stride[i] * height > UINT32_MAX) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS, "size overflow for plane %d", i); goto err_out; } off_t size = lseek(attribs.fd[i], 0, SEEK_END); if (size == -1) { // Skip checks if kernel does no support seek on buffer continue; } if (attribs.offset[i] > size) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS, "invalid offset %" PRIu32 " for plane %d", attribs.offset[i], i); goto err_out; } if (attribs.offset[i] + attribs.stride[i] > size || attribs.stride[i] == 0) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS, "invalid stride %" PRIu32 " for plane %d", attribs.stride[i], i); goto err_out; } // planes > 0 might be subsampled according to fourcc format if (i == 0 && attribs.offset[i] + attribs.stride[i] * height > size) { wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS, "invalid buffer stride or height for plane %d", i); goto err_out; } } /* Check if dmabuf is usable */ if (!check_import_dmabuf(linux_dmabuf, &attribs)) { goto err_failed; } struct wlr_dmabuf_v1_buffer *buffer = calloc(1, sizeof(*buffer)); if (!buffer) { wl_resource_post_no_memory(params_resource); goto err_failed; } wlr_buffer_init(&buffer->base, &buffer_impl, attribs.width, attribs.height); struct wl_client *client = wl_resource_get_client(params_resource); buffer->resource = wl_resource_create(client, &wl_buffer_interface, 1, buffer_id); if (!buffer->resource) { wl_resource_post_no_memory(params_resource); free(buffer); goto err_failed; } wl_resource_set_implementation(buffer->resource, &wl_buffer_impl, buffer, buffer_handle_resource_destroy); buffer->attributes = attribs; buffer->release.notify = buffer_handle_release; wl_signal_add(&buffer->base.events.release, &buffer->release); /* send 'created' event when the request is not for an immediate * import, that is buffer_id is zero */ if (buffer_id == 0) { zwp_linux_buffer_params_v1_send_created(params_resource, buffer->resource); } return; err_failed: if (buffer_id == 0) { zwp_linux_buffer_params_v1_send_failed(params_resource); } else { /* since the behavior is left implementation defined by the * protocol in case of create_immed failure due to an unknown cause, * we choose to treat it as a fatal error and immediately kill the * client instead of creating an invalid handle and waiting for it * to be used. */ wl_resource_post_error(params_resource, ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INVALID_WL_BUFFER, "importing the supplied dmabufs failed"); } err_out: wlr_dmabuf_attributes_finish(&attribs); } static void params_create(struct wl_client *client, struct wl_resource *params_resource, int32_t width, int32_t height, uint32_t format, uint32_t flags) { params_create_common(params_resource, 0, width, height, format, flags); } static void params_create_immed(struct wl_client *client, struct wl_resource *params_resource, uint32_t buffer_id, int32_t width, int32_t height, uint32_t format, uint32_t flags) { params_create_common(params_resource, buffer_id, width, height, format, flags); } static const struct zwp_linux_buffer_params_v1_interface buffer_params_impl = { .destroy = params_destroy, .add = params_add, .create = params_create, .create_immed = params_create_immed, }; static void params_handle_resource_destroy(struct wl_resource *resource) { struct wlr_linux_buffer_params_v1 *params = params_from_resource(resource); if (!params) { return; } wlr_dmabuf_attributes_finish(¶ms->attributes); free(params); } static const struct zwp_linux_dmabuf_v1_interface linux_dmabuf_impl; static struct wlr_linux_dmabuf_v1 *linux_dmabuf_from_resource( struct wl_resource *resource) { assert(wl_resource_instance_of(resource, &zwp_linux_dmabuf_v1_interface, &linux_dmabuf_impl)); struct wlr_linux_dmabuf_v1 *dmabuf = wl_resource_get_user_data(resource); assert(dmabuf); return dmabuf; } static void linux_dmabuf_create_params(struct wl_client *client, struct wl_resource *linux_dmabuf_resource, uint32_t params_id) { struct wlr_linux_dmabuf_v1 *linux_dmabuf = linux_dmabuf_from_resource(linux_dmabuf_resource); struct wlr_linux_buffer_params_v1 *params = calloc(1, sizeof(*params)); if (!params) { wl_resource_post_no_memory(linux_dmabuf_resource); return; } for (int i = 0; i < WLR_DMABUF_MAX_PLANES; i++) { params->attributes.fd[i] = -1; } params->linux_dmabuf = linux_dmabuf; uint32_t version = wl_resource_get_version(linux_dmabuf_resource); params->resource = wl_resource_create(client, &zwp_linux_buffer_params_v1_interface, version, params_id); if (!params->resource) { free(params); wl_resource_post_no_memory(linux_dmabuf_resource); return; } wl_resource_set_implementation(params->resource, &buffer_params_impl, params, params_handle_resource_destroy); } static void linux_dmabuf_feedback_destroy(struct wl_client *client, struct wl_resource *resource) { wl_resource_destroy(resource); } static const struct zwp_linux_dmabuf_feedback_v1_interface linux_dmabuf_feedback_impl = { .destroy = linux_dmabuf_feedback_destroy, }; static struct wlr_linux_dmabuf_feedback_v1_compiled *feedback_compile( const struct wlr_linux_dmabuf_feedback_v1 *feedback) { assert(feedback->tranches_len > 0); // Require the last tranche to be the fallback tranche and contain all // formats/modifiers const struct wlr_linux_dmabuf_feedback_v1_tranche *fallback_tranche = &feedback->tranches[feedback->tranches_len - 1]; size_t table_len = 0; for (size_t i = 0; i < fallback_tranche->formats->len; i++) { const struct wlr_drm_format *fmt = fallback_tranche->formats->formats[i]; table_len += 1 + fmt->len; } assert(table_len > 0); size_t table_size = table_len * sizeof(struct wlr_linux_dmabuf_feedback_v1_table_entry); int rw_fd, ro_fd; if (!allocate_shm_file_pair(table_size, &rw_fd, &ro_fd)) { wlr_log(WLR_ERROR, "Failed to allocate shm file for format table"); return NULL; } struct wlr_linux_dmabuf_feedback_v1_table_entry *table = mmap(NULL, table_size, PROT_READ | PROT_WRITE, MAP_SHARED, rw_fd, 0); if (table == MAP_FAILED) { wlr_log_errno(WLR_ERROR, "mmap failed"); close(rw_fd); close(ro_fd); return NULL; } close(rw_fd); size_t n = 0; for (size_t i = 0; i < fallback_tranche->formats->len; i++) { const struct wlr_drm_format *fmt = fallback_tranche->formats->formats[i]; table[n] = (struct wlr_linux_dmabuf_feedback_v1_table_entry){ .format = fmt->format, .modifier = DRM_FORMAT_MOD_INVALID, }; n++; for (size_t k = 0; k < fmt->len; k++) { table[n] = (struct wlr_linux_dmabuf_feedback_v1_table_entry){ .format = fmt->format, .modifier = fmt->modifiers[k], }; n++; } } assert(n == table_len); munmap(table, table_size); struct wlr_linux_dmabuf_feedback_v1_compiled *compiled = calloc(1, sizeof(struct wlr_linux_dmabuf_feedback_v1_compiled) + feedback->tranches_len * sizeof(struct wlr_linux_dmabuf_feedback_v1_compiled_tranche)); if (compiled == NULL) { close(ro_fd); return NULL; } compiled->main_device = feedback->main_device; compiled->tranches_len = feedback->tranches_len; compiled->table_fd = ro_fd; compiled->table_size = table_size; // Build the indices lists for all but the last (fallback) tranches for (size_t i = 0; i < feedback->tranches_len - 1; i++) { assert(false); // TODO: unimplemented } struct wlr_linux_dmabuf_feedback_v1_compiled_tranche *fallback_compiled_tranche = &compiled->tranches[compiled->tranches_len - 1]; fallback_compiled_tranche->target_device = fallback_tranche->target_device; fallback_compiled_tranche->flags = fallback_tranche->flags; // Build the indices list for the last (fallback) tranche wl_array_init(&fallback_compiled_tranche->indices); if (!wl_array_add(&fallback_compiled_tranche->indices, table_len * sizeof(uint16_t))) { wlr_log(WLR_ERROR, "Failed to allocate fallback tranche indices array"); goto error_compiled; } n = 0; uint16_t *index_ptr; wl_array_for_each(index_ptr, &fallback_compiled_tranche->indices) { *index_ptr = n; n++; } return compiled; error_compiled: close(compiled->table_fd); free(compiled); return NULL; } static void compiled_feedback_destroy( struct wlr_linux_dmabuf_feedback_v1_compiled *feedback) { for (size_t i = 0; i < feedback->tranches_len; i++) { wl_array_release(&feedback->tranches[i].indices); } close(feedback->table_fd); free(feedback); } static bool feedback_tranche_init_with_renderer( struct wlr_linux_dmabuf_feedback_v1_tranche *tranche, struct wlr_renderer *renderer) { memset(tranche, 0, sizeof(*tranche)); int drm_fd = wlr_renderer_get_drm_fd(renderer); if (drm_fd < 0) { wlr_log(WLR_ERROR, "Failed to get DRM FD from renderer"); return false; } struct stat stat; if (fstat(drm_fd, &stat) != 0) { wlr_log_errno(WLR_ERROR, "fstat failed"); return false; } tranche->target_device = stat.st_rdev; tranche->formats = wlr_renderer_get_dmabuf_texture_formats(renderer); if (tranche->formats == NULL) { wlr_log(WLR_ERROR, "Failed to get renderer DMA-BUF texture formats"); return false; } return true; } static struct wlr_linux_dmabuf_feedback_v1_compiled *compile_default_feedback( struct wlr_renderer *renderer) { struct wlr_linux_dmabuf_feedback_v1_tranche tranche = {0}; if (!feedback_tranche_init_with_renderer(&tranche, renderer)) { return NULL; } const struct wlr_linux_dmabuf_feedback_v1 feedback = { .main_device = tranche.target_device, .tranches = &tranche, .tranches_len = 1, }; return feedback_compile(&feedback); } static void feedback_tranche_send( const struct wlr_linux_dmabuf_feedback_v1_compiled_tranche *tranche, struct wl_resource *resource) { struct wl_array dev_array = { .size = sizeof(tranche->target_device), .data = (void *)&tranche->target_device, }; zwp_linux_dmabuf_feedback_v1_send_tranche_target_device(resource, &dev_array); zwp_linux_dmabuf_feedback_v1_send_tranche_flags(resource, tranche->flags); zwp_linux_dmabuf_feedback_v1_send_tranche_formats(resource, (struct wl_array *)&tranche->indices); zwp_linux_dmabuf_feedback_v1_send_tranche_done(resource); } static void feedback_send(const struct wlr_linux_dmabuf_feedback_v1_compiled *feedback, struct wl_resource *resource) { struct wl_array dev_array = { .size = sizeof(feedback->main_device), .data = (void *)&feedback->main_device, }; zwp_linux_dmabuf_feedback_v1_send_main_device(resource, &dev_array); zwp_linux_dmabuf_feedback_v1_send_format_table(resource, feedback->table_fd, feedback->table_size); for (size_t i = 0; i < feedback->tranches_len; i++) { feedback_tranche_send(&feedback->tranches[i], resource); } zwp_linux_dmabuf_feedback_v1_send_done(resource); } static void linux_dmabuf_get_default_feedback(struct wl_client *client, struct wl_resource *resource, uint32_t id) { struct wlr_linux_dmabuf_v1 *linux_dmabuf = linux_dmabuf_from_resource(resource); uint32_t version = wl_resource_get_version(resource); struct wl_resource *feedback_resource = wl_resource_create(client, &zwp_linux_dmabuf_feedback_v1_interface, version, id); if (feedback_resource == NULL) { wl_client_post_no_memory(client); return; } wl_resource_set_implementation(feedback_resource, &linux_dmabuf_feedback_impl, NULL, NULL); feedback_send(linux_dmabuf->default_feedback, feedback_resource); } static void linux_dmabuf_get_surface_feedback(struct wl_client *client, struct wl_resource *resource, uint32_t id, struct wl_resource *surface_resource) { // TODO: implement per-surface feedback linux_dmabuf_get_default_feedback(client, resource, id); } static void linux_dmabuf_destroy(struct wl_client *client, struct wl_resource *resource) { wl_resource_destroy(resource); } static const struct zwp_linux_dmabuf_v1_interface linux_dmabuf_impl = { .destroy = linux_dmabuf_destroy, .create_params = linux_dmabuf_create_params, .get_default_feedback = linux_dmabuf_get_default_feedback, .get_surface_feedback = linux_dmabuf_get_surface_feedback, }; static void linux_dmabuf_send_modifiers(struct wl_resource *resource, const struct wlr_drm_format *fmt) { if (wl_resource_get_version(resource) < ZWP_LINUX_DMABUF_V1_MODIFIER_SINCE_VERSION) { if (wlr_drm_format_has(fmt, DRM_FORMAT_MOD_INVALID)) { zwp_linux_dmabuf_v1_send_format(resource, fmt->format); } return; } // In case only INVALID and LINEAR are advertised, send INVALID only due to XWayland: // https://gitlab.freedesktop.org/xorg/xserver/-/issues/1166 if (fmt->len == 2 && wlr_drm_format_has(fmt, DRM_FORMAT_MOD_INVALID) && wlr_drm_format_has(fmt, DRM_FORMAT_MOD_LINEAR)) { uint64_t mod = DRM_FORMAT_MOD_INVALID; zwp_linux_dmabuf_v1_send_modifier(resource, fmt->format, mod >> 32, mod & 0xFFFFFFFF); return; } for (size_t i = 0; i < fmt->len; i++) { uint64_t mod = fmt->modifiers[i]; zwp_linux_dmabuf_v1_send_modifier(resource, fmt->format, mod >> 32, mod & 0xFFFFFFFF); } } static void linux_dmabuf_send_formats(struct wlr_linux_dmabuf_v1 *linux_dmabuf, struct wl_resource *resource) { const struct wlr_drm_format_set *formats = wlr_renderer_get_dmabuf_texture_formats(linux_dmabuf->renderer); if (formats == NULL) { return; } for (size_t i = 0; i < formats->len; i++) { const struct wlr_drm_format *fmt = formats->formats[i]; linux_dmabuf_send_modifiers(resource, fmt); } } static void linux_dmabuf_bind(struct wl_client *client, void *data, uint32_t version, uint32_t id) { struct wlr_linux_dmabuf_v1 *linux_dmabuf = data; struct wl_resource *resource = wl_resource_create(client, &zwp_linux_dmabuf_v1_interface, version, id); if (resource == NULL) { wl_client_post_no_memory(client); return; } wl_resource_set_implementation(resource, &linux_dmabuf_impl, linux_dmabuf, NULL); if (version < ZWP_LINUX_DMABUF_V1_GET_DEFAULT_FEEDBACK_SINCE_VERSION) { linux_dmabuf_send_formats(linux_dmabuf, resource); } } static void linux_dmabuf_v1_destroy(struct wlr_linux_dmabuf_v1 *linux_dmabuf) { wlr_signal_emit_safe(&linux_dmabuf->events.destroy, linux_dmabuf); compiled_feedback_destroy(linux_dmabuf->default_feedback); wl_list_remove(&linux_dmabuf->display_destroy.link); wl_list_remove(&linux_dmabuf->renderer_destroy.link); wl_global_destroy(linux_dmabuf->global); free(linux_dmabuf); } static void handle_display_destroy(struct wl_listener *listener, void *data) { struct wlr_linux_dmabuf_v1 *linux_dmabuf = wl_container_of(listener, linux_dmabuf, display_destroy); linux_dmabuf_v1_destroy(linux_dmabuf); } static void handle_renderer_destroy(struct wl_listener *listener, void *data) { struct wlr_linux_dmabuf_v1 *linux_dmabuf = wl_container_of(listener, linux_dmabuf, renderer_destroy); linux_dmabuf_v1_destroy(linux_dmabuf); } struct wlr_linux_dmabuf_v1 *wlr_linux_dmabuf_v1_create(struct wl_display *display, struct wlr_renderer *renderer) { struct wlr_linux_dmabuf_v1 *linux_dmabuf = calloc(1, sizeof(struct wlr_linux_dmabuf_v1)); if (linux_dmabuf == NULL) { wlr_log(WLR_ERROR, "could not create simple dmabuf manager"); return NULL; } linux_dmabuf->renderer = renderer; wl_signal_init(&linux_dmabuf->events.destroy); linux_dmabuf->global = wl_global_create(display, &zwp_linux_dmabuf_v1_interface, LINUX_DMABUF_VERSION, linux_dmabuf, linux_dmabuf_bind); if (!linux_dmabuf->global) { wlr_log(WLR_ERROR, "could not create linux dmabuf v1 wl global"); free(linux_dmabuf); return NULL; } linux_dmabuf->default_feedback = compile_default_feedback(renderer); if (linux_dmabuf->default_feedback == NULL) { wlr_log(WLR_ERROR, "Failed to init default linux-dmabuf feedback"); wl_global_destroy(linux_dmabuf->global); free(linux_dmabuf); return NULL; } linux_dmabuf->display_destroy.notify = handle_display_destroy; wl_display_add_destroy_listener(display, &linux_dmabuf->display_destroy); linux_dmabuf->renderer_destroy.notify = handle_renderer_destroy; wl_signal_add(&renderer->events.destroy, &linux_dmabuf->renderer_destroy); return linux_dmabuf; }