x86: initial changes to unify traps_32.c and traps_64.c

This patch does not change the generated object files.

Signed-off-by: Alexander van Heukelum <heukelum@fastmail.fm>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
This commit is contained in:
Alexander van Heukelum 2008-07-02 01:29:44 +02:00 committed by Ingo Molnar
parent e93ef949fd
commit a8c1be9d2e
2 changed files with 191 additions and 196 deletions

View file

@ -1,5 +1,6 @@
/* /*
* Copyright (C) 1991, 1992 Linus Torvalds * Copyright (C) 1991, 1992 Linus Torvalds
* Copyright (C) 2000, 2001, 2002 Andi Kleen, SuSE Labs
* *
* Pentium III FXSR, SSE support * Pentium III FXSR, SSE support
* Gareth Hughes <gareth@valinux.com>, May 2000 * Gareth Hughes <gareth@valinux.com>, May 2000
@ -130,7 +131,8 @@ void printk_address(unsigned long address, int reliable)
#endif #endif
} }
static inline int valid_stack_ptr(struct thread_info *tinfo, void *p, unsigned size) static inline int valid_stack_ptr(struct thread_info *tinfo,
void *p, unsigned int size)
{ {
return p > (void *)tinfo && return p > (void *)tinfo &&
p <= (void *)tinfo + THREAD_SIZE - size; p <= (void *)tinfo + THREAD_SIZE - size;
@ -138,14 +140,14 @@ static inline int valid_stack_ptr(struct thread_info *tinfo, void *p, unsigned s
/* The form of the top of the frame on the stack */ /* The form of the top of the frame on the stack */
struct stack_frame { struct stack_frame {
struct stack_frame *next_frame; struct stack_frame *next_frame;
unsigned long return_address; unsigned long return_address;
}; };
static inline unsigned long static inline unsigned long
print_context_stack(struct thread_info *tinfo, print_context_stack(struct thread_info *tinfo,
unsigned long *stack, unsigned long bp, unsigned long *stack, unsigned long bp,
const struct stacktrace_ops *ops, void *data) const struct stacktrace_ops *ops, void *data)
{ {
struct stack_frame *frame = (struct stack_frame *)bp; struct stack_frame *frame = (struct stack_frame *)bp;
@ -167,8 +169,6 @@ print_context_stack(struct thread_info *tinfo,
return bp; return bp;
} }
#define MSG(msg) ops->warning(data, msg)
void dump_trace(struct task_struct *task, struct pt_regs *regs, void dump_trace(struct task_struct *task, struct pt_regs *regs,
unsigned long *stack, unsigned long bp, unsigned long *stack, unsigned long bp,
const struct stacktrace_ops *ops, void *data) const struct stacktrace_ops *ops, void *data)
@ -178,7 +178,6 @@ void dump_trace(struct task_struct *task, struct pt_regs *regs,
if (!stack) { if (!stack) {
unsigned long dummy; unsigned long dummy;
stack = &dummy; stack = &dummy;
if (task != current) if (task != current)
stack = (unsigned long *)task->thread.sp; stack = (unsigned long *)task->thread.sp;
@ -196,7 +195,7 @@ void dump_trace(struct task_struct *task, struct pt_regs *regs,
} }
#endif #endif
while (1) { for (;;) {
struct thread_info *context; struct thread_info *context;
context = (struct thread_info *) context = (struct thread_info *)
@ -248,10 +247,10 @@ static void print_trace_address(void *data, unsigned long addr, int reliable)
} }
static const struct stacktrace_ops print_trace_ops = { static const struct stacktrace_ops print_trace_ops = {
.warning = print_trace_warning, .warning = print_trace_warning,
.warning_symbol = print_trace_warning_symbol, .warning_symbol = print_trace_warning_symbol,
.stack = print_trace_stack, .stack = print_trace_stack,
.address = print_trace_address, .address = print_trace_address,
}; };
static void static void
@ -351,15 +350,14 @@ void show_registers(struct pt_regs *regs)
printk(KERN_EMERG "Code: "); printk(KERN_EMERG "Code: ");
ip = (u8 *)regs->ip - code_prologue; ip = (u8 *)regs->ip - code_prologue;
if (ip < (u8 *)PAGE_OFFSET || if (ip < (u8 *)PAGE_OFFSET || probe_kernel_address(ip, c)) {
probe_kernel_address(ip, c)) {
/* try starting at EIP */ /* try starting at EIP */
ip = (u8 *)regs->ip; ip = (u8 *)regs->ip;
code_len = code_len - code_prologue + 1; code_len = code_len - code_prologue + 1;
} }
for (i = 0; i < code_len; i++, ip++) { for (i = 0; i < code_len; i++, ip++) {
if (ip < (u8 *)PAGE_OFFSET || if (ip < (u8 *)PAGE_OFFSET ||
probe_kernel_address(ip, c)) { probe_kernel_address(ip, c)) {
printk(" Bad EIP value."); printk(" Bad EIP value.");
break; break;
} }
@ -546,7 +544,7 @@ void do_##name(struct pt_regs *regs, long error_code) \
{ \ { \
trace_hardirqs_fixup(); \ trace_hardirqs_fixup(); \
if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \ if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \
== NOTIFY_STOP) \ == NOTIFY_STOP) \
return; \ return; \
do_trap(trapnr, signr, str, 0, regs, error_code, NULL); \ do_trap(trapnr, signr, str, 0, regs, error_code, NULL); \
} }
@ -562,7 +560,7 @@ void do_##name(struct pt_regs *regs, long error_code) \
info.si_code = sicode; \ info.si_code = sicode; \
info.si_addr = (void __user *)siaddr; \ info.si_addr = (void __user *)siaddr; \
if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \ if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \
== NOTIFY_STOP) \ == NOTIFY_STOP) \
return; \ return; \
do_trap(trapnr, signr, str, 0, regs, error_code, &info); \ do_trap(trapnr, signr, str, 0, regs, error_code, &info); \
} }
@ -571,7 +569,7 @@ void do_##name(struct pt_regs *regs, long error_code) \
void do_##name(struct pt_regs *regs, long error_code) \ void do_##name(struct pt_regs *regs, long error_code) \
{ \ { \
if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \ if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \
== NOTIFY_STOP) \ == NOTIFY_STOP) \
return; \ return; \
do_trap(trapnr, signr, str, 1, regs, error_code, NULL); \ do_trap(trapnr, signr, str, 1, regs, error_code, NULL); \
} }
@ -586,22 +584,22 @@ void do_##name(struct pt_regs *regs, long error_code) \
info.si_addr = (void __user *)siaddr; \ info.si_addr = (void __user *)siaddr; \
trace_hardirqs_fixup(); \ trace_hardirqs_fixup(); \
if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \ if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \
== NOTIFY_STOP) \ == NOTIFY_STOP) \
return; \ return; \
do_trap(trapnr, signr, str, 1, regs, error_code, &info); \ do_trap(trapnr, signr, str, 1, regs, error_code, &info); \
} }
DO_VM86_ERROR_INFO(0, SIGFPE, "divide error", divide_error, FPE_INTDIV, regs->ip) DO_VM86_ERROR_INFO(0, SIGFPE, "divide error", divide_error, FPE_INTDIV, regs->ip)
#ifndef CONFIG_KPROBES #ifndef CONFIG_KPROBES
DO_VM86_ERROR(3, SIGTRAP, "int3", int3) DO_VM86_ERROR(3, SIGTRAP, "int3", int3)
#endif #endif
DO_VM86_ERROR(4, SIGSEGV, "overflow", overflow) DO_VM86_ERROR(4, SIGSEGV, "overflow", overflow)
DO_VM86_ERROR(5, SIGSEGV, "bounds", bounds) DO_VM86_ERROR(5, SIGSEGV, "bounds", bounds)
DO_ERROR_INFO(6, SIGILL, "invalid opcode", invalid_op, ILL_ILLOPN, regs->ip, 0) DO_ERROR_INFO(6, SIGILL, "invalid opcode", invalid_op, ILL_ILLOPN, regs->ip, 0)
DO_ERROR(9, SIGFPE, "coprocessor segment overrun", coprocessor_segment_overrun) DO_ERROR(9, SIGFPE, "coprocessor segment overrun", coprocessor_segment_overrun)
DO_ERROR(10, SIGSEGV, "invalid TSS", invalid_TSS) DO_ERROR(10, SIGSEGV, "invalid TSS", invalid_TSS)
DO_ERROR(11, SIGBUS, "segment not present", segment_not_present) DO_ERROR(11, SIGBUS, "segment not present", segment_not_present)
DO_ERROR(12, SIGBUS, "stack segment", stack_segment) DO_ERROR(12, SIGBUS, "stack segment", stack_segment)
DO_ERROR_INFO(17, SIGBUS, "alignment check", alignment_check, BUS_ADRALN, 0, 0) DO_ERROR_INFO(17, SIGBUS, "alignment check", alignment_check, BUS_ADRALN, 0, 0)
DO_ERROR_INFO(32, SIGILL, "iret exception", iret_error, ILL_BADSTK, 0, 1) DO_ERROR_INFO(32, SIGILL, "iret exception", iret_error, ILL_BADSTK, 0, 1)
@ -799,7 +797,7 @@ static notrace __kprobes void default_do_nmi(struct pt_regs *regs)
if (!(reason & 0xc0)) { if (!(reason & 0xc0)) {
if (notify_die(DIE_NMI_IPI, "nmi_ipi", regs, reason, 2, SIGINT) if (notify_die(DIE_NMI_IPI, "nmi_ipi", regs, reason, 2, SIGINT)
== NOTIFY_STOP) == NOTIFY_STOP)
return; return;
#ifdef CONFIG_X86_LOCAL_APIC #ifdef CONFIG_X86_LOCAL_APIC
/* /*
@ -818,6 +816,8 @@ static notrace __kprobes void default_do_nmi(struct pt_regs *regs)
} }
if (notify_die(DIE_NMI, "nmi", regs, reason, 2, SIGINT) == NOTIFY_STOP) if (notify_die(DIE_NMI, "nmi", regs, reason, 2, SIGINT) == NOTIFY_STOP)
return; return;
/* AK: following checks seem to be broken on modern chipsets. FIXME */
if (reason & 0x80) if (reason & 0x80)
mem_parity_error(reason, regs); mem_parity_error(reason, regs);
if (reason & 0x40) if (reason & 0x40)
@ -915,7 +915,7 @@ void __kprobes do_debug(struct pt_regs *regs, long error_code)
tsk->thread.debugctlmsr = 0; tsk->thread.debugctlmsr = 0;
if (notify_die(DIE_DEBUG, "debug", regs, condition, error_code, if (notify_die(DIE_DEBUG, "debug", regs, condition, error_code,
SIGTRAP) == NOTIFY_STOP) SIGTRAP) == NOTIFY_STOP)
return; return;
/* It's safe to allow irq's after DR6 has been saved */ /* It's safe to allow irq's after DR6 has been saved */
if (regs->flags & X86_EFLAGS_IF) if (regs->flags & X86_EFLAGS_IF)
@ -997,7 +997,7 @@ void math_error(void __user *ip)
* C1 reg you need in case of a stack fault, 0x040 is the stack * C1 reg you need in case of a stack fault, 0x040 is the stack
* fault bit. We should only be taking one exception at a time, * fault bit. We should only be taking one exception at a time,
* so if this combination doesn't produce any single exception, * so if this combination doesn't produce any single exception,
* then we have a bad program that isn't syncronizing its FPU usage * then we have a bad program that isn't synchronizing its FPU usage
* and it will suffer the consequences since we won't be able to * and it will suffer the consequences since we won't be able to
* fully reproduce the context of the exception * fully reproduce the context of the exception
*/ */
@ -1006,7 +1006,7 @@ void math_error(void __user *ip)
switch (swd & ~cwd & 0x3f) { switch (swd & ~cwd & 0x3f) {
case 0x000: /* No unmasked exception */ case 0x000: /* No unmasked exception */
return; return;
default: /* Multiple exceptions */ default: /* Multiple exceptions */
break; break;
case 0x001: /* Invalid Op */ case 0x001: /* Invalid Op */
/* /*
@ -1198,16 +1198,16 @@ void __init trap_init(void)
early_iounmap(p, 4); early_iounmap(p, 4);
#endif #endif
set_trap_gate(0, &divide_error); set_trap_gate(0, &divide_error);
set_intr_gate(1, &debug); set_intr_gate(1, &debug);
set_intr_gate(2, &nmi); set_intr_gate(2, &nmi);
set_system_intr_gate(3, &int3); /* int3/4 can be called from all */ set_system_intr_gate(3, &int3); /* int3 can be called from all */
set_system_gate(4, &overflow); set_system_gate(4, &overflow); /* int4 can be called from all */
set_trap_gate(5, &bounds); set_trap_gate(5, &bounds);
set_trap_gate(6, &invalid_op); set_trap_gate(6, &invalid_op);
set_trap_gate(7, &device_not_available); set_trap_gate(7, &device_not_available);
set_task_gate(8, GDT_ENTRY_DOUBLEFAULT_TSS); set_task_gate(8, GDT_ENTRY_DOUBLEFAULT_TSS);
set_trap_gate(9, &coprocessor_segment_overrun); set_trap_gate(9, &coprocessor_segment_overrun);
set_trap_gate(10, &invalid_TSS); set_trap_gate(10, &invalid_TSS);
set_trap_gate(11, &segment_not_present); set_trap_gate(11, &segment_not_present);
set_trap_gate(12, &stack_segment); set_trap_gate(12, &stack_segment);

View file

@ -205,8 +205,6 @@ static unsigned long *in_exception_stack(unsigned cpu, unsigned long stack,
return NULL; return NULL;
} }
#define MSG(txt) ops->warning(data, txt)
/* /*
* x86-64 can have up to three kernel stacks: * x86-64 can have up to three kernel stacks:
* process stack * process stack
@ -233,11 +231,11 @@ struct stack_frame {
unsigned long return_address; unsigned long return_address;
}; };
static inline unsigned long
static inline unsigned long print_context_stack(struct thread_info *tinfo, print_context_stack(struct thread_info *tinfo,
unsigned long *stack, unsigned long bp, unsigned long *stack, unsigned long bp,
const struct stacktrace_ops *ops, void *data, const struct stacktrace_ops *ops, void *data,
unsigned long *end) unsigned long *end)
{ {
struct stack_frame *frame = (struct stack_frame *)bp; struct stack_frame *frame = (struct stack_frame *)bp;
@ -259,7 +257,7 @@ static inline unsigned long print_context_stack(struct thread_info *tinfo,
return bp; return bp;
} }
void dump_trace(struct task_struct *tsk, struct pt_regs *regs, void dump_trace(struct task_struct *task, struct pt_regs *regs,
unsigned long *stack, unsigned long bp, unsigned long *stack, unsigned long bp,
const struct stacktrace_ops *ops, void *data) const struct stacktrace_ops *ops, void *data)
{ {
@ -268,31 +266,29 @@ void dump_trace(struct task_struct *tsk, struct pt_regs *regs,
unsigned used = 0; unsigned used = 0;
struct thread_info *tinfo; struct thread_info *tinfo;
if (!tsk) if (!task)
tsk = current; task = current;
tinfo = task_thread_info(tsk); tinfo = task_thread_info(task);
if (!stack) { if (!stack) {
unsigned long dummy; unsigned long dummy;
stack = &dummy; stack = &dummy;
if (tsk && tsk != current) if (task && task != current)
stack = (unsigned long *)tsk->thread.sp; stack = (unsigned long *)task->thread.sp;
} }
#ifdef CONFIG_FRAME_POINTER #ifdef CONFIG_FRAME_POINTER
if (!bp) { if (!bp) {
if (tsk == current) { if (task == current) {
/* Grab bp right from our regs */ /* Grab bp right from our regs */
asm("movq %%rbp, %0" : "=r" (bp):); asm("movq %%rbp, %0" : "=r" (bp) :);
} else { } else {
/* bp is the last reg pushed by switch_to */ /* bp is the last reg pushed by switch_to */
bp = *(unsigned long *) tsk->thread.sp; bp = *(unsigned long *) task->thread.sp;
} }
} }
#endif #endif
/* /*
* Print function call entries in all stacks, starting at the * Print function call entries in all stacks, starting at the
* current stack address. If the stacks consist of nested * current stack address. If the stacks consist of nested
@ -382,18 +378,17 @@ static const struct stacktrace_ops print_trace_ops = {
.address = print_trace_address, .address = print_trace_address,
}; };
void void show_trace(struct task_struct *task, struct pt_regs *regs,
show_trace(struct task_struct *tsk, struct pt_regs *regs, unsigned long *stack, unsigned long *stack, unsigned long bp)
unsigned long bp)
{ {
printk("\nCall Trace:\n"); printk("\nCall Trace:\n");
dump_trace(tsk, regs, stack, bp, &print_trace_ops, NULL); dump_trace(task, regs, stack, bp, &print_trace_ops, NULL);
printk("\n"); printk("\n");
} }
static void static void
_show_stack(struct task_struct *tsk, struct pt_regs *regs, unsigned long *sp, _show_stack(struct task_struct *task, struct pt_regs *regs,
unsigned long bp) unsigned long *sp, unsigned long bp)
{ {
unsigned long *stack; unsigned long *stack;
int i; int i;
@ -405,14 +400,14 @@ _show_stack(struct task_struct *tsk, struct pt_regs *regs, unsigned long *sp,
// back trace for this cpu. // back trace for this cpu.
if (sp == NULL) { if (sp == NULL) {
if (tsk) if (task)
sp = (unsigned long *)tsk->thread.sp; sp = (unsigned long *)task->thread.sp;
else else
sp = (unsigned long *)&sp; sp = (unsigned long *)&sp;
} }
stack = sp; stack = sp;
for(i=0; i < kstack_depth_to_print; i++) { for (i = 0; i < kstack_depth_to_print; i++) {
if (stack >= irqstack && stack <= irqstack_end) { if (stack >= irqstack && stack <= irqstack_end) {
if (stack == irqstack_end) { if (stack == irqstack_end) {
stack = (unsigned long *) (irqstack_end[-1]); stack = (unsigned long *) (irqstack_end[-1]);
@ -427,12 +422,12 @@ _show_stack(struct task_struct *tsk, struct pt_regs *regs, unsigned long *sp,
printk(" %016lx", *stack++); printk(" %016lx", *stack++);
touch_nmi_watchdog(); touch_nmi_watchdog();
} }
show_trace(tsk, regs, sp, bp); show_trace(task, regs, sp, bp);
} }
void show_stack(struct task_struct *tsk, unsigned long * sp) void show_stack(struct task_struct *task, unsigned long *sp)
{ {
_show_stack(tsk, NULL, sp, 0); _show_stack(task, NULL, sp, 0);
} }
/* /*
@ -440,7 +435,7 @@ void show_stack(struct task_struct *tsk, unsigned long * sp)
*/ */
void dump_stack(void) void dump_stack(void)
{ {
unsigned long dummy; unsigned long stack;
unsigned long bp = 0; unsigned long bp = 0;
#ifdef CONFIG_FRAME_POINTER #ifdef CONFIG_FRAME_POINTER
@ -453,7 +448,7 @@ void dump_stack(void)
init_utsname()->release, init_utsname()->release,
(int)strcspn(init_utsname()->version, " "), (int)strcspn(init_utsname()->version, " "),
init_utsname()->version); init_utsname()->version);
show_trace(NULL, NULL, &dummy, bp); show_trace(NULL, NULL, &stack, bp);
} }
EXPORT_SYMBOL(dump_stack); EXPORT_SYMBOL(dump_stack);
@ -488,7 +483,7 @@ void show_registers(struct pt_regs *regs)
printk(KERN_EMERG "Code: "); printk(KERN_EMERG "Code: ");
if (ip < (u8 *)PAGE_OFFSET || probe_kernel_address(ip, c)) { if (ip < (u8 *)PAGE_OFFSET || probe_kernel_address(ip, c)) {
/* try starting at RIP */ /* try starting at RIP */
ip = (u8 *) regs->ip; ip = (u8 *)regs->ip;
code_len = code_len - code_prologue + 1; code_len = code_len - code_prologue + 1;
} }
for (i = 0; i < code_len; i++, ip++) { for (i = 0; i < code_len; i++, ip++) {
@ -504,7 +499,7 @@ void show_registers(struct pt_regs *regs)
} }
} }
printk("\n"); printk("\n");
} }
int is_valid_bugaddr(unsigned long ip) int is_valid_bugaddr(unsigned long ip)
{ {
@ -576,8 +571,10 @@ int __kprobes __die(const char * str, struct pt_regs * regs, long err)
printk("DEBUG_PAGEALLOC"); printk("DEBUG_PAGEALLOC");
#endif #endif
printk("\n"); printk("\n");
if (notify_die(DIE_OOPS, str, regs, err, current->thread.trap_no, SIGSEGV) == NOTIFY_STOP) if (notify_die(DIE_OOPS, str, regs, err,
current->thread.trap_no, SIGSEGV) == NOTIFY_STOP)
return 1; return 1;
show_registers(regs); show_registers(regs);
add_taint(TAINT_DIE); add_taint(TAINT_DIE);
/* Executive summary in case the oops scrolled away */ /* Executive summary in case the oops scrolled away */
@ -589,7 +586,7 @@ int __kprobes __die(const char * str, struct pt_regs * regs, long err)
return 0; return 0;
} }
void die(const char * str, struct pt_regs * regs, long err) void die(const char * str, struct pt_regs *regs, long err)
{ {
unsigned long flags = oops_begin(); unsigned long flags = oops_begin();
@ -606,8 +603,7 @@ die_nmi(char *str, struct pt_regs *regs, int do_panic)
{ {
unsigned long flags; unsigned long flags;
if (notify_die(DIE_NMIWATCHDOG, str, regs, 0, 2, SIGINT) == if (notify_die(DIE_NMIWATCHDOG, str, regs, 0, 2, SIGINT) == NOTIFY_STOP)
NOTIFY_STOP)
return; return;
flags = oops_begin(); flags = oops_begin();
@ -629,9 +625,9 @@ die_nmi(char *str, struct pt_regs *regs, int do_panic)
do_exit(SIGBUS); do_exit(SIGBUS);
} }
static void __kprobes do_trap(int trapnr, int signr, char *str, static void __kprobes
struct pt_regs * regs, long error_code, do_trap(int trapnr, int signr, char *str, struct pt_regs *regs,
siginfo_t *info) long error_code, siginfo_t *info)
{ {
struct task_struct *tsk = current; struct task_struct *tsk = current;
@ -676,38 +672,38 @@ static void __kprobes do_trap(int trapnr, int signr, char *str,
} }
#define DO_ERROR(trapnr, signr, str, name) \ #define DO_ERROR(trapnr, signr, str, name) \
asmlinkage void do_##name(struct pt_regs * regs, long error_code) \ asmlinkage void do_##name(struct pt_regs * regs, long error_code) \
{ \ { \
if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \ if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \
== NOTIFY_STOP) \ == NOTIFY_STOP) \
return; \ return; \
conditional_sti(regs); \ conditional_sti(regs); \
do_trap(trapnr, signr, str, regs, error_code, NULL); \ do_trap(trapnr, signr, str, regs, error_code, NULL); \
} }
#define DO_ERROR_INFO(trapnr, signr, str, name, sicode, siaddr) \ #define DO_ERROR_INFO(trapnr, signr, str, name, sicode, siaddr) \
asmlinkage void do_##name(struct pt_regs * regs, long error_code) \ asmlinkage void do_##name(struct pt_regs * regs, long error_code) \
{ \ { \
siginfo_t info; \ siginfo_t info; \
info.si_signo = signr; \ info.si_signo = signr; \
info.si_errno = 0; \ info.si_errno = 0; \
info.si_code = sicode; \ info.si_code = sicode; \
info.si_addr = (void __user *)siaddr; \ info.si_addr = (void __user *)siaddr; \
trace_hardirqs_fixup(); \ trace_hardirqs_fixup(); \
if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \ if (notify_die(DIE_TRAP, str, regs, error_code, trapnr, signr) \
== NOTIFY_STOP) \ == NOTIFY_STOP) \
return; \ return; \
conditional_sti(regs); \ conditional_sti(regs); \
do_trap(trapnr, signr, str, regs, error_code, &info); \ do_trap(trapnr, signr, str, regs, error_code, &info); \
} }
DO_ERROR_INFO( 0, SIGFPE, "divide error", divide_error, FPE_INTDIV, regs->ip) DO_ERROR_INFO(0, SIGFPE, "divide error", divide_error, FPE_INTDIV, regs->ip)
DO_ERROR( 4, SIGSEGV, "overflow", overflow) DO_ERROR(4, SIGSEGV, "overflow", overflow)
DO_ERROR( 5, SIGSEGV, "bounds", bounds) DO_ERROR(5, SIGSEGV, "bounds", bounds)
DO_ERROR_INFO( 6, SIGILL, "invalid opcode", invalid_op, ILL_ILLOPN, regs->ip) DO_ERROR_INFO(6, SIGILL, "invalid opcode", invalid_op, ILL_ILLOPN, regs->ip)
DO_ERROR( 9, SIGFPE, "coprocessor segment overrun", coprocessor_segment_overrun) DO_ERROR(9, SIGFPE, "coprocessor segment overrun", coprocessor_segment_overrun)
DO_ERROR(10, SIGSEGV, "invalid TSS", invalid_TSS) DO_ERROR(10, SIGSEGV, "invalid TSS", invalid_TSS)
DO_ERROR(11, SIGBUS, "segment not present", segment_not_present) DO_ERROR(11, SIGBUS, "segment not present", segment_not_present)
DO_ERROR_INFO(17, SIGBUS, "alignment check", alignment_check, BUS_ADRALN, 0) DO_ERROR_INFO(17, SIGBUS, "alignment check", alignment_check, BUS_ADRALN, 0)
/* Runs on IST stack */ /* Runs on IST stack */
@ -775,14 +771,14 @@ asmlinkage void __kprobes do_general_protection(struct pt_regs * regs,
} }
static notrace __kprobes void static notrace __kprobes void
mem_parity_error(unsigned char reason, struct pt_regs * regs) mem_parity_error(unsigned char reason, struct pt_regs *regs)
{ {
printk(KERN_EMERG "Uhhuh. NMI received for unknown reason %02x.\n", printk(KERN_EMERG "Uhhuh. NMI received for unknown reason %02x.\n",
reason); reason);
printk(KERN_EMERG "You have some hardware problem, likely on the PCI bus.\n"); printk(KERN_EMERG "You have some hardware problem, likely on the PCI bus.\n");
#if defined(CONFIG_EDAC) #if defined(CONFIG_EDAC)
if(edac_handler_set()) { if (edac_handler_set()) {
edac_atomic_assert_error(); edac_atomic_assert_error();
return; return;
} }
@ -799,7 +795,7 @@ mem_parity_error(unsigned char reason, struct pt_regs * regs)
} }
static notrace __kprobes void static notrace __kprobes void
io_check_error(unsigned char reason, struct pt_regs * regs) io_check_error(unsigned char reason, struct pt_regs *regs)
{ {
printk("NMI: IOCK error (debug interrupt?)\n"); printk("NMI: IOCK error (debug interrupt?)\n");
show_registers(regs); show_registers(regs);
@ -836,7 +832,7 @@ asmlinkage notrace __kprobes void default_do_nmi(struct pt_regs *regs)
cpu = smp_processor_id(); cpu = smp_processor_id();
/* Only the BSP gets external NMIs from the system. */ /* Only the BSP gets external NMIs from the system. */
if (!cpu) if (!cpu)
reason = get_nmi_reason(); reason = get_nmi_reason();
@ -848,18 +844,17 @@ asmlinkage notrace __kprobes void default_do_nmi(struct pt_regs *regs)
* Ok, so this is none of the documented NMI sources, * Ok, so this is none of the documented NMI sources,
* so it must be the NMI watchdog. * so it must be the NMI watchdog.
*/ */
if (nmi_watchdog_tick(regs,reason)) if (nmi_watchdog_tick(regs, reason))
return; return;
if (!do_nmi_callback(regs,cpu)) if (!do_nmi_callback(regs, cpu))
unknown_nmi_error(reason, regs); unknown_nmi_error(reason, regs);
return; return;
} }
if (notify_die(DIE_NMI, "nmi", regs, reason, 2, SIGINT) == NOTIFY_STOP) if (notify_die(DIE_NMI, "nmi", regs, reason, 2, SIGINT) == NOTIFY_STOP)
return; return;
/* AK: following checks seem to be broken on modern chipsets. FIXME */ /* AK: following checks seem to be broken on modern chipsets. FIXME */
if (reason & 0x80) if (reason & 0x80)
mem_parity_error(reason, regs); mem_parity_error(reason, regs);
if (reason & 0x40) if (reason & 0x40)
@ -870,9 +865,12 @@ asmlinkage notrace __kprobes void
do_nmi(struct pt_regs *regs, long error_code) do_nmi(struct pt_regs *regs, long error_code)
{ {
nmi_enter(); nmi_enter();
add_pda(__nmi_count, 1); add_pda(__nmi_count, 1);
if (!ignore_nmis) if (!ignore_nmis)
default_do_nmi(regs); default_do_nmi(regs);
nmi_exit(); nmi_exit();
} }
@ -889,13 +887,14 @@ void restart_nmi(void)
} }
/* runs on IST stack. */ /* runs on IST stack. */
asmlinkage void __kprobes do_int3(struct pt_regs * regs, long error_code) asmlinkage void __kprobes do_int3(struct pt_regs *regs, long error_code)
{ {
trace_hardirqs_fixup(); trace_hardirqs_fixup();
if (notify_die(DIE_INT3, "int3", regs, error_code, 3, SIGTRAP) == NOTIFY_STOP) { if (notify_die(DIE_INT3, "int3", regs, error_code, 3, SIGTRAP)
== NOTIFY_STOP)
return; return;
}
preempt_conditional_sti(regs); preempt_conditional_sti(regs);
do_trap(3, SIGTRAP, "int3", regs, error_code, NULL); do_trap(3, SIGTRAP, "int3", regs, error_code, NULL);
preempt_conditional_cli(regs); preempt_conditional_cli(regs);
@ -948,21 +947,19 @@ asmlinkage void __kprobes do_debug(struct pt_regs * regs,
/* Mask out spurious debug traps due to lazy DR7 setting */ /* Mask out spurious debug traps due to lazy DR7 setting */
if (condition & (DR_TRAP0|DR_TRAP1|DR_TRAP2|DR_TRAP3)) { if (condition & (DR_TRAP0|DR_TRAP1|DR_TRAP2|DR_TRAP3)) {
if (!tsk->thread.debugreg7) { if (!tsk->thread.debugreg7)
goto clear_dr7; goto clear_dr7;
}
} }
tsk->thread.debugreg6 = condition; tsk->thread.debugreg6 = condition;
/* /*
* Single-stepping through TF: make sure we ignore any events in * Single-stepping through TF: make sure we ignore any events in
* kernel space (but re-enable TF when returning to user mode). * kernel space (but re-enable TF when returning to user mode).
*/ */
if (condition & DR_STEP) { if (condition & DR_STEP) {
if (!user_mode(regs)) if (!user_mode(regs))
goto clear_TF_reenable; goto clear_TF_reenable;
} }
/* Ok, finally something we can handle */ /* Ok, finally something we can handle */
@ -975,7 +972,7 @@ asmlinkage void __kprobes do_debug(struct pt_regs * regs,
force_sig_info(SIGTRAP, &info, tsk); force_sig_info(SIGTRAP, &info, tsk);
clear_dr7: clear_dr7:
set_debugreg(0UL, 7); set_debugreg(0, 7);
preempt_conditional_cli(regs); preempt_conditional_cli(regs);
return; return;
@ -983,6 +980,7 @@ asmlinkage void __kprobes do_debug(struct pt_regs * regs,
set_tsk_thread_flag(tsk, TIF_SINGLESTEP); set_tsk_thread_flag(tsk, TIF_SINGLESTEP);
regs->flags &= ~X86_EFLAGS_TF; regs->flags &= ~X86_EFLAGS_TF;
preempt_conditional_cli(regs); preempt_conditional_cli(regs);
return;
} }
static int kernel_math_error(struct pt_regs *regs, const char *str, int trapnr) static int kernel_math_error(struct pt_regs *regs, const char *str, int trapnr)
@ -1005,7 +1003,7 @@ static int kernel_math_error(struct pt_regs *regs, const char *str, int trapnr)
asmlinkage void do_coprocessor_error(struct pt_regs *regs) asmlinkage void do_coprocessor_error(struct pt_regs *regs)
{ {
void __user *ip = (void __user *)(regs->ip); void __user *ip = (void __user *)(regs->ip);
struct task_struct * task; struct task_struct *task;
siginfo_t info; siginfo_t info;
unsigned short cwd, swd; unsigned short cwd, swd;
@ -1038,30 +1036,30 @@ asmlinkage void do_coprocessor_error(struct pt_regs *regs)
cwd = get_fpu_cwd(task); cwd = get_fpu_cwd(task);
swd = get_fpu_swd(task); swd = get_fpu_swd(task);
switch (swd & ~cwd & 0x3f) { switch (swd & ~cwd & 0x3f) {
case 0x000: case 0x000: /* No unmasked exception */
default: default: /* Multiple exceptions */
break; break;
case 0x001: /* Invalid Op */ case 0x001: /* Invalid Op */
/* /*
* swd & 0x240 == 0x040: Stack Underflow * swd & 0x240 == 0x040: Stack Underflow
* swd & 0x240 == 0x240: Stack Overflow * swd & 0x240 == 0x240: Stack Overflow
* User must clear the SF bit (0x40) if set * User must clear the SF bit (0x40) if set
*/ */
info.si_code = FPE_FLTINV; info.si_code = FPE_FLTINV;
break; break;
case 0x002: /* Denormalize */ case 0x002: /* Denormalize */
case 0x010: /* Underflow */ case 0x010: /* Underflow */
info.si_code = FPE_FLTUND; info.si_code = FPE_FLTUND;
break; break;
case 0x004: /* Zero Divide */ case 0x004: /* Zero Divide */
info.si_code = FPE_FLTDIV; info.si_code = FPE_FLTDIV;
break; break;
case 0x008: /* Overflow */ case 0x008: /* Overflow */
info.si_code = FPE_FLTOVF; info.si_code = FPE_FLTOVF;
break; break;
case 0x020: /* Precision */ case 0x020: /* Precision */
info.si_code = FPE_FLTRES; info.si_code = FPE_FLTRES;
break; break;
} }
force_sig_info(SIGFPE, &info, task); force_sig_info(SIGFPE, &info, task);
} }
@ -1074,7 +1072,7 @@ asmlinkage void bad_intr(void)
asmlinkage void do_simd_coprocessor_error(struct pt_regs *regs) asmlinkage void do_simd_coprocessor_error(struct pt_regs *regs)
{ {
void __user *ip = (void __user *)(regs->ip); void __user *ip = (void __user *)(regs->ip);
struct task_struct * task; struct task_struct *task;
siginfo_t info; siginfo_t info;
unsigned short mxcsr; unsigned short mxcsr;
@ -1102,25 +1100,25 @@ asmlinkage void do_simd_coprocessor_error(struct pt_regs *regs)
*/ */
mxcsr = get_fpu_mxcsr(task); mxcsr = get_fpu_mxcsr(task);
switch (~((mxcsr & 0x1f80) >> 7) & (mxcsr & 0x3f)) { switch (~((mxcsr & 0x1f80) >> 7) & (mxcsr & 0x3f)) {
case 0x000: case 0x000:
default: default:
break; break;
case 0x001: /* Invalid Op */ case 0x001: /* Invalid Op */
info.si_code = FPE_FLTINV; info.si_code = FPE_FLTINV;
break; break;
case 0x002: /* Denormalize */ case 0x002: /* Denormalize */
case 0x010: /* Underflow */ case 0x010: /* Underflow */
info.si_code = FPE_FLTUND; info.si_code = FPE_FLTUND;
break; break;
case 0x004: /* Zero Divide */ case 0x004: /* Zero Divide */
info.si_code = FPE_FLTDIV; info.si_code = FPE_FLTDIV;
break; break;
case 0x008: /* Overflow */ case 0x008: /* Overflow */
info.si_code = FPE_FLTOVF; info.si_code = FPE_FLTOVF;
break; break;
case 0x020: /* Precision */ case 0x020: /* Precision */
info.si_code = FPE_FLTRES; info.si_code = FPE_FLTRES;
break; break;
} }
force_sig_info(SIGFPE, &info, task); force_sig_info(SIGFPE, &info, task);
} }
@ -1138,7 +1136,7 @@ asmlinkage void __attribute__((weak)) mce_threshold_interrupt(void)
} }
/* /*
* 'math_state_restore()' saves the current math information in the * 'math_state_restore()' saves the current math information in the
* old math state array, and gets the new ones from the current task * old math state array, and gets the new ones from the current task
* *
* Careful.. There are problems with IBM-designed IRQ13 behaviour. * Careful.. There are problems with IBM-designed IRQ13 behaviour.
@ -1163,7 +1161,7 @@ asmlinkage void math_state_restore(void)
local_irq_disable(); local_irq_disable();
} }
clts(); /* Allow maths ops (or we recurse) */ clts(); /* Allow maths ops (or we recurse) */
restore_fpu_checking(&me->thread.xstate->fxsave); restore_fpu_checking(&me->thread.xstate->fxsave);
task_thread_info(me)->status |= TS_USEDFPU; task_thread_info(me)->status |= TS_USEDFPU;
me->fpu_counter++; me->fpu_counter++;
@ -1172,64 +1170,61 @@ EXPORT_SYMBOL_GPL(math_state_restore);
void __init trap_init(void) void __init trap_init(void)
{ {
set_intr_gate(0,&divide_error); set_intr_gate(0, &divide_error);
set_intr_gate_ist(1,&debug,DEBUG_STACK); set_intr_gate_ist(1, &debug, DEBUG_STACK);
set_intr_gate_ist(2,&nmi,NMI_STACK); set_intr_gate_ist(2, &nmi, NMI_STACK);
set_system_gate_ist(3,&int3,DEBUG_STACK); /* int3 can be called from all */ set_system_gate_ist(3, &int3, DEBUG_STACK); /* int3 can be called from all */
set_system_gate(4,&overflow); /* int4 can be called from all */ set_system_gate(4, &overflow); /* int4 can be called from all */
set_intr_gate(5,&bounds); set_intr_gate(5, &bounds);
set_intr_gate(6,&invalid_op); set_intr_gate(6, &invalid_op);
set_intr_gate(7,&device_not_available); set_intr_gate(7, &device_not_available);
set_intr_gate_ist(8,&double_fault, DOUBLEFAULT_STACK); set_intr_gate_ist(8, &double_fault, DOUBLEFAULT_STACK);
set_intr_gate(9,&coprocessor_segment_overrun); set_intr_gate(9, &coprocessor_segment_overrun);
set_intr_gate(10,&invalid_TSS); set_intr_gate(10, &invalid_TSS);
set_intr_gate(11,&segment_not_present); set_intr_gate(11, &segment_not_present);
set_intr_gate_ist(12,&stack_segment,STACKFAULT_STACK); set_intr_gate_ist(12, &stack_segment, STACKFAULT_STACK);
set_intr_gate(13,&general_protection); set_intr_gate(13, &general_protection);
set_intr_gate(14,&page_fault); set_intr_gate(14, &page_fault);
set_intr_gate(15,&spurious_interrupt_bug); set_intr_gate(15, &spurious_interrupt_bug);
set_intr_gate(16,&coprocessor_error); set_intr_gate(16, &coprocessor_error);
set_intr_gate(17,&alignment_check); set_intr_gate(17, &alignment_check);
#ifdef CONFIG_X86_MCE #ifdef CONFIG_X86_MCE
set_intr_gate_ist(18,&machine_check, MCE_STACK); set_intr_gate_ist(18, &machine_check, MCE_STACK);
#endif #endif
set_intr_gate(19,&simd_coprocessor_error); set_intr_gate(19, &simd_coprocessor_error);
#ifdef CONFIG_IA32_EMULATION #ifdef CONFIG_IA32_EMULATION
set_system_gate(IA32_SYSCALL_VECTOR, ia32_syscall); set_system_gate(IA32_SYSCALL_VECTOR, ia32_syscall);
#endif #endif
/* /*
* initialize the per thread extended state: * initialize the per thread extended state:
*/ */
init_thread_xstate(); init_thread_xstate();
/* /*
* Should be a barrier for any external CPU state. * Should be a barrier for any external CPU state:
*/ */
cpu_init(); cpu_init();
} }
static int __init oops_setup(char *s) static int __init oops_setup(char *s)
{ {
if (!s) if (!s)
return -EINVAL; return -EINVAL;
if (!strcmp(s, "panic")) if (!strcmp(s, "panic"))
panic_on_oops = 1; panic_on_oops = 1;
return 0; return 0;
} }
early_param("oops", oops_setup); early_param("oops", oops_setup);
static int __init kstack_setup(char *s) static int __init kstack_setup(char *s)
{ {
if (!s) if (!s)
return -EINVAL; return -EINVAL;
kstack_depth_to_print = simple_strtoul(s,NULL,0); kstack_depth_to_print = simple_strtoul(s, NULL, 0);
return 0; return 0;
} }
early_param("kstack", kstack_setup); early_param("kstack", kstack_setup);
static int __init code_bytes_setup(char *s) static int __init code_bytes_setup(char *s)
{ {
code_bytes = simple_strtoul(s, NULL, 0); code_bytes = simple_strtoul(s, NULL, 0);