2018-02-23 18:45:16 +01:00
|
|
|
#define _POSIX_C_SOURCE 200809L
|
|
|
|
#include <assert.h>
|
2019-01-29 19:33:38 +01:00
|
|
|
#include <drm_fourcc.h>
|
2022-11-30 13:00:09 +01:00
|
|
|
#include <fcntl.h>
|
2018-02-23 18:45:16 +01:00
|
|
|
#include <stdlib.h>
|
2018-11-13 12:35:08 +01:00
|
|
|
#include <sys/mman.h>
|
2018-02-23 18:45:16 +01:00
|
|
|
#include <unistd.h>
|
2022-12-09 18:30:16 +01:00
|
|
|
#include <wlr/backend.h>
|
2022-03-03 15:38:26 +01:00
|
|
|
#include <wlr/interfaces/wlr_buffer.h>
|
2018-04-08 17:00:56 +02:00
|
|
|
#include <wlr/render/wlr_renderer.h>
|
2022-01-13 09:55:09 +01:00
|
|
|
#include <wlr/types/wlr_compositor.h>
|
2018-07-19 22:50:00 +02:00
|
|
|
#include <wlr/types/wlr_linux_dmabuf_v1.h>
|
2023-02-07 10:57:17 +01:00
|
|
|
#include <wlr/types/wlr_output_layer.h>
|
2018-02-23 18:45:16 +01:00
|
|
|
#include <wlr/util/log.h>
|
2022-11-30 13:00:09 +01:00
|
|
|
#include <xf86drm.h>
|
2018-02-23 18:45:16 +01:00
|
|
|
#include "linux-dmabuf-unstable-v1-protocol.h"
|
2021-03-31 17:07:55 +02:00
|
|
|
#include "render/drm_format_set.h"
|
2018-11-13 12:35:08 +01:00
|
|
|
#include "util/shm.h"
|
2018-02-23 18:45:16 +01:00
|
|
|
|
2018-11-13 12:35:08 +01:00
|
|
|
#define LINUX_DMABUF_VERSION 4
|
2018-04-30 23:20:39 +02:00
|
|
|
|
2021-11-22 10:32:55 +01:00
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2018-11-13 12:35:08 +01:00
|
|
|
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");
|
|
|
|
|
2021-06-07 18:20:45 +02:00
|
|
|
struct wlr_linux_dmabuf_v1_surface {
|
|
|
|
struct wlr_surface *surface;
|
|
|
|
struct wlr_linux_dmabuf_v1 *linux_dmabuf;
|
|
|
|
struct wl_list link; // wlr_linux_dmabuf_v1.surfaces
|
|
|
|
|
|
|
|
struct wlr_addon addon;
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1_compiled *feedback;
|
|
|
|
|
|
|
|
struct wl_list feedback_resources; // wl_resource_get_link
|
|
|
|
};
|
|
|
|
|
2018-04-30 23:20:39 +02:00
|
|
|
static void buffer_handle_destroy(struct wl_client *client,
|
2018-02-23 18:45:16 +01:00
|
|
|
struct wl_resource *resource) {
|
|
|
|
wl_resource_destroy(resource);
|
|
|
|
}
|
|
|
|
|
2021-04-12 14:09:27 +02:00
|
|
|
static const struct wl_buffer_interface wl_buffer_impl = {
|
2018-04-30 23:20:39 +02:00
|
|
|
.destroy = buffer_handle_destroy,
|
2018-02-23 18:45:16 +01:00
|
|
|
};
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
bool wlr_dmabuf_v1_resource_is_buffer(struct wl_resource *resource) {
|
|
|
|
if (!wl_resource_instance_of(resource, &wl_buffer_interface,
|
2021-04-12 14:09:27 +02:00
|
|
|
&wl_buffer_impl)) {
|
2018-02-23 18:45:16 +01:00
|
|
|
return false;
|
|
|
|
}
|
2021-04-12 11:49:57 +02:00
|
|
|
return wl_resource_get_user_data(resource) != NULL;
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2018-07-19 22:50:00 +02:00
|
|
|
struct wlr_dmabuf_v1_buffer *wlr_dmabuf_v1_buffer_from_buffer_resource(
|
2021-04-12 11:49:57 +02:00
|
|
|
struct wl_resource *resource) {
|
|
|
|
assert(wl_resource_instance_of(resource, &wl_buffer_interface,
|
2021-04-12 14:09:27 +02:00
|
|
|
&wl_buffer_impl));
|
2021-04-12 11:49:57 +02:00
|
|
|
return wl_resource_get_user_data(resource);
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2021-04-12 14:09:27 +02:00
|
|
|
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);
|
|
|
|
}
|
2018-05-31 13:33:27 +02:00
|
|
|
wlr_dmabuf_attributes_finish(&buffer->attributes);
|
2021-04-12 14:09:27 +02:00
|
|
|
wl_list_remove(&buffer->release.link);
|
2018-02-23 18:45:16 +01:00
|
|
|
free(buffer);
|
|
|
|
}
|
|
|
|
|
2021-04-12 14:09:27 +02:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
static const struct zwp_linux_buffer_params_v1_interface buffer_params_impl;
|
2021-04-12 11:03:49 +02:00
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
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);
|
2021-04-12 11:03:49 +02:00
|
|
|
}
|
|
|
|
|
2018-04-08 17:00:56 +02:00
|
|
|
static void params_destroy(struct wl_client *client,
|
|
|
|
struct wl_resource *resource) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_destroy(resource);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void params_add(struct wl_client *client,
|
2018-05-29 23:38:00 +02:00
|
|
|
struct wl_resource *params_resource, int32_t fd,
|
2018-02-23 18:45:16 +01:00
|
|
|
uint32_t plane_idx, uint32_t offset, uint32_t stride,
|
|
|
|
uint32_t modifier_hi, uint32_t modifier_lo) {
|
2021-04-12 11:49:57 +02:00
|
|
|
struct wlr_linux_buffer_params_v1 *params =
|
|
|
|
params_from_resource(params_resource);
|
|
|
|
if (!params) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_ALREADY_USED,
|
|
|
|
"params was already used to create a wl_buffer");
|
2018-05-29 23:38:00 +02:00
|
|
|
close(fd);
|
2018-02-23 18:45:16 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-29 23:38:00 +02:00
|
|
|
if (plane_idx >= WLR_DMABUF_MAX_PLANES) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_PLANE_IDX,
|
2018-05-29 23:38:00 +02:00
|
|
|
"plane index %u > %u", plane_idx, WLR_DMABUF_MAX_PLANES);
|
|
|
|
close(fd);
|
2018-02-23 18:45:16 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
if (params->attributes.fd[plane_idx] != -1) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_PLANE_SET,
|
2018-05-29 23:38:00 +02:00
|
|
|
"a dmabuf with FD %d has already been added for plane %u",
|
2021-04-12 11:49:57 +02:00
|
|
|
params->attributes.fd[plane_idx], plane_idx);
|
2018-05-29 23:38:00 +02:00
|
|
|
close(fd);
|
2018-02-23 18:45:16 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-29 23:38:00 +02:00
|
|
|
uint64_t modifier = ((uint64_t)modifier_hi << 32) | modifier_lo;
|
2021-04-12 11:49:57 +02:00
|
|
|
if (params->has_modifier && modifier != params->attributes.modifier) {
|
2018-05-29 23:38:00 +02:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INVALID_FORMAT,
|
2018-06-04 15:20:09 +02:00
|
|
|
"sent modifier %" PRIu64 " for plane %u, expected"
|
|
|
|
" modifier %" PRIu64 " like other planes",
|
2021-04-12 11:49:57 +02:00
|
|
|
modifier, plane_idx, params->attributes.modifier);
|
2018-05-29 23:38:00 +02:00
|
|
|
close(fd);
|
2018-02-23 18:45:16 +01:00
|
|
|
return;
|
|
|
|
}
|
2018-10-12 22:04:12 +02:00
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
params->attributes.modifier = modifier;
|
|
|
|
params->has_modifier = true;
|
2018-02-23 18:45:16 +01:00
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
params->attributes.fd[plane_idx] = fd;
|
|
|
|
params->attributes.offset[plane_idx] = offset;
|
|
|
|
params->attributes.stride[plane_idx] = stride;
|
|
|
|
params->attributes.n_planes++;
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2018-04-30 23:20:39 +02:00
|
|
|
static void buffer_handle_resource_destroy(struct wl_resource *buffer_resource) {
|
2018-07-19 22:50:00 +02:00
|
|
|
struct wlr_dmabuf_v1_buffer *buffer =
|
|
|
|
wlr_dmabuf_v1_buffer_from_buffer_resource(buffer_resource);
|
2021-04-12 14:09:27 +02:00
|
|
|
buffer->resource = NULL;
|
|
|
|
wlr_buffer_drop(&buffer->base);
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
static bool check_import_dmabuf(struct wlr_linux_dmabuf_v1 *linux_dmabuf,
|
|
|
|
struct wlr_dmabuf_attributes *attribs) {
|
2022-11-30 13:00:09 +01:00
|
|
|
// TODO: check number of planes
|
|
|
|
for (int i = 0; i < attribs->n_planes; i++) {
|
|
|
|
uint32_t handle = 0;
|
|
|
|
if (drmPrimeFDToHandle(linux_dmabuf->main_device_fd, attribs->fd[i], &handle) != 0) {
|
|
|
|
wlr_log_errno(WLR_DEBUG, "Failed to import DMA-BUF FD");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (drmCloseBufferHandle(linux_dmabuf->main_device_fd, handle) != 0) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "Failed to close buffer handle");
|
|
|
|
return false;
|
|
|
|
}
|
2018-05-30 15:18:07 +02:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
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) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_ALREADY_USED,
|
|
|
|
"params was already used to create a wl_buffer");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
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);
|
2018-02-23 18:45:16 +01:00
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
if (!attribs.n_planes) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INCOMPLETE,
|
|
|
|
"no dmabuf has been added to the params");
|
|
|
|
goto err_out;
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
if (attribs.fd[0] == -1) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INCOMPLETE,
|
2018-03-15 11:22:08 +01:00
|
|
|
"no dmabuf has been added for plane 0");
|
2018-02-23 18:45:16 +01:00
|
|
|
goto err_out;
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
if ((attribs.fd[3] >= 0 || attribs.fd[2] >= 0) &&
|
|
|
|
(attribs.fd[2] == -1 || attribs.fd[1] == -1)) {
|
2018-04-30 23:20:39 +02:00
|
|
|
wl_resource_post_error(params_resource,
|
2018-02-23 18:45:16 +01:00
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_INCOMPLETE,
|
2018-03-15 11:22:08 +01:00
|
|
|
"gap in dmabuf planes");
|
2018-02-23 18:45:16 +01:00
|
|
|
goto err_out;
|
|
|
|
}
|
|
|
|
|
2021-11-16 22:55:54 +01:00
|
|
|
/* 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;
|
|
|
|
}
|
|
|
|
|
2021-11-16 22:51:06 +01:00
|
|
|
if (flags != 0) {
|
|
|
|
wlr_log(WLR_ERROR, "dmabuf flags aren't supported");
|
|
|
|
goto err_failed;
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
attribs.width = width;
|
|
|
|
attribs.height = height;
|
|
|
|
attribs.format = format;
|
2018-02-23 18:45:16 +01:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
for (int i = 0; i < attribs.n_planes; i++) {
|
|
|
|
if ((uint64_t)attribs.offset[i]
|
|
|
|
+ attribs.stride[i] > UINT32_MAX) {
|
2018-03-15 11:22:08 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS,
|
|
|
|
"size overflow for plane %d", i);
|
|
|
|
goto err_out;
|
|
|
|
}
|
2018-02-23 18:45:16 +01:00
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
if ((uint64_t)attribs.offset[i]
|
|
|
|
+ (uint64_t)attribs.stride[i] * height > UINT32_MAX) {
|
2018-03-15 11:22:08 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS,
|
|
|
|
"size overflow for plane %d", i);
|
|
|
|
goto err_out;
|
|
|
|
}
|
2018-02-23 18:45:16 +01:00
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
off_t size = lseek(attribs.fd[i], 0, SEEK_END);
|
2018-05-29 23:38:00 +02:00
|
|
|
if (size == -1) {
|
|
|
|
// Skip checks if kernel does no support seek on buffer
|
2018-03-15 11:22:08 +01:00
|
|
|
continue;
|
|
|
|
}
|
2021-04-12 11:49:57 +02:00
|
|
|
if (attribs.offset[i] > size) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS,
|
2020-06-04 14:33:32 +02:00
|
|
|
"invalid offset %" PRIu32 " for plane %d",
|
2021-04-12 11:49:57 +02:00
|
|
|
attribs.offset[i], i);
|
2018-02-23 18:45:16 +01:00
|
|
|
goto err_out;
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
if (attribs.offset[i] + attribs.stride[i] > size ||
|
|
|
|
attribs.stride[i] == 0) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS,
|
2020-06-04 14:33:32 +02:00
|
|
|
"invalid stride %" PRIu32 " for plane %d",
|
2021-04-12 11:49:57 +02:00
|
|
|
attribs.stride[i], i);
|
2018-02-23 18:45:16 +01:00
|
|
|
goto err_out;
|
|
|
|
}
|
|
|
|
|
2018-05-29 23:38:00 +02:00
|
|
|
// planes > 0 might be subsampled according to fourcc format
|
2021-04-12 11:49:57 +02:00
|
|
|
if (i == 0 && attribs.offset[i] +
|
|
|
|
attribs.stride[i] * height > size) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_post_error(params_resource,
|
|
|
|
ZWP_LINUX_BUFFER_PARAMS_V1_ERROR_OUT_OF_BOUNDS,
|
2018-03-15 11:22:08 +01:00
|
|
|
"invalid buffer stride or height for plane %d", i);
|
2018-03-11 19:05:11 +01:00
|
|
|
goto err_out;
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if dmabuf is usable */
|
2021-04-12 11:49:57 +02:00
|
|
|
if (!check_import_dmabuf(linux_dmabuf, &attribs)) {
|
2018-02-23 18:45:16 +01:00
|
|
|
goto err_failed;
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
struct wlr_dmabuf_v1_buffer *buffer = calloc(1, sizeof(*buffer));
|
|
|
|
if (!buffer) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_post_no_memory(params_resource);
|
|
|
|
goto err_failed;
|
|
|
|
}
|
2021-04-12 14:09:27 +02:00
|
|
|
wlr_buffer_init(&buffer->base, &buffer_impl, attribs.width, attribs.height);
|
2018-02-23 18:45:16 +01:00
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
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);
|
2021-04-12 14:09:27 +02:00
|
|
|
free(buffer);
|
2021-04-12 11:49:57 +02:00
|
|
|
goto err_failed;
|
|
|
|
}
|
|
|
|
wl_resource_set_implementation(buffer->resource,
|
2021-04-12 14:09:27 +02:00
|
|
|
&wl_buffer_impl, buffer, buffer_handle_resource_destroy);
|
2018-02-23 18:45:16 +01:00
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
buffer->attributes = attribs;
|
|
|
|
|
2021-04-12 14:09:27 +02:00
|
|
|
buffer->release.notify = buffer_handle_release;
|
|
|
|
wl_signal_add(&buffer->base.events.release, &buffer->release);
|
|
|
|
|
2018-02-23 18:45:16 +01:00
|
|
|
/* 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,
|
2021-04-12 11:49:57 +02:00
|
|
|
buffer->resource);
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
2021-04-12 11:49:57 +02:00
|
|
|
|
2018-02-23 18:45:16 +01:00
|
|
|
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:
|
2021-04-12 11:49:57 +02:00
|
|
|
wlr_dmabuf_attributes_finish(&attribs);
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void params_create(struct wl_client *client,
|
|
|
|
struct wl_resource *params_resource,
|
2018-04-08 17:00:56 +02:00
|
|
|
int32_t width, int32_t height, uint32_t format, uint32_t flags) {
|
2021-04-12 11:49:57 +02:00
|
|
|
params_create_common(params_resource, 0, width, height, format,
|
2018-04-08 17:00:56 +02:00
|
|
|
flags);
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void params_create_immed(struct wl_client *client,
|
|
|
|
struct wl_resource *params_resource, uint32_t buffer_id,
|
2018-04-08 17:00:56 +02:00
|
|
|
int32_t width, int32_t height, uint32_t format, uint32_t flags) {
|
2021-04-12 11:49:57 +02:00
|
|
|
params_create_common(params_resource, buffer_id, width, height,
|
2018-04-08 17:00:56 +02:00
|
|
|
format, flags);
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
static const struct zwp_linux_buffer_params_v1_interface buffer_params_impl = {
|
2018-04-08 17:00:56 +02:00
|
|
|
.destroy = params_destroy,
|
|
|
|
.add = params_add,
|
|
|
|
.create = params_create,
|
|
|
|
.create_immed = params_create_immed,
|
2018-02-23 18:45:16 +01:00
|
|
|
};
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
static void params_handle_resource_destroy(struct wl_resource *resource) {
|
|
|
|
struct wlr_linux_buffer_params_v1 *params = params_from_resource(resource);
|
|
|
|
if (!params) {
|
2018-02-23 18:45:16 +01:00
|
|
|
return;
|
|
|
|
}
|
2021-04-12 11:49:57 +02:00
|
|
|
wlr_dmabuf_attributes_finish(¶ms->attributes);
|
|
|
|
free(params);
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2021-04-12 11:03:49 +02:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2018-02-23 18:45:16 +01:00
|
|
|
static void linux_dmabuf_create_params(struct wl_client *client,
|
|
|
|
struct wl_resource *linux_dmabuf_resource,
|
|
|
|
uint32_t params_id) {
|
2018-07-19 22:50:00 +02:00
|
|
|
struct wlr_linux_dmabuf_v1 *linux_dmabuf =
|
2021-04-12 11:03:49 +02:00
|
|
|
linux_dmabuf_from_resource(linux_dmabuf_resource);
|
2018-02-23 18:45:16 +01:00
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
struct wlr_linux_buffer_params_v1 *params = calloc(1, sizeof(*params));
|
|
|
|
if (!params) {
|
|
|
|
wl_resource_post_no_memory(linux_dmabuf_resource);
|
|
|
|
return;
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2018-05-29 23:38:00 +02:00
|
|
|
for (int i = 0; i < WLR_DMABUF_MAX_PLANES; i++) {
|
2021-04-12 11:49:57 +02:00
|
|
|
params->attributes.fd[i] = -1;
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2021-04-12 11:49:57 +02:00
|
|
|
params->linux_dmabuf = linux_dmabuf;
|
|
|
|
|
|
|
|
uint32_t version = wl_resource_get_version(linux_dmabuf_resource);
|
|
|
|
params->resource = wl_resource_create(client,
|
2018-04-30 23:20:39 +02:00
|
|
|
&zwp_linux_buffer_params_v1_interface, version, params_id);
|
2021-04-12 11:49:57 +02:00
|
|
|
if (!params->resource) {
|
|
|
|
free(params);
|
|
|
|
wl_resource_post_no_memory(linux_dmabuf_resource);
|
|
|
|
return;
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
2021-04-12 11:49:57 +02:00
|
|
|
wl_resource_set_implementation(params->resource,
|
|
|
|
&buffer_params_impl, params, params_handle_resource_destroy);
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2018-11-13 12:35:08 +01:00
|
|
|
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,
|
|
|
|
};
|
|
|
|
|
2021-06-07 18:20:45 +02:00
|
|
|
static ssize_t get_drm_format_set_index(const struct wlr_drm_format_set *set,
|
|
|
|
uint32_t format, uint64_t modifier) {
|
|
|
|
bool format_found = false;
|
|
|
|
const struct wlr_drm_format *fmt;
|
|
|
|
size_t idx = 0;
|
|
|
|
for (size_t i = 0; i < set->len; i++) {
|
2023-05-11 09:53:11 +02:00
|
|
|
fmt = &set->formats[i];
|
2021-06-07 18:20:45 +02:00
|
|
|
if (fmt->format == format) {
|
|
|
|
format_found = true;
|
|
|
|
break;
|
|
|
|
}
|
2022-05-26 10:28:21 +02:00
|
|
|
idx += fmt->len;
|
2021-06-07 18:20:45 +02:00
|
|
|
}
|
|
|
|
if (!format_found) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (size_t i = 0; i < fmt->len; i++) {
|
|
|
|
if (fmt->modifiers[i] == modifier) {
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
idx++;
|
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2018-11-13 12:35:08 +01:00
|
|
|
static struct wlr_linux_dmabuf_feedback_v1_compiled *feedback_compile(
|
|
|
|
const struct wlr_linux_dmabuf_feedback_v1 *feedback) {
|
2022-12-09 17:58:40 +01:00
|
|
|
const struct wlr_linux_dmabuf_feedback_v1_tranche *tranches = feedback->tranches.data;
|
|
|
|
size_t tranches_len = feedback->tranches.size / sizeof(struct wlr_linux_dmabuf_feedback_v1_tranche);
|
|
|
|
assert(tranches_len > 0);
|
2018-11-13 12:35:08 +01:00
|
|
|
|
2023-05-17 18:06:53 +02:00
|
|
|
// Make one big format set that contains all formats across all tranches so that we
|
|
|
|
// can build an index
|
|
|
|
struct wlr_drm_format_set all_formats = {0};
|
|
|
|
for (size_t i = 0; i < tranches_len; i++) {
|
|
|
|
const struct wlr_linux_dmabuf_feedback_v1_tranche *tranche = &tranches[i];
|
|
|
|
if (!wlr_drm_format_set_union(&all_formats, &all_formats, &tranche->formats)) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to union scanout formats into one tranche");
|
2023-05-18 11:16:37 +02:00
|
|
|
goto err_all_formats;
|
2023-05-17 18:06:53 +02:00
|
|
|
}
|
|
|
|
}
|
2018-11-13 12:35:08 +01:00
|
|
|
|
|
|
|
size_t table_len = 0;
|
2023-05-17 18:06:53 +02:00
|
|
|
for (size_t i = 0; i < all_formats.len; i++) {
|
|
|
|
const struct wlr_drm_format *fmt = &all_formats.formats[i];
|
2022-05-26 10:28:21 +02:00
|
|
|
table_len += fmt->len;
|
2018-11-13 12:35:08 +01:00
|
|
|
}
|
|
|
|
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);
|
2023-05-18 11:16:37 +02:00
|
|
|
goto err_all_formats;
|
2018-11-13 12:35:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
close(rw_fd);
|
|
|
|
|
|
|
|
size_t n = 0;
|
2023-05-17 18:06:53 +02:00
|
|
|
for (size_t i = 0; i < all_formats.len; i++) {
|
|
|
|
const struct wlr_drm_format *fmt = &all_formats.formats[i];
|
2018-11-13 12:35:08 +01:00
|
|
|
|
|
|
|
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) +
|
2022-12-09 17:58:40 +01:00
|
|
|
tranches_len * sizeof(struct wlr_linux_dmabuf_feedback_v1_compiled_tranche));
|
2018-11-13 12:35:08 +01:00
|
|
|
if (compiled == NULL) {
|
|
|
|
close(ro_fd);
|
2023-05-18 11:16:37 +02:00
|
|
|
goto err_all_formats;
|
2018-11-13 12:35:08 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
compiled->main_device = feedback->main_device;
|
2022-12-09 17:58:40 +01:00
|
|
|
compiled->tranches_len = tranches_len;
|
2018-11-13 12:35:08 +01:00
|
|
|
compiled->table_fd = ro_fd;
|
|
|
|
compiled->table_size = table_size;
|
|
|
|
|
2023-05-17 18:06:53 +02:00
|
|
|
// Build the indices lists for all tranches
|
|
|
|
for (size_t i = 0; i < tranches_len; i++) {
|
2022-12-09 17:58:40 +01:00
|
|
|
const struct wlr_linux_dmabuf_feedback_v1_tranche *tranche = &tranches[i];
|
2021-06-07 18:20:45 +02:00
|
|
|
struct wlr_linux_dmabuf_feedback_v1_compiled_tranche *compiled_tranche =
|
|
|
|
&compiled->tranches[i];
|
|
|
|
|
|
|
|
compiled_tranche->target_device = tranche->target_device;
|
|
|
|
compiled_tranche->flags = tranche->flags;
|
|
|
|
|
|
|
|
wl_array_init(&compiled_tranche->indices);
|
|
|
|
if (!wl_array_add(&compiled_tranche->indices, table_len * sizeof(uint16_t))) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to allocate tranche indices array");
|
|
|
|
goto error_compiled;
|
|
|
|
}
|
|
|
|
|
|
|
|
n = 0;
|
|
|
|
uint16_t *indices = compiled_tranche->indices.data;
|
2022-12-09 17:58:40 +01:00
|
|
|
for (size_t j = 0; j < tranche->formats.len; j++) {
|
2023-05-11 09:53:11 +02:00
|
|
|
const struct wlr_drm_format *fmt = &tranche->formats.formats[j];
|
2021-06-07 18:20:45 +02:00
|
|
|
for (size_t k = 0; k < fmt->len; k++) {
|
|
|
|
ssize_t index = get_drm_format_set_index(
|
2023-05-17 18:06:53 +02:00
|
|
|
&all_formats, fmt->format, fmt->modifiers[k]);
|
2021-06-07 18:20:45 +02:00
|
|
|
if (index < 0) {
|
|
|
|
wlr_log(WLR_ERROR, "Format 0x%" PRIX32 " and modifier "
|
|
|
|
"0x%" PRIX64 " are in tranche #%zu but are missing "
|
|
|
|
"from the fallback tranche",
|
|
|
|
fmt->format, fmt->modifiers[k], i);
|
|
|
|
goto error_compiled;
|
|
|
|
}
|
|
|
|
indices[n] = index;
|
|
|
|
n++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
compiled_tranche->indices.size = n * sizeof(uint16_t);
|
2018-11-13 12:35:08 +01:00
|
|
|
}
|
|
|
|
|
2023-05-18 11:16:37 +02:00
|
|
|
wlr_drm_format_set_finish(&all_formats);
|
|
|
|
|
2018-11-13 12:35:08 +01:00
|
|
|
return compiled;
|
|
|
|
|
|
|
|
error_compiled:
|
|
|
|
close(compiled->table_fd);
|
|
|
|
free(compiled);
|
2023-05-18 11:16:37 +02:00
|
|
|
err_all_formats:
|
|
|
|
wlr_drm_format_set_finish(&all_formats);
|
2018-11-13 12:35:08 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void compiled_feedback_destroy(
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1_compiled *feedback) {
|
2021-06-07 18:20:45 +02:00
|
|
|
if (feedback == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
2018-11-13 12:35:08 +01:00
|
|
|
for (size_t i = 0; i < feedback->tranches_len; i++) {
|
|
|
|
wl_array_release(&feedback->tranches[i].indices);
|
|
|
|
}
|
|
|
|
close(feedback->table_fd);
|
|
|
|
free(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);
|
|
|
|
}
|
|
|
|
|
2021-06-07 18:20:45 +02:00
|
|
|
static void surface_destroy(struct wlr_linux_dmabuf_v1_surface *surface) {
|
|
|
|
struct wl_resource *resource, *resource_tmp;
|
|
|
|
wl_resource_for_each_safe(resource, resource_tmp, &surface->feedback_resources) {
|
|
|
|
struct wl_list *link = wl_resource_get_link(resource);
|
|
|
|
wl_list_remove(link);
|
|
|
|
wl_list_init(link);
|
|
|
|
}
|
|
|
|
|
|
|
|
compiled_feedback_destroy(surface->feedback);
|
|
|
|
|
|
|
|
wlr_addon_finish(&surface->addon);
|
|
|
|
wl_list_remove(&surface->link);
|
|
|
|
free(surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void surface_addon_destroy(struct wlr_addon *addon) {
|
|
|
|
struct wlr_linux_dmabuf_v1_surface *surface =
|
|
|
|
wl_container_of(addon, surface, addon);
|
|
|
|
surface_destroy(surface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wlr_addon_interface surface_addon_impl = {
|
|
|
|
.name = "wlr_linux_dmabuf_v1_surface",
|
|
|
|
.destroy = surface_addon_destroy,
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct wlr_linux_dmabuf_v1_surface *surface_get_or_create(
|
|
|
|
struct wlr_linux_dmabuf_v1 *linux_dmabuf,
|
|
|
|
struct wlr_surface *wlr_surface) {
|
|
|
|
struct wlr_addon *addon =
|
|
|
|
wlr_addon_find(&wlr_surface->addons, linux_dmabuf, &surface_addon_impl);
|
|
|
|
if (addon != NULL) {
|
|
|
|
struct wlr_linux_dmabuf_v1_surface *surface =
|
|
|
|
wl_container_of(addon, surface, addon);
|
|
|
|
return surface;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct wlr_linux_dmabuf_v1_surface *surface = calloc(1, sizeof(*surface));
|
|
|
|
if (surface == NULL) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
surface->surface = wlr_surface;
|
|
|
|
surface->linux_dmabuf = linux_dmabuf;
|
|
|
|
wl_list_init(&surface->feedback_resources);
|
|
|
|
wlr_addon_init(&surface->addon, &wlr_surface->addons, linux_dmabuf,
|
|
|
|
&surface_addon_impl);
|
|
|
|
wl_list_insert(&linux_dmabuf->surfaces, &surface->link);
|
|
|
|
|
|
|
|
return surface;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wlr_linux_dmabuf_feedback_v1_compiled *surface_get_feedback(
|
|
|
|
struct wlr_linux_dmabuf_v1_surface *surface) {
|
|
|
|
if (surface->feedback != NULL) {
|
|
|
|
return surface->feedback;
|
|
|
|
}
|
|
|
|
return surface->linux_dmabuf->default_feedback;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void surface_feedback_handle_resource_destroy(struct wl_resource *resource) {
|
|
|
|
wl_list_remove(wl_resource_get_link(resource));
|
|
|
|
}
|
|
|
|
|
2018-11-13 12:35:08 +01:00
|
|
|
static void linux_dmabuf_get_surface_feedback(struct wl_client *client,
|
|
|
|
struct wl_resource *resource, uint32_t id,
|
|
|
|
struct wl_resource *surface_resource) {
|
2021-06-07 18:20:45 +02:00
|
|
|
struct wlr_linux_dmabuf_v1 *linux_dmabuf =
|
|
|
|
linux_dmabuf_from_resource(resource);
|
|
|
|
struct wlr_surface *wlr_surface = wlr_surface_from_resource(surface_resource);
|
|
|
|
|
|
|
|
struct wlr_linux_dmabuf_v1_surface *surface =
|
|
|
|
surface_get_or_create(linux_dmabuf, wlr_surface);
|
|
|
|
if (surface == NULL) {
|
|
|
|
wl_client_post_no_memory(client);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
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, surface_feedback_handle_resource_destroy);
|
|
|
|
wl_list_insert(&surface->feedback_resources, wl_resource_get_link(feedback_resource));
|
|
|
|
|
|
|
|
feedback_send(surface_get_feedback(surface), feedback_resource);
|
2018-11-13 12:35:08 +01:00
|
|
|
}
|
|
|
|
|
2018-04-08 17:00:56 +02:00
|
|
|
static void linux_dmabuf_destroy(struct wl_client *client,
|
|
|
|
struct wl_resource *resource) {
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_resource_destroy(resource);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct zwp_linux_dmabuf_v1_interface linux_dmabuf_impl = {
|
2018-04-08 17:00:56 +02:00
|
|
|
.destroy = linux_dmabuf_destroy,
|
|
|
|
.create_params = linux_dmabuf_create_params,
|
2018-11-13 12:35:08 +01:00
|
|
|
.get_default_feedback = linux_dmabuf_get_default_feedback,
|
|
|
|
.get_surface_feedback = linux_dmabuf_get_surface_feedback,
|
2018-02-23 18:45:16 +01:00
|
|
|
};
|
|
|
|
|
2021-03-23 00:06:40 +01:00
|
|
|
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) {
|
2021-03-31 17:07:55 +02:00
|
|
|
if (wlr_drm_format_has(fmt, DRM_FORMAT_MOD_INVALID)) {
|
|
|
|
zwp_linux_dmabuf_v1_send_format(resource, fmt->format);
|
|
|
|
}
|
2021-03-23 00:06:40 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-11-27 19:08:18 +01:00
|
|
|
// 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)
|
2021-11-29 09:40:02 +01:00
|
|
|
&& wlr_drm_format_has(fmt, DRM_FORMAT_MOD_LINEAR)) {
|
2021-11-27 19:08:18 +01:00
|
|
|
uint64_t mod = DRM_FORMAT_MOD_INVALID;
|
|
|
|
zwp_linux_dmabuf_v1_send_modifier(resource, fmt->format,
|
|
|
|
mod >> 32, mod & 0xFFFFFFFF);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-03-23 00:06:40 +01:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-12 22:04:12 +02:00
|
|
|
static void linux_dmabuf_send_formats(struct wlr_linux_dmabuf_v1 *linux_dmabuf,
|
2021-03-23 00:06:40 +01:00
|
|
|
struct wl_resource *resource) {
|
2022-11-30 13:24:46 +01:00
|
|
|
for (size_t i = 0; i < linux_dmabuf->default_formats.len; i++) {
|
2023-05-11 09:53:11 +02:00
|
|
|
const struct wlr_drm_format *fmt = &linux_dmabuf->default_formats.formats[i];
|
2021-03-23 00:06:40 +01:00
|
|
|
linux_dmabuf_send_modifiers(resource, fmt);
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-08 17:00:56 +02:00
|
|
|
static void linux_dmabuf_bind(struct wl_client *client, void *data,
|
|
|
|
uint32_t version, uint32_t id) {
|
2018-07-19 22:50:00 +02:00
|
|
|
struct wlr_linux_dmabuf_v1 *linux_dmabuf = data;
|
2018-02-23 18:45:16 +01:00
|
|
|
|
2018-04-08 17:00:56 +02:00
|
|
|
struct wl_resource *resource = wl_resource_create(client,
|
|
|
|
&zwp_linux_dmabuf_v1_interface, version, id);
|
2018-02-23 18:45:16 +01:00
|
|
|
if (resource == NULL) {
|
|
|
|
wl_client_post_no_memory(client);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
wl_resource_set_implementation(resource, &linux_dmabuf_impl,
|
2019-11-16 18:31:33 +01:00
|
|
|
linux_dmabuf, NULL);
|
2018-11-13 12:35:08 +01:00
|
|
|
|
|
|
|
if (version < ZWP_LINUX_DMABUF_V1_GET_DEFAULT_FEEDBACK_SINCE_VERSION) {
|
|
|
|
linux_dmabuf_send_formats(linux_dmabuf, resource);
|
|
|
|
}
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2022-11-11 15:43:48 +01:00
|
|
|
static struct wlr_buffer *buffer_from_resource(struct wl_resource *resource) {
|
|
|
|
struct wlr_dmabuf_v1_buffer *buffer =
|
|
|
|
wlr_dmabuf_v1_buffer_from_buffer_resource(resource);
|
|
|
|
return &buffer->base;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct wlr_buffer_resource_interface buffer_resource_interface = {
|
|
|
|
.name = "wlr_dmabuf_v1_buffer",
|
|
|
|
.is_instance = wlr_dmabuf_v1_resource_is_buffer,
|
|
|
|
.from_resource = buffer_from_resource,
|
|
|
|
};
|
|
|
|
|
2019-11-16 18:31:33 +01:00
|
|
|
static void linux_dmabuf_v1_destroy(struct wlr_linux_dmabuf_v1 *linux_dmabuf) {
|
2022-08-18 13:16:16 +02:00
|
|
|
wl_signal_emit_mutable(&linux_dmabuf->events.destroy, linux_dmabuf);
|
2018-04-30 23:20:39 +02:00
|
|
|
|
2021-06-07 18:20:45 +02:00
|
|
|
struct wlr_linux_dmabuf_v1_surface *surface, *surface_tmp;
|
|
|
|
wl_list_for_each_safe(surface, surface_tmp, &linux_dmabuf->surfaces, link) {
|
|
|
|
surface_destroy(surface);
|
|
|
|
}
|
|
|
|
|
2018-11-13 12:35:08 +01:00
|
|
|
compiled_feedback_destroy(linux_dmabuf->default_feedback);
|
2022-11-30 13:24:46 +01:00
|
|
|
wlr_drm_format_set_finish(&linux_dmabuf->default_formats);
|
2022-11-30 13:00:09 +01:00
|
|
|
close(linux_dmabuf->main_device_fd);
|
2018-11-13 12:35:08 +01:00
|
|
|
|
2018-02-23 18:45:16 +01:00
|
|
|
wl_list_remove(&linux_dmabuf->display_destroy.link);
|
2018-04-30 23:20:39 +02:00
|
|
|
|
2018-07-08 20:21:31 +02:00
|
|
|
wl_global_destroy(linux_dmabuf->global);
|
2018-02-23 18:45:16 +01:00
|
|
|
free(linux_dmabuf);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void handle_display_destroy(struct wl_listener *listener, void *data) {
|
2018-07-19 22:50:00 +02:00
|
|
|
struct wlr_linux_dmabuf_v1 *linux_dmabuf =
|
2018-04-08 17:00:56 +02:00
|
|
|
wl_container_of(listener, linux_dmabuf, display_destroy);
|
2019-11-16 18:31:33 +01:00
|
|
|
linux_dmabuf_v1_destroy(linux_dmabuf);
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2022-11-30 13:00:09 +01:00
|
|
|
static bool set_default_feedback(struct wlr_linux_dmabuf_v1 *linux_dmabuf,
|
|
|
|
const struct wlr_linux_dmabuf_feedback_v1 *feedback) {
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1_compiled *compiled = feedback_compile(feedback);
|
|
|
|
if (compiled == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
drmDevice *device = NULL;
|
|
|
|
if (drmGetDeviceFromDevId(feedback->main_device, 0, &device) != 0) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "drmGetDeviceFromDevId failed");
|
|
|
|
goto error_compiled;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *name = NULL;
|
|
|
|
if (device->available_nodes & (1 << DRM_NODE_RENDER)) {
|
|
|
|
name = device->nodes[DRM_NODE_RENDER];
|
|
|
|
} else {
|
|
|
|
// Likely a split display/render setup
|
|
|
|
assert(device->available_nodes & (1 << DRM_NODE_PRIMARY));
|
|
|
|
name = device->nodes[DRM_NODE_PRIMARY];
|
|
|
|
wlr_log(WLR_DEBUG, "DRM device %s has no render node, "
|
|
|
|
"falling back to primary node", name);
|
|
|
|
}
|
|
|
|
|
|
|
|
int main_device_fd = open(name, O_RDWR | O_CLOEXEC);
|
|
|
|
drmFreeDevice(&device);
|
|
|
|
if (main_device_fd < 0) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "Failed to open DRM device %s", name);
|
|
|
|
goto error_compiled;
|
|
|
|
}
|
|
|
|
|
2022-12-09 17:58:40 +01:00
|
|
|
size_t tranches_len =
|
|
|
|
feedback->tranches.size / sizeof(struct wlr_linux_dmabuf_feedback_v1_tranche);
|
|
|
|
const struct wlr_linux_dmabuf_feedback_v1_tranche *tranches = feedback->tranches.data;
|
2022-11-30 13:24:46 +01:00
|
|
|
struct wlr_drm_format_set formats = {0};
|
2022-12-09 17:58:40 +01:00
|
|
|
for (size_t i = 0; i < tranches_len; i++) {
|
|
|
|
const struct wlr_linux_dmabuf_feedback_v1_tranche *tranche = &tranches[i];
|
2023-05-03 18:38:28 +02:00
|
|
|
|
2023-05-04 20:16:34 +02:00
|
|
|
if (!wlr_drm_format_set_union(&formats, &formats, &tranche->formats)) {
|
2023-05-03 18:38:28 +02:00
|
|
|
goto error_formats;
|
2022-11-30 13:24:46 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-30 13:00:09 +01:00
|
|
|
compiled_feedback_destroy(linux_dmabuf->default_feedback);
|
|
|
|
linux_dmabuf->default_feedback = compiled;
|
|
|
|
|
|
|
|
if (linux_dmabuf->main_device_fd >= 0) {
|
|
|
|
close(linux_dmabuf->main_device_fd);
|
|
|
|
}
|
|
|
|
linux_dmabuf->main_device_fd = main_device_fd;
|
|
|
|
|
2022-11-30 13:24:46 +01:00
|
|
|
wlr_drm_format_set_finish(&linux_dmabuf->default_formats);
|
|
|
|
linux_dmabuf->default_formats = formats;
|
|
|
|
|
2022-11-30 13:00:09 +01:00
|
|
|
return true;
|
|
|
|
|
2022-11-30 13:24:46 +01:00
|
|
|
error_formats:
|
|
|
|
wlr_drm_format_set_finish(&formats);
|
2022-11-30 13:00:09 +01:00
|
|
|
error_compiled:
|
|
|
|
compiled_feedback_destroy(compiled);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-11-30 13:28:52 +01:00
|
|
|
struct wlr_linux_dmabuf_v1 *wlr_linux_dmabuf_v1_create(struct wl_display *display,
|
|
|
|
uint32_t version, const struct wlr_linux_dmabuf_feedback_v1 *default_feedback) {
|
2022-11-24 10:07:19 +01:00
|
|
|
assert(version <= LINUX_DMABUF_VERSION);
|
|
|
|
|
2018-07-19 22:50:00 +02:00
|
|
|
struct wlr_linux_dmabuf_v1 *linux_dmabuf =
|
|
|
|
calloc(1, sizeof(struct wlr_linux_dmabuf_v1));
|
2018-02-23 18:45:16 +01:00
|
|
|
if (linux_dmabuf == NULL) {
|
2018-07-09 23:49:54 +02:00
|
|
|
wlr_log(WLR_ERROR, "could not create simple dmabuf manager");
|
2018-02-23 18:45:16 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
2022-11-30 13:00:09 +01:00
|
|
|
linux_dmabuf->main_device_fd = -1;
|
2018-02-23 18:45:16 +01:00
|
|
|
|
2021-06-07 18:20:45 +02:00
|
|
|
wl_list_init(&linux_dmabuf->surfaces);
|
2018-04-30 23:20:39 +02:00
|
|
|
wl_signal_init(&linux_dmabuf->events.destroy);
|
|
|
|
|
2022-11-24 10:07:19 +01:00
|
|
|
linux_dmabuf->global = wl_global_create(display, &zwp_linux_dmabuf_v1_interface,
|
|
|
|
version, linux_dmabuf, linux_dmabuf_bind);
|
2018-07-08 20:21:31 +02:00
|
|
|
if (!linux_dmabuf->global) {
|
2018-07-09 23:49:54 +02:00
|
|
|
wlr_log(WLR_ERROR, "could not create linux dmabuf v1 wl global");
|
2022-11-30 13:00:09 +01:00
|
|
|
goto error_linux_dmabuf;
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
|
|
|
|
2022-11-30 13:28:52 +01:00
|
|
|
if (!set_default_feedback(linux_dmabuf, default_feedback)) {
|
2022-11-30 13:00:09 +01:00
|
|
|
goto error_global;
|
2018-11-13 12:35:08 +01:00
|
|
|
}
|
|
|
|
|
2018-04-08 17:00:56 +02:00
|
|
|
linux_dmabuf->display_destroy.notify = handle_display_destroy;
|
|
|
|
wl_display_add_destroy_listener(display, &linux_dmabuf->display_destroy);
|
|
|
|
|
2022-11-11 15:43:48 +01:00
|
|
|
wlr_buffer_register_resource_interface(&buffer_resource_interface);
|
|
|
|
|
2018-02-23 18:45:16 +01:00
|
|
|
return linux_dmabuf;
|
2022-11-30 13:00:09 +01:00
|
|
|
|
|
|
|
error_global:
|
|
|
|
wl_global_destroy(linux_dmabuf->global);
|
|
|
|
error_linux_dmabuf:
|
|
|
|
free(linux_dmabuf);
|
|
|
|
return NULL;
|
2018-02-23 18:45:16 +01:00
|
|
|
}
|
2021-06-07 18:20:45 +02:00
|
|
|
|
2022-11-30 13:28:52 +01:00
|
|
|
struct wlr_linux_dmabuf_v1 *wlr_linux_dmabuf_v1_create_with_renderer(struct wl_display *display,
|
|
|
|
uint32_t version, struct wlr_renderer *renderer) {
|
2022-12-09 18:35:02 +01:00
|
|
|
const struct wlr_linux_dmabuf_feedback_v1_init_options options = {
|
|
|
|
.main_renderer = renderer,
|
|
|
|
};
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1 feedback = {0};
|
|
|
|
if (!wlr_linux_dmabuf_feedback_v1_init_with_options(&feedback, &options)) {
|
2022-11-30 13:28:52 +01:00
|
|
|
return NULL;
|
|
|
|
}
|
2022-12-09 18:35:02 +01:00
|
|
|
struct wlr_linux_dmabuf_v1 *linux_dmabuf =
|
|
|
|
wlr_linux_dmabuf_v1_create(display, version, &feedback);
|
|
|
|
wlr_linux_dmabuf_feedback_v1_finish(&feedback);
|
|
|
|
return linux_dmabuf;
|
2022-11-30 13:28:52 +01:00
|
|
|
}
|
|
|
|
|
2021-06-07 18:20:45 +02:00
|
|
|
bool wlr_linux_dmabuf_v1_set_surface_feedback(
|
|
|
|
struct wlr_linux_dmabuf_v1 *linux_dmabuf,
|
|
|
|
struct wlr_surface *wlr_surface,
|
|
|
|
const struct wlr_linux_dmabuf_feedback_v1 *feedback) {
|
|
|
|
struct wlr_linux_dmabuf_v1_surface *surface =
|
|
|
|
surface_get_or_create(linux_dmabuf, wlr_surface);
|
|
|
|
if (surface == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1_compiled *compiled = NULL;
|
|
|
|
if (feedback != NULL) {
|
|
|
|
compiled = feedback_compile(feedback);
|
|
|
|
if (compiled == NULL) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
compiled_feedback_destroy(surface->feedback);
|
|
|
|
surface->feedback = compiled;
|
|
|
|
|
|
|
|
struct wl_resource *resource;
|
|
|
|
wl_resource_for_each(resource, &surface->feedback_resources) {
|
|
|
|
feedback_send(surface_get_feedback(surface), resource);
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2022-12-09 18:06:26 +01:00
|
|
|
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1_tranche *wlr_linux_dmabuf_feedback_add_tranche(
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1 *feedback) {
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1_tranche *tranche =
|
|
|
|
wl_array_add(&feedback->tranches, sizeof(*tranche));
|
|
|
|
if (tranche == NULL) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "Allocation failed");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
memset(tranche, 0, sizeof(*tranche));
|
|
|
|
return tranche;
|
|
|
|
}
|
|
|
|
|
|
|
|
void wlr_linux_dmabuf_feedback_v1_finish(struct wlr_linux_dmabuf_feedback_v1 *feedback) {
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1_tranche *tranche;
|
|
|
|
wl_array_for_each(tranche, &feedback->tranches) {
|
|
|
|
wlr_drm_format_set_finish(&tranche->formats);
|
|
|
|
}
|
|
|
|
wl_array_release(&feedback->tranches);
|
|
|
|
}
|
2022-12-09 18:30:16 +01:00
|
|
|
|
|
|
|
static bool devid_from_fd(int fd, dev_t *devid) {
|
|
|
|
struct stat stat;
|
|
|
|
if (fstat(fd, &stat) != 0) {
|
|
|
|
wlr_log_errno(WLR_ERROR, "fstat failed");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
*devid = stat.st_rdev;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool wlr_linux_dmabuf_feedback_v1_init_with_options(struct wlr_linux_dmabuf_feedback_v1 *feedback,
|
|
|
|
const struct wlr_linux_dmabuf_feedback_v1_init_options *options) {
|
|
|
|
assert(options->main_renderer != NULL);
|
2023-02-07 10:57:17 +01:00
|
|
|
assert(options->scanout_primary_output == NULL ||
|
|
|
|
options->output_layer_feedback_event == NULL);
|
2022-12-09 18:30:16 +01:00
|
|
|
|
|
|
|
memset(feedback, 0, sizeof(*feedback));
|
|
|
|
|
|
|
|
int renderer_drm_fd = wlr_renderer_get_drm_fd(options->main_renderer);
|
|
|
|
if (renderer_drm_fd < 0) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to get renderer DRM FD");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
dev_t renderer_dev;
|
|
|
|
if (!devid_from_fd(renderer_drm_fd, &renderer_dev)) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
feedback->main_device = renderer_dev;
|
|
|
|
|
|
|
|
const struct wlr_drm_format_set *renderer_formats =
|
|
|
|
wlr_renderer_get_dmabuf_texture_formats(options->main_renderer);
|
|
|
|
if (renderer_formats == NULL) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to get renderer DMA-BUF texture formats");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
2023-02-07 10:57:17 +01:00
|
|
|
if (options->output_layer_feedback_event != NULL) {
|
|
|
|
const struct wlr_output_layer_feedback_event *event = options->output_layer_feedback_event;
|
|
|
|
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1_tranche *tranche =
|
|
|
|
wlr_linux_dmabuf_feedback_add_tranche(feedback);
|
|
|
|
if (tranche == NULL) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
tranche->target_device = event->target_device;
|
|
|
|
tranche->flags = ZWP_LINUX_DMABUF_FEEDBACK_V1_TRANCHE_FLAGS_SCANOUT;
|
|
|
|
if (!wlr_drm_format_set_intersect(&tranche->formats, event->formats, renderer_formats)) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to intersect renderer and scanout formats");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
} else if (options->scanout_primary_output != NULL) {
|
2022-12-09 18:30:16 +01:00
|
|
|
int backend_drm_fd = wlr_backend_get_drm_fd(options->scanout_primary_output->backend);
|
|
|
|
if (backend_drm_fd < 0) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to get backend DRM FD");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
dev_t backend_dev;
|
|
|
|
if (!devid_from_fd(backend_drm_fd, &backend_dev)) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct wlr_drm_format_set *scanout_formats =
|
|
|
|
wlr_output_get_primary_formats(options->scanout_primary_output, WLR_BUFFER_CAP_DMABUF);
|
|
|
|
if (scanout_formats == NULL) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to get output primary DMA-BUF formats");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1_tranche *tranche =
|
|
|
|
wlr_linux_dmabuf_feedback_add_tranche(feedback);
|
|
|
|
if (tranche == NULL) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
tranche->target_device = backend_dev;
|
2023-02-07 11:25:10 +01:00
|
|
|
tranche->flags = ZWP_LINUX_DMABUF_FEEDBACK_V1_TRANCHE_FLAGS_SCANOUT;
|
2022-12-09 18:30:16 +01:00
|
|
|
if (!wlr_drm_format_set_intersect(&tranche->formats, scanout_formats, renderer_formats)) {
|
|
|
|
wlr_log(WLR_ERROR, "Failed to intersect renderer and scanout formats");
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct wlr_linux_dmabuf_feedback_v1_tranche *tranche =
|
|
|
|
wlr_linux_dmabuf_feedback_add_tranche(feedback);
|
|
|
|
if (tranche == NULL) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
tranche->target_device = renderer_dev;
|
|
|
|
if (!wlr_drm_format_set_copy(&tranche->formats, renderer_formats)) {
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
error:
|
|
|
|
wlr_linux_dmabuf_feedback_v1_finish(feedback);
|
|
|
|
return false;
|
|
|
|
}
|