2005-04-16 15:20:36 -07:00
# include <linux/mm.h>
# include <linux/sched.h>
# include <linux/init_task.h>
# include <linux/fs.h>
# include <asm/uaccess.h>
# include <asm/pgtable.h>
# include <asm/processor.h>
# include <asm/desc.h>
2013-05-09 12:02:29 +02:00
# ifdef CONFIG_X86_32
2005-04-16 15:20:36 -07:00
# define DOUBLEFAULT_STACKSIZE (1024)
static unsigned long doublefault_stack [ DOUBLEFAULT_STACKSIZE ] ;
# define STACK_START (unsigned long)(doublefault_stack+DOUBLEFAULT_STACKSIZE)
2007-08-10 22:31:11 +02:00
# define ptr_ok(x) ((x) > PAGE_OFFSET && (x) < PAGE_OFFSET + MAXMEM)
2005-04-16 15:20:36 -07:00
static void doublefault_fn ( void )
{
2008-01-30 13:31:12 +01:00
struct desc_ptr gdt_desc = { 0 , 0 } ;
2005-04-16 15:20:36 -07:00
unsigned long gdt , tss ;
2013-04-05 16:42:23 -04:00
native_store_gdt ( & gdt_desc ) ;
2005-04-16 15:20:36 -07:00
gdt = gdt_desc . address ;
2007-08-10 22:31:11 +02:00
printk ( KERN_EMERG " PANIC: double fault, gdt at %08lx [%d bytes] \n " , gdt , gdt_desc . size ) ;
2005-04-16 15:20:36 -07:00
if ( ptr_ok ( gdt ) ) {
gdt + = GDT_ENTRY_TSS < < 3 ;
2009-07-19 00:08:54 +09:00
tss = get_desc_base ( ( struct desc_struct * ) gdt ) ;
2007-08-10 22:31:11 +02:00
printk ( KERN_EMERG " double fault, tss at %08lx \n " , tss ) ;
2005-04-16 15:20:36 -07:00
if ( ptr_ok ( tss ) ) {
2008-01-30 13:31:31 +01:00
struct x86_hw_tss * t = ( struct x86_hw_tss * ) tss ;
2005-04-16 15:20:36 -07:00
2008-01-30 13:31:02 +01:00
printk ( KERN_EMERG " eip = %08lx, esp = %08lx \n " ,
t - > ip , t - > sp ) ;
2005-04-16 15:20:36 -07:00
2007-08-10 22:31:11 +02:00
printk ( KERN_EMERG " eax = %08lx, ebx = %08lx, ecx = %08lx, edx = %08lx \n " ,
2008-01-30 13:31:02 +01:00
t - > ax , t - > bx , t - > cx , t - > dx ) ;
2007-08-10 22:31:11 +02:00
printk ( KERN_EMERG " esi = %08lx, edi = %08lx \n " ,
2008-01-30 13:31:02 +01:00
t - > si , t - > di ) ;
2005-04-16 15:20:36 -07:00
}
}
2006-06-25 05:46:53 -07:00
for ( ; ; )
cpu_relax ( ) ;
2005-04-16 15:20:36 -07:00
}
struct tss_struct doublefault_tss __cacheline_aligned = {
2007-05-02 19:27:13 +02:00
. x86_tss = {
2008-01-30 13:31:02 +01:00
. sp0 = STACK_START ,
2007-05-02 19:27:13 +02:00
. ss0 = __KERNEL_DS ,
. ldt = 0 ,
. io_bitmap_base = INVALID_IO_BITMAP_OFFSET ,
2005-04-16 15:20:36 -07:00
2008-01-30 13:31:02 +01:00
. ip = ( unsigned long ) doublefault_fn ,
2007-05-02 19:27:13 +02:00
/* 0x2 bit is always set */
2008-01-30 13:31:02 +01:00
. flags = X86_EFLAGS_SF | 0x2 ,
. sp = STACK_START ,
2007-05-02 19:27:13 +02:00
. es = __USER_DS ,
. cs = __KERNEL_CS ,
. ss = __KERNEL_DS ,
. ds = __USER_DS ,
2007-08-10 22:31:11 +02:00
. fs = __KERNEL_PERCPU ,
2005-04-16 15:20:36 -07:00
2008-10-03 17:54:25 +02:00
. __cr3 = __pa_nodebug ( swapper_pg_dir ) ,
2007-05-02 19:27:13 +02:00
}
2005-04-16 15:20:36 -07:00
} ;
2013-05-09 12:02:29 +02:00
/* dummy for do_double_fault() call */
void df_debug ( struct pt_regs * regs , long error_code ) { }
# else /* !CONFIG_X86_32 */
void df_debug ( struct pt_regs * regs , long error_code )
{
pr_emerg ( " PANIC: double fault, error_code: 0x%lx \n " , error_code ) ;
show_regs ( regs ) ;
panic ( " Machine halted. " ) ;
}
# endif