Commit b601c16b authored by Maxime Ripard's avatar Maxime Ripard

drm/vc4: crtc: Lookup the encoder from the register at boot

At boot, we can't rely on the vc4_get_crtc_encoder since we don't have a
state yet and thus will not be able to figure out which connector is
attached to our CRTC.

However, we have a muxing bit in the CRTC register we can use to get the
encoder currently connected to the pixelvalve. We can thus read that
register, lookup the associated register through the vc4_pv_data
structure, and then pass it to vc4_crtc_disable so that we can perform
the proper operations.

Fixes: 875a4d53 ("drm/vc4: drv: Disable the CRTC at boot time")
Signed-off-by: default avatarMaxime Ripard <maxime@cerno.tech>
Reviewed-by: default avatarDave Stevenson <dave.stevenson@raspberrypi.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20210507150515.257424-6-maxime@cerno.tech
parent 5a184d95
...@@ -430,11 +430,10 @@ static void require_hvs_enabled(struct drm_device *dev) ...@@ -430,11 +430,10 @@ static void require_hvs_enabled(struct drm_device *dev)
} }
static int vc4_crtc_disable(struct drm_crtc *crtc, static int vc4_crtc_disable(struct drm_crtc *crtc,
struct drm_encoder *encoder,
struct drm_atomic_state *state, struct drm_atomic_state *state,
unsigned int channel) unsigned int channel)
{ {
struct drm_encoder *encoder = vc4_get_crtc_encoder(crtc, state,
drm_atomic_get_old_connector_state);
struct vc4_encoder *vc4_encoder = to_vc4_encoder(encoder); struct vc4_encoder *vc4_encoder = to_vc4_encoder(encoder);
struct vc4_crtc *vc4_crtc = to_vc4_crtc(crtc); struct vc4_crtc *vc4_crtc = to_vc4_crtc(crtc);
struct drm_device *dev = crtc->dev; struct drm_device *dev = crtc->dev;
...@@ -475,10 +474,29 @@ static int vc4_crtc_disable(struct drm_crtc *crtc, ...@@ -475,10 +474,29 @@ static int vc4_crtc_disable(struct drm_crtc *crtc,
return 0; return 0;
} }
static struct drm_encoder *vc4_crtc_get_encoder_by_type(struct drm_crtc *crtc,
enum vc4_encoder_type type)
{
struct drm_encoder *encoder;
drm_for_each_encoder(encoder, crtc->dev) {
struct vc4_encoder *vc4_encoder = to_vc4_encoder(encoder);
if (vc4_encoder->type == type)
return encoder;
}
return NULL;
}
int vc4_crtc_disable_at_boot(struct drm_crtc *crtc) int vc4_crtc_disable_at_boot(struct drm_crtc *crtc)
{ {
struct drm_device *drm = crtc->dev; struct drm_device *drm = crtc->dev;
struct vc4_crtc *vc4_crtc = to_vc4_crtc(crtc); struct vc4_crtc *vc4_crtc = to_vc4_crtc(crtc);
enum vc4_encoder_type encoder_type;
const struct vc4_pv_data *pv_data;
struct drm_encoder *encoder;
unsigned encoder_sel;
int channel; int channel;
if (!(of_device_is_compatible(vc4_crtc->pdev->dev.of_node, if (!(of_device_is_compatible(vc4_crtc->pdev->dev.of_node,
...@@ -497,7 +515,17 @@ int vc4_crtc_disable_at_boot(struct drm_crtc *crtc) ...@@ -497,7 +515,17 @@ int vc4_crtc_disable_at_boot(struct drm_crtc *crtc)
if (channel < 0) if (channel < 0)
return 0; return 0;
return vc4_crtc_disable(crtc, NULL, channel); encoder_sel = VC4_GET_FIELD(CRTC_READ(PV_CONTROL), PV_CONTROL_CLK_SELECT);
if (WARN_ON(encoder_sel != 0))
return 0;
pv_data = vc4_crtc_to_vc4_pv_data(vc4_crtc);
encoder_type = pv_data->encoder_types[encoder_sel];
encoder = vc4_crtc_get_encoder_by_type(crtc, encoder_type);
if (WARN_ON(!encoder))
return 0;
return vc4_crtc_disable(crtc, encoder, NULL, channel);
} }
static void vc4_crtc_atomic_disable(struct drm_crtc *crtc, static void vc4_crtc_atomic_disable(struct drm_crtc *crtc,
...@@ -506,6 +534,8 @@ static void vc4_crtc_atomic_disable(struct drm_crtc *crtc, ...@@ -506,6 +534,8 @@ static void vc4_crtc_atomic_disable(struct drm_crtc *crtc,
struct drm_crtc_state *old_state = drm_atomic_get_old_crtc_state(state, struct drm_crtc_state *old_state = drm_atomic_get_old_crtc_state(state,
crtc); crtc);
struct vc4_crtc_state *old_vc4_state = to_vc4_crtc_state(old_state); struct vc4_crtc_state *old_vc4_state = to_vc4_crtc_state(old_state);
struct drm_encoder *encoder = vc4_get_crtc_encoder(crtc, state,
drm_atomic_get_old_connector_state);
struct drm_device *dev = crtc->dev; struct drm_device *dev = crtc->dev;
require_hvs_enabled(dev); require_hvs_enabled(dev);
...@@ -513,7 +543,7 @@ static void vc4_crtc_atomic_disable(struct drm_crtc *crtc, ...@@ -513,7 +543,7 @@ static void vc4_crtc_atomic_disable(struct drm_crtc *crtc,
/* Disable vblank irq handling before crtc is disabled. */ /* Disable vblank irq handling before crtc is disabled. */
drm_crtc_vblank_off(crtc); drm_crtc_vblank_off(crtc);
vc4_crtc_disable(crtc, state, old_vc4_state->assigned_channel); vc4_crtc_disable(crtc, encoder, state, old_vc4_state->assigned_channel);
/* /*
* Make sure we issue a vblank event after disabling the CRTC if * Make sure we issue a vblank event after disabling the CRTC if
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment