diff options
-rw-r--r-- | include/asm-x86/system.h | 46 |
1 files changed, 31 insertions, 15 deletions
diff --git a/include/asm-x86/system.h b/include/asm-x86/system.h index 428d9471497f..299ae9605cb1 100644 --- a/include/asm-x86/system.h +++ b/include/asm-x86/system.h | |||
@@ -27,22 +27,38 @@ struct task_struct *__switch_to(struct task_struct *prev, | |||
27 | * Saving eflags is important. It switches not only IOPL between tasks, | 27 | * Saving eflags is important. It switches not only IOPL between tasks, |
28 | * it also protects other tasks from NT leaking through sysenter etc. | 28 | * it also protects other tasks from NT leaking through sysenter etc. |
29 | */ | 29 | */ |
30 | #define switch_to(prev, next, last) do { \ | 30 | #define switch_to(prev, next, last) \ |
31 | do { \ | ||
31 | unsigned long esi, edi; \ | 32 | unsigned long esi, edi; \ |
32 | asm volatile("pushfl\n\t" /* Save flags */ \ | 33 | \ |
33 | "pushl %%ebp\n\t" \ | 34 | asm volatile( \ |
34 | "movl %%esp,%0\n\t" /* save ESP */ \ | 35 | "pushfl \n\t" /* save flags */ \ |
35 | "movl %5,%%esp\n\t" /* restore ESP */ \ | 36 | "pushl %%ebp \n\t" /* save EBP */ \ |
36 | "movl $1f,%1\n\t" /* save EIP */ \ | 37 | "movl %%esp,%[prev_sp] \n\t" /* save ESP */ \ |
37 | "pushl %6\n\t" /* restore EIP */ \ | 38 | "movl %[next_sp],%%esp \n\t" /* restore ESP */ \ |
38 | "jmp __switch_to\n" \ | 39 | "movl $1f,%[prev_ip] \n\t" /* save EIP */ \ |
39 | "1:\t" \ | 40 | "pushl %[next_ip] \n\t" /* restore EIP */ \ |
40 | "popl %%ebp\n\t" \ | 41 | "jmp __switch_to \n" /* regparm call */ \ |
41 | "popfl" \ | 42 | "1: \t" \ |
42 | :"=m" (prev->thread.sp), "=m" (prev->thread.ip), \ | 43 | "popl %%ebp \n\t" /* restore EBP */ \ |
43 | "=a" (last), "=S" (esi), "=D" (edi) \ | 44 | "popfl \n" /* restore flags */ \ |
44 | :"m" (next->thread.sp), "m" (next->thread.ip), \ | 45 | \ |
45 | "2" (prev), "d" (next)); \ | 46 | /* output parameters */ \ |
47 | : [prev_sp] "=m" (prev->thread.sp), \ | ||
48 | [prev_ip] "=m" (prev->thread.ip), \ | ||
49 | "=a" (last), \ | ||
50 | \ | ||
51 | /* clobbered output registers: */ \ | ||
52 | "=S" (esi), "=D" (edi) \ | ||
53 | \ | ||
54 | /* input parameters: */ \ | ||
55 | : [next_sp] "m" (next->thread.sp), \ | ||
56 | [next_ip] "m" (next->thread.ip), \ | ||
57 | \ | ||
58 | /* regparm parameters for __switch_to(): */ \ | ||
59 | [prev] "a" (prev), \ | ||
60 | [next] "d" (next) \ | ||
61 | ); \ | ||
46 | } while (0) | 62 | } while (0) |
47 | 63 | ||
48 | /* | 64 | /* |