2005-04-16 15:20:36 -07:00
/*
2012-07-20 11:15:04 +02:00
* Copyright IBM Corp . 2004 , 2011
2011-05-26 09:48:24 +02:00
* Author ( s ) : Martin Schwidefsky < schwidefsky @ de . ibm . com > ,
* Holger Smolinski < Holger . Smolinski @ de . ibm . com > ,
* Thomas Spatzier < tspat @ de . ibm . com > ,
2005-04-16 15:20:36 -07:00
*
* This file contains interrupt related functions .
*/
# include <linux/kernel_stat.h>
# include <linux/interrupt.h>
# include <linux/seq_file.h>
2007-02-05 21:16:44 +01:00
# include <linux/proc_fs.h>
# include <linux/profile.h>
2011-05-26 09:48:24 +02:00
# include <linux/module.h>
# include <linux/kernel.h>
# include <linux/ftrace.h>
# include <linux/errno.h>
# include <linux/slab.h>
# include <linux/cpu.h>
# include <asm/irq_regs.h>
# include <asm/cputime.h>
# include <asm/lowcore.h>
# include <asm/irq.h>
2013-06-27 09:01:09 +02:00
# include <asm/hw_irq.h>
2011-05-26 09:48:24 +02:00
# include "entry.h"
2005-04-16 15:20:36 -07:00
2013-01-02 15:18:18 +01:00
DEFINE_PER_CPU_SHARED_ALIGNED ( struct irq_stat , irq_stat ) ;
EXPORT_PER_CPU_SYMBOL_GPL ( irq_stat ) ;
2011-01-05 12:47:28 +01:00
struct irq_class {
char * name ;
char * desc ;
} ;
2013-01-02 15:18:18 +01:00
/*
2013-03-16 20:53:05 +09:00
* The list of " main " irq classes on s390 . This is the list of interrupts
2013-01-02 15:18:18 +01:00
* that appear both in / proc / stat ( " intr " line ) and / proc / interrupts .
* Historically only external and I / O interrupts have been part of / proc / stat .
* We can ' t add the split external and I / O sub classes since the first field
* in the " intr " line in / proc / stat is supposed to be the sum of all other
* fields .
* Since the external and I / O interrupt fields are already sums we would end
* up with having a sum which accounts each interrupt twice .
*/
2013-06-27 09:01:09 +02:00
static const struct irq_class irqclass_main_desc [ NR_IRQS_BASE ] = {
[ EXT_INTERRUPT ] = { . name = " EXT " } ,
[ IO_INTERRUPT ] = { . name = " I/O " } ,
[ THIN_INTERRUPT ] = { . name = " AIO " } ,
2013-01-02 15:18:18 +01:00
} ;
/*
* The list of split external and I / O interrupts that appear only in
* / proc / interrupts .
* In addition this list contains non external / I / O events like NMIs .
*/
static const struct irq_class irqclass_sub_desc [ NR_ARCH_IRQS ] = {
[ IRQEXT_CLK ] = { . name = " CLK " , . desc = " [EXT] Clock Comparator " } ,
[ IRQEXT_EXC ] = { . name = " EXC " , . desc = " [EXT] External Call " } ,
[ IRQEXT_EMS ] = { . name = " EMS " , . desc = " [EXT] Emergency Signal " } ,
[ IRQEXT_TMR ] = { . name = " TMR " , . desc = " [EXT] CPU Timer " } ,
[ IRQEXT_TLA ] = { . name = " TAL " , . desc = " [EXT] Timing Alert " } ,
[ IRQEXT_PFL ] = { . name = " PFL " , . desc = " [EXT] Pseudo Page Fault " } ,
[ IRQEXT_DSD ] = { . name = " DSD " , . desc = " [EXT] DASD Diag " } ,
[ IRQEXT_VRT ] = { . name = " VRT " , . desc = " [EXT] Virtio " } ,
[ IRQEXT_SCP ] = { . name = " SCP " , . desc = " [EXT] Service Call " } ,
[ IRQEXT_IUC ] = { . name = " IUC " , . desc = " [EXT] IUCV " } ,
[ IRQEXT_CMS ] = { . name = " CMS " , . desc = " [EXT] CPU-Measurement: Sampling " } ,
[ IRQEXT_CMC ] = { . name = " CMC " , . desc = " [EXT] CPU-Measurement: Counter " } ,
[ IRQEXT_CMR ] = { . name = " CMR " , . desc = " [EXT] CPU-Measurement: RI " } ,
[ IRQIO_CIO ] = { . name = " CIO " , . desc = " [I/O] Common I/O Layer Interrupt " } ,
[ IRQIO_QAI ] = { . name = " QAI " , . desc = " [I/O] QDIO Adapter Interrupt " } ,
[ IRQIO_DAS ] = { . name = " DAS " , . desc = " [I/O] DASD " } ,
[ IRQIO_C15 ] = { . name = " C15 " , . desc = " [I/O] 3215 " } ,
[ IRQIO_C70 ] = { . name = " C70 " , . desc = " [I/O] 3270 " } ,
[ IRQIO_TAP ] = { . name = " TAP " , . desc = " [I/O] Tape " } ,
[ IRQIO_VMR ] = { . name = " VMR " , . desc = " [I/O] Unit Record Devices " } ,
[ IRQIO_LCS ] = { . name = " LCS " , . desc = " [I/O] LCS " } ,
[ IRQIO_CLW ] = { . name = " CLW " , . desc = " [I/O] CLAW " } ,
[ IRQIO_CTC ] = { . name = " CTC " , . desc = " [I/O] CTC " } ,
[ IRQIO_APB ] = { . name = " APB " , . desc = " [I/O] AP Bus " } ,
[ IRQIO_ADM ] = { . name = " ADM " , . desc = " [I/O] EADM Subchannel " } ,
[ IRQIO_CSC ] = { . name = " CSC " , . desc = " [I/O] CHSC Subchannel " } ,
[ IRQIO_PCI ] = { . name = " PCI " , . desc = " [I/O] PCI Interrupt " } ,
[ IRQIO_MSI ] = { . name = " MSI " , . desc = " [I/O] MSI Interrupt " } ,
2013-02-24 13:07:18 -08:00
[ IRQIO_VIR ] = { . name = " VIR " , . desc = " [I/O] Virtual I/O Devices " } ,
2012-09-17 06:05:16 +02:00
[ NMI_NMI ] = { . name = " NMI " , . desc = " [NMI] Machine Check " } ,
2013-01-02 16:54:12 +01:00
[ CPU_RST ] = { . name = " RST " , . desc = " [CPU] CPU Restart " } ,
2011-01-05 12:47:28 +01:00
} ;
2013-06-27 09:01:09 +02:00
void __init init_IRQ ( void )
{
irq_reserve_irqs ( 0 , THIN_INTERRUPT ) ;
init_cio_interrupts ( ) ;
init_airq_interrupts ( ) ;
init_ext_interrupts ( ) ;
}
void do_IRQ ( struct pt_regs * regs , int irq )
{
struct pt_regs * old_regs ;
old_regs = set_irq_regs ( regs ) ;
irq_enter ( ) ;
if ( S390_lowcore . int_clock > = S390_lowcore . clock_comparator )
/* Serve timer interrupts first. */
clock_comparator_work ( ) ;
generic_handle_irq ( irq ) ;
irq_exit ( ) ;
set_irq_regs ( old_regs ) ;
}
2005-04-16 15:20:36 -07:00
/*
* show_interrupts is needed by / proc / interrupts .
*/
int show_interrupts ( struct seq_file * p , void * v )
{
2013-01-02 15:18:18 +01:00
int irq = * ( loff_t * ) v ;
int cpu ;
2005-04-16 15:20:36 -07:00
2008-05-15 16:52:39 +02:00
get_online_cpus ( ) ;
2013-01-02 15:18:18 +01:00
if ( irq = = 0 ) {
2005-04-16 15:20:36 -07:00
seq_puts ( p , " " ) ;
2013-01-02 15:18:18 +01:00
for_each_online_cpu ( cpu )
seq_printf ( p , " CPU%d " , cpu ) ;
2005-04-16 15:20:36 -07:00
seq_putc ( p , ' \n ' ) ;
2013-06-27 09:01:09 +02:00
goto out ;
2005-04-16 15:20:36 -07:00
}
2013-01-02 15:18:18 +01:00
if ( irq < NR_IRQS ) {
2013-06-27 09:01:09 +02:00
if ( irq > = NR_IRQS_BASE )
goto out ;
2013-01-02 15:18:18 +01:00
seq_printf ( p , " %s: " , irqclass_main_desc [ irq ] . name ) ;
for_each_online_cpu ( cpu )
2013-06-27 09:01:09 +02:00
seq_printf ( p , " %10u " , kstat_irqs_cpu ( irq , cpu ) ) ;
2013-01-02 15:18:18 +01:00
seq_putc ( p , ' \n ' ) ;
2013-06-27 09:01:09 +02:00
goto out ;
2013-01-02 15:18:18 +01:00
}
for ( irq = 0 ; irq < NR_ARCH_IRQS ; irq + + ) {
seq_printf ( p , " %s: " , irqclass_sub_desc [ irq ] . name ) ;
for_each_online_cpu ( cpu )
2013-06-27 09:01:09 +02:00
seq_printf ( p , " %10u " ,
per_cpu ( irq_stat , cpu ) . irqs [ irq ] ) ;
2013-01-02 15:18:18 +01:00
if ( irqclass_sub_desc [ irq ] . desc )
seq_printf ( p , " %s " , irqclass_sub_desc [ irq ] . desc ) ;
seq_putc ( p , ' \n ' ) ;
}
2013-06-27 09:01:09 +02:00
out :
2008-05-15 16:52:39 +02:00
put_online_cpus ( ) ;
2013-01-02 15:18:18 +01:00
return 0 ;
2005-04-16 15:20:36 -07:00
}
2013-06-27 09:01:09 +02:00
int arch_show_interrupts ( struct seq_file * p , int prec )
{
return 0 ;
}
2005-04-16 15:20:36 -07:00
/*
* Switch to the asynchronous interrupt stack for softirq execution .
*/
asmlinkage void do_softirq ( void )
{
unsigned long flags , old , new ;
if ( in_interrupt ( ) )
return ;
local_irq_save ( flags ) ;
if ( local_softirq_pending ( ) ) {
/* Get current stack pointer. */
asm volatile ( " la %0,0(15) " : " = a " (old)) ;
/* Check against async. stack address range. */
new = S390_lowcore . async_stack ;
if ( ( ( new - old ) > > ( PAGE_SHIFT + THREAD_ORDER ) ) ! = 0 ) {
/* Need to switch to the async. stack. */
new - = STACK_FRAME_OVERHEAD ;
( ( struct stack_frame * ) new ) - > back_chain = old ;
asm volatile ( " la 15,0(%0) \n "
" basr 14,%2 \n "
" la 15,0(%1) \n "
: : " a " ( new ) , " a " ( old ) ,
" a " ( __do_softirq )
: " 0 " , " 1 " , " 2 " , " 3 " , " 4 " , " 5 " , " 14 " ,
" cc " , " memory " ) ;
2012-04-11 14:28:09 +02:00
} else {
2005-04-16 15:20:36 -07:00
/* We are already on the async stack. */
__do_softirq ( ) ;
2012-04-11 14:28:09 +02:00
}
2005-04-16 15:20:36 -07:00
}
local_irq_restore ( flags ) ;
}
2007-02-05 21:16:44 +01:00
2011-05-26 09:48:24 +02:00
/*
2011-07-24 10:48:27 +02:00
* ext_int_hash [ index ] is the list head for all external interrupts that hash
* to this index .
2011-05-26 09:48:24 +02:00
*/
2013-09-03 14:12:07 +02:00
static struct hlist_head ext_int_hash [ 256 ] ;
2011-05-26 09:48:24 +02:00
struct ext_int_info {
ext_int_handler_t handler ;
2013-09-03 14:12:07 +02:00
struct hlist_node entry ;
2011-07-24 10:48:27 +02:00
struct rcu_head rcu ;
2013-09-03 14:12:07 +02:00
u16 code ;
2011-05-26 09:48:24 +02:00
} ;
2011-07-24 10:48:27 +02:00
/* ext_int_hash_lock protects the handler lists for external interrupts */
DEFINE_SPINLOCK ( ext_int_hash_lock ) ;
2011-05-26 09:48:24 +02:00
static inline int ext_hash ( u16 code )
{
return ( code + ( code > > 9 ) ) & 0xff ;
}
int register_external_interrupt ( u16 code , ext_int_handler_t handler )
{
struct ext_int_info * p ;
2011-07-24 10:48:27 +02:00
unsigned long flags ;
2011-05-26 09:48:24 +02:00
int index ;
p = kmalloc ( sizeof ( * p ) , GFP_ATOMIC ) ;
if ( ! p )
return - ENOMEM ;
p - > code = code ;
p - > handler = handler ;
index = ext_hash ( code ) ;
2011-07-24 10:48:27 +02:00
spin_lock_irqsave ( & ext_int_hash_lock , flags ) ;
2013-09-03 14:12:07 +02:00
hlist_add_head_rcu ( & p - > entry , & ext_int_hash [ index ] ) ;
2011-07-24 10:48:27 +02:00
spin_unlock_irqrestore ( & ext_int_hash_lock , flags ) ;
2011-05-26 09:48:24 +02:00
return 0 ;
}
EXPORT_SYMBOL ( register_external_interrupt ) ;
int unregister_external_interrupt ( u16 code , ext_int_handler_t handler )
{
2011-07-24 10:48:27 +02:00
struct ext_int_info * p ;
unsigned long flags ;
int index = ext_hash ( code ) ;
2011-05-26 09:48:24 +02:00
2011-07-24 10:48:27 +02:00
spin_lock_irqsave ( & ext_int_hash_lock , flags ) ;
2013-09-03 14:12:07 +02:00
hlist_for_each_entry_rcu ( p , & ext_int_hash [ index ] , entry ) {
2011-07-24 10:48:27 +02:00
if ( p - > code = = code & & p - > handler = = handler ) {
2013-09-03 14:12:07 +02:00
hlist_del_rcu ( & p - > entry ) ;
2012-01-06 16:59:51 -08:00
kfree_rcu ( p , rcu ) ;
2011-07-24 10:48:27 +02:00
}
2012-04-11 14:28:09 +02:00
}
2011-07-24 10:48:27 +02:00
spin_unlock_irqrestore ( & ext_int_hash_lock , flags ) ;
2011-05-26 09:48:24 +02:00
return 0 ;
}
EXPORT_SYMBOL ( unregister_external_interrupt ) ;
2013-06-27 09:01:09 +02:00
static irqreturn_t do_ext_interrupt ( int irq , void * dummy )
2011-05-26 09:48:24 +02:00
{
2013-06-27 09:01:09 +02:00
struct pt_regs * regs = get_irq_regs ( ) ;
2013-06-17 14:54:02 +02:00
struct ext_code ext_code ;
2011-05-26 09:48:24 +02:00
struct ext_int_info * p ;
int index ;
2013-06-17 14:54:02 +02:00
ext_code = * ( struct ext_code * ) & regs - > int_code ;
2012-03-11 11:59:31 -04:00
if ( ext_code . code ! = 0x1004 )
2011-05-26 09:48:24 +02:00
__get_cpu_var ( s390_idle ) . nohz_delay = 1 ;
2011-07-24 10:48:27 +02:00
2012-03-11 11:59:31 -04:00
index = ext_hash ( ext_code . code ) ;
2011-07-24 10:48:27 +02:00
rcu_read_lock ( ) ;
2013-09-03 14:12:07 +02:00
hlist_for_each_entry_rcu ( p , & ext_int_hash [ index ] , entry ) {
if ( unlikely ( p - > code ! = ext_code . code ) )
continue ;
p - > handler ( ext_code , regs - > int_parm , regs - > int_parm_long ) ;
}
2011-07-24 10:48:27 +02:00
rcu_read_unlock ( ) ;
2013-06-27 09:01:09 +02:00
return IRQ_HANDLED ;
2011-05-26 09:48:24 +02:00
}
2013-06-27 09:01:09 +02:00
static struct irqaction external_interrupt = {
. name = " EXT " ,
. handler = do_ext_interrupt ,
} ;
void __init init_ext_interrupts ( void )
2011-07-24 10:48:27 +02:00
{
2013-06-27 09:01:09 +02:00
int idx ;
for ( idx = 0 ; idx < ARRAY_SIZE ( ext_int_hash ) ; idx + + )
2013-09-03 14:12:07 +02:00
INIT_HLIST_HEAD ( & ext_int_hash [ idx ] ) ;
2013-06-27 09:01:09 +02:00
irq_set_chip_and_handler ( EXT_INTERRUPT ,
& dummy_irq_chip , handle_percpu_irq ) ;
setup_irq ( EXT_INTERRUPT , & external_interrupt ) ;
2011-07-24 10:48:27 +02:00
}
2011-05-26 09:48:24 +02:00
static DEFINE_SPINLOCK ( sc_irq_lock ) ;
static int sc_irq_refcount ;
void service_subclass_irq_register ( void )
{
spin_lock ( & sc_irq_lock ) ;
if ( ! sc_irq_refcount )
ctl_set_bit ( 0 , 9 ) ;
sc_irq_refcount + + ;
spin_unlock ( & sc_irq_lock ) ;
}
EXPORT_SYMBOL ( service_subclass_irq_register ) ;
void service_subclass_irq_unregister ( void )
{
spin_lock ( & sc_irq_lock ) ;
sc_irq_refcount - - ;
if ( ! sc_irq_refcount )
ctl_clear_bit ( 0 , 9 ) ;
spin_unlock ( & sc_irq_lock ) ;
}
EXPORT_SYMBOL ( service_subclass_irq_unregister ) ;
2012-03-23 11:13:05 +01:00
static DEFINE_SPINLOCK ( ma_subclass_lock ) ;
static int ma_subclass_refcount ;
void measurement_alert_subclass_register ( void )
{
spin_lock ( & ma_subclass_lock ) ;
if ( ! ma_subclass_refcount )
ctl_set_bit ( 0 , 5 ) ;
ma_subclass_refcount + + ;
spin_unlock ( & ma_subclass_lock ) ;
}
EXPORT_SYMBOL ( measurement_alert_subclass_register ) ;
void measurement_alert_subclass_unregister ( void )
{
spin_lock ( & ma_subclass_lock ) ;
ma_subclass_refcount - - ;
if ( ! ma_subclass_refcount )
ctl_clear_bit ( 0 , 5 ) ;
spin_unlock ( & ma_subclass_lock ) ;
}
EXPORT_SYMBOL ( measurement_alert_subclass_unregister ) ;