mirror of
https://gitlab.freedesktop.org/wlroots/wlroots.git
synced 2024-11-25 00:22:25 +00:00
c87ab6465d
Enable scene-tree direct scanout of a single buffer with various options for scaling and source crop. This is intended to support direct scanout for fullscreen video with/without scaling, letterboxing/pillarboxing (e.g. 4:3 content on a 16:9 display), and source crop (e.g. when 1920x1088 planes are used for 1920x1080 video). This works by explicitly specifying the source crop and destination box for the primary buffer in the output state. DRM atomic and libliftoff backends will turn this into a crop and scale of the plane (assuming the hardware supports that). For the Wayland/X11/DRM-legacy backends I just reject this so scanout will be disabled. The previous behaviour is preserved if buffer_src_box and buffer_dst_box are unset: the buffer is displayed at native size at the top-left of the output with no crop. The change to `struct wlr_output_state` makes this a binary breaking change (but this works transparently for scene-tree compositors like labwc after a recompile).
290 lines
8.3 KiB
C
290 lines
8.3 KiB
C
#include <assert.h>
|
|
#include <stdlib.h>
|
|
#include <wlr/util/log.h>
|
|
#include <xf86drm.h>
|
|
#include <xf86drmMode.h>
|
|
#include "backend/drm/drm.h"
|
|
#include "backend/drm/fb.h"
|
|
#include "backend/drm/iface.h"
|
|
#include "backend/drm/util.h"
|
|
#include "types/wlr_output.h"
|
|
|
|
static bool legacy_fb_props_match(struct wlr_drm_fb *fb1,
|
|
struct wlr_drm_fb *fb2) {
|
|
struct wlr_dmabuf_attributes dmabuf1 = {0}, dmabuf2 = {0};
|
|
if (!wlr_buffer_get_dmabuf(fb1->wlr_buf, &dmabuf1) ||
|
|
!wlr_buffer_get_dmabuf(fb2->wlr_buf, &dmabuf2)) {
|
|
return false;
|
|
}
|
|
|
|
if (dmabuf1.width != dmabuf2.width ||
|
|
dmabuf1.height != dmabuf2.height ||
|
|
dmabuf1.format != dmabuf2.format ||
|
|
dmabuf1.modifier != dmabuf2.modifier ||
|
|
dmabuf1.n_planes != dmabuf2.n_planes) {
|
|
return false;
|
|
}
|
|
|
|
for (int i = 0; i < dmabuf1.n_planes; i++) {
|
|
if (dmabuf1.stride[i] != dmabuf2.stride[i] ||
|
|
dmabuf1.offset[i] != dmabuf2.offset[i]) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool legacy_crtc_test(const struct wlr_drm_connector_state *state,
|
|
bool modeset) {
|
|
struct wlr_drm_connector *conn = state->connector;
|
|
struct wlr_drm_crtc *crtc = conn->crtc;
|
|
|
|
if (state->base->committed & WLR_OUTPUT_STATE_BUFFER) {
|
|
// If the size doesn't match, reject buffer (scaling is not supported)
|
|
int pending_width, pending_height;
|
|
output_pending_resolution(&state->connector->output, state->base,
|
|
&pending_width, &pending_height);
|
|
if (state->base->buffer->width != pending_width ||
|
|
state->base->buffer->height != pending_height) {
|
|
wlr_log(WLR_DEBUG, "Primary buffer size mismatch");
|
|
return false;
|
|
}
|
|
// Source crop is also not supported
|
|
struct wlr_fbox src_box;
|
|
output_state_get_buffer_src_box(state->base, &src_box);
|
|
if (src_box.x != 0.0 || src_box.y != 0.0 ||
|
|
src_box.width != (double)state->base->buffer->width ||
|
|
src_box.height != (double)state->base->buffer->height) {
|
|
wlr_log(WLR_DEBUG, "Source crop not supported in DRM-legacy output");
|
|
return false;
|
|
}
|
|
|
|
if (!modeset) {
|
|
struct wlr_drm_fb *pending_fb = state->primary_fb;
|
|
|
|
struct wlr_drm_fb *prev_fb = crtc->primary->queued_fb;
|
|
if (!prev_fb) {
|
|
prev_fb = crtc->primary->current_fb;
|
|
}
|
|
|
|
/* Legacy is only guaranteed to be able to display a FB if it's been
|
|
* allocated the same way as the previous one. */
|
|
if (prev_fb != NULL && !legacy_fb_props_match(prev_fb, pending_fb)) {
|
|
wlr_drm_conn_log(conn, WLR_DEBUG,
|
|
"Cannot change scan-out buffer parameters with legacy KMS API");
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool legacy_crtc_commit(const struct wlr_drm_connector_state *state,
|
|
struct wlr_drm_page_flip *page_flip, uint32_t flags, bool modeset) {
|
|
struct wlr_drm_connector *conn = state->connector;
|
|
struct wlr_drm_backend *drm = conn->backend;
|
|
struct wlr_output *output = &conn->output;
|
|
struct wlr_drm_crtc *crtc = conn->crtc;
|
|
struct wlr_drm_plane *cursor = crtc->cursor;
|
|
|
|
uint32_t fb_id = 0;
|
|
if (state->active) {
|
|
if (state->primary_fb == NULL) {
|
|
wlr_log(WLR_ERROR, "%s: failed to acquire primary FB",
|
|
conn->output.name);
|
|
return false;
|
|
}
|
|
fb_id = state->primary_fb->id;
|
|
}
|
|
|
|
if (modeset) {
|
|
uint32_t *conns = NULL;
|
|
size_t conns_len = 0;
|
|
drmModeModeInfo *mode = NULL;
|
|
if (state->active) {
|
|
conns = &conn->id;
|
|
conns_len = 1;
|
|
mode = (drmModeModeInfo *)&state->mode;
|
|
}
|
|
|
|
uint32_t dpms = state->active ? DRM_MODE_DPMS_ON : DRM_MODE_DPMS_OFF;
|
|
if (drmModeConnectorSetProperty(drm->fd, conn->id, conn->props.dpms,
|
|
dpms) != 0) {
|
|
wlr_drm_conn_log_errno(conn, WLR_ERROR,
|
|
"Failed to set DPMS property");
|
|
return false;
|
|
}
|
|
|
|
if (drmModeSetCrtc(drm->fd, crtc->id, fb_id, 0, 0,
|
|
conns, conns_len, mode)) {
|
|
wlr_drm_conn_log_errno(conn, WLR_ERROR, "Failed to set CRTC");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (state->base->committed & WLR_OUTPUT_STATE_GAMMA_LUT) {
|
|
if (!drm_legacy_crtc_set_gamma(drm, crtc,
|
|
state->base->gamma_lut_size, state->base->gamma_lut)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (state->base->committed & WLR_OUTPUT_STATE_ADAPTIVE_SYNC_ENABLED) {
|
|
if (state->base->adaptive_sync_enabled && !output->adaptive_sync_supported) {
|
|
return false;
|
|
}
|
|
if (crtc->props.vrr_enabled != 0 &&
|
|
drmModeObjectSetProperty(drm->fd, crtc->id, DRM_MODE_OBJECT_CRTC,
|
|
crtc->props.vrr_enabled,
|
|
state->base->adaptive_sync_enabled) != 0) {
|
|
wlr_drm_conn_log_errno(conn, WLR_ERROR,
|
|
"drmModeObjectSetProperty(VRR_ENABLED) failed");
|
|
return false;
|
|
}
|
|
output->adaptive_sync_status = state->base->adaptive_sync_enabled ?
|
|
WLR_OUTPUT_ADAPTIVE_SYNC_ENABLED :
|
|
WLR_OUTPUT_ADAPTIVE_SYNC_DISABLED;
|
|
wlr_drm_conn_log(conn, WLR_DEBUG, "VRR %s",
|
|
state->base->adaptive_sync_enabled ? "enabled" : "disabled");
|
|
}
|
|
|
|
if (cursor != NULL && drm_connector_is_cursor_visible(conn)) {
|
|
struct wlr_drm_fb *cursor_fb = state->cursor_fb;
|
|
if (cursor_fb == NULL) {
|
|
wlr_drm_conn_log(conn, WLR_DEBUG, "Failed to acquire cursor FB");
|
|
return false;
|
|
}
|
|
|
|
drmModeFB *drm_fb = drmModeGetFB(drm->fd, cursor_fb->id);
|
|
if (drm_fb == NULL) {
|
|
wlr_drm_conn_log_errno(conn, WLR_DEBUG, "Failed to get cursor "
|
|
"BO handle: drmModeGetFB failed");
|
|
return false;
|
|
}
|
|
uint32_t cursor_handle = drm_fb->handle;
|
|
uint32_t cursor_width = drm_fb->width;
|
|
uint32_t cursor_height = drm_fb->height;
|
|
drmModeFreeFB(drm_fb);
|
|
|
|
int ret = drmModeSetCursor(drm->fd, crtc->id, cursor_handle,
|
|
cursor_width, cursor_height);
|
|
int set_cursor_errno = errno;
|
|
if (drmCloseBufferHandle(drm->fd, cursor_handle) != 0) {
|
|
wlr_log_errno(WLR_ERROR, "drmCloseBufferHandle failed");
|
|
}
|
|
if (ret != 0) {
|
|
wlr_drm_conn_log(conn, WLR_DEBUG, "drmModeSetCursor failed: %s",
|
|
strerror(set_cursor_errno));
|
|
return false;
|
|
}
|
|
|
|
if (drmModeMoveCursor(drm->fd,
|
|
crtc->id, conn->cursor_x, conn->cursor_y) != 0) {
|
|
wlr_drm_conn_log_errno(conn, WLR_ERROR, "drmModeMoveCursor failed");
|
|
return false;
|
|
}
|
|
} else {
|
|
if (drmModeSetCursor(drm->fd, crtc->id, 0, 0, 0)) {
|
|
wlr_drm_conn_log_errno(conn, WLR_DEBUG, "drmModeSetCursor failed");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (flags & DRM_MODE_PAGE_FLIP_EVENT) {
|
|
if (drmModePageFlip(drm->fd, crtc->id, fb_id, flags, page_flip)) {
|
|
wlr_drm_conn_log_errno(conn, WLR_ERROR, "drmModePageFlip failed");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool legacy_commit(struct wlr_drm_backend *drm,
|
|
const struct wlr_drm_device_state *state,
|
|
struct wlr_drm_page_flip *page_flip, uint32_t flags,
|
|
bool test_only) {
|
|
for (size_t i = 0; i < state->connectors_len; i++) {
|
|
const struct wlr_drm_connector_state *conn_state = &state->connectors[i];
|
|
if (!legacy_crtc_test(conn_state, state->modeset)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (test_only) {
|
|
return true;
|
|
}
|
|
|
|
for (size_t i = 0; i < state->connectors_len; i++) {
|
|
const struct wlr_drm_connector_state *conn_state = &state->connectors[i];
|
|
if (!legacy_crtc_commit(conn_state, page_flip, flags,
|
|
state->modeset)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
static void fill_empty_gamma_table(size_t size,
|
|
uint16_t *r, uint16_t *g, uint16_t *b) {
|
|
assert(0xFFFF < UINT64_MAX / (size - 1));
|
|
for (uint32_t i = 0; i < size; ++i) {
|
|
uint16_t val = (uint64_t)0xFFFF * i / (size - 1);
|
|
r[i] = g[i] = b[i] = val;
|
|
}
|
|
}
|
|
|
|
bool drm_legacy_crtc_set_gamma(struct wlr_drm_backend *drm,
|
|
struct wlr_drm_crtc *crtc, size_t size, uint16_t *lut) {
|
|
uint16_t *linear_lut = NULL;
|
|
if (size == 0) {
|
|
// The legacy interface doesn't offer a way to reset the gamma LUT
|
|
size = drm_crtc_get_gamma_lut_size(drm, crtc);
|
|
if (size == 0) {
|
|
return false;
|
|
}
|
|
|
|
linear_lut = malloc(3 * size * sizeof(uint16_t));
|
|
if (linear_lut == NULL) {
|
|
wlr_log_errno(WLR_ERROR, "Allocation failed");
|
|
return false;
|
|
}
|
|
fill_empty_gamma_table(size, linear_lut, linear_lut + size,
|
|
linear_lut + 2 * size);
|
|
|
|
lut = linear_lut;
|
|
}
|
|
|
|
uint16_t *r = lut, *g = lut + size, *b = lut + 2 * size;
|
|
if (drmModeCrtcSetGamma(drm->fd, crtc->id, size, r, g, b) != 0) {
|
|
wlr_log_errno(WLR_ERROR, "Failed to set gamma LUT on CRTC %"PRIu32,
|
|
crtc->id);
|
|
free(linear_lut);
|
|
return false;
|
|
}
|
|
|
|
free(linear_lut);
|
|
return true;
|
|
}
|
|
|
|
static bool legacy_reset(struct wlr_drm_backend *drm) {
|
|
bool ok = true;
|
|
for (size_t i = 0; i < drm->num_crtcs; i++) {
|
|
struct wlr_drm_crtc *crtc = &drm->crtcs[i];
|
|
if (drmModeSetCrtc(drm->fd, crtc->id, 0, 0, 0, NULL, 0, NULL) != 0) {
|
|
wlr_log_errno(WLR_ERROR, "Failed to disable CRTC %"PRIu32,
|
|
crtc->id);
|
|
ok = false;
|
|
}
|
|
}
|
|
return ok;
|
|
}
|
|
|
|
const struct wlr_drm_interface legacy_iface = {
|
|
.commit = legacy_commit,
|
|
.reset = legacy_reset,
|
|
};
|