#include <linux/kprobes.h>
#include <linux/uaccess.h>
#include <linux/kdebug.h>
+#include <linux/magic.h>
#include <asm/system.h>
#include <asm/desc.h>
unsigned long error_code, unsigned long address)
{
struct task_struct *tsk = current;
+ unsigned long *stackend;
+
#ifdef CONFIG_X86_64
unsigned long flags;
int sig;
show_fault_oops(regs, error_code, address);
+ stackend = end_of_stack(tsk);
+ if (*stackend != STACK_END_MAGIC)
+ printk(KERN_ALERT "Thread overran stack, or stack corrupted\n");
+
tsk->thread.cr2 = address;
tsk->thread.trap_no = 14;
tsk->thread.error_code = error_code;
return 0;
}
+ static int fault_in_kernel_space(unsigned long address)
+ {
+ #ifdef CONFIG_X86_32
+ return address >= TASK_SIZE;
+ #else /* !CONFIG_X86_32 */
+ return address >= TASK_SIZE64;
+ #endif /* CONFIG_X86_32 */
+ }
+
/*
* This routine handles page faults. It determines the address,
* and the problem, and then passes it off to one of the appropriate
* (error_code & 4) == 0, and that the fault was not a
* protection error (error_code & 9) == 0.
*/
- #ifdef CONFIG_X86_32
- if (unlikely(address >= TASK_SIZE)) {
- #else
- if (unlikely(address >= TASK_SIZE64)) {
- #endif
+ if (unlikely(fault_in_kernel_space(address))) {
if (!(error_code & (PF_RSVD|PF_USER|PF_PROT)) &&
vmalloc_fault(address) >= 0)
return;
return;
}
down_read(&mm->mmap_sem);
+ } else {
+ /*
+ * The above down_read_trylock() might have succeeded in which
+ * case we'll have missed the might_sleep() from down_read().
+ */
+ might_sleep();
}
vma = find_vma(mm, address);