2005-04-17 02:20:36 +04:00
/*
* 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 .
*
2006-02-15 00:04:54 +03:00
* Copyright ( C ) 1994 - 1999 , 2000 , 01 , 06 Ralf Baechle
2005-04-17 02:20:36 +04:00
* Copyright ( C ) 1995 , 1996 Paul M . Antoine
* Copyright ( C ) 1998 Ulf Carlsson
* Copyright ( C ) 1999 Silicon Graphics , Inc .
* Kevin D . Kissell , kevink @ mips . com and Carsten Langgaard , carstenl @ mips . com
* Copyright ( C ) 2000 , 01 MIPS Technologies , Inc .
2005-06-23 00:43:29 +04:00
* Copyright ( C ) 2002 , 2003 , 2004 , 2005 Maciej W . Rozycki
2005-04-17 02:20:36 +04:00
*/
2007-05-31 17:00:19 +04:00
# include <linux/bug.h>
2005-04-17 02:20:36 +04:00
# include <linux/init.h>
# include <linux/mm.h>
# include <linux/module.h>
# include <linux/sched.h>
# include <linux/smp.h>
# include <linux/spinlock.h>
# include <linux/kallsyms.h>
2005-07-14 19:57:16 +04:00
# include <linux/bootmem.h>
2006-07-20 20:52:02 +04:00
# include <linux/interrupt.h>
2005-04-17 02:20:36 +04:00
# include <asm/bootinfo.h>
# include <asm/branch.h>
# include <asm/break.h>
# include <asm/cpu.h>
2005-05-31 15:49:19 +04:00
# include <asm/dsp.h>
2005-04-17 02:20:36 +04:00
# include <asm/fpu.h>
2005-08-17 21:44:08 +04:00
# include <asm/mipsregs.h>
# include <asm/mipsmtregs.h>
2005-04-17 02:20:36 +04:00
# include <asm/module.h>
# include <asm/pgtable.h>
# include <asm/ptrace.h>
# include <asm/sections.h>
# include <asm/system.h>
# include <asm/tlbdebug.h>
# include <asm/traps.h>
# include <asm/uaccess.h>
# include <asm/mmu_context.h>
# include <asm/watch.h>
# include <asm/types.h>
2006-09-26 18:44:01 +04:00
# include <asm/stacktrace.h>
2005-04-17 02:20:36 +04:00
2006-04-03 20:56:36 +04:00
extern asmlinkage void handle_int ( void ) ;
2005-04-17 02:20:36 +04:00
extern asmlinkage void handle_tlbm ( void ) ;
extern asmlinkage void handle_tlbl ( void ) ;
extern asmlinkage void handle_tlbs ( void ) ;
extern asmlinkage void handle_adel ( void ) ;
extern asmlinkage void handle_ades ( void ) ;
extern asmlinkage void handle_ibe ( void ) ;
extern asmlinkage void handle_dbe ( void ) ;
extern asmlinkage void handle_sys ( void ) ;
extern asmlinkage void handle_bp ( void ) ;
extern asmlinkage void handle_ri ( void ) ;
2006-09-11 12:50:29 +04:00
extern asmlinkage void handle_ri_rdhwr_vivt ( void ) ;
extern asmlinkage void handle_ri_rdhwr ( void ) ;
2005-04-17 02:20:36 +04:00
extern asmlinkage void handle_cpu ( void ) ;
extern asmlinkage void handle_ov ( void ) ;
extern asmlinkage void handle_tr ( void ) ;
extern asmlinkage void handle_fpe ( void ) ;
extern asmlinkage void handle_mdmx ( void ) ;
extern asmlinkage void handle_watch ( void ) ;
2005-08-17 21:44:08 +04:00
extern asmlinkage void handle_mt ( void ) ;
2005-05-31 15:49:19 +04:00
extern asmlinkage void handle_dsp ( void ) ;
2005-04-17 02:20:36 +04:00
extern asmlinkage void handle_mcheck ( void ) ;
extern asmlinkage void handle_reserved ( void ) ;
2005-10-18 13:26:46 +04:00
extern int fpu_emulator_cop1Handler ( struct pt_regs * xcp ,
2006-10-08 19:10:01 +04:00
struct mips_fpu_struct * ctx , int has_fpu ) ;
2005-04-17 02:20:36 +04:00
void ( * board_be_init ) ( void ) ;
int ( * board_be_handler ) ( struct pt_regs * regs , int is_fixup ) ;
2005-07-14 19:57:16 +04:00
void ( * board_nmi_handler_setup ) ( void ) ;
void ( * board_ejtag_handler_setup ) ( void ) ;
void ( * board_bind_eic_interrupt ) ( int irq , int regset ) ;
2005-04-17 02:20:36 +04:00
2006-08-03 11:29:21 +04:00
static void show_raw_backtrace ( unsigned long reg29 )
2006-07-25 18:51:36 +04:00
{
2006-08-03 11:29:21 +04:00
unsigned long * sp = ( unsigned long * ) reg29 ;
2006-07-25 18:51:36 +04:00
unsigned long addr ;
printk ( " Call Trace: " ) ;
# ifdef CONFIG_KALLSYMS
printk ( " \n " ) ;
# endif
2006-08-03 11:29:17 +04:00
while ( ! kstack_end ( sp ) ) {
addr = * sp + + ;
if ( __kernel_text_address ( addr ) )
print_ip_sym ( addr ) ;
2006-07-25 18:51:36 +04:00
}
printk ( " \n " ) ;
}
2006-07-29 18:27:20 +04:00
# ifdef CONFIG_KALLSYMS
2006-09-26 18:44:01 +04:00
int raw_show_trace ;
2006-07-29 18:27:20 +04:00
static int __init set_raw_show_trace ( char * str )
{
raw_show_trace = 1 ;
return 1 ;
}
__setup ( " raw_show_trace " , set_raw_show_trace ) ;
2006-09-26 18:44:01 +04:00
# endif
2006-08-03 11:29:21 +04:00
2006-08-03 11:29:17 +04:00
static void show_backtrace ( struct task_struct * task , struct pt_regs * regs )
2006-07-29 18:27:20 +04:00
{
2006-08-03 11:29:21 +04:00
unsigned long sp = regs - > regs [ 29 ] ;
unsigned long ra = regs - > regs [ 31 ] ;
2006-07-29 18:27:20 +04:00
unsigned long pc = regs - > cp0_epc ;
if ( raw_show_trace | | ! __kernel_text_address ( pc ) ) {
2006-08-03 11:29:17 +04:00
show_raw_backtrace ( sp ) ;
2006-07-29 18:27:20 +04:00
return ;
}
printk ( " Call Trace: \n " ) ;
2006-08-03 11:29:21 +04:00
do {
2006-08-03 11:29:17 +04:00
print_ip_sym ( pc ) ;
2006-09-29 13:02:51 +04:00
pc = unwind_stack ( task , & sp , pc , & ra ) ;
2006-08-03 11:29:21 +04:00
} while ( pc ) ;
2006-07-29 18:27:20 +04:00
printk ( " \n " ) ;
}
2005-04-17 02:20:36 +04:00
/*
* This routine abuses get_user ( ) / put_user ( ) to reference pointers
* with at least a bit of error checking . . .
*/
2006-07-29 18:27:20 +04:00
static void show_stacktrace ( struct task_struct * task , struct pt_regs * regs )
2005-04-17 02:20:36 +04:00
{
const int field = 2 * sizeof ( unsigned long ) ;
long stackdata ;
int i ;
2006-07-29 18:27:20 +04:00
unsigned long * sp = ( unsigned long * ) regs - > regs [ 29 ] ;
2005-04-17 02:20:36 +04:00
printk ( " Stack : " ) ;
i = 0 ;
while ( ( unsigned long ) sp & ( PAGE_SIZE - 1 ) ) {
if ( i & & ( ( i % ( 64 / field ) ) = = 0 ) )
printk ( " \n " ) ;
if ( i > 39 ) {
printk ( " ... " ) ;
break ;
}
if ( __get_user ( stackdata , sp + + ) ) {
printk ( " (Bad stack address) " ) ;
break ;
}
printk ( " %0*lx " , field , stackdata ) ;
i + + ;
}
printk ( " \n " ) ;
2006-08-03 11:29:17 +04:00
show_backtrace ( task , regs ) ;
2006-07-29 18:27:20 +04:00
}
void show_stack ( struct task_struct * task , unsigned long * sp )
{
struct pt_regs regs ;
if ( sp ) {
regs . regs [ 29 ] = ( unsigned long ) sp ;
regs . regs [ 31 ] = 0 ;
regs . cp0_epc = 0 ;
} else {
if ( task & & task ! = current ) {
regs . regs [ 29 ] = task - > thread . reg29 ;
regs . regs [ 31 ] = 0 ;
regs . cp0_epc = task - > thread . reg31 ;
} else {
prepare_frametrace ( & regs ) ;
}
}
show_stacktrace ( task , & regs ) ;
2005-04-17 02:20:36 +04:00
}
/*
* The architecture - independent dump_stack generator
*/
void dump_stack ( void )
{
2006-08-03 11:29:19 +04:00
struct pt_regs regs ;
2005-04-17 02:20:36 +04:00
2006-08-03 11:29:19 +04:00
prepare_frametrace ( & regs ) ;
show_backtrace ( current , & regs ) ;
2005-04-17 02:20:36 +04:00
}
EXPORT_SYMBOL ( dump_stack ) ;
void show_code ( unsigned int * pc )
{
long i ;
printk ( " \n Code: " ) ;
for ( i = - 3 ; i < 6 ; i + + ) {
unsigned int insn ;
if ( __get_user ( insn , pc + i ) ) {
printk ( " (Bad address in epc) \n " ) ;
break ;
}
printk ( " %c%08x%c " , ( i ? ' ' : ' < ' ) , insn , ( i ? ' ' : ' > ' ) ) ;
}
}
void show_regs ( struct pt_regs * regs )
{
const int field = 2 * sizeof ( unsigned long ) ;
unsigned int cause = regs - > cp0_cause ;
int i ;
printk ( " Cpu %d \n " , smp_processor_id ( ) ) ;
/*
* Saved main processor registers
*/
for ( i = 0 ; i < 32 ; ) {
if ( ( i % 4 ) = = 0 )
printk ( " $%2d : " , i ) ;
if ( i = = 0 )
printk ( " %0*lx " , field , 0UL ) ;
else if ( i = = 26 | | i = = 27 )
printk ( " %*s " , field , " " ) ;
else
printk ( " %0*lx " , field , regs - > regs [ i ] ) ;
i + + ;
if ( ( i % 4 ) = = 0 )
printk ( " \n " ) ;
}
2007-02-02 19:41:47 +03:00
# ifdef CONFIG_CPU_HAS_SMARTMIPS
printk ( " Acx : %0*lx \n " , field , regs - > acx ) ;
# endif
2005-04-17 02:20:36 +04:00
printk ( " Hi : %0*lx \n " , field , regs - > hi ) ;
printk ( " Lo : %0*lx \n " , field , regs - > lo ) ;
/*
* Saved cp0 registers
*/
printk ( " epc : %0*lx " , field , regs - > cp0_epc ) ;
print_symbol ( " %s " , regs - > cp0_epc ) ;
printk ( " %s \n " , print_tainted ( ) ) ;
printk ( " ra : %0*lx " , field , regs - > regs [ 31 ] ) ;
print_symbol ( " %s \n " , regs - > regs [ 31 ] ) ;
printk ( " Status: %08x " , ( uint32_t ) regs - > cp0_status ) ;
2005-06-23 00:43:29 +04:00
if ( current_cpu_data . isa_level = = MIPS_CPU_ISA_I ) {
if ( regs - > cp0_status & ST0_KUO )
printk ( " KUo " ) ;
if ( regs - > cp0_status & ST0_IEO )
printk ( " IEo " ) ;
if ( regs - > cp0_status & ST0_KUP )
printk ( " KUp " ) ;
if ( regs - > cp0_status & ST0_IEP )
printk ( " IEp " ) ;
if ( regs - > cp0_status & ST0_KUC )
printk ( " KUc " ) ;
if ( regs - > cp0_status & ST0_IEC )
printk ( " IEc " ) ;
} else {
if ( regs - > cp0_status & ST0_KX )
printk ( " KX " ) ;
if ( regs - > cp0_status & ST0_SX )
printk ( " SX " ) ;
if ( regs - > cp0_status & ST0_UX )
printk ( " UX " ) ;
switch ( regs - > cp0_status & ST0_KSU ) {
case KSU_USER :
printk ( " USER " ) ;
break ;
case KSU_SUPERVISOR :
printk ( " SUPERVISOR " ) ;
break ;
case KSU_KERNEL :
printk ( " KERNEL " ) ;
break ;
default :
printk ( " BAD_MODE " ) ;
break ;
}
if ( regs - > cp0_status & ST0_ERL )
printk ( " ERL " ) ;
if ( regs - > cp0_status & ST0_EXL )
printk ( " EXL " ) ;
if ( regs - > cp0_status & ST0_IE )
printk ( " IE " ) ;
2005-04-17 02:20:36 +04:00
}
printk ( " \n " ) ;
printk ( " Cause : %08x \n " , cause ) ;
cause = ( cause & CAUSEF_EXCCODE ) > > CAUSEB_EXCCODE ;
if ( 1 < = cause & & cause < = 5 )
printk ( " BadVA : %0*lx \n " , field , regs - > cp0_badvaddr ) ;
printk ( " PrId : %08x \n " , read_c0_prid ( ) ) ;
}
void show_registers ( struct pt_regs * regs )
{
show_regs ( regs ) ;
print_modules ( ) ;
printk ( " Process %s (pid: %d, threadinfo=%p, task=%p) \n " ,
current - > comm , current - > pid , current_thread_info ( ) , current ) ;
2006-07-29 18:27:20 +04:00
show_stacktrace ( current , regs ) ;
2005-04-17 02:20:36 +04:00
show_code ( ( unsigned int * ) regs - > cp0_epc ) ;
printk ( " \n " ) ;
}
static DEFINE_SPINLOCK ( die_lock ) ;
2005-10-13 20:07:54 +04:00
NORET_TYPE void ATTRIB_NORET die ( const char * str , struct pt_regs * regs )
2005-04-17 02:20:36 +04:00
{
static int die_counter ;
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_MIPS_MT_SMTC
unsigned long dvpret = dvpe ( ) ;
# endif /* CONFIG_MIPS_MT_SMTC */
2005-04-17 02:20:36 +04:00
console_verbose ( ) ;
spin_lock_irq ( & die_lock ) ;
2006-04-05 12:45:45 +04:00
bust_spinlocks ( 1 ) ;
# ifdef CONFIG_MIPS_MT_SMTC
mips_mt_regdump ( dvpret ) ;
# endif /* CONFIG_MIPS_MT_SMTC */
2005-10-13 20:07:54 +04:00
printk ( " %s[#%d]: \n " , str , + + die_counter ) ;
2005-04-17 02:20:36 +04:00
show_registers ( regs ) ;
spin_unlock_irq ( & die_lock ) ;
2006-07-20 20:52:02 +04:00
if ( in_interrupt ( ) )
panic ( " Fatal exception in interrupt " ) ;
if ( panic_on_oops ) {
printk ( KERN_EMERG " Fatal exception: panic in 5 seconds \n " ) ;
ssleep ( 5 ) ;
panic ( " Fatal exception " ) ;
}
2005-04-17 02:20:36 +04:00
do_exit ( SIGSEGV ) ;
}
extern const struct exception_table_entry __start___dbe_table [ ] ;
extern const struct exception_table_entry __stop___dbe_table [ ] ;
2007-02-18 18:57:09 +03:00
__asm__ (
" .section __dbe_table, \" a \" \n "
" .previous \n " ) ;
2005-04-17 02:20:36 +04:00
/* Given an address, look for it in the exception tables. */
static const struct exception_table_entry * search_dbe_tables ( unsigned long addr )
{
const struct exception_table_entry * e ;
e = search_extable ( __start___dbe_table , __stop___dbe_table - 1 , addr ) ;
if ( ! e )
e = search_module_dbetables ( addr ) ;
return e ;
}
asmlinkage void do_be ( struct pt_regs * regs )
{
const int field = 2 * sizeof ( unsigned long ) ;
const struct exception_table_entry * fixup = NULL ;
int data = regs - > cp0_cause & 4 ;
int action = MIPS_BE_FATAL ;
/* XXX For now. Fixme, this searches the wrong table ... */
if ( data & & ! user_mode ( regs ) )
fixup = search_dbe_tables ( exception_epc ( regs ) ) ;
if ( fixup )
action = MIPS_BE_FIXUP ;
if ( board_be_handler )
action = board_be_handler ( regs , fixup ! = 0 ) ;
switch ( action ) {
case MIPS_BE_DISCARD :
return ;
case MIPS_BE_FIXUP :
if ( fixup ) {
regs - > cp0_epc = fixup - > nextinsn ;
return ;
}
break ;
default :
break ;
}
/*
* Assume it would be too dangerous to continue . . .
*/
printk ( KERN_ALERT " %s bus error, epc == %0*lx, ra == %0*lx \n " ,
data ? " Data " : " Instruction " ,
field , regs - > cp0_epc , field , regs - > regs [ 31 ] ) ;
die_if_kernel ( " Oops " , regs ) ;
force_sig ( SIGBUS , current ) ;
}
/*
* ll / sc emulation
*/
# define OPCODE 0xfc000000
# define BASE 0x03e00000
# define RT 0x001f0000
# define OFFSET 0x0000ffff
# define LL 0xc0000000
# define SC 0xe0000000
2005-04-13 21:43:59 +04:00
# define SPEC3 0x7c000000
# define RD 0x0000f800
# define FUNC 0x0000003f
# define RDHWR 0x0000003b
2005-04-17 02:20:36 +04:00
/*
* The ll_bit is cleared by r * _switch . S
*/
unsigned long ll_bit ;
static struct task_struct * ll_task = NULL ;
static inline void simulate_ll ( struct pt_regs * regs , unsigned int opcode )
{
2005-03-01 22:22:29 +03:00
unsigned long value , __user * vaddr ;
2005-04-17 02:20:36 +04:00
long offset ;
int signal = 0 ;
/*
* analyse the ll instruction that just caused a ri exception
* and put the referenced address to addr .
*/
/* sign extend offset */
offset = opcode & OFFSET ;
offset < < = 16 ;
offset > > = 16 ;
2005-03-01 22:22:29 +03:00
vaddr = ( unsigned long __user * )
( ( unsigned long ) ( regs - > regs [ ( opcode & BASE ) > > 21 ] ) + offset ) ;
2005-04-17 02:20:36 +04:00
if ( ( unsigned long ) vaddr & 3 ) {
signal = SIGBUS ;
goto sig ;
}
if ( get_user ( value , vaddr ) ) {
signal = SIGSEGV ;
goto sig ;
}
preempt_disable ( ) ;
if ( ll_task = = NULL | | ll_task = = current ) {
ll_bit = 1 ;
} else {
ll_bit = 0 ;
}
ll_task = current ;
preempt_enable ( ) ;
2005-04-12 15:04:15 +04:00
compute_return_epc ( regs ) ;
2005-04-17 02:20:36 +04:00
regs - > regs [ ( opcode & RT ) > > 16 ] = value ;
return ;
sig :
force_sig ( signal , current ) ;
}
static inline void simulate_sc ( struct pt_regs * regs , unsigned int opcode )
{
2005-03-01 22:22:29 +03:00
unsigned long __user * vaddr ;
unsigned long reg ;
2005-04-17 02:20:36 +04:00
long offset ;
int signal = 0 ;
/*
* analyse the sc instruction that just caused a ri exception
* and put the referenced address to addr .
*/
/* sign extend offset */
offset = opcode & OFFSET ;
offset < < = 16 ;
offset > > = 16 ;
2005-03-01 22:22:29 +03:00
vaddr = ( unsigned long __user * )
( ( unsigned long ) ( regs - > regs [ ( opcode & BASE ) > > 21 ] ) + offset ) ;
2005-04-17 02:20:36 +04:00
reg = ( opcode & RT ) > > 16 ;
if ( ( unsigned long ) vaddr & 3 ) {
signal = SIGBUS ;
goto sig ;
}
preempt_disable ( ) ;
if ( ll_bit = = 0 | | ll_task ! = current ) {
2005-04-13 00:26:05 +04:00
compute_return_epc ( regs ) ;
2005-04-17 02:20:36 +04:00
regs - > regs [ reg ] = 0 ;
preempt_enable ( ) ;
return ;
}
preempt_enable ( ) ;
if ( put_user ( regs - > regs [ reg ] , vaddr ) ) {
signal = SIGSEGV ;
goto sig ;
}
2005-04-12 15:04:15 +04:00
compute_return_epc ( regs ) ;
2005-04-17 02:20:36 +04:00
regs - > regs [ reg ] = 1 ;
return ;
sig :
force_sig ( signal , current ) ;
}
/*
* ll uses the opcode of lwc0 and sc uses the opcode of swc0 . That is both
* opcodes are supposed to result in coprocessor unusable exceptions if
* executed on ll / sc - less processors . That ' s the theory . In practice a
* few processors such as NEC ' s VR4100 throw reserved instruction exceptions
* instead , so we ' re doing the emulation thing in both exception handlers .
*/
static inline int simulate_llsc ( struct pt_regs * regs )
{
unsigned int opcode ;
2006-11-30 04:14:47 +03:00
if ( get_user ( opcode , ( unsigned int __user * ) exception_epc ( regs ) ) )
goto out_sigsegv ;
2005-04-17 02:20:36 +04:00
if ( ( opcode & OPCODE ) = = LL ) {
simulate_ll ( regs , opcode ) ;
return 0 ;
}
if ( ( opcode & OPCODE ) = = SC ) {
simulate_sc ( regs , opcode ) ;
return 0 ;
}
return - EFAULT ; /* Strange things going on ... */
2006-11-30 04:14:47 +03:00
out_sigsegv :
force_sig ( SIGSEGV , current ) ;
return - EFAULT ;
2005-04-17 02:20:36 +04:00
}
2005-04-13 21:43:59 +04:00
/*
* Simulate trapping ' rdhwr ' instructions to provide user accessible
* registers not implemented in hardware . The only current use of this
* is the thread area pointer .
*/
static inline int simulate_rdhwr ( struct pt_regs * regs )
{
2006-01-12 12:06:07 +03:00
struct thread_info * ti = task_thread_info ( current ) ;
2005-04-13 21:43:59 +04:00
unsigned int opcode ;
2006-11-30 04:14:47 +03:00
if ( get_user ( opcode , ( unsigned int __user * ) exception_epc ( regs ) ) )
goto out_sigsegv ;
2005-04-13 21:43:59 +04:00
if ( unlikely ( compute_return_epc ( regs ) ) )
return - EFAULT ;
if ( ( opcode & OPCODE ) = = SPEC3 & & ( opcode & FUNC ) = = RDHWR ) {
int rd = ( opcode & RD ) > > 11 ;
int rt = ( opcode & RT ) > > 16 ;
switch ( rd ) {
case 29 :
regs - > regs [ rt ] = ti - > tp_value ;
2005-11-27 06:34:41 +03:00
return 0 ;
2005-04-13 21:43:59 +04:00
default :
return - EFAULT ;
}
}
2005-11-27 06:34:41 +03:00
/* Not ours. */
return - EFAULT ;
2006-11-30 04:14:47 +03:00
out_sigsegv :
force_sig ( SIGSEGV , current ) ;
return - EFAULT ;
2005-04-13 21:43:59 +04:00
}
2005-04-17 02:20:36 +04:00
asmlinkage void do_ov ( struct pt_regs * regs )
{
siginfo_t info ;
2006-02-15 00:04:54 +03:00
die_if_kernel ( " Integer overflow " , regs ) ;
2005-04-17 02:20:36 +04:00
info . si_code = FPE_INTOVF ;
info . si_signo = SIGFPE ;
info . si_errno = 0 ;
2005-03-01 22:22:29 +03:00
info . si_addr = ( void __user * ) regs - > cp0_epc ;
2005-04-17 02:20:36 +04:00
force_sig_info ( SIGFPE , & info , current ) ;
}
/*
* XXX Delayed fp exceptions when doing a lazy ctx switch XXX
*/
asmlinkage void do_fpe ( struct pt_regs * regs , unsigned long fcr31 )
{
2006-07-01 02:35:28 +04:00
die_if_kernel ( " FP exception in kernel code " , regs ) ;
2005-04-17 02:20:36 +04:00
if ( fcr31 & FPU_CSR_UNI_X ) {
int sig ;
/*
2006-03-11 11:18:41 +03:00
* Unimplemented operation exception . If we ' ve got the full
2005-04-17 02:20:36 +04:00
* software emulator on - board , let ' s use it . . .
*
* Force FPU to dump state into task / thread context . We ' re
* moving a lot of data here for what is probably a single
* instruction , but the alternative is to pre - decode the FP
* register operands before invoking the emulator , which seems
* a bit extreme for what should be an infrequent event .
*/
2005-04-28 17:39:10 +04:00
/* Ensure 'resume' not overwrite saved fp context again. */
2007-03-09 19:07:45 +03:00
lose_fpu ( 1 ) ;
2005-04-17 02:20:36 +04:00
/* Run the emulator */
2006-10-08 19:10:01 +04:00
sig = fpu_emulator_cop1Handler ( regs , & current - > thread . fpu , 1 ) ;
2005-04-17 02:20:36 +04:00
/*
* We can ' t allow the emulated instruction to leave any of
* the cause bit set in $ fcr31 .
*/
2006-05-15 20:26:03 +04:00
current - > thread . fpu . fcr31 & = ~ FPU_CSR_ALL_X ;
2005-04-17 02:20:36 +04:00
/* Restore the hardware register state */
2007-03-09 19:07:45 +03:00
own_fpu ( 1 ) ; /* Using the FPU again. */
2005-04-17 02:20:36 +04:00
/* If something went wrong, signal */
if ( sig )
force_sig ( sig , current ) ;
return ;
}
force_sig ( SIGFPE , current ) ;
}
asmlinkage void do_bp ( struct pt_regs * regs )
{
unsigned int opcode , bcode ;
siginfo_t info ;
2007-04-12 15:02:54 +04:00
if ( __get_user ( opcode , ( unsigned int __user * ) exception_epc ( regs ) ) )
2006-11-30 04:14:47 +03:00
goto out_sigsegv ;
2005-04-17 02:20:36 +04:00
/*
* There is the ancient bug in the MIPS assemblers that the break
* code starts left to bit 16 instead to bit 6 in the opcode .
* Gas is bug - compatible , but not always , grrr . . .
* We handle both cases with a simple heuristics . - - macro
*/
bcode = ( ( opcode > > 6 ) & ( ( 1 < < 20 ) - 1 ) ) ;
if ( bcode < ( 1 < < 10 ) )
bcode < < = 10 ;
/*
* ( A short test says that IRIX 5.3 sends SIGTRAP for all break
* insns , even for break codes that indicate arithmetic failures .
* Weird . . . )
* But should we continue the brokenness ? ? ? - - macro
*/
switch ( bcode ) {
case BRK_OVERFLOW < < 10 :
case BRK_DIVZERO < < 10 :
2006-10-16 04:38:50 +04:00
die_if_kernel ( " Break instruction in kernel code " , regs ) ;
2005-04-17 02:20:36 +04:00
if ( bcode = = ( BRK_DIVZERO < < 10 ) )
info . si_code = FPE_INTDIV ;
else
info . si_code = FPE_INTOVF ;
info . si_signo = SIGFPE ;
info . si_errno = 0 ;
2005-03-01 22:22:29 +03:00
info . si_addr = ( void __user * ) regs - > cp0_epc ;
2005-04-17 02:20:36 +04:00
force_sig_info ( SIGFPE , & info , current ) ;
break ;
2006-10-16 04:38:50 +04:00
case BRK_BUG :
die ( " Kernel bug detected " , regs ) ;
break ;
2005-04-17 02:20:36 +04:00
default :
2006-10-16 04:38:50 +04:00
die_if_kernel ( " Break instruction in kernel code " , regs ) ;
2005-04-17 02:20:36 +04:00
force_sig ( SIGTRAP , current ) ;
}
2007-02-06 10:02:21 +03:00
return ;
2006-11-30 04:14:47 +03:00
out_sigsegv :
force_sig ( SIGSEGV , current ) ;
2005-04-17 02:20:36 +04:00
}
asmlinkage void do_tr ( struct pt_regs * regs )
{
unsigned int opcode , tcode = 0 ;
siginfo_t info ;
2007-04-12 15:02:54 +04:00
if ( __get_user ( opcode , ( unsigned int __user * ) exception_epc ( regs ) ) )
2006-11-30 04:14:47 +03:00
goto out_sigsegv ;
2005-04-17 02:20:36 +04:00
/* Immediate versions don't provide a code. */
if ( ! ( opcode & OPCODE ) )
tcode = ( ( opcode > > 6 ) & ( ( 1 < < 10 ) - 1 ) ) ;
/*
* ( A short test says that IRIX 5.3 sends SIGTRAP for all trap
* insns , even for trap codes that indicate arithmetic failures .
* Weird . . . )
* But should we continue the brokenness ? ? ? - - macro
*/
switch ( tcode ) {
case BRK_OVERFLOW :
case BRK_DIVZERO :
2006-10-16 04:38:50 +04:00
die_if_kernel ( " Trap instruction in kernel code " , regs ) ;
2005-04-17 02:20:36 +04:00
if ( tcode = = BRK_DIVZERO )
info . si_code = FPE_INTDIV ;
else
info . si_code = FPE_INTOVF ;
info . si_signo = SIGFPE ;
info . si_errno = 0 ;
2005-03-01 22:22:29 +03:00
info . si_addr = ( void __user * ) regs - > cp0_epc ;
2005-04-17 02:20:36 +04:00
force_sig_info ( SIGFPE , & info , current ) ;
break ;
2006-10-16 04:38:50 +04:00
case BRK_BUG :
die ( " Kernel bug detected " , regs ) ;
break ;
2005-04-17 02:20:36 +04:00
default :
2006-10-16 04:38:50 +04:00
die_if_kernel ( " Trap instruction in kernel code " , regs ) ;
2005-04-17 02:20:36 +04:00
force_sig ( SIGTRAP , current ) ;
}
2007-02-06 10:02:21 +03:00
return ;
2006-11-30 04:14:47 +03:00
out_sigsegv :
force_sig ( SIGSEGV , current ) ;
2005-04-17 02:20:36 +04:00
}
asmlinkage void do_ri ( struct pt_regs * regs )
{
die_if_kernel ( " Reserved instruction in kernel code " , regs ) ;
if ( ! cpu_has_llsc )
if ( ! simulate_llsc ( regs ) )
return ;
2005-04-13 21:43:59 +04:00
if ( ! simulate_rdhwr ( regs ) )
return ;
2005-04-17 02:20:36 +04:00
force_sig ( SIGILL , current ) ;
}
asmlinkage void do_cpu ( struct pt_regs * regs )
{
unsigned int cpid ;
2007-04-13 21:37:26 +04:00
die_if_kernel ( " do_cpu invoked from kernel context! " , regs ) ;
2005-04-17 02:20:36 +04:00
cpid = ( regs - > cp0_cause > > CAUSEB_CE ) & 3 ;
switch ( cpid ) {
case 0 :
2005-04-13 21:43:59 +04:00
if ( ! cpu_has_llsc )
if ( ! simulate_llsc ( regs ) )
return ;
2005-04-17 02:20:36 +04:00
2005-04-13 21:43:59 +04:00
if ( ! simulate_rdhwr ( regs ) )
2005-04-17 02:20:36 +04:00
return ;
2005-04-13 21:43:59 +04:00
2005-04-17 02:20:36 +04:00
break ;
case 1 :
2007-03-09 19:07:45 +03:00
if ( used_math ( ) ) /* Using the FPU again. */
own_fpu ( 1 ) ;
else { /* First time FPU user. */
2005-04-17 02:20:36 +04:00
init_fpu ( ) ;
set_used_math ( ) ;
}
2007-04-13 21:37:26 +04:00
if ( ! raw_cpu_has_fpu ) {
2006-10-08 19:10:01 +04:00
int sig ;
sig = fpu_emulator_cop1Handler ( regs ,
& current - > thread . fpu , 0 ) ;
2005-04-17 02:20:36 +04:00
if ( sig )
force_sig ( sig , current ) ;
2006-04-05 12:45:47 +04:00
# ifdef CONFIG_MIPS_MT_FPAFF
else {
/*
* MIPS MT processors may have fewer FPU contexts
* than CPU threads . If we ' ve emulated more than
* some threshold number of instructions , force
* migration to a " CPU " that has FP support .
*/
if ( mt_fpemul_threshold > 0
& & ( ( current - > thread . emulated_fp + +
> mt_fpemul_threshold ) ) ) {
/*
* If there ' s no FPU present , or if the
* application has already restricted
* the allowed set to exclude any CPUs
* with FPUs , we ' ll skip the procedure .
*/
if ( cpus_intersects ( current - > cpus_allowed ,
mt_fpu_cpumask ) ) {
cpumask_t tmask ;
cpus_and ( tmask ,
current - > thread . user_cpus_allowed ,
mt_fpu_cpumask ) ;
set_cpus_allowed ( current , tmask ) ;
current - > thread . mflags | = MF_FPUBOUND ;
}
}
}
# endif /* CONFIG_MIPS_MT_FPAFF */
2005-04-17 02:20:36 +04:00
}
return ;
case 2 :
case 3 :
break ;
}
force_sig ( SIGILL , current ) ;
}
asmlinkage void do_mdmx ( struct pt_regs * regs )
{
force_sig ( SIGILL , current ) ;
}
asmlinkage void do_watch ( struct pt_regs * regs )
{
/*
* We use the watch exception where available to detect stack
* overflows .
*/
dump_tlb_all ( ) ;
show_regs ( regs ) ;
panic ( " Caught WATCH exception - probably caused by stack overflow. " ) ;
}
asmlinkage void do_mcheck ( struct pt_regs * regs )
{
2006-05-24 19:51:02 +04:00
const int field = 2 * sizeof ( unsigned long ) ;
int multi_match = regs - > cp0_status & ST0_TS ;
2005-04-17 02:20:36 +04:00
show_regs ( regs ) ;
2006-05-24 19:51:02 +04:00
if ( multi_match ) {
printk ( " Index : %0x \n " , read_c0_index ( ) ) ;
printk ( " Pagemask: %0x \n " , read_c0_pagemask ( ) ) ;
printk ( " EntryHi : %0*lx \n " , field , read_c0_entryhi ( ) ) ;
printk ( " EntryLo0: %0*lx \n " , field , read_c0_entrylo0 ( ) ) ;
printk ( " EntryLo1: %0*lx \n " , field , read_c0_entrylo1 ( ) ) ;
printk ( " \n " ) ;
dump_tlb_all ( ) ;
}
show_code ( ( unsigned int * ) regs - > cp0_epc ) ;
2005-04-17 02:20:36 +04:00
/*
* Some chips may have other causes of machine check ( e . g . SB1
* graduation timer )
*/
panic ( " Caught Machine Check exception - %scaused by multiple "
" matching entries in the TLB. " ,
2006-05-24 19:51:02 +04:00
( multi_match ) ? " " : " not " ) ;
2005-04-17 02:20:36 +04:00
}
2005-08-17 21:44:08 +04:00
asmlinkage void do_mt ( struct pt_regs * regs )
{
2006-04-05 12:45:45 +04:00
int subcode ;
subcode = ( read_vpe_c0_vpecontrol ( ) & VPECONTROL_EXCPT )
> > VPECONTROL_EXCPT_SHIFT ;
switch ( subcode ) {
case 0 :
2006-06-30 17:19:45 +04:00
printk ( KERN_DEBUG " Thread Underflow \n " ) ;
2006-04-05 12:45:45 +04:00
break ;
case 1 :
2006-06-30 17:19:45 +04:00
printk ( KERN_DEBUG " Thread Overflow \n " ) ;
2006-04-05 12:45:45 +04:00
break ;
case 2 :
2006-06-30 17:19:45 +04:00
printk ( KERN_DEBUG " Invalid YIELD Qualifier \n " ) ;
2006-04-05 12:45:45 +04:00
break ;
case 3 :
2006-06-30 17:19:45 +04:00
printk ( KERN_DEBUG " Gating Storage Exception \n " ) ;
2006-04-05 12:45:45 +04:00
break ;
case 4 :
2006-06-30 17:19:45 +04:00
printk ( KERN_DEBUG " YIELD Scheduler Exception \n " ) ;
2006-04-05 12:45:45 +04:00
break ;
case 5 :
2006-06-30 17:19:45 +04:00
printk ( KERN_DEBUG " Gating Storage Schedulier Exception \n " ) ;
2006-04-05 12:45:45 +04:00
break ;
default :
2006-06-30 17:19:45 +04:00
printk ( KERN_DEBUG " *** UNKNOWN THREAD EXCEPTION %d *** \n " ,
2006-04-05 12:45:45 +04:00
subcode ) ;
break ;
}
2005-08-17 21:44:08 +04:00
die_if_kernel ( " MIPS MT Thread exception in kernel " , regs ) ;
force_sig ( SIGILL , current ) ;
}
2005-05-31 15:49:19 +04:00
asmlinkage void do_dsp ( struct pt_regs * regs )
{
if ( cpu_has_dsp )
panic ( " Unexpected DSP exception \n " ) ;
force_sig ( SIGILL , current ) ;
}
2005-04-17 02:20:36 +04:00
asmlinkage void do_reserved ( struct pt_regs * regs )
{
/*
* Game over - no way to handle this if it ever occurs . Most probably
* caused by a new unknown cpu type or after another deadly
* hard / software error .
*/
show_regs ( regs ) ;
panic ( " Caught reserved exception %ld - should not happen. " ,
( regs - > cp0_cause & 0x7f ) > > 2 ) ;
}
/*
* Some MIPS CPUs can enable / disable for cache parity detection , but do
* it different ways .
*/
static inline void parity_protection_init ( void )
{
switch ( current_cpu_data . cputype ) {
case CPU_24K :
2006-04-27 18:50:32 +04:00
case CPU_34K :
2005-04-17 02:20:36 +04:00
case CPU_5KC :
2005-03-01 21:15:08 +03:00
write_c0_ecc ( 0x80000000 ) ;
back_to_back_c0_hazard ( ) ;
/* Set the PE bit (bit 31) in the c0_errctl register. */
printk ( KERN_INFO " Cache parity protection %sabled \n " ,
( read_c0_ecc ( ) & 0x80000000 ) ? " en " : " dis " ) ;
2005-04-17 02:20:36 +04:00
break ;
case CPU_20KC :
case CPU_25KF :
/* Clear the DE bit (bit 16) in the c0_status register. */
printk ( KERN_INFO " Enable cache parity protection for "
" MIPS 20KC/25KF CPUs. \n " ) ;
clear_c0_status ( ST0_DE ) ;
break ;
default :
break ;
}
}
asmlinkage void cache_parity_error ( void )
{
const int field = 2 * sizeof ( unsigned long ) ;
unsigned int reg_val ;
/* For the moment, report the problem and hang. */
printk ( " Cache error exception: \n " ) ;
printk ( " cp0_errorepc == %0*lx \n " , field , read_c0_errorepc ( ) ) ;
reg_val = read_c0_cacheerr ( ) ;
printk ( " c0_cacheerr == %08x \n " , reg_val ) ;
printk ( " Decoded c0_cacheerr: %s cache fault in %s reference. \n " ,
reg_val & ( 1 < < 30 ) ? " secondary " : " primary " ,
reg_val & ( 1 < < 31 ) ? " data " : " insn " ) ;
printk ( " Error bits: %s%s%s%s%s%s%s \n " ,
reg_val & ( 1 < < 29 ) ? " ED " : " " ,
reg_val & ( 1 < < 28 ) ? " ET " : " " ,
reg_val & ( 1 < < 26 ) ? " EE " : " " ,
reg_val & ( 1 < < 25 ) ? " EB " : " " ,
reg_val & ( 1 < < 24 ) ? " EI " : " " ,
reg_val & ( 1 < < 23 ) ? " E1 " : " " ,
reg_val & ( 1 < < 22 ) ? " E0 " : " " ) ;
printk ( " IDX: 0x%08x \n " , reg_val & ( ( 1 < < 22 ) - 1 ) ) ;
2005-10-07 19:58:15 +04:00
# if defined(CONFIG_CPU_MIPS32) || defined(CONFIG_CPU_MIPS64)
2005-04-17 02:20:36 +04:00
if ( reg_val & ( 1 < < 22 ) )
printk ( " DErrAddr0: 0x%0*lx \n " , field , read_c0_derraddr0 ( ) ) ;
if ( reg_val & ( 1 < < 23 ) )
printk ( " DErrAddr1: 0x%0*lx \n " , field , read_c0_derraddr1 ( ) ) ;
# endif
panic ( " Can't handle the cache error! " ) ;
}
/*
* SDBBP EJTAG debug exception handler .
* We skip the instruction and return to the next instruction .
*/
void ejtag_exception_handler ( struct pt_regs * regs )
{
const int field = 2 * sizeof ( unsigned long ) ;
unsigned long depc , old_epc ;
unsigned int debug ;
2006-06-30 15:32:37 +04:00
printk ( KERN_DEBUG " SDBBP EJTAG debug exception - not handled yet, just ignored! \n " ) ;
2005-04-17 02:20:36 +04:00
depc = read_c0_depc ( ) ;
debug = read_c0_debug ( ) ;
2006-06-30 15:32:37 +04:00
printk ( KERN_DEBUG " c0_depc = %0*lx, DEBUG = %08x \n " , field , depc , debug ) ;
2005-04-17 02:20:36 +04:00
if ( debug & 0x80000000 ) {
/*
* In branch delay slot .
* We cheat a little bit here and use EPC to calculate the
* debug return address ( DEPC ) . EPC is restored after the
* calculation .
*/
old_epc = regs - > cp0_epc ;
regs - > cp0_epc = depc ;
__compute_return_epc ( regs ) ;
depc = regs - > cp0_epc ;
regs - > cp0_epc = old_epc ;
} else
depc + = 4 ;
write_c0_depc ( depc ) ;
#if 0
2006-06-30 15:32:37 +04:00
printk ( KERN_DEBUG " \n \n ----- Enable EJTAG single stepping ---- \n \n " ) ;
2005-04-17 02:20:36 +04:00
write_c0_debug ( debug | 0x100 ) ;
# endif
}
/*
* NMI exception handler .
*/
void nmi_exception_handler ( struct pt_regs * regs )
{
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_MIPS_MT_SMTC
unsigned long dvpret = dvpe ( ) ;
bust_spinlocks ( 1 ) ;
printk ( " NMI taken!!!! \n " ) ;
mips_mt_regdump ( dvpret ) ;
# else
bust_spinlocks ( 1 ) ;
2005-04-17 02:20:36 +04:00
printk ( " NMI taken!!!! \n " ) ;
2006-04-05 12:45:45 +04:00
# endif /* CONFIG_MIPS_MT_SMTC */
2005-04-17 02:20:36 +04:00
die ( " NMI " , regs ) ;
while ( 1 ) ;
}
2005-07-14 19:57:16 +04:00
# define VECTORSPACING 0x100 /* for EI/VI mode */
unsigned long ebase ;
2005-04-17 02:20:36 +04:00
unsigned long exception_handlers [ 32 ] ;
2005-07-14 19:57:16 +04:00
unsigned long vi_handlers [ 64 ] ;
2005-04-17 02:20:36 +04:00
/*
* As a side effect of the way this is implemented we ' re limited
* to interrupt handlers in the address range from
* KSEG0 < = x < KSEG0 + 256 mb on the Nevada . Oh well . . .
*/
void * set_except_vector ( int n , void * addr )
{
unsigned long handler = ( unsigned long ) addr ;
unsigned long old_handler = exception_handlers [ n ] ;
exception_handlers [ n ] = handler ;
if ( n = = 0 & & cpu_has_divec ) {
2005-07-14 19:57:16 +04:00
* ( volatile u32 * ) ( ebase + 0x200 ) = 0x08000000 |
2005-04-17 02:20:36 +04:00
( 0x03ffffff & ( handler > > 2 ) ) ;
2005-07-14 19:57:16 +04:00
flush_icache_range ( ebase + 0x200 , ebase + 0x204 ) ;
}
return ( void * ) old_handler ;
}
2006-06-05 20:24:46 +04:00
# ifdef CONFIG_CPU_MIPSR2_SRS
2005-07-14 19:57:16 +04:00
/*
2006-04-04 18:09:06 +04:00
* MIPSR2 shadow register set allocation
2005-07-14 19:57:16 +04:00
* FIXME : SMP . . .
*/
2006-04-04 18:09:06 +04:00
static struct shadow_registers {
/*
* Number of shadow register sets supported
*/
unsigned long sr_supported ;
/*
* Bitmap of allocated shadow registers
*/
unsigned long sr_allocated ;
2005-07-14 19:57:16 +04:00
} shadow_registers ;
2006-04-05 12:45:49 +04:00
static void mips_srs_init ( void )
2005-07-14 19:57:16 +04:00
{
shadow_registers . sr_supported = ( ( read_c0_srsctl ( ) > > 26 ) & 0x0f ) + 1 ;
2006-10-31 07:44:38 +03:00
printk ( KERN_INFO " %ld MIPSR2 register sets available \n " ,
2006-03-29 17:11:22 +04:00
shadow_registers . sr_supported ) ;
2005-07-14 19:57:16 +04:00
shadow_registers . sr_allocated = 1 ; /* Set 0 used by kernel */
}
int mips_srs_max ( void )
{
return shadow_registers . sr_supported ;
}
2006-03-29 17:12:58 +04:00
int mips_srs_alloc ( void )
2005-07-14 19:57:16 +04:00
{
struct shadow_registers * sr = & shadow_registers ;
int set ;
2006-04-04 18:09:06 +04:00
again :
set = find_first_zero_bit ( & sr - > sr_allocated , sr - > sr_supported ) ;
if ( set > = sr - > sr_supported )
return - 1 ;
2005-07-14 19:57:16 +04:00
2006-04-04 18:09:06 +04:00
if ( test_and_set_bit ( set , & sr - > sr_allocated ) )
goto again ;
2005-07-14 19:57:16 +04:00
2006-04-04 18:09:06 +04:00
return set ;
2005-07-14 19:57:16 +04:00
}
2006-04-05 12:45:45 +04:00
void mips_srs_free ( int set )
2005-07-14 19:57:16 +04:00
{
struct shadow_registers * sr = & shadow_registers ;
2006-04-04 18:09:06 +04:00
clear_bit ( set , & sr - > sr_allocated ) ;
2005-07-14 19:57:16 +04:00
}
2007-05-21 18:45:38 +04:00
static asmlinkage void do_default_vi ( void )
{
show_regs ( get_irq_regs ( ) ) ;
panic ( " Caught unexpected vectored interrupt. " ) ;
}
2007-05-06 21:31:18 +04:00
static void * set_vi_srs_handler ( int n , vi_handler_t addr , int srs )
2005-07-14 19:57:16 +04:00
{
unsigned long handler ;
unsigned long old_handler = vi_handlers [ n ] ;
u32 * w ;
unsigned char * b ;
if ( ! cpu_has_veic & & ! cpu_has_vint )
BUG ( ) ;
if ( addr = = NULL ) {
handler = ( unsigned long ) do_default_vi ;
srs = 0 ;
2006-04-05 12:45:45 +04:00
} else
2005-07-14 19:57:16 +04:00
handler = ( unsigned long ) addr ;
vi_handlers [ n ] = ( unsigned long ) addr ;
b = ( unsigned char * ) ( ebase + 0x200 + n * VECTORSPACING ) ;
if ( srs > = mips_srs_max ( ) )
panic ( " Shadow register set %d not supported " , srs ) ;
if ( cpu_has_veic ) {
if ( board_bind_eic_interrupt )
board_bind_eic_interrupt ( n , srs ) ;
2006-04-05 12:45:45 +04:00
} else if ( cpu_has_vint ) {
2005-07-14 19:57:16 +04:00
/* SRSMap is only defined if shadow sets are implemented */
if ( mips_srs_max ( ) > 1 )
change_c0_srsmap ( 0xf < < n * 4 , srs < < n * 4 ) ;
}
if ( srs = = 0 ) {
/*
* If no shadow set is selected then use the default handler
* that does normal register saving and a standard interrupt exit
*/
extern char except_vec_vi , except_vec_vi_lui ;
extern char except_vec_vi_ori , except_vec_vi_end ;
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_MIPS_MT_SMTC
/*
* We need to provide the SMTC vectored interrupt handler
* not only with the address of the handler , but with the
* Status . IM bit to be masked before going there .
*/
extern char except_vec_vi_mori ;
const int mori_offset = & except_vec_vi_mori - & except_vec_vi ;
# endif /* CONFIG_MIPS_MT_SMTC */
2005-07-14 19:57:16 +04:00
const int handler_len = & except_vec_vi_end - & except_vec_vi ;
const int lui_offset = & except_vec_vi_lui - & except_vec_vi ;
const int ori_offset = & except_vec_vi_ori - & except_vec_vi ;
if ( handler_len > VECTORSPACING ) {
/*
* Sigh . . . panicing won ' t help as the console
* is probably not configured : (
*/
panic ( " VECTORSPACING too small " ) ;
}
memcpy ( b , & except_vec_vi , handler_len ) ;
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_MIPS_MT_SMTC
2007-05-31 17:00:19 +04:00
BUG_ON ( n > 7 ) ; /* Vector index %d exceeds SMTC maximum. */
2006-04-05 12:45:45 +04:00
w = ( u32 * ) ( b + mori_offset ) ;
* w = ( * w & 0xffff0000 ) | ( 0x100 < < n ) ;
# endif /* CONFIG_MIPS_MT_SMTC */
2005-07-14 19:57:16 +04:00
w = ( u32 * ) ( b + lui_offset ) ;
* w = ( * w & 0xffff0000 ) | ( ( ( u32 ) handler > > 16 ) & 0xffff ) ;
w = ( u32 * ) ( b + ori_offset ) ;
* w = ( * w & 0xffff0000 ) | ( ( u32 ) handler & 0xffff ) ;
flush_icache_range ( ( unsigned long ) b , ( unsigned long ) ( b + handler_len ) ) ;
}
else {
/*
* In other cases jump directly to the interrupt handler
*
* It is the handlers responsibility to save registers if required
* ( eg hi / lo ) and return from the exception using " eret "
*/
w = ( u32 * ) b ;
* w + + = 0x08000000 | ( ( ( u32 ) handler > > 2 ) & 0x03fffff ) ; /* j handler */
* w = 0 ;
flush_icache_range ( ( unsigned long ) b , ( unsigned long ) ( b + 8 ) ) ;
2005-04-17 02:20:36 +04:00
}
2005-07-14 19:57:16 +04:00
2005-04-17 02:20:36 +04:00
return ( void * ) old_handler ;
}
2007-05-06 21:31:18 +04:00
void * set_vi_handler ( int n , vi_handler_t addr )
2005-07-14 19:57:16 +04:00
{
2006-03-29 17:12:58 +04:00
return set_vi_srs_handler ( n , addr , 0 ) ;
2005-07-14 19:57:16 +04:00
}
2006-06-05 20:24:46 +04:00
# else
static inline void mips_srs_init ( void )
{
}
# endif /* CONFIG_CPU_MIPSR2_SRS */
2005-07-14 19:57:16 +04:00
2005-04-17 02:20:36 +04:00
/*
* This is used by native signal handling
*/
2007-03-09 19:07:45 +03:00
asmlinkage int ( * save_fp_context ) ( struct sigcontext __user * sc ) ;
asmlinkage int ( * restore_fp_context ) ( struct sigcontext __user * sc ) ;
2005-04-17 02:20:36 +04:00
2007-03-09 19:07:45 +03:00
extern asmlinkage int _save_fp_context ( struct sigcontext __user * sc ) ;
extern asmlinkage int _restore_fp_context ( struct sigcontext __user * sc ) ;
2005-04-17 02:20:36 +04:00
2007-03-09 19:07:45 +03:00
extern asmlinkage int fpu_emulator_save_context ( struct sigcontext __user * sc ) ;
extern asmlinkage int fpu_emulator_restore_context ( struct sigcontext __user * sc ) ;
2005-04-17 02:20:36 +04:00
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_SMP
2007-03-09 19:07:45 +03:00
static int smp_save_fp_context ( struct sigcontext __user * sc )
2006-04-05 12:45:45 +04:00
{
2007-03-09 19:07:45 +03:00
return raw_cpu_has_fpu
2006-04-05 12:45:45 +04:00
? _save_fp_context ( sc )
: fpu_emulator_save_context ( sc ) ;
}
2007-03-09 19:07:45 +03:00
static int smp_restore_fp_context ( struct sigcontext __user * sc )
2006-04-05 12:45:45 +04:00
{
2007-03-09 19:07:45 +03:00
return raw_cpu_has_fpu
2006-04-05 12:45:45 +04:00
? _restore_fp_context ( sc )
: fpu_emulator_restore_context ( sc ) ;
}
# endif
2005-04-17 02:20:36 +04:00
static inline void signal_init ( void )
{
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_SMP
/* For now just do the cpu_has_fpu check when the functions are invoked */
save_fp_context = smp_save_fp_context ;
restore_fp_context = smp_restore_fp_context ;
# else
2005-04-17 02:20:36 +04:00
if ( cpu_has_fpu ) {
save_fp_context = _save_fp_context ;
restore_fp_context = _restore_fp_context ;
} else {
save_fp_context = fpu_emulator_save_context ;
restore_fp_context = fpu_emulator_restore_context ;
}
2006-04-05 12:45:45 +04:00
# endif
2005-04-17 02:20:36 +04:00
}
# ifdef CONFIG_MIPS32_COMPAT
/*
* This is used by 32 - bit signal stuff on the 64 - bit kernel
*/
2007-03-09 19:07:45 +03:00
asmlinkage int ( * save_fp_context32 ) ( struct sigcontext32 __user * sc ) ;
asmlinkage int ( * restore_fp_context32 ) ( struct sigcontext32 __user * sc ) ;
2005-04-17 02:20:36 +04:00
2007-03-09 19:07:45 +03:00
extern asmlinkage int _save_fp_context32 ( struct sigcontext32 __user * sc ) ;
extern asmlinkage int _restore_fp_context32 ( struct sigcontext32 __user * sc ) ;
2005-04-17 02:20:36 +04:00
2007-03-09 19:07:45 +03:00
extern asmlinkage int fpu_emulator_save_context32 ( struct sigcontext32 __user * sc ) ;
extern asmlinkage int fpu_emulator_restore_context32 ( struct sigcontext32 __user * sc ) ;
2005-04-17 02:20:36 +04:00
static inline void signal32_init ( void )
{
if ( cpu_has_fpu ) {
save_fp_context32 = _save_fp_context32 ;
restore_fp_context32 = _restore_fp_context32 ;
} else {
save_fp_context32 = fpu_emulator_save_context32 ;
restore_fp_context32 = fpu_emulator_restore_context32 ;
}
}
# endif
extern void cpu_cache_init ( void ) ;
extern void tlb_init ( void ) ;
2005-07-15 19:23:23 +04:00
extern void flush_tlb_handlers ( void ) ;
2005-04-17 02:20:36 +04:00
void __init per_cpu_trap_init ( void )
{
unsigned int cpu = smp_processor_id ( ) ;
unsigned int status_set = ST0_CU0 ;
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_MIPS_MT_SMTC
int secondaryTC = 0 ;
int bootTC = ( cpu = = 0 ) ;
/*
* Only do per_cpu_trap_init ( ) for first TC of Each VPE .
* Note that this hack assumes that the SMTC init code
* assigns TCs consecutively and in ascending order .
*/
if ( ( ( read_c0_tcbind ( ) & TCBIND_CURTC ) ! = 0 ) & &
( ( read_c0_tcbind ( ) & TCBIND_CURVPE ) = = cpu_data [ cpu - 1 ] . vpe_id ) )
secondaryTC = 1 ;
# endif /* CONFIG_MIPS_MT_SMTC */
2005-04-17 02:20:36 +04:00
/*
* Disable coprocessors and select 32 - bit or 64 - bit addressing
* and the 16 / 32 or 32 / 32 FPR register model . Reset the BEV
* flag that some firmware may have left set and the TS bit ( for
* IP27 ) . Set XX for ISA IV code to work .
*/
2005-09-04 02:56:16 +04:00
# ifdef CONFIG_64BIT
2005-04-17 02:20:36 +04:00
status_set | = ST0_FR | ST0_KX | ST0_SX | ST0_UX ;
# endif
if ( current_cpu_data . isa_level = = MIPS_CPU_ISA_IV )
status_set | = ST0_XX ;
2006-02-07 04:20:43 +03:00
change_c0_status ( ST0_CU | ST0_MX | ST0_RE | ST0_FR | ST0_BEV | ST0_TS | ST0_KX | ST0_SX | ST0_UX ,
2005-04-17 02:20:36 +04:00
status_set ) ;
2005-05-31 15:49:19 +04:00
if ( cpu_has_dsp )
set_c0_status ( ST0_MX ) ;
2005-07-14 19:57:16 +04:00
# ifdef CONFIG_CPU_MIPSR2
write_c0_hwrena ( 0x0000000f ) ; /* Allow rdhwr to all registers */
# endif
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_MIPS_MT_SMTC
if ( ! secondaryTC ) {
# endif /* CONFIG_MIPS_MT_SMTC */
2005-04-17 02:20:36 +04:00
/*
2005-07-14 19:57:16 +04:00
* Interrupt handling .
2005-04-17 02:20:36 +04:00
*/
2005-07-14 19:57:16 +04:00
if ( cpu_has_veic | | cpu_has_vint ) {
write_c0_ebase ( ebase ) ;
/* Setting vector spacing enables EI/VI mode */
change_c0_intctl ( 0x3e0 , VECTORSPACING ) ;
}
2005-08-17 17:44:26 +04:00
if ( cpu_has_divec ) {
if ( cpu_has_mipsmt ) {
unsigned int vpflags = dvpe ( ) ;
set_c0_cause ( CAUSEF_IV ) ;
evpe ( vpflags ) ;
} else
set_c0_cause ( CAUSEF_IV ) ;
}
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_MIPS_MT_SMTC
}
# endif /* CONFIG_MIPS_MT_SMTC */
2005-04-17 02:20:36 +04:00
cpu_data [ cpu ] . asid_cache = ASID_FIRST_VERSION ;
TLBMISS_HANDLER_SETUP ( ) ;
atomic_inc ( & init_mm . mm_count ) ;
current - > active_mm = & init_mm ;
BUG_ON ( current - > mm ) ;
enter_lazy_tlb ( & init_mm , current ) ;
2006-04-05 12:45:45 +04:00
# ifdef CONFIG_MIPS_MT_SMTC
if ( bootTC ) {
# endif /* CONFIG_MIPS_MT_SMTC */
cpu_cache_init ( ) ;
tlb_init ( ) ;
# ifdef CONFIG_MIPS_MT_SMTC
}
# endif /* CONFIG_MIPS_MT_SMTC */
2005-04-17 02:20:36 +04:00
}
2005-07-14 19:57:16 +04:00
/* Install CPU exception handler */
void __init set_handler ( unsigned long offset , void * addr , unsigned long size )
{
memcpy ( ( void * ) ( ebase + offset ) , addr , size ) ;
flush_icache_range ( ebase + offset , ebase + offset + size ) ;
}
/* Install uncached CPU exception handler */
void __init set_uncached_handler ( unsigned long offset , void * addr , unsigned long size )
{
# ifdef CONFIG_32BIT
unsigned long uncached_ebase = KSEG1ADDR ( ebase ) ;
# endif
# ifdef CONFIG_64BIT
unsigned long uncached_ebase = TO_UNCAC ( ebase ) ;
# endif
memcpy ( ( void * ) ( uncached_ebase + offset ) , addr , size ) ;
}
2006-09-11 12:50:29 +04:00
static int __initdata rdhwr_noopt ;
static int __init set_rdhwr_noopt ( char * str )
{
rdhwr_noopt = 1 ;
return 1 ;
}
__setup ( " rdhwr_noopt " , set_rdhwr_noopt ) ;
2005-04-17 02:20:36 +04:00
void __init trap_init ( void )
{
extern char except_vec3_generic , except_vec3_r4000 ;
extern char except_vec4 ;
unsigned long i ;
2005-07-14 19:57:16 +04:00
if ( cpu_has_veic | | cpu_has_vint )
ebase = ( unsigned long ) alloc_bootmem_low_pages ( 0x200 + VECTORSPACING * 64 ) ;
else
ebase = CAC_BASE ;
mips_srs_init ( ) ;
2005-04-17 02:20:36 +04:00
per_cpu_trap_init ( ) ;
/*
* Copy the generic exception handlers to their final destination .
* This will be overriden later as suitable for a particular
* configuration .
*/
2005-07-14 19:57:16 +04:00
set_handler ( 0x180 , & except_vec3_generic , 0x80 ) ;
2005-04-17 02:20:36 +04:00
/*
* Setup default vectors
*/
for ( i = 0 ; i < = 31 ; i + + )
set_except_vector ( i , handle_reserved ) ;
/*
* Copy the EJTAG debug exception vector handler code to it ' s final
* destination .
*/
2005-07-14 19:57:16 +04:00
if ( cpu_has_ejtag & & board_ejtag_handler_setup )
board_ejtag_handler_setup ( ) ;
2005-04-17 02:20:36 +04:00
/*
* Only some CPUs have the watch exceptions .
*/
if ( cpu_has_watch )
set_except_vector ( 23 , handle_watch ) ;
/*
2005-07-14 19:57:16 +04:00
* Initialise interrupt handlers
2005-04-17 02:20:36 +04:00
*/
2005-07-14 19:57:16 +04:00
if ( cpu_has_veic | | cpu_has_vint ) {
int nvec = cpu_has_veic ? 64 : 8 ;
for ( i = 0 ; i < nvec ; i + + )
2006-03-29 17:12:58 +04:00
set_vi_handler ( i , NULL ) ;
2005-07-14 19:57:16 +04:00
}
else if ( cpu_has_divec )
set_handler ( 0x200 , & except_vec4 , 0x8 ) ;
2005-04-17 02:20:36 +04:00
/*
* Some CPUs can enable / disable for cache parity detection , but does
* it different ways .
*/
parity_protection_init ( ) ;
/*
* The Data Bus Errors / Instruction Bus Errors are signaled
* by external hardware . Therefore these two exceptions
* may have board specific handlers .
*/
if ( board_be_init )
board_be_init ( ) ;
2006-04-03 20:56:36 +04:00
set_except_vector ( 0 , handle_int ) ;
2005-04-17 02:20:36 +04:00
set_except_vector ( 1 , handle_tlbm ) ;
set_except_vector ( 2 , handle_tlbl ) ;
set_except_vector ( 3 , handle_tlbs ) ;
set_except_vector ( 4 , handle_adel ) ;
set_except_vector ( 5 , handle_ades ) ;
set_except_vector ( 6 , handle_ibe ) ;
set_except_vector ( 7 , handle_dbe ) ;
set_except_vector ( 8 , handle_sys ) ;
set_except_vector ( 9 , handle_bp ) ;
2006-09-11 12:50:29 +04:00
set_except_vector ( 10 , rdhwr_noopt ? handle_ri :
( cpu_has_vtag_icache ?
handle_ri_rdhwr_vivt : handle_ri_rdhwr ) ) ;
2005-04-17 02:20:36 +04:00
set_except_vector ( 11 , handle_cpu ) ;
set_except_vector ( 12 , handle_ov ) ;
set_except_vector ( 13 , handle_tr ) ;
if ( current_cpu_data . cputype = = CPU_R6000 | |
current_cpu_data . cputype = = CPU_R6000A ) {
/*
* The R6000 is the only R - series CPU that features a machine
* check exception ( similar to the R4000 cache error ) and
* unaligned ldc1 / sdc1 exception . The handlers have not been
* written yet . Well , anyway there is no R6000 machine on the
* current list of targets for Linux / MIPS .
* ( Duh , crap , there is someone with a triple R6k machine )
*/
//set_except_vector(14, handle_mc);
//set_except_vector(15, handle_ndc);
}
2005-07-14 19:57:16 +04:00
if ( board_nmi_handler_setup )
board_nmi_handler_setup ( ) ;
2005-05-31 15:49:19 +04:00
if ( cpu_has_fpu & & ! cpu_has_nofpuex )
set_except_vector ( 15 , handle_fpe ) ;
set_except_vector ( 22 , handle_mdmx ) ;
if ( cpu_has_mcheck )
set_except_vector ( 24 , handle_mcheck ) ;
2005-08-17 21:44:08 +04:00
if ( cpu_has_mipsmt )
set_except_vector ( 25 , handle_mt ) ;
2007-05-25 01:30:18 +04:00
set_except_vector ( 26 , handle_dsp ) ;
2005-05-31 15:49:19 +04:00
if ( cpu_has_vce )
/* Special exception: R4[04]00 uses also the divec space. */
memcpy ( ( void * ) ( CAC_BASE + 0x180 ) , & except_vec3_r4000 , 0x100 ) ;
else if ( cpu_has_4kex )
memcpy ( ( void * ) ( CAC_BASE + 0x180 ) , & except_vec3_generic , 0x80 ) ;
else
memcpy ( ( void * ) ( CAC_BASE + 0x080 ) , & except_vec3_generic , 0x80 ) ;
2005-04-17 02:20:36 +04:00
signal_init ( ) ;
# ifdef CONFIG_MIPS32_COMPAT
signal32_init ( ) ;
# endif
2005-07-14 19:57:16 +04:00
flush_icache_range ( ebase , ebase + 0x400 ) ;
2005-07-15 19:23:23 +04:00
flush_tlb_handlers ( ) ;
2005-04-17 02:20:36 +04:00
}