diff options
Diffstat (limited to 'arch/x86/entry/entry_32.S')
| -rw-r--r-- | arch/x86/entry/entry_32.S | 30 | 
1 files changed, 22 insertions, 8 deletions
| diff --git a/arch/x86/entry/entry_32.S b/arch/x86/entry/entry_32.S index 4838037f97f6..60c4c342316c 100644 --- a/arch/x86/entry/entry_32.S +++ b/arch/x86/entry/entry_32.S @@ -44,6 +44,7 @@  #include <asm/asm.h>  #include <asm/smap.h>  #include <asm/frame.h> +#include <asm/nospec-branch.h>  	.section .entry.text, "ax" @@ -243,6 +244,17 @@ ENTRY(__switch_to_asm)  	movl	%ebx, PER_CPU_VAR(stack_canary)+stack_canary_offset  #endif +#ifdef CONFIG_RETPOLINE +	/* +	 * When switching from a shallower to a deeper call stack +	 * the RSB may either underflow or use entries populated +	 * with userspace addresses. On CPUs where those concerns +	 * exist, overwrite the RSB with entries which capture +	 * speculative execution to prevent attack. +	 */ +	FILL_RETURN_BUFFER %ebx, RSB_CLEAR_LOOPS, X86_FEATURE_RSB_CTXSW +#endif +  	/* restore callee-saved registers */  	popl	%esi  	popl	%edi @@ -290,7 +302,7 @@ ENTRY(ret_from_fork)  	/* kernel thread */  1:	movl	%edi, %eax -	call	*%ebx +	CALL_NOSPEC %ebx  	/*  	 * A kernel thread is allowed to return here after successfully  	 * calling do_execve().  Exit to userspace to complete the execve() @@ -919,7 +931,7 @@ common_exception:  	movl	%ecx, %es  	TRACE_IRQS_OFF  	movl	%esp, %eax			# pt_regs pointer -	call	*%edi +	CALL_NOSPEC %edi  	jmp	ret_from_exception  END(common_exception) @@ -941,9 +953,10 @@ ENTRY(debug)  	movl	%esp, %eax			# pt_regs pointer  	/* Are we currently on the SYSENTER stack? */ -	PER_CPU(cpu_tss + CPU_TSS_SYSENTER_stack + SIZEOF_SYSENTER_stack, %ecx) -	subl	%eax, %ecx	/* ecx = (end of SYSENTER_stack) - esp */ -	cmpl	$SIZEOF_SYSENTER_stack, %ecx +	movl	PER_CPU_VAR(cpu_entry_area), %ecx +	addl	$CPU_ENTRY_AREA_entry_stack + SIZEOF_entry_stack, %ecx +	subl	%eax, %ecx	/* ecx = (end of entry_stack) - esp */ +	cmpl	$SIZEOF_entry_stack, %ecx  	jb	.Ldebug_from_sysenter_stack  	TRACE_IRQS_OFF @@ -984,9 +997,10 @@ ENTRY(nmi)  	movl	%esp, %eax			# pt_regs pointer  	/* Are we currently on the SYSENTER stack? */ -	PER_CPU(cpu_tss + CPU_TSS_SYSENTER_stack + SIZEOF_SYSENTER_stack, %ecx) -	subl	%eax, %ecx	/* ecx = (end of SYSENTER_stack) - esp */ -	cmpl	$SIZEOF_SYSENTER_stack, %ecx +	movl	PER_CPU_VAR(cpu_entry_area), %ecx +	addl	$CPU_ENTRY_AREA_entry_stack + SIZEOF_entry_stack, %ecx +	subl	%eax, %ecx	/* ecx = (end of entry_stack) - esp */ +	cmpl	$SIZEOF_entry_stack, %ecx  	jb	.Lnmi_from_sysenter_stack  	/* Not on SYSENTER stack. */ |