2007-07-10 16:03:43 +04:00
/*
* TI DaVinci GPIO Support
*
2008-09-08 10:41:04 +04:00
* Copyright ( c ) 2006 - 2007 David Brownell
2007-07-10 16:03:43 +04:00
* Copyright ( c ) 2007 , MontaVista Software , Inc . < source @ mvista . com >
*
* 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 .
*/
# include <linux/errno.h>
# include <linux/kernel.h>
# include <linux/clk.h>
# include <linux/err.h>
# include <linux/io.h>
2008-08-05 19:14:15 +04:00
# include <mach/gpio.h>
2007-07-10 16:03:43 +04:00
# include <asm/mach/irq.h>
2008-09-08 10:41:04 +04:00
static DEFINE_SPINLOCK ( gpio_lock ) ;
2007-07-10 16:03:43 +04:00
2008-09-08 10:41:04 +04:00
struct davinci_gpio {
struct gpio_chip chip ;
2010-02-26 02:36:38 +03:00
struct gpio_controller __iomem * regs ;
2009-06-26 04:01:31 +04:00
int irq_base ;
2008-09-08 10:41:04 +04:00
} ;
2007-07-10 16:03:43 +04:00
2010-05-02 02:37:51 +04:00
# define chip2controller(chip) \
container_of ( chip , struct davinci_gpio , chip )
2008-09-08 10:41:04 +04:00
static struct davinci_gpio chips [ DIV_ROUND_UP ( DAVINCI_N_GPIO , 32 ) ] ;
2007-07-10 16:03:43 +04:00
/* create a non-inlined version */
2010-02-26 02:36:38 +03:00
static struct gpio_controller __iomem __init * gpio2controller ( unsigned gpio )
2007-07-10 16:03:43 +04:00
{
return __gpio_to_controller ( gpio ) ;
}
2010-02-26 03:49:56 +03:00
static inline struct gpio_controller __iomem * irq2controller ( int irq )
{
struct gpio_controller __iomem * g ;
g = ( __force struct gpio_controller __iomem * ) get_irq_chip_data ( irq ) ;
return g ;
}
2009-05-11 22:04:53 +04:00
static int __init davinci_gpio_irq_setup ( void ) ;
2008-09-08 10:41:04 +04:00
/*--------------------------------------------------------------------------*/
2007-07-10 16:03:43 +04:00
/*
2008-09-08 10:41:04 +04:00
* board setup code * MUST * set PINMUX0 and PINMUX1 as
* needed , and enable the GPIO clock .
2007-07-10 16:03:43 +04:00
*/
2008-09-08 10:41:04 +04:00
2010-05-02 02:37:51 +04:00
static inline int __davinci_direction ( struct gpio_chip * chip ,
unsigned offset , bool out , int value )
2007-07-10 16:03:43 +04:00
{
2010-05-02 02:37:51 +04:00
struct davinci_gpio * d = chip2controller ( chip ) ;
2010-02-26 02:36:38 +03:00
struct gpio_controller __iomem * g = d - > regs ;
2008-09-08 10:41:04 +04:00
u32 temp ;
2010-05-02 02:37:51 +04:00
u32 mask = 1 < < offset ;
2007-07-10 16:03:43 +04:00
2008-09-08 10:41:04 +04:00
spin_lock ( & gpio_lock ) ;
temp = __raw_readl ( & g - > dir ) ;
2010-05-02 02:37:51 +04:00
if ( out ) {
temp & = ~ mask ;
__raw_writel ( mask , value ? & g - > set_data : & g - > clr_data ) ;
} else {
temp | = mask ;
}
2008-09-08 10:41:04 +04:00
__raw_writel ( temp , & g - > dir ) ;
spin_unlock ( & gpio_lock ) ;
2007-07-10 16:03:43 +04:00
2008-09-08 10:41:04 +04:00
return 0 ;
}
2007-07-10 16:03:43 +04:00
2010-05-02 02:37:51 +04:00
static int davinci_direction_in ( struct gpio_chip * chip , unsigned offset )
{
return __davinci_direction ( chip , offset , false , 0 ) ;
}
static int
davinci_direction_out ( struct gpio_chip * chip , unsigned offset , int value )
{
return __davinci_direction ( chip , offset , true , value ) ;
}
2007-07-10 16:03:43 +04:00
/*
* Read the pin ' s value ( works even if it ' s set up as output ) ;
* returns zero / nonzero .
*
* Note that changes are synched to the GPIO clock , so reading values back
* right after you ' ve set them may give old values .
*/
2008-09-08 10:41:04 +04:00
static int davinci_gpio_get ( struct gpio_chip * chip , unsigned offset )
2007-07-10 16:03:43 +04:00
{
2010-05-02 02:37:51 +04:00
struct davinci_gpio * d = chip2controller ( chip ) ;
2010-02-26 02:36:38 +03:00
struct gpio_controller __iomem * g = d - > regs ;
2007-07-10 16:03:43 +04:00
2008-09-08 10:41:04 +04:00
return ( 1 < < offset ) & __raw_readl ( & g - > in_data ) ;
2007-07-10 16:03:43 +04:00
}
2008-09-08 10:41:04 +04:00
/*
* Assuming the pin is muxed as a gpio output , set its output value .
*/
static void
davinci_gpio_set ( struct gpio_chip * chip , unsigned offset , int value )
2007-07-10 16:03:43 +04:00
{
2010-05-02 02:37:51 +04:00
struct davinci_gpio * d = chip2controller ( chip ) ;
2010-02-26 02:36:38 +03:00
struct gpio_controller __iomem * g = d - > regs ;
2007-07-10 16:03:43 +04:00
2008-09-08 10:41:04 +04:00
__raw_writel ( ( 1 < < offset ) , value ? & g - > set_data : & g - > clr_data ) ;
}
static int __init davinci_gpio_setup ( void )
{
int i , base ;
2009-04-15 23:40:35 +04:00
unsigned ngpio ;
struct davinci_soc_info * soc_info = & davinci_soc_info ;
2008-09-08 10:41:04 +04:00
2009-04-15 23:40:35 +04:00
/*
* The gpio banks conceptually expose a segmented bitmap ,
2008-12-07 22:46:23 +03:00
* and " ngpio " is one more than the largest zero - based
* bit index that ' s valid .
*/
2009-04-15 23:40:35 +04:00
ngpio = soc_info - > gpio_num ;
if ( ngpio = = 0 ) {
2008-12-07 22:46:23 +03:00
pr_err ( " GPIO setup: how many GPIOs? \n " ) ;
return - EINVAL ;
}
if ( WARN_ON ( DAVINCI_N_GPIO < ngpio ) )
ngpio = DAVINCI_N_GPIO ;
for ( i = 0 , base = 0 ; base < ngpio ; i + + , base + = 32 ) {
2008-09-08 10:41:04 +04:00
chips [ i ] . chip . label = " DaVinci " ;
chips [ i ] . chip . direction_input = davinci_direction_in ;
chips [ i ] . chip . get = davinci_gpio_get ;
chips [ i ] . chip . direction_output = davinci_direction_out ;
chips [ i ] . chip . set = davinci_gpio_set ;
chips [ i ] . chip . base = base ;
2008-12-07 22:46:23 +03:00
chips [ i ] . chip . ngpio = ngpio - base ;
2008-09-08 10:41:04 +04:00
if ( chips [ i ] . chip . ngpio > 32 )
chips [ i ] . chip . ngpio = 32 ;
chips [ i ] . regs = gpio2controller ( base ) ;
gpiochip_add ( & chips [ i ] . chip ) ;
}
2007-07-10 16:03:43 +04:00
2009-05-11 22:04:53 +04:00
davinci_gpio_irq_setup ( ) ;
2007-07-10 16:03:43 +04:00
return 0 ;
}
2008-09-08 10:41:04 +04:00
pure_initcall ( davinci_gpio_setup ) ;
2007-07-10 16:03:43 +04:00
2008-09-08 10:41:04 +04:00
/*--------------------------------------------------------------------------*/
2007-07-10 16:03:43 +04:00
/*
* We expect irqs will normally be set up as input pins , but they can also be
* used as output pins . . . which is convenient for testing .
*
2008-12-07 22:46:23 +03:00
* NOTE : The first few GPIOs also have direct INTC hookups in addition
2009-06-26 04:01:31 +04:00
* to their GPIOBNK0 irq , with a bit less overhead .
2007-07-10 16:03:43 +04:00
*
2008-12-07 22:46:23 +03:00
* All those INTC hookups ( direct , plus several IRQ banks ) can also
2007-07-10 16:03:43 +04:00
* serve as EDMA event triggers .
*/
static void gpio_irq_disable ( unsigned irq )
{
2010-02-26 03:49:56 +03:00
struct gpio_controller __iomem * g = irq2controller ( irq ) ;
2009-06-26 04:01:31 +04:00
u32 mask = ( u32 ) get_irq_data ( irq ) ;
2007-07-10 16:03:43 +04:00
__raw_writel ( mask , & g - > clr_falling ) ;
__raw_writel ( mask , & g - > clr_rising ) ;
}
static void gpio_irq_enable ( unsigned irq )
{
2010-02-26 03:49:56 +03:00
struct gpio_controller __iomem * g = irq2controller ( irq ) ;
2009-06-26 04:01:31 +04:00
u32 mask = ( u32 ) get_irq_data ( irq ) ;
2009-05-05 00:14:27 +04:00
unsigned status = irq_desc [ irq ] . status ;
2007-07-10 16:03:43 +04:00
2009-05-05 00:14:27 +04:00
status & = IRQ_TYPE_EDGE_FALLING | IRQ_TYPE_EDGE_RISING ;
if ( ! status )
status = IRQ_TYPE_EDGE_FALLING | IRQ_TYPE_EDGE_RISING ;
if ( status & IRQ_TYPE_EDGE_FALLING )
2007-07-10 16:03:43 +04:00
__raw_writel ( mask , & g - > set_falling ) ;
2009-05-05 00:14:27 +04:00
if ( status & IRQ_TYPE_EDGE_RISING )
2007-07-10 16:03:43 +04:00
__raw_writel ( mask , & g - > set_rising ) ;
}
static int gpio_irq_type ( unsigned irq , unsigned trigger )
{
2010-02-26 03:49:56 +03:00
struct gpio_controller __iomem * g = irq2controller ( irq ) ;
2009-06-26 04:01:31 +04:00
u32 mask = ( u32 ) get_irq_data ( irq ) ;
2007-07-10 16:03:43 +04:00
if ( trigger & ~ ( IRQ_TYPE_EDGE_FALLING | IRQ_TYPE_EDGE_RISING ) )
return - EINVAL ;
irq_desc [ irq ] . status & = ~ IRQ_TYPE_SENSE_MASK ;
irq_desc [ irq ] . status | = trigger ;
2009-05-05 00:14:27 +04:00
/* don't enable the IRQ if it's currently disabled */
if ( irq_desc [ irq ] . depth = = 0 ) {
__raw_writel ( mask , ( trigger & IRQ_TYPE_EDGE_FALLING )
? & g - > set_falling : & g - > clr_falling ) ;
__raw_writel ( mask , ( trigger & IRQ_TYPE_EDGE_RISING )
? & g - > set_rising : & g - > clr_rising ) ;
}
2007-07-10 16:03:43 +04:00
return 0 ;
}
static struct irq_chip gpio_irqchip = {
. name = " GPIO " ,
. enable = gpio_irq_enable ,
. disable = gpio_irq_disable ,
. set_type = gpio_irq_type ,
} ;
static void
gpio_irq_handler ( unsigned irq , struct irq_desc * desc )
{
2010-02-26 03:49:56 +03:00
struct gpio_controller __iomem * g = irq2controller ( irq ) ;
2007-07-10 16:03:43 +04:00
u32 mask = 0xffff ;
/* we only care about one bank */
if ( irq & 1 )
mask < < = 16 ;
/* temporarily mask (level sensitive) parent IRQ */
2009-05-11 22:04:53 +04:00
desc - > chip - > mask ( irq ) ;
2007-07-10 16:03:43 +04:00
desc - > chip - > ack ( irq ) ;
while ( 1 ) {
u32 status ;
int n ;
int res ;
/* ack any irqs */
status = __raw_readl ( & g - > intstat ) & mask ;
if ( ! status )
break ;
__raw_writel ( status , & g - > intstat ) ;
if ( irq & 1 )
status > > = 16 ;
/* now demux them to the right lowlevel handler */
n = ( int ) get_irq_data ( irq ) ;
while ( status ) {
res = ffs ( status ) ;
n + = res ;
2008-10-09 16:36:24 +04:00
generic_handle_irq ( n - 1 ) ;
2007-07-10 16:03:43 +04:00
status > > = res ;
}
}
desc - > chip - > unmask ( irq ) ;
/* now it may re-trigger */
}
2009-06-26 04:01:31 +04:00
static int gpio_to_irq_banked ( struct gpio_chip * chip , unsigned offset )
{
2010-05-02 02:37:51 +04:00
struct davinci_gpio * d = chip2controller ( chip ) ;
2009-06-26 04:01:31 +04:00
if ( d - > irq_base > = 0 )
return d - > irq_base + offset ;
else
return - ENODEV ;
}
static int gpio_to_irq_unbanked ( struct gpio_chip * chip , unsigned offset )
{
struct davinci_soc_info * soc_info = & davinci_soc_info ;
/* NOTE: we assume for now that only irqs in the first gpio_chip
* can provide direct - mapped IRQs to AINTC ( up to 32 GPIOs ) .
*/
if ( offset < soc_info - > gpio_unbanked )
return soc_info - > gpio_irq + offset ;
else
return - ENODEV ;
}
static int gpio_irq_type_unbanked ( unsigned irq , unsigned trigger )
{
2010-02-26 03:49:56 +03:00
struct gpio_controller __iomem * g = irq2controller ( irq ) ;
2009-06-26 04:01:31 +04:00
u32 mask = ( u32 ) get_irq_data ( irq ) ;
if ( trigger & ~ ( IRQ_TYPE_EDGE_FALLING | IRQ_TYPE_EDGE_RISING ) )
return - EINVAL ;
__raw_writel ( mask , ( trigger & IRQ_TYPE_EDGE_FALLING )
? & g - > set_falling : & g - > clr_falling ) ;
__raw_writel ( mask , ( trigger & IRQ_TYPE_EDGE_RISING )
? & g - > set_rising : & g - > clr_rising ) ;
return 0 ;
}
2007-07-10 16:03:43 +04:00
/*
2008-12-07 22:46:23 +03:00
* NOTE : for suspend / resume , probably best to make a platform_device with
* suspend_late / resume_resume calls hooking into results of the set_wake ( )
2007-07-10 16:03:43 +04:00
* calls . . . so if no gpios are wakeup events the clock can be disabled ,
* with outputs left at previously set levels , and so that VDD3P3V . IOPWDN0
2008-12-07 22:46:23 +03:00
* ( dm6446 ) can be set appropriately for GPIOV33 pins .
2007-07-10 16:03:43 +04:00
*/
static int __init davinci_gpio_irq_setup ( void )
{
unsigned gpio , irq , bank ;
struct clk * clk ;
2008-12-07 22:46:23 +03:00
u32 binten = 0 ;
2009-04-15 23:40:35 +04:00
unsigned ngpio , bank_irq ;
struct davinci_soc_info * soc_info = & davinci_soc_info ;
2010-02-26 02:36:38 +03:00
struct gpio_controller __iomem * g ;
2009-04-15 23:40:35 +04:00
ngpio = soc_info - > gpio_num ;
2008-12-07 22:46:23 +03:00
2009-04-15 23:40:35 +04:00
bank_irq = soc_info - > gpio_irq ;
if ( bank_irq = = 0 ) {
2008-12-07 22:46:23 +03:00
printk ( KERN_ERR " Don't know first GPIO bank IRQ. \n " ) ;
return - EINVAL ;
}
2007-07-10 16:03:43 +04:00
clk = clk_get ( NULL , " gpio " ) ;
if ( IS_ERR ( clk ) ) {
printk ( KERN_ERR " Error %ld getting gpio clock? \n " ,
PTR_ERR ( clk ) ) ;
2008-12-07 22:46:23 +03:00
return PTR_ERR ( clk ) ;
2007-07-10 16:03:43 +04:00
}
clk_enable ( clk ) ;
2009-06-26 04:01:31 +04:00
/* Arrange gpio_to_irq() support, handling either direct IRQs or
* banked IRQs . Having GPIOs in the first GPIO bank use direct
* IRQs , while the others use banked IRQs , would need some setup
* tweaks to recognize hardware which can do that .
*/
for ( gpio = 0 , bank = 0 ; gpio < ngpio ; bank + + , gpio + = 32 ) {
chips [ bank ] . chip . to_irq = gpio_to_irq_banked ;
chips [ bank ] . irq_base = soc_info - > gpio_unbanked
? - EINVAL
: ( soc_info - > intc_irq_num + gpio ) ;
}
/*
* AINTC can handle direct / unbanked IRQs for GPIOs , with the GPIO
* controller only handling trigger modes . We currently assume no
* IRQ mux conflicts ; gpio_irq_type_unbanked ( ) is only for GPIOs .
*/
if ( soc_info - > gpio_unbanked ) {
static struct irq_chip gpio_irqchip_unbanked ;
/* pass "bank 0" GPIO IRQs to AINTC */
chips [ 0 ] . chip . to_irq = gpio_to_irq_unbanked ;
binten = BIT ( 0 ) ;
/* AINTC handles mask/unmask; GPIO handles triggering */
irq = bank_irq ;
gpio_irqchip_unbanked = * get_irq_desc_chip ( irq_to_desc ( irq ) ) ;
gpio_irqchip_unbanked . name = " GPIO-AINTC " ;
gpio_irqchip_unbanked . set_type = gpio_irq_type_unbanked ;
/* default trigger: both edges */
g = gpio2controller ( 0 ) ;
__raw_writel ( ~ 0 , & g - > set_falling ) ;
__raw_writel ( ~ 0 , & g - > set_rising ) ;
/* set the direct IRQs up to use that irqchip */
for ( gpio = 0 ; gpio < soc_info - > gpio_unbanked ; gpio + + , irq + + ) {
set_irq_chip ( irq , & gpio_irqchip_unbanked ) ;
set_irq_data ( irq , ( void * ) __gpio_mask ( gpio ) ) ;
2010-02-26 03:49:56 +03:00
set_irq_chip_data ( irq , ( __force void * ) g ) ;
2009-06-26 04:01:31 +04:00
irq_desc [ irq ] . status | = IRQ_TYPE_EDGE_BOTH ;
}
goto done ;
}
/*
* Or , AINTC can handle IRQs for banks of 16 GPIO IRQs , which we
* then chain through our own handler .
*/
2008-12-07 22:46:23 +03:00
for ( gpio = 0 , irq = gpio_to_irq ( 0 ) , bank = 0 ;
gpio < ngpio ;
bank + + , bank_irq + + ) {
2007-07-10 16:03:43 +04:00
unsigned i ;
2009-06-26 04:01:31 +04:00
/* disabled by default, enabled only as needed */
g = gpio2controller ( gpio ) ;
2007-07-10 16:03:43 +04:00
__raw_writel ( ~ 0 , & g - > clr_falling ) ;
__raw_writel ( ~ 0 , & g - > clr_rising ) ;
/* set up all irqs in this bank */
2008-12-07 22:46:23 +03:00
set_irq_chained_handler ( bank_irq , gpio_irq_handler ) ;
2010-02-26 03:49:56 +03:00
set_irq_chip_data ( bank_irq , ( __force void * ) g ) ;
set_irq_data ( bank_irq , ( void * ) irq ) ;
2007-07-10 16:03:43 +04:00
2008-12-07 22:46:23 +03:00
for ( i = 0 ; i < 16 & & gpio < ngpio ; i + + , irq + + , gpio + + ) {
2007-07-10 16:03:43 +04:00
set_irq_chip ( irq , & gpio_irqchip ) ;
2010-02-26 03:49:56 +03:00
set_irq_chip_data ( irq , ( __force void * ) g ) ;
2009-06-26 04:01:31 +04:00
set_irq_data ( irq , ( void * ) __gpio_mask ( gpio ) ) ;
2007-07-10 16:03:43 +04:00
set_irq_handler ( irq , handle_simple_irq ) ;
set_irq_flags ( irq , IRQF_VALID ) ;
}
2008-12-07 22:46:23 +03:00
binten | = BIT ( bank ) ;
2007-07-10 16:03:43 +04:00
}
2009-06-26 04:01:31 +04:00
done :
2007-07-10 16:03:43 +04:00
/* BINTEN -- per-bank interrupt enable. genirq would also let these
* bits be set / cleared dynamically .
*/
2009-04-15 23:40:35 +04:00
__raw_writel ( binten , soc_info - > gpio_base + 0x08 ) ;
2007-07-10 16:03:43 +04:00
printk ( KERN_INFO " DaVinci: %d gpio irqs \n " , irq - gpio_to_irq ( 0 ) ) ;
return 0 ;
}