On Thu, Sep 29, 2011 at 06:09:50PM -0700, Keith Packard wrote: > There's no good reason to turn off the eDP force VDD bit synchronously > while probing devices; that just sticks a huge delay into all mode > setting paths. Instead, queue a delayed work proc to disable the VDD > force bit and then remember when that fires to ensure that the > appropriate delay is respected before trying to turn it back on. > > Signed-off-by: Keith Packard <keithp@xxxxxxxxxx> > --- > drivers/gpu/drm/i915/intel_dp.c | 121 +++++++++++++++++++++++++++++++-------- > 1 files changed, 97 insertions(+), 24 deletions(-) > > diff --git a/drivers/gpu/drm/i915/intel_dp.c b/drivers/gpu/drm/i915/intel_dp.c > index c7ccb46..08817b0 100644 > --- a/drivers/gpu/drm/i915/intel_dp.c > +++ b/drivers/gpu/drm/i915/intel_dp.c > @@ -62,6 +62,9 @@ struct intel_dp { > int panel_power_up_delay; > int panel_power_down_delay; > struct drm_display_mode *panel_fixed_mode; /* for eDP */ > + struct delayed_work panel_vdd_work; > + bool want_panel_vdd; > + unsigned long panel_off_jiffies; > }; > > /** > @@ -611,7 +614,7 @@ intel_dp_i2c_aux_ch(struct i2c_adapter *adapter, int mode, > } > > static void ironlake_edp_panel_vdd_on(struct intel_dp *intel_dp); > -static void ironlake_edp_panel_vdd_off(struct intel_dp *intel_dp); > +static void ironlake_edp_panel_vdd_off(struct intel_dp *intel_dp, bool sync); > > static int > intel_dp_i2c_init(struct intel_dp *intel_dp, > @@ -634,7 +637,7 @@ intel_dp_i2c_init(struct intel_dp *intel_dp, > > ironlake_edp_panel_vdd_on(intel_dp); > ret = i2c_dp_aux_add_bus(&intel_dp->adapter); > - ironlake_edp_panel_vdd_off(intel_dp); > + ironlake_edp_panel_vdd_off(intel_dp, false); > return ret; > } > > @@ -848,6 +851,23 @@ intel_dp_mode_set(struct drm_encoder *encoder, struct drm_display_mode *mode, > } > } > > +static void ironlake_wait_panel_off(struct intel_dp *intel_dp) > +{ > + unsigned long off_time; > + unsigned long delay; > + DRM_DEBUG_KMS("Wait for panel power off time\n"); > + off_time = intel_dp->panel_off_jiffies + msecs_to_jiffies(intel_dp->panel_power_down_delay); > + if (time_after(jiffies, off_time)) { > + DRM_DEBUG_KMS("Time already passed"); > + return; > + } > + delay = jiffies_to_msecs(off_time - jiffies); > + if (delay > intel_dp->panel_power_down_delay) > + delay = intel_dp->panel_power_down_delay; > + DRM_DEBUG_KMS("Waiting an additional %ld ms\n", delay); > + msleep(delay); > +} A cancel_work might be good here, not point in waking up the cpu for no reason. And can you add "panel off timer: " to the deboug output? > + > static void ironlake_edp_panel_vdd_on(struct intel_dp *intel_dp) > { > struct drm_device *dev = intel_dp->base.base.dev; > @@ -858,6 +878,16 @@ static void ironlake_edp_panel_vdd_on(struct intel_dp *intel_dp) > return; > DRM_DEBUG_KMS("Turn eDP VDD on\n"); > > + WARN(intel_dp->want_panel_vdd, > + "eDP VDD already requested on\n"); > + > + intel_dp->want_panel_vdd = true; > + if (ironlake_edp_have_panel_vdd(intel_dp)) { > + DRM_DEBUG_KMS("eDP VDD already on\n"); > + return; > + } > + > + ironlake_wait_panel_off(intel_dp); > pp = I915_READ(PCH_PP_CONTROL); > pp &= ~PANEL_UNLOCK_MASK; > pp |= PANEL_UNLOCK_REGS; > @@ -871,31 +901,64 @@ static void ironlake_edp_panel_vdd_on(struct intel_dp *intel_dp) > * If the panel wasn't on, delay before accessing aux channel > */ > if (!ironlake_edp_have_panel_power(intel_dp)) { > + DRM_DEBUG_KMS("eDP was not running\n"); > msleep(intel_dp->panel_power_up_delay); > - DRM_DEBUG_KMS("eDP VDD was not on\n"); > } > } > > -static void ironlake_edp_panel_vdd_off(struct intel_dp *intel_dp) > +static void ironlake_panel_vdd_off_sync(struct intel_dp *intel_dp) > { > struct drm_device *dev = intel_dp->base.base.dev; > struct drm_i915_private *dev_priv = dev->dev_private; > u32 pp; > > + if (!intel_dp->want_panel_vdd && ironlake_edp_have_panel_vdd(intel_dp)) { > + pp = I915_READ(PCH_PP_CONTROL); > + pp &= ~PANEL_UNLOCK_MASK; > + pp |= PANEL_UNLOCK_REGS; > + pp &= ~EDP_FORCE_VDD; > + I915_WRITE(PCH_PP_CONTROL, pp); > + POSTING_READ(PCH_PP_CONTROL); > + > + /* Make sure sequencer is idle before allowing subsequent activity */ > + DRM_DEBUG_KMS("PCH_PP_STATUS: 0x%08x PCH_PP_CONTROL: 0x%08x\n", > + I915_READ(PCH_PP_STATUS), I915_READ(PCH_PP_CONTROL)); > + intel_dp->panel_off_jiffies = jiffies; > + } > +} > + > +static void ironlake_panel_vdd_work(struct work_struct *__work) > +{ > + struct intel_dp *intel_dp = container_of(to_delayed_work(__work), > + struct intel_dp, panel_vdd_work); > + struct drm_device *dev = intel_dp->base.base.dev; > + > + mutex_lock(&dev->struct_mutex); > + ironlake_panel_vdd_off_sync(intel_dp); > + mutex_unlock(&dev->struct_mutex); > +} Like Chris already mentioned, s/struct_mutex/mode_config.mutex/ Maybe also move the intel_dp->want_panel_vdd check in here - we need it only here to avoid a race with vdd_on. I've almost overlooked it and already started to write the complaint about the race ;-) > + > +static void ironlake_edp_panel_vdd_off(struct intel_dp *intel_dp, bool sync) > +{ > if (!is_edp(intel_dp)) > return; > - DRM_DEBUG_KMS("Turn eDP VDD off\n"); > - pp = I915_READ(PCH_PP_CONTROL); > - pp &= ~PANEL_UNLOCK_MASK; > - pp |= PANEL_UNLOCK_REGS; > - pp &= ~EDP_FORCE_VDD; > - I915_WRITE(PCH_PP_CONTROL, pp); > - POSTING_READ(PCH_PP_CONTROL); > > - /* Make sure sequencer is idle before allowing subsequent activity */ > - DRM_DEBUG_KMS("PCH_PP_STATUS: 0x%08x PCH_PP_CONTROL: 0x%08x\n", > - I915_READ(PCH_PP_STATUS), I915_READ(PCH_PP_CONTROL)); > - msleep(intel_dp->panel_power_down_delay); > + DRM_DEBUG_KMS("Turn eDP VDD off %d\n", intel_dp->want_panel_vdd); > + WARN(!intel_dp->want_panel_vdd, "eDP VDD not forced on"); > + > + intel_dp->want_panel_vdd = false; > + > + if (sync) { > + ironlake_panel_vdd_off_sync(intel_dp); > + } else { > + /* > + * Queue the timer to fire a long > + * time from now (relative to the power down delay) > + * to keep the panel power up across a sequence of operations > + */ > + schedule_delayed_work(&intel_dp->panel_vdd_work, > + msecs_to_jiffies(intel_dp->panel_power_down_delay * 5)); > + } > } > > /* Returns true if the panel was already on when called */ > @@ -908,6 +971,7 @@ static void ironlake_edp_panel_on (struct intel_dp *intel_dp) > if (ironlake_edp_have_panel_power(intel_dp)) > return; > > + ironlake_wait_panel_off(intel_dp); > pp = I915_READ(PCH_PP_CONTROL); > pp &= ~PANEL_UNLOCK_MASK; > pp |= PANEL_UNLOCK_REGS; > @@ -951,7 +1015,6 @@ static void ironlake_edp_panel_off(struct drm_encoder *encoder) > pp &= ~POWER_TARGET_ON; > I915_WRITE(PCH_PP_CONTROL, pp); > POSTING_READ(PCH_PP_CONTROL); > - msleep(intel_dp->panel_power_down_delay); > > if (wait_for((I915_READ(PCH_PP_STATUS) & idle_off_mask) == 0, 5000)) > DRM_ERROR("panel off wait timed out: 0x%08x\n", > @@ -960,6 +1023,7 @@ static void ironlake_edp_panel_off(struct drm_encoder *encoder) > pp |= PANEL_POWER_RESET; /* restore panel reset bit */ > I915_WRITE(PCH_PP_CONTROL, pp); > POSTING_READ(PCH_PP_CONTROL); > + intel_dp->panel_off_jiffies = jiffies; > } > > static void ironlake_edp_backlight_on (struct drm_device *dev) > @@ -1053,7 +1117,7 @@ static void intel_dp_prepare(struct drm_encoder *encoder) > /* Wake up the sink first */ > ironlake_edp_panel_vdd_on(intel_dp); > intel_dp_sink_dpms(intel_dp, DRM_MODE_DPMS_ON); > - ironlake_edp_panel_vdd_off(intel_dp); > + ironlake_edp_panel_vdd_off(intel_dp, false); > > if (is_edp(intel_dp)) { > ironlake_edp_backlight_off(dev); > @@ -1077,7 +1141,7 @@ static void intel_dp_commit(struct drm_encoder *encoder) > > if (is_edp(intel_dp)) > ironlake_edp_panel_on(intel_dp); > - ironlake_edp_panel_vdd_off(intel_dp); > + ironlake_edp_panel_vdd_off(intel_dp, true); > > intel_dp_complete_link_train(intel_dp); > > @@ -1111,10 +1175,10 @@ intel_dp_dpms(struct drm_encoder *encoder, int mode) > intel_dp_start_link_train(intel_dp); > if (is_edp(intel_dp)) > ironlake_edp_panel_on(intel_dp); > - ironlake_edp_panel_vdd_off(intel_dp); > + ironlake_edp_panel_vdd_off(intel_dp, true); > intel_dp_complete_link_train(intel_dp); > } else > - ironlake_edp_panel_vdd_off(intel_dp); > + ironlake_edp_panel_vdd_off(intel_dp, false); Any reason why one vdd_off is sync while the other isn't? > if (is_edp(intel_dp)) > ironlake_edp_backlight_on(dev); > } > @@ -1752,7 +1816,7 @@ intel_dp_get_edid(struct drm_connector *connector, struct i2c_adapter *adapter) > > ironlake_edp_panel_vdd_on(intel_dp); > edid = drm_get_edid(connector, adapter); > - ironlake_edp_panel_vdd_off(intel_dp); > + ironlake_edp_panel_vdd_off(intel_dp, false); > return edid; > } > > @@ -1764,7 +1828,7 @@ intel_dp_get_edid_modes(struct drm_connector *connector, struct i2c_adapter *ada > > ironlake_edp_panel_vdd_on(intel_dp); > ret = intel_ddc_get_modes(connector, adapter); > - ironlake_edp_panel_vdd_off(intel_dp); > + ironlake_edp_panel_vdd_off(intel_dp, false); > return ret; > } > > @@ -1951,6 +2015,10 @@ static void intel_dp_encoder_destroy(struct drm_encoder *encoder) > > i2c_del_adapter(&intel_dp->adapter); > drm_encoder_cleanup(encoder); > + if (is_edp(intel_dp)) { > + cancel_delayed_work_sync(&intel_dp->panel_vdd_work); > + ironlake_panel_vdd_off_sync(intel_dp); > + } > kfree(intel_dp); > } > > @@ -2087,8 +2155,11 @@ intel_dp_init(struct drm_device *dev, int output_reg) > else if (output_reg == DP_D || output_reg == PCH_DP_D) > intel_encoder->clone_mask = (1 << INTEL_DP_D_CLONE_BIT); > > - if (is_edp(intel_dp)) > + if (is_edp(intel_dp)) { > intel_encoder->clone_mask = (1 << INTEL_EDP_CLONE_BIT); > + INIT_DELAYED_WORK(&intel_dp->panel_vdd_work, > + ironlake_panel_vdd_work); > + } > > intel_encoder->crtc_mask = (1 << 0) | (1 << 1); > connector->interlace_allowed = true; > @@ -2163,9 +2234,11 @@ intel_dp_init(struct drm_device *dev, int output_reg) > DRM_DEBUG_KMS("panel power up delay %d, power down delay %d\n", > intel_dp->panel_power_up_delay, intel_dp->panel_power_down_delay); > > + intel_dp->panel_off_jiffies = jiffies - intel_dp->panel_power_down_delay; > + > ironlake_edp_panel_vdd_on(intel_dp); > ret = intel_dp_get_dpcd(intel_dp); > - ironlake_edp_panel_vdd_off(intel_dp); > + ironlake_edp_panel_vdd_off(intel_dp, false); > if (ret) { > if (intel_dp->dpcd[DP_DPCD_REV] >= 0x11) > dev_priv->no_aux_handshake = > -- > 1.7.6.3 > > _______________________________________________ > dri-devel mailing list > dri-devel@xxxxxxxxxxxxxxxxxxxxx > http://lists.freedesktop.org/mailman/listinfo/dri-devel -- Daniel Vetter Mail: daniel@xxxxxxxx Mobile: +41 (0)79 365 57 48 _______________________________________________ dri-devel mailing list dri-devel@xxxxxxxxxxxxxxxxxxxxx http://lists.freedesktop.org/mailman/listinfo/dri-devel