[Intel-gfx] [PATCH v2] drm/i915: hsw: fix link training for eDP on port-A
Imre Deak
imre.deak at intel.com
Thu May 2 20:56:28 CEST 2013
On Thu, 2013-05-02 at 14:45 -0300, Paulo Zanoni wrote:
> Hi
>
> 2013/4/29 Imre Deak <imre.deak at intel.com>:
> > According to BSpec the link training sequence for eDP on HSW port-A
> > should be as follows:
> >
> > 1. link training: clock recovery
> > 2. link training: equalization
> > 3. link training: set idle transmission mode
> > 4. display pipe enable
> > 5. link training: disable (set normal mode)
> >
> > Contrary to this at the moment we don't do step 3. and we do step 5.
> > before step 4. Fix this by setting idle transmission mode for eDP at
> > the end of intel_dp_complete_link_train and adding a new
> > intel_dp_stop_link_training function to disable link training. With
> > these changes we'll end up with the following functions corresponding
> > to the above steps:
> >
> > intel_dp_start_link_train -> step 1.
> > intel_dp_complete_link_train -> step 2., step 3.
> > intel_dp_stop_link_train -> step 5.
> >
> > For port-A we'll call intel_dp_stop_link_train only after enabling the
> > pipe, for everything else we'll call it right after
> > intel_dp_complete_link_train to preserve the current behavior.
> >
> > Tested on HSW/HSW-ULT.
> >
> > In v2:
> > - Due to a HW issue we must set idle transmission mode for port-A too
> > before enabling the pipe. Thanks for Arthur Runyan for explaining
> > this.
> > - Update the patch subject to make it clear that it's an eDP fix, DP is
> > not affected.
> >
> > Signed-off-by: Imre Deak <imre.deak at intel.com>
> > ---
> > drivers/gpu/drm/i915/intel_ddi.c | 5 ++++
> > drivers/gpu/drm/i915/intel_dp.c | 59 ++++++++++++++++++++++++++++----------
> > drivers/gpu/drm/i915/intel_drv.h | 1 +
> > 3 files changed, 50 insertions(+), 15 deletions(-)
> >
> > diff --git a/drivers/gpu/drm/i915/intel_ddi.c b/drivers/gpu/drm/i915/intel_ddi.c
> > index 0d7cf31..f3d0f21 100644
> > --- a/drivers/gpu/drm/i915/intel_ddi.c
> > +++ b/drivers/gpu/drm/i915/intel_ddi.c
> > @@ -1265,6 +1265,8 @@ static void intel_ddi_pre_enable(struct intel_encoder *intel_encoder)
> > intel_dp_sink_dpms(intel_dp, DRM_MODE_DPMS_ON);
> > intel_dp_start_link_train(intel_dp);
> > intel_dp_complete_link_train(intel_dp);
> > + if (port != PORT_A)
> > + intel_dp_stop_link_train(intel_dp);
> > }
> > }
> >
> > @@ -1326,6 +1328,9 @@ static void intel_enable_ddi(struct intel_encoder *intel_encoder)
> > } else if (type == INTEL_OUTPUT_EDP) {
> > struct intel_dp *intel_dp = enc_to_intel_dp(encoder);
> >
> > + if (port == PORT_A)
> > + intel_dp_stop_link_train(intel_dp);
> > +
> > ironlake_edp_backlight_on(intel_dp);
> > }
> >
> > diff --git a/drivers/gpu/drm/i915/intel_dp.c b/drivers/gpu/drm/i915/intel_dp.c
> > index 13d2fd6..d4ddeb9 100644
> > --- a/drivers/gpu/drm/i915/intel_dp.c
> > +++ b/drivers/gpu/drm/i915/intel_dp.c
> > @@ -1437,6 +1437,7 @@ static void intel_enable_dp(struct intel_encoder *encoder)
> > ironlake_edp_panel_on(intel_dp);
> > ironlake_edp_panel_vdd_off(intel_dp, true);
> > intel_dp_complete_link_train(intel_dp);
> > + intel_dp_stop_link_train(intel_dp);
> > ironlake_edp_backlight_on(intel_dp);
> >
> > if (IS_VALLEYVIEW(dev)) {
> > @@ -1935,10 +1936,9 @@ intel_dp_set_link_train(struct intel_dp *intel_dp,
> > struct drm_i915_private *dev_priv = dev->dev_private;
> > enum port port = intel_dig_port->port;
> > int ret;
> > - uint32_t temp;
> >
> > if (HAS_DDI(dev)) {
> > - temp = I915_READ(DP_TP_CTL(port));
> > + uint32_t temp = I915_READ(DP_TP_CTL(port));
> >
> > if (dp_train_pat & DP_LINK_SCRAMBLING_DISABLE)
> > temp |= DP_TP_CTL_SCRAMBLE_DISABLE;
> > @@ -1948,18 +1948,6 @@ intel_dp_set_link_train(struct intel_dp *intel_dp,
> > temp &= ~DP_TP_CTL_LINK_TRAIN_MASK;
> > switch (dp_train_pat & DP_TRAINING_PATTERN_MASK) {
> > case DP_TRAINING_PATTERN_DISABLE:
> > -
> > - if (port != PORT_A) {
> > - temp |= DP_TP_CTL_LINK_TRAIN_IDLE;
> > - I915_WRITE(DP_TP_CTL(port), temp);
> > -
> > - if (wait_for((I915_READ(DP_TP_STATUS(port)) &
> > - DP_TP_STATUS_IDLE_DONE), 1))
> > - DRM_ERROR("Timed out waiting for DP idle patterns\n");
> > -
> > - temp &= ~DP_TP_CTL_LINK_TRAIN_MASK;
> > - }
> > -
> > temp |= DP_TP_CTL_LINK_TRAIN_NORMAL;
> >
> > break;
> > @@ -2035,6 +2023,37 @@ intel_dp_set_link_train(struct intel_dp *intel_dp,
> > return true;
> > }
> >
> > +static void intel_dp_idle_link_train(struct intel_dp *intel_dp)
>
> Maybe intel_dp_set_idle_link_train?
Yea, that's more like the other link_train functions. Will change it.
>
>
> > +{
> > + struct intel_digital_port *intel_dig_port = dp_to_dig_port(intel_dp);
> > + struct drm_device *dev = intel_dig_port->base.base.dev;
> > + struct drm_i915_private *dev_priv = dev->dev_private;
> > + enum port port = intel_dig_port->port;
> > + uint32_t l;
>
> We usually use "val", "tmp" or "temp". L is unusual.
Ok, will change.
>
>
> > +
> > + if (!HAS_DDI(dev))
> > + return;
> > +
> > + l = I915_READ(DP_TP_CTL(port));
> > + l &= ~DP_TP_CTL_LINK_TRAIN_MASK;
> > + l |= DP_TP_CTL_LINK_TRAIN_IDLE;
> > + I915_WRITE(DP_TP_CTL(port), l);
> > +
> > + /*
> > + * On PORT_A we can have only eDP in SST mode. There the only reason
> > + * we need to set idle transmission mode is to work around a HW issue
> > + * where we enable the pipe while not in idle link-training mode.
> > + * In this case there is requirement to wait for a minimum number of
> > + * idle patterns to be sent.
> > + */
> > + if (port == PORT_A)
> > + return;
> > +
> > + if (wait_for((I915_READ(DP_TP_STATUS(port)) & DP_TP_STATUS_IDLE_DONE),
> > + 1))
> > + DRM_ERROR("Timed out waiting for DP idle patterns\n");
> > +}
> > +
> > /* Enable corresponding port and start training pattern 1 */
> > void
> > intel_dp_start_link_train(struct intel_dp *intel_dp)
> > @@ -2177,10 +2196,19 @@ intel_dp_complete_link_train(struct intel_dp *intel_dp)
> > ++tries;
> > }
> >
> > + intel_dp_idle_link_train(intel_dp);
> > +
> > + intel_dp->DP = DP;
>
> I know this is not your fault, but I kinda _really_ don't like the
> fact that we have 3 places to store the contents of the same register:
> (i) the register itself, (ii) intel_dp->DP and (iii) these temporary
> DP variables. Whoever writes a patch to improve that without adding
> any regressions will get a free $favorite_beverage. I guess the first
> step is to try to remove the temporary variables only.
Yea, I agree we should improve on this. You didn't insist, so I suggest
that we do it after this patch gets merged. If there's no one else I can
also volunteer to do this a bit later.
> So I've also booted your patch and both DP and eDP still work for me.
> With or without the changes I proposed above:
> Reviewed-by: Paulo Zanoni <paulo.r.zanoni at intel.com>
> Tested-by: Paulo Zanoni <paulo.r.zanoni at intel.com>
Thanks,
Imre
>
> > +
> > if (channel_eq)
> > DRM_DEBUG_KMS("Channel EQ done. DP Training successfull\n");
> >
> > - intel_dp_set_link_train(intel_dp, DP, DP_TRAINING_PATTERN_DISABLE);
> > +}
> > +
> > +void intel_dp_stop_link_train(struct intel_dp *intel_dp)
> > +{
> > + intel_dp_set_link_train(intel_dp, intel_dp->DP,
> > + DP_TRAINING_PATTERN_DISABLE);
> > }
> >
> > static void
> > @@ -2388,6 +2416,7 @@ intel_dp_check_link_status(struct intel_dp *intel_dp)
> > drm_get_encoder_name(&intel_encoder->base));
> > intel_dp_start_link_train(intel_dp);
> > intel_dp_complete_link_train(intel_dp);
> > + intel_dp_stop_link_train(intel_dp);
> > }
> > }
> >
> > diff --git a/drivers/gpu/drm/i915/intel_drv.h b/drivers/gpu/drm/i915/intel_drv.h
> > index a8c6960..6d1e63a 100644
> > --- a/drivers/gpu/drm/i915/intel_drv.h
> > +++ b/drivers/gpu/drm/i915/intel_drv.h
> > @@ -542,6 +542,7 @@ extern void intel_dp_init_connector(struct intel_digital_port *intel_dig_port,
> > extern void intel_dp_init_link_config(struct intel_dp *intel_dp);
> > extern void intel_dp_start_link_train(struct intel_dp *intel_dp);
> > extern void intel_dp_complete_link_train(struct intel_dp *intel_dp);
> > +extern void intel_dp_stop_link_train(struct intel_dp *intel_dp);
> > extern void intel_dp_sink_dpms(struct intel_dp *intel_dp, int mode);
> > extern void intel_dp_encoder_destroy(struct drm_encoder *encoder);
> > extern void intel_dp_check_link_status(struct intel_dp *intel_dp);
> > --
> > 1.7.10.4
> >
> > _______________________________________________
> > Intel-gfx mailing list
> > Intel-gfx at lists.freedesktop.org
> > http://lists.freedesktop.org/mailman/listinfo/intel-gfx
>
More information about the Intel-gfx
mailing list