2008-07-05 10:02:49 +02:00
/*
* MXC GPIO support . ( c ) 2008 Daniel Mack < daniel @ caiaq . de >
* Copyright 2008 Juergen Beisert , kernel @ pengutronix . de
*
* Based on code from Freescale ,
2010-04-22 16:28:42 +03:00
* Copyright ( C ) 2004 - 2010 Freescale Semiconductor , Inc . All Rights Reserved .
2008-07-05 10:02:49 +02: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 . , 51 Franklin Street , Fifth Floor , Boston , MA 02110 - 1301 , USA .
*/
# include <linux/init.h>
2010-10-23 09:12:48 -05:00
# include <linux/interrupt.h>
2008-07-05 10:02:49 +02:00
# include <linux/io.h>
# include <linux/irq.h>
# include <linux/gpio.h>
2011-06-06 00:07:55 +08:00
# include <linux/platform_device.h>
# include <linux/slab.h>
2011-06-06 13:22:41 +08:00
# include <linux/basic_mmio_gpio.h>
2011-07-07 00:37:43 +08:00
# include <linux/of.h>
# include <linux/of_device.h>
2011-07-03 13:38:09 -04:00
# include <linux/module.h>
2008-07-05 10:02:49 +02:00
# include <asm-generic/bug.h>
2011-09-21 21:24:04 +08:00
# include <asm/mach/irq.h>
2008-07-05 10:02:49 +02:00
2011-08-14 00:14:04 +08:00
# define irq_to_gpio(irq) ((irq) - MXC_GPIO_IRQ_START)
2011-07-07 00:37:41 +08:00
enum mxc_gpio_hwtype {
IMX1_GPIO , /* runs on i.mx1 */
IMX21_GPIO , /* runs on i.mx21 and i.mx27 */
IMX31_GPIO , /* runs on all other i.mx */
} ;
/* device type dependent stuff */
struct mxc_gpio_hwdata {
unsigned dr_reg ;
unsigned gdir_reg ;
unsigned psr_reg ;
unsigned icr1_reg ;
unsigned icr2_reg ;
unsigned imr_reg ;
unsigned isr_reg ;
unsigned low_level ;
unsigned high_level ;
unsigned rise_edge ;
unsigned fall_edge ;
} ;
2011-06-06 00:07:55 +08:00
struct mxc_gpio_port {
struct list_head node ;
void __iomem * base ;
int irq ;
int irq_high ;
int virtual_irq_start ;
2011-06-06 13:22:41 +08:00
struct bgpio_chip bgc ;
2011-06-06 00:07:55 +08:00
u32 both_edges ;
} ;
2011-07-07 00:37:41 +08:00
static struct mxc_gpio_hwdata imx1_imx21_gpio_hwdata = {
. dr_reg = 0x1c ,
. gdir_reg = 0x00 ,
. psr_reg = 0x24 ,
. icr1_reg = 0x28 ,
. icr2_reg = 0x2c ,
. imr_reg = 0x30 ,
. isr_reg = 0x34 ,
. low_level = 0x03 ,
. high_level = 0x02 ,
. rise_edge = 0x00 ,
. fall_edge = 0x01 ,
} ;
static struct mxc_gpio_hwdata imx31_gpio_hwdata = {
. dr_reg = 0x00 ,
. gdir_reg = 0x04 ,
. psr_reg = 0x08 ,
. icr1_reg = 0x0c ,
. icr2_reg = 0x10 ,
. imr_reg = 0x14 ,
. isr_reg = 0x18 ,
. low_level = 0x00 ,
. high_level = 0x01 ,
. rise_edge = 0x02 ,
. fall_edge = 0x03 ,
} ;
static enum mxc_gpio_hwtype mxc_gpio_hwtype ;
static struct mxc_gpio_hwdata * mxc_gpio_hwdata ;
# define GPIO_DR (mxc_gpio_hwdata->dr_reg)
# define GPIO_GDIR (mxc_gpio_hwdata->gdir_reg)
# define GPIO_PSR (mxc_gpio_hwdata->psr_reg)
# define GPIO_ICR1 (mxc_gpio_hwdata->icr1_reg)
# define GPIO_ICR2 (mxc_gpio_hwdata->icr2_reg)
# define GPIO_IMR (mxc_gpio_hwdata->imr_reg)
# define GPIO_ISR (mxc_gpio_hwdata->isr_reg)
# define GPIO_INT_LOW_LEV (mxc_gpio_hwdata->low_level)
# define GPIO_INT_HIGH_LEV (mxc_gpio_hwdata->high_level)
# define GPIO_INT_RISE_EDGE (mxc_gpio_hwdata->rise_edge)
# define GPIO_INT_FALL_EDGE (mxc_gpio_hwdata->fall_edge)
# define GPIO_INT_NONE 0x4
static struct platform_device_id mxc_gpio_devtype [ ] = {
{
. name = " imx1-gpio " ,
. driver_data = IMX1_GPIO ,
} , {
. name = " imx21-gpio " ,
. driver_data = IMX21_GPIO ,
} , {
. name = " imx31-gpio " ,
. driver_data = IMX31_GPIO ,
} , {
/* sentinel */
}
} ;
2011-07-07 00:37:43 +08:00
static const struct of_device_id mxc_gpio_dt_ids [ ] = {
{ . compatible = " fsl,imx1-gpio " , . data = & mxc_gpio_devtype [ IMX1_GPIO ] , } ,
{ . compatible = " fsl,imx21-gpio " , . data = & mxc_gpio_devtype [ IMX21_GPIO ] , } ,
{ . compatible = " fsl,imx31-gpio " , . data = & mxc_gpio_devtype [ IMX31_GPIO ] , } ,
{ /* sentinel */ }
} ;
2011-06-06 00:07:55 +08:00
/*
* MX2 has one interrupt * for all * gpio ports . The list is used
* to save the references to all ports , so that mx2_gpio_irq_handler
* can walk through all interrupt status registers .
*/
static LIST_HEAD ( mxc_gpio_ports ) ;
2008-07-05 10:02:49 +02:00
/* Note: This driver assumes 32 GPIOs are handled in one register */
2010-11-29 11:16:23 +01:00
static int gpio_set_irq_type ( struct irq_data * d , u32 type )
2008-07-05 10:02:49 +02:00
{
2010-11-29 11:16:23 +01:00
u32 gpio = irq_to_gpio ( d - > irq ) ;
2011-06-07 16:25:37 +08:00
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct mxc_gpio_port * port = gc - > private ;
2008-07-05 10:02:49 +02:00
u32 bit , val ;
int edge ;
void __iomem * reg = port - > base ;
2009-03-12 12:46:41 +01:00
port - > both_edges & = ~ ( 1 < < ( gpio & 31 ) ) ;
2008-07-05 10:02:49 +02:00
switch ( type ) {
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_EDGE_RISING :
2008-07-05 10:02:49 +02:00
edge = GPIO_INT_RISE_EDGE ;
break ;
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_EDGE_FALLING :
2008-07-05 10:02:49 +02:00
edge = GPIO_INT_FALL_EDGE ;
break ;
2009-03-12 12:46:41 +01:00
case IRQ_TYPE_EDGE_BOTH :
2011-06-12 01:33:29 +08:00
val = gpio_get_value ( gpio ) ;
2009-03-12 12:46:41 +01:00
if ( val ) {
edge = GPIO_INT_LOW_LEV ;
pr_debug ( " mxc: set GPIO %d to low trigger \n " , gpio ) ;
} else {
edge = GPIO_INT_HIGH_LEV ;
pr_debug ( " mxc: set GPIO %d to high trigger \n " , gpio ) ;
}
port - > both_edges | = 1 < < ( gpio & 31 ) ;
break ;
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_LEVEL_LOW :
2008-07-05 10:02:49 +02:00
edge = GPIO_INT_LOW_LEV ;
break ;
2008-07-27 04:23:31 +01:00
case IRQ_TYPE_LEVEL_HIGH :
2008-07-05 10:02:49 +02:00
edge = GPIO_INT_HIGH_LEV ;
break ;
2009-03-12 12:46:41 +01:00
default :
2008-07-05 10:02:49 +02:00
return - EINVAL ;
}
reg + = GPIO_ICR1 + ( ( gpio & 0x10 ) > > 2 ) ; /* lower or upper register */
bit = gpio & 0xf ;
2011-06-06 00:07:55 +08:00
val = readl ( reg ) & ~ ( 0x3 < < ( bit < < 1 ) ) ;
writel ( val | ( edge < < ( bit < < 1 ) ) , reg ) ;
2011-06-07 16:25:37 +08:00
writel ( 1 < < ( gpio & 0x1f ) , port - > base + GPIO_ISR ) ;
2008-07-05 10:02:49 +02:00
return 0 ;
}
2009-03-12 12:46:41 +01:00
static void mxc_flip_edge ( struct mxc_gpio_port * port , u32 gpio )
{
void __iomem * reg = port - > base ;
u32 bit , val ;
int edge ;
reg + = GPIO_ICR1 + ( ( gpio & 0x10 ) > > 2 ) ; /* lower or upper register */
bit = gpio & 0xf ;
2011-06-06 00:07:55 +08:00
val = readl ( reg ) ;
2009-03-12 12:46:41 +01:00
edge = ( val > > ( bit < < 1 ) ) & 3 ;
val & = ~ ( 0x3 < < ( bit < < 1 ) ) ;
2010-02-05 22:14:37 +01:00
if ( edge = = GPIO_INT_HIGH_LEV ) {
2009-03-12 12:46:41 +01:00
edge = GPIO_INT_LOW_LEV ;
pr_debug ( " mxc: switch GPIO %d to low trigger \n " , gpio ) ;
2010-02-05 22:14:37 +01:00
} else if ( edge = = GPIO_INT_LOW_LEV ) {
2009-03-12 12:46:41 +01:00
edge = GPIO_INT_HIGH_LEV ;
pr_debug ( " mxc: switch GPIO %d to high trigger \n " , gpio ) ;
2010-02-05 22:14:37 +01:00
} else {
2009-03-12 12:46:41 +01:00
pr_err ( " mxc: invalid configuration for GPIO %d: %x \n " ,
gpio , edge ) ;
return ;
}
2011-06-06 00:07:55 +08:00
writel ( val | ( edge < < ( bit < < 1 ) ) , reg ) ;
2009-03-12 12:46:41 +01:00
}
2010-02-08 21:02:30 +01:00
/* handle 32 interrupts in one status register */
2008-07-05 10:02:49 +02:00
static void mxc_gpio_irq_handler ( struct mxc_gpio_port * port , u32 irq_stat )
{
2010-02-08 21:02:30 +01:00
u32 gpio_irq_no_base = port - > virtual_irq_start ;
2008-07-05 10:02:49 +02:00
2010-02-08 21:02:30 +01:00
while ( irq_stat ! = 0 ) {
int irqoffset = fls ( irq_stat ) - 1 ;
2008-07-05 10:02:49 +02:00
2010-02-08 21:02:30 +01:00
if ( port - > both_edges & ( 1 < < irqoffset ) )
mxc_flip_edge ( port , irqoffset ) ;
2009-03-12 12:46:41 +01:00
2010-02-08 21:02:30 +01:00
generic_handle_irq ( gpio_irq_no_base + irqoffset ) ;
2009-03-12 12:46:41 +01:00
2010-02-08 21:02:30 +01:00
irq_stat & = ~ ( 1 < < irqoffset ) ;
2008-07-05 10:02:49 +02:00
}
}
2008-11-14 11:01:38 +01:00
/* MX1 and MX3 has one interrupt *per* gpio port */
2008-07-05 10:02:49 +02:00
static void mx3_gpio_irq_handler ( u32 irq , struct irq_desc * desc )
{
u32 irq_stat ;
2011-03-24 13:25:22 +01:00
struct mxc_gpio_port * port = irq_get_handler_data ( irq ) ;
2011-09-21 21:24:04 +08:00
struct irq_chip * chip = irq_get_chip ( irq ) ;
chained_irq_enter ( chip , desc ) ;
2008-07-05 10:02:49 +02:00
2011-06-06 00:07:55 +08:00
irq_stat = readl ( port - > base + GPIO_ISR ) & readl ( port - > base + GPIO_IMR ) ;
2009-04-21 12:39:59 +02:00
2008-07-05 10:02:49 +02:00
mxc_gpio_irq_handler ( port , irq_stat ) ;
2011-09-21 21:24:04 +08:00
chained_irq_exit ( chip , desc ) ;
2008-07-05 10:02:49 +02:00
}
/* MX2 has one interrupt *for all* gpio ports */
static void mx2_gpio_irq_handler ( u32 irq , struct irq_desc * desc )
{
u32 irq_msk , irq_stat ;
2011-06-06 00:07:55 +08:00
struct mxc_gpio_port * port ;
2008-07-05 10:02:49 +02:00
/* walk through all interrupt status registers */
2011-06-06 00:07:55 +08:00
list_for_each_entry ( port , & mxc_gpio_ports , node ) {
irq_msk = readl ( port - > base + GPIO_IMR ) ;
2008-07-05 10:02:49 +02:00
if ( ! irq_msk )
continue ;
2011-06-06 00:07:55 +08:00
irq_stat = readl ( port - > base + GPIO_ISR ) & irq_msk ;
2008-07-05 10:02:49 +02:00
if ( irq_stat )
2011-06-06 00:07:55 +08:00
mxc_gpio_irq_handler ( port , irq_stat ) ;
2008-07-05 10:02:49 +02:00
}
}
2010-10-23 09:12:48 -05:00
/*
* Set interrupt number " irq " in the GPIO as a wake - up source .
* While system is running , all registered GPIO interrupts need to have
* wake - up enabled . When system is suspended , only selected GPIO interrupts
* need to have wake - up enabled .
* @ param irq interrupt source number
* @ param enable enable as wake - up if equal to non - zero
* @ return This function returns 0 on success .
*/
2010-11-29 11:16:23 +01:00
static int gpio_set_wake_irq ( struct irq_data * d , u32 enable )
2010-10-23 09:12:48 -05:00
{
2010-11-29 11:16:23 +01:00
u32 gpio = irq_to_gpio ( d - > irq ) ;
2010-10-23 09:12:48 -05:00
u32 gpio_idx = gpio & 0x1F ;
2011-06-07 16:25:37 +08:00
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct mxc_gpio_port * port = gc - > private ;
2010-10-23 09:12:48 -05:00
if ( enable ) {
if ( port - > irq_high & & ( gpio_idx > = 16 ) )
enable_irq_wake ( port - > irq_high ) ;
else
enable_irq_wake ( port - > irq ) ;
} else {
if ( port - > irq_high & & ( gpio_idx > = 16 ) )
disable_irq_wake ( port - > irq_high ) ;
else
disable_irq_wake ( port - > irq ) ;
}
return 0 ;
}
2011-06-07 16:25:37 +08:00
static void __init mxc_gpio_init_gc ( struct mxc_gpio_port * port )
{
struct irq_chip_generic * gc ;
struct irq_chip_type * ct ;
gc = irq_alloc_generic_chip ( " gpio-mxc " , 1 , port - > virtual_irq_start ,
port - > base , handle_level_irq ) ;
gc - > private = port ;
ct = gc - > chip_types ;
2011-07-19 21:16:56 +08:00
ct - > chip . irq_ack = irq_gc_ack_set_bit ;
2011-06-07 16:25:37 +08:00
ct - > chip . irq_mask = irq_gc_mask_clr_bit ;
ct - > chip . irq_unmask = irq_gc_mask_set_bit ;
ct - > chip . irq_set_type = gpio_set_irq_type ;
2011-07-19 21:16:56 +08:00
ct - > chip . irq_set_wake = gpio_set_wake_irq ;
2011-06-07 16:25:37 +08:00
ct - > regs . ack = GPIO_ISR ;
ct - > regs . mask = GPIO_IMR ;
irq_setup_generic_chip ( gc , IRQ_MSK ( 32 ) , IRQ_GC_INIT_NESTED_LOCK ,
IRQ_NOREQUEST , 0 ) ;
}
2011-04-04 14:29:58 +02:00
2011-07-07 00:37:41 +08:00
static void __devinit mxc_gpio_get_hw ( struct platform_device * pdev )
{
2011-07-07 00:37:43 +08:00
const struct of_device_id * of_id =
of_match_device ( mxc_gpio_dt_ids , & pdev - > dev ) ;
enum mxc_gpio_hwtype hwtype ;
if ( of_id )
pdev - > id_entry = of_id - > data ;
hwtype = pdev - > id_entry - > driver_data ;
2011-07-07 00:37:41 +08:00
if ( mxc_gpio_hwtype ) {
/*
* The driver works with a reasonable presupposition ,
* that is all gpio ports must be the same type when
* running on one soc .
*/
BUG_ON ( mxc_gpio_hwtype ! = hwtype ) ;
return ;
}
if ( hwtype = = IMX31_GPIO )
mxc_gpio_hwdata = & imx31_gpio_hwdata ;
else
mxc_gpio_hwdata = & imx1_imx21_gpio_hwdata ;
mxc_gpio_hwtype = hwtype ;
}
2011-08-14 00:14:02 +08:00
static int mxc_gpio_to_irq ( struct gpio_chip * gc , unsigned offset )
{
struct bgpio_chip * bgc = to_bgpio_chip ( gc ) ;
struct mxc_gpio_port * port =
container_of ( bgc , struct mxc_gpio_port , bgc ) ;
return port - > virtual_irq_start + offset ;
}
2011-06-06 00:07:55 +08:00
static int __devinit mxc_gpio_probe ( struct platform_device * pdev )
2008-07-05 10:02:49 +02:00
{
2011-07-07 00:37:43 +08:00
struct device_node * np = pdev - > dev . of_node ;
2011-06-06 00:07:55 +08:00
struct mxc_gpio_port * port ;
struct resource * iores ;
2011-06-07 16:25:37 +08:00
int err ;
2011-06-06 00:07:55 +08:00
2011-07-07 00:37:41 +08:00
mxc_gpio_get_hw ( pdev ) ;
2011-06-06 00:07:55 +08:00
port = kzalloc ( sizeof ( struct mxc_gpio_port ) , GFP_KERNEL ) ;
if ( ! port )
return - ENOMEM ;
2008-07-05 10:02:49 +02:00
2011-06-06 00:07:55 +08:00
iores = platform_get_resource ( pdev , IORESOURCE_MEM , 0 ) ;
if ( ! iores ) {
err = - ENODEV ;
goto out_kfree ;
}
2010-07-06 14:03:22 +03:00
2011-06-06 00:07:55 +08:00
if ( ! request_mem_region ( iores - > start , resource_size ( iores ) ,
pdev - > name ) ) {
err = - EBUSY ;
goto out_kfree ;
}
2008-07-05 10:02:49 +02:00
2011-06-06 00:07:55 +08:00
port - > base = ioremap ( iores - > start , resource_size ( iores ) ) ;
if ( ! port - > base ) {
err = - ENOMEM ;
goto out_release_mem ;
}
port - > irq_high = platform_get_irq ( pdev , 1 ) ;
port - > irq = platform_get_irq ( pdev , 0 ) ;
if ( port - > irq < 0 ) {
err = - EINVAL ;
goto out_iounmap ;
}
/* disable the interrupt and clear the status */
writel ( 0 , port - > base + GPIO_IMR ) ;
writel ( ~ 0 , port - > base + GPIO_ISR ) ;
2011-07-07 00:37:41 +08:00
if ( mxc_gpio_hwtype = = IMX21_GPIO ) {
2009-06-15 12:36:25 +02:00
/* setup one handler for all GPIO interrupts */
2011-06-06 00:07:55 +08:00
if ( pdev - > id = = 0 )
irq_set_chained_handler ( port - > irq ,
mx2_gpio_irq_handler ) ;
} else {
/* setup one handler for each entry */
irq_set_chained_handler ( port - > irq , mx3_gpio_irq_handler ) ;
irq_set_handler_data ( port - > irq , port ) ;
if ( port - > irq_high > 0 ) {
/* setup handler for GPIO 16 to 31 */
irq_set_chained_handler ( port - > irq_high ,
mx3_gpio_irq_handler ) ;
irq_set_handler_data ( port - > irq_high , port ) ;
}
2008-07-05 10:02:49 +02:00
}
2011-06-06 13:22:41 +08:00
err = bgpio_init ( & port - > bgc , & pdev - > dev , 4 ,
port - > base + GPIO_PSR ,
port - > base + GPIO_DR , NULL ,
port - > base + GPIO_GDIR , NULL , false ) ;
if ( err )
goto out_iounmap ;
2011-06-06 00:07:55 +08:00
2011-08-14 00:14:02 +08:00
port - > bgc . gc . to_irq = mxc_gpio_to_irq ;
2011-06-06 13:22:41 +08:00
port - > bgc . gc . base = pdev - > id * 32 ;
2011-07-07 14:50:16 +02:00
port - > bgc . dir = port - > bgc . read_reg ( port - > bgc . reg_dir ) ;
port - > bgc . data = port - > bgc . read_reg ( port - > bgc . reg_set ) ;
2011-06-06 00:07:55 +08:00
2011-06-06 13:22:41 +08:00
err = gpiochip_add ( & port - > bgc . gc ) ;
2011-06-06 00:07:55 +08:00
if ( err )
2011-06-06 13:22:41 +08:00
goto out_bgpio_remove ;
2011-06-06 00:07:55 +08:00
2011-07-07 00:37:43 +08:00
/*
* In dt case , we use gpio number range dynamically
* allocated by gpio core .
*/
port - > virtual_irq_start = MXC_GPIO_IRQ_START + ( np ? port - > bgc . gc . base :
pdev - > id * 32 ) ;
/* gpio-mxc can be a generic irq chip */
mxc_gpio_init_gc ( port ) ;
2011-06-06 00:07:55 +08:00
list_add_tail ( & port - > node , & mxc_gpio_ports ) ;
2008-07-05 10:02:49 +02:00
return 0 ;
2011-06-06 00:07:55 +08:00
2011-06-06 13:22:41 +08:00
out_bgpio_remove :
bgpio_remove ( & port - > bgc ) ;
2011-06-06 00:07:55 +08:00
out_iounmap :
iounmap ( port - > base ) ;
out_release_mem :
release_mem_region ( iores - > start , resource_size ( iores ) ) ;
out_kfree :
kfree ( port ) ;
dev_info ( & pdev - > dev , " %s failed with errno %d \n " , __func__ , err ) ;
return err ;
2008-07-05 10:02:49 +02:00
}
2011-06-06 00:07:55 +08:00
static struct platform_driver mxc_gpio_driver = {
. driver = {
. name = " gpio-mxc " ,
. owner = THIS_MODULE ,
2011-07-07 00:37:43 +08:00
. of_match_table = mxc_gpio_dt_ids ,
2011-06-06 00:07:55 +08:00
} ,
. probe = mxc_gpio_probe ,
2011-07-07 00:37:41 +08:00
. id_table = mxc_gpio_devtype ,
2011-06-06 00:07:55 +08:00
} ;
static int __init gpio_mxc_init ( void )
{
return platform_driver_register ( & mxc_gpio_driver ) ;
}
postcore_initcall ( gpio_mxc_init ) ;
MODULE_AUTHOR ( " Freescale Semiconductor, "
" Daniel Mack <danielncaiaq.de>, "
" Juergen Beisert <kernel@pengutronix.de> " ) ;
MODULE_DESCRIPTION ( " Freescale MXC GPIO " ) ;
MODULE_LICENSE ( " GPL " ) ;