Re: [PATCH 5/7] drm/i915: pass status instead of enable flags to i915_enable_pipestat

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



On Tue,  4 Feb 2014 21:35:49 +0200
Imre Deak <imre.deak@xxxxxxxxx> wrote:

> There isn't any PSR interrupt enable bit for pipe A, so we couldn't
> enable it through the current API. Passing the corresponding status bits
> solves this and also makes the mapping between enable and status bits
> simpler on VLV (addressed in an upcoming patch).
> 
> Except of checking for invalid status bit arguments, no functional
> change.
> 
> Signed-off-by: Imre Deak <imre.deak@xxxxxxxxx>
> ---
>  drivers/gpu/drm/i915/i915_drv.h |  6 ++--
>  drivers/gpu/drm/i915/i915_irq.c | 68 +++++++++++++++++++++++++----------------
>  drivers/gpu/drm/i915/i915_reg.h |  3 ++
>  drivers/gpu/drm/i915/intel_tv.c |  8 ++---
>  4 files changed, 53 insertions(+), 32 deletions(-)
> 
> diff --git a/drivers/gpu/drm/i915/i915_drv.h b/drivers/gpu/drm/i915/i915_drv.h
> index fa37dfd..43f37ca 100644
> --- a/drivers/gpu/drm/i915/i915_drv.h
> +++ b/drivers/gpu/drm/i915/i915_drv.h
> @@ -1994,10 +1994,12 @@ extern void intel_uncore_check_errors(struct drm_device *dev);
>  extern void intel_uncore_fini(struct drm_device *dev);
>  
>  void
> -i915_enable_pipestat(drm_i915_private_t *dev_priv, enum pipe pipe, u32 mask);
> +i915_enable_pipestat(drm_i915_private_t *dev_priv, enum pipe pipe,
> +		     u32 status_mask);
>  
>  void
> -i915_disable_pipestat(drm_i915_private_t *dev_priv, enum pipe pipe, u32 mask);
> +i915_disable_pipestat(drm_i915_private_t *dev_priv, enum pipe pipe,
> +		      u32 status_mask);
>  
>  /* i915_gem.c */
>  int i915_gem_init_ioctl(struct drm_device *dev, void *data,
> diff --git a/drivers/gpu/drm/i915/i915_irq.c b/drivers/gpu/drm/i915/i915_irq.c
> index 3b876ee..ec56a70 100644
> --- a/drivers/gpu/drm/i915/i915_irq.c
> +++ b/drivers/gpu/drm/i915/i915_irq.c
> @@ -471,36 +471,52 @@ done:
>  	return ret;
>  }
>  
> +static u32 pipestat_to_enable_mask(struct drm_device *dev, u32 status_mask)
> +{
> +	return status_mask << 16;
> +}
>  
>  void
> -i915_enable_pipestat(drm_i915_private_t *dev_priv, enum pipe pipe, u32 mask)
> +i915_enable_pipestat(drm_i915_private_t *dev_priv, enum pipe pipe,
> +		     u32 status_mask)
>  {
>  	u32 reg = PIPESTAT(pipe);
> -	u32 pipestat = I915_READ(reg) & 0x7fff0000;
> +	u32 pipestat = I915_READ(reg) & PIPESTAT_INT_ENABLE_MASK;
> +	u32 enable_mask;
>  
>  	assert_spin_locked(&dev_priv->irq_lock);
>  
> -	if ((pipestat & mask) == mask)
> +	if (WARN_ON_ONCE(status_mask & ~PIPESTAT_INT_STATUS_MASK))
> +		return;
> +
> +	enable_mask = pipestat_to_enable_mask(dev_priv->dev, status_mask);
> +	if ((pipestat & enable_mask) == enable_mask)
>  		return;
>  
>  	/* Enable the interrupt, clear any pending status */
> -	pipestat |= mask | (mask >> 16);
> +	pipestat |= enable_mask | status_mask;
>  	I915_WRITE(reg, pipestat);
>  	POSTING_READ(reg);
>  }
>  
>  void
> -i915_disable_pipestat(drm_i915_private_t *dev_priv, enum pipe pipe, u32 mask)
> +i915_disable_pipestat(drm_i915_private_t *dev_priv, enum pipe pipe,
> +		      u32 status_mask)
>  {
>  	u32 reg = PIPESTAT(pipe);
> -	u32 pipestat = I915_READ(reg) & 0x7fff0000;
> +	u32 pipestat = I915_READ(reg) & PIPESTAT_INT_ENABLE_MASK;
> +	u32 enable_mask;
>  
>  	assert_spin_locked(&dev_priv->irq_lock);
>  
> -	if ((pipestat & mask) == 0)
> +	if (WARN_ON_ONCE(status_mask & ~PIPESTAT_INT_STATUS_MASK))
>  		return;
>  
> -	pipestat &= ~mask;
> +	enable_mask = pipestat_to_enable_mask(dev_priv->dev, status_mask);
> +	if ((pipestat & enable_mask) == 0)
> +		return;
> +
> +	pipestat &= ~enable_mask;
>  	I915_WRITE(reg, pipestat);
>  	POSTING_READ(reg);
>  }
> @@ -518,10 +534,10 @@ static void i915_enable_asle_pipestat(struct drm_device *dev)
>  
>  	spin_lock_irqsave(&dev_priv->irq_lock, irqflags);
>  
> -	i915_enable_pipestat(dev_priv, PIPE_B, PIPE_LEGACY_BLC_EVENT_ENABLE);
> +	i915_enable_pipestat(dev_priv, PIPE_B, PIPE_LEGACY_BLC_EVENT_STATUS);
>  	if (INTEL_INFO(dev)->gen >= 4)
>  		i915_enable_pipestat(dev_priv, PIPE_A,
> -				     PIPE_LEGACY_BLC_EVENT_ENABLE);
> +				     PIPE_LEGACY_BLC_EVENT_STATUS);
>  
>  	spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags);
>  }
> @@ -2270,10 +2286,10 @@ static int i915_enable_vblank(struct drm_device *dev, int pipe)
>  	spin_lock_irqsave(&dev_priv->irq_lock, irqflags);
>  	if (INTEL_INFO(dev)->gen >= 4)
>  		i915_enable_pipestat(dev_priv, pipe,
> -				     PIPE_START_VBLANK_INTERRUPT_ENABLE);
> +				     PIPE_START_VBLANK_INTERRUPT_STATUS);
>  	else
>  		i915_enable_pipestat(dev_priv, pipe,
> -				     PIPE_VBLANK_INTERRUPT_ENABLE);
> +				     PIPE_VBLANK_INTERRUPT_STATUS);
>  
>  	/* maintain vblank delivery even in deep C-states */
>  	if (dev_priv->info->gen == 3)
> @@ -2310,7 +2326,7 @@ static int valleyview_enable_vblank(struct drm_device *dev, int pipe)
>  
>  	spin_lock_irqsave(&dev_priv->irq_lock, irqflags);
>  	i915_enable_pipestat(dev_priv, pipe,
> -			     PIPE_START_VBLANK_INTERRUPT_ENABLE);
> +			     PIPE_START_VBLANK_INTERRUPT_STATUS);
>  	spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags);
>  
>  	return 0;
> @@ -2345,8 +2361,8 @@ static void i915_disable_vblank(struct drm_device *dev, int pipe)
>  		I915_WRITE(INSTPM, _MASKED_BIT_ENABLE(INSTPM_AGPBUSY_DIS));
>  
>  	i915_disable_pipestat(dev_priv, pipe,
> -			      PIPE_VBLANK_INTERRUPT_ENABLE |
> -			      PIPE_START_VBLANK_INTERRUPT_ENABLE);
> +			      PIPE_VBLANK_INTERRUPT_STATUS |
> +			      PIPE_START_VBLANK_INTERRUPT_STATUS);
>  	spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags);
>  }
>  
> @@ -2369,7 +2385,7 @@ static void valleyview_disable_vblank(struct drm_device *dev, int pipe)
>  
>  	spin_lock_irqsave(&dev_priv->irq_lock, irqflags);
>  	i915_disable_pipestat(dev_priv, pipe,
> -			      PIPE_START_VBLANK_INTERRUPT_ENABLE);
> +			      PIPE_START_VBLANK_INTERRUPT_STATUS);
>  	spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags);
>  }
>  
> @@ -2918,8 +2934,8 @@ static int valleyview_irq_postinstall(struct drm_device *dev)
>  {
>  	drm_i915_private_t *dev_priv = (drm_i915_private_t *) dev->dev_private;
>  	u32 enable_mask;
> -	u32 pipestat_enable = PLANE_FLIP_DONE_INT_EN_VLV |
> -		PIPE_CRC_DONE_ENABLE;
> +	u32 pipestat_enable = PLANE_FLIP_DONE_INT_STATUS_VLV |
> +		PIPE_CRC_DONE_INTERRUPT_STATUS;
>  	unsigned long irqflags;
>  
>  	enable_mask = I915_DISPLAY_PORT_INTERRUPT;
> @@ -2950,7 +2966,7 @@ static int valleyview_irq_postinstall(struct drm_device *dev)
>  	 * just to make the assert_spin_locked check happy. */
>  	spin_lock_irqsave(&dev_priv->irq_lock, irqflags);
>  	i915_enable_pipestat(dev_priv, PIPE_A, pipestat_enable);
> -	i915_enable_pipestat(dev_priv, PIPE_A, PIPE_GMBUS_EVENT_ENABLE);
> +	i915_enable_pipestat(dev_priv, PIPE_A, PIPE_GMBUS_INTERRUPT_STATUS);
>  	i915_enable_pipestat(dev_priv, PIPE_B, pipestat_enable);
>  	spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags);
>  
> @@ -3173,8 +3189,8 @@ static int i8xx_irq_postinstall(struct drm_device *dev)
>  	/* Interrupt setup is already guaranteed to be single-threaded, this is
>  	 * just to make the assert_spin_locked check happy. */
>  	spin_lock_irqsave(&dev_priv->irq_lock, irqflags);
> -	i915_enable_pipestat(dev_priv, PIPE_A, PIPE_CRC_DONE_ENABLE);
> -	i915_enable_pipestat(dev_priv, PIPE_B, PIPE_CRC_DONE_ENABLE);
> +	i915_enable_pipestat(dev_priv, PIPE_A, PIPE_CRC_DONE_INTERRUPT_STATUS);
> +	i915_enable_pipestat(dev_priv, PIPE_B, PIPE_CRC_DONE_INTERRUPT_STATUS);
>  	spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags);
>  
>  	return 0;
> @@ -3356,8 +3372,8 @@ static int i915_irq_postinstall(struct drm_device *dev)
>  	/* Interrupt setup is already guaranteed to be single-threaded, this is
>  	 * just to make the assert_spin_locked check happy. */
>  	spin_lock_irqsave(&dev_priv->irq_lock, irqflags);
> -	i915_enable_pipestat(dev_priv, PIPE_A, PIPE_CRC_DONE_ENABLE);
> -	i915_enable_pipestat(dev_priv, PIPE_B, PIPE_CRC_DONE_ENABLE);
> +	i915_enable_pipestat(dev_priv, PIPE_A, PIPE_CRC_DONE_INTERRUPT_STATUS);
> +	i915_enable_pipestat(dev_priv, PIPE_B, PIPE_CRC_DONE_INTERRUPT_STATUS);
>  	spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags);
>  
>  	return 0;
> @@ -3566,9 +3582,9 @@ static int i965_irq_postinstall(struct drm_device *dev)
>  	/* Interrupt setup is already guaranteed to be single-threaded, this is
>  	 * just to make the assert_spin_locked check happy. */
>  	spin_lock_irqsave(&dev_priv->irq_lock, irqflags);
> -	i915_enable_pipestat(dev_priv, PIPE_A, PIPE_GMBUS_EVENT_ENABLE);
> -	i915_enable_pipestat(dev_priv, PIPE_A, PIPE_CRC_DONE_ENABLE);
> -	i915_enable_pipestat(dev_priv, PIPE_B, PIPE_CRC_DONE_ENABLE);
> +	i915_enable_pipestat(dev_priv, PIPE_A, PIPE_GMBUS_INTERRUPT_STATUS);
> +	i915_enable_pipestat(dev_priv, PIPE_A, PIPE_CRC_DONE_INTERRUPT_STATUS);
> +	i915_enable_pipestat(dev_priv, PIPE_B, PIPE_CRC_DONE_INTERRUPT_STATUS);
>  	spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags);
>  
>  	/*
> diff --git a/drivers/gpu/drm/i915/i915_reg.h b/drivers/gpu/drm/i915/i915_reg.h
> index 8eefb26..599c7f6 100644
> --- a/drivers/gpu/drm/i915/i915_reg.h
> +++ b/drivers/gpu/drm/i915/i915_reg.h
> @@ -3263,6 +3263,9 @@
>  #define   PIPE_VBLANK_INTERRUPT_STATUS		(1UL<<1)
>  #define   PIPE_OVERLAY_UPDATED_STATUS		(1UL<<0)
>  
> +#define PIPESTAT_INT_ENABLE_MASK		0x7fff0000
> +#define PIPESTAT_INT_STATUS_MASK		0x0000ffff
> +
>  #define PIPESRC(pipe) _PIPE(pipe, _PIPEASRC, _PIPEBSRC)
>  #define PIPECONF(tran) _TRANSCODER(tran, _PIPEACONF, _PIPEBCONF)
>  #define PIPEDSL(pipe)  _PIPE(pipe, _PIPEADSL, _PIPEBDSL)
> diff --git a/drivers/gpu/drm/i915/intel_tv.c b/drivers/gpu/drm/i915/intel_tv.c
> index 22cf0f4..ccd02ec 100644
> --- a/drivers/gpu/drm/i915/intel_tv.c
> +++ b/drivers/gpu/drm/i915/intel_tv.c
> @@ -1189,8 +1189,8 @@ intel_tv_detect_type(struct intel_tv *intel_tv,
>  	if (connector->polled & DRM_CONNECTOR_POLL_HPD) {
>  		spin_lock_irqsave(&dev_priv->irq_lock, irqflags);
>  		i915_disable_pipestat(dev_priv, 0,
> -				      PIPE_HOTPLUG_INTERRUPT_ENABLE |
> -				      PIPE_HOTPLUG_TV_INTERRUPT_ENABLE);
> +				      PIPE_HOTPLUG_INTERRUPT_STATUS |
> +				      PIPE_HOTPLUG_TV_INTERRUPT_STATUS);
>  		spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags);
>  	}
>  
> @@ -1266,8 +1266,8 @@ intel_tv_detect_type(struct intel_tv *intel_tv,
>  	if (connector->polled & DRM_CONNECTOR_POLL_HPD) {
>  		spin_lock_irqsave(&dev_priv->irq_lock, irqflags);
>  		i915_enable_pipestat(dev_priv, 0,
> -				     PIPE_HOTPLUG_INTERRUPT_ENABLE |
> -				     PIPE_HOTPLUG_TV_INTERRUPT_ENABLE);
> +				     PIPE_HOTPLUG_INTERRUPT_STATUS |
> +				     PIPE_HOTPLUG_TV_INTERRUPT_STATUS);
>  		spin_unlock_irqrestore(&dev_priv->irq_lock, irqflags);
>  	}
>  

I almost think we should just separate enable vs status entirely.  As
long as the bits are named consistently it may be easier to follow (as
Ville found in your next patch with the subtle remapping of status
bits).

But if everyone else is comfortable with this, it looks fine.

Reviewed-by: Jesse Barnes <jbarnes@xxxxxxxxxxxxxxxx>
_______________________________________________
Intel-gfx mailing list
Intel-gfx@xxxxxxxxxxxxxxxxxxxxx
http://lists.freedesktop.org/mailman/listinfo/intel-gfx




[Index of Archives]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]
  Powered by Linux