2005-04-17 02:20:36 +04:00
/*
Copyright ( c ) 1998 - 2002 Frodo Looijaard < frodol @ dds . nl > ,
Philip Edelbrock < phil @ netroedge . com > , and Mark D . Studebaker
< mdsxyz123 @ yahoo . com >
2008-07-15 00:38:33 +04:00
Copyright ( C ) 2007 , 2008 Jean Delvare < khali @ linux - fr . org >
2010-10-31 23:06:59 +03:00
Copyright ( C ) 2010 Intel Corporation ,
David Woodhouse < dwmw2 @ infradead . org >
2005-04-17 02:20:36 +04:00
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation ; either version 2 of the License , or
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
along with this program ; if not , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
/*
2008-01-27 20:14:49 +03:00
Supports the following Intel I / O Controller Hubs ( ICH ) :
I / O Block I2C
region SMBus Block proc . block
Chip name PCI ID size PEC buffer call read
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
82801 AA ( ICH ) 0x2413 16 no no no no
82801 AB ( ICH0 ) 0x2423 16 no no no no
82801 BA ( ICH2 ) 0x2443 16 no no no no
82801 CA ( ICH3 ) 0x2483 32 soft no no no
82801 DB ( ICH4 ) 0x24c3 32 hard yes no no
82801 E ( ICH5 ) 0x24d3 32 hard yes yes yes
6300 ESB 0x25a4 32 hard yes yes yes
82801F ( ICH6 ) 0x266a 32 hard yes yes yes
6310 ESB / 6320 ESB 0x269b 32 hard yes yes yes
82801 G ( ICH7 ) 0x27da 32 hard yes yes yes
82801 H ( ICH8 ) 0x283e 32 hard yes yes yes
82801 I ( ICH9 ) 0x2930 32 hard yes yes yes
2010-10-05 00:27:14 +04:00
EP80579 ( Tolapai ) 0x5032 32 hard yes yes yes
2008-02-24 22:03:42 +03:00
ICH10 0x3a30 32 hard yes yes yes
ICH10 0x3a60 32 hard yes yes yes
2010-10-05 00:27:14 +04:00
5 / 3400 Series ( PCH ) 0x3b30 32 hard yes yes yes
2010-03-02 14:23:39 +03:00
Cougar Point ( PCH ) 0x1c22 32 hard yes yes yes
2010-10-31 23:06:59 +03:00
Patsburg ( PCH ) 0x1d22 32 hard yes yes yes
2010-10-31 23:07:00 +03:00
Patsburg ( PCH ) IDF 0x1d70 32 hard yes yes yes
Patsburg ( PCH ) IDF 0x1d71 32 hard yes yes yes
Patsburg ( PCH ) IDF 0x1d72 32 hard yes yes yes
2008-01-27 20:14:49 +03:00
Features supported by this driver :
Software PEC no
Hardware PEC yes
Block buffer yes
Block process call transaction no
2008-01-27 20:14:50 +03:00
I2C block read transaction yes ( doesn ' t use the block buffer )
2010-10-31 23:07:00 +03:00
Slave mode no
2008-01-27 20:14:49 +03:00
See the file Documentation / i2c / busses / i2c - i801 for details .
2005-04-17 02:20:36 +04:00
*/
# include <linux/module.h>
# include <linux/pci.h>
# include <linux/kernel.h>
# include <linux/stddef.h>
# include <linux/delay.h>
# include <linux/ioport.h>
# include <linux/init.h>
# include <linux/i2c.h>
2008-07-15 00:38:33 +04:00
# include <linux/acpi.h>
2009-01-07 16:29:17 +03:00
# include <linux/io.h>
2009-03-30 23:46:44 +04:00
# include <linux/dmi.h>
2005-04-17 02:20:36 +04:00
/* I801 SMBus address offsets */
2010-10-31 23:06:59 +03:00
# define SMBHSTSTS(p) (0 + (p)->smba)
# define SMBHSTCNT(p) (2 + (p)->smba)
# define SMBHSTCMD(p) (3 + (p)->smba)
# define SMBHSTADD(p) (4 + (p)->smba)
# define SMBHSTDAT0(p) (5 + (p)->smba)
# define SMBHSTDAT1(p) (6 + (p)->smba)
# define SMBBLKDAT(p) (7 + (p)->smba)
# define SMBPEC(p) (8 + (p)->smba) /* ICH3 and later */
# define SMBAUXSTS(p) (12 + (p)->smba) /* ICH4 and later */
# define SMBAUXCTL(p) (13 + (p)->smba) /* ICH4 and later */
2005-04-17 02:20:36 +04:00
/* PCI Address Constants */
2006-06-12 23:53:02 +04:00
# define SMBBAR 4
2005-04-17 02:20:36 +04:00
# define SMBHSTCFG 0x040
/* Host configuration bits for SMBHSTCFG */
# define SMBHSTCFG_HST_EN 1
# define SMBHSTCFG_SMB_SMI_EN 2
# define SMBHSTCFG_I2C_EN 4
2007-07-12 16:12:31 +04:00
/* Auxillary control register bits, ICH4+ only */
# define SMBAUXCTL_CRC 1
# define SMBAUXCTL_E32B 2
/* kill bit for SMBHSTCNT */
# define SMBHSTCNT_KILL 2
2005-04-17 02:20:36 +04:00
/* Other settings */
# define MAX_TIMEOUT 100
# define ENABLE_INT9 0 /* set to 0x01 to enable - untested */
/* I801 command constants */
# define I801_QUICK 0x00
# define I801_BYTE 0x04
# define I801_BYTE_DATA 0x08
# define I801_WORD_DATA 0x0C
2008-01-27 20:14:49 +03:00
# define I801_PROC_CALL 0x10 /* unimplemented */
2005-04-17 02:20:36 +04:00
# define I801_BLOCK_DATA 0x14
2008-01-27 20:14:50 +03:00
# define I801_I2C_BLOCK_DATA 0x18 /* ICH5 and later */
2005-04-17 02:20:36 +04:00
# define I801_BLOCK_LAST 0x34
2008-01-27 20:14:50 +03:00
# define I801_I2C_BLOCK_LAST 0x38 /* ICH5 and later */
2005-04-17 02:20:36 +04:00
# define I801_START 0x40
2008-01-27 20:14:49 +03:00
# define I801_PEC_EN 0x80 /* ICH3 and later */
2005-04-17 02:20:36 +04:00
2007-07-12 16:12:31 +04:00
/* I801 Hosts Status register bits */
# define SMBHSTSTS_BYTE_DONE 0x80
# define SMBHSTSTS_INUSE_STS 0x40
# define SMBHSTSTS_SMBALERT_STS 0x20
# define SMBHSTSTS_FAILED 0x10
# define SMBHSTSTS_BUS_ERR 0x08
# define SMBHSTSTS_DEV_ERR 0x04
# define SMBHSTSTS_INTR 0x02
# define SMBHSTSTS_HOST_BUSY 0x01
2005-04-17 02:20:36 +04:00
2008-07-15 00:38:33 +04:00
# define STATUS_FLAGS (SMBHSTSTS_BYTE_DONE | SMBHSTSTS_FAILED | \
SMBHSTSTS_BUS_ERR | SMBHSTSTS_DEV_ERR | \
SMBHSTSTS_INTR )
2010-10-31 23:07:00 +03:00
/* Patsburg also has three 'Integrated Device Function' SMBus controllers */
# define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF0 0x1d70
# define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF1 0x1d71
# define PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF2 0x1d72
2010-10-31 23:06:59 +03:00
struct i801_priv {
struct i2c_adapter adapter ;
unsigned long smba ;
unsigned char original_hstcfg ;
struct pci_dev * pci_dev ;
unsigned int features ;
} ;
2005-09-25 18:37:04 +04:00
static struct pci_driver i801_driver ;
2008-01-27 20:14:50 +03:00
# define FEATURE_SMBUS_PEC (1 << 0)
# define FEATURE_BLOCK_BUFFER (1 << 1)
# define FEATURE_BLOCK_PROC (1 << 2)
# define FEATURE_I2C_BLOCK_READ (1 << 3)
2005-04-17 02:20:36 +04:00
2010-05-21 20:40:54 +04:00
static const char * i801_feature_names [ ] = {
" SMBus PEC " ,
" Block buffer " ,
" Block process call " ,
" I2C block read " ,
} ;
static unsigned int disable_features ;
module_param ( disable_features , uint , S_IRUGO | S_IWUSR ) ;
MODULE_PARM_DESC ( disable_features , " Disable selected driver features " ) ;
2008-07-15 00:38:33 +04:00
/* Make sure the SMBus host is ready to start transmitting.
Return 0 if it is , - EBUSY if it is not . */
2010-10-31 23:06:59 +03:00
static int i801_check_pre ( struct i801_priv * priv )
2005-04-17 02:20:36 +04:00
{
2008-07-15 00:38:32 +04:00
int status ;
2005-04-17 02:20:36 +04:00
2010-10-31 23:06:59 +03:00
status = inb_p ( SMBHSTSTS ( priv ) ) ;
2008-07-15 00:38:33 +04:00
if ( status & SMBHSTSTS_HOST_BUSY ) {
2010-10-31 23:06:59 +03:00
dev_err ( & priv - > pci_dev - > dev , " SMBus is busy, can't use it! \n " ) ;
2008-07-15 00:38:33 +04:00
return - EBUSY ;
}
status & = STATUS_FLAGS ;
if ( status ) {
2010-10-31 23:06:59 +03:00
dev_dbg ( & priv - > pci_dev - > dev , " Clearing status flags (%02x) \n " ,
2008-07-15 00:38:32 +04:00
status ) ;
2010-10-31 23:06:59 +03:00
outb_p ( status , SMBHSTSTS ( priv ) ) ;
status = inb_p ( SMBHSTSTS ( priv ) ) & STATUS_FLAGS ;
2008-07-15 00:38:33 +04:00
if ( status ) {
2010-10-31 23:06:59 +03:00
dev_err ( & priv - > pci_dev - > dev ,
2008-07-15 00:38:33 +04:00
" Failed clearing status flags (%02x) \n " ,
status ) ;
2008-07-15 00:38:25 +04:00
return - EBUSY ;
2005-04-17 02:20:36 +04:00
}
}
2008-07-15 00:38:33 +04:00
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-07-15 00:38:33 +04:00
/* Convert the status register to an error code, and clear it. */
2010-10-31 23:06:59 +03:00
static int i801_check_post ( struct i801_priv * priv , int status , int timeout )
2008-07-15 00:38:33 +04:00
{
int result = 0 ;
2005-04-17 02:20:36 +04:00
/* If the SMBus is still busy, we give up */
2008-07-15 00:38:33 +04:00
if ( timeout ) {
2010-10-31 23:06:59 +03:00
dev_err ( & priv - > pci_dev - > dev , " Transaction timeout \n " ) ;
2007-07-12 16:12:31 +04:00
/* try to stop the current command */
2010-10-31 23:06:59 +03:00
dev_dbg ( & priv - > pci_dev - > dev , " Terminating the current operation \n " ) ;
outb_p ( inb_p ( SMBHSTCNT ( priv ) ) | SMBHSTCNT_KILL ,
SMBHSTCNT ( priv ) ) ;
2007-07-12 16:12:31 +04:00
msleep ( 1 ) ;
2010-10-31 23:06:59 +03:00
outb_p ( inb_p ( SMBHSTCNT ( priv ) ) & ( ~ SMBHSTCNT_KILL ) ,
SMBHSTCNT ( priv ) ) ;
2008-07-15 00:38:33 +04:00
/* Check if it worked */
2010-10-31 23:06:59 +03:00
status = inb_p ( SMBHSTSTS ( priv ) ) ;
2008-07-15 00:38:33 +04:00
if ( ( status & SMBHSTSTS_HOST_BUSY ) | |
! ( status & SMBHSTSTS_FAILED ) )
2010-10-31 23:06:59 +03:00
dev_err ( & priv - > pci_dev - > dev ,
2008-07-15 00:38:33 +04:00
" Failed terminating the transaction \n " ) ;
2010-10-31 23:06:59 +03:00
outb_p ( STATUS_FLAGS , SMBHSTSTS ( priv ) ) ;
2008-07-15 00:38:33 +04:00
return - ETIMEDOUT ;
2005-04-17 02:20:36 +04:00
}
2008-07-15 00:38:32 +04:00
if ( status & SMBHSTSTS_FAILED ) {
2008-07-15 00:38:25 +04:00
result = - EIO ;
2010-10-31 23:06:59 +03:00
dev_err ( & priv - > pci_dev - > dev , " Transaction failed \n " ) ;
2008-07-15 00:38:33 +04:00
}
if ( status & SMBHSTSTS_DEV_ERR ) {
result = - ENXIO ;
2010-10-31 23:06:59 +03:00
dev_dbg ( & priv - > pci_dev - > dev , " No response \n " ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-15 00:38:32 +04:00
if ( status & SMBHSTSTS_BUS_ERR ) {
2008-07-15 00:38:32 +04:00
result = - EAGAIN ;
2010-10-31 23:06:59 +03:00
dev_dbg ( & priv - > pci_dev - > dev , " Lost arbitration \n " ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-15 00:38:33 +04:00
if ( result ) {
/* Clear error flags */
2010-10-31 23:06:59 +03:00
outb_p ( status & STATUS_FLAGS , SMBHSTSTS ( priv ) ) ;
status = inb_p ( SMBHSTSTS ( priv ) ) & STATUS_FLAGS ;
2008-07-15 00:38:33 +04:00
if ( status ) {
2010-10-31 23:06:59 +03:00
dev_warn ( & priv - > pci_dev - > dev , " Failed clearing status "
2008-07-15 00:38:33 +04:00
" flags at end of transaction (%02x) \n " ,
status ) ;
}
2005-04-17 02:20:36 +04:00
}
return result ;
}
2010-10-31 23:06:59 +03:00
static int i801_transaction ( struct i801_priv * priv , int xact )
2008-07-15 00:38:33 +04:00
{
int status ;
int result ;
int timeout = 0 ;
2010-10-31 23:06:59 +03:00
result = i801_check_pre ( priv ) ;
2008-07-15 00:38:33 +04:00
if ( result < 0 )
return result ;
/* the current contents of SMBHSTCNT can be overwritten, since PEC,
* INTREN , SMBSCMD are passed in xact */
2010-10-31 23:06:59 +03:00
outb_p ( xact | I801_START , SMBHSTCNT ( priv ) ) ;
2008-07-15 00:38:33 +04:00
/* We will always wait for a fraction of a second! */
do {
msleep ( 1 ) ;
2010-10-31 23:06:59 +03:00
status = inb_p ( SMBHSTSTS ( priv ) ) ;
2008-07-15 00:38:33 +04:00
} while ( ( status & SMBHSTSTS_HOST_BUSY ) & & ( timeout + + < MAX_TIMEOUT ) ) ;
2010-10-31 23:06:59 +03:00
result = i801_check_post ( priv , status , timeout > MAX_TIMEOUT ) ;
2008-07-15 00:38:33 +04:00
if ( result < 0 )
return result ;
2010-10-31 23:06:59 +03:00
outb_p ( SMBHSTSTS_INTR , SMBHSTSTS ( priv ) ) ;
2008-07-15 00:38:33 +04:00
return 0 ;
}
2007-07-12 16:12:31 +04:00
/* wait for INTR bit as advised by Intel */
2010-10-31 23:06:59 +03:00
static void i801_wait_hwpec ( struct i801_priv * priv )
2007-07-12 16:12:31 +04:00
{
int timeout = 0 ;
2008-07-15 00:38:32 +04:00
int status ;
2007-07-12 16:12:31 +04:00
do {
msleep ( 1 ) ;
2010-10-31 23:06:59 +03:00
status = inb_p ( SMBHSTSTS ( priv ) ) ;
2008-07-15 00:38:32 +04:00
} while ( ( ! ( status & SMBHSTSTS_INTR ) )
2007-07-12 16:12:31 +04:00
& & ( timeout + + < MAX_TIMEOUT ) ) ;
2009-05-05 10:39:24 +04:00
if ( timeout > MAX_TIMEOUT )
2010-10-31 23:06:59 +03:00
dev_dbg ( & priv - > pci_dev - > dev , " PEC Timeout! \n " ) ;
2009-05-05 10:39:24 +04:00
2010-10-31 23:06:59 +03:00
outb_p ( status , SMBHSTSTS ( priv ) ) ;
2007-07-12 16:12:31 +04:00
}
2010-10-31 23:06:59 +03:00
static int i801_block_transaction_by_block ( struct i801_priv * priv ,
union i2c_smbus_data * data ,
2007-07-12 16:12:31 +04:00
char read_write , int hwpec )
{
int i , len ;
2008-07-15 00:38:25 +04:00
int status ;
2007-07-12 16:12:31 +04:00
2010-10-31 23:06:59 +03:00
inb_p ( SMBHSTCNT ( priv ) ) ; /* reset the data buffer index */
2007-07-12 16:12:31 +04:00
/* Use 32-byte buffer to process this transaction */
if ( read_write = = I2C_SMBUS_WRITE ) {
len = data - > block [ 0 ] ;
2010-10-31 23:06:59 +03:00
outb_p ( len , SMBHSTDAT0 ( priv ) ) ;
2007-07-12 16:12:31 +04:00
for ( i = 0 ; i < len ; i + + )
2010-10-31 23:06:59 +03:00
outb_p ( data - > block [ i + 1 ] , SMBBLKDAT ( priv ) ) ;
2007-07-12 16:12:31 +04:00
}
2010-10-31 23:06:59 +03:00
status = i801_transaction ( priv , I801_BLOCK_DATA | ENABLE_INT9 |
2008-07-15 00:38:25 +04:00
I801_PEC_EN * hwpec ) ;
if ( status )
return status ;
2007-07-12 16:12:31 +04:00
if ( read_write = = I2C_SMBUS_READ ) {
2010-10-31 23:06:59 +03:00
len = inb_p ( SMBHSTDAT0 ( priv ) ) ;
2007-07-12 16:12:31 +04:00
if ( len < 1 | | len > I2C_SMBUS_BLOCK_MAX )
2008-07-15 00:38:25 +04:00
return - EPROTO ;
2007-07-12 16:12:31 +04:00
data - > block [ 0 ] = len ;
for ( i = 0 ; i < len ; i + + )
2010-10-31 23:06:59 +03:00
data - > block [ i + 1 ] = inb_p ( SMBBLKDAT ( priv ) ) ;
2007-07-12 16:12:31 +04:00
}
return 0 ;
}
2010-10-31 23:06:59 +03:00
static int i801_block_transaction_byte_by_byte ( struct i801_priv * priv ,
union i2c_smbus_data * data ,
2008-01-27 20:14:50 +03:00
char read_write , int command ,
int hwpec )
2005-04-17 02:20:36 +04:00
{
int i , len ;
int smbcmd ;
2008-07-15 00:38:32 +04:00
int status ;
2008-07-15 00:38:33 +04:00
int result ;
2005-04-17 02:20:36 +04:00
int timeout ;
2008-07-15 00:38:33 +04:00
2010-10-31 23:06:59 +03:00
result = i801_check_pre ( priv ) ;
2008-07-15 00:38:33 +04:00
if ( result < 0 )
return result ;
2005-04-17 02:20:36 +04:00
2007-07-12 16:12:31 +04:00
len = data - > block [ 0 ] ;
2005-04-17 02:20:36 +04:00
if ( read_write = = I2C_SMBUS_WRITE ) {
2010-10-31 23:06:59 +03:00
outb_p ( len , SMBHSTDAT0 ( priv ) ) ;
outb_p ( data - > block [ 1 ] , SMBBLKDAT ( priv ) ) ;
2005-04-17 02:20:36 +04:00
}
for ( i = 1 ; i < = len ; i + + ) {
2008-01-27 20:14:50 +03:00
if ( i = = len & & read_write = = I2C_SMBUS_READ ) {
if ( command = = I2C_SMBUS_I2C_BLOCK_DATA )
smbcmd = I801_I2C_BLOCK_LAST ;
else
smbcmd = I801_BLOCK_LAST ;
} else {
if ( command = = I2C_SMBUS_I2C_BLOCK_DATA
& & read_write = = I2C_SMBUS_READ )
smbcmd = I801_I2C_BLOCK_DATA ;
else
smbcmd = I801_BLOCK_DATA ;
}
2010-10-31 23:06:59 +03:00
outb_p ( smbcmd | ENABLE_INT9 , SMBHSTCNT ( priv ) ) ;
2005-04-17 02:20:36 +04:00
if ( i = = 1 )
2010-10-31 23:06:59 +03:00
outb_p ( inb ( SMBHSTCNT ( priv ) ) | I801_START ,
SMBHSTCNT ( priv ) ) ;
2005-04-17 02:20:36 +04:00
/* We will always wait for a fraction of a second! */
timeout = 0 ;
do {
msleep ( 1 ) ;
2010-10-31 23:06:59 +03:00
status = inb_p ( SMBHSTSTS ( priv ) ) ;
2010-05-21 20:40:55 +04:00
} while ( ( ! ( status & SMBHSTSTS_BYTE_DONE ) )
& & ( timeout + + < MAX_TIMEOUT ) ) ;
2005-04-17 02:20:36 +04:00
2010-10-31 23:06:59 +03:00
result = i801_check_post ( priv , status , timeout > MAX_TIMEOUT ) ;
2008-07-15 00:38:33 +04:00
if ( result < 0 )
return result ;
2005-04-17 02:20:36 +04:00
2008-01-27 20:14:50 +03:00
if ( i = = 1 & & read_write = = I2C_SMBUS_READ
& & command ! = I2C_SMBUS_I2C_BLOCK_DATA ) {
2010-10-31 23:06:59 +03:00
len = inb_p ( SMBHSTDAT0 ( priv ) ) ;
2008-07-15 00:38:33 +04:00
if ( len < 1 | | len > I2C_SMBUS_BLOCK_MAX ) {
2010-10-31 23:06:59 +03:00
dev_err ( & priv - > pci_dev - > dev ,
2008-07-15 00:38:33 +04:00
" Illegal SMBus block read size %d \n " ,
len ) ;
/* Recover */
2010-10-31 23:06:59 +03:00
while ( inb_p ( SMBHSTSTS ( priv ) ) &
SMBHSTSTS_HOST_BUSY )
outb_p ( SMBHSTSTS_BYTE_DONE ,
SMBHSTSTS ( priv ) ) ;
outb_p ( SMBHSTSTS_INTR , SMBHSTSTS ( priv ) ) ;
2008-07-15 00:38:25 +04:00
return - EPROTO ;
2008-07-15 00:38:33 +04:00
}
2005-04-17 02:20:36 +04:00
data - > block [ 0 ] = len ;
}
/* Retrieve/store value in SMBBLKDAT */
if ( read_write = = I2C_SMBUS_READ )
2010-10-31 23:06:59 +03:00
data - > block [ i ] = inb_p ( SMBBLKDAT ( priv ) ) ;
2005-04-17 02:20:36 +04:00
if ( read_write = = I2C_SMBUS_WRITE & & i + 1 < = len )
2010-10-31 23:06:59 +03:00
outb_p ( data - > block [ i + 1 ] , SMBBLKDAT ( priv ) ) ;
2005-04-17 02:20:36 +04:00
2008-07-15 00:38:33 +04:00
/* signals SMBBLKDAT ready */
2010-10-31 23:06:59 +03:00
outb_p ( SMBHSTSTS_BYTE_DONE | SMBHSTSTS_INTR , SMBHSTSTS ( priv ) ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-15 00:38:33 +04:00
return 0 ;
2007-07-12 16:12:31 +04:00
}
2005-04-17 02:20:36 +04:00
2010-10-31 23:06:59 +03:00
static int i801_set_block_buffer_mode ( struct i801_priv * priv )
2007-07-12 16:12:31 +04:00
{
2010-10-31 23:06:59 +03:00
outb_p ( inb_p ( SMBAUXCTL ( priv ) ) | SMBAUXCTL_E32B , SMBAUXCTL ( priv ) ) ;
if ( ( inb_p ( SMBAUXCTL ( priv ) ) & SMBAUXCTL_E32B ) = = 0 )
2008-07-15 00:38:25 +04:00
return - EIO ;
2007-07-12 16:12:31 +04:00
return 0 ;
}
/* Block transaction function */
2010-10-31 23:06:59 +03:00
static int i801_block_transaction ( struct i801_priv * priv ,
union i2c_smbus_data * data , char read_write ,
2007-07-12 16:12:31 +04:00
int command , int hwpec )
{
int result = 0 ;
unsigned char hostc ;
if ( command = = I2C_SMBUS_I2C_BLOCK_DATA ) {
if ( read_write = = I2C_SMBUS_WRITE ) {
/* set I2C_EN bit in configuration register */
2010-10-31 23:06:59 +03:00
pci_read_config_byte ( priv - > pci_dev , SMBHSTCFG , & hostc ) ;
pci_write_config_byte ( priv - > pci_dev , SMBHSTCFG ,
2007-07-12 16:12:31 +04:00
hostc | SMBHSTCFG_I2C_EN ) ;
2010-10-31 23:06:59 +03:00
} else if ( ! ( priv - > features & FEATURE_I2C_BLOCK_READ ) ) {
dev_err ( & priv - > pci_dev - > dev ,
2008-01-27 20:14:50 +03:00
" I2C block read is unsupported! \n " ) ;
2008-07-15 00:38:25 +04:00
return - EOPNOTSUPP ;
2007-07-12 16:12:31 +04:00
}
}
2008-01-27 20:14:50 +03:00
if ( read_write = = I2C_SMBUS_WRITE
| | command = = I2C_SMBUS_I2C_BLOCK_DATA ) {
2007-07-12 16:12:31 +04:00
if ( data - > block [ 0 ] < 1 )
data - > block [ 0 ] = 1 ;
if ( data - > block [ 0 ] > I2C_SMBUS_BLOCK_MAX )
data - > block [ 0 ] = I2C_SMBUS_BLOCK_MAX ;
} else {
2008-01-27 20:14:50 +03:00
data - > block [ 0 ] = 32 ; /* max for SMBus block reads */
2007-07-12 16:12:31 +04:00
}
2010-03-13 22:56:53 +03:00
/* Experience has shown that the block buffer can only be used for
SMBus ( not I2C ) block transactions , even though the datasheet
doesn ' t mention this limitation . */
2010-10-31 23:06:59 +03:00
if ( ( priv - > features & FEATURE_BLOCK_BUFFER )
2010-03-13 22:56:53 +03:00
& & command ! = I2C_SMBUS_I2C_BLOCK_DATA
2010-10-31 23:06:59 +03:00
& & i801_set_block_buffer_mode ( priv ) = = 0 )
result = i801_block_transaction_by_block ( priv , data ,
read_write , hwpec ) ;
2007-07-12 16:12:31 +04:00
else
2010-10-31 23:06:59 +03:00
result = i801_block_transaction_byte_by_byte ( priv , data ,
read_write ,
2008-01-27 20:14:50 +03:00
command , hwpec ) ;
2007-07-12 16:12:31 +04:00
if ( result = = 0 & & hwpec )
2010-10-31 23:06:59 +03:00
i801_wait_hwpec ( priv ) ;
2005-04-17 02:20:36 +04:00
2008-01-27 20:14:50 +03:00
if ( command = = I2C_SMBUS_I2C_BLOCK_DATA
& & read_write = = I2C_SMBUS_WRITE ) {
2005-04-17 02:20:36 +04:00
/* restore saved configuration register value */
2010-10-31 23:06:59 +03:00
pci_write_config_byte ( priv - > pci_dev , SMBHSTCFG , hostc ) ;
2005-04-17 02:20:36 +04:00
}
return result ;
}
2008-07-15 00:38:25 +04:00
/* Return negative errno on error. */
2010-05-21 20:40:55 +04:00
static s32 i801_access ( struct i2c_adapter * adap , u16 addr ,
2005-04-17 02:20:36 +04:00
unsigned short flags , char read_write , u8 command ,
2010-05-21 20:40:55 +04:00
int size , union i2c_smbus_data * data )
2005-04-17 02:20:36 +04:00
{
2005-10-26 23:34:42 +04:00
int hwpec ;
2005-04-17 02:20:36 +04:00
int block = 0 ;
int ret , xact = 0 ;
2010-10-31 23:06:59 +03:00
struct i801_priv * priv = i2c_get_adapdata ( adap ) ;
2005-04-17 02:20:36 +04:00
2010-10-31 23:06:59 +03:00
hwpec = ( priv - > features & FEATURE_SMBUS_PEC ) & & ( flags & I2C_CLIENT_PEC )
2005-10-26 23:34:42 +04:00
& & size ! = I2C_SMBUS_QUICK
& & size ! = I2C_SMBUS_I2C_BLOCK_DATA ;
2005-04-17 02:20:36 +04:00
switch ( size ) {
case I2C_SMBUS_QUICK :
outb_p ( ( ( addr & 0x7f ) < < 1 ) | ( read_write & 0x01 ) ,
2010-10-31 23:06:59 +03:00
SMBHSTADD ( priv ) ) ;
2005-04-17 02:20:36 +04:00
xact = I801_QUICK ;
break ;
case I2C_SMBUS_BYTE :
outb_p ( ( ( addr & 0x7f ) < < 1 ) | ( read_write & 0x01 ) ,
2010-10-31 23:06:59 +03:00
SMBHSTADD ( priv ) ) ;
2005-04-17 02:20:36 +04:00
if ( read_write = = I2C_SMBUS_WRITE )
2010-10-31 23:06:59 +03:00
outb_p ( command , SMBHSTCMD ( priv ) ) ;
2005-04-17 02:20:36 +04:00
xact = I801_BYTE ;
break ;
case I2C_SMBUS_BYTE_DATA :
outb_p ( ( ( addr & 0x7f ) < < 1 ) | ( read_write & 0x01 ) ,
2010-10-31 23:06:59 +03:00
SMBHSTADD ( priv ) ) ;
outb_p ( command , SMBHSTCMD ( priv ) ) ;
2005-04-17 02:20:36 +04:00
if ( read_write = = I2C_SMBUS_WRITE )
2010-10-31 23:06:59 +03:00
outb_p ( data - > byte , SMBHSTDAT0 ( priv ) ) ;
2005-04-17 02:20:36 +04:00
xact = I801_BYTE_DATA ;
break ;
case I2C_SMBUS_WORD_DATA :
outb_p ( ( ( addr & 0x7f ) < < 1 ) | ( read_write & 0x01 ) ,
2010-10-31 23:06:59 +03:00
SMBHSTADD ( priv ) ) ;
outb_p ( command , SMBHSTCMD ( priv ) ) ;
2005-04-17 02:20:36 +04:00
if ( read_write = = I2C_SMBUS_WRITE ) {
2010-10-31 23:06:59 +03:00
outb_p ( data - > word & 0xff , SMBHSTDAT0 ( priv ) ) ;
outb_p ( ( data - > word & 0xff00 ) > > 8 , SMBHSTDAT1 ( priv ) ) ;
2005-04-17 02:20:36 +04:00
}
xact = I801_WORD_DATA ;
break ;
case I2C_SMBUS_BLOCK_DATA :
outb_p ( ( ( addr & 0x7f ) < < 1 ) | ( read_write & 0x01 ) ,
2010-10-31 23:06:59 +03:00
SMBHSTADD ( priv ) ) ;
outb_p ( command , SMBHSTCMD ( priv ) ) ;
2005-04-17 02:20:36 +04:00
block = 1 ;
break ;
2008-01-27 20:14:50 +03:00
case I2C_SMBUS_I2C_BLOCK_DATA :
/* NB: page 240 of ICH5 datasheet shows that the R/#W
* bit should be cleared here , even when reading */
2010-10-31 23:06:59 +03:00
outb_p ( ( addr & 0x7f ) < < 1 , SMBHSTADD ( priv ) ) ;
2008-01-27 20:14:50 +03:00
if ( read_write = = I2C_SMBUS_READ ) {
/* NB: page 240 of ICH5 datasheet also shows
* that DATA1 is the cmd field when reading */
2010-10-31 23:06:59 +03:00
outb_p ( command , SMBHSTDAT1 ( priv ) ) ;
2008-01-27 20:14:50 +03:00
} else
2010-10-31 23:06:59 +03:00
outb_p ( command , SMBHSTCMD ( priv ) ) ;
2008-01-27 20:14:50 +03:00
block = 1 ;
break ;
2005-04-17 02:20:36 +04:00
default :
2010-10-31 23:06:59 +03:00
dev_err ( & priv - > pci_dev - > dev , " Unsupported transaction %d \n " ,
size ) ;
2008-07-15 00:38:25 +04:00
return - EOPNOTSUPP ;
2005-04-17 02:20:36 +04:00
}
2007-07-12 16:12:31 +04:00
if ( hwpec ) /* enable/disable hardware PEC */
2010-10-31 23:06:59 +03:00
outb_p ( inb_p ( SMBAUXCTL ( priv ) ) | SMBAUXCTL_CRC , SMBAUXCTL ( priv ) ) ;
2007-07-12 16:12:31 +04:00
else
2010-10-31 23:06:59 +03:00
outb_p ( inb_p ( SMBAUXCTL ( priv ) ) & ( ~ SMBAUXCTL_CRC ) ,
SMBAUXCTL ( priv ) ) ;
2005-10-26 23:34:42 +04:00
2010-05-21 20:40:55 +04:00
if ( block )
2010-10-31 23:06:59 +03:00
ret = i801_block_transaction ( priv , data , read_write , size ,
hwpec ) ;
2007-07-12 16:12:31 +04:00
else
2010-10-31 23:06:59 +03:00
ret = i801_transaction ( priv , xact | ENABLE_INT9 ) ;
2005-04-17 02:20:36 +04:00
2006-04-20 13:43:18 +04:00
/* Some BIOSes don't like it when PEC is enabled at reboot or resume
2007-07-12 16:12:31 +04:00
time , so we forcibly disable it after every transaction . Turn off
E32B for the same reason . */
2008-01-27 20:14:50 +03:00
if ( hwpec | | block )
2010-10-31 23:06:59 +03:00
outb_p ( inb_p ( SMBAUXCTL ( priv ) ) &
~ ( SMBAUXCTL_CRC | SMBAUXCTL_E32B ) , SMBAUXCTL ( priv ) ) ;
2006-04-20 13:43:18 +04:00
2010-05-21 20:40:55 +04:00
if ( block )
2005-04-17 02:20:36 +04:00
return ret ;
2010-05-21 20:40:55 +04:00
if ( ret )
2008-07-15 00:38:25 +04:00
return ret ;
2005-04-17 02:20:36 +04:00
if ( ( read_write = = I2C_SMBUS_WRITE ) | | ( xact = = I801_QUICK ) )
return 0 ;
switch ( xact & 0x7f ) {
case I801_BYTE : /* Result put in SMBHSTDAT0 */
case I801_BYTE_DATA :
2010-10-31 23:06:59 +03:00
data - > byte = inb_p ( SMBHSTDAT0 ( priv ) ) ;
2005-04-17 02:20:36 +04:00
break ;
case I801_WORD_DATA :
2010-10-31 23:06:59 +03:00
data - > word = inb_p ( SMBHSTDAT0 ( priv ) ) +
( inb_p ( SMBHSTDAT1 ( priv ) ) < < 8 ) ;
2005-04-17 02:20:36 +04:00
break ;
}
return 0 ;
}
static u32 i801_func ( struct i2c_adapter * adapter )
{
2010-10-31 23:06:59 +03:00
struct i801_priv * priv = i2c_get_adapdata ( adapter ) ;
2005-04-17 02:20:36 +04:00
return I2C_FUNC_SMBUS_QUICK | I2C_FUNC_SMBUS_BYTE |
2008-01-27 20:14:50 +03:00
I2C_FUNC_SMBUS_BYTE_DATA | I2C_FUNC_SMBUS_WORD_DATA |
I2C_FUNC_SMBUS_BLOCK_DATA | I2C_FUNC_SMBUS_WRITE_I2C_BLOCK |
2010-10-31 23:06:59 +03:00
( ( priv - > features & FEATURE_SMBUS_PEC ) ? I2C_FUNC_SMBUS_PEC : 0 ) |
( ( priv - > features & FEATURE_I2C_BLOCK_READ ) ?
2008-01-27 20:14:50 +03:00
I2C_FUNC_SMBUS_READ_I2C_BLOCK : 0 ) ;
2005-04-17 02:20:36 +04:00
}
2006-09-04 00:39:46 +04:00
static const struct i2c_algorithm smbus_algorithm = {
2005-04-17 02:20:36 +04:00
. smbus_xfer = i801_access ,
. functionality = i801_func ,
} ;
2010-03-02 14:23:37 +03:00
static const struct pci_device_id i801_ids [ ] = {
2005-04-17 02:20:36 +04:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801AA_3 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801AB_3 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801BA_2 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801CA_3 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801DB_3 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_82801EB_3 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ESB_4 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH6_16 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH7_17 ) } ,
2005-04-17 02:24:45 +04:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ESB2_17 ) } ,
2006-01-09 21:58:08 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH8_5 ) } ,
2006-11-23 02:19:12 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH9_6 ) } ,
2010-10-05 00:27:14 +04:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_EP80579_1 ) } ,
2008-02-24 22:03:42 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH10_4 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ICH10_5 ) } ,
2010-10-05 00:27:14 +04:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_5_3400_SERIES_SMBUS ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_COUGARPOINT_SMBUS ) } ,
2010-10-31 23:06:59 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS ) } ,
2010-10-31 23:07:00 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF0 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF1 ) } ,
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_PATSBURG_SMBUS_IDF2 ) } ,
2005-04-17 02:20:36 +04:00
{ 0 , }
} ;
2010-05-21 20:40:55 +04:00
MODULE_DEVICE_TABLE ( pci , i801_ids ) ;
2005-04-17 02:20:36 +04:00
2009-01-07 16:29:17 +03:00
# if defined CONFIG_INPUT_APANEL || defined CONFIG_INPUT_APANEL_MODULE
static unsigned char apanel_addr ;
/* Scan the system ROM for the signature "FJKEYINF" */
static __init const void __iomem * bios_signature ( const void __iomem * bios )
{
ssize_t offset ;
const unsigned char signature [ ] = " FJKEYINF " ;
for ( offset = 0 ; offset < 0x10000 ; offset + = 0x10 ) {
if ( check_signature ( bios + offset , signature ,
sizeof ( signature ) - 1 ) )
return bios + offset ;
}
return NULL ;
}
static void __init input_apanel_init ( void )
{
void __iomem * bios ;
const void __iomem * p ;
bios = ioremap ( 0xF0000 , 0x10000 ) ; /* Can't fail */
p = bios_signature ( bios ) ;
if ( p ) {
/* just use the first address */
apanel_addr = readb ( p + 8 + 3 ) > > 1 ;
}
iounmap ( bios ) ;
}
# else
static void __init input_apanel_init ( void ) { }
# endif
2009-03-30 23:46:44 +04:00
# if defined CONFIG_SENSORS_FSCHMD || defined CONFIG_SENSORS_FSCHMD_MODULE
struct dmi_onboard_device_info {
const char * name ;
u8 type ;
unsigned short i2c_addr ;
const char * i2c_type ;
} ;
static struct dmi_onboard_device_info __devinitdata dmi_devices [ ] = {
{ " Syleus " , DMI_DEV_TYPE_OTHER , 0x73 , " fscsyl " } ,
{ " Hermes " , DMI_DEV_TYPE_OTHER , 0x73 , " fscher " } ,
{ " Hades " , DMI_DEV_TYPE_OTHER , 0x73 , " fschds " } ,
} ;
static void __devinit dmi_check_onboard_device ( u8 type , const char * name ,
struct i2c_adapter * adap )
{
int i ;
struct i2c_board_info info ;
for ( i = 0 ; i < ARRAY_SIZE ( dmi_devices ) ; i + + ) {
/* & ~0x80, ignore enabled/disabled bit */
if ( ( type & ~ 0x80 ) ! = dmi_devices [ i ] . type )
continue ;
2010-07-09 18:22:51 +04:00
if ( strcasecmp ( name , dmi_devices [ i ] . name ) )
2009-03-30 23:46:44 +04:00
continue ;
memset ( & info , 0 , sizeof ( struct i2c_board_info ) ) ;
info . addr = dmi_devices [ i ] . i2c_addr ;
strlcpy ( info . type , dmi_devices [ i ] . i2c_type , I2C_NAME_SIZE ) ;
i2c_new_device ( adap , & info ) ;
break ;
}
}
/* We use our own function to check for onboard devices instead of
dmi_find_device ( ) as some buggy BIOS ' s have the devices we are interested
in marked as disabled */
static void __devinit dmi_check_onboard_devices ( const struct dmi_header * dm ,
void * adap )
{
int i , count ;
if ( dm - > type ! = 10 )
return ;
count = ( dm - > length - sizeof ( struct dmi_header ) ) / 2 ;
for ( i = 0 ; i < count ; i + + ) {
const u8 * d = ( char * ) ( dm + 1 ) + ( i * 2 ) ;
const char * name = ( ( char * ) dm ) + dm - > length ;
u8 type = d [ 0 ] ;
u8 s = d [ 1 ] ;
if ( ! s )
continue ;
s - - ;
while ( s > 0 & & name [ 0 ] ) {
name + = strlen ( name ) + 1 ;
s - - ;
}
if ( name [ 0 ] = = 0 ) /* Bogus string reference */
continue ;
dmi_check_onboard_device ( type , name , adap ) ;
}
}
# endif
2010-05-21 20:40:55 +04:00
static int __devinit i801_probe ( struct pci_dev * dev ,
const struct pci_device_id * id )
2005-04-17 02:20:36 +04:00
{
2006-06-12 23:53:41 +04:00
unsigned char temp ;
2010-05-21 20:40:54 +04:00
int err , i ;
2010-10-31 23:06:59 +03:00
struct i801_priv * priv ;
priv = kzalloc ( sizeof ( * priv ) , GFP_KERNEL ) ;
if ( ! priv )
return - ENOMEM ;
i2c_set_adapdata ( & priv - > adapter , priv ) ;
priv - > adapter . owner = THIS_MODULE ;
priv - > adapter . class = I2C_CLASS_HWMON | I2C_CLASS_SPD ;
priv - > adapter . algo = & smbus_algorithm ;
2005-04-17 02:20:36 +04:00
2010-10-31 23:06:59 +03:00
priv - > pci_dev = dev ;
2006-12-10 23:21:33 +03:00
switch ( dev - > device ) {
2010-05-21 20:40:55 +04:00
default :
2010-10-31 23:06:59 +03:00
priv - > features | = FEATURE_I2C_BLOCK_READ ;
2008-01-27 20:14:50 +03:00
/* fall through */
case PCI_DEVICE_ID_INTEL_82801DB_3 :
2010-10-31 23:06:59 +03:00
priv - > features | = FEATURE_SMBUS_PEC ;
priv - > features | = FEATURE_BLOCK_BUFFER ;
2010-05-21 20:40:55 +04:00
/* fall through */
case PCI_DEVICE_ID_INTEL_82801CA_3 :
case PCI_DEVICE_ID_INTEL_82801BA_2 :
case PCI_DEVICE_ID_INTEL_82801AB_3 :
case PCI_DEVICE_ID_INTEL_82801AA_3 :
2006-12-10 23:21:33 +03:00
break ;
}
2006-06-12 23:53:41 +04:00
2010-05-21 20:40:54 +04:00
/* Disable features on user request */
for ( i = 0 ; i < ARRAY_SIZE ( i801_feature_names ) ; i + + ) {
2010-10-31 23:06:59 +03:00
if ( priv - > features & disable_features & ( 1 < < i ) )
2010-05-21 20:40:54 +04:00
dev_notice ( & dev - > dev , " %s disabled by user \n " ,
i801_feature_names [ i ] ) ;
}
2010-10-31 23:06:59 +03:00
priv - > features & = ~ disable_features ;
2010-05-21 20:40:54 +04:00
2006-06-12 23:53:41 +04:00
err = pci_enable_device ( dev ) ;
if ( err ) {
dev_err ( & dev - > dev , " Failed to enable SMBus PCI device (%d) \n " ,
err ) ;
goto exit ;
}
/* Determine the address of the SMBus area */
2010-10-31 23:06:59 +03:00
priv - > smba = pci_resource_start ( dev , SMBBAR ) ;
if ( ! priv - > smba ) {
2006-06-12 23:53:41 +04:00
dev_err ( & dev - > dev , " SMBus base address uninitialized, "
" upgrade BIOS \n " ) ;
err = - ENODEV ;
2006-06-27 20:40:54 +04:00
goto exit ;
2006-06-12 23:53:41 +04:00
}
2008-07-15 00:38:33 +04:00
err = acpi_check_resource_conflict ( & dev - > resource [ SMBBAR ] ) ;
2009-10-05 00:53:45 +04:00
if ( err ) {
err = - ENODEV ;
2008-07-15 00:38:33 +04:00
goto exit ;
2009-10-05 00:53:45 +04:00
}
2008-07-15 00:38:33 +04:00
2006-06-12 23:53:41 +04:00
err = pci_request_region ( dev , SMBBAR , i801_driver . name ) ;
if ( err ) {
dev_err ( & dev - > dev , " Failed to request SMBus region "
2010-10-31 23:06:59 +03:00
" 0x%lx-0x%Lx \n " , priv - > smba ,
2006-06-30 12:56:20 +04:00
( unsigned long long ) pci_resource_end ( dev , SMBBAR ) ) ;
2006-06-27 20:40:54 +04:00
goto exit ;
2006-06-12 23:53:41 +04:00
}
2010-10-31 23:06:59 +03:00
pci_read_config_byte ( priv - > pci_dev , SMBHSTCFG , & temp ) ;
priv - > original_hstcfg = temp ;
2006-06-12 23:53:41 +04:00
temp & = ~ SMBHSTCFG_I2C_EN ; /* SMBus timing */
if ( ! ( temp & SMBHSTCFG_HST_EN ) ) {
dev_info ( & dev - > dev , " Enabling SMBus device \n " ) ;
temp | = SMBHSTCFG_HST_EN ;
}
2010-10-31 23:06:59 +03:00
pci_write_config_byte ( priv - > pci_dev , SMBHSTCFG , temp ) ;
2006-06-12 23:53:41 +04:00
if ( temp & SMBHSTCFG_SMB_SMI_EN )
dev_dbg ( & dev - > dev , " SMBus using interrupt SMI# \n " ) ;
else
dev_dbg ( & dev - > dev , " SMBus using PCI Interrupt \n " ) ;
2005-04-17 02:20:36 +04:00
2008-01-27 20:14:50 +03:00
/* Clear special mode bits */
2010-10-31 23:06:59 +03:00
if ( priv - > features & ( FEATURE_SMBUS_PEC | FEATURE_BLOCK_BUFFER ) )
outb_p ( inb_p ( SMBAUXCTL ( priv ) ) &
~ ( SMBAUXCTL_CRC | SMBAUXCTL_E32B ) , SMBAUXCTL ( priv ) ) ;
2008-01-27 20:14:50 +03:00
2007-02-17 21:13:42 +03:00
/* set up the sysfs linkage to our parent device */
2010-10-31 23:06:59 +03:00
priv - > adapter . dev . parent = & dev - > dev ;
2005-04-17 02:20:36 +04:00
2009-12-06 19:06:27 +03:00
/* Retry up to 3 times on lost arbitration */
2010-10-31 23:06:59 +03:00
priv - > adapter . retries = 3 ;
2009-12-06 19:06:27 +03:00
2010-10-31 23:06:59 +03:00
snprintf ( priv - > adapter . name , sizeof ( priv - > adapter . name ) ,
" SMBus I801 adapter at %04lx " , priv - > smba ) ;
err = i2c_add_adapter ( & priv - > adapter ) ;
2006-06-12 23:53:41 +04:00
if ( err ) {
dev_err ( & dev - > dev , " Failed to add SMBus adapter \n " ) ;
2006-06-27 20:40:54 +04:00
goto exit_release ;
2006-06-12 23:53:41 +04:00
}
2009-01-07 16:29:17 +03:00
/* Register optional slaves */
# if defined CONFIG_INPUT_APANEL || defined CONFIG_INPUT_APANEL_MODULE
if ( apanel_addr ) {
struct i2c_board_info info ;
memset ( & info , 0 , sizeof ( struct i2c_board_info ) ) ;
info . addr = apanel_addr ;
strlcpy ( info . type , " fujitsu_apanel " , I2C_NAME_SIZE ) ;
2010-10-31 23:06:59 +03:00
i2c_new_device ( & priv - > adapter , & info ) ;
2009-01-07 16:29:17 +03:00
}
# endif
2009-03-30 23:46:44 +04:00
# if defined CONFIG_SENSORS_FSCHMD || defined CONFIG_SENSORS_FSCHMD_MODULE
2010-07-09 18:22:51 +04:00
if ( dmi_name_in_vendors ( " FUJITSU " ) )
2010-10-31 23:06:59 +03:00
dmi_walk ( dmi_check_onboard_devices , & priv - > adapter ) ;
2009-03-30 23:46:44 +04:00
# endif
2009-01-07 16:29:17 +03:00
2010-10-31 23:06:59 +03:00
pci_set_drvdata ( dev , priv ) ;
2006-06-27 20:40:54 +04:00
return 0 ;
2006-06-12 23:53:41 +04:00
2006-06-27 20:40:54 +04:00
exit_release :
pci_release_region ( dev , SMBBAR ) ;
2006-06-12 23:53:41 +04:00
exit :
2010-10-31 23:06:59 +03:00
kfree ( priv ) ;
2006-06-12 23:53:41 +04:00
return err ;
2005-04-17 02:20:36 +04:00
}
static void __devexit i801_remove ( struct pci_dev * dev )
{
2010-10-31 23:06:59 +03:00
struct i801_priv * priv = pci_get_drvdata ( dev ) ;
i2c_del_adapter ( & priv - > adapter ) ;
pci_write_config_byte ( dev , SMBHSTCFG , priv - > original_hstcfg ) ;
2006-06-12 23:53:02 +04:00
pci_release_region ( dev , SMBBAR ) ;
2010-10-31 23:06:59 +03:00
pci_set_drvdata ( dev , NULL ) ;
kfree ( priv ) ;
2006-06-27 20:40:54 +04:00
/*
* do not call pci_disable_device ( dev ) since it can cause hard hangs on
* some systems during power - off ( eg . Fujitsu - Siemens Lifebook E8010 )
*/
2005-04-17 02:20:36 +04:00
}
2007-03-22 21:49:01 +03:00
# ifdef CONFIG_PM
static int i801_suspend ( struct pci_dev * dev , pm_message_t mesg )
{
2010-10-31 23:06:59 +03:00
struct i801_priv * priv = pci_get_drvdata ( dev ) ;
2007-03-22 21:49:01 +03:00
pci_save_state ( dev ) ;
2010-10-31 23:06:59 +03:00
pci_write_config_byte ( dev , SMBHSTCFG , priv - > original_hstcfg ) ;
2007-03-22 21:49:01 +03:00
pci_set_power_state ( dev , pci_choose_state ( dev , mesg ) ) ;
return 0 ;
}
static int i801_resume ( struct pci_dev * dev )
{
pci_set_power_state ( dev , PCI_D0 ) ;
pci_restore_state ( dev ) ;
return pci_enable_device ( dev ) ;
}
# else
# define i801_suspend NULL
# define i801_resume NULL
# endif
2005-04-17 02:20:36 +04:00
static struct pci_driver i801_driver = {
. name = " i801_smbus " ,
. id_table = i801_ids ,
. probe = i801_probe ,
. remove = __devexit_p ( i801_remove ) ,
2007-03-22 21:49:01 +03:00
. suspend = i801_suspend ,
. resume = i801_resume ,
2005-04-17 02:20:36 +04:00
} ;
static int __init i2c_i801_init ( void )
{
2009-01-07 16:29:17 +03:00
input_apanel_init ( ) ;
2005-04-17 02:20:36 +04:00
return pci_register_driver ( & i801_driver ) ;
}
static void __exit i2c_i801_exit ( void )
{
pci_unregister_driver ( & i801_driver ) ;
}
2008-01-27 20:14:50 +03:00
MODULE_AUTHOR ( " Mark D. Studebaker <mdsxyz123@yahoo.com>, "
" Jean Delvare <khali@linux-fr.org> " ) ;
2005-04-17 02:20:36 +04:00
MODULE_DESCRIPTION ( " I801 SMBus driver " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_init ( i2c_i801_init ) ;
module_exit ( i2c_i801_exit ) ;