226 lines
5.5 KiB
C
226 lines
5.5 KiB
C
// SPDX-License-Identifier: MIT
|
|
/*
|
|
* Copyright © 2023 Intel Corporation
|
|
*/
|
|
|
|
#include <drm/drm_atomic.h>
|
|
#include <drm/drm_atomic_helper.h>
|
|
#include <drm/drm_atomic_uapi.h>
|
|
|
|
#include "i915_drv.h"
|
|
#include "intel_atomic.h"
|
|
#include "intel_crtc.h"
|
|
#include "intel_display_types.h"
|
|
#include "intel_load_detect.h"
|
|
|
|
/* VESA 640x480x72Hz mode to set on the pipe */
|
|
static const struct drm_display_mode load_detect_mode = {
|
|
DRM_MODE("640x480", DRM_MODE_TYPE_DEFAULT, 31500, 640, 664,
|
|
704, 832, 0, 480, 489, 491, 520, 0, DRM_MODE_FLAG_NHSYNC | DRM_MODE_FLAG_NVSYNC),
|
|
};
|
|
|
|
static int intel_modeset_disable_planes(struct drm_atomic_state *state,
|
|
struct drm_crtc *crtc)
|
|
{
|
|
struct drm_plane *plane;
|
|
struct drm_plane_state *plane_state;
|
|
int ret, i;
|
|
|
|
ret = drm_atomic_add_affected_planes(state, crtc);
|
|
if (ret)
|
|
return ret;
|
|
|
|
for_each_new_plane_in_state(state, plane, plane_state, i) {
|
|
if (plane_state->crtc != crtc)
|
|
continue;
|
|
|
|
ret = drm_atomic_set_crtc_for_plane(plane_state, NULL);
|
|
if (ret)
|
|
return ret;
|
|
|
|
drm_atomic_set_fb_for_plane(plane_state, NULL);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
struct drm_atomic_state *
|
|
intel_load_detect_get_pipe(struct drm_connector *connector,
|
|
struct drm_modeset_acquire_ctx *ctx)
|
|
{
|
|
struct intel_encoder *encoder =
|
|
intel_attached_encoder(to_intel_connector(connector));
|
|
struct intel_crtc *possible_crtc;
|
|
struct intel_crtc *crtc = NULL;
|
|
struct drm_device *dev = encoder->base.dev;
|
|
struct drm_i915_private *dev_priv = to_i915(dev);
|
|
struct drm_mode_config *config = &dev->mode_config;
|
|
struct drm_atomic_state *state = NULL, *restore_state = NULL;
|
|
struct drm_connector_state *connector_state;
|
|
struct intel_crtc_state *crtc_state;
|
|
int ret;
|
|
|
|
drm_dbg_kms(&dev_priv->drm, "[CONNECTOR:%d:%s], [ENCODER:%d:%s]\n",
|
|
connector->base.id, connector->name,
|
|
encoder->base.base.id, encoder->base.name);
|
|
|
|
drm_WARN_ON(dev, !drm_modeset_is_locked(&config->connection_mutex));
|
|
|
|
/*
|
|
* Algorithm gets a little messy:
|
|
*
|
|
* - if the connector already has an assigned crtc, use it (but make
|
|
* sure it's on first)
|
|
*
|
|
* - try to find the first unused crtc that can drive this connector,
|
|
* and use that if we find one
|
|
*/
|
|
|
|
/* See if we already have a CRTC for this connector */
|
|
if (connector->state->crtc) {
|
|
crtc = to_intel_crtc(connector->state->crtc);
|
|
|
|
ret = drm_modeset_lock(&crtc->base.mutex, ctx);
|
|
if (ret)
|
|
goto fail;
|
|
|
|
/* Make sure the crtc and connector are running */
|
|
goto found;
|
|
}
|
|
|
|
/* Find an unused one (if possible) */
|
|
for_each_intel_crtc(dev, possible_crtc) {
|
|
if (!(encoder->base.possible_crtcs &
|
|
drm_crtc_mask(&possible_crtc->base)))
|
|
continue;
|
|
|
|
ret = drm_modeset_lock(&possible_crtc->base.mutex, ctx);
|
|
if (ret)
|
|
goto fail;
|
|
|
|
if (possible_crtc->base.state->enable) {
|
|
drm_modeset_unlock(&possible_crtc->base.mutex);
|
|
continue;
|
|
}
|
|
|
|
crtc = possible_crtc;
|
|
break;
|
|
}
|
|
|
|
/*
|
|
* If we didn't find an unused CRTC, don't use any.
|
|
*/
|
|
if (!crtc) {
|
|
drm_dbg_kms(&dev_priv->drm,
|
|
"no pipe available for load-detect\n");
|
|
ret = -ENODEV;
|
|
goto fail;
|
|
}
|
|
|
|
found:
|
|
state = drm_atomic_state_alloc(dev);
|
|
restore_state = drm_atomic_state_alloc(dev);
|
|
if (!state || !restore_state) {
|
|
ret = -ENOMEM;
|
|
goto fail;
|
|
}
|
|
|
|
state->acquire_ctx = ctx;
|
|
to_intel_atomic_state(state)->internal = true;
|
|
|
|
restore_state->acquire_ctx = ctx;
|
|
to_intel_atomic_state(restore_state)->internal = true;
|
|
|
|
connector_state = drm_atomic_get_connector_state(state, connector);
|
|
if (IS_ERR(connector_state)) {
|
|
ret = PTR_ERR(connector_state);
|
|
goto fail;
|
|
}
|
|
|
|
ret = drm_atomic_set_crtc_for_connector(connector_state, &crtc->base);
|
|
if (ret)
|
|
goto fail;
|
|
|
|
crtc_state = intel_atomic_get_crtc_state(state, crtc);
|
|
if (IS_ERR(crtc_state)) {
|
|
ret = PTR_ERR(crtc_state);
|
|
goto fail;
|
|
}
|
|
|
|
crtc_state->uapi.active = true;
|
|
|
|
ret = drm_atomic_set_mode_for_crtc(&crtc_state->uapi,
|
|
&load_detect_mode);
|
|
if (ret)
|
|
goto fail;
|
|
|
|
ret = intel_modeset_disable_planes(state, &crtc->base);
|
|
if (ret)
|
|
goto fail;
|
|
|
|
ret = PTR_ERR_OR_ZERO(drm_atomic_get_connector_state(restore_state, connector));
|
|
if (!ret)
|
|
ret = PTR_ERR_OR_ZERO(drm_atomic_get_crtc_state(restore_state, &crtc->base));
|
|
if (!ret)
|
|
ret = drm_atomic_add_affected_planes(restore_state, &crtc->base);
|
|
if (ret) {
|
|
drm_dbg_kms(&dev_priv->drm,
|
|
"Failed to create a copy of old state to restore: %i\n",
|
|
ret);
|
|
goto fail;
|
|
}
|
|
|
|
ret = drm_atomic_commit(state);
|
|
if (ret) {
|
|
drm_dbg_kms(&dev_priv->drm,
|
|
"failed to set mode on load-detect pipe\n");
|
|
goto fail;
|
|
}
|
|
|
|
drm_atomic_state_put(state);
|
|
|
|
/* let the connector get through one full cycle before testing */
|
|
intel_crtc_wait_for_next_vblank(crtc);
|
|
|
|
return restore_state;
|
|
|
|
fail:
|
|
if (state) {
|
|
drm_atomic_state_put(state);
|
|
state = NULL;
|
|
}
|
|
if (restore_state) {
|
|
drm_atomic_state_put(restore_state);
|
|
restore_state = NULL;
|
|
}
|
|
|
|
if (ret == -EDEADLK)
|
|
return ERR_PTR(ret);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void intel_load_detect_release_pipe(struct drm_connector *connector,
|
|
struct drm_atomic_state *state,
|
|
struct drm_modeset_acquire_ctx *ctx)
|
|
{
|
|
struct intel_encoder *intel_encoder =
|
|
intel_attached_encoder(to_intel_connector(connector));
|
|
struct drm_i915_private *i915 = to_i915(intel_encoder->base.dev);
|
|
struct drm_encoder *encoder = &intel_encoder->base;
|
|
int ret;
|
|
|
|
drm_dbg_kms(&i915->drm, "[CONNECTOR:%d:%s], [ENCODER:%d:%s]\n",
|
|
connector->base.id, connector->name,
|
|
encoder->base.id, encoder->name);
|
|
|
|
if (IS_ERR_OR_NULL(state))
|
|
return;
|
|
|
|
ret = drm_atomic_helper_commit_duplicated_state(state, ctx);
|
|
if (ret)
|
|
drm_dbg_kms(&i915->drm,
|
|
"Couldn't release load detect pipe: %i\n", ret);
|
|
drm_atomic_state_put(state);
|
|
}
|