diff options
Diffstat (limited to 'arch/x86/kernel/nmi.c')
| -rw-r--r-- | arch/x86/kernel/nmi.c | 55 | 
1 files changed, 47 insertions, 8 deletions
| diff --git a/arch/x86/kernel/nmi.c b/arch/x86/kernel/nmi.c index 17e955ab69fe..9a5b372c706f 100644 --- a/arch/x86/kernel/nmi.c +++ b/arch/x86/kernel/nmi.c @@ -35,6 +35,7 @@  #include <asm/nospec-branch.h>  #include <asm/microcode.h>  #include <asm/sev.h> +#include <asm/fred.h>  #define CREATE_TRACE_POINTS  #include <trace/events/nmi.h> @@ -303,13 +304,13 @@ unknown_nmi_error(unsigned char reason, struct pt_regs *regs)  	__this_cpu_add(nmi_stats.unknown, 1); -	pr_emerg("Uhhuh. NMI received for unknown reason %02x on CPU %d.\n", -		 reason, smp_processor_id()); +	pr_emerg_ratelimited("Uhhuh. NMI received for unknown reason %02x on CPU %d.\n", +			     reason, smp_processor_id());  	if (unknown_nmi_panic || panic_on_unrecovered_nmi)  		nmi_panic(regs, "NMI: Not continuing"); -	pr_emerg("Dazed and confused, but trying to continue\n"); +	pr_emerg_ratelimited("Dazed and confused, but trying to continue\n");  }  NOKPROBE_SYMBOL(unknown_nmi_error); @@ -502,7 +503,7 @@ DEFINE_IDTENTRY_RAW(exc_nmi)  	if (IS_ENABLED(CONFIG_NMI_CHECK_CPU))  		raw_atomic_long_inc(&nsp->idt_calls); -	if (IS_ENABLED(CONFIG_SMP) && arch_cpu_is_offline(smp_processor_id())) { +	if (arch_cpu_is_offline(smp_processor_id())) {  		if (microcode_nmi_handler_enabled())  			microcode_offline_nmi_handler();  		return; @@ -563,9 +564,6 @@ nmi_restart:  	}  	if (this_cpu_dec_return(nmi_state))  		goto nmi_restart; - -	if (user_mode(regs)) -		mds_user_clear_cpu_buffers();  }  #if IS_ENABLED(CONFIG_KVM_INTEL) @@ -639,7 +637,7 @@ void nmi_backtrace_stall_check(const struct cpumask *btp)  			msgp = nmi_check_stall_msg[idx];  			if (nsp->idt_ignored_snap != READ_ONCE(nsp->idt_ignored) && (idx & 0x1))  				modp = ", but OK because ignore_nmis was set"; -			if (nmi_seq & ~0x1) +			if (nmi_seq & 0x1)  				msghp = " (CPU currently in NMI handler function)";  			else if (nsp->idt_nmi_seq_snap + 1 == nmi_seq)  				msghp = " (CPU exited one NMI handler function)"; @@ -651,6 +649,47 @@ void nmi_backtrace_stall_check(const struct cpumask *btp)  #endif +#ifdef CONFIG_X86_FRED +/* + * With FRED, CR2/DR6 is pushed to #PF/#DB stack frame during FRED + * event delivery, i.e., there is no problem of transient states. + * And NMI unblocking only happens when the stack frame indicates + * that so should happen. + * + * Thus, the NMI entry stub for FRED is really straightforward and + * as simple as most exception handlers. As such, #DB is allowed + * during NMI handling. + */ +DEFINE_FREDENTRY_NMI(exc_nmi) +{ +	irqentry_state_t irq_state; + +	if (arch_cpu_is_offline(smp_processor_id())) { +		if (microcode_nmi_handler_enabled()) +			microcode_offline_nmi_handler(); +		return; +	} + +	/* +	 * Save CR2 for eventual restore to cover the case where the NMI +	 * hits the VMENTER/VMEXIT region where guest CR2 is life. This +	 * prevents guest state corruption in case that the NMI handler +	 * takes a page fault. +	 */ +	this_cpu_write(nmi_cr2, read_cr2()); + +	irq_state = irqentry_nmi_enter(regs); + +	inc_irq_stat(__nmi_count); +	default_do_nmi(regs); + +	irqentry_nmi_exit(regs, irq_state); + +	if (unlikely(this_cpu_read(nmi_cr2) != read_cr2())) +		write_cr2(this_cpu_read(nmi_cr2)); +} +#endif +  void stop_nmi(void)  {  	ignore_nmis++; |