[PATCH 18/21] drm/i915: Disable eDP VDD in a delayed work proc instead of synchronously

Daniel Vetter daniel at ffwll.ch
Fri Sep 30 11:47:00 PDT 2011


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 at keithp.com>
> ---
>  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 at lists.freedesktop.org
> http://lists.freedesktop.org/mailman/listinfo/dri-devel

-- 
Daniel Vetter
Mail: daniel at ffwll.ch
Mobile: +41 (0)79 365 57 48


More information about the dri-devel mailing list