mirror of
https://gitee.com/bianbu-linux/linux-6.6
synced 2025-04-24 14:07:52 -04:00
KVM: Use syscore_ops instead of reboot_notifier to hook restart/shutdown
Use syscore_ops.shutdown to disable hardware virtualization during a reboot instead of using the dedicated reboot_notifier so that KVM disables virtualization _after_ system_state has been updated. This will allow fixing a race in KVM's handling of a forced reboot where KVM can end up enabling hardware virtualization between kernel_restart_prepare() and machine_restart(). Rename KVM's hook to match the syscore op to avoid any possible confusion from wiring up a "reboot" helper to a "shutdown" hook (neither "shutdown nor "reboot" is completely accurate as the hook handles both). Opportunistically rewrite kvm_shutdown()'s comment to make it less VMX specific, and to explain why kvm_rebooting exists. Cc: Marc Zyngier <maz@kernel.org> Cc: Oliver Upton <oliver.upton@linux.dev> Cc: James Morse <james.morse@arm.com> Cc: Suzuki K Poulose <suzuki.poulose@arm.com> Cc: Zenghui Yu <yuzenghui@huawei.com> Cc: kvmarm@lists.linux.dev Cc: Huacai Chen <chenhuacai@kernel.org> Cc: Aleksandar Markovic <aleksandar.qemu.devel@gmail.com> Cc: Anup Patel <anup@brainfault.org> Cc: Atish Patra <atishp@atishpatra.org> Cc: kvm-riscv@lists.infradead.org Signed-off-by: Sean Christopherson <seanjc@google.com> Acked-by: Marc Zyngier <maz@kernel.org> Message-Id: <20230512233127.804012-2-seanjc@google.com> Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
This commit is contained in:
parent
39428f6ea9
commit
6735150b69
1 changed files with 11 additions and 15 deletions
|
@ -5213,26 +5213,24 @@ static int hardware_enable_all(void)
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int kvm_reboot(struct notifier_block *notifier, unsigned long val,
|
static void kvm_shutdown(void)
|
||||||
void *v)
|
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
* Some (well, at least mine) BIOSes hang on reboot if
|
* Disable hardware virtualization and set kvm_rebooting to indicate
|
||||||
* in vmx root mode.
|
* that KVM has asynchronously disabled hardware virtualization, i.e.
|
||||||
*
|
* that relevant errors and exceptions aren't entirely unexpected.
|
||||||
* And Intel TXT required VMX off for all cpu when system shutdown.
|
* Some flavors of hardware virtualization need to be disabled before
|
||||||
|
* transferring control to firmware (to perform shutdown/reboot), e.g.
|
||||||
|
* on x86, virtualization can block INIT interrupts, which are used by
|
||||||
|
* firmware to pull APs back under firmware control. Note, this path
|
||||||
|
* is used for both shutdown and reboot scenarios, i.e. neither name is
|
||||||
|
* 100% comprehensive.
|
||||||
*/
|
*/
|
||||||
pr_info("kvm: exiting hardware virtualization\n");
|
pr_info("kvm: exiting hardware virtualization\n");
|
||||||
kvm_rebooting = true;
|
kvm_rebooting = true;
|
||||||
on_each_cpu(hardware_disable_nolock, NULL, 1);
|
on_each_cpu(hardware_disable_nolock, NULL, 1);
|
||||||
return NOTIFY_OK;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct notifier_block kvm_reboot_notifier = {
|
|
||||||
.notifier_call = kvm_reboot,
|
|
||||||
.priority = 0,
|
|
||||||
};
|
|
||||||
|
|
||||||
static int kvm_suspend(void)
|
static int kvm_suspend(void)
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
|
@ -5263,6 +5261,7 @@ static void kvm_resume(void)
|
||||||
static struct syscore_ops kvm_syscore_ops = {
|
static struct syscore_ops kvm_syscore_ops = {
|
||||||
.suspend = kvm_suspend,
|
.suspend = kvm_suspend,
|
||||||
.resume = kvm_resume,
|
.resume = kvm_resume,
|
||||||
|
.shutdown = kvm_shutdown,
|
||||||
};
|
};
|
||||||
#else /* CONFIG_KVM_GENERIC_HARDWARE_ENABLING */
|
#else /* CONFIG_KVM_GENERIC_HARDWARE_ENABLING */
|
||||||
static int hardware_enable_all(void)
|
static int hardware_enable_all(void)
|
||||||
|
@ -5967,7 +5966,6 @@ int kvm_init(unsigned vcpu_size, unsigned vcpu_align, struct module *module)
|
||||||
if (r)
|
if (r)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
register_reboot_notifier(&kvm_reboot_notifier);
|
|
||||||
register_syscore_ops(&kvm_syscore_ops);
|
register_syscore_ops(&kvm_syscore_ops);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -6039,7 +6037,6 @@ err_cpu_kick_mask:
|
||||||
err_vcpu_cache:
|
err_vcpu_cache:
|
||||||
#ifdef CONFIG_KVM_GENERIC_HARDWARE_ENABLING
|
#ifdef CONFIG_KVM_GENERIC_HARDWARE_ENABLING
|
||||||
unregister_syscore_ops(&kvm_syscore_ops);
|
unregister_syscore_ops(&kvm_syscore_ops);
|
||||||
unregister_reboot_notifier(&kvm_reboot_notifier);
|
|
||||||
cpuhp_remove_state_nocalls(CPUHP_AP_KVM_ONLINE);
|
cpuhp_remove_state_nocalls(CPUHP_AP_KVM_ONLINE);
|
||||||
#endif
|
#endif
|
||||||
return r;
|
return r;
|
||||||
|
@ -6065,7 +6062,6 @@ void kvm_exit(void)
|
||||||
kvm_async_pf_deinit();
|
kvm_async_pf_deinit();
|
||||||
#ifdef CONFIG_KVM_GENERIC_HARDWARE_ENABLING
|
#ifdef CONFIG_KVM_GENERIC_HARDWARE_ENABLING
|
||||||
unregister_syscore_ops(&kvm_syscore_ops);
|
unregister_syscore_ops(&kvm_syscore_ops);
|
||||||
unregister_reboot_notifier(&kvm_reboot_notifier);
|
|
||||||
cpuhp_remove_state_nocalls(CPUHP_AP_KVM_ONLINE);
|
cpuhp_remove_state_nocalls(CPUHP_AP_KVM_ONLINE);
|
||||||
#endif
|
#endif
|
||||||
kvm_irqfd_exit();
|
kvm_irqfd_exit();
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue