2005-04-17 02:20:36 +04:00
/*
* intelfb
*
* Linux framebuffer driver for Intel ( R ) 865 G integrated graphics chips .
*
2007-10-20 01:21:04 +04:00
* Copyright © 2002 , 2003 David Dawes < dawes @ xfree86 . org >
2005-04-17 02:20:36 +04:00
* 2004 Sylvain Meyer
*
* This driver consists of two parts . The first part ( intelfbdrv . c ) provides
* the basic fbdev interfaces , is derived in part from the radeonfb and
* vesafb drivers , and is covered by the GPL . The second part ( intelfbhw . c )
* provides the code to program the hardware . Most of it is derived from
* the i810 / i830 XFree86 driver . The HW - specific code is covered here
* under a dual license ( GPL and MIT / XFree86 license ) .
*
* Author : David Dawes
*
*/
/* $DHD: intelfb/intelfbhw.c,v 1.9 2003/06/27 15:06:25 dawes Exp $ */
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/string.h>
# include <linux/mm.h>
# include <linux/slab.h>
# include <linux/delay.h>
# include <linux/fb.h>
# include <linux/ioport.h>
# include <linux/init.h>
# include <linux/pci.h>
# include <linux/vmalloc.h>
# include <linux/pagemap.h>
2006-06-20 22:36:41 +04:00
# include <linux/interrupt.h>
2005-04-17 02:20:36 +04:00
# include <asm/io.h>
# include "intelfb.h"
# include "intelfbhw.h"
2006-03-20 12:02:24 +03:00
struct pll_min_max {
2006-04-03 10:19:26 +04:00
int min_m , max_m , min_m1 , max_m1 ;
int min_m2 , max_m2 , min_n , max_n ;
int min_p , max_p , min_p1 , max_p1 ;
int min_vco , max_vco , p_transition_clk , ref_clk ;
2006-03-20 13:22:09 +03:00
int p_inc_lo , p_inc_hi ;
2006-03-20 12:02:24 +03:00
} ;
# define PLLS_I8xx 0
# define PLLS_I9xx 1
# define PLLS_MAX 2
2006-03-24 04:31:14 +03:00
static struct pll_min_max plls [ PLLS_MAX ] = {
2006-04-03 10:19:26 +04:00
{ 108 , 140 , 18 , 26 ,
6 , 16 , 3 , 16 ,
4 , 128 , 0 , 31 ,
930000 , 1400000 , 165000 , 48000 ,
2007-10-16 12:29:31 +04:00
4 , 2 } , /* I8xx */
2006-04-03 10:19:26 +04:00
{ 75 , 120 , 10 , 20 ,
5 , 9 , 4 , 7 ,
5 , 80 , 1 , 8 ,
1400000 , 2800000 , 200000 , 96000 ,
2007-10-16 12:29:31 +04:00
10 , 5 } /* I9xx */
2006-03-20 12:02:24 +03:00
} ;
2007-10-16 12:29:31 +04:00
int intelfbhw_get_chipset ( struct pci_dev * pdev , struct intelfb_info * dinfo )
2005-04-17 02:20:36 +04:00
{
u32 tmp ;
2006-03-20 12:26:45 +03:00
if ( ! pdev | | ! dinfo )
2005-04-17 02:20:36 +04:00
return 1 ;
switch ( pdev - > device ) {
case PCI_DEVICE_ID_INTEL_830M :
2006-03-20 12:26:45 +03:00
dinfo - > name = " Intel(R) 830M " ;
dinfo - > chipset = INTEL_830M ;
dinfo - > mobile = 1 ;
dinfo - > pll_index = PLLS_I8xx ;
2005-04-17 02:20:36 +04:00
return 0 ;
case PCI_DEVICE_ID_INTEL_845G :
2006-03-20 12:26:45 +03:00
dinfo - > name = " Intel(R) 845G " ;
dinfo - > chipset = INTEL_845G ;
dinfo - > mobile = 0 ;
dinfo - > pll_index = PLLS_I8xx ;
2005-04-17 02:20:36 +04:00
return 0 ;
case PCI_DEVICE_ID_INTEL_85XGM :
tmp = 0 ;
2006-03-20 12:26:45 +03:00
dinfo - > mobile = 1 ;
dinfo - > pll_index = PLLS_I8xx ;
2005-04-17 02:20:36 +04:00
pci_read_config_dword ( pdev , INTEL_85X_CAPID , & tmp ) ;
switch ( ( tmp > > INTEL_85X_VARIANT_SHIFT ) &
INTEL_85X_VARIANT_MASK ) {
case INTEL_VAR_855GME :
2006-03-20 12:26:45 +03:00
dinfo - > name = " Intel(R) 855GME " ;
dinfo - > chipset = INTEL_855GME ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_VAR_855GM :
2006-03-20 12:26:45 +03:00
dinfo - > name = " Intel(R) 855GM " ;
dinfo - > chipset = INTEL_855GM ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_VAR_852GME :
2006-03-20 12:26:45 +03:00
dinfo - > name = " Intel(R) 852GME " ;
dinfo - > chipset = INTEL_852GME ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_VAR_852GM :
2006-03-20 12:26:45 +03:00
dinfo - > name = " Intel(R) 852GM " ;
dinfo - > chipset = INTEL_852GM ;
2005-04-17 02:20:36 +04:00
return 0 ;
default :
2006-03-20 12:26:45 +03:00
dinfo - > name = " Intel(R) 852GM/855GM " ;
dinfo - > chipset = INTEL_85XGM ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
break ;
case PCI_DEVICE_ID_INTEL_865G :
2006-03-20 12:26:45 +03:00
dinfo - > name = " Intel(R) 865G " ;
dinfo - > chipset = INTEL_865G ;
dinfo - > mobile = 0 ;
dinfo - > pll_index = PLLS_I8xx ;
2005-04-17 02:20:36 +04:00
return 0 ;
case PCI_DEVICE_ID_INTEL_915G :
2006-03-20 12:26:45 +03:00
dinfo - > name = " Intel(R) 915G " ;
dinfo - > chipset = INTEL_915G ;
dinfo - > mobile = 0 ;
dinfo - > pll_index = PLLS_I9xx ;
2005-04-17 02:20:36 +04:00
return 0 ;
2005-11-07 12:00:33 +03:00
case PCI_DEVICE_ID_INTEL_915GM :
2006-03-20 12:26:45 +03:00
dinfo - > name = " Intel(R) 915GM " ;
dinfo - > chipset = INTEL_915GM ;
dinfo - > mobile = 1 ;
dinfo - > pll_index = PLLS_I9xx ;
2005-11-07 12:00:33 +03:00
return 0 ;
2006-03-23 03:23:55 +03:00
case PCI_DEVICE_ID_INTEL_945G :
dinfo - > name = " Intel(R) 945G " ;
dinfo - > chipset = INTEL_945G ;
dinfo - > mobile = 0 ;
dinfo - > pll_index = PLLS_I9xx ;
return 0 ;
2006-03-23 13:53:05 +03:00
case PCI_DEVICE_ID_INTEL_945GM :
dinfo - > name = " Intel(R) 945GM " ;
dinfo - > chipset = INTEL_945GM ;
dinfo - > mobile = 1 ;
dinfo - > pll_index = PLLS_I9xx ;
return 0 ;
2005-04-17 02:20:36 +04:00
default :
return 1 ;
}
}
2007-10-16 12:29:31 +04:00
int intelfbhw_get_memory ( struct pci_dev * pdev , int * aperture_size ,
int * stolen_size )
2005-04-17 02:20:36 +04:00
{
struct pci_dev * bridge_dev ;
u16 tmp ;
2006-05-27 12:30:00 +04:00
int stolen_overhead ;
2005-04-17 02:20:36 +04:00
if ( ! pdev | | ! aperture_size | | ! stolen_size )
return 1 ;
/* Find the bridge device. It is always 0:0.0 */
2006-10-21 01:36:00 +04:00
if ( ! ( bridge_dev = pci_get_bus_and_slot ( 0 , PCI_DEVFN ( 0 , 0 ) ) ) ) {
2005-04-17 02:20:36 +04:00
ERR_MSG ( " cannot find bridge device \n " ) ;
return 1 ;
}
/* Get the fb aperture size and "stolen" memory amount. */
tmp = 0 ;
pci_read_config_word ( bridge_dev , INTEL_GMCH_CTRL , & tmp ) ;
2006-10-21 01:36:00 +04:00
pci_dev_put ( bridge_dev ) ;
2005-04-17 02:20:36 +04:00
switch ( pdev - > device ) {
2006-05-27 12:30:00 +04:00
case PCI_DEVICE_ID_INTEL_915G :
case PCI_DEVICE_ID_INTEL_915GM :
case PCI_DEVICE_ID_INTEL_945G :
case PCI_DEVICE_ID_INTEL_945GM :
/* 915 and 945 chipsets support a 256MB aperture.
Aperture size is determined by inspected the
base address of the aperture . */
if ( pci_resource_start ( pdev , 2 ) & 0x08000000 )
* aperture_size = MB ( 128 ) ;
else
* aperture_size = MB ( 256 ) ;
break ;
default :
2005-04-17 02:20:36 +04:00
if ( ( tmp & INTEL_GMCH_MEM_MASK ) = = INTEL_GMCH_MEM_64M )
* aperture_size = MB ( 64 ) ;
else
* aperture_size = MB ( 128 ) ;
2006-05-27 12:30:00 +04:00
break ;
}
/* Stolen memory size is reduced by the GTT and the popup.
GTT is 1 K per MB of aperture size , and popup is 4 K . */
stolen_overhead = ( * aperture_size / MB ( 1 ) ) + 4 ;
switch ( pdev - > device ) {
case PCI_DEVICE_ID_INTEL_830M :
case PCI_DEVICE_ID_INTEL_845G :
2005-04-17 02:20:36 +04:00
switch ( tmp & INTEL_830_GMCH_GMS_MASK ) {
case INTEL_830_GMCH_GMS_STOLEN_512 :
2006-05-27 12:30:00 +04:00
* stolen_size = KB ( 512 ) - KB ( stolen_overhead ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_830_GMCH_GMS_STOLEN_1024 :
2006-05-27 12:30:00 +04:00
* stolen_size = MB ( 1 ) - KB ( stolen_overhead ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_830_GMCH_GMS_STOLEN_8192 :
2006-05-27 12:30:00 +04:00
* stolen_size = MB ( 8 ) - KB ( stolen_overhead ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_830_GMCH_GMS_LOCAL :
ERR_MSG ( " only local memory found \n " ) ;
return 1 ;
case INTEL_830_GMCH_GMS_DISABLED :
ERR_MSG ( " video memory is disabled \n " ) ;
return 1 ;
default :
ERR_MSG ( " unexpected GMCH_GMS value: 0x%02x \n " ,
tmp & INTEL_830_GMCH_GMS_MASK ) ;
return 1 ;
}
break ;
default :
switch ( tmp & INTEL_855_GMCH_GMS_MASK ) {
case INTEL_855_GMCH_GMS_STOLEN_1M :
2006-05-27 12:30:00 +04:00
* stolen_size = MB ( 1 ) - KB ( stolen_overhead ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_855_GMCH_GMS_STOLEN_4M :
2006-05-27 12:30:00 +04:00
* stolen_size = MB ( 4 ) - KB ( stolen_overhead ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_855_GMCH_GMS_STOLEN_8M :
2006-05-27 12:30:00 +04:00
* stolen_size = MB ( 8 ) - KB ( stolen_overhead ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_855_GMCH_GMS_STOLEN_16M :
2006-05-27 12:30:00 +04:00
* stolen_size = MB ( 16 ) - KB ( stolen_overhead ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_855_GMCH_GMS_STOLEN_32M :
2006-05-27 12:30:00 +04:00
* stolen_size = MB ( 32 ) - KB ( stolen_overhead ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_915G_GMCH_GMS_STOLEN_48M :
2006-05-27 12:30:00 +04:00
* stolen_size = MB ( 48 ) - KB ( stolen_overhead ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_915G_GMCH_GMS_STOLEN_64M :
2006-05-27 12:30:00 +04:00
* stolen_size = MB ( 64 ) - KB ( stolen_overhead ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
case INTEL_855_GMCH_GMS_DISABLED :
ERR_MSG ( " video memory is disabled \n " ) ;
return 0 ;
default :
ERR_MSG ( " unexpected GMCH_GMS value: 0x%02x \n " ,
tmp & INTEL_855_GMCH_GMS_MASK ) ;
return 1 ;
}
}
}
2007-10-16 12:29:31 +04:00
int intelfbhw_check_non_crt ( struct intelfb_info * dinfo )
2005-04-17 02:20:36 +04:00
{
int dvo = 0 ;
if ( INREG ( LVDS ) & PORT_ENABLE )
dvo | = LVDS_PORT ;
if ( INREG ( DVOA ) & PORT_ENABLE )
dvo | = DVOA_PORT ;
if ( INREG ( DVOB ) & PORT_ENABLE )
dvo | = DVOB_PORT ;
if ( INREG ( DVOC ) & PORT_ENABLE )
dvo | = DVOC_PORT ;
return dvo ;
}
2007-10-16 12:29:31 +04:00
const char * intelfbhw_dvo_to_string ( int dvo )
2005-04-17 02:20:36 +04:00
{
if ( dvo & DVOA_PORT )
return " DVO port A " ;
else if ( dvo & DVOB_PORT )
return " DVO port B " ;
else if ( dvo & DVOC_PORT )
return " DVO port C " ;
else if ( dvo & LVDS_PORT )
return " LVDS port " ;
else
return NULL ;
}
2007-10-16 12:29:31 +04:00
int intelfbhw_validate_mode ( struct intelfb_info * dinfo ,
struct fb_var_screeninfo * var )
2005-04-17 02:20:36 +04:00
{
int bytes_per_pixel ;
int tmp ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_validate_mode \n " ) ;
# endif
bytes_per_pixel = var - > bits_per_pixel / 8 ;
if ( bytes_per_pixel = = 3 )
bytes_per_pixel = 4 ;
/* Check if enough video memory. */
tmp = var - > yres_virtual * var - > xres_virtual * bytes_per_pixel ;
if ( tmp > dinfo - > fb . size ) {
WRN_MSG ( " Not enough video ram for mode "
" (%d KByte vs %d KByte). \n " ,
BtoKB ( tmp ) , BtoKB ( dinfo - > fb . size ) ) ;
return 1 ;
}
/* Check if x/y limits are OK. */
if ( var - > xres - 1 > HACTIVE_MASK ) {
WRN_MSG ( " X resolution too large (%d vs %d). \n " ,
var - > xres , HACTIVE_MASK + 1 ) ;
return 1 ;
}
if ( var - > yres - 1 > VACTIVE_MASK ) {
WRN_MSG ( " Y resolution too large (%d vs %d). \n " ,
var - > yres , VACTIVE_MASK + 1 ) ;
return 1 ;
}
2007-10-16 12:29:33 +04:00
if ( var - > xres < 4 ) {
WRN_MSG ( " X resolution too small (%d vs 4). \n " , var - > xres ) ;
return 1 ;
}
if ( var - > yres < 4 ) {
WRN_MSG ( " Y resolution too small (%d vs 4). \n " , var - > yres ) ;
return 1 ;
}
2005-04-17 02:20:36 +04:00
2007-10-16 12:29:18 +04:00
/* Check for doublescan modes. */
2005-04-17 02:20:36 +04:00
if ( var - > vmode & FB_VMODE_DOUBLE ) {
WRN_MSG ( " Mode is double-scan. \n " ) ;
return 1 ;
}
2007-10-16 12:29:33 +04:00
if ( ( var - > vmode & FB_VMODE_INTERLACED ) & & ( var - > yres & 1 ) ) {
WRN_MSG ( " Odd number of lines in interlaced mode \n " ) ;
return 1 ;
}
2005-04-17 02:20:36 +04:00
/* Check if clock is OK. */
tmp = 1000000000 / var - > pixclock ;
if ( tmp < MIN_CLOCK ) {
WRN_MSG ( " Pixel clock is too low (%d MHz vs %d MHz). \n " ,
( tmp + 500 ) / 1000 , MIN_CLOCK / 1000 ) ;
return 1 ;
}
if ( tmp > MAX_CLOCK ) {
WRN_MSG ( " Pixel clock is too high (%d MHz vs %d MHz). \n " ,
( tmp + 500 ) / 1000 , MAX_CLOCK / 1000 ) ;
return 1 ;
}
return 0 ;
}
2007-10-16 12:29:31 +04:00
int intelfbhw_pan_display ( struct fb_var_screeninfo * var , struct fb_info * info )
2005-04-17 02:20:36 +04:00
{
struct intelfb_info * dinfo = GET_DINFO ( info ) ;
u32 offset , xoffset , yoffset ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_pan_display \n " ) ;
# endif
xoffset = ROUND_DOWN_TO ( var - > xoffset , 8 ) ;
yoffset = var - > yoffset ;
if ( ( xoffset + var - > xres > var - > xres_virtual ) | |
( yoffset + var - > yres > var - > yres_virtual ) )
return - EINVAL ;
offset = ( yoffset * dinfo - > pitch ) +
( xoffset * var - > bits_per_pixel ) / 8 ;
offset + = dinfo - > fb . offset < < 12 ;
2006-06-20 22:36:42 +04:00
dinfo - > vsync . pan_offset = offset ;
2007-10-16 12:29:31 +04:00
if ( ( var - > activate & FB_ACTIVATE_VBL ) & &
2007-10-16 12:29:34 +04:00
! intelfbhw_enable_irq ( dinfo ) )
2006-06-20 22:36:42 +04:00
dinfo - > vsync . pan_display = 1 ;
2007-10-16 12:29:31 +04:00
else {
2006-06-20 22:36:42 +04:00
dinfo - > vsync . pan_display = 0 ;
OUTREG ( DSPABASE , offset ) ;
}
2005-04-17 02:20:36 +04:00
return 0 ;
}
/* Blank the screen. */
2007-10-16 12:29:31 +04:00
void intelfbhw_do_blank ( int blank , struct fb_info * info )
2005-04-17 02:20:36 +04:00
{
struct intelfb_info * dinfo = GET_DINFO ( info ) ;
u32 tmp ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_do_blank: blank is %d \n " , blank ) ;
# endif
/* Turn plane A on or off */
tmp = INREG ( DSPACNTR ) ;
if ( blank )
tmp & = ~ DISPPLANE_PLANE_ENABLE ;
else
tmp | = DISPPLANE_PLANE_ENABLE ;
OUTREG ( DSPACNTR , tmp ) ;
/* Flush */
tmp = INREG ( DSPABASE ) ;
OUTREG ( DSPABASE , tmp ) ;
/* Turn off/on the HW cursor */
# if VERBOSE > 0
DBG_MSG ( " cursor_on is %d \n " , dinfo - > cursor_on ) ;
# endif
if ( dinfo - > cursor_on ) {
2007-10-16 12:29:31 +04:00
if ( blank )
2005-04-17 02:20:36 +04:00
intelfbhw_cursor_hide ( dinfo ) ;
2007-10-16 12:29:31 +04:00
else
2005-04-17 02:20:36 +04:00
intelfbhw_cursor_show ( dinfo ) ;
dinfo - > cursor_on = 1 ;
}
dinfo - > cursor_blanked = blank ;
/* Set DPMS level */
tmp = INREG ( ADPA ) & ~ ADPA_DPMS_CONTROL_MASK ;
switch ( blank ) {
case FB_BLANK_UNBLANK :
case FB_BLANK_NORMAL :
tmp | = ADPA_DPMS_D0 ;
break ;
case FB_BLANK_VSYNC_SUSPEND :
tmp | = ADPA_DPMS_D1 ;
break ;
case FB_BLANK_HSYNC_SUSPEND :
tmp | = ADPA_DPMS_D2 ;
break ;
case FB_BLANK_POWERDOWN :
tmp | = ADPA_DPMS_D3 ;
break ;
}
OUTREG ( ADPA , tmp ) ;
return ;
}
2007-10-16 12:29:31 +04:00
void intelfbhw_setcolreg ( struct intelfb_info * dinfo , unsigned regno ,
unsigned red , unsigned green , unsigned blue ,
unsigned transp )
2005-04-17 02:20:36 +04:00
{
2007-10-16 12:29:33 +04:00
u32 palette_reg = ( dinfo - > pipe = = PIPE_A ) ?
PALETTE_A : PALETTE_B ;
2005-04-17 02:20:36 +04:00
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_setcolreg: %d: (%d, %d, %d) \n " ,
regno , red , green , blue ) ;
# endif
OUTREG ( palette_reg + ( regno < < 2 ) ,
( red < < PALETTE_8_RED_SHIFT ) |
( green < < PALETTE_8_GREEN_SHIFT ) |
( blue < < PALETTE_8_BLUE_SHIFT ) ) ;
}
2007-10-16 12:29:31 +04:00
int intelfbhw_read_hw_state ( struct intelfb_info * dinfo ,
struct intelfb_hwstate * hw , int flag )
2005-04-17 02:20:36 +04:00
{
int i ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_read_hw_state \n " ) ;
# endif
if ( ! hw | | ! dinfo )
return - 1 ;
/* Read in as much of the HW state as possible. */
hw - > vga0_divisor = INREG ( VGA0_DIVISOR ) ;
hw - > vga1_divisor = INREG ( VGA1_DIVISOR ) ;
hw - > vga_pd = INREG ( VGAPD ) ;
hw - > dpll_a = INREG ( DPLL_A ) ;
hw - > dpll_b = INREG ( DPLL_B ) ;
hw - > fpa0 = INREG ( FPA0 ) ;
hw - > fpa1 = INREG ( FPA1 ) ;
hw - > fpb0 = INREG ( FPB0 ) ;
hw - > fpb1 = INREG ( FPB1 ) ;
if ( flag = = 1 )
return flag ;
#if 0
/* This seems to be a problem with the 852GM/855GM */
for ( i = 0 ; i < PALETTE_8_ENTRIES ; i + + ) {
hw - > palette_a [ i ] = INREG ( PALETTE_A + ( i < < 2 ) ) ;
hw - > palette_b [ i ] = INREG ( PALETTE_B + ( i < < 2 ) ) ;
}
# endif
if ( flag = = 2 )
return flag ;
hw - > htotal_a = INREG ( HTOTAL_A ) ;
hw - > hblank_a = INREG ( HBLANK_A ) ;
hw - > hsync_a = INREG ( HSYNC_A ) ;
hw - > vtotal_a = INREG ( VTOTAL_A ) ;
hw - > vblank_a = INREG ( VBLANK_A ) ;
hw - > vsync_a = INREG ( VSYNC_A ) ;
hw - > src_size_a = INREG ( SRC_SIZE_A ) ;
hw - > bclrpat_a = INREG ( BCLRPAT_A ) ;
hw - > htotal_b = INREG ( HTOTAL_B ) ;
hw - > hblank_b = INREG ( HBLANK_B ) ;
hw - > hsync_b = INREG ( HSYNC_B ) ;
hw - > vtotal_b = INREG ( VTOTAL_B ) ;
hw - > vblank_b = INREG ( VBLANK_B ) ;
hw - > vsync_b = INREG ( VSYNC_B ) ;
hw - > src_size_b = INREG ( SRC_SIZE_B ) ;
hw - > bclrpat_b = INREG ( BCLRPAT_B ) ;
if ( flag = = 3 )
return flag ;
hw - > adpa = INREG ( ADPA ) ;
hw - > dvoa = INREG ( DVOA ) ;
hw - > dvob = INREG ( DVOB ) ;
hw - > dvoc = INREG ( DVOC ) ;
hw - > dvoa_srcdim = INREG ( DVOA_SRCDIM ) ;
hw - > dvob_srcdim = INREG ( DVOB_SRCDIM ) ;
hw - > dvoc_srcdim = INREG ( DVOC_SRCDIM ) ;
hw - > lvds = INREG ( LVDS ) ;
if ( flag = = 4 )
return flag ;
hw - > pipe_a_conf = INREG ( PIPEACONF ) ;
hw - > pipe_b_conf = INREG ( PIPEBCONF ) ;
hw - > disp_arb = INREG ( DISPARB ) ;
if ( flag = = 5 )
return flag ;
hw - > cursor_a_control = INREG ( CURSOR_A_CONTROL ) ;
hw - > cursor_b_control = INREG ( CURSOR_B_CONTROL ) ;
hw - > cursor_a_base = INREG ( CURSOR_A_BASEADDR ) ;
hw - > cursor_b_base = INREG ( CURSOR_B_BASEADDR ) ;
if ( flag = = 6 )
return flag ;
for ( i = 0 ; i < 4 ; i + + ) {
hw - > cursor_a_palette [ i ] = INREG ( CURSOR_A_PALETTE0 + ( i < < 2 ) ) ;
hw - > cursor_b_palette [ i ] = INREG ( CURSOR_B_PALETTE0 + ( i < < 2 ) ) ;
}
if ( flag = = 7 )
return flag ;
hw - > cursor_size = INREG ( CURSOR_SIZE ) ;
if ( flag = = 8 )
return flag ;
hw - > disp_a_ctrl = INREG ( DSPACNTR ) ;
hw - > disp_b_ctrl = INREG ( DSPBCNTR ) ;
hw - > disp_a_base = INREG ( DSPABASE ) ;
hw - > disp_b_base = INREG ( DSPBBASE ) ;
hw - > disp_a_stride = INREG ( DSPASTRIDE ) ;
hw - > disp_b_stride = INREG ( DSPBSTRIDE ) ;
if ( flag = = 9 )
return flag ;
hw - > vgacntrl = INREG ( VGACNTRL ) ;
if ( flag = = 10 )
return flag ;
hw - > add_id = INREG ( ADD_ID ) ;
if ( flag = = 11 )
return flag ;
for ( i = 0 ; i < 7 ; i + + ) {
hw - > swf0x [ i ] = INREG ( SWF00 + ( i < < 2 ) ) ;
hw - > swf1x [ i ] = INREG ( SWF10 + ( i < < 2 ) ) ;
if ( i < 3 )
hw - > swf3x [ i ] = INREG ( SWF30 + ( i < < 2 ) ) ;
}
for ( i = 0 ; i < 8 ; i + + )
hw - > fence [ i ] = INREG ( FENCE + ( i < < 2 ) ) ;
hw - > instpm = INREG ( INSTPM ) ;
hw - > mem_mode = INREG ( MEM_MODE ) ;
hw - > fw_blc_0 = INREG ( FW_BLC_0 ) ;
hw - > fw_blc_1 = INREG ( FW_BLC_1 ) ;
2006-06-20 22:36:41 +04:00
hw - > hwstam = INREG16 ( HWSTAM ) ;
hw - > ier = INREG16 ( IER ) ;
hw - > iir = INREG16 ( IIR ) ;
hw - > imr = INREG16 ( IMR ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2006-03-20 12:26:45 +03:00
static int calc_vclock3 ( int index , int m , int n , int p )
{
2006-03-23 04:30:05 +03:00
if ( p = = 0 | | n = = 0 )
return 0 ;
2006-03-31 11:08:52 +04:00
return plls [ index ] . ref_clk * m / n / p ;
2006-03-20 12:26:45 +03:00
}
2006-03-23 11:23:48 +03:00
2007-10-16 12:29:31 +04:00
static int calc_vclock ( int index , int m1 , int m2 , int n , int p1 , int p2 ,
int lvds )
2006-03-20 12:26:45 +03:00
{
2006-05-27 12:44:02 +04:00
struct pll_min_max * pll = & plls [ index ] ;
u32 m , vco , p ;
m = ( 5 * ( m1 + 2 ) ) + ( m2 + 2 ) ;
n + = 2 ;
vco = pll - > ref_clk * m / n ;
2007-10-16 12:29:31 +04:00
if ( index = = PLLS_I8xx )
2006-05-27 12:44:02 +04:00
p = ( ( p1 + 2 ) * ( 1 < < ( p2 + 1 ) ) ) ;
2007-10-16 12:29:31 +04:00
else
2006-05-27 12:44:02 +04:00
p = ( ( p1 ) * ( p2 ? 5 : 10 ) ) ;
return vco / p ;
2006-03-20 12:26:45 +03:00
}
2006-08-22 04:12:58 +04:00
# if REGDUMP
2007-10-16 12:29:31 +04:00
static void intelfbhw_get_p1p2 ( struct intelfb_info * dinfo , int dpll ,
int * o_p1 , int * o_p2 )
2006-06-18 10:12:27 +04:00
{
int p1 , p2 ;
if ( IS_I9XX ( dinfo ) ) {
if ( dpll & DPLL_P1_FORCE_DIV2 )
p1 = 1 ;
else
p1 = ( dpll > > DPLL_P1_SHIFT ) & 0xff ;
2007-10-16 12:29:31 +04:00
2006-06-18 10:12:27 +04:00
p1 = ffs ( p1 ) ;
p2 = ( dpll > > DPLL_I9XX_P2_SHIFT ) & DPLL_P2_MASK ;
} else {
if ( dpll & DPLL_P1_FORCE_DIV2 )
p1 = 0 ;
else
p1 = ( dpll > > DPLL_P1_SHIFT ) & DPLL_P1_MASK ;
p2 = ( dpll > > DPLL_P2_SHIFT ) & DPLL_P2_MASK ;
}
* o_p1 = p1 ;
* o_p2 = p2 ;
}
2006-08-22 04:12:58 +04:00
# endif
2006-06-18 10:12:27 +04:00
2007-10-16 12:29:31 +04:00
void intelfbhw_print_hw_state ( struct intelfb_info * dinfo ,
struct intelfb_hwstate * hw )
2005-04-17 02:20:36 +04:00
{
# if REGDUMP
int i , m1 , m2 , n , p1 , p2 ;
2006-03-20 12:26:45 +03:00
int index = dinfo - > pll_index ;
2005-04-17 02:20:36 +04:00
DBG_MSG ( " intelfbhw_print_hw_state \n " ) ;
2006-03-31 11:08:52 +04:00
2006-10-21 01:35:59 +04:00
if ( ! hw )
2005-04-17 02:20:36 +04:00
return ;
/* Read in as much of the HW state as possible. */
printk ( " hw state dump start \n " ) ;
printk ( " VGA0_DIVISOR: 0x%08x \n " , hw - > vga0_divisor ) ;
printk ( " VGA1_DIVISOR: 0x%08x \n " , hw - > vga1_divisor ) ;
2007-10-16 12:29:31 +04:00
printk ( " VGAPD: 0x%08x \n " , hw - > vga_pd ) ;
2005-04-17 02:20:36 +04:00
n = ( hw - > vga0_divisor > > FP_N_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
m1 = ( hw - > vga0_divisor > > FP_M1_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
m2 = ( hw - > vga0_divisor > > FP_M2_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
2006-03-31 11:08:52 +04:00
2006-06-18 10:12:27 +04:00
intelfbhw_get_p1p2 ( dinfo , hw - > vga_pd , & p1 , & p2 ) ;
2006-03-31 11:08:52 +04:00
2005-04-17 02:20:36 +04:00
printk ( " VGA0: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d) \n " ,
2006-03-20 12:26:45 +03:00
m1 , m2 , n , p1 , p2 ) ;
2006-03-23 11:23:48 +03:00
printk ( " VGA0: clock is %d \n " ,
2006-03-31 11:08:52 +04:00
calc_vclock ( index , m1 , m2 , n , p1 , p2 , 0 ) ) ;
2005-04-17 02:20:36 +04:00
n = ( hw - > vga1_divisor > > FP_N_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
m1 = ( hw - > vga1_divisor > > FP_M1_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
m2 = ( hw - > vga1_divisor > > FP_M2_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
2006-06-18 10:12:27 +04:00
intelfbhw_get_p1p2 ( dinfo , hw - > vga_pd , & p1 , & p2 ) ;
2005-04-17 02:20:36 +04:00
printk ( " VGA1: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d) \n " ,
2006-03-20 12:26:45 +03:00
m1 , m2 , n , p1 , p2 ) ;
2007-10-16 12:29:31 +04:00
printk ( " VGA1: clock is %d \n " ,
calc_vclock ( index , m1 , m2 , n , p1 , p2 , 0 ) ) ;
2006-03-31 11:08:52 +04:00
2005-04-17 02:20:36 +04:00
printk ( " DPLL_A: 0x%08x \n " , hw - > dpll_a ) ;
printk ( " DPLL_B: 0x%08x \n " , hw - > dpll_b ) ;
printk ( " FPA0: 0x%08x \n " , hw - > fpa0 ) ;
printk ( " FPA1: 0x%08x \n " , hw - > fpa1 ) ;
printk ( " FPB0: 0x%08x \n " , hw - > fpb0 ) ;
printk ( " FPB1: 0x%08x \n " , hw - > fpb1 ) ;
2006-03-31 11:08:52 +04:00
2005-04-17 02:20:36 +04:00
n = ( hw - > fpa0 > > FP_N_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
m1 = ( hw - > fpa0 > > FP_M1_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
m2 = ( hw - > fpa0 > > FP_M2_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
2006-03-31 11:08:52 +04:00
2006-06-18 10:12:27 +04:00
intelfbhw_get_p1p2 ( dinfo , hw - > dpll_a , & p1 , & p2 ) ;
2006-03-31 11:08:52 +04:00
2005-04-17 02:20:36 +04:00
printk ( " PLLA0: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d) \n " ,
2006-03-20 12:26:45 +03:00
m1 , m2 , n , p1 , p2 ) ;
2007-10-16 12:29:31 +04:00
printk ( " PLLA0: clock is %d \n " ,
calc_vclock ( index , m1 , m2 , n , p1 , p2 , 0 ) ) ;
2006-03-31 11:08:52 +04:00
2005-04-17 02:20:36 +04:00
n = ( hw - > fpa1 > > FP_N_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
m1 = ( hw - > fpa1 > > FP_M1_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
m2 = ( hw - > fpa1 > > FP_M2_DIVISOR_SHIFT ) & FP_DIVISOR_MASK ;
2006-03-31 11:08:52 +04:00
2006-06-18 10:12:27 +04:00
intelfbhw_get_p1p2 ( dinfo , hw - > dpll_a , & p1 , & p2 ) ;
2006-03-31 11:08:52 +04:00
2005-04-17 02:20:36 +04:00
printk ( " PLLA1: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d) \n " ,
2006-03-20 12:26:45 +03:00
m1 , m2 , n , p1 , p2 ) ;
2007-10-16 12:29:31 +04:00
printk ( " PLLA1: clock is %d \n " ,
calc_vclock ( index , m1 , m2 , n , p1 , p2 , 0 ) ) ;
2006-03-31 11:08:52 +04:00
2005-04-17 02:20:36 +04:00
#if 0
printk ( " PALETTE_A: \n " ) ;
for ( i = 0 ; i < PALETTE_8_ENTRIES )
2006-03-20 12:26:45 +03:00
printk ( " %3d: 0x%08x \n " , i , hw - > palette_a [ i ] ) ;
2005-04-17 02:20:36 +04:00
printk ( " PALETTE_B: \n " ) ;
for ( i = 0 ; i < PALETTE_8_ENTRIES )
2006-03-20 12:26:45 +03:00
printk ( " %3d: 0x%08x \n " , i , hw - > palette_b [ i ] ) ;
2005-04-17 02:20:36 +04:00
# endif
printk ( " HTOTAL_A: 0x%08x \n " , hw - > htotal_a ) ;
printk ( " HBLANK_A: 0x%08x \n " , hw - > hblank_a ) ;
printk ( " HSYNC_A: 0x%08x \n " , hw - > hsync_a ) ;
printk ( " VTOTAL_A: 0x%08x \n " , hw - > vtotal_a ) ;
printk ( " VBLANK_A: 0x%08x \n " , hw - > vblank_a ) ;
printk ( " VSYNC_A: 0x%08x \n " , hw - > vsync_a ) ;
printk ( " SRC_SIZE_A: 0x%08x \n " , hw - > src_size_a ) ;
printk ( " BCLRPAT_A: 0x%08x \n " , hw - > bclrpat_a ) ;
printk ( " HTOTAL_B: 0x%08x \n " , hw - > htotal_b ) ;
printk ( " HBLANK_B: 0x%08x \n " , hw - > hblank_b ) ;
printk ( " HSYNC_B: 0x%08x \n " , hw - > hsync_b ) ;
printk ( " VTOTAL_B: 0x%08x \n " , hw - > vtotal_b ) ;
printk ( " VBLANK_B: 0x%08x \n " , hw - > vblank_b ) ;
printk ( " VSYNC_B: 0x%08x \n " , hw - > vsync_b ) ;
printk ( " SRC_SIZE_B: 0x%08x \n " , hw - > src_size_b ) ;
printk ( " BCLRPAT_B: 0x%08x \n " , hw - > bclrpat_b ) ;
printk ( " ADPA: 0x%08x \n " , hw - > adpa ) ;
printk ( " DVOA: 0x%08x \n " , hw - > dvoa ) ;
printk ( " DVOB: 0x%08x \n " , hw - > dvob ) ;
printk ( " DVOC: 0x%08x \n " , hw - > dvoc ) ;
printk ( " DVOA_SRCDIM: 0x%08x \n " , hw - > dvoa_srcdim ) ;
printk ( " DVOB_SRCDIM: 0x%08x \n " , hw - > dvob_srcdim ) ;
printk ( " DVOC_SRCDIM: 0x%08x \n " , hw - > dvoc_srcdim ) ;
printk ( " LVDS: 0x%08x \n " , hw - > lvds ) ;
printk ( " PIPEACONF: 0x%08x \n " , hw - > pipe_a_conf ) ;
printk ( " PIPEBCONF: 0x%08x \n " , hw - > pipe_b_conf ) ;
printk ( " DISPARB: 0x%08x \n " , hw - > disp_arb ) ;
printk ( " CURSOR_A_CONTROL: 0x%08x \n " , hw - > cursor_a_control ) ;
printk ( " CURSOR_B_CONTROL: 0x%08x \n " , hw - > cursor_b_control ) ;
printk ( " CURSOR_A_BASEADDR: 0x%08x \n " , hw - > cursor_a_base ) ;
printk ( " CURSOR_B_BASEADDR: 0x%08x \n " , hw - > cursor_b_base ) ;
printk ( " CURSOR_A_PALETTE: " ) ;
for ( i = 0 ; i < 4 ; i + + ) {
printk ( " 0x%08x " , hw - > cursor_a_palette [ i ] ) ;
if ( i < 3 )
printk ( " , " ) ;
}
printk ( " \n " ) ;
printk ( " CURSOR_B_PALETTE: " ) ;
for ( i = 0 ; i < 4 ; i + + ) {
printk ( " 0x%08x " , hw - > cursor_b_palette [ i ] ) ;
if ( i < 3 )
printk ( " , " ) ;
}
printk ( " \n " ) ;
printk ( " CURSOR_SIZE: 0x%08x \n " , hw - > cursor_size ) ;
printk ( " DSPACNTR: 0x%08x \n " , hw - > disp_a_ctrl ) ;
printk ( " DSPBCNTR: 0x%08x \n " , hw - > disp_b_ctrl ) ;
printk ( " DSPABASE: 0x%08x \n " , hw - > disp_a_base ) ;
printk ( " DSPBBASE: 0x%08x \n " , hw - > disp_b_base ) ;
printk ( " DSPASTRIDE: 0x%08x \n " , hw - > disp_a_stride ) ;
printk ( " DSPBSTRIDE: 0x%08x \n " , hw - > disp_b_stride ) ;
printk ( " VGACNTRL: 0x%08x \n " , hw - > vgacntrl ) ;
printk ( " ADD_ID: 0x%08x \n " , hw - > add_id ) ;
for ( i = 0 ; i < 7 ; i + + ) {
printk ( " SWF0%d 0x%08x \n " , i ,
hw - > swf0x [ i ] ) ;
}
for ( i = 0 ; i < 7 ; i + + ) {
printk ( " SWF1%d 0x%08x \n " , i ,
hw - > swf1x [ i ] ) ;
}
for ( i = 0 ; i < 3 ; i + + ) {
printk ( " SWF3%d 0x%08x \n " , i ,
2006-03-20 12:26:45 +03:00
hw - > swf3x [ i ] ) ;
2005-04-17 02:20:36 +04:00
}
for ( i = 0 ; i < 8 ; i + + )
printk ( " FENCE%d 0x%08x \n " , i ,
2006-03-20 12:26:45 +03:00
hw - > fence [ i ] ) ;
2006-03-23 11:23:48 +03:00
2005-04-17 02:20:36 +04:00
printk ( " INSTPM 0x%08x \n " , hw - > instpm ) ;
printk ( " MEM_MODE 0x%08x \n " , hw - > mem_mode ) ;
printk ( " FW_BLC_0 0x%08x \n " , hw - > fw_blc_0 ) ;
printk ( " FW_BLC_1 0x%08x \n " , hw - > fw_blc_1 ) ;
2006-06-20 22:36:41 +04:00
printk ( " HWSTAM 0x%04x \n " , hw - > hwstam ) ;
printk ( " IER 0x%04x \n " , hw - > ier ) ;
printk ( " IIR 0x%04x \n " , hw - > iir ) ;
printk ( " IMR 0x%04x \n " , hw - > imr ) ;
2005-04-17 02:20:36 +04:00
printk ( " hw state dump end \n " ) ;
# endif
}
2006-03-23 11:23:48 +03:00
2006-03-20 12:26:45 +03:00
2005-04-17 02:20:36 +04:00
/* Split the M parameter into M1 and M2. */
2007-10-16 12:29:31 +04:00
static int splitm ( int index , unsigned int m , unsigned int * retm1 ,
unsigned int * retm2 )
2005-04-17 02:20:36 +04:00
{
int m1 , m2 ;
2006-03-20 12:54:12 +03:00
int testm ;
2006-05-27 12:44:02 +04:00
struct pll_min_max * pll = & plls [ index ] ;
2006-03-20 12:54:12 +03:00
/* no point optimising too much - brute force m */
2006-05-27 12:44:02 +04:00
for ( m1 = pll - > min_m1 ; m1 < pll - > max_m1 + 1 ; m1 + + ) {
for ( m2 = pll - > min_m2 ; m2 < pll - > max_m2 + 1 ; m2 + + ) {
2006-03-31 11:08:52 +04:00
testm = ( 5 * ( m1 + 2 ) ) + ( m2 + 2 ) ;
2006-03-23 11:23:48 +03:00
if ( testm = = m ) {
* retm1 = ( unsigned int ) m1 ;
* retm2 = ( unsigned int ) m2 ;
return 0 ;
}
}
2005-04-17 02:20:36 +04:00
}
2006-03-20 12:54:12 +03:00
return 1 ;
2005-04-17 02:20:36 +04:00
}
/* Split the P parameter into P1 and P2. */
2007-10-16 12:29:31 +04:00
static int splitp ( int index , unsigned int p , unsigned int * retp1 ,
unsigned int * retp2 )
2005-04-17 02:20:36 +04:00
{
int p1 , p2 ;
2006-05-27 12:44:02 +04:00
struct pll_min_max * pll = & plls [ index ] ;
2005-04-17 02:20:36 +04:00
2006-03-23 11:23:48 +03:00
if ( index = = PLLS_I9xx ) {
2006-04-03 10:19:26 +04:00
p2 = ( p % 10 ) ? 1 : 0 ;
2006-03-31 11:08:52 +04:00
p1 = p / ( p2 ? 5 : 10 ) ;
2006-03-20 12:26:45 +03:00
* retp1 = ( unsigned int ) p1 ;
* retp2 = ( unsigned int ) p2 ;
return 0 ;
}
2006-05-27 12:44:02 +04:00
if ( p % 4 = = 0 )
p2 = 1 ;
else
p2 = 0 ;
p1 = ( p / ( 1 < < ( p2 + 1 ) ) ) - 2 ;
if ( p % 4 = = 0 & & p1 < pll - > min_p1 ) {
p2 = 0 ;
2005-04-17 02:20:36 +04:00
p1 = ( p / ( 1 < < ( p2 + 1 ) ) ) - 2 ;
}
2006-05-27 12:44:02 +04:00
if ( p1 < pll - > min_p1 | | p1 > pll - > max_p1 | |
( p1 + 2 ) * ( 1 < < ( p2 + 1 ) ) ! = p ) {
return 1 ;
} else {
* retp1 = ( unsigned int ) p1 ;
* retp2 = ( unsigned int ) p2 ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
}
2007-10-16 12:29:31 +04:00
static int calc_pll_params ( int index , int clock , u32 * retm1 , u32 * retm2 ,
u32 * retn , u32 * retp1 , u32 * retp2 , u32 * retclock )
2005-04-17 02:20:36 +04:00
{
2006-03-23 04:30:05 +03:00
u32 m1 , m2 , n , p1 , p2 , n1 , testm ;
u32 f_vco , p , p_best = 0 , m , f_out = 0 ;
2005-04-17 02:20:36 +04:00
u32 err_max , err_target , err_best = 10000000 ;
u32 n_best = 0 , m_best = 0 , f_best , f_err ;
2006-04-03 10:19:26 +04:00
u32 p_min , p_max , p_inc , div_max ;
struct pll_min_max * pll = & plls [ index ] ;
2005-04-17 02:20:36 +04:00
/* Accept 0.5% difference, but aim for 0.1% */
err_max = 5 * clock / 1000 ;
err_target = clock / 1000 ;
DBG_MSG ( " Clock is %d \n " , clock ) ;
2006-04-03 10:19:26 +04:00
div_max = pll - > max_vco / clock ;
2005-04-17 02:20:36 +04:00
2006-04-03 10:19:26 +04:00
p_inc = ( clock < = pll - > p_transition_clk ) ? pll - > p_inc_lo : pll - > p_inc_hi ;
p_min = p_inc ;
2005-04-17 02:20:36 +04:00
p_max = ROUND_DOWN_TO ( div_max , p_inc ) ;
2006-04-03 10:19:26 +04:00
if ( p_min < pll - > min_p )
p_min = pll - > min_p ;
if ( p_max > pll - > max_p )
p_max = pll - > max_p ;
2005-04-17 02:20:36 +04:00
DBG_MSG ( " p range is %d-%d (%d) \n " , p_min , p_max , p_inc ) ;
p = p_min ;
do {
2006-03-20 12:02:24 +03:00
if ( splitp ( index , p , & p1 , & p2 ) ) {
2005-04-17 02:20:36 +04:00
WRN_MSG ( " cannot split p = %d \n " , p ) ;
p + = p_inc ;
continue ;
}
2006-04-03 10:19:26 +04:00
n = pll - > min_n ;
2005-04-17 02:20:36 +04:00
f_vco = clock * p ;
do {
2006-04-03 10:19:26 +04:00
m = ROUND_UP_TO ( f_vco * n , pll - > ref_clk ) / pll - > ref_clk ;
if ( m < pll - > min_m )
m = pll - > min_m + 1 ;
if ( m > pll - > max_m )
m = pll - > max_m - 1 ;
2006-03-23 04:30:05 +03:00
for ( testm = m - 1 ; testm < = m ; testm + + ) {
2007-09-12 02:24:12 +04:00
f_out = calc_vclock3 ( index , testm , n , p ) ;
2006-05-27 12:44:02 +04:00
if ( splitm ( index , testm , & m1 , & m2 ) ) {
2007-09-12 02:24:12 +04:00
WRN_MSG ( " cannot split m = %d \n " ,
testm ) ;
2006-03-23 04:30:05 +03:00
continue ;
}
if ( clock > f_out )
f_err = clock - f_out ;
else /* slightly bias the error for bigger clocks */
f_err = f_out - clock + 1 ;
2006-03-31 11:08:52 +04:00
2006-03-23 04:30:05 +03:00
if ( f_err < err_best ) {
2006-05-27 12:44:02 +04:00
m_best = testm ;
2006-03-23 04:30:05 +03:00
n_best = n ;
p_best = p ;
f_best = f_out ;
err_best = f_err ;
}
2005-04-17 02:20:36 +04:00
}
n + + ;
2006-04-03 10:19:26 +04:00
} while ( ( n < = pll - > max_n ) & & ( f_out > = clock ) ) ;
2005-04-17 02:20:36 +04:00
p + = p_inc ;
} while ( ( p < = p_max ) ) ;
if ( ! m_best ) {
WRN_MSG ( " cannot find parameters for clock %d \n " , clock ) ;
return 1 ;
}
m = m_best ;
n = n_best ;
p = p_best ;
2006-03-20 12:02:24 +03:00
splitm ( index , m , & m1 , & m2 ) ;
splitp ( index , p , & p1 , & p2 ) ;
2005-04-17 02:20:36 +04:00
n1 = n - 2 ;
DBG_MSG ( " m, n, p: %d (%d,%d), %d (%d), %d (%d,%d), "
" f: %d (%d), VCO: %d \n " ,
m , m1 , m2 , n , n1 , p , p1 , p2 ,
2006-03-23 11:23:48 +03:00
calc_vclock3 ( index , m , n , p ) ,
2006-03-31 11:08:52 +04:00
calc_vclock ( index , m1 , m2 , n1 , p1 , p2 , 0 ) ,
2006-03-20 12:26:45 +03:00
calc_vclock3 ( index , m , n , p ) * p ) ;
2005-04-17 02:20:36 +04:00
* retm1 = m1 ;
* retm2 = m2 ;
* retn = n1 ;
* retp1 = p1 ;
* retp2 = p2 ;
2006-03-31 11:08:52 +04:00
* retclock = calc_vclock ( index , m1 , m2 , n1 , p1 , p2 , 0 ) ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
2007-10-16 12:29:31 +04:00
static __inline__ int check_overflow ( u32 value , u32 limit ,
const char * description )
2005-04-17 02:20:36 +04:00
{
if ( value > limit ) {
WRN_MSG ( " %s value %d exceeds limit %d \n " ,
description , value , limit ) ;
return 1 ;
}
return 0 ;
}
/* It is assumed that hw is filled in with the initial state information. */
2007-10-16 12:29:31 +04:00
int intelfbhw_mode_to_hw ( struct intelfb_info * dinfo ,
struct intelfb_hwstate * hw ,
struct fb_var_screeninfo * var )
2005-04-17 02:20:36 +04:00
{
int pipe = PIPE_A ;
u32 * dpll , * fp0 , * fp1 ;
u32 m1 , m2 , n , p1 , p2 , clock_target , clock ;
u32 hsync_start , hsync_end , hblank_start , hblank_end , htotal , hactive ;
u32 vsync_start , vsync_end , vblank_start , vblank_end , vtotal , vactive ;
u32 vsync_pol , hsync_pol ;
u32 * vs , * vb , * vt , * hs , * hb , * ht , * ss , * pipe_conf ;
2006-05-27 12:17:52 +04:00
u32 stride_alignment ;
2005-04-17 02:20:36 +04:00
DBG_MSG ( " intelfbhw_mode_to_hw \n " ) ;
/* Disable VGA */
hw - > vgacntrl | = VGA_DISABLE ;
/* Check whether pipe A or pipe B is enabled. */
if ( hw - > pipe_a_conf & PIPECONF_ENABLE )
pipe = PIPE_A ;
else if ( hw - > pipe_b_conf & PIPECONF_ENABLE )
pipe = PIPE_B ;
/* Set which pipe's registers will be set. */
if ( pipe = = PIPE_B ) {
dpll = & hw - > dpll_b ;
fp0 = & hw - > fpb0 ;
fp1 = & hw - > fpb1 ;
hs = & hw - > hsync_b ;
hb = & hw - > hblank_b ;
ht = & hw - > htotal_b ;
vs = & hw - > vsync_b ;
vb = & hw - > vblank_b ;
vt = & hw - > vtotal_b ;
ss = & hw - > src_size_b ;
pipe_conf = & hw - > pipe_b_conf ;
} else {
dpll = & hw - > dpll_a ;
fp0 = & hw - > fpa0 ;
fp1 = & hw - > fpa1 ;
hs = & hw - > hsync_a ;
hb = & hw - > hblank_a ;
ht = & hw - > htotal_a ;
vs = & hw - > vsync_a ;
vb = & hw - > vblank_a ;
vt = & hw - > vtotal_a ;
ss = & hw - > src_size_a ;
pipe_conf = & hw - > pipe_a_conf ;
}
/* Use ADPA register for sync control. */
hw - > adpa & = ~ ADPA_USE_VGA_HVPOLARITY ;
/* sync polarity */
hsync_pol = ( var - > sync & FB_SYNC_HOR_HIGH_ACT ) ?
ADPA_SYNC_ACTIVE_HIGH : ADPA_SYNC_ACTIVE_LOW ;
vsync_pol = ( var - > sync & FB_SYNC_VERT_HIGH_ACT ) ?
ADPA_SYNC_ACTIVE_HIGH : ADPA_SYNC_ACTIVE_LOW ;
hw - > adpa & = ~ ( ( ADPA_SYNC_ACTIVE_MASK < < ADPA_VSYNC_ACTIVE_SHIFT ) |
( ADPA_SYNC_ACTIVE_MASK < < ADPA_HSYNC_ACTIVE_SHIFT ) ) ;
hw - > adpa | = ( hsync_pol < < ADPA_HSYNC_ACTIVE_SHIFT ) |
( vsync_pol < < ADPA_VSYNC_ACTIVE_SHIFT ) ;
/* Connect correct pipe to the analog port DAC */
hw - > adpa & = ~ ( PIPE_MASK < < ADPA_PIPE_SELECT_SHIFT ) ;
hw - > adpa | = ( pipe < < ADPA_PIPE_SELECT_SHIFT ) ;
/* Set DPMS state to D0 (on) */
hw - > adpa & = ~ ADPA_DPMS_CONTROL_MASK ;
hw - > adpa | = ADPA_DPMS_D0 ;
hw - > adpa | = ADPA_DAC_ENABLE ;
* dpll | = ( DPLL_VCO_ENABLE | DPLL_VGA_MODE_DISABLE ) ;
* dpll & = ~ ( DPLL_RATE_SELECT_MASK | DPLL_REFERENCE_SELECT_MASK ) ;
* dpll | = ( DPLL_REFERENCE_DEFAULT | DPLL_RATE_SELECT_FP0 ) ;
/* Desired clock in kHz */
clock_target = 1000000000 / var - > pixclock ;
2006-03-31 11:08:52 +04:00
if ( calc_pll_params ( dinfo - > pll_index , clock_target , & m1 , & m2 ,
2006-03-23 11:23:48 +03:00
& n , & p1 , & p2 , & clock ) ) {
2005-04-17 02:20:36 +04:00
WRN_MSG ( " calc_pll_params failed \n " ) ;
return 1 ;
}
/* Check for overflow. */
if ( check_overflow ( p1 , DPLL_P1_MASK , " PLL P1 parameter " ) )
return 1 ;
if ( check_overflow ( p2 , DPLL_P2_MASK , " PLL P2 parameter " ) )
return 1 ;
if ( check_overflow ( m1 , FP_DIVISOR_MASK , " PLL M1 parameter " ) )
return 1 ;
if ( check_overflow ( m2 , FP_DIVISOR_MASK , " PLL M2 parameter " ) )
return 1 ;
if ( check_overflow ( n , FP_DIVISOR_MASK , " PLL N parameter " ) )
return 1 ;
* dpll & = ~ DPLL_P1_FORCE_DIV2 ;
* dpll & = ~ ( ( DPLL_P2_MASK < < DPLL_P2_SHIFT ) |
( DPLL_P1_MASK < < DPLL_P1_SHIFT ) ) ;
2006-03-31 11:08:52 +04:00
if ( IS_I9XX ( dinfo ) ) {
* dpll | = ( p2 < < DPLL_I9XX_P2_SHIFT ) ;
* dpll | = ( 1 < < ( p1 - 1 ) ) < < DPLL_P1_SHIFT ;
2007-10-16 12:29:31 +04:00
} else
2006-03-31 11:08:52 +04:00
* dpll | = ( p2 < < DPLL_P2_SHIFT ) | ( p1 < < DPLL_P1_SHIFT ) ;
2005-04-17 02:20:36 +04:00
* fp0 = ( n < < FP_N_DIVISOR_SHIFT ) |
( m1 < < FP_M1_DIVISOR_SHIFT ) |
( m2 < < FP_M2_DIVISOR_SHIFT ) ;
* fp1 = * fp0 ;
hw - > dvob & = ~ PORT_ENABLE ;
hw - > dvoc & = ~ PORT_ENABLE ;
/* Use display plane A. */
hw - > disp_a_ctrl | = DISPPLANE_PLANE_ENABLE ;
hw - > disp_a_ctrl & = ~ DISPPLANE_GAMMA_ENABLE ;
hw - > disp_a_ctrl & = ~ DISPPLANE_PIXFORMAT_MASK ;
switch ( intelfb_var_to_depth ( var ) ) {
case 8 :
hw - > disp_a_ctrl | = DISPPLANE_8BPP | DISPPLANE_GAMMA_ENABLE ;
break ;
case 15 :
hw - > disp_a_ctrl | = DISPPLANE_15_16BPP ;
break ;
case 16 :
hw - > disp_a_ctrl | = DISPPLANE_16BPP ;
break ;
case 24 :
hw - > disp_a_ctrl | = DISPPLANE_32BPP_NO_ALPHA ;
break ;
}
hw - > disp_a_ctrl & = ~ ( PIPE_MASK < < DISPPLANE_SEL_PIPE_SHIFT ) ;
hw - > disp_a_ctrl | = ( pipe < < DISPPLANE_SEL_PIPE_SHIFT ) ;
/* Set CRTC registers. */
hactive = var - > xres ;
hsync_start = hactive + var - > right_margin ;
hsync_end = hsync_start + var - > hsync_len ;
htotal = hsync_end + var - > left_margin ;
hblank_start = hactive ;
hblank_end = htotal ;
DBG_MSG ( " H: act %d, ss %d, se %d, tot %d bs %d, be %d \n " ,
hactive , hsync_start , hsync_end , htotal , hblank_start ,
hblank_end ) ;
vactive = var - > yres ;
2007-10-16 12:29:33 +04:00
if ( var - > vmode & FB_VMODE_INTERLACED )
vactive - - ; /* the chip adds 2 halflines automatically */
2005-04-17 02:20:36 +04:00
vsync_start = vactive + var - > lower_margin ;
vsync_end = vsync_start + var - > vsync_len ;
vtotal = vsync_end + var - > upper_margin ;
vblank_start = vactive ;
vblank_end = vtotal ;
vblank_end = vsync_end + 1 ;
DBG_MSG ( " V: act %d, ss %d, se %d, tot %d bs %d, be %d \n " ,
vactive , vsync_start , vsync_end , vtotal , vblank_start ,
vblank_end ) ;
/* Adjust for register values, and check for overflow. */
hactive - - ;
if ( check_overflow ( hactive , HACTIVE_MASK , " CRTC hactive " ) )
return 1 ;
hsync_start - - ;
if ( check_overflow ( hsync_start , HSYNCSTART_MASK , " CRTC hsync_start " ) )
return 1 ;
hsync_end - - ;
if ( check_overflow ( hsync_end , HSYNCEND_MASK , " CRTC hsync_end " ) )
return 1 ;
htotal - - ;
if ( check_overflow ( htotal , HTOTAL_MASK , " CRTC htotal " ) )
return 1 ;
hblank_start - - ;
if ( check_overflow ( hblank_start , HBLANKSTART_MASK , " CRTC hblank_start " ) )
return 1 ;
hblank_end - - ;
if ( check_overflow ( hblank_end , HBLANKEND_MASK , " CRTC hblank_end " ) )
return 1 ;
vactive - - ;
if ( check_overflow ( vactive , VACTIVE_MASK , " CRTC vactive " ) )
return 1 ;
vsync_start - - ;
if ( check_overflow ( vsync_start , VSYNCSTART_MASK , " CRTC vsync_start " ) )
return 1 ;
vsync_end - - ;
if ( check_overflow ( vsync_end , VSYNCEND_MASK , " CRTC vsync_end " ) )
return 1 ;
vtotal - - ;
if ( check_overflow ( vtotal , VTOTAL_MASK , " CRTC vtotal " ) )
return 1 ;
vblank_start - - ;
if ( check_overflow ( vblank_start , VBLANKSTART_MASK , " CRTC vblank_start " ) )
return 1 ;
vblank_end - - ;
if ( check_overflow ( vblank_end , VBLANKEND_MASK , " CRTC vblank_end " ) )
return 1 ;
* ht = ( htotal < < HTOTAL_SHIFT ) | ( hactive < < HACTIVE_SHIFT ) ;
* hb = ( hblank_start < < HBLANKSTART_SHIFT ) |
( hblank_end < < HSYNCEND_SHIFT ) ;
* hs = ( hsync_start < < HSYNCSTART_SHIFT ) | ( hsync_end < < HSYNCEND_SHIFT ) ;
* vt = ( vtotal < < VTOTAL_SHIFT ) | ( vactive < < VACTIVE_SHIFT ) ;
* vb = ( vblank_start < < VBLANKSTART_SHIFT ) |
( vblank_end < < VSYNCEND_SHIFT ) ;
* vs = ( vsync_start < < VSYNCSTART_SHIFT ) | ( vsync_end < < VSYNCEND_SHIFT ) ;
* ss = ( hactive < < SRC_SIZE_HORIZ_SHIFT ) |
( vactive < < SRC_SIZE_VERT_SHIFT ) ;
2006-04-03 08:46:55 +04:00
hw - > disp_a_stride = dinfo - > pitch ;
2005-04-17 02:20:36 +04:00
DBG_MSG ( " pitch is %d \n " , hw - > disp_a_stride ) ;
hw - > disp_a_base = hw - > disp_a_stride * var - > yoffset +
var - > xoffset * var - > bits_per_pixel / 8 ;
hw - > disp_a_base + = dinfo - > fb . offset < < 12 ;
/* Check stride alignment. */
2006-05-27 12:17:52 +04:00
stride_alignment = IS_I9XX ( dinfo ) ? STRIDE_ALIGNMENT_I9XX :
STRIDE_ALIGNMENT ;
if ( hw - > disp_a_stride % stride_alignment ! = 0 ) {
2005-04-17 02:20:36 +04:00
WRN_MSG ( " display stride %d has bad alignment %d \n " ,
2006-05-27 12:17:52 +04:00
hw - > disp_a_stride , stride_alignment ) ;
2005-04-17 02:20:36 +04:00
return 1 ;
}
/* Set the palette to 8-bit mode. */
* pipe_conf & = ~ PIPECONF_GAMMA ;
2007-10-16 12:29:18 +04:00
if ( var - > vmode & FB_VMODE_INTERLACED )
* pipe_conf | = PIPECONF_INTERLACE_W_FIELD_INDICATION ;
else
* pipe_conf & = ~ PIPECONF_INTERLACE_MASK ;
2005-04-17 02:20:36 +04:00
return 0 ;
}
/* Program a (non-VGA) video mode. */
2007-10-16 12:29:31 +04:00
int intelfbhw_program_mode ( struct intelfb_info * dinfo ,
const struct intelfb_hwstate * hw , int blank )
2005-04-17 02:20:36 +04:00
{
int pipe = PIPE_A ;
u32 tmp ;
const u32 * dpll , * fp0 , * fp1 , * pipe_conf ;
const u32 * hs , * ht , * hb , * vs , * vt , * vb , * ss ;
2007-10-16 12:29:34 +04:00
u32 dpll_reg , fp0_reg , fp1_reg , pipe_conf_reg , pipe_stat_reg ;
2005-04-17 02:20:36 +04:00
u32 hsync_reg , htotal_reg , hblank_reg ;
u32 vsync_reg , vtotal_reg , vblank_reg ;
u32 src_size_reg ;
2006-03-23 04:30:05 +03:00
u32 count , tmp_val [ 3 ] ;
2005-04-17 02:20:36 +04:00
/* Assume single pipe, display plane A, analog CRT. */
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_program_mode \n " ) ;
# endif
/* Disable VGA */
tmp = INREG ( VGACNTRL ) ;
tmp | = VGA_DISABLE ;
OUTREG ( VGACNTRL , tmp ) ;
/* Check whether pipe A or pipe B is enabled. */
if ( hw - > pipe_a_conf & PIPECONF_ENABLE )
pipe = PIPE_A ;
else if ( hw - > pipe_b_conf & PIPECONF_ENABLE )
pipe = PIPE_B ;
dinfo - > pipe = pipe ;
if ( pipe = = PIPE_B ) {
dpll = & hw - > dpll_b ;
fp0 = & hw - > fpb0 ;
fp1 = & hw - > fpb1 ;
pipe_conf = & hw - > pipe_b_conf ;
hs = & hw - > hsync_b ;
hb = & hw - > hblank_b ;
ht = & hw - > htotal_b ;
vs = & hw - > vsync_b ;
vb = & hw - > vblank_b ;
vt = & hw - > vtotal_b ;
ss = & hw - > src_size_b ;
dpll_reg = DPLL_B ;
fp0_reg = FPB0 ;
fp1_reg = FPB1 ;
pipe_conf_reg = PIPEBCONF ;
2007-10-16 12:29:34 +04:00
pipe_stat_reg = PIPEBSTAT ;
2005-04-17 02:20:36 +04:00
hsync_reg = HSYNC_B ;
htotal_reg = HTOTAL_B ;
hblank_reg = HBLANK_B ;
vsync_reg = VSYNC_B ;
vtotal_reg = VTOTAL_B ;
vblank_reg = VBLANK_B ;
src_size_reg = SRC_SIZE_B ;
} else {
dpll = & hw - > dpll_a ;
fp0 = & hw - > fpa0 ;
fp1 = & hw - > fpa1 ;
pipe_conf = & hw - > pipe_a_conf ;
hs = & hw - > hsync_a ;
hb = & hw - > hblank_a ;
ht = & hw - > htotal_a ;
vs = & hw - > vsync_a ;
vb = & hw - > vblank_a ;
vt = & hw - > vtotal_a ;
ss = & hw - > src_size_a ;
dpll_reg = DPLL_A ;
fp0_reg = FPA0 ;
fp1_reg = FPA1 ;
pipe_conf_reg = PIPEACONF ;
2007-10-16 12:29:34 +04:00
pipe_stat_reg = PIPEASTAT ;
2005-04-17 02:20:36 +04:00
hsync_reg = HSYNC_A ;
htotal_reg = HTOTAL_A ;
hblank_reg = HBLANK_A ;
vsync_reg = VSYNC_A ;
vtotal_reg = VTOTAL_A ;
vblank_reg = VBLANK_A ;
src_size_reg = SRC_SIZE_A ;
}
2006-03-23 04:30:05 +03:00
/* turn off pipe */
tmp = INREG ( pipe_conf_reg ) ;
tmp & = ~ PIPECONF_ENABLE ;
OUTREG ( pipe_conf_reg , tmp ) ;
2006-03-31 11:08:52 +04:00
2006-03-23 04:30:05 +03:00
count = 0 ;
2006-03-23 11:23:48 +03:00
do {
2007-10-16 12:29:33 +04:00
tmp_val [ count % 3 ] = INREG ( PIPEA_DSL ) ;
if ( ( tmp_val [ 0 ] = = tmp_val [ 1 ] ) & & ( tmp_val [ 1 ] = = tmp_val [ 2 ] ) )
2006-03-31 11:08:52 +04:00
break ;
count + + ;
udelay ( 1 ) ;
if ( count % 200 = = 0 ) {
tmp = INREG ( pipe_conf_reg ) ;
tmp & = ~ PIPECONF_ENABLE ;
OUTREG ( pipe_conf_reg , tmp ) ;
}
2007-10-16 12:29:31 +04:00
} while ( count < 2000 ) ;
2006-03-23 04:30:05 +03:00
OUTREG ( ADPA , INREG ( ADPA ) & ~ ADPA_DAC_ENABLE ) ;
2005-04-17 02:20:36 +04:00
/* Disable planes A and B. */
tmp = INREG ( DSPACNTR ) ;
tmp & = ~ DISPPLANE_PLANE_ENABLE ;
OUTREG ( DSPACNTR , tmp ) ;
tmp = INREG ( DSPBCNTR ) ;
tmp & = ~ DISPPLANE_PLANE_ENABLE ;
OUTREG ( DSPBCNTR , tmp ) ;
2006-03-31 11:08:52 +04:00
/* Wait for vblank. For now, just wait for a 50Hz cycle (20ms)) */
2005-04-17 02:20:36 +04:00
mdelay ( 20 ) ;
2006-05-27 12:56:02 +04:00
OUTREG ( DVOB , INREG ( DVOB ) & ~ PORT_ENABLE ) ;
OUTREG ( DVOC , INREG ( DVOC ) & ~ PORT_ENABLE ) ;
OUTREG ( ADPA , INREG ( ADPA ) & ~ ADPA_DAC_ENABLE ) ;
2005-04-17 02:20:36 +04:00
/* Disable Sync */
tmp = INREG ( ADPA ) ;
tmp & = ~ ADPA_DPMS_CONTROL_MASK ;
tmp | = ADPA_DPMS_D3 ;
OUTREG ( ADPA , tmp ) ;
2006-03-23 04:30:05 +03:00
/* do some funky magic - xyzzy */
OUTREG ( 0x61204 , 0xabcd0000 ) ;
2005-04-17 02:20:36 +04:00
/* turn off PLL */
tmp = INREG ( dpll_reg ) ;
2007-09-19 09:46:34 +04:00
tmp & = ~ DPLL_VCO_ENABLE ;
2005-04-17 02:20:36 +04:00
OUTREG ( dpll_reg , tmp ) ;
/* Set PLL parameters */
OUTREG ( fp0_reg , * fp0 ) ;
OUTREG ( fp1_reg , * fp1 ) ;
2006-03-23 04:30:05 +03:00
/* Enable PLL */
2006-05-27 12:56:02 +04:00
OUTREG ( dpll_reg , * dpll ) ;
2005-04-17 02:20:36 +04:00
/* Set DVOs B/C */
OUTREG ( DVOB , hw - > dvob ) ;
OUTREG ( DVOC , hw - > dvoc ) ;
2006-03-23 04:30:05 +03:00
/* undo funky magic */
OUTREG ( 0x61204 , 0x00000000 ) ;
2005-04-17 02:20:36 +04:00
/* Set ADPA */
2006-03-23 04:30:05 +03:00
OUTREG ( ADPA , INREG ( ADPA ) | ADPA_DAC_ENABLE ) ;
2005-04-17 02:20:36 +04:00
OUTREG ( ADPA , ( hw - > adpa & ~ ( ADPA_DPMS_CONTROL_MASK ) ) | ADPA_DPMS_D3 ) ;
2006-03-23 04:30:05 +03:00
/* Set pipe parameters */
OUTREG ( hsync_reg , * hs ) ;
OUTREG ( hblank_reg , * hb ) ;
OUTREG ( htotal_reg , * ht ) ;
OUTREG ( vsync_reg , * vs ) ;
OUTREG ( vblank_reg , * vb ) ;
OUTREG ( vtotal_reg , * vt ) ;
OUTREG ( src_size_reg , * ss ) ;
2005-04-17 02:20:36 +04:00
2007-10-16 12:29:34 +04:00
switch ( dinfo - > info - > var . vmode & ( FB_VMODE_INTERLACED |
FB_VMODE_ODD_FLD_FIRST ) ) {
case FB_VMODE_INTERLACED | FB_VMODE_ODD_FLD_FIRST :
OUTREG ( pipe_stat_reg , 0xFFFF | PIPESTAT_FLD_EVT_ODD_EN ) ;
break ;
case FB_VMODE_INTERLACED : /* even lines first */
OUTREG ( pipe_stat_reg , 0xFFFF | PIPESTAT_FLD_EVT_EVEN_EN ) ;
break ;
default : /* non-interlaced */
OUTREG ( pipe_stat_reg , 0xFFFF ) ; /* clear all status bits only */
}
2005-04-17 02:20:36 +04:00
/* Enable pipe */
OUTREG ( pipe_conf_reg , * pipe_conf | PIPECONF_ENABLE ) ;
/* Enable sync */
tmp = INREG ( ADPA ) ;
tmp & = ~ ADPA_DPMS_CONTROL_MASK ;
tmp | = ADPA_DPMS_D0 ;
OUTREG ( ADPA , tmp ) ;
/* setup display plane */
if ( dinfo - > pdev - > device = = PCI_DEVICE_ID_INTEL_830M ) {
/*
* i830M errata : the display plane must be enabled
* to allow writes to the other bits in the plane
* control register .
*/
tmp = INREG ( DSPACNTR ) ;
if ( ( tmp & DISPPLANE_PLANE_ENABLE ) ! = DISPPLANE_PLANE_ENABLE ) {
tmp | = DISPPLANE_PLANE_ENABLE ;
OUTREG ( DSPACNTR , tmp ) ;
OUTREG ( DSPACNTR ,
hw - > disp_a_ctrl | DISPPLANE_PLANE_ENABLE ) ;
mdelay ( 1 ) ;
2006-03-31 11:08:52 +04:00
}
2005-04-17 02:20:36 +04:00
}
OUTREG ( DSPACNTR , hw - > disp_a_ctrl & ~ DISPPLANE_PLANE_ENABLE ) ;
OUTREG ( DSPASTRIDE , hw - > disp_a_stride ) ;
OUTREG ( DSPABASE , hw - > disp_a_base ) ;
/* Enable plane */
if ( ! blank ) {
tmp = INREG ( DSPACNTR ) ;
tmp | = DISPPLANE_PLANE_ENABLE ;
OUTREG ( DSPACNTR , tmp ) ;
OUTREG ( DSPABASE , hw - > disp_a_base ) ;
}
return 0 ;
}
/* forward declarations */
static void refresh_ring ( struct intelfb_info * dinfo ) ;
static void reset_state ( struct intelfb_info * dinfo ) ;
static void do_flush ( struct intelfb_info * dinfo ) ;
2007-05-08 11:37:25 +04:00
static u32 get_ring_space ( struct intelfb_info * dinfo )
{
u32 ring_space ;
if ( dinfo - > ring_tail > = dinfo - > ring_head )
ring_space = dinfo - > ring . size -
( dinfo - > ring_tail - dinfo - > ring_head ) ;
else
ring_space = dinfo - > ring_head - dinfo - > ring_tail ;
if ( ring_space > RING_MIN_FREE )
ring_space - = RING_MIN_FREE ;
else
ring_space = 0 ;
return ring_space ;
}
2007-10-16 12:29:31 +04:00
static int wait_ring ( struct intelfb_info * dinfo , int n )
2005-04-17 02:20:36 +04:00
{
int i = 0 ;
unsigned long end ;
u32 last_head = INREG ( PRI_RING_HEAD ) & RING_HEAD_MASK ;
# if VERBOSE > 0
DBG_MSG ( " wait_ring: %d \n " , n ) ;
# endif
end = jiffies + ( HZ * 3 ) ;
while ( dinfo - > ring_space < n ) {
2006-06-23 09:05:39 +04:00
dinfo - > ring_head = INREG ( PRI_RING_HEAD ) & RING_HEAD_MASK ;
2007-05-08 11:37:25 +04:00
dinfo - > ring_space = get_ring_space ( dinfo ) ;
2006-06-23 09:05:39 +04:00
if ( dinfo - > ring_head ! = last_head ) {
2005-04-17 02:20:36 +04:00
end = jiffies + ( HZ * 3 ) ;
2006-06-23 09:05:39 +04:00
last_head = dinfo - > ring_head ;
2005-04-17 02:20:36 +04:00
}
i + + ;
if ( time_before ( end , jiffies ) ) {
if ( ! i ) {
/* Try again */
reset_state ( dinfo ) ;
refresh_ring ( dinfo ) ;
do_flush ( dinfo ) ;
end = jiffies + ( HZ * 3 ) ;
i = 1 ;
} else {
WRN_MSG ( " ring buffer : space: %d wanted %d \n " ,
dinfo - > ring_space , n ) ;
WRN_MSG ( " lockup - turning off hardware "
" acceleration \n " ) ;
dinfo - > ring_lockup = 1 ;
break ;
}
}
udelay ( 1 ) ;
}
return i ;
}
2007-10-16 12:29:31 +04:00
static void do_flush ( struct intelfb_info * dinfo )
{
2005-04-17 02:20:36 +04:00
START_RING ( 2 ) ;
OUT_RING ( MI_FLUSH | MI_WRITE_DIRTY_STATE | MI_INVALIDATE_MAP_CACHE ) ;
OUT_RING ( MI_NOOP ) ;
ADVANCE_RING ( ) ;
}
2007-10-16 12:29:31 +04:00
void intelfbhw_do_sync ( struct intelfb_info * dinfo )
2005-04-17 02:20:36 +04:00
{
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_do_sync \n " ) ;
# endif
if ( ! dinfo - > accel )
return ;
/*
* Send a flush , then wait until the ring is empty . This is what
* the XFree86 driver does , and actually it doesn ' t seem a lot worse
* than the recommended method ( both have problems ) .
*/
do_flush ( dinfo ) ;
wait_ring ( dinfo , dinfo - > ring . size - RING_MIN_FREE ) ;
dinfo - > ring_space = dinfo - > ring . size - RING_MIN_FREE ;
}
2007-10-16 12:29:31 +04:00
static void refresh_ring ( struct intelfb_info * dinfo )
2005-04-17 02:20:36 +04:00
{
# if VERBOSE > 0
DBG_MSG ( " refresh_ring \n " ) ;
# endif
2006-06-23 09:05:39 +04:00
dinfo - > ring_head = INREG ( PRI_RING_HEAD ) & RING_HEAD_MASK ;
2005-04-17 02:20:36 +04:00
dinfo - > ring_tail = INREG ( PRI_RING_TAIL ) & RING_TAIL_MASK ;
2007-05-08 11:37:25 +04:00
dinfo - > ring_space = get_ring_space ( dinfo ) ;
2005-04-17 02:20:36 +04:00
}
2007-10-16 12:29:31 +04:00
static void reset_state ( struct intelfb_info * dinfo )
2005-04-17 02:20:36 +04:00
{
int i ;
u32 tmp ;
# if VERBOSE > 0
DBG_MSG ( " reset_state \n " ) ;
# endif
for ( i = 0 ; i < FENCE_NUM ; i + + )
OUTREG ( FENCE + ( i < < 2 ) , 0 ) ;
/* Flush the ring buffer if it's enabled. */
tmp = INREG ( PRI_RING_LENGTH ) ;
if ( tmp & RING_ENABLE ) {
# if VERBOSE > 0
DBG_MSG ( " reset_state: ring was enabled \n " ) ;
# endif
refresh_ring ( dinfo ) ;
intelfbhw_do_sync ( dinfo ) ;
DO_RING_IDLE ( ) ;
}
OUTREG ( PRI_RING_LENGTH , 0 ) ;
OUTREG ( PRI_RING_HEAD , 0 ) ;
OUTREG ( PRI_RING_TAIL , 0 ) ;
OUTREG ( PRI_RING_START , 0 ) ;
}
/* Stop the 2D engine, and turn off the ring buffer. */
2007-10-16 12:29:31 +04:00
void intelfbhw_2d_stop ( struct intelfb_info * dinfo )
2005-04-17 02:20:36 +04:00
{
# if VERBOSE > 0
2007-10-16 12:29:31 +04:00
DBG_MSG ( " intelfbhw_2d_stop: accel: %d, ring_active: %d \n " ,
dinfo - > accel , dinfo - > ring_active ) ;
2005-04-17 02:20:36 +04:00
# endif
if ( ! dinfo - > accel )
return ;
dinfo - > ring_active = 0 ;
reset_state ( dinfo ) ;
}
/*
* Enable the ring buffer , and initialise the 2 D engine .
* It is assumed that the graphics engine has been stopped by previously
* calling intelfb_2d_stop ( ) .
*/
2007-10-16 12:29:31 +04:00
void intelfbhw_2d_start ( struct intelfb_info * dinfo )
2005-04-17 02:20:36 +04:00
{
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_2d_start: accel: %d, ring_active: %d \n " ,
dinfo - > accel , dinfo - > ring_active ) ;
# endif
if ( ! dinfo - > accel )
return ;
/* Initialise the primary ring buffer. */
OUTREG ( PRI_RING_LENGTH , 0 ) ;
OUTREG ( PRI_RING_TAIL , 0 ) ;
OUTREG ( PRI_RING_HEAD , 0 ) ;
OUTREG ( PRI_RING_START , dinfo - > ring . physical & RING_START_MASK ) ;
OUTREG ( PRI_RING_LENGTH ,
( ( dinfo - > ring . size - GTT_PAGE_SIZE ) & RING_LENGTH_MASK ) |
RING_NO_REPORT | RING_ENABLE ) ;
refresh_ring ( dinfo ) ;
dinfo - > ring_active = 1 ;
}
/* 2D fillrect (solid fill or invert) */
2007-10-16 12:29:31 +04:00
void intelfbhw_do_fillrect ( struct intelfb_info * dinfo , u32 x , u32 y , u32 w ,
u32 h , u32 color , u32 pitch , u32 bpp , u32 rop )
2005-04-17 02:20:36 +04:00
{
u32 br00 , br09 , br13 , br14 , br16 ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_do_fillrect: (%d,%d) %dx%d, c 0x%06x, p %d bpp %d, "
" rop 0x%02x \n " , x , y , w , h , color , pitch , bpp , rop ) ;
# endif
br00 = COLOR_BLT_CMD ;
br09 = dinfo - > fb_start + ( y * pitch + x * ( bpp / 8 ) ) ;
br13 = ( rop < < ROP_SHIFT ) | pitch ;
br14 = ( h < < HEIGHT_SHIFT ) | ( ( w * ( bpp / 8 ) ) < < WIDTH_SHIFT ) ;
br16 = color ;
switch ( bpp ) {
case 8 :
br13 | = COLOR_DEPTH_8 ;
break ;
case 16 :
br13 | = COLOR_DEPTH_16 ;
break ;
case 32 :
br13 | = COLOR_DEPTH_32 ;
br00 | = WRITE_ALPHA | WRITE_RGB ;
break ;
}
START_RING ( 6 ) ;
OUT_RING ( br00 ) ;
OUT_RING ( br13 ) ;
OUT_RING ( br14 ) ;
OUT_RING ( br09 ) ;
OUT_RING ( br16 ) ;
OUT_RING ( MI_NOOP ) ;
ADVANCE_RING ( ) ;
# if VERBOSE > 0
DBG_MSG ( " ring = 0x%08x, 0x%08x (%d) \n " , dinfo - > ring_head ,
dinfo - > ring_tail , dinfo - > ring_space ) ;
# endif
}
void
intelfbhw_do_bitblt ( struct intelfb_info * dinfo , u32 curx , u32 cury ,
u32 dstx , u32 dsty , u32 w , u32 h , u32 pitch , u32 bpp )
{
u32 br00 , br09 , br11 , br12 , br13 , br22 , br23 , br26 ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_do_bitblt: (%d,%d)->(%d,%d) %dx%d, p %d bpp %d \n " ,
curx , cury , dstx , dsty , w , h , pitch , bpp ) ;
# endif
br00 = XY_SRC_COPY_BLT_CMD ;
br09 = dinfo - > fb_start ;
br11 = ( pitch < < PITCH_SHIFT ) ;
br12 = dinfo - > fb_start ;
br13 = ( SRC_ROP_GXCOPY < < ROP_SHIFT ) | ( pitch < < PITCH_SHIFT ) ;
br22 = ( dstx < < WIDTH_SHIFT ) | ( dsty < < HEIGHT_SHIFT ) ;
br23 = ( ( dstx + w ) < < WIDTH_SHIFT ) |
( ( dsty + h ) < < HEIGHT_SHIFT ) ;
br26 = ( curx < < WIDTH_SHIFT ) | ( cury < < HEIGHT_SHIFT ) ;
switch ( bpp ) {
case 8 :
br13 | = COLOR_DEPTH_8 ;
break ;
case 16 :
br13 | = COLOR_DEPTH_16 ;
break ;
case 32 :
br13 | = COLOR_DEPTH_32 ;
br00 | = WRITE_ALPHA | WRITE_RGB ;
break ;
}
START_RING ( 8 ) ;
OUT_RING ( br00 ) ;
OUT_RING ( br13 ) ;
OUT_RING ( br22 ) ;
OUT_RING ( br23 ) ;
OUT_RING ( br09 ) ;
OUT_RING ( br26 ) ;
OUT_RING ( br11 ) ;
OUT_RING ( br12 ) ;
ADVANCE_RING ( ) ;
}
2007-10-16 12:29:31 +04:00
int intelfbhw_do_drawglyph ( struct intelfb_info * dinfo , u32 fg , u32 bg , u32 w ,
u32 h , const u8 * cdat , u32 x , u32 y , u32 pitch ,
u32 bpp )
2005-04-17 02:20:36 +04:00
{
int nbytes , ndwords , pad , tmp ;
u32 br00 , br09 , br13 , br18 , br19 , br22 , br23 ;
int dat , ix , iy , iw ;
int i , j ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_do_drawglyph: (%d,%d) %dx%d \n " , x , y , w , h ) ;
# endif
/* size in bytes of a padded scanline */
nbytes = ROUND_UP_TO ( w , 16 ) / 8 ;
/* Total bytes of padded scanline data to write out. */
nbytes = nbytes * h ;
/*
* Check if the glyph data exceeds the immediate mode limit .
* It would take a large font ( 1 K pixels ) to hit this limit .
*/
if ( nbytes > MAX_MONO_IMM_SIZE )
return 0 ;
/* Src data is packaged a dword (32-bit) at a time. */
ndwords = ROUND_UP_TO ( nbytes , 4 ) / 4 ;
/*
* Ring has to be padded to a quad word . But because the command starts
with 7 bytes , pad only if there is an even number of ndwords
*/
pad = ! ( ndwords % 2 ) ;
tmp = ( XY_MONO_SRC_IMM_BLT_CMD & DW_LENGTH_MASK ) + ndwords ;
br00 = ( XY_MONO_SRC_IMM_BLT_CMD & ~ DW_LENGTH_MASK ) | tmp ;
br09 = dinfo - > fb_start ;
br13 = ( SRC_ROP_GXCOPY < < ROP_SHIFT ) | ( pitch < < PITCH_SHIFT ) ;
br18 = bg ;
br19 = fg ;
br22 = ( x < < WIDTH_SHIFT ) | ( y < < HEIGHT_SHIFT ) ;
br23 = ( ( x + w ) < < WIDTH_SHIFT ) | ( ( y + h ) < < HEIGHT_SHIFT ) ;
switch ( bpp ) {
case 8 :
br13 | = COLOR_DEPTH_8 ;
break ;
case 16 :
br13 | = COLOR_DEPTH_16 ;
break ;
case 32 :
br13 | = COLOR_DEPTH_32 ;
br00 | = WRITE_ALPHA | WRITE_RGB ;
break ;
}
START_RING ( 8 + ndwords ) ;
OUT_RING ( br00 ) ;
OUT_RING ( br13 ) ;
OUT_RING ( br22 ) ;
OUT_RING ( br23 ) ;
OUT_RING ( br09 ) ;
OUT_RING ( br18 ) ;
OUT_RING ( br19 ) ;
ix = iy = 0 ;
iw = ROUND_UP_TO ( w , 8 ) / 8 ;
while ( ndwords - - ) {
dat = 0 ;
for ( j = 0 ; j < 2 ; + + j ) {
for ( i = 0 ; i < 2 ; + + i ) {
if ( ix ! = iw | | i = = 0 )
dat | = cdat [ iy * iw + ix + + ] < < ( i + j * 2 ) * 8 ;
}
if ( ix = = iw & & iy ! = ( h - 1 ) ) {
ix = 0 ;
+ + iy ;
}
}
OUT_RING ( dat ) ;
}
if ( pad )
OUT_RING ( MI_NOOP ) ;
ADVANCE_RING ( ) ;
return 1 ;
}
/* HW cursor functions. */
2007-10-16 12:29:31 +04:00
void intelfbhw_cursor_init ( struct intelfb_info * dinfo )
2005-04-17 02:20:36 +04:00
{
u32 tmp ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_cursor_init \n " ) ;
# endif
2006-03-31 11:08:52 +04:00
if ( dinfo - > mobile | | IS_I9XX ( dinfo ) ) {
2005-04-17 02:20:36 +04:00
if ( ! dinfo - > cursor . physical )
return ;
tmp = INREG ( CURSOR_A_CONTROL ) ;
tmp & = ~ ( CURSOR_MODE_MASK | CURSOR_MOBILE_GAMMA_ENABLE |
CURSOR_MEM_TYPE_LOCAL |
( 1 < < CURSOR_PIPE_SELECT_SHIFT ) ) ;
tmp | = CURSOR_MODE_DISABLE ;
OUTREG ( CURSOR_A_CONTROL , tmp ) ;
OUTREG ( CURSOR_A_BASEADDR , dinfo - > cursor . physical ) ;
} else {
tmp = INREG ( CURSOR_CONTROL ) ;
tmp & = ~ ( CURSOR_FORMAT_MASK | CURSOR_GAMMA_ENABLE |
CURSOR_ENABLE | CURSOR_STRIDE_MASK ) ;
tmp = CURSOR_FORMAT_3C ;
OUTREG ( CURSOR_CONTROL , tmp ) ;
OUTREG ( CURSOR_A_BASEADDR , dinfo - > cursor . offset < < 12 ) ;
tmp = ( 64 < < CURSOR_SIZE_H_SHIFT ) |
( 64 < < CURSOR_SIZE_V_SHIFT ) ;
OUTREG ( CURSOR_SIZE , tmp ) ;
}
}
2007-10-16 12:29:31 +04:00
void intelfbhw_cursor_hide ( struct intelfb_info * dinfo )
2005-04-17 02:20:36 +04:00
{
u32 tmp ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_cursor_hide \n " ) ;
# endif
dinfo - > cursor_on = 0 ;
2006-03-31 11:08:52 +04:00
if ( dinfo - > mobile | | IS_I9XX ( dinfo ) ) {
2005-04-17 02:20:36 +04:00
if ( ! dinfo - > cursor . physical )
return ;
tmp = INREG ( CURSOR_A_CONTROL ) ;
tmp & = ~ CURSOR_MODE_MASK ;
tmp | = CURSOR_MODE_DISABLE ;
OUTREG ( CURSOR_A_CONTROL , tmp ) ;
/* Flush changes */
OUTREG ( CURSOR_A_BASEADDR , dinfo - > cursor . physical ) ;
} else {
tmp = INREG ( CURSOR_CONTROL ) ;
tmp & = ~ CURSOR_ENABLE ;
OUTREG ( CURSOR_CONTROL , tmp ) ;
}
}
2007-10-16 12:29:31 +04:00
void intelfbhw_cursor_show ( struct intelfb_info * dinfo )
2005-04-17 02:20:36 +04:00
{
u32 tmp ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_cursor_show \n " ) ;
# endif
dinfo - > cursor_on = 1 ;
if ( dinfo - > cursor_blanked )
return ;
2006-03-31 11:08:52 +04:00
if ( dinfo - > mobile | | IS_I9XX ( dinfo ) ) {
2005-04-17 02:20:36 +04:00
if ( ! dinfo - > cursor . physical )
return ;
tmp = INREG ( CURSOR_A_CONTROL ) ;
tmp & = ~ CURSOR_MODE_MASK ;
tmp | = CURSOR_MODE_64_4C_AX ;
OUTREG ( CURSOR_A_CONTROL , tmp ) ;
/* Flush changes */
OUTREG ( CURSOR_A_BASEADDR , dinfo - > cursor . physical ) ;
} else {
tmp = INREG ( CURSOR_CONTROL ) ;
tmp | = CURSOR_ENABLE ;
OUTREG ( CURSOR_CONTROL , tmp ) ;
}
}
2007-10-16 12:29:31 +04:00
void intelfbhw_cursor_setpos ( struct intelfb_info * dinfo , int x , int y )
2005-04-17 02:20:36 +04:00
{
u32 tmp ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_cursor_setpos: (%d, %d) \n " , x , y ) ;
# endif
/*
2006-03-31 11:08:52 +04:00
* Sets the position . The coordinates are assumed to already
* have any offset adjusted . Assume that the cursor is never
2005-04-17 02:20:36 +04:00
* completely off - screen , and that x , y are always > = 0.
*/
tmp = ( ( x & CURSOR_POS_MASK ) < < CURSOR_X_SHIFT ) |
( ( y & CURSOR_POS_MASK ) < < CURSOR_Y_SHIFT ) ;
OUTREG ( CURSOR_A_POSITION , tmp ) ;
2006-03-23 05:06:32 +03:00
2007-10-16 12:29:31 +04:00
if ( IS_I9XX ( dinfo ) )
2006-03-23 05:06:32 +03:00
OUTREG ( CURSOR_A_BASEADDR , dinfo - > cursor . physical ) ;
2005-04-17 02:20:36 +04:00
}
2007-10-16 12:29:31 +04:00
void intelfbhw_cursor_setcolor ( struct intelfb_info * dinfo , u32 bg , u32 fg )
2005-04-17 02:20:36 +04:00
{
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_cursor_setcolor \n " ) ;
# endif
OUTREG ( CURSOR_A_PALETTE0 , bg & CURSOR_PALETTE_MASK ) ;
OUTREG ( CURSOR_A_PALETTE1 , fg & CURSOR_PALETTE_MASK ) ;
OUTREG ( CURSOR_A_PALETTE2 , fg & CURSOR_PALETTE_MASK ) ;
OUTREG ( CURSOR_A_PALETTE3 , bg & CURSOR_PALETTE_MASK ) ;
}
2007-10-16 12:29:31 +04:00
void intelfbhw_cursor_load ( struct intelfb_info * dinfo , int width , int height ,
u8 * data )
2005-04-17 02:20:36 +04:00
{
u8 __iomem * addr = ( u8 __iomem * ) dinfo - > cursor . virtual ;
int i , j , w = width / 8 ;
int mod = width % 8 , t_mask , d_mask ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_cursor_load \n " ) ;
# endif
if ( ! dinfo - > cursor . virtual )
return ;
t_mask = 0xff > > mod ;
d_mask = ~ ( 0xff > > mod ) ;
for ( i = height ; i - - ; ) {
for ( j = 0 ; j < w ; j + + ) {
writeb ( 0x00 , addr + j ) ;
writeb ( * ( data + + ) , addr + j + 8 ) ;
}
if ( mod ) {
writeb ( t_mask , addr + j ) ;
writeb ( * ( data + + ) & d_mask , addr + j + 8 ) ;
}
addr + = 16 ;
}
}
2007-10-16 12:29:31 +04:00
void intelfbhw_cursor_reset ( struct intelfb_info * dinfo )
{
2005-04-17 02:20:36 +04:00
u8 __iomem * addr = ( u8 __iomem * ) dinfo - > cursor . virtual ;
int i , j ;
# if VERBOSE > 0
DBG_MSG ( " intelfbhw_cursor_reset \n " ) ;
# endif
if ( ! dinfo - > cursor . virtual )
return ;
for ( i = 64 ; i - - ; ) {
for ( j = 0 ; j < 8 ; j + + ) {
writeb ( 0xff , addr + j + 0 ) ;
writeb ( 0x00 , addr + j + 8 ) ;
}
addr + = 16 ;
}
}
2006-06-20 22:36:41 +04:00
2007-10-16 12:29:34 +04:00
static irqreturn_t intelfbhw_irq ( int irq , void * dev_id )
{
2006-06-20 22:36:41 +04:00
u16 tmp ;
2008-02-06 12:36:20 +03:00
struct intelfb_info * dinfo = dev_id ;
2006-06-20 22:36:41 +04:00
spin_lock ( & dinfo - > int_lock ) ;
tmp = INREG16 ( IIR ) ;
2007-10-16 12:29:34 +04:00
if ( dinfo - > info - > var . vmode & FB_VMODE_INTERLACED )
tmp & = PIPE_A_EVENT_INTERRUPT ;
else
tmp & = VSYNC_PIPE_A_INTERRUPT ; /* non-interlaced */
2006-06-20 22:36:41 +04:00
if ( tmp = = 0 ) {
spin_unlock ( & dinfo - > int_lock ) ;
2007-10-16 12:29:34 +04:00
return IRQ_RETVAL ( 0 ) ; /* not us */
2006-06-20 22:36:41 +04:00
}
2007-10-16 12:29:34 +04:00
/* clear status bits 0-15 ASAP and don't touch bits 16-31 */
OUTREG ( PIPEASTAT , INREG ( PIPEASTAT ) ) ;
2006-06-20 22:36:41 +04:00
2007-10-16 12:29:34 +04:00
OUTREG16 ( IIR , tmp ) ;
if ( dinfo - > vsync . pan_display ) {
dinfo - > vsync . pan_display = 0 ;
OUTREG ( DSPABASE , dinfo - > vsync . pan_offset ) ;
2006-06-20 22:36:41 +04:00
}
2007-10-16 12:29:34 +04:00
dinfo - > vsync . count + + ;
wake_up_interruptible ( & dinfo - > vsync . wait ) ;
2006-06-20 22:36:41 +04:00
spin_unlock ( & dinfo - > int_lock ) ;
2007-10-16 12:29:34 +04:00
return IRQ_RETVAL ( 1 ) ;
2006-06-20 22:36:41 +04:00
}
2007-10-16 12:29:34 +04:00
int intelfbhw_enable_irq ( struct intelfb_info * dinfo )
{
u16 tmp ;
2006-06-20 22:36:41 +04:00
if ( ! test_and_set_bit ( 0 , & dinfo - > irq_flags ) ) {
2007-02-14 11:33:16 +03:00
if ( request_irq ( dinfo - > pdev - > irq , intelfbhw_irq , IRQF_SHARED ,
2007-10-16 12:29:34 +04:00
" intelfb " , dinfo ) ) {
2006-06-20 22:36:41 +04:00
clear_bit ( 0 , & dinfo - > irq_flags ) ;
return - EINVAL ;
}
spin_lock_irq ( & dinfo - > int_lock ) ;
2007-10-16 12:29:34 +04:00
OUTREG16 ( HWSTAM , 0xfffe ) ; /* i830 DRM uses ffff */
OUTREG16 ( IMR , 0 ) ;
} else
2006-06-20 22:36:41 +04:00
spin_lock_irq ( & dinfo - > int_lock ) ;
2007-10-16 12:29:34 +04:00
if ( dinfo - > info - > var . vmode & FB_VMODE_INTERLACED )
tmp = PIPE_A_EVENT_INTERRUPT ;
else
tmp = VSYNC_PIPE_A_INTERRUPT ; /* non-interlaced */
if ( tmp ! = INREG16 ( IER ) ) {
DBG_MSG ( " changing IER to 0x%X \n " , tmp ) ;
OUTREG16 ( IER , tmp ) ;
2006-06-20 22:36:41 +04:00
}
2007-10-16 12:29:34 +04:00
spin_unlock_irq ( & dinfo - > int_lock ) ;
2006-06-20 22:36:41 +04:00
return 0 ;
}
2007-10-16 12:29:34 +04:00
void intelfbhw_disable_irq ( struct intelfb_info * dinfo )
{
2006-06-20 22:36:41 +04:00
if ( test_and_clear_bit ( 0 , & dinfo - > irq_flags ) ) {
2006-06-20 22:36:42 +04:00
if ( dinfo - > vsync . pan_display ) {
dinfo - > vsync . pan_display = 0 ;
OUTREG ( DSPABASE , dinfo - > vsync . pan_offset ) ;
}
2006-06-20 22:36:41 +04:00
spin_lock_irq ( & dinfo - > int_lock ) ;
OUTREG16 ( HWSTAM , 0xffff ) ;
OUTREG16 ( IMR , 0xffff ) ;
OUTREG16 ( IER , 0x0 ) ;
2007-10-16 12:29:33 +04:00
OUTREG16 ( IIR , INREG16 ( IIR ) ) ; /* clear IRQ requests */
2006-06-20 22:36:41 +04:00
spin_unlock_irq ( & dinfo - > int_lock ) ;
free_irq ( dinfo - > pdev - > irq , dinfo ) ;
}
}
2006-06-20 22:36:42 +04:00
2007-10-16 12:29:31 +04:00
int intelfbhw_wait_for_vsync ( struct intelfb_info * dinfo , u32 pipe )
{
2006-06-20 22:36:42 +04:00
struct intelfb_vsync * vsync ;
unsigned int count ;
int ret ;
switch ( pipe ) {
case 0 :
vsync = & dinfo - > vsync ;
break ;
default :
return - ENODEV ;
}
2007-10-16 12:29:34 +04:00
ret = intelfbhw_enable_irq ( dinfo ) ;
2007-10-16 12:29:31 +04:00
if ( ret )
2006-06-20 22:36:42 +04:00
return ret ;
count = vsync - > count ;
2007-10-16 12:29:31 +04:00
ret = wait_event_interruptible_timeout ( vsync - > wait ,
count ! = vsync - > count , HZ / 10 ) ;
if ( ret < 0 )
2006-06-20 22:36:42 +04:00
return ret ;
if ( ret = = 0 ) {
DBG_MSG ( " wait_for_vsync timed out! \n " ) ;
return - ETIMEDOUT ;
}
return 0 ;
}