2008-10-21 17:07:08 +04:00
/* arch/arm/plat-s3c64xx/irq-eint.c
*
* Copyright 2008 Openmoko , Inc .
* Copyright 2008 Simtec Electronics
* Ben Dooks < ben @ simtec . co . uk >
* http : //armlinux.simtec.co.uk/
*
* S3C64XX - Interrupt handling for IRQ_EINT ( x )
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
# include <linux/kernel.h>
# include <linux/interrupt.h>
2009-03-10 21:19:35 +03:00
# include <linux/sysdev.h>
2008-12-12 03:24:33 +03:00
# include <linux/gpio.h>
2008-10-21 17:07:08 +04:00
# include <linux/irq.h>
# include <linux/io.h>
# include <asm/hardware/vic.h>
# include <plat/regs-irqtype.h>
2008-12-12 03:24:33 +03:00
# include <plat/regs-gpio.h>
# include <plat/gpio-cfg.h>
2008-10-21 17:07:08 +04:00
# include <mach/map.h>
# include <plat/cpu.h>
2009-03-10 21:19:35 +03:00
# include <plat/pm.h>
2008-10-21 17:07:08 +04:00
# define eint_offset(irq) ((irq) - IRQ_EINT(0))
# define eint_irq_to_bit(irq) (1 << eint_offset(irq))
static inline void s3c_irq_eint_mask ( unsigned int irq )
{
u32 mask ;
mask = __raw_readl ( S3C64XX_EINT0MASK ) ;
mask | = eint_irq_to_bit ( irq ) ;
__raw_writel ( mask , S3C64XX_EINT0MASK ) ;
}
static void s3c_irq_eint_unmask ( unsigned int irq )
{
u32 mask ;
mask = __raw_readl ( S3C64XX_EINT0MASK ) ;
2009-02-24 17:55:48 +03:00
mask & = ~ eint_irq_to_bit ( irq ) ;
2008-10-21 17:07:08 +04:00
__raw_writel ( mask , S3C64XX_EINT0MASK ) ;
}
static inline void s3c_irq_eint_ack ( unsigned int irq )
{
__raw_writel ( eint_irq_to_bit ( irq ) , S3C64XX_EINT0PEND ) ;
}
static void s3c_irq_eint_maskack ( unsigned int irq )
{
/* compiler should in-line these */
s3c_irq_eint_mask ( irq ) ;
s3c_irq_eint_ack ( irq ) ;
}
static int s3c_irq_eint_set_type ( unsigned int irq , unsigned int type )
{
int offs = eint_offset ( irq ) ;
2009-11-20 15:04:13 +03:00
int pin , pin_val ;
2008-10-21 17:07:08 +04:00
int shift ;
u32 ctrl , mask ;
u32 newvalue = 0 ;
void __iomem * reg ;
if ( offs > 27 )
return - EINVAL ;
2008-12-02 22:03:28 +03:00
if ( offs < = 15 )
2008-10-21 17:07:08 +04:00
reg = S3C64XX_EINT0CON0 ;
else
reg = S3C64XX_EINT0CON1 ;
switch ( type ) {
case IRQ_TYPE_NONE :
printk ( KERN_WARNING " No edge setting! \n " ) ;
break ;
case IRQ_TYPE_EDGE_RISING :
newvalue = S3C2410_EXTINT_RISEEDGE ;
break ;
case IRQ_TYPE_EDGE_FALLING :
newvalue = S3C2410_EXTINT_FALLEDGE ;
break ;
case IRQ_TYPE_EDGE_BOTH :
newvalue = S3C2410_EXTINT_BOTHEDGE ;
break ;
case IRQ_TYPE_LEVEL_LOW :
newvalue = S3C2410_EXTINT_LOWLEV ;
break ;
case IRQ_TYPE_LEVEL_HIGH :
newvalue = S3C2410_EXTINT_HILEV ;
break ;
default :
printk ( KERN_ERR " No such irq type %d " , type ) ;
return - 1 ;
}
2009-11-20 15:04:13 +03:00
if ( offs < = 15 )
shift = ( offs / 2 ) * 4 ;
else
shift = ( ( offs - 16 ) / 2 ) * 4 ;
2008-10-21 17:07:08 +04:00
mask = 0x7 < < shift ;
ctrl = __raw_readl ( reg ) ;
ctrl & = ~ mask ;
ctrl | = newvalue < < shift ;
__raw_writel ( ctrl , reg ) ;
2008-12-12 03:24:33 +03:00
/* set the GPIO pin appropriately */
2009-11-20 15:04:13 +03:00
if ( offs < 16 ) {
2008-12-12 03:24:33 +03:00
pin = S3C64XX_GPN ( offs ) ;
2009-11-20 15:04:13 +03:00
pin_val = S3C_GPIO_SFN ( 2 ) ;
} else if ( offs < 23 ) {
pin = S3C64XX_GPL ( offs + 8 - 16 ) ;
pin_val = S3C_GPIO_SFN ( 3 ) ;
} else {
2008-12-12 03:24:33 +03:00
pin = S3C64XX_GPM ( offs - 23 ) ;
2009-11-20 15:04:13 +03:00
pin_val = S3C_GPIO_SFN ( 3 ) ;
}
2008-12-12 03:24:33 +03:00
2009-11-20 15:04:13 +03:00
s3c_gpio_cfgpin ( pin , pin_val ) ;
2008-12-12 03:24:33 +03:00
2008-10-21 17:07:08 +04:00
return 0 ;
}
static struct irq_chip s3c_irq_eint = {
. name = " s3c-eint " ,
. mask = s3c_irq_eint_mask ,
. unmask = s3c_irq_eint_unmask ,
. mask_ack = s3c_irq_eint_maskack ,
. ack = s3c_irq_eint_ack ,
. set_type = s3c_irq_eint_set_type ,
2009-03-10 21:19:35 +03:00
. set_wake = s3c_irqext_wake ,
2008-10-21 17:07:08 +04:00
} ;
/* s3c_irq_demux_eint
*
* This function demuxes the IRQ from the group0 external interrupts ,
* from IRQ_EINT ( 0 ) to IRQ_EINT ( 27 ) . It is designed to be inlined into
* the specific handlers s3c_irq_demux_eintX_Y .
*/
static inline void s3c_irq_demux_eint ( unsigned int start , unsigned int end )
{
u32 status = __raw_readl ( S3C64XX_EINT0PEND ) ;
u32 mask = __raw_readl ( S3C64XX_EINT0MASK ) ;
unsigned int irq ;
status & = ~ mask ;
status > > = start ;
status & = ( 1 < < ( end - start + 1 ) ) - 1 ;
for ( irq = IRQ_EINT ( start ) ; irq < = IRQ_EINT ( end ) ; irq + + ) {
if ( status & 1 )
generic_handle_irq ( irq ) ;
status > > = 1 ;
}
}
static void s3c_irq_demux_eint0_3 ( unsigned int irq , struct irq_desc * desc )
{
s3c_irq_demux_eint ( 0 , 3 ) ;
}
static void s3c_irq_demux_eint4_11 ( unsigned int irq , struct irq_desc * desc )
{
s3c_irq_demux_eint ( 4 , 11 ) ;
}
static void s3c_irq_demux_eint12_19 ( unsigned int irq , struct irq_desc * desc )
{
s3c_irq_demux_eint ( 12 , 19 ) ;
}
static void s3c_irq_demux_eint20_27 ( unsigned int irq , struct irq_desc * desc )
{
s3c_irq_demux_eint ( 20 , 27 ) ;
}
2009-01-23 19:29:44 +03:00
static int __init s3c64xx_init_irq_eint ( void )
2008-10-21 17:07:08 +04:00
{
int irq ;
for ( irq = IRQ_EINT ( 0 ) ; irq < = IRQ_EINT ( 27 ) ; irq + + ) {
set_irq_chip ( irq , & s3c_irq_eint ) ;
set_irq_handler ( irq , handle_level_irq ) ;
set_irq_flags ( irq , IRQF_VALID ) ;
}
set_irq_chained_handler ( IRQ_EINT0_3 , s3c_irq_demux_eint0_3 ) ;
set_irq_chained_handler ( IRQ_EINT4_11 , s3c_irq_demux_eint4_11 ) ;
set_irq_chained_handler ( IRQ_EINT12_19 , s3c_irq_demux_eint12_19 ) ;
set_irq_chained_handler ( IRQ_EINT20_27 , s3c_irq_demux_eint20_27 ) ;
return 0 ;
}
arch_initcall ( s3c64xx_init_irq_eint ) ;