gma500: Add VBLANK support for Poulsbo hardware
Signed-off-by: Patrik Jakobsson <patrik.r.jakobsson@gmail.com> Signed-off-by: Alan Cox <alan@linux.intel.com> Signed-off-by: Greg Kroah-Hartman <gregkh@suse.de>
This commit is contained in:
parent
24dd55b4ee
commit
0a0691a20d
2 changed files with 44 additions and 30 deletions
|
@ -135,6 +135,9 @@ enum {
|
||||||
#define _PSB_IRQ_MSVDX_FLAG (1<<19)
|
#define _PSB_IRQ_MSVDX_FLAG (1<<19)
|
||||||
#define _LNC_IRQ_TOPAZ_FLAG (1<<20)
|
#define _LNC_IRQ_TOPAZ_FLAG (1<<20)
|
||||||
|
|
||||||
|
#define _PSB_PIPE_EVENT_FLAG (_PSB_VSYNC_PIPEA_FLAG | \
|
||||||
|
_PSB_VSYNC_PIPEB_FLAG)
|
||||||
|
|
||||||
/* This flag includes all the display IRQ bits excepts the vblank irqs. */
|
/* This flag includes all the display IRQ bits excepts the vblank irqs. */
|
||||||
#define _MDFLD_DISP_ALL_IRQ_FLAG (_MDFLD_PIPEC_EVENT_FLAG | \
|
#define _MDFLD_DISP_ALL_IRQ_FLAG (_MDFLD_PIPEC_EVENT_FLAG | \
|
||||||
_MDFLD_PIPEB_EVENT_FLAG | \
|
_MDFLD_PIPEB_EVENT_FLAG | \
|
||||||
|
|
|
@ -138,22 +138,11 @@ void mid_disable_pipe_event(struct drm_psb_private *dev_priv, int pipe)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Display controller interrupt handler for vsync/vblank.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
static void mid_vblank_handler(struct drm_device *dev, uint32_t pipe)
|
|
||||||
{
|
|
||||||
drm_handle_vblank(dev, pipe);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Display controller interrupt handler for pipe event.
|
* Display controller interrupt handler for pipe event.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
#define WAIT_STATUS_CLEAR_LOOP_COUNT 0xffff
|
static void mid_pipe_event_handler(struct drm_device *dev, int pipe)
|
||||||
static void mid_pipe_event_handler(struct drm_device *dev, uint32_t pipe)
|
|
||||||
{
|
{
|
||||||
struct drm_psb_private *dev_priv =
|
struct drm_psb_private *dev_priv =
|
||||||
(struct drm_psb_private *) dev->dev_private;
|
(struct drm_psb_private *) dev->dev_private;
|
||||||
|
@ -162,6 +151,7 @@ static void mid_pipe_event_handler(struct drm_device *dev, uint32_t pipe)
|
||||||
uint32_t pipe_stat_reg = psb_pipestat(pipe);
|
uint32_t pipe_stat_reg = psb_pipestat(pipe);
|
||||||
uint32_t pipe_enable = dev_priv->pipestat[pipe];
|
uint32_t pipe_enable = dev_priv->pipestat[pipe];
|
||||||
uint32_t pipe_status = dev_priv->pipestat[pipe] >> 16;
|
uint32_t pipe_status = dev_priv->pipestat[pipe] >> 16;
|
||||||
|
uint32_t pipe_clear;
|
||||||
uint32_t i = 0;
|
uint32_t i = 0;
|
||||||
|
|
||||||
spin_lock(&dev_priv->irqmask_lock);
|
spin_lock(&dev_priv->irqmask_lock);
|
||||||
|
@ -172,27 +162,23 @@ static void mid_pipe_event_handler(struct drm_device *dev, uint32_t pipe)
|
||||||
|
|
||||||
spin_unlock(&dev_priv->irqmask_lock);
|
spin_unlock(&dev_priv->irqmask_lock);
|
||||||
|
|
||||||
/* clear the 2nd level interrupt status bits */
|
/* Clear the 2nd level interrupt status bits
|
||||||
/**
|
* Sometimes the bits are very sticky so we repeat until they unstick */
|
||||||
* FIXME: shouldn't use while loop here. However, the interrupt
|
for (i = 0; i < 0xffff; i++) {
|
||||||
* status 'sticky' bits cannot be cleared by setting '1' to that
|
|
||||||
* bit once...
|
|
||||||
*/
|
|
||||||
for (i = 0; i < WAIT_STATUS_CLEAR_LOOP_COUNT; i++) {
|
|
||||||
PSB_WVDC32(PSB_RVDC32(pipe_stat_reg), pipe_stat_reg);
|
PSB_WVDC32(PSB_RVDC32(pipe_stat_reg), pipe_stat_reg);
|
||||||
(void) PSB_RVDC32(pipe_stat_reg);
|
pipe_clear = PSB_RVDC32(pipe_stat_reg) & pipe_status;
|
||||||
|
|
||||||
if ((PSB_RVDC32(pipe_stat_reg) & pipe_status) == 0)
|
if (pipe_clear == 0)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (i == WAIT_STATUS_CLEAR_LOOP_COUNT)
|
if (pipe_clear)
|
||||||
dev_err(dev->dev,
|
dev_err(dev->dev,
|
||||||
"%s, can't clear the status bits in pipe_stat_reg, its value = 0x%x.\n",
|
"%s, can't clear status bits for pipe %d, its value = 0x%x.\n",
|
||||||
__func__, PSB_RVDC32(pipe_stat_reg));
|
__func__, pipe, PSB_RVDC32(pipe_stat_reg));
|
||||||
|
|
||||||
if (pipe_stat_val & PIPE_VBLANK_STATUS)
|
if (pipe_stat_val & PIPE_VBLANK_STATUS)
|
||||||
mid_vblank_handler(dev, pipe);
|
drm_handle_vblank(dev, pipe);
|
||||||
|
|
||||||
if (pipe_stat_val & PIPE_TE_STATUS)
|
if (pipe_stat_val & PIPE_TE_STATUS)
|
||||||
drm_handle_vblank(dev, pipe);
|
drm_handle_vblank(dev, pipe);
|
||||||
|
@ -203,8 +189,11 @@ static void mid_pipe_event_handler(struct drm_device *dev, uint32_t pipe)
|
||||||
*/
|
*/
|
||||||
static void psb_vdc_interrupt(struct drm_device *dev, uint32_t vdc_stat)
|
static void psb_vdc_interrupt(struct drm_device *dev, uint32_t vdc_stat)
|
||||||
{
|
{
|
||||||
if (vdc_stat & _PSB_PIPEA_EVENT_FLAG)
|
if (vdc_stat & _PSB_VSYNC_PIPEA_FLAG)
|
||||||
mid_pipe_event_handler(dev, 0);
|
mid_pipe_event_handler(dev, 0);
|
||||||
|
|
||||||
|
if (vdc_stat & _PSB_VSYNC_PIPEB_FLAG)
|
||||||
|
mid_pipe_event_handler(dev, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
irqreturn_t psb_irq_handler(DRM_IRQ_ARGS)
|
irqreturn_t psb_irq_handler(DRM_IRQ_ARGS)
|
||||||
|
@ -220,8 +209,13 @@ irqreturn_t psb_irq_handler(DRM_IRQ_ARGS)
|
||||||
|
|
||||||
vdc_stat = PSB_RVDC32(PSB_INT_IDENTITY_R);
|
vdc_stat = PSB_RVDC32(PSB_INT_IDENTITY_R);
|
||||||
|
|
||||||
|
if (vdc_stat & _PSB_PIPE_EVENT_FLAG)
|
||||||
|
dsp_int = 1;
|
||||||
|
|
||||||
|
/* FIXME: Handle Medfield
|
||||||
if (vdc_stat & _MDFLD_DISP_ALL_IRQ_FLAG)
|
if (vdc_stat & _MDFLD_DISP_ALL_IRQ_FLAG)
|
||||||
dsp_int = 1;
|
dsp_int = 1;
|
||||||
|
*/
|
||||||
|
|
||||||
if (vdc_stat & _PSB_IRQ_SGX_FLAG)
|
if (vdc_stat & _PSB_IRQ_SGX_FLAG)
|
||||||
sgx_int = 1;
|
sgx_int = 1;
|
||||||
|
@ -267,13 +261,18 @@ void psb_irq_preinstall(struct drm_device *dev)
|
||||||
if (gma_power_is_on(dev))
|
if (gma_power_is_on(dev))
|
||||||
PSB_WVDC32(0xFFFFFFFF, PSB_HWSTAM);
|
PSB_WVDC32(0xFFFFFFFF, PSB_HWSTAM);
|
||||||
if (dev->vblank_enabled[0])
|
if (dev->vblank_enabled[0])
|
||||||
dev_priv->vdc_irq_mask |= _PSB_PIPEA_EVENT_FLAG;
|
dev_priv->vdc_irq_mask |= _PSB_VSYNC_PIPEA_FLAG;
|
||||||
|
if (dev->vblank_enabled[1])
|
||||||
|
dev_priv->vdc_irq_mask |= _PSB_VSYNC_PIPEB_FLAG;
|
||||||
|
|
||||||
|
/* FIXME: Handle Medfield irq mask
|
||||||
if (dev->vblank_enabled[1])
|
if (dev->vblank_enabled[1])
|
||||||
dev_priv->vdc_irq_mask |= _MDFLD_PIPEB_EVENT_FLAG;
|
dev_priv->vdc_irq_mask |= _MDFLD_PIPEB_EVENT_FLAG;
|
||||||
if (dev->vblank_enabled[2])
|
if (dev->vblank_enabled[2])
|
||||||
dev_priv->vdc_irq_mask |= _MDFLD_PIPEC_EVENT_FLAG;
|
dev_priv->vdc_irq_mask |= _MDFLD_PIPEC_EVENT_FLAG;
|
||||||
|
*/
|
||||||
|
|
||||||
/*This register is safe even if display island is off*/
|
/* This register is safe even if display island is off */
|
||||||
PSB_WVDC32(~dev_priv->vdc_irq_mask, PSB_INT_MASK_R);
|
PSB_WVDC32(~dev_priv->vdc_irq_mask, PSB_INT_MASK_R);
|
||||||
spin_unlock_irqrestore(&dev_priv->irqmask_lock, irqflags);
|
spin_unlock_irqrestore(&dev_priv->irqmask_lock, irqflags);
|
||||||
}
|
}
|
||||||
|
@ -471,7 +470,13 @@ int psb_enable_vblank(struct drm_device *dev, int pipe)
|
||||||
|
|
||||||
spin_lock_irqsave(&dev_priv->irqmask_lock, irqflags);
|
spin_lock_irqsave(&dev_priv->irqmask_lock, irqflags);
|
||||||
|
|
||||||
mid_enable_pipe_event(dev_priv, pipe);
|
if (pipe == 0)
|
||||||
|
dev_priv->vdc_irq_mask |= _PSB_VSYNC_PIPEA_FLAG;
|
||||||
|
else if (pipe == 1)
|
||||||
|
dev_priv->vdc_irq_mask |= _PSB_VSYNC_PIPEB_FLAG;
|
||||||
|
|
||||||
|
PSB_WVDC32(~dev_priv->vdc_irq_mask, PSB_INT_MASK_R);
|
||||||
|
PSB_WVDC32(dev_priv->vdc_irq_mask, PSB_INT_ENABLE_R);
|
||||||
psb_enable_pipestat(dev_priv, pipe, PIPE_VBLANK_INTERRUPT_ENABLE);
|
psb_enable_pipestat(dev_priv, pipe, PIPE_VBLANK_INTERRUPT_ENABLE);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&dev_priv->irqmask_lock, irqflags);
|
spin_unlock_irqrestore(&dev_priv->irqmask_lock, irqflags);
|
||||||
|
@ -493,7 +498,13 @@ void psb_disable_vblank(struct drm_device *dev, int pipe)
|
||||||
#endif
|
#endif
|
||||||
spin_lock_irqsave(&dev_priv->irqmask_lock, irqflags);
|
spin_lock_irqsave(&dev_priv->irqmask_lock, irqflags);
|
||||||
|
|
||||||
mid_disable_pipe_event(dev_priv, pipe);
|
if (pipe == 0)
|
||||||
|
dev_priv->vdc_irq_mask &= ~_PSB_VSYNC_PIPEA_FLAG;
|
||||||
|
else if (pipe == 1)
|
||||||
|
dev_priv->vdc_irq_mask &= ~_PSB_VSYNC_PIPEB_FLAG;
|
||||||
|
|
||||||
|
PSB_WVDC32(~dev_priv->vdc_irq_mask, PSB_INT_MASK_R);
|
||||||
|
PSB_WVDC32(dev_priv->vdc_irq_mask, PSB_INT_ENABLE_R);
|
||||||
psb_disable_pipestat(dev_priv, pipe, PIPE_VBLANK_INTERRUPT_ENABLE);
|
psb_disable_pipestat(dev_priv, pipe, PIPE_VBLANK_INTERRUPT_ENABLE);
|
||||||
|
|
||||||
spin_unlock_irqrestore(&dev_priv->irqmask_lock, irqflags);
|
spin_unlock_irqrestore(&dev_priv->irqmask_lock, irqflags);
|
||||||
|
|
Loading…
Add table
Reference in a new issue