2005-04-16 15:20:36 -07:00
/*
* linux / arch / arm / mach - pxa / irq . c
*
* Generic PXA IRQ handling , GPIO IRQ demultiplexing , etc .
*
* Author : Nicolas Pitre
* Created : Jun 15 , 2001
* Copyright : MontaVista Software Inc .
*
* 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/init.h>
# include <linux/module.h>
# include <linux/interrupt.h>
# include <asm/hardware.h>
# include <asm/irq.h>
# include <asm/mach/irq.h>
# include <asm/arch/pxa-regs.h>
# include "generic.h"
/*
* This is for peripheral IRQs internal to the PXA chip .
*/
static void pxa_mask_low_irq ( unsigned int irq )
{
2007-06-06 06:22:20 +01:00
ICMR & = ~ ( 1 < < irq ) ;
2005-04-16 15:20:36 -07:00
}
static void pxa_unmask_low_irq ( unsigned int irq )
{
2007-06-06 06:22:20 +01:00
ICMR | = ( 1 < < irq ) ;
2005-04-16 15:20:36 -07:00
}
2007-02-26 01:44:57 +01:00
static int pxa_set_wake ( unsigned int irq , unsigned int on )
{
u32 mask ;
switch ( irq ) {
case IRQ_RTCAlrm :
mask = PWER_RTC ;
break ;
# ifdef CONFIG_PXA27x
/* REVISIT can handle USBH1, USBH2, USB, MSL, USIM, ... */
# endif
default :
return - EINVAL ;
}
if ( on )
PWER | = mask ;
else
PWER & = ~ mask ;
return 0 ;
}
2006-08-01 22:26:25 +01:00
static struct irq_chip pxa_internal_chip_low = {
. name = " SC " ,
2005-04-16 15:20:36 -07:00
. ack = pxa_mask_low_irq ,
. mask = pxa_mask_low_irq ,
. unmask = pxa_unmask_low_irq ,
2007-02-26 01:44:57 +01:00
. set_wake = pxa_set_wake ,
2005-04-16 15:20:36 -07:00
} ;
2007-06-06 06:36:04 +01:00
void __init pxa_init_irq_low ( void )
{
int irq ;
/* disable all IRQs */
ICMR = 0 ;
/* all IRQs are IRQ, not FIQ */
ICLR = 0 ;
/* only unmasked interrupts kick us out of idle */
ICCR = 1 ;
for ( irq = PXA_IRQ ( 0 ) ; irq < = PXA_IRQ ( 31 ) ; irq + + ) {
set_irq_chip ( irq , & pxa_internal_chip_low ) ;
set_irq_handler ( irq , handle_level_irq ) ;
set_irq_flags ( irq , IRQF_VALID ) ;
}
}
2007-06-06 06:32:38 +01:00
# ifdef CONFIG_PXA27x
2005-04-16 15:20:36 -07:00
/*
* This is for the second set of internal IRQs as found on the PXA27x .
*/
static void pxa_mask_high_irq ( unsigned int irq )
{
2007-06-06 06:22:20 +01:00
ICMR2 & = ~ ( 1 < < ( irq - 32 ) ) ;
2005-04-16 15:20:36 -07:00
}
static void pxa_unmask_high_irq ( unsigned int irq )
{
2007-06-06 06:22:20 +01:00
ICMR2 | = ( 1 < < ( irq - 32 ) ) ;
2005-04-16 15:20:36 -07:00
}
2006-08-01 22:26:25 +01:00
static struct irq_chip pxa_internal_chip_high = {
. name = " SC-hi " ,
2005-04-16 15:20:36 -07:00
. ack = pxa_mask_high_irq ,
. mask = pxa_mask_high_irq ,
. unmask = pxa_unmask_high_irq ,
} ;
2007-06-06 06:32:38 +01:00
void __init pxa_init_irq_high ( void )
{
int irq ;
ICMR2 = 0 ;
ICLR2 = 0 ;
for ( irq = PXA_IRQ ( 32 ) ; irq < PXA_IRQ ( 64 ) ; irq + + ) {
set_irq_chip ( irq , & pxa_internal_chip_high ) ;
set_irq_handler ( irq , handle_level_irq ) ;
set_irq_flags ( irq , IRQF_VALID ) ;
}
}
2005-04-16 15:20:36 -07:00
# endif
2007-02-26 01:44:57 +01:00
/* Note that if an input/irq line ever gets changed to an output during
* suspend , the relevant PWER , PRER , and PFER bits should be cleared .
*/
# ifdef CONFIG_PXA27x
/* PXA27x: Various gpios can issue wakeup events. This logic only
* handles the simple cases , not the WEMUX2 and WEMUX3 options
*/
# define PXA27x_GPIO_NOWAKE_MASK \
( ( 1 < < 8 ) | ( 1 < < 7 ) | ( 1 < < 6 ) | ( 1 < < 5 ) | ( 1 < < 2 ) )
# define WAKEMASK(gpio) \
( ( ( gpio ) < = 15 ) \
? ( ( 1 < < ( gpio ) ) & ~ PXA27x_GPIO_NOWAKE_MASK ) \
: ( ( gpio = = 35 ) ? ( 1 < < 24 ) : 0 ) )
# else
/* pxa 210, 250, 255, 26x: gpios 0..15 can issue wakeups */
# define WAKEMASK(gpio) (((gpio) <= 15) ? (1 << (gpio)) : 0)
# endif
2005-04-16 15:20:36 -07:00
/*
* PXA GPIO edge detection for IRQs :
* IRQs are generated on Falling - Edge , Rising - Edge , or both .
* Use this instead of directly setting GRER / GFER .
*/
static long GPIO_IRQ_rising_edge [ 4 ] ;
static long GPIO_IRQ_falling_edge [ 4 ] ;
static long GPIO_IRQ_mask [ 4 ] ;
static int pxa_gpio_irq_type ( unsigned int irq , unsigned int type )
{
int gpio , idx ;
2007-02-26 01:44:57 +01:00
u32 mask ;
2005-04-16 15:20:36 -07:00
gpio = IRQ_TO_GPIO ( irq ) ;
idx = gpio > > 5 ;
2007-02-26 01:44:57 +01:00
mask = WAKEMASK ( gpio ) ;
2005-04-16 15:20:36 -07:00
if ( type = = IRQT_PROBE ) {
/* Don't mess with enabled GPIOs using preconfigured edges or
2006-06-28 16:42:02 +01:00
GPIOs set to alternate function or to output during probe */
if ( ( GPIO_IRQ_rising_edge [ idx ] | GPIO_IRQ_falling_edge [ idx ] | GPDR ( gpio ) ) &
2005-04-16 15:20:36 -07:00
GPIO_bit ( gpio ) )
return 0 ;
if ( GAFR ( gpio ) & ( 0x3 < < ( ( ( gpio ) & 0xf ) * 2 ) ) )
return 0 ;
type = __IRQT_RISEDGE | __IRQT_FALEDGE ;
}
/* printk(KERN_DEBUG "IRQ%d (GPIO%d): ", irq, gpio); */
pxa_gpio_mode ( gpio | GPIO_IN ) ;
if ( type & __IRQT_RISEDGE ) {
/* printk("rising "); */
__set_bit ( gpio , GPIO_IRQ_rising_edge ) ;
2007-02-26 01:44:57 +01:00
PRER | = mask ;
} else {
2005-04-16 15:20:36 -07:00
__clear_bit ( gpio , GPIO_IRQ_rising_edge ) ;
2007-02-26 01:44:57 +01:00
PRER & = ~ mask ;
}
2005-04-16 15:20:36 -07:00
if ( type & __IRQT_FALEDGE ) {
/* printk("falling "); */
__set_bit ( gpio , GPIO_IRQ_falling_edge ) ;
2007-02-26 01:44:57 +01:00
PFER | = mask ;
} else {
2005-04-16 15:20:36 -07:00
__clear_bit ( gpio , GPIO_IRQ_falling_edge ) ;
2007-02-26 01:44:57 +01:00
PFER & = ~ mask ;
}
2005-04-16 15:20:36 -07:00
/* printk("edges\n"); */
GRER ( gpio ) = GPIO_IRQ_rising_edge [ idx ] & GPIO_IRQ_mask [ idx ] ;
GFER ( gpio ) = GPIO_IRQ_falling_edge [ idx ] & GPIO_IRQ_mask [ idx ] ;
return 0 ;
}
/*
* GPIO IRQs must be acknowledged . This is for GPIO 0 and 1.
*/
static void pxa_ack_low_gpio ( unsigned int irq )
{
GEDR0 = ( 1 < < ( irq - IRQ_GPIO0 ) ) ;
}
2007-02-26 01:44:57 +01:00
static int pxa_set_gpio_wake ( unsigned int irq , unsigned int on )
{
int gpio = IRQ_TO_GPIO ( irq ) ;
u32 mask = WAKEMASK ( gpio ) ;
if ( ! mask )
return - EINVAL ;
if ( on )
PWER | = mask ;
else
PWER & = ~ mask ;
return 0 ;
}
2006-08-01 22:26:25 +01:00
static struct irq_chip pxa_low_gpio_chip = {
. name = " GPIO-l " ,
2005-04-16 15:20:36 -07:00
. ack = pxa_ack_low_gpio ,
. mask = pxa_mask_low_irq ,
. unmask = pxa_unmask_low_irq ,
2005-09-04 19:43:13 +01:00
. set_type = pxa_gpio_irq_type ,
2007-02-26 01:44:57 +01:00
. set_wake = pxa_set_gpio_wake ,
2005-04-16 15:20:36 -07:00
} ;
/*
* Demux handler for GPIO > = 2 edge detect interrupts
*/
2006-11-23 11:41:32 +00:00
static void pxa_gpio_demux_handler ( unsigned int irq , struct irq_desc * desc )
2005-04-16 15:20:36 -07:00
{
unsigned int mask ;
int loop ;
do {
loop = 0 ;
2007-06-06 06:45:18 +01:00
mask = GEDR0 & GPIO_IRQ_mask [ 0 ] & ~ 3 ;
2005-04-16 15:20:36 -07:00
if ( mask ) {
GEDR0 = mask ;
irq = IRQ_GPIO ( 2 ) ;
desc = irq_desc + irq ;
mask > > = 2 ;
do {
if ( mask & 1 )
2006-10-06 10:53:39 -07:00
desc_handle_irq ( irq , desc ) ;
2005-04-16 15:20:36 -07:00
irq + + ;
desc + + ;
mask > > = 1 ;
} while ( mask ) ;
loop = 1 ;
}
2007-06-06 06:45:18 +01:00
mask = GEDR1 & GPIO_IRQ_mask [ 1 ] ;
2005-04-16 15:20:36 -07:00
if ( mask ) {
GEDR1 = mask ;
irq = IRQ_GPIO ( 32 ) ;
desc = irq_desc + irq ;
do {
if ( mask & 1 )
2006-10-06 10:53:39 -07:00
desc_handle_irq ( irq , desc ) ;
2005-04-16 15:20:36 -07:00
irq + + ;
desc + + ;
mask > > = 1 ;
} while ( mask ) ;
loop = 1 ;
}
2007-06-06 06:45:18 +01:00
mask = GEDR2 & GPIO_IRQ_mask [ 2 ] ;
2005-04-16 15:20:36 -07:00
if ( mask ) {
GEDR2 = mask ;
irq = IRQ_GPIO ( 64 ) ;
desc = irq_desc + irq ;
do {
if ( mask & 1 )
2006-10-06 10:53:39 -07:00
desc_handle_irq ( irq , desc ) ;
2005-04-16 15:20:36 -07:00
irq + + ;
desc + + ;
mask > > = 1 ;
} while ( mask ) ;
loop = 1 ;
}
2007-06-06 06:45:18 +01:00
mask = GEDR3 & GPIO_IRQ_mask [ 3 ] ;
2005-04-16 15:20:36 -07:00
if ( mask ) {
GEDR3 = mask ;
irq = IRQ_GPIO ( 96 ) ;
desc = irq_desc + irq ;
do {
if ( mask & 1 )
2006-10-06 10:53:39 -07:00
desc_handle_irq ( irq , desc ) ;
2005-04-16 15:20:36 -07:00
irq + + ;
desc + + ;
mask > > = 1 ;
} while ( mask ) ;
loop = 1 ;
}
} while ( loop ) ;
}
static void pxa_ack_muxed_gpio ( unsigned int irq )
{
int gpio = irq - IRQ_GPIO ( 2 ) + 2 ;
GEDR ( gpio ) = GPIO_bit ( gpio ) ;
}
static void pxa_mask_muxed_gpio ( unsigned int irq )
{
int gpio = irq - IRQ_GPIO ( 2 ) + 2 ;
__clear_bit ( gpio , GPIO_IRQ_mask ) ;
GRER ( gpio ) & = ~ GPIO_bit ( gpio ) ;
GFER ( gpio ) & = ~ GPIO_bit ( gpio ) ;
}
static void pxa_unmask_muxed_gpio ( unsigned int irq )
{
int gpio = irq - IRQ_GPIO ( 2 ) + 2 ;
int idx = gpio > > 5 ;
__set_bit ( gpio , GPIO_IRQ_mask ) ;
GRER ( gpio ) = GPIO_IRQ_rising_edge [ idx ] & GPIO_IRQ_mask [ idx ] ;
GFER ( gpio ) = GPIO_IRQ_falling_edge [ idx ] & GPIO_IRQ_mask [ idx ] ;
}
2006-08-01 22:26:25 +01:00
static struct irq_chip pxa_muxed_gpio_chip = {
. name = " GPIO " ,
2005-04-16 15:20:36 -07:00
. ack = pxa_ack_muxed_gpio ,
. mask = pxa_mask_muxed_gpio ,
. unmask = pxa_unmask_muxed_gpio ,
2005-09-04 19:43:13 +01:00
. set_type = pxa_gpio_irq_type ,
2007-02-26 01:44:57 +01:00
. set_wake = pxa_set_gpio_wake ,
2005-04-16 15:20:36 -07:00
} ;
2007-06-06 06:37:15 +01:00
void __init pxa_init_irq_gpio ( int gpio_nr )
2005-04-16 15:20:36 -07:00
{
2007-06-06 06:37:15 +01:00
int irq , i ;
2005-04-16 15:20:36 -07:00
2007-08-29 10:18:47 +01:00
pxa_last_gpio = gpio_nr - 1 ;
2005-04-16 15:20:36 -07:00
/* clear all GPIO edge detects */
2007-06-06 06:37:15 +01:00
for ( i = 0 ; i < gpio_nr ; i + = 32 ) {
GFER ( i ) = 0 ;
GRER ( i ) = 0 ;
GEDR ( i ) = GEDR ( i ) ;
}
2005-04-16 15:20:36 -07:00
/* GPIO 0 and 1 must have their mask bit always set */
GPIO_IRQ_mask [ 0 ] = 3 ;
for ( irq = IRQ_GPIO0 ; irq < = IRQ_GPIO1 ; irq + + ) {
set_irq_chip ( irq , & pxa_low_gpio_chip ) ;
2006-11-23 11:41:32 +00:00
set_irq_handler ( irq , handle_edge_irq ) ;
2005-04-16 15:20:36 -07:00
set_irq_flags ( irq , IRQF_VALID | IRQF_PROBE ) ;
}
2007-08-28 19:56:34 +01:00
for ( irq = IRQ_GPIO ( 2 ) ; irq < IRQ_GPIO ( gpio_nr ) ; irq + + ) {
2005-04-16 15:20:36 -07:00
set_irq_chip ( irq , & pxa_muxed_gpio_chip ) ;
2006-11-23 11:41:32 +00:00
set_irq_handler ( irq , handle_edge_irq ) ;
2005-04-16 15:20:36 -07:00
set_irq_flags ( irq , IRQF_VALID | IRQF_PROBE ) ;
}
/* Install handler for GPIO>=2 edge detect interrupts */
set_irq_chip ( IRQ_GPIO_2_x , & pxa_internal_chip_low ) ;
set_irq_chained_handler ( IRQ_GPIO_2_x , pxa_gpio_demux_handler ) ;
}