diff options
Diffstat (limited to 'backend')
-rw-r--r-- | backend/drm/drm.c | 91 |
1 files changed, 18 insertions, 73 deletions
diff --git a/backend/drm/drm.c b/backend/drm/drm.c index b7565d28..60e49db6 100644 --- a/backend/drm/drm.c +++ b/backend/drm/drm.c @@ -561,9 +561,6 @@ bool drm_connector_supports_vrr(struct wlr_drm_connector *conn) { return true; } -static bool drm_connector_set_mode(struct wlr_drm_connector *conn, - const struct wlr_drm_connector_state *state); - bool drm_connector_commit_state(struct wlr_drm_connector *conn, const struct wlr_output_state *base) { struct wlr_drm_backend *drm = conn->backend; @@ -594,8 +591,8 @@ bool drm_connector_commit_state(struct wlr_drm_connector *conn, } } - if (pending.modeset) { - if (!drm_connector_set_mode(conn, &pending)) { + if (!pending.active) { + if (conn->crtc != NULL && !drm_crtc_commit(conn, &pending, 0, false)) { return false; } } else if (pending.base->committed & WLR_OUTPUT_STATE_BUFFER) { @@ -611,6 +608,22 @@ bool drm_connector_commit_state(struct wlr_drm_connector *conn, } } + if (pending.base->committed & WLR_OUTPUT_STATE_ENABLED) { + wlr_output_update_enabled(&conn->output, pending.active); + } + if (pending.base->committed & WLR_OUTPUT_STATE_MODE) { + struct wlr_output_mode *mode = NULL; + switch (pending.base->mode_type) { + case WLR_OUTPUT_STATE_MODE_FIXED: + mode = pending.base->mode; + break; + case WLR_OUTPUT_STATE_MODE_CUSTOM: + mode = wlr_drm_connector_add_mode(&conn->output, &pending.mode); + break; + } + wlr_output_update_mode(&conn->output, mode); + } + return true; } @@ -673,74 +686,6 @@ static bool drm_connector_alloc_crtc(struct wlr_drm_connector *conn) { return conn->crtc != NULL; } -static bool drm_connector_set_mode(struct wlr_drm_connector *conn, - const struct wlr_drm_connector_state *state) { - struct wlr_output_mode *wlr_mode = NULL; - if (state->active) { - if (state->base->committed & WLR_OUTPUT_STATE_MODE) { - switch (state->base->mode_type) { - case WLR_OUTPUT_STATE_MODE_FIXED: - wlr_mode = state->base->mode; - break; - case WLR_OUTPUT_STATE_MODE_CUSTOM: - wlr_mode = wlr_drm_connector_add_mode(&conn->output, &state->mode); - if (wlr_mode == NULL) { - return false; - } - break; - } - } else { - wlr_mode = conn->output.current_mode; - } - } - - if (wlr_mode == NULL) { - if (conn->crtc != NULL) { - if (!drm_crtc_commit(conn, state, 0, false)) { - return false; - } - } - wlr_output_update_enabled(&conn->output, false); - return true; - } - - if (conn->status != DRM_MODE_CONNECTED) { - wlr_drm_conn_log(conn, WLR_ERROR, - "Cannot modeset a disconnected output"); - return false; - } - - if (!drm_connector_alloc_crtc(conn)) { - wlr_drm_conn_log(conn, WLR_ERROR, - "Cannot perform modeset: no CRTC for this connector"); - return false; - } - - wlr_drm_conn_log(conn, WLR_INFO, - "Modesetting with '%" PRId32 "x%" PRId32 "@%" PRId32 "mHz'", - wlr_mode->width, wlr_mode->height, wlr_mode->refresh); - - // drm_crtc_page_flip expects a FB to be available - struct wlr_drm_plane *plane = conn->crtc->primary; - if (!plane_get_next_fb(plane)) { - wlr_drm_conn_log(conn, WLR_ERROR, "Missing FB in modeset"); - return false; - } - - if (!drm_crtc_page_flip(conn, state)) { - return false; - } - - wlr_output_update_mode(&conn->output, wlr_mode); - wlr_output_update_enabled(&conn->output, true); - - // When switching VTs, the mode is not updated but the buffers become - // invalid, so we need to manually damage the output here - wlr_output_damage_whole(&conn->output); - - return true; -} - static struct wlr_drm_mode *drm_mode_create(const drmModeModeInfo *modeinfo) { struct wlr_drm_mode *mode = calloc(1, sizeof(*mode)); if (!mode) { |