aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86
diff options
context:
space:
mode:
authorAlexander van Heukelum <heukelum@fastmail.fm>2008-09-30 07:12:14 -0400
committerIngo Molnar <mingo@elte.hu>2008-10-13 04:33:16 -0400
commit2bc5f927d489f9e47b6fa71f323b653e8ec81782 (patch)
tree8f2608d3af2675227d01c0678bfe0366994a050f /arch/x86
parentaf5c2bd16ac2e5688c3bf46ea1f95112d696d294 (diff)
i386: split out dumpstack code from traps_32.c
The dumpstack code is logically quite independent from the hardware traps. Split it out into its own file. Signed-off-by: Alexander van Heukelum <heukelum@fastmail.fm> Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'arch/x86')
-rw-r--r--arch/x86/kernel/Makefile2
-rw-r--r--arch/x86/kernel/dumpstack_32.c422
-rw-r--r--arch/x86/kernel/traps_32.c404
3 files changed, 423 insertions, 405 deletions
diff --git a/arch/x86/kernel/Makefile b/arch/x86/kernel/Makefile
index 5098585f87ce..d5bde5d8c2b6 100644
--- a/arch/x86/kernel/Makefile
+++ b/arch/x86/kernel/Makefile
@@ -28,7 +28,7 @@ obj-y += time_$(BITS).o ioport.o ldt.o
28obj-y += setup.o i8259.o irqinit_$(BITS).o setup_percpu.o 28obj-y += setup.o i8259.o irqinit_$(BITS).o setup_percpu.o
29obj-$(CONFIG_X86_VISWS) += visws_quirks.o 29obj-$(CONFIG_X86_VISWS) += visws_quirks.o
30obj-$(CONFIG_X86_32) += probe_roms_32.o 30obj-$(CONFIG_X86_32) += probe_roms_32.o
31obj-$(CONFIG_X86_32) += sys_i386_32.o i386_ksyms_32.o 31obj-$(CONFIG_X86_32) += sys_i386_32.o i386_ksyms_32.o dumpstack_32.o
32obj-$(CONFIG_X86_64) += sys_x86_64.o x8664_ksyms_64.o 32obj-$(CONFIG_X86_64) += sys_x86_64.o x8664_ksyms_64.o
33obj-$(CONFIG_X86_64) += syscall_64.o vsyscall_64.o 33obj-$(CONFIG_X86_64) += syscall_64.o vsyscall_64.o
34obj-y += bootflag.o e820.o 34obj-y += bootflag.o e820.o
diff --git a/arch/x86/kernel/dumpstack_32.c b/arch/x86/kernel/dumpstack_32.c
new file mode 100644
index 000000000000..c398b27df6cd
--- /dev/null
+++ b/arch/x86/kernel/dumpstack_32.c
@@ -0,0 +1,422 @@
1/*
2 * Copyright (C) 1991, 1992 Linus Torvalds
3 * Copyright (C) 2000, 2001, 2002 Andi Kleen, SuSE Labs
4 */
5#include <linux/kallsyms.h>
6#include <linux/kprobes.h>
7#include <linux/uaccess.h>
8#include <linux/utsname.h>
9#include <linux/hardirq.h>
10#include <linux/kdebug.h>
11#include <linux/module.h>
12#include <linux/ptrace.h>
13#include <linux/kexec.h>
14#include <linux/bug.h>
15#include <linux/nmi.h>
16
17#include <asm/stacktrace.h>
18
19int panic_on_unrecovered_nmi;
20int kstack_depth_to_print = 24;
21static unsigned int code_bytes = 64;
22static int die_counter;
23
24void printk_address(unsigned long address, int reliable)
25{
26#ifdef CONFIG_KALLSYMS
27 unsigned long offset = 0;
28 unsigned long symsize;
29 const char *symname;
30 char *modname;
31 char *delim = ":";
32 char namebuf[KSYM_NAME_LEN];
33 char reliab[4] = "";
34
35 symname = kallsyms_lookup(address, &symsize, &offset,
36 &modname, namebuf);
37 if (!symname) {
38 printk(" [<%08lx>]\n", address);
39 return;
40 }
41 if (!reliable)
42 strcpy(reliab, "? ");
43
44 if (!modname)
45 modname = delim = "";
46 printk(" [<%08lx>] %s%s%s%s%s+0x%lx/0x%lx\n",
47 address, reliab, delim, modname, delim, symname, offset, symsize);
48#else
49 printk(" [<%08lx>]\n", address);
50#endif
51}
52
53static inline int valid_stack_ptr(struct thread_info *tinfo,
54 void *p, unsigned int size)
55{
56 void *t = tinfo;
57 return p > t && p <= t + THREAD_SIZE - size;
58}
59
60/* The form of the top of the frame on the stack */
61struct stack_frame {
62 struct stack_frame *next_frame;
63 unsigned long return_address;
64};
65
66static inline unsigned long
67print_context_stack(struct thread_info *tinfo,
68 unsigned long *stack, unsigned long bp,
69 const struct stacktrace_ops *ops, void *data)
70{
71 struct stack_frame *frame = (struct stack_frame *)bp;
72
73 while (valid_stack_ptr(tinfo, stack, sizeof(*stack))) {
74 unsigned long addr;
75
76 addr = *stack;
77 if (__kernel_text_address(addr)) {
78 if ((unsigned long) stack == bp + 4) {
79 ops->address(data, addr, 1);
80 frame = frame->next_frame;
81 bp = (unsigned long) frame;
82 } else {
83 ops->address(data, addr, bp == 0);
84 }
85 }
86 stack++;
87 }
88 return bp;
89}
90
91void dump_trace(struct task_struct *task, struct pt_regs *regs,
92 unsigned long *stack, unsigned long bp,
93 const struct stacktrace_ops *ops, void *data)
94{
95 if (!task)
96 task = current;
97
98 if (!stack) {
99 unsigned long dummy;
100 stack = &dummy;
101 if (task != current)
102 stack = (unsigned long *)task->thread.sp;
103 }
104
105#ifdef CONFIG_FRAME_POINTER
106 if (!bp) {
107 if (task == current) {
108 /* Grab bp right from our regs */
109 asm("movl %%ebp, %0" : "=r" (bp) :);
110 } else {
111 /* bp is the last reg pushed by switch_to */
112 bp = *(unsigned long *) task->thread.sp;
113 }
114 }
115#endif
116
117 for (;;) {
118 struct thread_info *context;
119
120 context = (struct thread_info *)
121 ((unsigned long)stack & (~(THREAD_SIZE - 1)));
122 bp = print_context_stack(context, stack, bp, ops, data);
123 /*
124 * Should be after the line below, but somewhere
125 * in early boot context comes out corrupted and we
126 * can't reference it:
127 */
128 if (ops->stack(data, "IRQ") < 0)
129 break;
130 stack = (unsigned long *)context->previous_esp;
131 if (!stack)
132 break;
133 touch_nmi_watchdog();
134 }
135}
136EXPORT_SYMBOL(dump_trace);
137
138static void
139print_trace_warning_symbol(void *data, char *msg, unsigned long symbol)
140{
141 printk(data);
142 print_symbol(msg, symbol);
143 printk("\n");
144}
145
146static void print_trace_warning(void *data, char *msg)
147{
148 printk("%s%s\n", (char *)data, msg);
149}
150
151static int print_trace_stack(void *data, char *name)
152{
153 return 0;
154}
155
156/*
157 * Print one address/symbol entries per line.
158 */
159static void print_trace_address(void *data, unsigned long addr, int reliable)
160{
161 printk("%s [<%08lx>] ", (char *)data, addr);
162 if (!reliable)
163 printk("? ");
164 print_symbol("%s\n", addr);
165 touch_nmi_watchdog();
166}
167
168static const struct stacktrace_ops print_trace_ops = {
169 .warning = print_trace_warning,
170 .warning_symbol = print_trace_warning_symbol,
171 .stack = print_trace_stack,
172 .address = print_trace_address,
173};
174
175static void
176show_trace_log_lvl(struct task_struct *task, struct pt_regs *regs,
177 unsigned long *stack, unsigned long bp, char *log_lvl)
178{
179 dump_trace(task, regs, stack, bp, &print_trace_ops, log_lvl);
180 printk("%s =======================\n", log_lvl);
181}
182
183void show_trace(struct task_struct *task, struct pt_regs *regs,
184 unsigned long *stack, unsigned long bp)
185{
186 show_trace_log_lvl(task, regs, stack, bp, "");
187}
188
189static void
190show_stack_log_lvl(struct task_struct *task, struct pt_regs *regs,
191 unsigned long *sp, unsigned long bp, char *log_lvl)
192{
193 unsigned long *stack;
194 int i;
195
196 if (sp == NULL) {
197 if (task)
198 sp = (unsigned long *)task->thread.sp;
199 else
200 sp = (unsigned long *)&sp;
201 }
202
203 stack = sp;
204 for (i = 0; i < kstack_depth_to_print; i++) {
205 if (kstack_end(stack))
206 break;
207 if (i && ((i % 8) == 0))
208 printk("\n%s ", log_lvl);
209 printk("%08lx ", *stack++);
210 }
211 printk("\n%sCall Trace:\n", log_lvl);
212
213 show_trace_log_lvl(task, regs, sp, bp, log_lvl);
214}
215
216void show_stack(struct task_struct *task, unsigned long *sp)
217{
218 printk(" ");
219 show_stack_log_lvl(task, NULL, sp, 0, "");
220}
221
222/*
223 * The architecture-independent dump_stack generator
224 */
225void dump_stack(void)
226{
227 unsigned long bp = 0;
228 unsigned long stack;
229
230#ifdef CONFIG_FRAME_POINTER
231 if (!bp)
232 asm("movl %%ebp, %0" : "=r" (bp):);
233#endif
234
235 printk("Pid: %d, comm: %.20s %s %s %.*s\n",
236 current->pid, current->comm, print_tainted(),
237 init_utsname()->release,
238 (int)strcspn(init_utsname()->version, " "),
239 init_utsname()->version);
240
241 show_trace(current, NULL, &stack, bp);
242}
243
244EXPORT_SYMBOL(dump_stack);
245
246void show_registers(struct pt_regs *regs)
247{
248 int i;
249
250 print_modules();
251 __show_regs(regs, 0);
252
253 printk(KERN_EMERG "Process %.*s (pid: %d, ti=%p task=%p task.ti=%p)",
254 TASK_COMM_LEN, current->comm, task_pid_nr(current),
255 current_thread_info(), current, task_thread_info(current));
256 /*
257 * When in-kernel, we also print out the stack and code at the
258 * time of the fault..
259 */
260 if (!user_mode_vm(regs)) {
261 unsigned int code_prologue = code_bytes * 43 / 64;
262 unsigned int code_len = code_bytes;
263 unsigned char c;
264 u8 *ip;
265
266 printk("\n" KERN_EMERG "Stack: ");
267 show_stack_log_lvl(NULL, regs, &regs->sp, 0, KERN_EMERG);
268
269 printk(KERN_EMERG "Code: ");
270
271 ip = (u8 *)regs->ip - code_prologue;
272 if (ip < (u8 *)PAGE_OFFSET || probe_kernel_address(ip, c)) {
273 /* try starting at EIP */
274 ip = (u8 *)regs->ip;
275 code_len = code_len - code_prologue + 1;
276 }
277 for (i = 0; i < code_len; i++, ip++) {
278 if (ip < (u8 *)PAGE_OFFSET ||
279 probe_kernel_address(ip, c)) {
280 printk(" Bad EIP value.");
281 break;
282 }
283 if (ip == (u8 *)regs->ip)
284 printk("<%02x> ", c);
285 else
286 printk("%02x ", c);
287 }
288 }
289 printk("\n");
290}
291
292int is_valid_bugaddr(unsigned long ip)
293{
294 unsigned short ud2;
295
296 if (ip < PAGE_OFFSET)
297 return 0;
298 if (probe_kernel_address((unsigned short *)ip, ud2))
299 return 0;
300
301 return ud2 == 0x0b0f;
302}
303
304static raw_spinlock_t die_lock = __RAW_SPIN_LOCK_UNLOCKED;
305static int die_owner = -1;
306static unsigned int die_nest_count;
307
308unsigned __kprobes long oops_begin(void)
309{
310 unsigned long flags;
311
312 oops_enter();
313
314 if (die_owner != raw_smp_processor_id()) {
315 console_verbose();
316 raw_local_irq_save(flags);
317 __raw_spin_lock(&die_lock);
318 die_owner = smp_processor_id();
319 die_nest_count = 0;
320 bust_spinlocks(1);
321 } else {
322 raw_local_irq_save(flags);
323 }
324 die_nest_count++;
325 return flags;
326}
327
328void __kprobes oops_end(unsigned long flags, struct pt_regs *regs, int signr)
329{
330 bust_spinlocks(0);
331 die_owner = -1;
332 add_taint(TAINT_DIE);
333 __raw_spin_unlock(&die_lock);
334 raw_local_irq_restore(flags);
335
336 if (!regs)
337 return;
338
339 if (kexec_should_crash(current))
340 crash_kexec(regs);
341
342 if (in_interrupt())
343 panic("Fatal exception in interrupt");
344
345 if (panic_on_oops)
346 panic("Fatal exception");
347
348 oops_exit();
349 do_exit(signr);
350}
351
352int __kprobes __die(const char *str, struct pt_regs *regs, long err)
353{
354 unsigned short ss;
355 unsigned long sp;
356
357 printk(KERN_EMERG "%s: %04lx [#%d] ", str, err & 0xffff, ++die_counter);
358#ifdef CONFIG_PREEMPT
359 printk("PREEMPT ");
360#endif
361#ifdef CONFIG_SMP
362 printk("SMP ");
363#endif
364#ifdef CONFIG_DEBUG_PAGEALLOC
365 printk("DEBUG_PAGEALLOC");
366#endif
367 printk("\n");
368 if (notify_die(DIE_OOPS, str, regs, err,
369 current->thread.trap_no, SIGSEGV) == NOTIFY_STOP)
370 return 1;
371
372 show_registers(regs);
373 /* Executive summary in case the oops scrolled away */
374 sp = (unsigned long) (&regs->sp);
375 savesegment(ss, ss);
376 if (user_mode(regs)) {
377 sp = regs->sp;
378 ss = regs->ss & 0xffff;
379 }
380 printk(KERN_EMERG "EIP: [<%08lx>] ", regs->ip);
381 print_symbol("%s", regs->ip);
382 printk(" SS:ESP %04x:%08lx\n", ss, sp);
383 return 0;
384}
385
386/*
387 * This is gone through when something in the kernel has done something bad
388 * and is about to be terminated:
389 */
390void die(const char *str, struct pt_regs *regs, long err)
391{
392 unsigned long flags = oops_begin();
393
394 if (die_nest_count < 3) {
395 report_bug(regs->ip, regs);
396
397 if (__die(str, regs, err))
398 regs = NULL;
399 } else {
400 printk(KERN_EMERG "Recursive die() failure, output suppressed\n");
401 }
402
403 oops_end(flags, regs, SIGSEGV);
404}
405
406static int __init kstack_setup(char *s)
407{
408 kstack_depth_to_print = simple_strtoul(s, NULL, 0);
409
410 return 1;
411}
412__setup("kstack=", kstack_setup);
413
414static int __init code_bytes_setup(char *s)
415{
416 code_bytes = simple_strtoul(s, NULL, 0);
417 if (code_bytes > 8192)
418 code_bytes = 8192;
419
420 return 1;
421}
422__setup("code_bytes=", code_bytes_setup);
diff --git a/arch/x86/kernel/traps_32.c b/arch/x86/kernel/traps_32.c
index 4f0831593e32..ce1063c141fc 100644
--- a/arch/x86/kernel/traps_32.c
+++ b/arch/x86/kernel/traps_32.c
@@ -80,11 +80,7 @@ char ignore_fpu_irq;
80gate_desc idt_table[256] 80gate_desc idt_table[256]
81 __attribute__((__section__(".data.idt"))) = { { { { 0, 0 } } }, }; 81 __attribute__((__section__(".data.idt"))) = { { { { 0, 0 } } }, };
82 82
83int panic_on_unrecovered_nmi;
84int kstack_depth_to_print = 24;
85static unsigned int code_bytes = 64;
86static int ignore_nmis; 83static int ignore_nmis;
87static int die_counter;
88 84
89static inline void conditional_sti(struct pt_regs *regs) 85static inline void conditional_sti(struct pt_regs *regs)
90{ 86{
@@ -92,388 +88,6 @@ static inline void conditional_sti(struct pt_regs *regs)
92 local_irq_enable(); 88 local_irq_enable();
93} 89}
94 90
95void printk_address(unsigned long address, int reliable)
96{
97#ifdef CONFIG_KALLSYMS
98 unsigned long offset = 0;
99 unsigned long symsize;
100 const char *symname;
101 char *modname;
102 char *delim = ":";
103 char namebuf[KSYM_NAME_LEN];
104 char reliab[4] = "";
105
106 symname = kallsyms_lookup(address, &symsize, &offset,
107 &modname, namebuf);
108 if (!symname) {
109 printk(" [<%08lx>]\n", address);
110 return;
111 }
112 if (!reliable)
113 strcpy(reliab, "? ");
114
115 if (!modname)
116 modname = delim = "";
117 printk(" [<%08lx>] %s%s%s%s%s+0x%lx/0x%lx\n",
118 address, reliab, delim, modname, delim, symname, offset, symsize);
119#else
120 printk(" [<%08lx>]\n", address);
121#endif
122}
123
124static inline int valid_stack_ptr(struct thread_info *tinfo,
125 void *p, unsigned int size)
126{
127 void *t = tinfo;
128 return p > t && p <= t + THREAD_SIZE - size;
129}
130
131/* The form of the top of the frame on the stack */
132struct stack_frame {
133 struct stack_frame *next_frame;
134 unsigned long return_address;
135};
136
137static inline unsigned long
138print_context_stack(struct thread_info *tinfo,
139 unsigned long *stack, unsigned long bp,
140 const struct stacktrace_ops *ops, void *data)
141{
142 struct stack_frame *frame = (struct stack_frame *)bp;
143
144 while (valid_stack_ptr(tinfo, stack, sizeof(*stack))) {
145 unsigned long addr;
146
147 addr = *stack;
148 if (__kernel_text_address(addr)) {
149 if ((unsigned long) stack == bp + 4) {
150 ops->address(data, addr, 1);
151 frame = frame->next_frame;
152 bp = (unsigned long) frame;
153 } else {
154 ops->address(data, addr, bp == 0);
155 }
156 }
157 stack++;
158 }
159 return bp;
160}
161
162void dump_trace(struct task_struct *task, struct pt_regs *regs,
163 unsigned long *stack, unsigned long bp,
164 const struct stacktrace_ops *ops, void *data)
165{
166 if (!task)
167 task = current;
168
169 if (!stack) {
170 unsigned long dummy;
171 stack = &dummy;
172 if (task != current)
173 stack = (unsigned long *)task->thread.sp;
174 }
175
176#ifdef CONFIG_FRAME_POINTER
177 if (!bp) {
178 if (task == current) {
179 /* Grab bp right from our regs */
180 asm("movl %%ebp, %0" : "=r" (bp) :);
181 } else {
182 /* bp is the last reg pushed by switch_to */
183 bp = *(unsigned long *) task->thread.sp;
184 }
185 }
186#endif
187
188 for (;;) {
189 struct thread_info *context;
190
191 context = (struct thread_info *)
192 ((unsigned long)stack & (~(THREAD_SIZE - 1)));
193 bp = print_context_stack(context, stack, bp, ops, data);
194 /*
195 * Should be after the line below, but somewhere
196 * in early boot context comes out corrupted and we
197 * can't reference it:
198 */
199 if (ops->stack(data, "IRQ") < 0)
200 break;
201 stack = (unsigned long *)context->previous_esp;
202 if (!stack)
203 break;
204 touch_nmi_watchdog();
205 }
206}
207EXPORT_SYMBOL(dump_trace);
208
209static void
210print_trace_warning_symbol(void *data, char *msg, unsigned long symbol)
211{
212 printk(data);
213 print_symbol(msg, symbol);
214 printk("\n");
215}
216
217static void print_trace_warning(void *data, char *msg)
218{
219 printk("%s%s\n", (char *)data, msg);
220}
221
222static int print_trace_stack(void *data, char *name)
223{
224 return 0;
225}
226
227/*
228 * Print one address/symbol entries per line.
229 */
230static void print_trace_address(void *data, unsigned long addr, int reliable)
231{
232 printk("%s [<%08lx>] ", (char *)data, addr);
233 if (!reliable)
234 printk("? ");
235 print_symbol("%s\n", addr);
236 touch_nmi_watchdog();
237}
238
239static const struct stacktrace_ops print_trace_ops = {
240 .warning = print_trace_warning,
241 .warning_symbol = print_trace_warning_symbol,
242 .stack = print_trace_stack,
243 .address = print_trace_address,
244};
245
246static void
247show_trace_log_lvl(struct task_struct *task, struct pt_regs *regs,
248 unsigned long *stack, unsigned long bp, char *log_lvl)
249{
250 dump_trace(task, regs, stack, bp, &print_trace_ops, log_lvl);
251 printk("%s =======================\n", log_lvl);
252}
253
254void show_trace(struct task_struct *task, struct pt_regs *regs,
255 unsigned long *stack, unsigned long bp)
256{
257 show_trace_log_lvl(task, regs, stack, bp, "");
258}
259
260static void
261show_stack_log_lvl(struct task_struct *task, struct pt_regs *regs,
262 unsigned long *sp, unsigned long bp, char *log_lvl)
263{
264 unsigned long *stack;
265 int i;
266
267 if (sp == NULL) {
268 if (task)
269 sp = (unsigned long *)task->thread.sp;
270 else
271 sp = (unsigned long *)&sp;
272 }
273
274 stack = sp;
275 for (i = 0; i < kstack_depth_to_print; i++) {
276 if (kstack_end(stack))
277 break;
278 if (i && ((i % 8) == 0))
279 printk("\n%s ", log_lvl);
280 printk("%08lx ", *stack++);
281 }
282 printk("\n%sCall Trace:\n", log_lvl);
283
284 show_trace_log_lvl(task, regs, sp, bp, log_lvl);
285}
286
287void show_stack(struct task_struct *task, unsigned long *sp)
288{
289 printk(" ");
290 show_stack_log_lvl(task, NULL, sp, 0, "");
291}
292
293/*
294 * The architecture-independent dump_stack generator
295 */
296void dump_stack(void)
297{
298 unsigned long bp = 0;
299 unsigned long stack;
300
301#ifdef CONFIG_FRAME_POINTER
302 if (!bp)
303 asm("movl %%ebp, %0" : "=r" (bp):);
304#endif
305
306 printk("Pid: %d, comm: %.20s %s %s %.*s\n",
307 current->pid, current->comm, print_tainted(),
308 init_utsname()->release,
309 (int)strcspn(init_utsname()->version, " "),
310 init_utsname()->version);
311
312 show_trace(current, NULL, &stack, bp);
313}
314
315EXPORT_SYMBOL(dump_stack);
316
317void show_registers(struct pt_regs *regs)
318{
319 int i;
320
321 print_modules();
322 __show_regs(regs, 0);
323
324 printk(KERN_EMERG "Process %.*s (pid: %d, ti=%p task=%p task.ti=%p)",
325 TASK_COMM_LEN, current->comm, task_pid_nr(current),
326 current_thread_info(), current, task_thread_info(current));
327 /*
328 * When in-kernel, we also print out the stack and code at the
329 * time of the fault..
330 */
331 if (!user_mode_vm(regs)) {
332 unsigned int code_prologue = code_bytes * 43 / 64;
333 unsigned int code_len = code_bytes;
334 unsigned char c;
335 u8 *ip;
336
337 printk("\n" KERN_EMERG "Stack: ");
338 show_stack_log_lvl(NULL, regs, &regs->sp, 0, KERN_EMERG);
339
340 printk(KERN_EMERG "Code: ");
341
342 ip = (u8 *)regs->ip - code_prologue;
343 if (ip < (u8 *)PAGE_OFFSET || probe_kernel_address(ip, c)) {
344 /* try starting at EIP */
345 ip = (u8 *)regs->ip;
346 code_len = code_len - code_prologue + 1;
347 }
348 for (i = 0; i < code_len; i++, ip++) {
349 if (ip < (u8 *)PAGE_OFFSET ||
350 probe_kernel_address(ip, c)) {
351 printk(" Bad EIP value.");
352 break;
353 }
354 if (ip == (u8 *)regs->ip)
355 printk("<%02x> ", c);
356 else
357 printk("%02x ", c);
358 }
359 }
360 printk("\n");
361}
362
363int is_valid_bugaddr(unsigned long ip)
364{
365 unsigned short ud2;
366
367 if (ip < PAGE_OFFSET)
368 return 0;
369 if (probe_kernel_address((unsigned short *)ip, ud2))
370 return 0;
371
372 return ud2 == 0x0b0f;
373}
374
375static raw_spinlock_t die_lock = __RAW_SPIN_LOCK_UNLOCKED;
376static int die_owner = -1;
377static unsigned int die_nest_count;
378
379unsigned __kprobes long oops_begin(void)
380{
381 unsigned long flags;
382
383 oops_enter();
384
385 if (die_owner != raw_smp_processor_id()) {
386 console_verbose();
387 raw_local_irq_save(flags);
388 __raw_spin_lock(&die_lock);
389 die_owner = smp_processor_id();
390 die_nest_count = 0;
391 bust_spinlocks(1);
392 } else {
393 raw_local_irq_save(flags);
394 }
395 die_nest_count++;
396 return flags;
397}
398
399void __kprobes oops_end(unsigned long flags, struct pt_regs *regs, int signr)
400{
401 bust_spinlocks(0);
402 die_owner = -1;
403 add_taint(TAINT_DIE);
404 __raw_spin_unlock(&die_lock);
405 raw_local_irq_restore(flags);
406
407 if (!regs)
408 return;
409
410 if (kexec_should_crash(current))
411 crash_kexec(regs);
412
413 if (in_interrupt())
414 panic("Fatal exception in interrupt");
415
416 if (panic_on_oops)
417 panic("Fatal exception");
418
419 oops_exit();
420 do_exit(signr);
421}
422
423int __kprobes __die(const char *str, struct pt_regs *regs, long err)
424{
425 unsigned short ss;
426 unsigned long sp;
427
428 printk(KERN_EMERG "%s: %04lx [#%d] ", str, err & 0xffff, ++die_counter);
429#ifdef CONFIG_PREEMPT
430 printk("PREEMPT ");
431#endif
432#ifdef CONFIG_SMP
433 printk("SMP ");
434#endif
435#ifdef CONFIG_DEBUG_PAGEALLOC
436 printk("DEBUG_PAGEALLOC");
437#endif
438 printk("\n");
439 if (notify_die(DIE_OOPS, str, regs, err,
440 current->thread.trap_no, SIGSEGV) == NOTIFY_STOP)
441 return 1;
442
443 show_registers(regs);
444 /* Executive summary in case the oops scrolled away */
445 sp = (unsigned long) (&regs->sp);
446 savesegment(ss, ss);
447 if (user_mode(regs)) {
448 sp = regs->sp;
449 ss = regs->ss & 0xffff;
450 }
451 printk(KERN_EMERG "EIP: [<%08lx>] ", regs->ip);
452 print_symbol("%s", regs->ip);
453 printk(" SS:ESP %04x:%08lx\n", ss, sp);
454 return 0;
455}
456
457/*
458 * This is gone through when something in the kernel has done something bad
459 * and is about to be terminated:
460 */
461void die(const char *str, struct pt_regs *regs, long err)
462{
463 unsigned long flags = oops_begin();
464
465 if (die_nest_count < 3) {
466 report_bug(regs->ip, regs);
467
468 if (__die(str, regs, err))
469 regs = NULL;
470 } else {
471 printk(KERN_EMERG "Recursive die() failure, output suppressed\n");
472 }
473
474 oops_end(flags, regs, SIGSEGV);
475}
476
477static inline void 91static inline void
478die_if_kernel(const char *str, struct pt_regs *regs, long err) 92die_if_kernel(const char *str, struct pt_regs *regs, long err)
479{ 93{
@@ -1256,21 +870,3 @@ void __init trap_init(void)
1256 870
1257 trap_init_hook(); 871 trap_init_hook();
1258} 872}
1259
1260static int __init kstack_setup(char *s)
1261{
1262 kstack_depth_to_print = simple_strtoul(s, NULL, 0);
1263
1264 return 1;
1265}
1266__setup("kstack=", kstack_setup);
1267
1268static int __init code_bytes_setup(char *s)
1269{
1270 code_bytes = simple_strtoul(s, NULL, 0);
1271 if (code_bytes > 8192)
1272 code_bytes = 8192;
1273
1274 return 1;
1275}
1276__setup("code_bytes=", code_bytes_setup);