2005-04-16 15:20:36 -07:00
/*
* arch / s390 / kernel / traps . c
*
* S390 version
* Copyright ( C ) 1999 , 2000 IBM Deutschland Entwicklung GmbH , IBM Corporation
* Author ( s ) : Martin Schwidefsky ( schwidefsky @ de . ibm . com ) ,
* Denis Joseph Barrow ( djbarrow @ de . ibm . com , barrow_dj @ yahoo . com ) ,
*
* Derived from " arch/i386/kernel/traps.c "
* Copyright ( C ) 1991 , 1992 Linus Torvalds
*/
/*
* ' Traps . c ' handles hardware traps and faults after we have saved some
* state in ' asm . s ' .
*/
# include <linux/sched.h>
# include <linux/kernel.h>
# include <linux/string.h>
# include <linux/errno.h>
# include <linux/ptrace.h>
# include <linux/timer.h>
# include <linux/mm.h>
# include <linux/smp.h>
# include <linux/init.h>
# include <linux/interrupt.h>
# include <linux/delay.h>
# include <linux/module.h>
2007-05-08 00:27:03 -07:00
# include <linux/kdebug.h>
2005-04-16 15:20:36 -07:00
# include <linux/kallsyms.h>
2005-08-01 21:11:33 -07:00
# include <linux/reboot.h>
2006-09-20 15:58:39 +02:00
# include <linux/kprobes.h>
2007-04-27 16:01:42 +02:00
# include <linux/bug.h>
2005-04-16 15:20:36 -07:00
# include <asm/system.h>
# include <asm/uaccess.h>
# include <asm/io.h>
# include <asm/atomic.h>
# include <asm/mathemu.h>
# include <asm/cpcmd.h>
# include <asm/s390_ext.h>
# include <asm/lowcore.h>
# include <asm/debug.h>
/* Called from entry.S only */
extern void handle_per_exception ( struct pt_regs * regs ) ;
typedef void pgm_check_handler_t ( struct pt_regs * , long ) ;
pgm_check_handler_t * pgm_check_table [ 128 ] ;
# ifdef CONFIG_SYSCTL
# ifdef CONFIG_PROCESS_DEBUG
int sysctl_userprocess_debug = 1 ;
# else
int sysctl_userprocess_debug = 0 ;
# endif
# endif
extern pgm_check_handler_t do_protection_exception ;
extern pgm_check_handler_t do_dat_exception ;
extern pgm_check_handler_t do_monitor_call ;
# define stack_pointer ({ void **sp; asm("la %0,0(15)" : "=&d" (sp)); sp; })
2006-01-06 00:19:28 -08:00
# ifndef CONFIG_64BIT
2005-04-16 15:20:36 -07:00
# define FOURLONG "%08lx %08lx %08lx %08lx\n"
static int kstack_depth_to_print = 12 ;
2006-01-06 00:19:28 -08:00
# else /* CONFIG_64BIT */
2005-04-16 15:20:36 -07:00
# define FOURLONG "%016lx %016lx %016lx %016lx\n"
static int kstack_depth_to_print = 20 ;
2006-01-06 00:19:28 -08:00
# endif /* CONFIG_64BIT */
2005-04-16 15:20:36 -07:00
/*
* For show_trace we have tree different stack to consider :
* - the panic stack which is used if the kernel stack has overflown
* - the asynchronous interrupt stack ( cpu related )
* - the synchronous kernel stack ( process related )
* The stack trace can start at any of the three stack and can potentially
* touch all of them . The order is : panic stack , async stack , sync stack .
*/
static unsigned long
__show_trace ( unsigned long sp , unsigned long low , unsigned long high )
{
struct stack_frame * sf ;
struct pt_regs * regs ;
while ( 1 ) {
sp = sp & PSW_ADDR_INSN ;
if ( sp < low | | sp > high - sizeof ( * sf ) )
return sp ;
sf = ( struct stack_frame * ) sp ;
printk ( " ([<%016lx>] " , sf - > gprs [ 8 ] & PSW_ADDR_INSN ) ;
print_symbol ( " %s) \n " , sf - > gprs [ 8 ] & PSW_ADDR_INSN ) ;
/* Follow the backchain. */
while ( 1 ) {
low = sp ;
sp = sf - > back_chain & PSW_ADDR_INSN ;
if ( ! sp )
break ;
if ( sp < = low | | sp > high - sizeof ( * sf ) )
return sp ;
sf = ( struct stack_frame * ) sp ;
printk ( " [<%016lx>] " , sf - > gprs [ 8 ] & PSW_ADDR_INSN ) ;
print_symbol ( " %s \n " , sf - > gprs [ 8 ] & PSW_ADDR_INSN ) ;
}
/* Zero backchain detected, check for interrupt frame. */
sp = ( unsigned long ) ( sf + 1 ) ;
if ( sp < = low | | sp > high - sizeof ( * regs ) )
return sp ;
regs = ( struct pt_regs * ) sp ;
printk ( " [<%016lx>] " , regs - > psw . addr & PSW_ADDR_INSN ) ;
print_symbol ( " %s \n " , regs - > psw . addr & PSW_ADDR_INSN ) ;
low = sp ;
sp = regs - > gprs [ 15 ] ;
}
}
2006-12-04 15:40:47 +01:00
void show_trace ( struct task_struct * task , unsigned long * stack )
2005-04-16 15:20:36 -07:00
{
register unsigned long __r15 asm ( " 15 " ) ;
unsigned long sp ;
sp = ( unsigned long ) stack ;
if ( ! sp )
sp = task ? task - > thread . ksp : __r15 ;
printk ( " Call Trace: \n " ) ;
# ifdef CONFIG_CHECK_STACK
sp = __show_trace ( sp , S390_lowcore . panic_stack - 4096 ,
S390_lowcore . panic_stack ) ;
# endif
sp = __show_trace ( sp , S390_lowcore . async_stack - ASYNC_SIZE ,
S390_lowcore . async_stack ) ;
if ( task )
2006-01-12 01:05:50 -08:00
__show_trace ( sp , ( unsigned long ) task_stack_page ( task ) ,
( unsigned long ) task_stack_page ( task ) + THREAD_SIZE ) ;
2005-04-16 15:20:36 -07:00
else
__show_trace ( sp , S390_lowcore . thread_info ,
S390_lowcore . thread_info + THREAD_SIZE ) ;
printk ( " \n " ) ;
2006-12-04 15:40:47 +01:00
if ( ! task )
task = current ;
debug_show_held_locks ( task ) ;
2005-04-16 15:20:36 -07:00
}
void show_stack ( struct task_struct * task , unsigned long * sp )
{
register unsigned long * __r15 asm ( " 15 " ) ;
unsigned long * stack ;
int i ;
if ( ! sp )
2006-06-29 14:56:23 +02:00
stack = task ? ( unsigned long * ) task - > thread . ksp : __r15 ;
else
stack = sp ;
2005-04-16 15:20:36 -07:00
for ( i = 0 ; i < kstack_depth_to_print ; i + + ) {
if ( ( ( addr_t ) stack & ( THREAD_SIZE - 1 ) ) = = 0 )
break ;
if ( i & & ( ( i * sizeof ( long ) % 32 ) = = 0 ) )
printk ( " \n " ) ;
printk ( " %p " , ( void * ) * stack + + ) ;
}
printk ( " \n " ) ;
show_trace ( task , sp ) ;
}
/*
* The architecture - independent dump_stack generator
*/
void dump_stack ( void )
{
2006-07-12 16:41:55 +02:00
show_stack ( NULL , NULL ) ;
2005-04-16 15:20:36 -07:00
}
EXPORT_SYMBOL ( dump_stack ) ;
[S390] Improved oops output.
This patch adds two improvements to the oops output. First it adds an
additional line after the PSW which decodes the different fields of it.
Second a disassembler is added that decodes the instructions surrounding
the faulting PSW. The output of a test oops now looks like this:
kernel BUG at init/main.c:419
illegal operation: 0001 [#1]
CPU: 0 Not tainted
Process swapper (pid: 0, task: 0000000000464968, ksp: 00000000004be000)
Krnl PSW : 0700000180000000 00000000000120b6 (rest_init+0x36/0x38)
R:0 T:1 IO:1 EX:1 Key:0 M:0 W:0 P:0 AS:0 CC:0 PM:0 EA:3
Krnl GPRS: 0000000000000003 00000000004ba017 0000000000000022 0000000000000001
000000000003a5f6 0000000000000000 00000000004be6a8 0000000000000000
0000000000000000 00000000004b8200 0000000000003a50 0000000000008000
0000000000516368 000000000033d008 00000000000120b2 00000000004bdee0
Krnl Code: 00000000000120a6: e3e0f0980024 stg %r14,152(%r15)
00000000000120ac: c0e500014296 brasl %r14,3a5d8
00000000000120b2: a7f40001 brc 15,120b4
>00000000000120b6: 0707 bcr 0,%r7
00000000000120b8: eb7ff0500024 stmg %r7,%r15,80(%r15)
00000000000120be: c0d000195825 larl %r13,33d108
00000000000120c4: a7f13f00 tmll %r15,16128
00000000000120c8: a7840001 brc 8,120ca
Call Trace:
([<00000000000120b2>] rest_init+0x32/0x38)
[<00000000004be614>] start_kernel+0x37c/0x410
[<0000000000012020>] _ehead+0x20/0x80
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
2007-04-27 16:01:41 +02:00
static inline int mask_bits ( struct pt_regs * regs , unsigned long bits )
{
return ( regs - > psw . mask & bits ) / ( ( ~ bits + 1 ) & bits ) ;
}
2005-04-16 15:20:36 -07:00
void show_registers ( struct pt_regs * regs )
{
char * mode ;
mode = ( regs - > psw . mask & PSW_MASK_PSTATE ) ? " User " : " Krnl " ;
printk ( " %s PSW : %p %p " ,
mode , ( void * ) regs - > psw . mask ,
( void * ) regs - > psw . addr ) ;
print_symbol ( " (%s) \n " , regs - > psw . addr & PSW_ADDR_INSN ) ;
[S390] Improved oops output.
This patch adds two improvements to the oops output. First it adds an
additional line after the PSW which decodes the different fields of it.
Second a disassembler is added that decodes the instructions surrounding
the faulting PSW. The output of a test oops now looks like this:
kernel BUG at init/main.c:419
illegal operation: 0001 [#1]
CPU: 0 Not tainted
Process swapper (pid: 0, task: 0000000000464968, ksp: 00000000004be000)
Krnl PSW : 0700000180000000 00000000000120b6 (rest_init+0x36/0x38)
R:0 T:1 IO:1 EX:1 Key:0 M:0 W:0 P:0 AS:0 CC:0 PM:0 EA:3
Krnl GPRS: 0000000000000003 00000000004ba017 0000000000000022 0000000000000001
000000000003a5f6 0000000000000000 00000000004be6a8 0000000000000000
0000000000000000 00000000004b8200 0000000000003a50 0000000000008000
0000000000516368 000000000033d008 00000000000120b2 00000000004bdee0
Krnl Code: 00000000000120a6: e3e0f0980024 stg %r14,152(%r15)
00000000000120ac: c0e500014296 brasl %r14,3a5d8
00000000000120b2: a7f40001 brc 15,120b4
>00000000000120b6: 0707 bcr 0,%r7
00000000000120b8: eb7ff0500024 stmg %r7,%r15,80(%r15)
00000000000120be: c0d000195825 larl %r13,33d108
00000000000120c4: a7f13f00 tmll %r15,16128
00000000000120c8: a7840001 brc 8,120ca
Call Trace:
([<00000000000120b2>] rest_init+0x32/0x38)
[<00000000004be614>] start_kernel+0x37c/0x410
[<0000000000012020>] _ehead+0x20/0x80
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
2007-04-27 16:01:41 +02:00
printk ( " R:%x T:%x IO:%x EX:%x Key:%x M:%x W:%x "
" P:%x AS:%x CC:%x PM:%x " , mask_bits ( regs , PSW_MASK_PER ) ,
mask_bits ( regs , PSW_MASK_DAT ) , mask_bits ( regs , PSW_MASK_IO ) ,
mask_bits ( regs , PSW_MASK_EXT ) , mask_bits ( regs , PSW_MASK_KEY ) ,
mask_bits ( regs , PSW_MASK_MCHECK ) , mask_bits ( regs , PSW_MASK_WAIT ) ,
mask_bits ( regs , PSW_MASK_PSTATE ) , mask_bits ( regs , PSW_MASK_ASC ) ,
mask_bits ( regs , PSW_MASK_CC ) , mask_bits ( regs , PSW_MASK_PM ) ) ;
# ifdef CONFIG_64BIT
printk ( " EA:%x " , mask_bits ( regs , PSW_BASE_BITS ) ) ;
# endif
printk ( " \n %s GPRS: " FOURLONG , mode ,
2005-04-16 15:20:36 -07:00
regs - > gprs [ 0 ] , regs - > gprs [ 1 ] , regs - > gprs [ 2 ] , regs - > gprs [ 3 ] ) ;
printk ( " " FOURLONG ,
regs - > gprs [ 4 ] , regs - > gprs [ 5 ] , regs - > gprs [ 6 ] , regs - > gprs [ 7 ] ) ;
printk ( " " FOURLONG ,
regs - > gprs [ 8 ] , regs - > gprs [ 9 ] , regs - > gprs [ 10 ] , regs - > gprs [ 11 ] ) ;
printk ( " " FOURLONG ,
regs - > gprs [ 12 ] , regs - > gprs [ 13 ] , regs - > gprs [ 14 ] , regs - > gprs [ 15 ] ) ;
[S390] Improved oops output.
This patch adds two improvements to the oops output. First it adds an
additional line after the PSW which decodes the different fields of it.
Second a disassembler is added that decodes the instructions surrounding
the faulting PSW. The output of a test oops now looks like this:
kernel BUG at init/main.c:419
illegal operation: 0001 [#1]
CPU: 0 Not tainted
Process swapper (pid: 0, task: 0000000000464968, ksp: 00000000004be000)
Krnl PSW : 0700000180000000 00000000000120b6 (rest_init+0x36/0x38)
R:0 T:1 IO:1 EX:1 Key:0 M:0 W:0 P:0 AS:0 CC:0 PM:0 EA:3
Krnl GPRS: 0000000000000003 00000000004ba017 0000000000000022 0000000000000001
000000000003a5f6 0000000000000000 00000000004be6a8 0000000000000000
0000000000000000 00000000004b8200 0000000000003a50 0000000000008000
0000000000516368 000000000033d008 00000000000120b2 00000000004bdee0
Krnl Code: 00000000000120a6: e3e0f0980024 stg %r14,152(%r15)
00000000000120ac: c0e500014296 brasl %r14,3a5d8
00000000000120b2: a7f40001 brc 15,120b4
>00000000000120b6: 0707 bcr 0,%r7
00000000000120b8: eb7ff0500024 stmg %r7,%r15,80(%r15)
00000000000120be: c0d000195825 larl %r13,33d108
00000000000120c4: a7f13f00 tmll %r15,16128
00000000000120c8: a7840001 brc 8,120ca
Call Trace:
([<00000000000120b2>] rest_init+0x32/0x38)
[<00000000004be614>] start_kernel+0x37c/0x410
[<0000000000012020>] _ehead+0x20/0x80
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Signed-off-by: Heiko Carstens <heiko.carstens@de.ibm.com>
2007-04-27 16:01:41 +02:00
show_code ( regs ) ;
2005-04-16 15:20:36 -07:00
}
/* This is called from fs/proc/array.c */
char * task_show_regs ( struct task_struct * task , char * buffer )
{
struct pt_regs * regs ;
2006-01-12 01:05:49 -08:00
regs = task_pt_regs ( task ) ;
2005-04-16 15:20:36 -07:00
buffer + = sprintf ( buffer , " task: %p, ksp: %p \n " ,
task , ( void * ) task - > thread . ksp ) ;
buffer + = sprintf ( buffer , " User PSW : %p %p \n " ,
( void * ) regs - > psw . mask , ( void * ) regs - > psw . addr ) ;
buffer + = sprintf ( buffer , " User GPRS: " FOURLONG ,
regs - > gprs [ 0 ] , regs - > gprs [ 1 ] ,
regs - > gprs [ 2 ] , regs - > gprs [ 3 ] ) ;
buffer + = sprintf ( buffer , " " FOURLONG ,
regs - > gprs [ 4 ] , regs - > gprs [ 5 ] ,
regs - > gprs [ 6 ] , regs - > gprs [ 7 ] ) ;
buffer + = sprintf ( buffer , " " FOURLONG ,
regs - > gprs [ 8 ] , regs - > gprs [ 9 ] ,
regs - > gprs [ 10 ] , regs - > gprs [ 11 ] ) ;
buffer + = sprintf ( buffer , " " FOURLONG ,
regs - > gprs [ 12 ] , regs - > gprs [ 13 ] ,
regs - > gprs [ 14 ] , regs - > gprs [ 15 ] ) ;
buffer + = sprintf ( buffer , " User ACRS: %08x %08x %08x %08x \n " ,
task - > thread . acrs [ 0 ] , task - > thread . acrs [ 1 ] ,
task - > thread . acrs [ 2 ] , task - > thread . acrs [ 3 ] ) ;
buffer + = sprintf ( buffer , " %08x %08x %08x %08x \n " ,
task - > thread . acrs [ 4 ] , task - > thread . acrs [ 5 ] ,
task - > thread . acrs [ 6 ] , task - > thread . acrs [ 7 ] ) ;
buffer + = sprintf ( buffer , " %08x %08x %08x %08x \n " ,
task - > thread . acrs [ 8 ] , task - > thread . acrs [ 9 ] ,
task - > thread . acrs [ 10 ] , task - > thread . acrs [ 11 ] ) ;
buffer + = sprintf ( buffer , " %08x %08x %08x %08x \n " ,
task - > thread . acrs [ 12 ] , task - > thread . acrs [ 13 ] ,
task - > thread . acrs [ 14 ] , task - > thread . acrs [ 15 ] ) ;
return buffer ;
}
2007-02-05 21:16:47 +01:00
static DEFINE_SPINLOCK ( die_lock ) ;
2005-04-16 15:20:36 -07:00
void die ( const char * str , struct pt_regs * regs , long err )
{
static int die_counter ;
2007-06-19 13:10:05 +02:00
oops_enter ( ) ;
2005-04-16 15:20:36 -07:00
debug_stop_all ( ) ;
console_verbose ( ) ;
spin_lock_irq ( & die_lock ) ;
bust_spinlocks ( 1 ) ;
printk ( " %s: %04lx [#%d] \n " , str , err & 0xffff , + + die_counter ) ;
2007-06-19 13:10:04 +02:00
print_modules ( ) ;
2007-06-19 13:10:05 +02:00
show_regs ( regs ) ;
2005-04-16 15:20:36 -07:00
bust_spinlocks ( 0 ) ;
2007-07-17 04:03:42 -07:00
add_taint ( TAINT_DIE ) ;
2007-06-19 13:10:05 +02:00
spin_unlock_irq ( & die_lock ) ;
2005-04-16 15:20:36 -07:00
if ( in_interrupt ( ) )
panic ( " Fatal exception in interrupt " ) ;
if ( panic_on_oops )
panic ( " Fatal exception: panic_on_oops " ) ;
2007-06-19 13:10:05 +02:00
oops_exit ( ) ;
do_exit ( SIGSEGV ) ;
2005-04-16 15:20:36 -07:00
}
static void inline
report_user_fault ( long interruption_code , struct pt_regs * regs )
{
# if defined(CONFIG_SYSCTL)
if ( ! sysctl_userprocess_debug )
return ;
# endif
# if defined(CONFIG_SYSCTL) || defined(CONFIG_PROCESS_DEBUG)
printk ( " User process fault: interruption code 0x%lX \n " ,
interruption_code ) ;
show_regs ( regs ) ;
# endif
}
2007-04-27 16:01:42 +02:00
int is_valid_bugaddr ( unsigned long addr )
{
return 1 ;
}
2006-09-20 15:58:39 +02:00
static void __kprobes inline do_trap ( long interruption_code , int signr ,
char * str , struct pt_regs * regs ,
siginfo_t * info )
2005-04-16 15:20:36 -07:00
{
/*
* We got all needed information from the lowcore and can
* now safely switch on interrupts .
*/
if ( regs - > psw . mask & PSW_MASK_PSTATE )
local_irq_enable ( ) ;
2006-09-20 15:58:39 +02:00
if ( notify_die ( DIE_TRAP , str , regs , interruption_code ,
interruption_code , signr ) = = NOTIFY_STOP )
return ;
2005-04-16 15:20:36 -07:00
if ( regs - > psw . mask & PSW_MASK_PSTATE ) {
struct task_struct * tsk = current ;
tsk - > thread . trap_no = interruption_code & 0xffff ;
force_sig_info ( signr , info , tsk ) ;
report_user_fault ( interruption_code , regs ) ;
} else {
const struct exception_table_entry * fixup ;
fixup = search_exception_tables ( regs - > psw . addr & PSW_ADDR_INSN ) ;
if ( fixup )
regs - > psw . addr = fixup - > fixup | PSW_ADDR_AMODE ;
2007-04-27 16:01:42 +02:00
else {
enum bug_trap_type btt ;
2007-07-15 23:41:39 -07:00
btt = report_bug ( regs - > psw . addr & PSW_ADDR_INSN , regs ) ;
2007-04-27 16:01:42 +02:00
if ( btt = = BUG_TRAP_TYPE_WARN )
return ;
die ( str , regs , interruption_code ) ;
}
2005-04-16 15:20:36 -07:00
}
}
2006-07-12 16:41:55 +02:00
static inline void __user * get_check_address ( struct pt_regs * regs )
2005-04-16 15:20:36 -07:00
{
2006-07-12 16:41:55 +02:00
return ( void __user * ) ( ( regs - > psw . addr - S390_lowcore . pgm_ilc ) & PSW_ADDR_INSN ) ;
2005-04-16 15:20:36 -07:00
}
2006-09-20 15:58:39 +02:00
void __kprobes do_single_step ( struct pt_regs * regs )
2005-04-16 15:20:36 -07:00
{
2006-09-20 15:58:39 +02:00
if ( notify_die ( DIE_SSTEP , " sstep " , regs , 0 , 0 ,
SIGTRAP ) = = NOTIFY_STOP ) {
return ;
}
2005-04-16 15:20:36 -07:00
if ( ( current - > ptrace & PT_PTRACED ) ! = 0 )
force_sig ( SIGTRAP , current ) ;
}
2007-02-05 21:16:47 +01:00
static void default_trap_handler ( struct pt_regs * regs , long interruption_code )
2005-04-16 15:20:36 -07:00
{
if ( regs - > psw . mask & PSW_MASK_PSTATE ) {
local_irq_enable ( ) ;
do_exit ( SIGSEGV ) ;
report_user_fault ( interruption_code , regs ) ;
} else
die ( " Unknown program exception " , regs , interruption_code ) ;
}
# define DO_ERROR_INFO(signr, str, name, sicode, siaddr) \
2007-02-05 21:16:47 +01:00
static void name ( struct pt_regs * regs , long interruption_code ) \
2005-04-16 15:20:36 -07:00
{ \
siginfo_t info ; \
info . si_signo = signr ; \
info . si_errno = 0 ; \
info . si_code = sicode ; \
2006-07-12 16:41:55 +02:00
info . si_addr = siaddr ; \
2005-04-16 15:20:36 -07:00
do_trap ( interruption_code , signr , str , regs , & info ) ; \
}
DO_ERROR_INFO ( SIGILL , " addressing exception " , addressing_exception ,
ILL_ILLADR , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGILL , " execute exception " , execute_exception ,
ILL_ILLOPN , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGFPE , " fixpoint divide exception " , divide_exception ,
FPE_INTDIV , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGFPE , " fixpoint overflow exception " , overflow_exception ,
FPE_INTOVF , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGFPE , " HFP overflow exception " , hfp_overflow_exception ,
FPE_FLTOVF , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGFPE , " HFP underflow exception " , hfp_underflow_exception ,
FPE_FLTUND , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGFPE , " HFP significance exception " , hfp_significance_exception ,
FPE_FLTRES , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGFPE , " HFP divide exception " , hfp_divide_exception ,
FPE_FLTDIV , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGFPE , " HFP square root exception " , hfp_sqrt_exception ,
FPE_FLTINV , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGILL , " operand exception " , operand_exception ,
ILL_ILLOPN , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGILL , " privileged operation " , privileged_op ,
ILL_PRVOPC , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGILL , " special operation exception " , special_op_exception ,
ILL_ILLOPN , get_check_address ( regs ) )
DO_ERROR_INFO ( SIGILL , " translation exception " , translation_exception ,
ILL_ILLOPN , get_check_address ( regs ) )
static inline void
2006-07-12 16:41:55 +02:00
do_fp_trap ( struct pt_regs * regs , void __user * location ,
2005-04-16 15:20:36 -07:00
int fpc , long interruption_code )
{
siginfo_t si ;
si . si_signo = SIGFPE ;
si . si_errno = 0 ;
si . si_addr = location ;
si . si_code = 0 ;
/* FPC[2] is Data Exception Code */
if ( ( fpc & 0x00000300 ) = = 0 ) {
/* bits 6 and 7 of DXC are 0 iff IEEE exception */
if ( fpc & 0x8000 ) /* invalid fp operation */
si . si_code = FPE_FLTINV ;
else if ( fpc & 0x4000 ) /* div by 0 */
si . si_code = FPE_FLTDIV ;
else if ( fpc & 0x2000 ) /* overflow */
si . si_code = FPE_FLTOVF ;
else if ( fpc & 0x1000 ) /* underflow */
si . si_code = FPE_FLTUND ;
else if ( fpc & 0x0800 ) /* inexact */
si . si_code = FPE_FLTRES ;
}
current - > thread . ieee_instruction_pointer = ( addr_t ) location ;
do_trap ( interruption_code , SIGFPE ,
" floating point exception " , regs , & si ) ;
}
2007-02-05 21:16:47 +01:00
static void illegal_op ( struct pt_regs * regs , long interruption_code )
2005-04-16 15:20:36 -07:00
{
siginfo_t info ;
__u8 opcode [ 6 ] ;
2006-07-12 16:41:55 +02:00
__u16 __user * location ;
2005-04-16 15:20:36 -07:00
int signal = 0 ;
2006-07-12 16:41:55 +02:00
location = get_check_address ( regs ) ;
2005-04-16 15:20:36 -07:00
/*
* We got all needed information from the lowcore and can
* now safely switch on interrupts .
*/
if ( regs - > psw . mask & PSW_MASK_PSTATE )
local_irq_enable ( ) ;
if ( regs - > psw . mask & PSW_MASK_PSTATE ) {
2006-10-27 12:39:22 +02:00
if ( get_user ( * ( ( __u16 * ) opcode ) , ( __u16 __user * ) location ) )
return ;
2005-04-16 15:20:36 -07:00
if ( * ( ( __u16 * ) opcode ) = = S390_BREAKPOINT_U16 ) {
if ( current - > ptrace & PT_PTRACED )
force_sig ( SIGTRAP , current ) ;
else
signal = SIGILL ;
# ifdef CONFIG_MATHEMU
} else if ( opcode [ 0 ] = = 0xb3 ) {
2006-10-27 12:39:22 +02:00
if ( get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) )
return ;
2005-04-16 15:20:36 -07:00
signal = math_emu_b3 ( opcode , regs ) ;
} else if ( opcode [ 0 ] = = 0xed ) {
2006-10-27 12:39:22 +02:00
if ( get_user ( * ( ( __u32 * ) ( opcode + 2 ) ) ,
( __u32 __user * ) ( location + 1 ) ) )
return ;
2005-04-16 15:20:36 -07:00
signal = math_emu_ed ( opcode , regs ) ;
} else if ( * ( ( __u16 * ) opcode ) = = 0xb299 ) {
2006-10-27 12:39:22 +02:00
if ( get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) )
return ;
2005-04-16 15:20:36 -07:00
signal = math_emu_srnm ( opcode , regs ) ;
} else if ( * ( ( __u16 * ) opcode ) = = 0xb29c ) {
2006-10-27 12:39:22 +02:00
if ( get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) )
return ;
2005-04-16 15:20:36 -07:00
signal = math_emu_stfpc ( opcode , regs ) ;
} else if ( * ( ( __u16 * ) opcode ) = = 0xb29d ) {
2006-10-27 12:39:22 +02:00
if ( get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) )
return ;
2005-04-16 15:20:36 -07:00
signal = math_emu_lfpc ( opcode , regs ) ;
# endif
} else
signal = SIGILL ;
2007-02-05 21:17:29 +01:00
} else {
/*
* If we get an illegal op in kernel mode , send it through the
* kprobes notifier . If kprobes doesn ' t pick it up , SIGILL
*/
if ( notify_die ( DIE_BPT , " bpt " , regs , interruption_code ,
3 , SIGTRAP ) ! = NOTIFY_STOP )
signal = SIGILL ;
}
2005-04-16 15:20:36 -07:00
# ifdef CONFIG_MATHEMU
if ( signal = = SIGFPE )
do_fp_trap ( regs , location ,
current - > thread . fp_regs . fpc , interruption_code ) ;
else if ( signal = = SIGSEGV ) {
info . si_signo = signal ;
info . si_errno = 0 ;
info . si_code = SEGV_MAPERR ;
2006-10-09 20:28:03 +01:00
info . si_addr = ( void __user * ) location ;
2005-04-16 15:20:36 -07:00
do_trap ( interruption_code , signal ,
" user address fault " , regs , & info ) ;
} else
# endif
if ( signal ) {
info . si_signo = signal ;
info . si_errno = 0 ;
info . si_code = ILL_ILLOPC ;
2006-02-01 06:55:59 -05:00
info . si_addr = ( void __user * ) location ;
2005-04-16 15:20:36 -07:00
do_trap ( interruption_code , signal ,
" illegal operation " , regs , & info ) ;
}
}
# ifdef CONFIG_MATHEMU
asmlinkage void
specification_exception ( struct pt_regs * regs , long interruption_code )
{
__u8 opcode [ 6 ] ;
2006-10-09 20:28:03 +01:00
__u16 __user * location = NULL ;
2005-04-16 15:20:36 -07:00
int signal = 0 ;
2006-10-09 20:28:03 +01:00
location = ( __u16 __user * ) get_check_address ( regs ) ;
2005-04-16 15:20:36 -07:00
/*
* We got all needed information from the lowcore and can
* now safely switch on interrupts .
*/
if ( regs - > psw . mask & PSW_MASK_PSTATE )
local_irq_enable ( ) ;
if ( regs - > psw . mask & PSW_MASK_PSTATE ) {
get_user ( * ( ( __u16 * ) opcode ) , location ) ;
switch ( opcode [ 0 ] ) {
case 0x28 : /* LDR Rx,Ry */
signal = math_emu_ldr ( opcode ) ;
break ;
case 0x38 : /* LER Rx,Ry */
signal = math_emu_ler ( opcode ) ;
break ;
case 0x60 : /* STD R,D(X,B) */
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_std ( opcode , regs ) ;
break ;
case 0x68 : /* LD R,D(X,B) */
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_ld ( opcode , regs ) ;
break ;
case 0x70 : /* STE R,D(X,B) */
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_ste ( opcode , regs ) ;
break ;
case 0x78 : /* LE R,D(X,B) */
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_le ( opcode , regs ) ;
break ;
default :
signal = SIGILL ;
break ;
}
} else
signal = SIGILL ;
if ( signal = = SIGFPE )
do_fp_trap ( regs , location ,
current - > thread . fp_regs . fpc , interruption_code ) ;
else if ( signal ) {
siginfo_t info ;
info . si_signo = signal ;
info . si_errno = 0 ;
info . si_code = ILL_ILLOPN ;
info . si_addr = location ;
do_trap ( interruption_code , signal ,
" specification exception " , regs , & info ) ;
}
}
# else
DO_ERROR_INFO ( SIGILL , " specification exception " , specification_exception ,
ILL_ILLOPN , get_check_address ( regs ) ) ;
# endif
2007-02-05 21:16:47 +01:00
static void data_exception ( struct pt_regs * regs , long interruption_code )
2005-04-16 15:20:36 -07:00
{
2006-07-12 16:41:55 +02:00
__u16 __user * location ;
2005-04-16 15:20:36 -07:00
int signal = 0 ;
2006-07-12 16:41:55 +02:00
location = get_check_address ( regs ) ;
2005-04-16 15:20:36 -07:00
/*
* We got all needed information from the lowcore and can
* now safely switch on interrupts .
*/
if ( regs - > psw . mask & PSW_MASK_PSTATE )
local_irq_enable ( ) ;
if ( MACHINE_HAS_IEEE )
2006-09-28 16:56:43 +02:00
asm volatile ( " stfpc %0 " : " =m " ( current - > thread . fp_regs . fpc ) ) ;
2005-04-16 15:20:36 -07:00
# ifdef CONFIG_MATHEMU
else if ( regs - > psw . mask & PSW_MASK_PSTATE ) {
__u8 opcode [ 6 ] ;
get_user ( * ( ( __u16 * ) opcode ) , location ) ;
switch ( opcode [ 0 ] ) {
case 0x28 : /* LDR Rx,Ry */
signal = math_emu_ldr ( opcode ) ;
break ;
case 0x38 : /* LER Rx,Ry */
signal = math_emu_ler ( opcode ) ;
break ;
case 0x60 : /* STD R,D(X,B) */
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_std ( opcode , regs ) ;
break ;
case 0x68 : /* LD R,D(X,B) */
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_ld ( opcode , regs ) ;
break ;
case 0x70 : /* STE R,D(X,B) */
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_ste ( opcode , regs ) ;
break ;
case 0x78 : /* LE R,D(X,B) */
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_le ( opcode , regs ) ;
break ;
case 0xb3 :
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_b3 ( opcode , regs ) ;
break ;
case 0xed :
get_user ( * ( ( __u32 * ) ( opcode + 2 ) ) ,
2006-10-09 20:28:03 +01:00
( __u32 __user * ) ( location + 1 ) ) ;
2005-04-16 15:20:36 -07:00
signal = math_emu_ed ( opcode , regs ) ;
break ;
case 0xb2 :
if ( opcode [ 1 ] = = 0x99 ) {
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_srnm ( opcode , regs ) ;
} else if ( opcode [ 1 ] = = 0x9c ) {
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_stfpc ( opcode , regs ) ;
} else if ( opcode [ 1 ] = = 0x9d ) {
get_user ( * ( ( __u16 * ) ( opcode + 2 ) ) , location + 1 ) ;
signal = math_emu_lfpc ( opcode , regs ) ;
} else
signal = SIGILL ;
break ;
default :
signal = SIGILL ;
break ;
}
}
# endif
if ( current - > thread . fp_regs . fpc & FPC_DXC_MASK )
signal = SIGFPE ;
else
signal = SIGILL ;
if ( signal = = SIGFPE )
do_fp_trap ( regs , location ,
current - > thread . fp_regs . fpc , interruption_code ) ;
else if ( signal ) {
siginfo_t info ;
info . si_signo = signal ;
info . si_errno = 0 ;
info . si_code = ILL_ILLOPN ;
info . si_addr = location ;
do_trap ( interruption_code , signal ,
" data exception " , regs , & info ) ;
}
}
2007-02-05 21:16:47 +01:00
static void space_switch_exception ( struct pt_regs * regs , long int_code )
2005-04-16 15:20:36 -07:00
{
siginfo_t info ;
/* Set user psw back to home space mode. */
if ( regs - > psw . mask & PSW_MASK_PSTATE )
regs - > psw . mask | = PSW_ASC_HOME ;
/* Send SIGILL. */
info . si_signo = SIGILL ;
info . si_errno = 0 ;
info . si_code = ILL_PRVOPC ;
info . si_addr = get_check_address ( regs ) ;
do_trap ( int_code , SIGILL , " space switch event " , regs , & info ) ;
}
asmlinkage void kernel_stack_overflow ( struct pt_regs * regs )
{
2005-06-21 17:16:28 -07:00
bust_spinlocks ( 1 ) ;
printk ( " Kernel stack overflow. \n " ) ;
show_regs ( regs ) ;
bust_spinlocks ( 0 ) ;
2005-04-16 15:20:36 -07:00
panic ( " Corrupt kernel stack, can't continue. " ) ;
}
/* init is done in lowcore.S and head.S */
void __init trap_init ( void )
{
int i ;
for ( i = 0 ; i < 128 ; i + + )
pgm_check_table [ i ] = & default_trap_handler ;
pgm_check_table [ 1 ] = & illegal_op ;
pgm_check_table [ 2 ] = & privileged_op ;
pgm_check_table [ 3 ] = & execute_exception ;
pgm_check_table [ 4 ] = & do_protection_exception ;
pgm_check_table [ 5 ] = & addressing_exception ;
pgm_check_table [ 6 ] = & specification_exception ;
pgm_check_table [ 7 ] = & data_exception ;
pgm_check_table [ 8 ] = & overflow_exception ;
pgm_check_table [ 9 ] = & divide_exception ;
pgm_check_table [ 0x0A ] = & overflow_exception ;
pgm_check_table [ 0x0B ] = & divide_exception ;
pgm_check_table [ 0x0C ] = & hfp_overflow_exception ;
pgm_check_table [ 0x0D ] = & hfp_underflow_exception ;
pgm_check_table [ 0x0E ] = & hfp_significance_exception ;
pgm_check_table [ 0x0F ] = & hfp_divide_exception ;
pgm_check_table [ 0x10 ] = & do_dat_exception ;
pgm_check_table [ 0x11 ] = & do_dat_exception ;
pgm_check_table [ 0x12 ] = & translation_exception ;
pgm_check_table [ 0x13 ] = & special_op_exception ;
2006-01-06 00:19:28 -08:00
# ifdef CONFIG_64BIT
2005-04-16 15:20:36 -07:00
pgm_check_table [ 0x38 ] = & do_dat_exception ;
pgm_check_table [ 0x39 ] = & do_dat_exception ;
pgm_check_table [ 0x3A ] = & do_dat_exception ;
pgm_check_table [ 0x3B ] = & do_dat_exception ;
2006-01-06 00:19:28 -08:00
# endif /* CONFIG_64BIT */
2005-04-16 15:20:36 -07:00
pgm_check_table [ 0x15 ] = & operand_exception ;
pgm_check_table [ 0x1C ] = & space_switch_exception ;
pgm_check_table [ 0x1D ] = & hfp_sqrt_exception ;
pgm_check_table [ 0x40 ] = & do_monitor_call ;
2006-12-04 15:40:40 +01:00
pfault_irq_init ( ) ;
2005-04-16 15:20:36 -07:00
}