2007-07-18 17:25:09 +09:00
/*
* Shared interrupt handling code for IPR and INTC2 types of IRQs .
*
2008-04-24 21:36:34 +09:00
* Copyright ( C ) 2007 , 2008 Magnus Damm
2007-07-18 17:25:09 +09:00
*
* Based on intc2 . c and ipr . c
*
* Copyright ( C ) 1999 Niibe Yutaka & Takeshi Yaegashi
* Copyright ( C ) 2000 Kazumoto Kojima
* Copyright ( C ) 2001 David J . Mckay ( david . mckay @ st . com )
* Copyright ( C ) 2003 Takashi Kusuda < kusuda - takashi @ hitachi - ul . co . jp >
* Copyright ( C ) 2005 , 2006 Paul Mundt
*
* This file is subject to the terms and conditions of the GNU General Public
* License . See the file " COPYING " in the main directory of this archive
* for more details .
*/
# include <linux/init.h>
# include <linux/irq.h>
# include <linux/module.h>
# include <linux/io.h>
# include <linux/interrupt.h>
2007-08-12 15:26:12 +09:00
# include <linux/bootmem.h>
2008-10-01 16:13:54 +09:00
# include <linux/sh_intc.h>
2009-04-01 14:30:59 +00:00
# include <linux/sysdev.h>
# include <linux/list.h>
2007-08-12 15:26:12 +09:00
# define _INTC_MK(fn, mode, addr_e, addr_d, width, shift) \
( ( shift ) | ( ( width ) < < 5 ) | ( ( fn ) < < 9 ) | ( ( mode ) < < 13 ) | \
( ( addr_e ) < < 16 ) | ( ( addr_d < < 24 ) ) )
# define _INTC_SHIFT(h) (h & 0x1f)
# define _INTC_WIDTH(h) ((h >> 5) & 0xf)
# define _INTC_FN(h) ((h >> 9) & 0xf)
# define _INTC_MODE(h) ((h >> 13) & 0x7)
# define _INTC_ADDR_E(h) ((h >> 16) & 0xff)
# define _INTC_ADDR_D(h) ((h >> 24) & 0xff)
struct intc_handle_int {
unsigned int irq ;
unsigned long handle ;
} ;
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
struct intc_desc_int {
2009-04-01 14:30:59 +00:00
struct list_head list ;
struct sys_device sysdev ;
2009-04-06 07:17:04 +00:00
pm_message_t state ;
2007-08-12 15:26:12 +09:00
unsigned long * reg ;
2007-09-21 18:16:42 +09:00
# ifdef CONFIG_SMP
unsigned long * smp ;
# endif
2007-08-12 15:26:12 +09:00
unsigned int nr_reg ;
struct intc_handle_int * prio ;
unsigned int nr_prio ;
struct intc_handle_int * sense ;
unsigned int nr_sense ;
struct irq_chip chip ;
} ;
2007-07-18 17:25:09 +09:00
2009-04-01 14:30:59 +00:00
static LIST_HEAD ( intc_list ) ;
2007-09-21 18:16:42 +09:00
# ifdef CONFIG_SMP
# define IS_SMP(x) x.smp
# define INTC_REG(d, x, c) (d->reg[(x)] + ((d->smp[(x)] & 0xff) * c))
# define SMP_NR(d, x) ((d->smp[(x)] >> 8) ? (d->smp[(x)] >> 8) : 1)
# else
# define IS_SMP(x) 0
# define INTC_REG(d, x, c) (d->reg[(x)])
# define SMP_NR(d, x) 1
# endif
2007-08-12 15:26:12 +09:00
static unsigned int intc_prio_level [ NR_IRQS ] ; /* for now */
2008-07-04 12:37:12 +09:00
# if defined(CONFIG_CPU_SH3) || defined(CONFIG_CPU_SH4A)
2008-04-24 21:36:34 +09:00
static unsigned long ack_handle [ NR_IRQS ] ;
# endif
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
static inline struct intc_desc_int * get_intc_desc ( unsigned int irq )
2007-07-18 17:25:09 +09:00
{
struct irq_chip * chip = get_irq_chip ( irq ) ;
2007-08-12 15:26:12 +09:00
return ( void * ) ( ( char * ) chip - offsetof ( struct intc_desc_int , chip ) ) ;
2007-07-18 17:25:09 +09:00
}
static inline unsigned int set_field ( unsigned int value ,
unsigned int field_value ,
2007-08-12 15:26:12 +09:00
unsigned int handle )
2007-07-18 17:25:09 +09:00
{
2007-08-12 15:26:12 +09:00
unsigned int width = _INTC_WIDTH ( handle ) ;
unsigned int shift = _INTC_SHIFT ( handle ) ;
2007-07-18 17:25:09 +09:00
value & = ~ ( ( ( 1 < < width ) - 1 ) < < shift ) ;
value | = field_value < < shift ;
return value ;
}
2007-08-12 15:26:12 +09:00
static void write_8 ( unsigned long addr , unsigned long h , unsigned long data )
2007-07-18 17:25:09 +09:00
{
2008-10-01 15:19:10 +09:00
__raw_writeb ( set_field ( 0 , data , h ) , addr ) ;
2007-07-18 17:25:09 +09:00
}
2007-08-12 15:26:12 +09:00
static void write_16 ( unsigned long addr , unsigned long h , unsigned long data )
2007-07-18 17:25:09 +09:00
{
2008-10-01 15:19:10 +09:00
__raw_writew ( set_field ( 0 , data , h ) , addr ) ;
2007-07-18 17:25:09 +09:00
}
2007-08-12 15:26:12 +09:00
static void write_32 ( unsigned long addr , unsigned long h , unsigned long data )
2007-07-18 17:25:09 +09:00
{
2008-10-01 15:19:10 +09:00
__raw_writel ( set_field ( 0 , data , h ) , addr ) ;
2007-07-18 17:25:09 +09:00
}
2007-08-12 15:26:12 +09:00
static void modify_8 ( unsigned long addr , unsigned long h , unsigned long data )
2007-07-18 17:25:09 +09:00
{
2008-04-24 21:53:07 +09:00
unsigned long flags ;
local_irq_save ( flags ) ;
2008-10-01 15:19:10 +09:00
__raw_writeb ( set_field ( __raw_readb ( addr ) , data , h ) , addr ) ;
2008-04-24 21:53:07 +09:00
local_irq_restore ( flags ) ;
2007-07-18 17:25:09 +09:00
}
2007-08-12 15:26:12 +09:00
static void modify_16 ( unsigned long addr , unsigned long h , unsigned long data )
2007-07-18 17:25:09 +09:00
{
2008-04-24 21:53:07 +09:00
unsigned long flags ;
local_irq_save ( flags ) ;
2008-10-01 15:19:10 +09:00
__raw_writew ( set_field ( __raw_readw ( addr ) , data , h ) , addr ) ;
2008-04-24 21:53:07 +09:00
local_irq_restore ( flags ) ;
2007-07-18 17:25:09 +09:00
}
2007-08-12 15:26:12 +09:00
static void modify_32 ( unsigned long addr , unsigned long h , unsigned long data )
2007-07-18 17:25:09 +09:00
{
2008-04-24 21:53:07 +09:00
unsigned long flags ;
local_irq_save ( flags ) ;
2008-10-01 15:19:10 +09:00
__raw_writel ( set_field ( __raw_readl ( addr ) , data , h ) , addr ) ;
2008-04-24 21:53:07 +09:00
local_irq_restore ( flags ) ;
2007-07-18 17:25:09 +09:00
}
2007-08-12 15:26:12 +09:00
enum { REG_FN_ERR = 0 , REG_FN_WRITE_BASE = 1 , REG_FN_MODIFY_BASE = 5 } ;
static void ( * intc_reg_fns [ ] ) ( unsigned long addr ,
unsigned long h ,
unsigned long data ) = {
[ REG_FN_WRITE_BASE + 0 ] = write_8 ,
[ REG_FN_WRITE_BASE + 1 ] = write_16 ,
[ REG_FN_WRITE_BASE + 3 ] = write_32 ,
[ REG_FN_MODIFY_BASE + 0 ] = modify_8 ,
[ REG_FN_MODIFY_BASE + 1 ] = modify_16 ,
[ REG_FN_MODIFY_BASE + 3 ] = modify_32 ,
} ;
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
enum { MODE_ENABLE_REG = 0 , /* Bit(s) set -> interrupt enabled */
MODE_MASK_REG , /* Bit(s) set -> interrupt disabled */
MODE_DUAL_REG , /* Two registers, set bit to enable / disable */
MODE_PRIO_REG , /* Priority value written to enable interrupt */
MODE_PCLR_REG , /* Above plus all bits set to disable interrupt */
} ;
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
static void intc_mode_field ( unsigned long addr ,
unsigned long handle ,
void ( * fn ) ( unsigned long ,
unsigned long ,
unsigned long ) ,
unsigned int irq )
2007-07-18 17:25:09 +09:00
{
2007-08-12 15:26:12 +09:00
fn ( addr , handle , ( ( 1 < < _INTC_WIDTH ( handle ) ) - 1 ) ) ;
2007-07-18 17:25:09 +09:00
}
2007-08-12 15:26:12 +09:00
static void intc_mode_zero ( unsigned long addr ,
unsigned long handle ,
void ( * fn ) ( unsigned long ,
unsigned long ,
unsigned long ) ,
unsigned int irq )
2007-08-03 14:25:32 +09:00
{
2007-08-12 15:26:12 +09:00
fn ( addr , handle , 0 ) ;
2007-08-03 14:25:32 +09:00
}
2007-08-12 15:26:12 +09:00
static void intc_mode_prio ( unsigned long addr ,
unsigned long handle ,
void ( * fn ) ( unsigned long ,
unsigned long ,
unsigned long ) ,
unsigned int irq )
2007-08-03 14:25:32 +09:00
{
2007-08-12 15:26:12 +09:00
fn ( addr , handle , intc_prio_level [ irq ] ) ;
2007-08-03 14:25:32 +09:00
}
2007-08-12 15:26:12 +09:00
static void ( * intc_enable_fns [ ] ) ( unsigned long addr ,
unsigned long handle ,
void ( * fn ) ( unsigned long ,
unsigned long ,
unsigned long ) ,
unsigned int irq ) = {
[ MODE_ENABLE_REG ] = intc_mode_field ,
[ MODE_MASK_REG ] = intc_mode_zero ,
[ MODE_DUAL_REG ] = intc_mode_field ,
[ MODE_PRIO_REG ] = intc_mode_prio ,
[ MODE_PCLR_REG ] = intc_mode_prio ,
} ;
2007-08-03 14:25:32 +09:00
2007-08-12 15:26:12 +09:00
static void ( * intc_disable_fns [ ] ) ( unsigned long addr ,
unsigned long handle ,
void ( * fn ) ( unsigned long ,
unsigned long ,
unsigned long ) ,
unsigned int irq ) = {
[ MODE_ENABLE_REG ] = intc_mode_zero ,
[ MODE_MASK_REG ] = intc_mode_field ,
[ MODE_DUAL_REG ] = intc_mode_field ,
[ MODE_PRIO_REG ] = intc_mode_zero ,
[ MODE_PCLR_REG ] = intc_mode_field ,
} ;
2007-08-03 14:25:32 +09:00
2007-08-12 15:26:12 +09:00
static inline void _intc_enable ( unsigned int irq , unsigned long handle )
2007-08-03 14:25:32 +09:00
{
2007-08-12 15:26:12 +09:00
struct intc_desc_int * d = get_intc_desc ( irq ) ;
2007-09-21 18:16:42 +09:00
unsigned long addr ;
unsigned int cpu ;
2007-08-03 14:25:32 +09:00
2007-09-21 18:16:42 +09:00
for ( cpu = 0 ; cpu < SMP_NR ( d , _INTC_ADDR_E ( handle ) ) ; cpu + + ) {
addr = INTC_REG ( d , _INTC_ADDR_E ( handle ) , cpu ) ;
intc_enable_fns [ _INTC_MODE ( handle ) ] ( addr , handle , intc_reg_fns \
[ _INTC_FN ( handle ) ] , irq ) ;
}
2007-08-03 14:25:32 +09:00
}
2007-07-18 17:25:09 +09:00
static void intc_enable ( unsigned int irq )
{
2007-08-12 15:26:12 +09:00
_intc_enable ( irq , ( unsigned long ) get_irq_chip_data ( irq ) ) ;
2007-07-18 17:25:09 +09:00
}
static void intc_disable ( unsigned int irq )
{
2007-09-21 18:16:42 +09:00
struct intc_desc_int * d = get_intc_desc ( irq ) ;
2007-08-12 15:26:12 +09:00
unsigned long handle = ( unsigned long ) get_irq_chip_data ( irq ) ;
2007-09-21 18:16:42 +09:00
unsigned long addr ;
unsigned int cpu ;
2007-07-18 17:25:09 +09:00
2007-09-21 18:16:42 +09:00
for ( cpu = 0 ; cpu < SMP_NR ( d , _INTC_ADDR_D ( handle ) ) ; cpu + + ) {
addr = INTC_REG ( d , _INTC_ADDR_D ( handle ) , cpu ) ;
intc_disable_fns [ _INTC_MODE ( handle ) ] ( addr , handle , intc_reg_fns \
[ _INTC_FN ( handle ) ] , irq ) ;
}
2007-07-18 17:25:09 +09:00
}
2009-04-01 14:30:59 +00:00
static int intc_set_wake ( unsigned int irq , unsigned int on )
{
return 0 ; /* allow wakeup, but setup hardware in intc_suspend() */
}
2008-07-04 12:37:12 +09:00
# if defined(CONFIG_CPU_SH3) || defined(CONFIG_CPU_SH4A)
2008-04-24 21:36:34 +09:00
static void intc_mask_ack ( unsigned int irq )
{
struct intc_desc_int * d = get_intc_desc ( irq ) ;
unsigned long handle = ack_handle [ irq ] ;
unsigned long addr ;
intc_disable ( irq ) ;
/* read register and write zero only to the assocaited bit */
if ( handle ) {
addr = INTC_REG ( d , _INTC_ADDR_D ( handle ) , 0 ) ;
2008-07-04 12:37:12 +09:00
switch ( _INTC_FN ( handle ) ) {
case REG_FN_MODIFY_BASE + 0 : /* 8bit */
2008-10-01 15:19:10 +09:00
__raw_readb ( addr ) ;
__raw_writeb ( 0xff ^ set_field ( 0 , 1 , handle ) , addr ) ;
2008-07-04 12:37:12 +09:00
break ;
case REG_FN_MODIFY_BASE + 1 : /* 16bit */
2008-10-01 15:19:10 +09:00
__raw_readw ( addr ) ;
__raw_writew ( 0xffff ^ set_field ( 0 , 1 , handle ) , addr ) ;
2008-07-04 12:37:12 +09:00
break ;
case REG_FN_MODIFY_BASE + 3 : /* 32bit */
2008-10-01 15:19:10 +09:00
__raw_readl ( addr ) ;
__raw_writel ( 0xffffffff ^ set_field ( 0 , 1 , handle ) , addr ) ;
2008-07-04 12:37:12 +09:00
break ;
default :
BUG ( ) ;
break ;
}
2008-04-24 21:36:34 +09:00
}
}
# endif
2007-08-12 15:26:12 +09:00
static struct intc_handle_int * intc_find_irq ( struct intc_handle_int * hp ,
unsigned int nr_hp ,
unsigned int irq )
2007-07-18 17:25:09 +09:00
{
2007-08-12 15:26:12 +09:00
int i ;
2007-08-17 00:50:44 +09:00
/* this doesn't scale well, but...
*
* this function should only be used for cerain uncommon
* operations such as intc_set_priority ( ) and intc_set_sense ( )
* and in those rare cases performance doesn ' t matter that much .
* keeping the memory footprint low is more important .
*
* one rather simple way to speed this up and still keep the
* memory footprint down is to make sure the array is sorted
* and then perform a bisect to lookup the irq .
*/
2007-08-12 15:26:12 +09:00
for ( i = 0 ; i < nr_hp ; i + + ) {
if ( ( hp + i ) - > irq ! = irq )
continue ;
return hp + i ;
}
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
return NULL ;
2007-07-18 17:25:09 +09:00
}
2007-08-12 15:26:12 +09:00
int intc_set_priority ( unsigned int irq , unsigned int prio )
2007-07-18 17:25:09 +09:00
{
2007-08-12 15:26:12 +09:00
struct intc_desc_int * d = get_intc_desc ( irq ) ;
struct intc_handle_int * ihp ;
if ( ! intc_prio_level [ irq ] | | prio < = 1 )
return - EINVAL ;
ihp = intc_find_irq ( d - > prio , d - > nr_prio , irq ) ;
if ( ihp ) {
2007-08-17 00:50:44 +09:00
if ( prio > = ( 1 < < _INTC_WIDTH ( ihp - > handle ) ) )
2007-08-12 15:26:12 +09:00
return - EINVAL ;
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
intc_prio_level [ irq ] = prio ;
/*
* only set secondary masking method directly
* primary masking method is using intc_prio_level [ irq ]
* priority level will be set during next enable ( )
*/
2007-08-17 00:50:44 +09:00
if ( _INTC_FN ( ihp - > handle ) ! = REG_FN_ERR )
2007-08-12 15:26:12 +09:00
_intc_enable ( irq , ihp - > handle ) ;
}
return 0 ;
2007-07-18 17:25:09 +09:00
}
# define VALID(x) (x | 0x80)
static unsigned char intc_irq_sense_table [ IRQ_TYPE_SENSE_MASK + 1 ] = {
[ IRQ_TYPE_EDGE_FALLING ] = VALID ( 0 ) ,
[ IRQ_TYPE_EDGE_RISING ] = VALID ( 1 ) ,
[ IRQ_TYPE_LEVEL_LOW ] = VALID ( 2 ) ,
2008-04-24 21:47:15 +09:00
/* SH7706, SH7707 and SH7709 do not support high level triggered */
# if !defined(CONFIG_CPU_SUBTYPE_SH7706) && \
! defined ( CONFIG_CPU_SUBTYPE_SH7707 ) & & \
! defined ( CONFIG_CPU_SUBTYPE_SH7709 )
2007-07-18 17:25:09 +09:00
[ IRQ_TYPE_LEVEL_HIGH ] = VALID ( 3 ) ,
2008-04-24 21:47:15 +09:00
# endif
2007-07-18 17:25:09 +09:00
} ;
static int intc_set_sense ( unsigned int irq , unsigned int type )
{
2007-08-12 15:26:12 +09:00
struct intc_desc_int * d = get_intc_desc ( irq ) ;
2007-07-18 17:25:09 +09:00
unsigned char value = intc_irq_sense_table [ type & IRQ_TYPE_SENSE_MASK ] ;
2007-08-12 15:26:12 +09:00
struct intc_handle_int * ihp ;
unsigned long addr ;
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
if ( ! value )
2007-07-18 17:25:09 +09:00
return - EINVAL ;
2007-08-12 15:26:12 +09:00
ihp = intc_find_irq ( d - > sense , d - > nr_sense , irq ) ;
if ( ihp ) {
2007-09-21 18:16:42 +09:00
addr = INTC_REG ( d , _INTC_ADDR_E ( ihp - > handle ) , 0 ) ;
2007-08-12 15:26:12 +09:00
intc_reg_fns [ _INTC_FN ( ihp - > handle ) ] ( addr , ihp - > handle , value ) ;
2007-07-18 17:25:09 +09:00
}
2007-08-12 15:26:12 +09:00
return 0 ;
2007-07-18 17:25:09 +09:00
}
2007-08-12 15:26:12 +09:00
static unsigned int __init intc_get_reg ( struct intc_desc_int * d ,
unsigned long address )
2007-07-18 17:25:09 +09:00
{
2007-08-12 15:26:12 +09:00
unsigned int k ;
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
for ( k = 0 ; k < d - > nr_reg ; k + + ) {
if ( d - > reg [ k ] = = address )
return k ;
2007-08-03 14:25:32 +09:00
}
BUG ( ) ;
2007-08-12 15:26:12 +09:00
return 0 ;
2007-08-03 14:25:32 +09:00
}
2007-08-12 15:26:12 +09:00
static intc_enum __init intc_grp_id ( struct intc_desc * desc ,
intc_enum enum_id )
2007-07-20 12:09:29 +09:00
{
struct intc_group * g = desc - > groups ;
unsigned int i , j ;
for ( i = 0 ; g & & enum_id & & i < desc - > nr_groups ; i + + ) {
g = desc - > groups + i ;
for ( j = 0 ; g - > enum_ids [ j ] ; j + + ) {
if ( g - > enum_ids [ j ] ! = enum_id )
continue ;
return g - > enum_id ;
}
}
return 0 ;
}
2007-07-18 17:25:09 +09:00
static unsigned int __init intc_mask_data ( struct intc_desc * desc ,
2007-08-12 15:26:12 +09:00
struct intc_desc_int * d ,
2007-07-20 12:09:29 +09:00
intc_enum enum_id , int do_grps )
2007-07-18 17:25:09 +09:00
{
2007-07-20 12:09:29 +09:00
struct intc_mask_reg * mr = desc - > mask_regs ;
2007-08-12 15:26:12 +09:00
unsigned int i , j , fn , mode ;
unsigned long reg_e , reg_d ;
2007-07-18 17:25:09 +09:00
2007-07-20 12:09:29 +09:00
for ( i = 0 ; mr & & enum_id & & i < desc - > nr_mask_regs ; i + + ) {
mr = desc - > mask_regs + i ;
2007-07-18 17:25:09 +09:00
for ( j = 0 ; j < ARRAY_SIZE ( mr - > enum_ids ) ; j + + ) {
if ( mr - > enum_ids [ j ] ! = enum_id )
continue ;
2007-08-12 15:26:12 +09:00
if ( mr - > set_reg & & mr - > clr_reg ) {
fn = REG_FN_WRITE_BASE ;
mode = MODE_DUAL_REG ;
reg_e = mr - > clr_reg ;
reg_d = mr - > set_reg ;
} else {
fn = REG_FN_MODIFY_BASE ;
if ( mr - > set_reg ) {
mode = MODE_ENABLE_REG ;
reg_e = mr - > set_reg ;
reg_d = mr - > set_reg ;
} else {
mode = MODE_MASK_REG ;
reg_e = mr - > clr_reg ;
reg_d = mr - > clr_reg ;
}
2007-08-03 14:25:32 +09:00
}
2007-08-12 15:26:12 +09:00
fn + = ( mr - > reg_width > > 3 ) - 1 ;
return _INTC_MK ( fn , mode ,
intc_get_reg ( d , reg_e ) ,
intc_get_reg ( d , reg_d ) ,
1 ,
( mr - > reg_width - 1 ) - j ) ;
2007-07-18 17:25:09 +09:00
}
}
2007-07-20 12:09:29 +09:00
if ( do_grps )
2007-08-12 15:26:12 +09:00
return intc_mask_data ( desc , d , intc_grp_id ( desc , enum_id ) , 0 ) ;
2007-07-20 12:09:29 +09:00
2007-07-18 17:25:09 +09:00
return 0 ;
}
static unsigned int __init intc_prio_data ( struct intc_desc * desc ,
2007-08-12 15:26:12 +09:00
struct intc_desc_int * d ,
2007-07-20 12:09:29 +09:00
intc_enum enum_id , int do_grps )
2007-07-18 17:25:09 +09:00
{
2007-07-20 12:09:29 +09:00
struct intc_prio_reg * pr = desc - > prio_regs ;
2007-08-12 15:26:12 +09:00
unsigned int i , j , fn , mode , bit ;
unsigned long reg_e , reg_d ;
2007-07-18 17:25:09 +09:00
2007-07-20 12:09:29 +09:00
for ( i = 0 ; pr & & enum_id & & i < desc - > nr_prio_regs ; i + + ) {
pr = desc - > prio_regs + i ;
2007-07-18 17:25:09 +09:00
for ( j = 0 ; j < ARRAY_SIZE ( pr - > enum_ids ) ; j + + ) {
if ( pr - > enum_ids [ j ] ! = enum_id )
continue ;
2007-08-12 15:26:12 +09:00
if ( pr - > set_reg & & pr - > clr_reg ) {
fn = REG_FN_WRITE_BASE ;
mode = MODE_PCLR_REG ;
reg_e = pr - > set_reg ;
reg_d = pr - > clr_reg ;
} else {
fn = REG_FN_MODIFY_BASE ;
mode = MODE_PRIO_REG ;
if ( ! pr - > set_reg )
BUG ( ) ;
reg_e = pr - > set_reg ;
reg_d = pr - > set_reg ;
}
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
fn + = ( pr - > reg_width > > 3 ) - 1 ;
2007-07-18 17:25:09 +09:00
2008-09-09 23:02:43 +02:00
BUG_ON ( ( j + 1 ) * pr - > field_width > pr - > reg_width ) ;
bit = pr - > reg_width - ( ( j + 1 ) * pr - > field_width ) ;
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
return _INTC_MK ( fn , mode ,
intc_get_reg ( d , reg_e ) ,
intc_get_reg ( d , reg_d ) ,
pr - > field_width , bit ) ;
2007-07-18 17:25:09 +09:00
}
}
2007-07-20 12:09:29 +09:00
if ( do_grps )
2007-08-12 15:26:12 +09:00
return intc_prio_data ( desc , d , intc_grp_id ( desc , enum_id ) , 0 ) ;
return 0 ;
}
2008-07-04 12:37:12 +09:00
# if defined(CONFIG_CPU_SH3) || defined(CONFIG_CPU_SH4A)
2008-04-24 21:36:34 +09:00
static unsigned int __init intc_ack_data ( struct intc_desc * desc ,
struct intc_desc_int * d ,
intc_enum enum_id )
{
struct intc_mask_reg * mr = desc - > ack_regs ;
unsigned int i , j , fn , mode ;
unsigned long reg_e , reg_d ;
for ( i = 0 ; mr & & enum_id & & i < desc - > nr_ack_regs ; i + + ) {
mr = desc - > ack_regs + i ;
for ( j = 0 ; j < ARRAY_SIZE ( mr - > enum_ids ) ; j + + ) {
if ( mr - > enum_ids [ j ] ! = enum_id )
continue ;
fn = REG_FN_MODIFY_BASE ;
mode = MODE_ENABLE_REG ;
reg_e = mr - > set_reg ;
reg_d = mr - > set_reg ;
fn + = ( mr - > reg_width > > 3 ) - 1 ;
return _INTC_MK ( fn , mode ,
intc_get_reg ( d , reg_e ) ,
intc_get_reg ( d , reg_d ) ,
1 ,
( mr - > reg_width - 1 ) - j ) ;
}
}
return 0 ;
}
# endif
2007-08-12 15:26:12 +09:00
static unsigned int __init intc_sense_data ( struct intc_desc * desc ,
struct intc_desc_int * d ,
intc_enum enum_id )
{
struct intc_sense_reg * sr = desc - > sense_regs ;
unsigned int i , j , fn , bit ;
for ( i = 0 ; sr & & enum_id & & i < desc - > nr_sense_regs ; i + + ) {
sr = desc - > sense_regs + i ;
for ( j = 0 ; j < ARRAY_SIZE ( sr - > enum_ids ) ; j + + ) {
if ( sr - > enum_ids [ j ] ! = enum_id )
continue ;
fn = REG_FN_MODIFY_BASE ;
fn + = ( sr - > reg_width > > 3 ) - 1 ;
2008-09-09 23:02:43 +02:00
BUG_ON ( ( j + 1 ) * sr - > field_width > sr - > reg_width ) ;
bit = sr - > reg_width - ( ( j + 1 ) * sr - > field_width ) ;
2007-08-12 15:26:12 +09:00
return _INTC_MK ( fn , 0 , intc_get_reg ( d , sr - > reg ) ,
0 , sr - > field_width , bit ) ;
}
}
2007-07-20 12:09:29 +09:00
2007-07-18 17:25:09 +09:00
return 0 ;
}
2007-08-12 15:26:12 +09:00
static void __init intc_register_irq ( struct intc_desc * desc ,
struct intc_desc_int * d ,
intc_enum enum_id ,
2007-07-18 17:25:09 +09:00
unsigned int irq )
{
2007-08-17 00:50:44 +09:00
struct intc_handle_int * hp ;
2007-07-20 12:09:29 +09:00
unsigned int data [ 2 ] , primary ;
/* Prefer single interrupt source bitmap over other combinations:
* 1. bitmap , single interrupt source
* 2. priority , single interrupt source
* 3. bitmap , multiple interrupt sources ( groups )
* 4. priority , multiple interrupt sources ( groups )
*/
2007-07-18 17:25:09 +09:00
2007-08-12 15:26:12 +09:00
data [ 0 ] = intc_mask_data ( desc , d , enum_id , 0 ) ;
data [ 1 ] = intc_prio_data ( desc , d , enum_id , 0 ) ;
2007-07-20 12:09:29 +09:00
primary = 0 ;
if ( ! data [ 0 ] & & data [ 1 ] )
primary = 1 ;
2009-02-24 22:58:57 +09:00
if ( ! data [ 0 ] & & ! data [ 1 ] )
2009-03-06 17:56:58 +09:00
pr_warning ( " intc: missing unique irq mask for "
" irq %d (vect 0x%04x) \n " , irq , irq2evt ( irq ) ) ;
2009-02-24 22:58:57 +09:00
2007-08-12 15:26:12 +09:00
data [ 0 ] = data [ 0 ] ? data [ 0 ] : intc_mask_data ( desc , d , enum_id , 1 ) ;
data [ 1 ] = data [ 1 ] ? data [ 1 ] : intc_prio_data ( desc , d , enum_id , 1 ) ;
2007-07-20 12:09:29 +09:00
if ( ! data [ primary ] )
primary ^ = 1 ;
BUG_ON ( ! data [ primary ] ) ; /* must have primary masking method */
2007-07-18 17:25:09 +09:00
disable_irq_nosync ( irq ) ;
2007-08-12 15:26:12 +09:00
set_irq_chip_and_handler_name ( irq , & d - > chip ,
2007-07-18 17:25:09 +09:00
handle_level_irq , " level " ) ;
2007-07-20 12:09:29 +09:00
set_irq_chip_data ( irq , ( void * ) data [ primary ] ) ;
2007-07-18 17:25:09 +09:00
2008-01-10 14:08:55 +09:00
/* set priority level
* - this needs to be at least 2 for 5 - bit priorities on 7780
*/
intc_prio_level [ irq ] = 2 ;
2007-08-12 15:26:12 +09:00
2007-07-20 12:09:29 +09:00
/* enable secondary masking method if present */
if ( data [ ! primary ] )
2007-08-12 15:26:12 +09:00
_intc_enable ( irq , data [ ! primary ] ) ;
/* add irq to d->prio list if priority is available */
if ( data [ 1 ] ) {
2007-08-17 00:50:44 +09:00
hp = d - > prio + d - > nr_prio ;
hp - > irq = irq ;
hp - > handle = data [ 1 ] ;
if ( primary ) {
/*
* only secondary priority should access registers , so
* set _INTC_FN ( h ) = REG_FN_ERR for intc_set_priority ( )
*/
hp - > handle & = ~ _INTC_MK ( 0x0f , 0 , 0 , 0 , 0 , 0 ) ;
hp - > handle | = _INTC_MK ( REG_FN_ERR , 0 , 0 , 0 , 0 , 0 ) ;
}
2007-08-12 15:26:12 +09:00
d - > nr_prio + + ;
}
/* add irq to d->sense list if sense is available */
data [ 0 ] = intc_sense_data ( desc , d , enum_id ) ;
if ( data [ 0 ] ) {
( d - > sense + d - > nr_sense ) - > irq = irq ;
( d - > sense + d - > nr_sense ) - > handle = data [ 0 ] ;
d - > nr_sense + + ;
}
2007-07-18 17:25:09 +09:00
/* irq should be disabled by default */
2007-08-12 15:26:12 +09:00
d - > chip . mask ( irq ) ;
2008-04-24 21:36:34 +09:00
2008-07-04 12:37:12 +09:00
# if defined(CONFIG_CPU_SH3) || defined(CONFIG_CPU_SH4A)
2008-04-24 21:36:34 +09:00
if ( desc - > ack_regs )
ack_handle [ irq ] = intc_ack_data ( desc , d , enum_id ) ;
# endif
2007-07-18 17:25:09 +09:00
}
2007-09-21 18:16:42 +09:00
static unsigned int __init save_reg ( struct intc_desc_int * d ,
unsigned int cnt ,
unsigned long value ,
unsigned int smp )
{
if ( value ) {
d - > reg [ cnt ] = value ;
# ifdef CONFIG_SMP
d - > smp [ cnt ] = smp ;
# endif
return 1 ;
}
return 0 ;
}
2009-02-24 22:58:57 +09:00
static unsigned char * intc_evt2irq_table ;
unsigned int intc_evt2irq ( unsigned int vector )
{
unsigned int irq = evt2irq ( vector ) ;
if ( intc_evt2irq_table & & intc_evt2irq_table [ irq ] )
irq = intc_evt2irq_table [ irq ] ;
return irq ;
}
2007-09-21 18:16:42 +09:00
2007-07-18 17:25:09 +09:00
void __init register_intc_controller ( struct intc_desc * desc )
{
2009-05-22 01:28:33 +09:00
unsigned int i , k , smp , cpu = smp_processor_id ( ) ;
2007-08-12 15:26:12 +09:00
struct intc_desc_int * d ;
d = alloc_bootmem ( sizeof ( * d ) ) ;
2009-04-01 14:30:59 +00:00
INIT_LIST_HEAD ( & d - > list ) ;
list_add ( & d - > list , & intc_list ) ;
2007-08-12 15:26:12 +09:00
d - > nr_reg = desc - > mask_regs ? desc - > nr_mask_regs * 2 : 0 ;
d - > nr_reg + = desc - > prio_regs ? desc - > nr_prio_regs * 2 : 0 ;
d - > nr_reg + = desc - > sense_regs ? desc - > nr_sense_regs : 0 ;
2008-07-04 12:37:12 +09:00
# if defined(CONFIG_CPU_SH3) || defined(CONFIG_CPU_SH4A)
2008-04-24 21:36:34 +09:00
d - > nr_reg + = desc - > ack_regs ? desc - > nr_ack_regs : 0 ;
# endif
2007-08-12 15:26:12 +09:00
d - > reg = alloc_bootmem ( d - > nr_reg * sizeof ( * d - > reg ) ) ;
2007-09-21 18:16:42 +09:00
# ifdef CONFIG_SMP
d - > smp = alloc_bootmem ( d - > nr_reg * sizeof ( * d - > smp ) ) ;
# endif
2007-08-12 15:26:12 +09:00
k = 0 ;
if ( desc - > mask_regs ) {
for ( i = 0 ; i < desc - > nr_mask_regs ; i + + ) {
2007-09-21 18:16:42 +09:00
smp = IS_SMP ( desc - > mask_regs [ i ] ) ;
k + = save_reg ( d , k , desc - > mask_regs [ i ] . set_reg , smp ) ;
k + = save_reg ( d , k , desc - > mask_regs [ i ] . clr_reg , smp ) ;
2007-08-12 15:26:12 +09:00
}
}
if ( desc - > prio_regs ) {
d - > prio = alloc_bootmem ( desc - > nr_vectors * sizeof ( * d - > prio ) ) ;
for ( i = 0 ; i < desc - > nr_prio_regs ; i + + ) {
2007-09-21 18:16:42 +09:00
smp = IS_SMP ( desc - > prio_regs [ i ] ) ;
k + = save_reg ( d , k , desc - > prio_regs [ i ] . set_reg , smp ) ;
k + = save_reg ( d , k , desc - > prio_regs [ i ] . clr_reg , smp ) ;
2007-08-12 15:26:12 +09:00
}
}
if ( desc - > sense_regs ) {
d - > sense = alloc_bootmem ( desc - > nr_vectors * sizeof ( * d - > sense ) ) ;
for ( i = 0 ; i < desc - > nr_sense_regs ; i + + ) {
2007-09-21 18:16:42 +09:00
k + = save_reg ( d , k , desc - > sense_regs [ i ] . reg , 0 ) ;
2007-08-12 15:26:12 +09:00
}
}
d - > chip . name = desc - > name ;
d - > chip . mask = intc_disable ;
d - > chip . unmask = intc_enable ;
d - > chip . mask_ack = intc_disable ;
2009-04-01 14:20:58 +00:00
d - > chip . enable = intc_enable ;
d - > chip . disable = intc_disable ;
d - > chip . shutdown = intc_disable ;
2007-08-12 15:26:12 +09:00
d - > chip . set_type = intc_set_sense ;
2009-04-01 14:30:59 +00:00
d - > chip . set_wake = intc_set_wake ;
2007-07-18 17:25:09 +09:00
2008-07-04 12:37:12 +09:00
# if defined(CONFIG_CPU_SH3) || defined(CONFIG_CPU_SH4A)
2008-04-24 21:36:34 +09:00
if ( desc - > ack_regs ) {
for ( i = 0 ; i < desc - > nr_ack_regs ; i + + )
k + = save_reg ( d , k , desc - > ack_regs [ i ] . set_reg , 0 ) ;
d - > chip . mask_ack = intc_mask_ack ;
}
# endif
BUG_ON ( k > 256 ) ; /* _INTC_ADDR_E() and _INTC_ADDR_D() are 8 bits */
2009-02-24 22:58:57 +09:00
/* keep the first vector only if same enum is used multiple times */
for ( i = 0 ; i < desc - > nr_vectors ; i + + ) {
struct intc_vect * vect = desc - > vectors + i ;
int first_irq = evt2irq ( vect - > vect ) ;
if ( ! vect - > enum_id )
continue ;
for ( k = i + 1 ; k < desc - > nr_vectors ; k + + ) {
struct intc_vect * vect2 = desc - > vectors + k ;
if ( vect - > enum_id ! = vect2 - > enum_id )
continue ;
vect2 - > enum_id = 0 ;
if ( ! intc_evt2irq_table )
intc_evt2irq_table = alloc_bootmem ( NR_IRQS ) ;
if ( ! intc_evt2irq_table ) {
pr_warning ( " intc: cannot allocate evt2irq! \n " ) ;
continue ;
}
intc_evt2irq_table [ evt2irq ( vect2 - > vect ) ] = first_irq ;
}
}
/* register the vectors one by one */
2007-07-18 17:25:09 +09:00
for ( i = 0 ; i < desc - > nr_vectors ; i + + ) {
struct intc_vect * vect = desc - > vectors + i ;
2009-05-22 01:28:33 +09:00
unsigned int irq = evt2irq ( vect - > vect ) ;
2009-05-22 13:47:52 +09:00
# ifdef CONFIG_SPARSE_IRQ
2009-05-22 01:28:33 +09:00
struct irq_desc * irq_desc ;
2009-05-22 13:47:52 +09:00
# endif
2009-02-24 22:58:57 +09:00
if ( ! vect - > enum_id )
continue ;
2009-05-22 13:47:52 +09:00
# ifdef CONFIG_SPARSE_IRQ
2009-05-22 01:28:33 +09:00
irq_desc = irq_to_desc_alloc_cpu ( irq , cpu ) ;
if ( unlikely ( ! irq_desc ) ) {
printk ( KERN_INFO " can not get irq_desc for %d \n " , irq ) ;
continue ;
}
2009-05-22 13:47:52 +09:00
# endif
2009-05-22 01:28:33 +09:00
intc_register_irq ( desc , d , vect - > enum_id , irq ) ;
2007-07-18 17:25:09 +09:00
}
}
2009-04-01 14:30:59 +00:00
static int intc_suspend ( struct sys_device * dev , pm_message_t state )
{
struct intc_desc_int * d ;
struct irq_desc * desc ;
int irq ;
/* get intc controller associated with this sysdev */
d = container_of ( dev , struct intc_desc_int , sysdev ) ;
2009-04-06 07:17:04 +00:00
switch ( state . event ) {
case PM_EVENT_ON :
if ( d - > state . event ! = PM_EVENT_FREEZE )
break ;
for_each_irq_desc ( irq , desc ) {
if ( desc - > chip ! = & d - > chip )
continue ;
if ( desc - > status & IRQ_DISABLED )
intc_disable ( irq ) ;
else
intc_enable ( irq ) ;
}
break ;
case PM_EVENT_FREEZE :
/* nothing has to be done */
break ;
case PM_EVENT_SUSPEND :
/* enable wakeup irqs belonging to this intc controller */
for_each_irq_desc ( irq , desc ) {
if ( ( desc - > status & IRQ_WAKEUP ) & & ( desc - > chip = = & d - > chip ) )
intc_enable ( irq ) ;
}
break ;
2009-04-01 14:30:59 +00:00
}
2009-04-06 07:17:04 +00:00
d - > state = state ;
2009-04-01 14:30:59 +00:00
return 0 ;
}
2009-04-06 07:17:04 +00:00
static int intc_resume ( struct sys_device * dev )
{
return intc_suspend ( dev , PMSG_ON ) ;
}
2009-04-01 14:30:59 +00:00
static struct sysdev_class intc_sysdev_class = {
. name = " intc " ,
. suspend = intc_suspend ,
2009-04-06 07:17:04 +00:00
. resume = intc_resume ,
2009-04-01 14:30:59 +00:00
} ;
/* register this intc as sysdev to allow suspend/resume */
static int __init register_intc_sysdevs ( void )
{
struct intc_desc_int * d ;
int error ;
int id = 0 ;
error = sysdev_class_register ( & intc_sysdev_class ) ;
if ( ! error ) {
list_for_each_entry ( d , & intc_list , list ) {
d - > sysdev . id = id ;
d - > sysdev . cls = & intc_sysdev_class ;
error = sysdev_register ( & d - > sysdev ) ;
if ( error )
break ;
id + + ;
}
}
if ( error )
pr_warning ( " intc: sysdev registration error \n " ) ;
return error ;
}
device_initcall ( register_intc_sysdevs ) ;