2010-03-02 10:48:55 +01:00
/*
2011-06-04 18:38:28 -06:00
* GPIO interface for Intel Poulsbo SCH
2010-03-02 10:48:55 +01:00
*
* 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/init.h>
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/io.h>
# include <linux/errno.h>
# include <linux/acpi.h>
# include <linux/platform_device.h>
2011-03-14 12:53:05 +02:00
# include <linux/pci_ids.h>
2010-03-02 10:48:55 +01:00
# include <linux/gpio.h>
2014-10-21 13:33:56 +02:00
# define GEN 0x00
# define GIO 0x04
# define GLV 0x08
struct sch_gpio {
struct gpio_chip chip ;
spinlock_t lock ;
unsigned short iobase ;
unsigned short core_base ;
unsigned short resume_base ;
} ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
# define to_sch_gpio(c) container_of(c, struct sch_gpio, chip)
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
static unsigned sch_gpio_offset ( struct sch_gpio * sch , unsigned gpio ,
unsigned reg )
2010-03-02 10:48:55 +01:00
{
2014-10-21 13:33:56 +02:00
unsigned base = 0 ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
if ( gpio > = sch - > resume_base ) {
gpio - = sch - > resume_base ;
base + = 0x20 ;
}
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
return base + reg + gpio / 8 ;
2010-03-02 10:48:55 +01:00
}
2014-10-21 13:33:56 +02:00
static unsigned sch_gpio_bit ( struct sch_gpio * sch , unsigned gpio )
2010-03-02 10:48:55 +01:00
{
2014-10-21 13:33:56 +02:00
if ( gpio > = sch - > resume_base )
gpio - = sch - > resume_base ;
return gpio % 8 ;
2010-03-02 10:48:55 +01:00
}
2014-10-21 13:33:56 +02:00
static void sch_gpio_enable ( struct sch_gpio * sch , unsigned gpio )
2010-03-02 10:48:55 +01:00
{
unsigned short offset , bit ;
2014-10-21 13:33:56 +02:00
u8 enable ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
spin_lock ( & sch - > lock ) ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
offset = sch_gpio_offset ( sch , gpio , GEN ) ;
bit = sch_gpio_bit ( sch , gpio ) ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
enable = inb ( sch - > iobase + offset ) ;
if ( ! ( enable & ( 1 < < bit ) ) )
outb ( enable | ( 1 < < bit ) , sch - > iobase + offset ) ;
2014-04-07 14:20:32 +02:00
2014-10-21 13:33:56 +02:00
spin_unlock ( & sch - > lock ) ;
2010-03-02 10:48:55 +01:00
}
2014-10-21 13:33:56 +02:00
static int sch_gpio_direction_in ( struct gpio_chip * gc , unsigned gpio_num )
2010-03-02 10:48:55 +01:00
{
2014-10-21 13:33:56 +02:00
struct sch_gpio * sch = to_sch_gpio ( gc ) ;
2010-03-02 10:48:55 +01:00
u8 curr_dirs ;
2013-03-06 13:49:36 -08:00
unsigned short offset , bit ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
spin_lock ( & sch - > lock ) ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
offset = sch_gpio_offset ( sch , gpio_num , GIO ) ;
bit = sch_gpio_bit ( sch , gpio_num ) ;
2013-03-06 13:49:36 -08:00
2014-10-21 13:33:56 +02:00
curr_dirs = inb ( sch - > iobase + offset ) ;
2010-03-02 10:48:55 +01:00
2013-03-06 13:49:36 -08:00
if ( ! ( curr_dirs & ( 1 < < bit ) ) )
2014-10-21 13:33:56 +02:00
outb ( curr_dirs | ( 1 < < bit ) , sch - > iobase + offset ) ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
spin_unlock ( & sch - > lock ) ;
2010-03-02 10:48:55 +01:00
return 0 ;
}
2014-10-21 13:33:56 +02:00
static int sch_gpio_get ( struct gpio_chip * gc , unsigned gpio_num )
2010-03-02 10:48:55 +01:00
{
2014-10-21 13:33:56 +02:00
struct sch_gpio * sch = to_sch_gpio ( gc ) ;
int res ;
2013-03-06 13:49:36 -08:00
unsigned short offset , bit ;
2014-10-21 13:33:56 +02:00
offset = sch_gpio_offset ( sch , gpio_num , GLV ) ;
bit = sch_gpio_bit ( sch , gpio_num ) ;
res = ! ! ( inb ( sch - > iobase + offset ) & ( 1 < < bit ) ) ;
2013-03-06 13:49:36 -08:00
2014-10-21 13:33:56 +02:00
return res ;
2010-03-02 10:48:55 +01:00
}
2014-10-21 13:33:56 +02:00
static void sch_gpio_set ( struct gpio_chip * gc , unsigned gpio_num , int val )
2010-03-02 10:48:55 +01:00
{
2014-10-21 13:33:56 +02:00
struct sch_gpio * sch = to_sch_gpio ( gc ) ;
2010-03-02 10:48:55 +01:00
u8 curr_vals ;
2013-03-06 13:49:36 -08:00
unsigned short offset , bit ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
spin_lock ( & sch - > lock ) ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
offset = sch_gpio_offset ( sch , gpio_num , GLV ) ;
bit = sch_gpio_bit ( sch , gpio_num ) ;
2013-03-06 13:49:36 -08:00
2014-10-21 13:33:56 +02:00
curr_vals = inb ( sch - > iobase + offset ) ;
2010-03-02 10:48:55 +01:00
if ( val )
2014-10-21 13:33:56 +02:00
outb ( curr_vals | ( 1 < < bit ) , sch - > iobase + offset ) ;
2010-03-02 10:48:55 +01:00
else
2014-10-21 13:33:56 +02:00
outb ( ( curr_vals & ~ ( 1 < < bit ) ) , sch - > iobase + offset ) ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
spin_unlock ( & sch - > lock ) ;
2010-03-02 10:48:55 +01:00
}
2014-10-21 13:33:56 +02:00
static int sch_gpio_direction_out ( struct gpio_chip * gc , unsigned gpio_num ,
int val )
2010-03-02 10:48:55 +01:00
{
2014-10-21 13:33:56 +02:00
struct sch_gpio * sch = to_sch_gpio ( gc ) ;
2010-03-02 10:48:55 +01:00
u8 curr_dirs ;
2013-03-06 13:49:36 -08:00
unsigned short offset , bit ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
spin_lock ( & sch - > lock ) ;
2013-03-06 13:49:36 -08:00
2014-10-21 13:33:56 +02:00
offset = sch_gpio_offset ( sch , gpio_num , GIO ) ;
bit = sch_gpio_bit ( sch , gpio_num ) ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
curr_dirs = inb ( sch - > iobase + offset ) ;
2013-03-06 13:49:36 -08:00
if ( curr_dirs & ( 1 < < bit ) )
2014-10-21 13:33:56 +02:00
outb ( curr_dirs & ~ ( 1 < < bit ) , sch - > iobase + offset ) ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
spin_unlock ( & sch - > lock ) ;
2014-04-07 14:20:32 +02:00
/*
2014-10-21 13:33:56 +02:00
* according to the datasheet , writing to the level register has no
* effect when GPIO is programmed as input .
* Actually the the level register is read - only when configured as input .
* Thus presetting the output level before switching to output is _NOT_ possible .
* Hence we set the level after configuring the GPIO as output .
* But we cannot prevent a short low pulse if direction is set to high
* and an external pull - up is connected .
*/
sch_gpio_set ( gc , gpio_num , val ) ;
2010-03-02 10:48:55 +01:00
return 0 ;
}
2014-10-21 13:33:56 +02:00
static struct gpio_chip sch_gpio_chip = {
. label = " sch_gpio " ,
2010-03-02 10:48:55 +01:00
. owner = THIS_MODULE ,
2014-10-21 13:33:56 +02:00
. direction_input = sch_gpio_direction_in ,
. get = sch_gpio_get ,
. direction_output = sch_gpio_direction_out ,
. set = sch_gpio_set ,
2010-03-02 10:48:55 +01:00
} ;
2012-11-19 13:22:34 -05:00
static int sch_gpio_probe ( struct platform_device * pdev )
2010-03-02 10:48:55 +01:00
{
2014-10-21 13:33:56 +02:00
struct sch_gpio * sch ;
2010-03-02 10:48:55 +01:00
struct resource * res ;
2011-03-14 12:53:05 +02:00
2014-10-21 13:33:56 +02:00
sch = devm_kzalloc ( & pdev - > dev , sizeof ( * sch ) , GFP_KERNEL ) ;
if ( ! sch )
return - ENOMEM ;
2010-03-02 10:48:55 +01:00
res = platform_get_resource ( pdev , IORESOURCE_IO , 0 ) ;
if ( ! res )
return - EBUSY ;
2014-10-21 13:33:56 +02:00
if ( ! devm_request_region ( & pdev - > dev , res - > start , resource_size ( res ) ,
pdev - > name ) )
2010-03-02 10:48:55 +01:00
return - EBUSY ;
2014-10-21 13:33:56 +02:00
spin_lock_init ( & sch - > lock ) ;
sch - > iobase = res - > start ;
sch - > chip = sch_gpio_chip ;
sch - > chip . label = dev_name ( & pdev - > dev ) ;
sch - > chip . dev = & pdev - > dev ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
switch ( pdev - > id ) {
2013-03-20 13:16:00 +01:00
case PCI_DEVICE_ID_INTEL_SCH_LPC :
2014-10-21 13:33:56 +02:00
sch - > core_base = 0 ;
sch - > resume_base = 10 ;
sch - > chip . ngpio = 14 ;
2013-03-20 13:16:00 +01:00
/*
* GPIO [ 6 : 0 ] enabled by default
* GPIO7 is configured by the CMC as SLPIOVR
* Enable GPIO [ 9 : 8 ] core powered gpios explicitly
*/
2014-10-21 13:33:56 +02:00
sch_gpio_enable ( sch , 8 ) ;
sch_gpio_enable ( sch , 9 ) ;
2013-03-20 13:16:00 +01:00
/*
* SUS_GPIO [ 2 : 0 ] enabled by default
* Enable SUS_GPIO3 resume powered gpio explicitly
*/
2014-10-21 13:33:56 +02:00
sch_gpio_enable ( sch , 13 ) ;
2013-03-20 13:16:00 +01:00
break ;
case PCI_DEVICE_ID_INTEL_ITC_LPC :
2014-10-21 13:33:56 +02:00
sch - > core_base = 0 ;
sch - > resume_base = 5 ;
sch - > chip . ngpio = 14 ;
2013-03-20 13:16:00 +01:00
break ;
case PCI_DEVICE_ID_INTEL_CENTERTON_ILB :
2014-10-21 13:33:56 +02:00
sch - > core_base = 0 ;
sch - > resume_base = 21 ;
sch - > chip . ngpio = 30 ;
2013-03-20 13:16:00 +01:00
break ;
default :
2014-10-21 13:33:56 +02:00
return - ENODEV ;
2011-03-14 12:53:05 +02:00
}
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
platform_set_drvdata ( pdev , sch ) ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
return gpiochip_add ( & sch - > chip ) ;
2010-03-02 10:48:55 +01:00
}
2012-11-19 13:25:50 -05:00
static int sch_gpio_remove ( struct platform_device * pdev )
2010-03-02 10:48:55 +01:00
{
2014-10-21 13:33:56 +02:00
struct sch_gpio * sch = platform_get_drvdata ( pdev ) ;
2010-03-02 10:48:55 +01:00
2014-10-21 13:33:56 +02:00
gpiochip_remove ( & sch - > chip ) ;
2010-03-02 10:48:55 +01:00
return 0 ;
}
static struct platform_driver sch_gpio_driver = {
. driver = {
. name = " sch_gpio " ,
} ,
. probe = sch_gpio_probe ,
2012-11-19 13:20:08 -05:00
. remove = sch_gpio_remove ,
2010-03-02 10:48:55 +01:00
} ;
2011-12-08 00:24:00 +08:00
module_platform_driver ( sch_gpio_driver ) ;
2010-03-02 10:48:55 +01:00
MODULE_AUTHOR ( " Denis Turischev <denis@compulab.co.il> " ) ;
MODULE_DESCRIPTION ( " GPIO interface for Intel Poulsbo SCH " ) ;
MODULE_LICENSE ( " GPL " ) ;
MODULE_ALIAS ( " platform:sch_gpio " ) ;