2010-05-28 23:09:12 -04:00
/*
* Copyright 2010 Tilera Corporation . All Rights Reserved .
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation , version 2.
*
* This program is distributed in the hope that it will be useful , but
* WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE , GOOD TITLE or
* NON INFRINGEMENT . See the GNU General Public License for
* more details .
*/
# include <linux/sched.h>
# include <linux/kernel.h>
# include <linux/kprobes.h>
# include <linux/module.h>
# include <linux/reboot.h>
# include <linux/uaccess.h>
# include <linux/ptrace.h>
2010-06-25 17:04:17 -04:00
# include <asm/stack.h>
# include <asm/traps.h>
2012-03-28 18:30:03 +01:00
# include <asm/setup.h>
2010-05-28 23:09:12 -04:00
# include <arch/interrupts.h>
# include <arch/spr_def.h>
2011-11-02 23:02:17 -04:00
# include <arch/opcode.h>
2010-05-28 23:09:12 -04:00
void __init trap_init ( void )
{
/* Nothing needed here since we link code at .intrpt1 */
}
int unaligned_fixup = 1 ;
static int __init setup_unaligned_fixup ( char * str )
{
/*
* Say " =-1 " to completely disable it . If you just do " =0 " , we
* will still parse the instruction , then fire a SIGBUS with
* the correct address from inside the single_step code .
*/
long val ;
if ( strict_strtol ( str , 0 , & val ) ! = 0 )
return 0 ;
unaligned_fixup = val ;
2010-06-25 17:04:17 -04:00
pr_info ( " Fixups for unaligned data accesses are %s \n " ,
2010-05-28 23:09:12 -04:00
unaligned_fixup > = 0 ?
( unaligned_fixup ? " enabled " : " disabled " ) :
" completely disabled " ) ;
return 1 ;
}
__setup ( " unaligned_fixup= " , setup_unaligned_fixup ) ;
# if CHIP_HAS_TILE_DMA()
static int dma_disabled ;
static int __init nodma ( char * str )
{
2010-06-25 17:04:17 -04:00
pr_info ( " User-space DMA is disabled \n " ) ;
2010-05-28 23:09:12 -04:00
dma_disabled = 1 ;
return 1 ;
}
__setup ( " nodma " , nodma ) ;
/* How to decode SPR_GPV_REASON */
# define IRET_ERROR (1U << 31)
# define MT_ERROR (1U << 30)
# define MF_ERROR (1U << 29)
# define SPR_INDEX ((1U << 15) - 1)
# define SPR_MPL_SHIFT 9 /* starting bit position for MPL encoded in SPR */
/*
* See if this GPV is just to notify the kernel of SPR use and we can
* retry the user instruction after adjusting some MPLs suitably .
*/
static int retry_gpv ( unsigned int gpv_reason )
{
int mpl ;
if ( gpv_reason & IRET_ERROR )
return 0 ;
BUG_ON ( ( gpv_reason & ( MT_ERROR | MF_ERROR ) ) = = 0 ) ;
mpl = ( gpv_reason & SPR_INDEX ) > > SPR_MPL_SHIFT ;
if ( mpl = = INT_DMA_NOTIFY & & ! dma_disabled ) {
/* User is turning on DMA. Allow it and retry. */
printk ( KERN_DEBUG " Process %d/%s is now enabled for DMA \n " ,
current - > pid , current - > comm ) ;
BUG_ON ( current - > thread . tile_dma_state . enabled ) ;
current - > thread . tile_dma_state . enabled = 1 ;
grant_dma_mpls ( ) ;
return 1 ;
}
return 0 ;
}
# endif /* CHIP_HAS_TILE_DMA() */
# ifdef __tilegx__
2010-06-25 17:04:17 -04:00
# define bundle_bits tilegx_bundle_bits
2010-05-28 23:09:12 -04:00
# else
2010-06-25 17:04:17 -04:00
# define bundle_bits tile_bundle_bits
2010-05-28 23:09:12 -04:00
# endif
2010-06-25 17:04:17 -04:00
extern bundle_bits bpt_code ;
asm ( " .pushsection .rodata.bpt_code, \" a \" ; "
" .align 8; "
" bpt_code: bpt; "
" .size bpt_code,.-bpt_code; "
" .popsection " ) ;
static int special_ill ( bundle_bits bundle , int * sigp , int * codep )
{
int sig , code , maxcode ;
if ( bundle = = bpt_code ) {
* sigp = SIGTRAP ;
* codep = TRAP_BRKPT ;
return 1 ;
}
/* If it's a "raise" bundle, then "ill" must be in pipe X1. */
# ifdef __tilegx__
if ( ( bundle & TILEGX_BUNDLE_MODE_MASK ) ! = 0 )
return 0 ;
2010-08-13 08:40:57 -04:00
if ( get_Opcode_X1 ( bundle ) ! = RRR_0_OPCODE_X1 )
return 0 ;
if ( get_RRROpcodeExtension_X1 ( bundle ) ! = UNARY_RRR_0_OPCODE_X1 )
2010-06-25 17:04:17 -04:00
return 0 ;
if ( get_UnaryOpcodeExtension_X1 ( bundle ) ! = ILL_UNARY_OPCODE_X1 )
return 0 ;
# else
2011-11-02 23:02:17 -04:00
if ( bundle & TILEPRO_BUNDLE_Y_ENCODING_MASK )
2010-06-25 17:04:17 -04:00
return 0 ;
if ( get_Opcode_X1 ( bundle ) ! = SHUN_0_OPCODE_X1 )
return 0 ;
if ( get_UnShOpcodeExtension_X1 ( bundle ) ! = UN_0_SHUN_0_OPCODE_X1 )
return 0 ;
if ( get_UnOpcodeExtension_X1 ( bundle ) ! = ILL_UN_0_SHUN_0_OPCODE_X1 )
return 0 ;
# endif
/* Check that the magic distinguishers are set to mean "raise". */
if ( get_Dest_X1 ( bundle ) ! = 29 | | get_SrcA_X1 ( bundle ) ! = 37 )
return 0 ;
/* There must be an "addli zero, zero, VAL" in X0. */
if ( get_Opcode_X0 ( bundle ) ! = ADDLI_OPCODE_X0 )
return 0 ;
if ( get_Dest_X0 ( bundle ) ! = TREG_ZERO )
return 0 ;
if ( get_SrcA_X0 ( bundle ) ! = TREG_ZERO )
return 0 ;
/*
* Validate the proposed signal number and si_code value .
* Note that we embed these in the static instruction itself
* so that we perturb the register state as little as possible
* at the time of the actual fault ; it ' s unlikely you ' d ever
* need to dynamically choose which kind of fault to raise
* from user space .
*/
sig = get_Imm16_X0 ( bundle ) & 0x3f ;
switch ( sig ) {
case SIGILL :
maxcode = NSIGILL ;
break ;
case SIGFPE :
maxcode = NSIGFPE ;
break ;
case SIGSEGV :
maxcode = NSIGSEGV ;
break ;
case SIGBUS :
maxcode = NSIGBUS ;
break ;
case SIGTRAP :
maxcode = NSIGTRAP ;
break ;
default :
return 0 ;
}
code = ( get_Imm16_X0 ( bundle ) > > 6 ) & 0xf ;
if ( code < = 0 | | code > maxcode )
return 0 ;
/* Make it the requested signal. */
* sigp = sig ;
* codep = code | __SI_FAULT ;
return 1 ;
}
2012-03-30 16:31:08 -04:00
static const char * const int_name [ ] = {
[ INT_MEM_ERROR ] = " Memory error " ,
[ INT_ILL ] = " Illegal instruction " ,
[ INT_GPV ] = " General protection violation " ,
[ INT_UDN_ACCESS ] = " UDN access " ,
[ INT_IDN_ACCESS ] = " IDN access " ,
# if CHIP_HAS_SN()
[ INT_SN_ACCESS ] = " SN access " ,
# endif
[ INT_SWINT_3 ] = " Software interrupt 3 " ,
[ INT_SWINT_2 ] = " Software interrupt 2 " ,
[ INT_SWINT_0 ] = " Software interrupt 0 " ,
[ INT_UNALIGN_DATA ] = " Unaligned data " ,
[ INT_DOUBLE_FAULT ] = " Double fault " ,
# ifdef __tilegx__
[ INT_ILL_TRANS ] = " Illegal virtual address " ,
# endif
} ;
2010-05-28 23:09:12 -04:00
void __kprobes do_trap ( struct pt_regs * regs , int fault_num ,
unsigned long reason )
{
siginfo_t info = { 0 } ;
int signo , code ;
2012-03-29 15:23:54 -04:00
unsigned long address = 0 ;
2010-06-25 17:04:17 -04:00
bundle_bits instr ;
2010-05-28 23:09:12 -04:00
2013-08-07 12:11:56 -04:00
/* Re-enable interrupts, if they were previously enabled. */
if ( ! ( regs - > flags & PT_FLAGS_DISABLE_IRQ ) )
local_irq_enable ( ) ;
2010-05-28 23:09:12 -04:00
/*
* If it hits in kernel mode and we can ' t fix it up , just exit the
* current process and hope for the best .
*/
if ( ! user_mode ( regs ) ) {
2012-03-30 16:31:08 -04:00
const char * name ;
2013-08-07 12:11:56 -04:00
char buf [ 100 ] ;
if ( fixup_exception ( regs ) ) /* ILL_TRANS or UNALIGN_DATA */
2010-05-28 23:09:12 -04:00
return ;
2012-03-30 16:31:08 -04:00
if ( fault_num > = 0 & &
fault_num < sizeof ( int_name ) / sizeof ( int_name [ 0 ] ) & &
int_name [ fault_num ] ! = NULL )
name = int_name [ fault_num ] ;
else
name = " Unknown interrupt " ;
2010-05-28 23:09:12 -04:00
if ( fault_num = = INT_GPV )
2013-08-07 12:11:56 -04:00
snprintf ( buf , sizeof ( buf ) , " ; GPV_REASON %#lx " , reason ) ;
# ifdef __tilegx__
else if ( fault_num = = INT_ILL_TRANS )
snprintf ( buf , sizeof ( buf ) , " ; address %#lx " , reason ) ;
# endif
else
buf [ 0 ] = ' \0 ' ;
pr_alert ( " Kernel took bad trap %d (%s) at PC %#lx%s \n " ,
fault_num , name , regs - > pc , buf ) ;
2010-05-28 23:09:12 -04:00
show_regs ( regs ) ;
do_exit ( SIGKILL ) ; /* FIXME: implement i386 die() */
return ;
}
switch ( fault_num ) {
2012-03-29 15:23:54 -04:00
case INT_MEM_ERROR :
signo = SIGBUS ;
code = BUS_OBJERR ;
break ;
2010-05-28 23:09:12 -04:00
case INT_ILL :
2010-06-25 17:04:17 -04:00
if ( copy_from_user ( & instr , ( void __user * ) regs - > pc ,
sizeof ( instr ) ) ) {
pr_err ( " Unreadable instruction for INT_ILL: "
2010-05-28 23:09:12 -04:00
" %#lx \n " , regs - > pc ) ;
do_exit ( SIGKILL ) ;
return ;
}
2010-06-25 17:04:17 -04:00
if ( ! special_ill ( instr , & signo , & code ) ) {
2010-05-28 23:09:12 -04:00
signo = SIGILL ;
code = ILL_ILLOPC ;
}
address = regs - > pc ;
break ;
case INT_GPV :
# if CHIP_HAS_TILE_DMA()
if ( retry_gpv ( reason ) )
return ;
# endif
/*FALLTHROUGH*/
case INT_UDN_ACCESS :
case INT_IDN_ACCESS :
# if CHIP_HAS_SN()
case INT_SN_ACCESS :
# endif
signo = SIGILL ;
code = ILL_PRVREG ;
address = regs - > pc ;
break ;
case INT_SWINT_3 :
case INT_SWINT_2 :
case INT_SWINT_0 :
signo = SIGILL ;
code = ILL_ILLTRP ;
address = regs - > pc ;
break ;
case INT_UNALIGN_DATA :
2010-10-14 16:32:41 -04:00
# ifndef __tilegx__ /* Emulated support for single step debugging */
2010-05-28 23:09:12 -04:00
if ( unaligned_fixup > = 0 ) {
struct single_step_state * state =
current_thread_info ( ) - > step_state ;
2010-06-25 17:04:17 -04:00
if ( ! state | |
( void __user * ) ( regs - > pc ) ! = state - > buffer ) {
2010-05-28 23:09:12 -04:00
single_step_once ( regs ) ;
return ;
}
}
# endif
signo = SIGBUS ;
code = BUS_ADRALN ;
address = 0 ;
break ;
case INT_DOUBLE_FAULT :
/*
* For double fault , " reason " is actually passed as
2010-10-14 16:23:03 -04:00
* SYSTEM_SAVE_K_2 , the hypervisor ' s double - fault info , so
2010-05-28 23:09:12 -04:00
* we can provide the original fault number rather than
* the uninteresting " INT_DOUBLE_FAULT " so the user can
* learn what actually struck while PL0 ICS was set .
*/
fault_num = reason ;
signo = SIGILL ;
code = ILL_DBLFLT ;
address = regs - > pc ;
break ;
# ifdef __tilegx__
2012-03-29 14:52:00 -04:00
case INT_ILL_TRANS : {
/* Avoid a hardware erratum with the return address stack. */
fill_ra_stack ( ) ;
2010-05-28 23:09:12 -04:00
signo = SIGSEGV ;
2013-08-07 12:11:56 -04:00
address = reason ;
2010-05-28 23:09:12 -04:00
code = SEGV_MAPERR ;
break ;
2012-03-29 14:52:00 -04:00
}
2010-05-28 23:09:12 -04:00
# endif
default :
panic ( " Unexpected do_trap interrupt number %d " , fault_num ) ;
return ;
}
info . si_signo = signo ;
info . si_code = code ;
2010-06-25 17:04:17 -04:00
info . si_addr = ( void __user * ) address ;
2010-05-28 23:09:12 -04:00
if ( signo = = SIGILL )
info . si_trapno = fault_num ;
2012-03-29 15:23:54 -04:00
if ( signo ! = SIGTRAP )
trace_unhandled_signal ( " trap " , regs , address , signo ) ;
2010-05-28 23:09:12 -04:00
force_sig_info ( signo , & info , current ) ;
}
void kernel_double_fault ( int dummy , ulong pc , ulong lr , ulong sp , ulong r52 )
{
_dump_stack ( dummy , pc , lr , sp , r52 ) ;
2010-06-25 17:04:17 -04:00
pr_emerg ( " Double fault: exiting \n " ) ;
2010-05-28 23:09:12 -04:00
machine_halt ( ) ;
}