| #ifndef _ASM_X86_SWITCH_TO_H |
| #define _ASM_X86_SWITCH_TO_H |
| |
| struct task_struct; /* one of the stranger aspects of C forward declarations */ |
| struct task_struct *__switch_to(struct task_struct *prev, |
| struct task_struct *next); |
| struct tss_struct; |
| void __switch_to_xtra(struct task_struct *prev_p, struct task_struct *next_p, |
| struct tss_struct *tss); |
| |
| #ifdef CONFIG_X86_32 |
| |
| #ifdef CONFIG_CC_STACKPROTECTOR |
| #define __switch_canary \ |
| "movl %P[task_canary](%[next]), %%ebx\n\t" \ |
| "movl %%ebx, "__percpu_arg([stack_canary])"\n\t" |
| #define __switch_canary_oparam \ |
| , [stack_canary] "=m" (stack_canary.canary) |
| #define __switch_canary_iparam \ |
| , [task_canary] "i" (offsetof(struct task_struct, stack_canary)) |
| #else /* CC_STACKPROTECTOR */ |
| #define __switch_canary |
| #define __switch_canary_oparam |
| #define __switch_canary_iparam |
| #endif /* CC_STACKPROTECTOR */ |
| |
| /* |
| * Saving eflags is important. It switches not only IOPL between tasks, |
| * it also protects other tasks from NT leaking through sysenter etc. |
| */ |
| #define switch_to(prev, next, last) \ |
| do { \ |
| /* \ |
| * Context-switching clobbers all registers, so we clobber \ |
| * them explicitly, via unused output variables. \ |
| * (EAX and EBP is not listed because EBP is saved/restored \ |
| * explicitly for wchan access and EAX is the return value of \ |
| * __switch_to()) \ |
| */ \ |
| unsigned long ebx, ecx, edx, esi, edi; \ |
| \ |
| asm volatile("pushfl\n\t" /* save flags */ \ |
| "pushl %%ebp\n\t" /* save EBP */ \ |
| "movl %%esp,%[prev_sp]\n\t" /* save ESP */ \ |
| "movl %[next_sp],%%esp\n\t" /* restore ESP */ \ |
| "movl $1f,%[prev_ip]\n\t" /* save EIP */ \ |
| "pushl %[next_ip]\n\t" /* restore EIP */ \ |
| __switch_canary \ |
| "jmp __switch_to\n" /* regparm call */ \ |
| "1:\t" \ |
| "popl %%ebp\n\t" /* restore EBP */ \ |
| "popfl\n" /* restore flags */ \ |
| \ |
| /* output parameters */ \ |
| : [prev_sp] "=m" (prev->thread.sp), \ |
| [prev_ip] "=m" (prev->thread.ip), \ |
| "=a" (last), \ |
| \ |
| /* clobbered output registers: */ \ |
| "=b" (ebx), "=c" (ecx), "=d" (edx), \ |
| "=S" (esi), "=D" (edi) \ |
| \ |
| __switch_canary_oparam \ |
| \ |
| /* input parameters: */ \ |
| : [next_sp] "m" (next->thread.sp), \ |
| [next_ip] "m" (next->thread.ip), \ |
| \ |
| /* regparm parameters for __switch_to(): */ \ |
| [prev] "a" (prev), \ |
| [next] "d" (next) \ |
| \ |
| __switch_canary_iparam \ |
| \ |
| : /* reloaded segment registers */ \ |
| "memory"); \ |
| } while (0) |
| |
| #else /* CONFIG_X86_32 */ |
| |
| /* frame pointer must be last for get_wchan */ |
| #define SAVE_CONTEXT "pushf ; pushq %%rbp ; movq %%rsi,%%rbp\n\t" |
| #define RESTORE_CONTEXT "movq %%rbp,%%rsi ; popq %%rbp ; popf\t" |
| |
| #define __EXTRA_CLOBBER \ |
| , "rcx", "rbx", "rdx", "r8", "r9", "r10", "r11", \ |
| "r12", "r13", "r14", "r15" |
| |
| #ifdef CONFIG_CC_STACKPROTECTOR |
| #define __switch_canary \ |
| "movq %P[task_canary](%%rsi),%%r8\n\t" \ |
| "movq %%r8,"__percpu_arg([gs_canary])"\n\t" |
| #define __switch_canary_oparam \ |
| , [gs_canary] "=m" (irq_stack_union.stack_canary) |
| #define __switch_canary_iparam \ |
| , [task_canary] "i" (offsetof(struct task_struct, stack_canary)) |
| #else /* CC_STACKPROTECTOR */ |
| #define __switch_canary |
| #define __switch_canary_oparam |
| #define __switch_canary_iparam |
| #endif /* CC_STACKPROTECTOR */ |
| |
| /* Save restore flags to clear handle leaking NT */ |
| #define switch_to(prev, next, last) \ |
| asm volatile(SAVE_CONTEXT \ |
| "movq %%rsp,%P[threadrsp](%[prev])\n\t" /* save RSP */ \ |
| "movq %P[threadrsp](%[next]),%%rsp\n\t" /* restore RSP */ \ |
| "call __switch_to\n\t" \ |
| "movq "__percpu_arg([current_task])",%%rsi\n\t" \ |
| __switch_canary \ |
| "movq %P[thread_info](%%rsi),%%r8\n\t" \ |
| "movq %%rax,%%rdi\n\t" \ |
| "testl %[_tif_fork],%P[ti_flags](%%r8)\n\t" \ |
| "jnz ret_from_fork\n\t" \ |
| RESTORE_CONTEXT \ |
| : "=a" (last) \ |
| __switch_canary_oparam \ |
| : [next] "S" (next), [prev] "D" (prev), \ |
| [threadrsp] "i" (offsetof(struct task_struct, thread.sp)), \ |
| [ti_flags] "i" (offsetof(struct thread_info, flags)), \ |
| [_tif_fork] "i" (_TIF_FORK), \ |
| [thread_info] "i" (offsetof(struct task_struct, stack)), \ |
| [current_task] "m" (current_task) \ |
| __switch_canary_iparam \ |
| : "memory", "cc" __EXTRA_CLOBBER) |
| |
| #endif /* CONFIG_X86_32 */ |
| |
| #endif /* _ASM_X86_SWITCH_TO_H */ |