2006-01-02 20:14:23 +11:00
/* i915_irq.c -- IRQ support for the I915 -*- linux-c -*-
2005-04-16 15:20:36 -07:00
*/
2006-01-02 20:14:23 +11:00
/*
2005-04-16 15:20:36 -07:00
* Copyright 2003 Tungsten Graphics , Inc . , Cedar Park , Texas .
* All Rights Reserved .
2005-06-23 22:46:46 +10:00
*
* Permission is hereby granted , free of charge , to any person obtaining a
* copy of this software and associated documentation files ( the
* " Software " ) , to deal in the Software without restriction , including
* without limitation the rights to use , copy , modify , merge , publish ,
* distribute , sub license , and / or sell copies of the Software , and to
* permit persons to whom the Software is furnished to do so , subject to
* the following conditions :
*
* The above copyright notice and this permission notice ( including the
* next paragraph ) shall be included in all copies or substantial portions
* of the Software .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND , EXPRESS
* OR IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY , FITNESS FOR A PARTICULAR PURPOSE AND NON - INFRINGEMENT .
* IN NO EVENT SHALL TUNGSTEN GRAPHICS AND / OR ITS SUPPLIERS BE LIABLE FOR
* ANY CLAIM , DAMAGES OR OTHER LIABILITY , WHETHER IN AN ACTION OF CONTRACT ,
* TORT OR OTHERWISE , ARISING FROM , OUT OF OR IN CONNECTION WITH THE
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE .
*
2006-01-02 20:14:23 +11:00
*/
2005-04-16 15:20:36 -07:00
# include "drmP.h"
# include "drm.h"
# include "i915_drm.h"
# include "i915_drv.h"
# define MAX_NOPID ((u32)~0)
2008-11-04 02:03:27 -08:00
/**
* Interrupts that are always left unmasked .
*
* Since pipe events are edge - triggered from the PIPESTAT register to IIR ,
* we leave them always unmasked in IMR and then control enabling them through
* PIPESTAT alone .
*/
# define I915_INTERRUPT_ENABLE_FIX (I915_ASLE_INTERRUPT | \
I915_DISPLAY_PIPE_A_EVENT_INTERRUPT | \
I915_DISPLAY_PIPE_B_EVENT_INTERRUPT )
/** Interrupts that we mask and unmask at runtime. */
# define I915_INTERRUPT_ENABLE_VAR (I915_USER_INTERRUPT)
/** These are all of the interrupts used by the driver */
# define I915_INTERRUPT_ENABLE_MASK (I915_INTERRUPT_ENABLE_FIX | \
I915_INTERRUPT_ENABLE_VAR )
2008-07-29 12:10:39 -07:00
2008-08-05 19:37:25 +01:00
void
2008-07-29 12:10:39 -07:00
i915_enable_irq ( drm_i915_private_t * dev_priv , u32 mask )
{
if ( ( dev_priv - > irq_mask_reg & mask ) ! = 0 ) {
dev_priv - > irq_mask_reg & = ~ mask ;
I915_WRITE ( IMR , dev_priv - > irq_mask_reg ) ;
( void ) I915_READ ( IMR ) ;
}
}
static inline void
i915_disable_irq ( drm_i915_private_t * dev_priv , u32 mask )
{
if ( ( dev_priv - > irq_mask_reg & mask ) ! = mask ) {
dev_priv - > irq_mask_reg | = mask ;
I915_WRITE ( IMR , dev_priv - > irq_mask_reg ) ;
( void ) I915_READ ( IMR ) ;
}
}
2008-11-04 02:03:27 -08:00
static inline u32
i915_pipestat ( int pipe )
{
if ( pipe = = 0 )
return PIPEASTAT ;
if ( pipe = = 1 )
return PIPEBSTAT ;
BUG_ON ( 1 ) ;
}
void
i915_enable_pipestat ( drm_i915_private_t * dev_priv , int pipe , u32 mask )
{
if ( ( dev_priv - > pipestat [ pipe ] & mask ) ! = mask ) {
u32 reg = i915_pipestat ( pipe ) ;
dev_priv - > pipestat [ pipe ] | = mask ;
/* Enable the interrupt, clear any pending status */
I915_WRITE ( reg , dev_priv - > pipestat [ pipe ] | ( mask > > 16 ) ) ;
( void ) I915_READ ( reg ) ;
}
}
void
i915_disable_pipestat ( drm_i915_private_t * dev_priv , int pipe , u32 mask )
{
if ( ( dev_priv - > pipestat [ pipe ] & mask ) ! = 0 ) {
u32 reg = i915_pipestat ( pipe ) ;
dev_priv - > pipestat [ pipe ] & = ~ mask ;
I915_WRITE ( reg , dev_priv - > pipestat [ pipe ] ) ;
( void ) I915_READ ( reg ) ;
}
}
2008-09-30 12:14:26 -07:00
/**
* i915_pipe_enabled - check if a pipe is enabled
* @ dev : DRM device
* @ pipe : pipe to check
*
* Reading certain registers when the pipe is disabled can hang the chip .
* Use this routine to make sure the PLL is running and the pipe is active
* before reading such registers if unsure .
*/
static int
i915_pipe_enabled ( struct drm_device * dev , int pipe )
{
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
unsigned long pipeconf = pipe ? PIPEBCONF : PIPEACONF ;
if ( I915_READ ( pipeconf ) & PIPEACONF_ENABLE )
return 1 ;
return 0 ;
}
2008-10-18 19:39:29 -07:00
/* Called from drm generic code, passed a 'crtc', which
* we use as a pipe index
*/
u32 i915_get_vblank_counter ( struct drm_device * dev , int pipe )
2008-09-30 12:14:26 -07:00
{
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
unsigned long high_frame ;
unsigned long low_frame ;
u32 high1 , high2 , low , count ;
high_frame = pipe ? PIPEBFRAMEHIGH : PIPEAFRAMEHIGH ;
low_frame = pipe ? PIPEBFRAMEPIXEL : PIPEAFRAMEPIXEL ;
if ( ! i915_pipe_enabled ( dev , pipe ) ) {
DRM_ERROR ( " trying to get vblank count for disabled pipe %d \n " , pipe ) ;
return 0 ;
}
/*
* High & low register fields aren ' t synchronized , so make sure
* we get a low value that ' s stable across two reads of the high
* register .
*/
do {
high1 = ( ( I915_READ ( high_frame ) & PIPE_FRAME_HIGH_MASK ) > >
PIPE_FRAME_HIGH_SHIFT ) ;
low = ( ( I915_READ ( low_frame ) & PIPE_FRAME_LOW_MASK ) > >
PIPE_FRAME_LOW_SHIFT ) ;
high2 = ( ( I915_READ ( high_frame ) & PIPE_FRAME_HIGH_MASK ) > >
PIPE_FRAME_HIGH_SHIFT ) ;
} while ( high1 ! = high2 ) ;
count = ( high1 < < 8 ) | low ;
return count ;
}
2005-04-16 15:20:36 -07:00
irqreturn_t i915_driver_irq_handler ( DRM_IRQ_ARGS )
{
2007-07-11 15:53:27 +10:00
struct drm_device * dev = ( struct drm_device * ) arg ;
2005-04-16 15:20:36 -07:00
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
2008-11-04 15:50:30 -08:00
u32 iir , new_iir ;
u32 pipea_stats , pipeb_stats ;
2008-09-30 12:14:26 -07:00
int vblank = 0 ;
2008-11-04 02:03:27 -08:00
unsigned long irqflags ;
2006-03-20 18:34:29 +11:00
2008-10-06 15:14:12 -07:00
atomic_inc ( & dev_priv - > irq_received ) ;
2008-07-29 12:10:39 -07:00
iir = I915_READ ( IIR ) ;
2006-10-24 23:37:43 +10:00
2008-11-04 15:50:30 -08:00
if ( iir = = 0 )
2008-05-07 12:15:39 +10:00
return IRQ_NONE ;
2008-11-04 15:50:30 -08:00
do {
pipea_stats = 0 ;
pipeb_stats = 0 ;
/*
* Clear the PIPE ( A | B ) STAT regs before the IIR
*/
if ( iir & I915_DISPLAY_PIPE_A_EVENT_INTERRUPT ) {
spin_lock_irqsave ( & dev_priv - > user_irq_lock , irqflags ) ;
pipea_stats = I915_READ ( PIPEASTAT ) ;
I915_WRITE ( PIPEASTAT , pipea_stats ) ;
spin_unlock_irqrestore ( & dev_priv - > user_irq_lock ,
irqflags ) ;
}
2005-04-16 15:20:36 -07:00
2008-11-04 15:50:30 -08:00
if ( iir & I915_DISPLAY_PIPE_B_EVENT_INTERRUPT ) {
spin_lock_irqsave ( & dev_priv - > user_irq_lock , irqflags ) ;
pipeb_stats = I915_READ ( PIPEBSTAT ) ;
I915_WRITE ( PIPEBSTAT , pipeb_stats ) ;
spin_unlock_irqrestore ( & dev_priv - > user_irq_lock ,
irqflags ) ;
}
2008-08-05 19:37:25 +01:00
2008-11-04 15:50:30 -08:00
I915_WRITE ( IIR , iir ) ;
new_iir = I915_READ ( IIR ) ; /* Flush posted writes */
2008-11-04 02:03:27 -08:00
2008-11-04 15:50:30 -08:00
if ( dev_priv - > sarea_priv )
dev_priv - > sarea_priv - > last_dispatch =
READ_BREADCRUMB ( dev_priv ) ;
2008-09-30 12:14:26 -07:00
2008-11-04 15:50:30 -08:00
if ( iir & I915_USER_INTERRUPT ) {
dev_priv - > mm . irq_gem_seqno = i915_get_gem_seqno ( dev ) ;
DRM_WAKEUP ( & dev_priv - > irq_queue ) ;
}
2008-07-30 12:06:12 -07:00
2008-11-04 15:50:30 -08:00
if ( pipea_stats & I915_VBLANK_INTERRUPT_STATUS ) {
vblank + + ;
drm_handle_vblank ( dev , 0 ) ;
}
2008-11-04 02:03:27 -08:00
2008-11-04 15:50:30 -08:00
if ( pipeb_stats & I915_VBLANK_INTERRUPT_STATUS ) {
vblank + + ;
drm_handle_vblank ( dev , 1 ) ;
}
2008-11-04 02:03:27 -08:00
2008-11-04 15:50:30 -08:00
if ( ( pipeb_stats & I915_LEGACY_BLC_EVENT_STATUS ) | |
( iir & I915_ASLE_INTERRUPT ) )
opregion_asle_intr ( dev ) ;
/* With MSI, interrupts are only generated when iir
* transitions from zero to nonzero . If another bit got
* set while we were handling the existing iir bits , then
* we would never get another interrupt .
*
* This is fine on non - MSI as well , as if we hit this path
* we avoid exiting the interrupt handler only to generate
* another one .
*
* Note that for MSI this could cause a stray interrupt report
* if an interrupt landed in the time between writing IIR and
* the posting read . This should be rare enough to never
* trigger the 99 % of 100 , 000 interrupts test for disabling
* stray interrupts .
*/
iir = new_iir ;
} while ( iir ! = 0 ) ;
2008-09-30 12:14:26 -07:00
2005-04-16 15:20:36 -07:00
return IRQ_HANDLED ;
}
2008-05-07 12:15:39 +10:00
static int i915_emit_irq ( struct drm_device * dev )
2005-04-16 15:20:36 -07:00
{
drm_i915_private_t * dev_priv = dev - > dev_private ;
RING_LOCALS ;
i915_kernel_lost_context ( dev ) ;
2008-01-24 15:58:57 +10:00
DRM_DEBUG ( " \n " ) ;
2005-04-16 15:20:36 -07:00
2008-08-20 11:20:13 -04:00
dev_priv - > counter + + ;
2006-08-12 16:29:24 +10:00
if ( dev_priv - > counter > 0x7FFFFFFFUL )
2008-08-20 11:20:13 -04:00
dev_priv - > counter = 1 ;
if ( dev_priv - > sarea_priv )
dev_priv - > sarea_priv - > last_enqueue = dev_priv - > counter ;
2006-08-12 16:29:24 +10:00
2008-11-08 11:44:14 +10:00
BEGIN_LP_RING ( 4 ) ;
2008-07-29 11:54:06 -07:00
OUT_RING ( MI_STORE_DWORD_INDEX ) ;
2008-11-08 11:44:14 +10:00
OUT_RING ( I915_BREADCRUMB_INDEX < < MI_STORE_DWORD_INDEX_SHIFT ) ;
2006-08-12 16:29:24 +10:00
OUT_RING ( dev_priv - > counter ) ;
2008-07-29 11:54:06 -07:00
OUT_RING ( MI_USER_INTERRUPT ) ;
2005-04-16 15:20:36 -07:00
ADVANCE_LP_RING ( ) ;
2007-11-05 12:50:58 +10:00
2006-08-12 16:29:24 +10:00
return dev_priv - > counter ;
2005-04-16 15:20:36 -07:00
}
2008-07-30 12:06:12 -07:00
void i915_user_irq_get ( struct drm_device * dev )
2008-07-29 12:10:39 -07:00
{
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
2008-10-16 11:31:38 -07:00
unsigned long irqflags ;
2008-07-29 12:10:39 -07:00
2008-10-16 11:31:38 -07:00
spin_lock_irqsave ( & dev_priv - > user_irq_lock , irqflags ) ;
2008-07-29 12:10:39 -07:00
if ( dev - > irq_enabled & & ( + + dev_priv - > user_irq_refcount = = 1 ) )
i915_enable_irq ( dev_priv , I915_USER_INTERRUPT ) ;
2008-10-16 11:31:38 -07:00
spin_unlock_irqrestore ( & dev_priv - > user_irq_lock , irqflags ) ;
2008-07-29 12:10:39 -07:00
}
2008-09-30 12:14:26 -07:00
void i915_user_irq_put ( struct drm_device * dev )
2008-07-29 12:10:39 -07:00
{
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
2008-10-16 11:31:38 -07:00
unsigned long irqflags ;
2008-07-29 12:10:39 -07:00
2008-10-16 11:31:38 -07:00
spin_lock_irqsave ( & dev_priv - > user_irq_lock , irqflags ) ;
2008-07-29 12:10:39 -07:00
BUG_ON ( dev - > irq_enabled & & dev_priv - > user_irq_refcount < = 0 ) ;
if ( dev - > irq_enabled & & ( - - dev_priv - > user_irq_refcount = = 0 ) )
i915_disable_irq ( dev_priv , I915_USER_INTERRUPT ) ;
2008-10-16 11:31:38 -07:00
spin_unlock_irqrestore ( & dev_priv - > user_irq_lock , irqflags ) ;
2008-07-29 12:10:39 -07:00
}
2007-07-11 15:53:27 +10:00
static int i915_wait_irq ( struct drm_device * dev , int irq_nr )
2005-04-16 15:20:36 -07:00
{
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
int ret = 0 ;
2008-01-24 15:58:57 +10:00
DRM_DEBUG ( " irq_nr=%d breadcrumb=%d \n " , irq_nr ,
2005-04-16 15:20:36 -07:00
READ_BREADCRUMB ( dev_priv ) ) ;
2008-07-29 12:10:39 -07:00
if ( READ_BREADCRUMB ( dev_priv ) > = irq_nr ) {
2008-08-20 11:20:13 -04:00
if ( dev_priv - > sarea_priv ) {
dev_priv - > sarea_priv - > last_dispatch =
READ_BREADCRUMB ( dev_priv ) ;
}
2005-04-16 15:20:36 -07:00
return 0 ;
2008-07-29 12:10:39 -07:00
}
2005-04-16 15:20:36 -07:00
2008-08-20 11:20:13 -04:00
if ( dev_priv - > sarea_priv )
dev_priv - > sarea_priv - > perf_boxes | = I915_BOX_WAIT ;
2005-04-16 15:20:36 -07:00
2008-07-29 12:10:39 -07:00
i915_user_irq_get ( dev ) ;
2005-04-16 15:20:36 -07:00
DRM_WAIT_ON ( ret , dev_priv - > irq_queue , 3 * DRM_HZ ,
READ_BREADCRUMB ( dev_priv ) > = irq_nr ) ;
2008-07-29 12:10:39 -07:00
i915_user_irq_put ( dev ) ;
2005-04-16 15:20:36 -07:00
2007-08-25 19:22:43 +10:00
if ( ret = = - EBUSY ) {
2008-01-24 15:58:57 +10:00
DRM_ERROR ( " EBUSY -- rec: %d emitted: %d \n " ,
2005-04-16 15:20:36 -07:00
READ_BREADCRUMB ( dev_priv ) , ( int ) dev_priv - > counter ) ;
}
2008-08-20 11:20:13 -04:00
if ( dev_priv - > sarea_priv )
dev_priv - > sarea_priv - > last_dispatch =
READ_BREADCRUMB ( dev_priv ) ;
2008-05-07 12:15:39 +10:00
return ret ;
}
2005-04-16 15:20:36 -07:00
/* Needs the lock as it touches the ring.
*/
2007-09-03 12:06:45 +10:00
int i915_irq_emit ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
2005-04-16 15:20:36 -07:00
{
drm_i915_private_t * dev_priv = dev - > dev_private ;
2007-09-03 12:06:45 +10:00
drm_i915_irq_emit_t * emit = data ;
2005-04-16 15:20:36 -07:00
int result ;
2008-09-01 16:45:29 -07:00
RING_LOCK_TEST_WITH_RETURN ( dev , file_priv ) ;
2005-04-16 15:20:36 -07:00
if ( ! dev_priv ) {
2008-01-24 15:58:57 +10:00
DRM_ERROR ( " called with no initialization \n " ) ;
2007-08-25 19:22:43 +10:00
return - EINVAL ;
2005-04-16 15:20:36 -07:00
}
2008-09-01 16:45:29 -07:00
mutex_lock ( & dev - > struct_mutex ) ;
2005-04-16 15:20:36 -07:00
result = i915_emit_irq ( dev ) ;
2008-09-01 16:45:29 -07:00
mutex_unlock ( & dev - > struct_mutex ) ;
2005-04-16 15:20:36 -07:00
2007-09-03 12:06:45 +10:00
if ( DRM_COPY_TO_USER ( emit - > irq_seq , & result , sizeof ( int ) ) ) {
2005-04-16 15:20:36 -07:00
DRM_ERROR ( " copy_to_user \n " ) ;
2007-08-25 19:22:43 +10:00
return - EFAULT ;
2005-04-16 15:20:36 -07:00
}
return 0 ;
}
/* Doesn't need the hardware lock.
*/
2007-09-03 12:06:45 +10:00
int i915_irq_wait ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
2005-04-16 15:20:36 -07:00
{
drm_i915_private_t * dev_priv = dev - > dev_private ;
2007-09-03 12:06:45 +10:00
drm_i915_irq_wait_t * irqwait = data ;
2005-04-16 15:20:36 -07:00
if ( ! dev_priv ) {
2008-01-24 15:58:57 +10:00
DRM_ERROR ( " called with no initialization \n " ) ;
2007-08-25 19:22:43 +10:00
return - EINVAL ;
2005-04-16 15:20:36 -07:00
}
2007-09-03 12:06:45 +10:00
return i915_wait_irq ( dev , irqwait - > irq_seq ) ;
2005-04-16 15:20:36 -07:00
}
2008-10-18 19:39:29 -07:00
/* Called from drm generic code, passed 'crtc' which
* we use as a pipe index
*/
int i915_enable_vblank ( struct drm_device * dev , int pipe )
2008-09-30 12:14:26 -07:00
{
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
2008-10-16 11:31:38 -07:00
unsigned long irqflags ;
2008-09-30 12:14:26 -07:00
2008-10-16 11:31:38 -07:00
spin_lock_irqsave ( & dev_priv - > user_irq_lock , irqflags ) ;
if ( IS_I965G ( dev ) )
2008-11-04 02:03:27 -08:00
i915_enable_pipestat ( dev_priv , pipe ,
PIPE_START_VBLANK_INTERRUPT_ENABLE ) ;
2008-10-16 11:31:38 -07:00
else
2008-11-04 02:03:27 -08:00
i915_enable_pipestat ( dev_priv , pipe ,
PIPE_VBLANK_INTERRUPT_ENABLE ) ;
2008-10-16 11:31:38 -07:00
spin_unlock_irqrestore ( & dev_priv - > user_irq_lock , irqflags ) ;
2008-09-30 12:14:26 -07:00
return 0 ;
}
2008-10-18 19:39:29 -07:00
/* Called from drm generic code, passed 'crtc' which
* we use as a pipe index
*/
void i915_disable_vblank ( struct drm_device * dev , int pipe )
2008-09-30 12:14:26 -07:00
{
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
2008-10-16 11:31:38 -07:00
unsigned long irqflags ;
2008-09-30 12:14:26 -07:00
2008-10-16 11:31:38 -07:00
spin_lock_irqsave ( & dev_priv - > user_irq_lock , irqflags ) ;
2008-11-04 02:03:27 -08:00
i915_disable_pipestat ( dev_priv , pipe ,
PIPE_VBLANK_INTERRUPT_ENABLE |
PIPE_START_VBLANK_INTERRUPT_ENABLE ) ;
2008-10-16 11:31:38 -07:00
spin_unlock_irqrestore ( & dev_priv - > user_irq_lock , irqflags ) ;
2008-09-30 12:14:26 -07:00
}
2006-06-24 17:07:34 +10:00
/* Set the vblank monitor pipe
*/
2007-09-03 12:06:45 +10:00
int i915_vblank_pipe_set ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
2006-06-24 17:07:34 +10:00
{
drm_i915_private_t * dev_priv = dev - > dev_private ;
if ( ! dev_priv ) {
2008-01-24 15:58:57 +10:00
DRM_ERROR ( " called with no initialization \n " ) ;
2007-08-25 19:22:43 +10:00
return - EINVAL ;
2006-06-24 17:07:34 +10:00
}
2006-10-25 00:08:23 +10:00
return 0 ;
2006-06-24 17:07:34 +10:00
}
2007-09-03 12:06:45 +10:00
int i915_vblank_pipe_get ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
2006-06-24 17:07:34 +10:00
{
drm_i915_private_t * dev_priv = dev - > dev_private ;
2007-09-03 12:06:45 +10:00
drm_i915_vblank_pipe_t * pipe = data ;
2006-06-24 17:07:34 +10:00
if ( ! dev_priv ) {
2008-01-24 15:58:57 +10:00
DRM_ERROR ( " called with no initialization \n " ) ;
2007-08-25 19:22:43 +10:00
return - EINVAL ;
2006-06-24 17:07:34 +10:00
}
2008-09-30 12:14:26 -07:00
pipe - > pipe = DRM_I915_VBLANK_PIPE_A | DRM_I915_VBLANK_PIPE_B ;
2007-09-03 12:06:45 +10:00
2006-06-24 17:07:34 +10:00
return 0 ;
}
2006-10-24 23:37:43 +10:00
/**
* Schedule buffer swap at given vertical blank .
*/
2007-09-03 12:06:45 +10:00
int i915_vblank_swap ( struct drm_device * dev , void * data ,
struct drm_file * file_priv )
2006-10-24 23:37:43 +10:00
{
2008-11-04 12:01:24 -08:00
/* The delayed swap mechanism was fundamentally racy, and has been
* removed . The model was that the client requested a delayed flip / swap
* from the kernel , then waited for vblank before continuing to perform
* rendering . The problem was that the kernel might wake the client
* up before it dispatched the vblank swap ( since the lock has to be
* held while touching the ringbuffer ) , in which case the client would
* clear and start the next frame before the swap occurred , and
* flicker would occur in addition to likely missing the vblank .
*
* In the absence of this ioctl , userland falls back to a correct path
* of waiting for a vblank , then dispatching the swap on its own .
* Context switching to userland and back is plenty fast enough for
* meeting the requirements of vblank swapping .
2008-09-30 12:14:26 -07:00
*/
2008-11-04 12:01:24 -08:00
return - EINVAL ;
2006-10-24 23:37:43 +10:00
}
2005-04-16 15:20:36 -07:00
/* drm_dma.h hooks
*/
2007-07-11 15:53:27 +10:00
void i915_driver_irq_preinstall ( struct drm_device * dev )
2005-04-16 15:20:36 -07:00
{
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
2008-09-30 12:14:26 -07:00
I915_WRITE ( HWSTAM , 0xeffe ) ;
2008-11-04 02:03:27 -08:00
I915_WRITE ( PIPEASTAT , 0 ) ;
I915_WRITE ( PIPEBSTAT , 0 ) ;
2008-09-30 12:14:26 -07:00
I915_WRITE ( IMR , 0xffffffff ) ;
2008-07-29 12:10:39 -07:00
I915_WRITE ( IER , 0x0 ) ;
2008-11-04 02:03:27 -08:00
( void ) I915_READ ( IER ) ;
2005-04-16 15:20:36 -07:00
}
2008-09-30 12:14:26 -07:00
int i915_driver_irq_postinstall ( struct drm_device * dev )
2005-04-16 15:20:36 -07:00
{
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
2008-09-30 12:14:26 -07:00
int ret , num_pipes = 2 ;
2005-04-16 15:20:36 -07:00
2008-09-30 12:14:26 -07:00
ret = drm_vblank_init ( dev , num_pipes ) ;
if ( ret )
return ret ;
dev_priv - > vblank_pipe = DRM_I915_VBLANK_PIPE_A | DRM_I915_VBLANK_PIPE_B ;
dev - > max_vblank_count = 0xffffff ; /* only 24 bits of frame count */
2008-07-29 12:10:39 -07:00
2008-11-04 02:03:27 -08:00
/* Unmask the interrupts that we always want on. */
dev_priv - > irq_mask_reg = ~ I915_INTERRUPT_ENABLE_FIX ;
dev_priv - > pipestat [ 0 ] = 0 ;
dev_priv - > pipestat [ 1 ] = 0 ;
/* Disable pipe interrupt enables, clear pending pipe status */
I915_WRITE ( PIPEASTAT , I915_READ ( PIPEASTAT ) & 0x8000ffff ) ;
I915_WRITE ( PIPEBSTAT , I915_READ ( PIPEBSTAT ) & 0x8000ffff ) ;
/* Clear pending interrupt status */
I915_WRITE ( IIR , I915_READ ( IIR ) ) ;
2008-08-05 19:37:25 +01:00
2008-07-29 12:10:39 -07:00
I915_WRITE ( IER , I915_INTERRUPT_ENABLE_MASK ) ;
2008-11-04 02:03:27 -08:00
I915_WRITE ( IMR , dev_priv - > irq_mask_reg ) ;
2008-07-29 12:10:39 -07:00
( void ) I915_READ ( IER ) ;
2008-08-05 19:37:25 +01:00
opregion_enable_asle ( dev ) ;
2005-04-16 15:20:36 -07:00
DRM_INIT_WAITQUEUE ( & dev_priv - > irq_queue ) ;
2008-09-30 12:14:26 -07:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2007-07-11 15:53:27 +10:00
void i915_driver_irq_uninstall ( struct drm_device * dev )
2005-04-16 15:20:36 -07:00
{
drm_i915_private_t * dev_priv = ( drm_i915_private_t * ) dev - > dev_private ;
2006-02-18 15:17:04 +11:00
2005-04-16 15:20:36 -07:00
if ( ! dev_priv )
return ;
2008-09-30 12:14:26 -07:00
dev_priv - > vblank_pipe = 0 ;
I915_WRITE ( HWSTAM , 0xffffffff ) ;
2008-11-04 02:03:27 -08:00
I915_WRITE ( PIPEASTAT , 0 ) ;
I915_WRITE ( PIPEBSTAT , 0 ) ;
2008-09-30 12:14:26 -07:00
I915_WRITE ( IMR , 0xffffffff ) ;
2008-07-29 12:10:39 -07:00
I915_WRITE ( IER , 0x0 ) ;
2008-05-07 12:15:39 +10:00
2008-11-04 02:03:27 -08:00
I915_WRITE ( PIPEASTAT , I915_READ ( PIPEASTAT ) & 0x8000ffff ) ;
I915_WRITE ( PIPEBSTAT , I915_READ ( PIPEBSTAT ) & 0x8000ffff ) ;
I915_WRITE ( IIR , I915_READ ( IIR ) ) ;
2005-04-16 15:20:36 -07:00
}