2010-02-19 13:26:25 +03:00
/*
* lpc_sch . c - LPC interface for Intel Poulsbo SCH
*
* LPC bridge function of the Intel SCH contains many other
* functional units , such as Interrupt controllers , Timers ,
* Power Management , System Management , GPIO , RTC , and LPC
* Configuration Registers .
*
* Copyright ( c ) 2010 CompuLab Ltd
* Author : Denis Turischev < denis @ compulab . co . il >
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License 2 as published
* by the Free Software Foundation .
*
* 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 ; see the file COPYING . If not , write to
* the Free Software Foundation , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/errno.h>
# include <linux/acpi.h>
# include <linux/pci.h>
# include <linux/mfd/core.h>
# define SMBASE 0x40
# define SMBUS_IO_SIZE 64
# define GPIOBASE 0x44
# define GPIO_IO_SIZE 64
2012-04-18 01:09:22 +04:00
# define GPIO_IO_SIZE_CENTERTON 128
2010-02-19 13:26:25 +03:00
2011-06-16 15:05:49 +04:00
# define WDTBASE 0x84
# define WDT_IO_SIZE 64
2010-02-19 13:26:25 +03:00
static struct resource smbus_sch_resource = {
. flags = IORESOURCE_IO ,
} ;
static struct resource gpio_sch_resource = {
. flags = IORESOURCE_IO ,
} ;
2011-06-16 15:05:49 +04:00
static struct resource wdt_sch_resource = {
. flags = IORESOURCE_IO ,
} ;
2013-02-09 03:20:36 +04:00
static struct mfd_cell lpc_sch_cells [ 3 ] ;
static struct mfd_cell isch_smbus_cell = {
. name = " isch_smbus " ,
. num_resources = 1 ,
. resources = & smbus_sch_resource ,
2013-10-23 15:31:00 +04:00
. ignore_resource_conflicts = true ,
2013-02-09 03:20:36 +04:00
} ;
static struct mfd_cell sch_gpio_cell = {
. name = " sch_gpio " ,
. num_resources = 1 ,
. resources = & gpio_sch_resource ,
2013-10-23 15:31:00 +04:00
. ignore_resource_conflicts = true ,
2013-02-09 03:20:36 +04:00
} ;
static struct mfd_cell wdt_sch_cell = {
. name = " ie6xx_wdt " ,
. num_resources = 1 ,
. resources = & wdt_sch_resource ,
2013-10-23 15:31:00 +04:00
. ignore_resource_conflicts = true ,
2011-06-16 15:05:49 +04:00
} ;
2013-12-03 03:15:39 +04:00
static const struct pci_device_id lpc_sch_ids [ ] = {
2010-02-19 13:26:25 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_SCH_LPC ) } ,
2011-03-13 18:28:59 +03:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_ITC_LPC ) } ,
2012-04-18 01:09:22 +04:00
{ PCI_DEVICE ( PCI_VENDOR_ID_INTEL , PCI_DEVICE_ID_INTEL_CENTERTON_ILB ) } ,
2010-02-19 13:26:25 +03:00
{ 0 , }
} ;
MODULE_DEVICE_TABLE ( pci , lpc_sch_ids ) ;
2012-11-19 22:23:04 +04:00
static int lpc_sch_probe ( struct pci_dev * dev ,
2010-02-19 13:26:25 +03:00
const struct pci_device_id * id )
{
unsigned int base_addr_cfg ;
unsigned short base_addr ;
2013-02-09 03:20:36 +04:00
int i , cells = 0 ;
2011-06-16 15:05:49 +04:00
int ret ;
2010-02-19 13:26:25 +03:00
pci_read_config_dword ( dev , SMBASE , & base_addr_cfg ) ;
2013-02-09 03:20:36 +04:00
base_addr = 0 ;
if ( ! ( base_addr_cfg & ( 1 < < 31 ) ) )
dev_warn ( & dev - > dev , " Decode of the SMBus I/O range disabled \n " ) ;
else
base_addr = ( unsigned short ) base_addr_cfg ;
2010-02-19 13:26:25 +03:00
if ( base_addr = = 0 ) {
2013-02-09 03:20:36 +04:00
dev_warn ( & dev - > dev , " I/O space for SMBus uninitialized \n " ) ;
} else {
lpc_sch_cells [ cells + + ] = isch_smbus_cell ;
smbus_sch_resource . start = base_addr ;
smbus_sch_resource . end = base_addr + SMBUS_IO_SIZE - 1 ;
2010-02-19 13:26:25 +03:00
}
2013-02-09 03:20:36 +04:00
pci_read_config_dword ( dev , GPIOBASE , & base_addr_cfg ) ;
base_addr = 0 ;
if ( ! ( base_addr_cfg & ( 1 < < 31 ) ) )
dev_warn ( & dev - > dev , " Decode of the GPIO I/O range disabled \n " ) ;
2012-04-18 01:09:22 +04:00
else
2013-02-09 03:20:36 +04:00
base_addr = ( unsigned short ) base_addr_cfg ;
2011-03-13 18:28:59 +03:00
2013-02-09 03:20:36 +04:00
if ( base_addr = = 0 ) {
dev_warn ( & dev - > dev , " I/O space for GPIO uninitialized \n " ) ;
} else {
lpc_sch_cells [ cells + + ] = sch_gpio_cell ;
gpio_sch_resource . start = base_addr ;
if ( id - > device = = PCI_DEVICE_ID_INTEL_CENTERTON_ILB )
gpio_sch_resource . end = base_addr + GPIO_IO_SIZE_CENTERTON - 1 ;
else
gpio_sch_resource . end = base_addr + GPIO_IO_SIZE - 1 ;
}
2011-06-16 15:05:49 +04:00
2012-04-18 01:09:22 +04:00
if ( id - > device = = PCI_DEVICE_ID_INTEL_ITC_LPC
2013-02-09 03:20:36 +04:00
| | id - > device = = PCI_DEVICE_ID_INTEL_CENTERTON_ILB ) {
2011-06-16 15:05:49 +04:00
pci_read_config_dword ( dev , WDTBASE , & base_addr_cfg ) ;
2013-02-09 03:20:36 +04:00
base_addr = 0 ;
if ( ! ( base_addr_cfg & ( 1 < < 31 ) ) )
dev_warn ( & dev - > dev , " Decode of the WDT I/O range disabled \n " ) ;
else
base_addr = ( unsigned short ) base_addr_cfg ;
if ( base_addr = = 0 )
dev_warn ( & dev - > dev , " I/O space for WDT uninitialized \n " ) ;
else {
lpc_sch_cells [ cells + + ] = wdt_sch_cell ;
wdt_sch_resource . start = base_addr ;
wdt_sch_resource . end = base_addr + WDT_IO_SIZE - 1 ;
2011-06-16 15:05:49 +04:00
}
2013-02-09 03:20:36 +04:00
}
2011-06-16 15:05:49 +04:00
2013-02-09 03:20:36 +04:00
if ( WARN_ON ( cells > ARRAY_SIZE ( lpc_sch_cells ) ) ) {
dev_err ( & dev - > dev , " Cell count exceeds array size " ) ;
return - ENODEV ;
}
2011-06-16 15:05:49 +04:00
2013-02-09 03:20:36 +04:00
if ( cells = = 0 ) {
dev_err ( & dev - > dev , " All decode registers disabled. \n " ) ;
return - ENODEV ;
2011-06-16 15:05:49 +04:00
}
2013-02-09 03:20:36 +04:00
for ( i = 0 ; i < cells ; i + + )
lpc_sch_cells [ i ] . id = id - > device ;
ret = mfd_add_devices ( & dev - > dev , 0 , lpc_sch_cells , cells , NULL , 0 , NULL ) ;
if ( ret )
mfd_remove_devices ( & dev - > dev ) ;
2011-06-16 15:05:49 +04:00
return ret ;
2010-02-19 13:26:25 +03:00
}
2012-11-19 22:26:01 +04:00
static void lpc_sch_remove ( struct pci_dev * dev )
2010-02-19 13:26:25 +03:00
{
mfd_remove_devices ( & dev - > dev ) ;
}
static struct pci_driver lpc_sch_driver = {
. name = " lpc_sch " ,
. id_table = lpc_sch_ids ,
. probe = lpc_sch_probe ,
2012-11-19 22:20:24 +04:00
. remove = lpc_sch_remove ,
2010-02-19 13:26:25 +03:00
} ;
2012-04-03 05:09:19 +04:00
module_pci_driver ( lpc_sch_driver ) ;
2010-02-19 13:26:25 +03:00
MODULE_AUTHOR ( " Denis Turischev <denis@compulab.co.il> " ) ;
MODULE_DESCRIPTION ( " LPC interface for Intel Poulsbo SCH " ) ;
MODULE_LICENSE ( " GPL " ) ;