2019-05-29 07:18:00 -07:00
// SPDX-License-Identifier: GPL-2.0-only
2017-07-10 18:00:26 -07:00
/*
* Copyright ( C ) 2012 Regents of the University of California
*/
2019-10-17 15:21:28 -07:00
# include <linux/cpu.h>
2017-07-10 18:00:26 -07:00
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/sched.h>
# include <linux/sched/debug.h>
# include <linux/sched/signal.h>
# include <linux/signal.h>
# include <linux/kdebug.h>
# include <linux/uaccess.h>
2020-12-17 16:01:42 +00:00
# include <linux/kprobes.h>
2017-07-10 18:00:26 -07:00
# include <linux/mm.h>
# include <linux/module.h>
# include <linux/irq.h>
# include <asm/processor.h>
# include <asm/ptrace.h>
# include <asm/csr.h>
int show_unhandled_signals = 1 ;
extern asmlinkage void handle_exception ( void ) ;
static DEFINE_SPINLOCK ( die_lock ) ;
void die ( struct pt_regs * regs , const char * str )
{
static int die_counter ;
int ret ;
oops_enter ( ) ;
spin_lock_irq ( & die_lock ) ;
console_verbose ( ) ;
bust_spinlocks ( 1 ) ;
pr_emerg ( " %s [#%d] \n " , str , + + die_counter ) ;
print_modules ( ) ;
show_regs ( regs ) ;
2019-10-28 13:10:32 +01:00
ret = notify_die ( DIE_OOPS , str , regs , 0 , regs - > cause , SIGSEGV ) ;
2017-07-10 18:00:26 -07:00
bust_spinlocks ( 0 ) ;
add_taint ( TAINT_DIE , LOCKDEP_NOW_UNRELIABLE ) ;
spin_unlock_irq ( & die_lock ) ;
oops_exit ( ) ;
if ( in_interrupt ( ) )
panic ( " Fatal exception in interrupt " ) ;
if ( panic_on_oops )
panic ( " Fatal exception " ) ;
if ( ret ! = NOTIFY_STOP )
do_exit ( SIGSEGV ) ;
}
2019-02-05 19:10:48 -06:00
void do_trap ( struct pt_regs * regs , int signo , int code , unsigned long addr )
2017-07-10 18:00:26 -07:00
{
2019-02-05 19:10:48 -06:00
struct task_struct * tsk = current ;
2017-07-10 18:00:26 -07:00
if ( show_unhandled_signals & & unhandled_signal ( tsk , signo )
& & printk_ratelimit ( ) ) {
pr_info ( " %s[%d]: unhandled signal %d code 0x%x at 0x " REG_FMT ,
tsk - > comm , task_pid_nr ( tsk ) , signo , code , addr ) ;
2019-04-15 11:14:40 +02:00
print_vma_addr ( KERN_CONT " in " , instruction_pointer ( regs ) ) ;
2017-07-10 18:00:26 -07:00
pr_cont ( " \n " ) ;
show_regs ( regs ) ;
}
2019-05-23 11:04:24 -05:00
force_sig_fault ( signo , code , ( void __user * ) addr ) ;
2017-07-10 18:00:26 -07:00
}
static void do_trap_error ( struct pt_regs * regs , int signo , int code ,
unsigned long addr , const char * str )
{
2020-12-17 16:01:44 +00:00
current - > thread . bad_cause = regs - > cause ;
2017-07-10 18:00:26 -07:00
if ( user_mode ( regs ) ) {
2019-02-05 19:10:48 -06:00
do_trap ( regs , signo , code , addr ) ;
2017-07-10 18:00:26 -07:00
} else {
if ( ! fixup_exception ( regs ) )
die ( regs , str ) ;
}
}
# define DO_ERROR_INFO(name, signo, code, str) \
2019-10-17 22:20:05 -07:00
asmlinkage __visible void name ( struct pt_regs * regs ) \
2017-07-10 18:00:26 -07:00
{ \
2019-10-28 13:10:32 +01:00
do_trap_error ( regs , signo , code , regs - > epc , " Oops - " str ) ; \
2017-07-10 18:00:26 -07:00
}
DO_ERROR_INFO ( do_trap_unknown ,
SIGILL , ILL_ILLTRP , " unknown exception " ) ;
DO_ERROR_INFO ( do_trap_insn_misaligned ,
SIGBUS , BUS_ADRALN , " instruction address misaligned " ) ;
DO_ERROR_INFO ( do_trap_insn_fault ,
SIGSEGV , SEGV_ACCERR , " instruction access fault " ) ;
DO_ERROR_INFO ( do_trap_insn_illegal ,
SIGILL , ILL_ILLOPC , " illegal instruction " ) ;
DO_ERROR_INFO ( do_trap_load_fault ,
SIGSEGV , SEGV_ACCERR , " load access fault " ) ;
2020-03-16 09:47:36 +09:00
# ifndef CONFIG_RISCV_M_MODE
DO_ERROR_INFO ( do_trap_load_misaligned ,
SIGBUS , BUS_ADRALN , " Oops - load address misaligned " ) ;
2017-07-10 18:00:26 -07:00
DO_ERROR_INFO ( do_trap_store_misaligned ,
2020-03-16 09:47:36 +09:00
SIGBUS , BUS_ADRALN , " Oops - store (or AMO) address misaligned " ) ;
# else
int handle_misaligned_load ( struct pt_regs * regs ) ;
int handle_misaligned_store ( struct pt_regs * regs ) ;
asmlinkage void do_trap_load_misaligned ( struct pt_regs * regs )
{
if ( ! handle_misaligned_load ( regs ) )
return ;
do_trap_error ( regs , SIGBUS , BUS_ADRALN , regs - > epc ,
" Oops - load address misaligned " ) ;
}
asmlinkage void do_trap_store_misaligned ( struct pt_regs * regs )
{
if ( ! handle_misaligned_store ( regs ) )
return ;
do_trap_error ( regs , SIGBUS , BUS_ADRALN , regs - > epc ,
" Oops - store (or AMO) address misaligned " ) ;
}
# endif
2017-07-10 18:00:26 -07:00
DO_ERROR_INFO ( do_trap_store_fault ,
SIGSEGV , SEGV_ACCERR , " store (or AMO) access fault " ) ;
DO_ERROR_INFO ( do_trap_ecall_u ,
SIGILL , ILL_ILLTRP , " environment call from U-mode " ) ;
DO_ERROR_INFO ( do_trap_ecall_s ,
SIGILL , ILL_ILLTRP , " environment call from S-mode " ) ;
DO_ERROR_INFO ( do_trap_ecall_m ,
SIGILL , ILL_ILLTRP , " environment call from M-mode " ) ;
2019-03-05 11:23:34 +08:00
static inline unsigned long get_break_insn_length ( unsigned long pc )
{
bug_insn_t insn ;
2020-06-17 09:37:55 +02:00
if ( get_kernel_nofault ( insn , ( bug_insn_t * ) pc ) )
2019-03-05 11:23:34 +08:00
return 0 ;
2020-03-10 00:55:42 +08:00
return GET_INSN_LENGTH ( insn ) ;
2019-03-05 11:23:34 +08:00
}
2019-10-17 22:20:05 -07:00
asmlinkage __visible void do_trap_break ( struct pt_regs * regs )
2017-07-10 18:00:26 -07:00
{
2020-12-17 16:01:42 +00:00
# ifdef CONFIG_KPROBES
if ( kprobe_single_step_handler ( regs ) )
return ;
if ( kprobe_breakpoint_handler ( regs ) )
return ;
# endif
2020-12-17 16:01:44 +00:00
# ifdef CONFIG_UPROBES
if ( uprobe_single_step_handler ( regs ) )
return ;
if ( uprobe_breakpoint_handler ( regs ) )
return ;
# endif
current - > thread . bad_cause = regs - > cause ;
2020-12-17 16:01:42 +00:00
2019-10-17 19:37:30 +02:00
if ( user_mode ( regs ) )
2019-10-28 13:10:32 +01:00
force_sig_fault ( SIGTRAP , TRAP_BRKPT , ( void __user * ) regs - > epc ) ;
2020-04-16 10:38:05 +08:00
# ifdef CONFIG_KGDB
else if ( notify_die ( DIE_TRAP , " EBREAK " , regs , 0 , regs - > cause , SIGTRAP )
= = NOTIFY_STOP )
return ;
# endif
2019-10-28 13:10:32 +01:00
else if ( report_bug ( regs - > epc , regs ) = = BUG_TRAP_TYPE_WARN )
regs - > epc + = get_break_insn_length ( regs - > epc ) ;
2019-10-17 19:37:30 +02:00
else
die ( regs , " Kernel BUG " ) ;
2017-07-10 18:00:26 -07:00
}
# ifdef CONFIG_GENERIC_BUG
int is_valid_bugaddr ( unsigned long pc )
{
bug_insn_t insn ;
2019-03-05 11:23:35 +08:00
if ( pc < VMALLOC_START )
2017-07-10 18:00:26 -07:00
return 0 ;
2020-06-17 09:37:55 +02:00
if ( get_kernel_nofault ( insn , ( bug_insn_t * ) pc ) )
2017-07-10 18:00:26 -07:00
return 0 ;
2019-03-05 11:23:34 +08:00
if ( ( insn & __INSN_LENGTH_MASK ) = = __INSN_LENGTH_32 )
return ( insn = = __BUG_INSN_32 ) ;
else
return ( ( insn & __COMPRESSED_INSN_MASK ) = = __BUG_INSN_16 ) ;
2017-07-10 18:00:26 -07:00
}
# endif /* CONFIG_GENERIC_BUG */
2020-07-15 16:30:06 -07:00
/* stvec & scratch is already set from head.S */
2020-03-17 18:11:43 -07:00
void trap_init ( void )
2017-07-10 18:00:26 -07:00
{
}