2007-05-06 14:50:35 -07:00
/*
* linux / arch / h8300 / kernel / irq . c
*
* Copyright 2007 Yoshinori Sato < ysato @ users . sourceforge . jp >
*/
# include <linux/module.h>
# include <linux/types.h>
# include <linux/kernel.h>
# include <linux/sched.h>
# include <linux/kernel_stat.h>
# include <linux/seq_file.h>
# include <linux/init.h>
# include <linux/random.h>
# include <linux/bootmem.h>
# include <linux/irq.h>
2008-02-06 01:36:35 -08:00
# include <linux/interrupt.h>
2007-05-06 14:50:35 -07:00
# include <asm/system.h>
# include <asm/traps.h>
# include <asm/io.h>
# include <asm/setup.h>
# include <asm/errno.h>
/*#define DEBUG*/
extern unsigned long * interrupt_redirect_table ;
extern const int h8300_saved_vectors [ ] ;
2008-02-23 15:23:59 -08:00
extern const h8300_vector h8300_trap_table [ ] ;
2007-05-06 14:50:35 -07:00
int h8300_enable_irq_pin ( unsigned int irq ) ;
void h8300_disable_irq_pin ( unsigned int irq ) ;
# define CPU_VECTOR ((unsigned long *)0x000000)
# define ADDR_MASK (0xffffff)
static inline int is_ext_irq ( unsigned int irq )
{
return ( irq > = EXT_IRQ0 & & irq < = ( EXT_IRQ0 + EXT_IRQS ) ) ;
}
2011-01-19 12:15:29 +01:00
static void h8300_enable_irq ( struct irq_data * data )
2007-05-06 14:50:35 -07:00
{
2011-01-19 12:15:29 +01:00
if ( is_ext_irq ( data - > irq ) )
IER_REGS | = 1 < < ( data - > irq - EXT_IRQ0 ) ;
2007-05-06 14:50:35 -07:00
}
2011-01-19 12:15:29 +01:00
static void h8300_disable_irq ( struct irq_data * data )
2007-05-06 14:50:35 -07:00
{
2011-01-19 12:15:29 +01:00
if ( is_ext_irq ( data - > irq ) )
IER_REGS & = ~ ( 1 < < ( data - > irq - EXT_IRQ0 ) ) ;
2007-05-06 14:50:35 -07:00
}
2011-01-19 12:15:29 +01:00
static unsigned int h8300_startup_irq ( struct irq_data * data )
2007-05-06 14:50:35 -07:00
{
2011-01-19 12:15:29 +01:00
if ( is_ext_irq ( data - > irq ) )
return h8300_enable_irq_pin ( data - > irq ) ;
2007-05-06 14:50:35 -07:00
else
return 0 ;
}
2011-01-19 12:15:29 +01:00
static void h8300_shutdown_irq ( struct irq_data * data )
2007-05-06 14:50:35 -07:00
{
2011-01-19 12:15:29 +01:00
if ( is_ext_irq ( data - > irq ) )
h8300_disable_irq_pin ( data - > irq ) ;
2007-05-06 14:50:35 -07:00
}
/*
2007-10-19 23:10:43 +02:00
* h8300 interrupt controller implementation
2007-05-06 14:50:35 -07:00
*/
struct irq_chip h8300irq_chip = {
. name = " H8300-INTC " ,
2011-01-19 12:15:29 +01:00
. irq_startup = h8300_startup_irq ,
. irq_shutdown = h8300_shutdown_irq ,
. irq_enable = h8300_enable_irq ,
. irq_disable = h8300_disable_irq ,
2007-05-06 14:50:35 -07:00
} ;
# if defined(CONFIG_RAMKERNEL)
static unsigned long __init * get_vector_address ( void )
{
unsigned long * rom_vector = CPU_VECTOR ;
unsigned long base , tmp ;
int vec_no ;
base = rom_vector [ EXT_IRQ0 ] & ADDR_MASK ;
/* check romvector format */
for ( vec_no = EXT_IRQ1 ; vec_no < = EXT_IRQ0 + EXT_IRQS ; vec_no + + ) {
if ( ( base + ( vec_no - EXT_IRQ0 ) * 4 ) ! = ( rom_vector [ vec_no ] & ADDR_MASK ) )
return NULL ;
}
/* ramvector base address */
base - = EXT_IRQ0 * 4 ;
/* writerble check */
tmp = ~ ( * ( volatile unsigned long * ) base ) ;
( * ( volatile unsigned long * ) base ) = tmp ;
if ( ( * ( volatile unsigned long * ) base ) ! = tmp )
return NULL ;
return ( unsigned long * ) base ;
}
static void __init setup_vector ( void )
{
int i ;
unsigned long * ramvec , * ramvec_p ;
2008-02-23 15:23:59 -08:00
const h8300_vector * trap_entry ;
2007-05-06 14:50:35 -07:00
const int * saved_vector ;
ramvec = get_vector_address ( ) ;
if ( ramvec = = NULL )
panic ( " interrupt vector serup failed. " ) ;
else
printk ( KERN_INFO " virtual vector at 0x%08lx \n " , ( unsigned long ) ramvec ) ;
/* create redirect table */
ramvec_p = ramvec ;
trap_entry = h8300_trap_table ;
saved_vector = h8300_saved_vectors ;
for ( i = 0 ; i < NR_IRQS ; i + + ) {
if ( i = = * saved_vector ) {
ramvec_p + + ;
saved_vector + + ;
} else {
if ( i < NR_TRAPS ) {
if ( * trap_entry )
* ramvec_p = VECTOR ( * trap_entry ) ;
ramvec_p + + ;
trap_entry + + ;
} else
* ramvec_p + + = REDIRECT ( interrupt_entry ) ;
}
}
interrupt_redirect_table = ramvec ;
# ifdef DEBUG
ramvec_p = ramvec ;
for ( i = 0 ; i < NR_IRQS ; i + + ) {
if ( ( i % 8 ) = = 0 )
printk ( KERN_DEBUG " \n %p: " , ramvec_p ) ;
printk ( KERN_DEBUG " %p " , * ramvec_p ) ;
ramvec_p + + ;
}
printk ( KERN_DEBUG " \n " ) ;
# endif
}
# else
# define setup_vector() do { } while(0)
# endif
void __init init_IRQ ( void )
{
int c ;
setup_vector ( ) ;
2011-01-19 12:18:57 +01:00
for ( c = 0 ; c < NR_IRQS ; c + + )
2011-03-24 17:32:22 +01:00
irq_set_chip_and_handler ( c , & h8300irq_chip , handle_simple_irq ) ;
2007-05-06 14:50:35 -07:00
}
asmlinkage void do_IRQ ( int irq )
{
irq_enter ( ) ;
2011-01-19 12:18:57 +01:00
generic_handle_irq ( irq ) ;
2007-05-06 14:50:35 -07:00
irq_exit ( ) ;
}