2009-03-11 19:49:05 +04:00
/*
* TI Common Platform Interrupt Controller ( cp_intc ) driver
*
* Author : Steve Chen < schen @ mvista . com >
* Copyright ( C ) 2008 - 2009 , MontaVista Software , Inc . < source @ mvista . com >
*
* This file is licensed under the terms of the GNU General Public License
* version 2. This program is licensed " as is " without any warranty of any
* kind , whether express or implied .
*/
# include <linux/init.h>
# include <linux/irq.h>
# include <linux/io.h>
2010-05-07 17:06:37 -04:00
# include <mach/common.h>
2009-03-11 19:49:05 +04:00
# include <mach/cp_intc.h>
static inline unsigned int cp_intc_read ( unsigned offset )
{
2010-05-07 17:06:37 -04:00
return __raw_readl ( davinci_intc_base + offset ) ;
2009-03-11 19:49:05 +04:00
}
static inline void cp_intc_write ( unsigned long value , unsigned offset )
{
2010-05-07 17:06:37 -04:00
__raw_writel ( value , davinci_intc_base + offset ) ;
2009-03-11 19:49:05 +04:00
}
static void cp_intc_ack_irq ( unsigned int irq )
{
cp_intc_write ( irq , CP_INTC_SYS_STAT_IDX_CLR ) ;
}
/* Disable interrupt */
static void cp_intc_mask_irq ( unsigned int irq )
{
/* XXX don't know why we need to disable nIRQ here... */
cp_intc_write ( 1 , CP_INTC_HOST_ENABLE_IDX_CLR ) ;
cp_intc_write ( irq , CP_INTC_SYS_ENABLE_IDX_CLR ) ;
cp_intc_write ( 1 , CP_INTC_HOST_ENABLE_IDX_SET ) ;
}
/* Enable interrupt */
static void cp_intc_unmask_irq ( unsigned int irq )
{
cp_intc_write ( irq , CP_INTC_SYS_ENABLE_IDX_SET ) ;
}
static int cp_intc_set_irq_type ( unsigned int irq , unsigned int flow_type )
{
unsigned reg = BIT_WORD ( irq ) ;
unsigned mask = BIT_MASK ( irq ) ;
unsigned polarity = cp_intc_read ( CP_INTC_SYS_POLARITY ( reg ) ) ;
unsigned type = cp_intc_read ( CP_INTC_SYS_TYPE ( reg ) ) ;
switch ( flow_type ) {
case IRQ_TYPE_EDGE_RISING :
polarity | = mask ;
type | = mask ;
break ;
case IRQ_TYPE_EDGE_FALLING :
polarity & = ~ mask ;
type | = mask ;
break ;
case IRQ_TYPE_LEVEL_HIGH :
polarity | = mask ;
type & = ~ mask ;
break ;
case IRQ_TYPE_LEVEL_LOW :
polarity & = ~ mask ;
type & = ~ mask ;
break ;
default :
return - EINVAL ;
}
cp_intc_write ( polarity , CP_INTC_SYS_POLARITY ( reg ) ) ;
cp_intc_write ( type , CP_INTC_SYS_TYPE ( reg ) ) ;
return 0 ;
}
2009-11-16 17:21:30 +05:30
/*
* Faking this allows us to to work with suspend functions of
* generic drivers which call { enable | disable } _irq_wake for
* wake up interrupt sources ( eg RTC on DA850 ) .
*/
static int cp_intc_set_wake ( unsigned int irq , unsigned int on )
{
return 0 ;
}
2009-03-11 19:49:05 +04:00
static struct irq_chip cp_intc_irq_chip = {
. name = " cp_intc " ,
. ack = cp_intc_ack_irq ,
. mask = cp_intc_mask_irq ,
. unmask = cp_intc_unmask_irq ,
. set_type = cp_intc_set_irq_type ,
2009-11-16 17:21:30 +05:30
. set_wake = cp_intc_set_wake ,
2009-03-11 19:49:05 +04:00
} ;
2010-05-07 17:06:37 -04:00
void __init cp_intc_init ( void )
2009-03-11 19:49:05 +04:00
{
2010-05-07 17:06:37 -04:00
unsigned long num_irq = davinci_soc_info . intc_irq_num ;
u8 * irq_prio = davinci_soc_info . intc_irq_prios ;
u32 * host_map = davinci_soc_info . intc_host_map ;
2009-03-11 19:49:05 +04:00
unsigned num_reg = BITS_TO_LONGS ( num_irq ) ;
int i ;
2010-05-07 17:06:37 -04:00
davinci_intc_type = DAVINCI_INTC_TYPE_CP_INTC ;
davinci_intc_base = ioremap ( davinci_soc_info . intc_base , SZ_8K ) ;
if ( WARN_ON ( ! davinci_intc_base ) )
return ;
2009-03-11 19:49:05 +04:00
cp_intc_write ( 0 , CP_INTC_GLOBAL_ENABLE ) ;
/* Disable all host interrupts */
cp_intc_write ( 0 , CP_INTC_HOST_ENABLE ( 0 ) ) ;
/* Disable system interrupts */
for ( i = 0 ; i < num_reg ; i + + )
cp_intc_write ( ~ 0 , CP_INTC_SYS_ENABLE_CLR ( i ) ) ;
/* Set to normal mode, no nesting, no priority hold */
cp_intc_write ( 0 , CP_INTC_CTRL ) ;
cp_intc_write ( 0 , CP_INTC_HOST_CTRL ) ;
/* Clear system interrupt status */
for ( i = 0 ; i < num_reg ; i + + )
cp_intc_write ( ~ 0 , CP_INTC_SYS_STAT_CLR ( i ) ) ;
/* Enable nIRQ (what about nFIQ?) */
cp_intc_write ( 1 , CP_INTC_HOST_ENABLE_IDX_SET ) ;
/*
* Priority is determined by host channel : lower channel number has
* higher priority i . e . channel 0 has highest priority and channel 31
* had the lowest priority .
*/
num_reg = ( num_irq + 3 ) > > 2 ; /* 4 channels per register */
if ( irq_prio ) {
unsigned j , k ;
u32 val ;
for ( k = i = 0 ; i < num_reg ; i + + ) {
for ( val = j = 0 ; j < 4 ; j + + , k + + ) {
val > > = 8 ;
if ( k < num_irq )
val | = irq_prio [ k ] < < 24 ;
}
cp_intc_write ( val , CP_INTC_CHAN_MAP ( i ) ) ;
}
} else {
/*
* Default everything to channel 15 if priority not specified .
* Note that channel 0 - 1 are mapped to nFIQ and channels 2 - 31
* are mapped to nIRQ .
*/
for ( i = 0 ; i < num_reg ; i + + )
cp_intc_write ( 0x0f0f0f0f , CP_INTC_CHAN_MAP ( i ) ) ;
}
2010-03-25 17:43:46 -04:00
if ( host_map )
for ( i = 0 ; host_map [ i ] ! = - 1 ; i + + )
cp_intc_write ( host_map [ i ] , CP_INTC_HOST_MAP ( i ) ) ;
2009-03-11 19:49:05 +04:00
/* Set up genirq dispatching for cp_intc */
for ( i = 0 ; i < num_irq ; i + + ) {
set_irq_chip ( i , & cp_intc_irq_chip ) ;
set_irq_flags ( i , IRQF_VALID | IRQF_PROBE ) ;
set_irq_handler ( i , handle_edge_irq ) ;
}
/* Enable global interrupt */
cp_intc_write ( 1 , CP_INTC_GLOBAL_ENABLE ) ;
}