]> git.ipfire.org Git - thirdparty/kernel/stable-queue.git/commitdiff
5.12-stable patches
authorGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Tue, 8 Jun 2021 15:43:43 +0000 (17:43 +0200)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Tue, 8 Jun 2021 15:43:43 +0000 (17:43 +0200)
added patches:
kvm-svm-truncate-gpr-value-for-dr-and-cr-accesses-in-64-bit-mode.patch
x86-kvm-disable-all-pv-features-on-crash.patch
x86-kvm-disable-kvmclock-on-all-cpus-on-shutdown.patch
x86-kvm-teardown-pv-features-on-boot-cpu-as-well.patch

queue-5.12/kvm-svm-truncate-gpr-value-for-dr-and-cr-accesses-in-64-bit-mode.patch [new file with mode: 0644]
queue-5.12/series
queue-5.12/x86-kvm-disable-all-pv-features-on-crash.patch [new file with mode: 0644]
queue-5.12/x86-kvm-disable-kvmclock-on-all-cpus-on-shutdown.patch [new file with mode: 0644]
queue-5.12/x86-kvm-teardown-pv-features-on-boot-cpu-as-well.patch [new file with mode: 0644]

diff --git a/queue-5.12/kvm-svm-truncate-gpr-value-for-dr-and-cr-accesses-in-64-bit-mode.patch b/queue-5.12/kvm-svm-truncate-gpr-value-for-dr-and-cr-accesses-in-64-bit-mode.patch
new file mode 100644 (file)
index 0000000..b012fef
--- /dev/null
@@ -0,0 +1,63 @@
+From foo@baz Tue Jun  8 05:38:02 PM CEST 2021
+From: Sean Christopherson <seanjc@google.com>
+Date: Wed, 21 Apr 2021 19:21:22 -0700
+Subject: KVM: SVM: Truncate GPR value for DR and CR accesses in !64-bit mode
+
+From: Sean Christopherson <seanjc@google.com>
+
+commit 0884335a2e653b8a045083aa1d57ce74269ac81d upstream.
+
+Drop bits 63:32 on loads/stores to/from DRs and CRs when the vCPU is not
+in 64-bit mode.  The APM states bits 63:32 are dropped for both DRs and
+CRs:
+
+  In 64-bit mode, the operand size is fixed at 64 bits without the need
+  for a REX prefix. In non-64-bit mode, the operand size is fixed at 32
+  bits and the upper 32 bits of the destination are forced to 0.
+
+Fixes: 7ff76d58a9dc ("KVM: SVM: enhance MOV CR intercept handler")
+Fixes: cae3797a4639 ("KVM: SVM: enhance mov DR intercept handler")
+Cc: stable@vger.kernel.org
+Signed-off-by: Sean Christopherson <seanjc@google.com>
+Message-Id: <20210422022128.3464144-4-seanjc@google.com>
+Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
+Signed-off-by: Sudip Mukherjee <sudipm.mukherjee@gmail.com>
+Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
+---
+ arch/x86/kvm/svm/svm.c |    8 ++++----
+ 1 file changed, 4 insertions(+), 4 deletions(-)
+
+--- a/arch/x86/kvm/svm/svm.c
++++ b/arch/x86/kvm/svm/svm.c
+@@ -2532,7 +2532,7 @@ static int cr_interception(struct vcpu_s
+       err = 0;
+       if (cr >= 16) { /* mov to cr */
+               cr -= 16;
+-              val = kvm_register_read(&svm->vcpu, reg);
++              val = kvm_register_readl(&svm->vcpu, reg);
+               trace_kvm_cr_write(cr, val);
+               switch (cr) {
+               case 0:
+@@ -2578,7 +2578,7 @@ static int cr_interception(struct vcpu_s
+                       kvm_queue_exception(&svm->vcpu, UD_VECTOR);
+                       return 1;
+               }
+-              kvm_register_write(&svm->vcpu, reg, val);
++              kvm_register_writel(&svm->vcpu, reg, val);
+               trace_kvm_cr_read(cr, val);
+       }
+       return kvm_complete_insn_gp(&svm->vcpu, err);
+@@ -2643,11 +2643,11 @@ static int dr_interception(struct vcpu_s
+       dr = svm->vmcb->control.exit_code - SVM_EXIT_READ_DR0;
+       if (dr >= 16) { /* mov to DRn  */
+               dr -= 16;
+-              val = kvm_register_read(&svm->vcpu, reg);
++              val = kvm_register_readl(&svm->vcpu, reg);
+               err = kvm_set_dr(&svm->vcpu, dr, val);
+       } else {
+               kvm_get_dr(&svm->vcpu, dr, &val);
+-              kvm_register_write(&svm->vcpu, reg, val);
++              kvm_register_writel(&svm->vcpu, reg, val);
+       }
+       return kvm_complete_insn_gp(&svm->vcpu, err);
index c99abfd598dda544c9292d6f6ca8b8de392b19e3..5ab53cd88e818144c6f90a574e15c0775a56fe8d 100644 (file)
@@ -147,3 +147,7 @@ btrfs-abort-in-rename_exchange-if-we-fail-to-insert-the-second-ref.patch
 btrfs-fix-deadlock-when-cloning-inline-extents-and-low-on-available-space.patch
 mm-hugetlb-fix-simple-resv_huge_pages-underflow-on-u.patch
 drm-msm-dpu-always-use-mdp-device-to-scale-bandwidth.patch
+kvm-svm-truncate-gpr-value-for-dr-and-cr-accesses-in-64-bit-mode.patch
+x86-kvm-teardown-pv-features-on-boot-cpu-as-well.patch
+x86-kvm-disable-kvmclock-on-all-cpus-on-shutdown.patch
+x86-kvm-disable-all-pv-features-on-crash.patch
diff --git a/queue-5.12/x86-kvm-disable-all-pv-features-on-crash.patch b/queue-5.12/x86-kvm-disable-all-pv-features-on-crash.patch
new file mode 100644 (file)
index 0000000..163899d
--- /dev/null
@@ -0,0 +1,201 @@
+From foo@baz Tue Jun  8 05:43:06 PM CEST 2021
+From: Krzysztof Kozlowski <krzysztof.kozlowski@canonical.com>
+Date: Tue,  1 Jun 2021 09:16:44 +0200
+Subject: x86/kvm: Disable all PV features on crash
+To: stable@vger.kernel.org
+Cc: Andrea Righi <andrea.righi@canonical.com>, Paolo Bonzini <pbonzini@redhat.com>, Vitaly Kuznetsov <vkuznets@redhat.com>, Krzysztof Kozlowski <krzysztof.kozlowski@canonical.com>
+Message-ID: <20210601071644.6055-3-krzysztof.kozlowski@canonical.com>
+
+From: Vitaly Kuznetsov <vkuznets@redhat.com>
+
+commit 3d6b84132d2a57b5a74100f6923a8feb679ac2ce upstream.
+
+Crash shutdown handler only disables kvmclock and steal time, other PV
+features remain active so we risk corrupting memory or getting some
+side-effects in kdump kernel. Move crash handler to kvm.c and unify
+with CPU offline.
+
+Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
+Message-Id: <20210414123544.1060604-5-vkuznets@redhat.com>
+Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
+Signed-off-by: Krzysztof Kozlowski <krzysztof.kozlowski@canonical.com>
+Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
+---
+ arch/x86/include/asm/kvm_para.h |    6 -----
+ arch/x86/kernel/kvm.c           |   44 +++++++++++++++++++++++++++++-----------
+ arch/x86/kernel/kvmclock.c      |   21 -------------------
+ 3 files changed, 32 insertions(+), 39 deletions(-)
+
+--- a/arch/x86/include/asm/kvm_para.h
++++ b/arch/x86/include/asm/kvm_para.h
+@@ -92,7 +92,6 @@ unsigned int kvm_arch_para_hints(void);
+ void kvm_async_pf_task_wait_schedule(u32 token);
+ void kvm_async_pf_task_wake(u32 token);
+ u32 kvm_read_and_reset_apf_flags(void);
+-void kvm_disable_steal_time(void);
+ bool __kvm_handle_async_pf(struct pt_regs *regs, u32 token);
+ DECLARE_STATIC_KEY_FALSE(kvm_async_pf_enabled);
+@@ -137,11 +136,6 @@ static inline u32 kvm_read_and_reset_apf
+       return 0;
+ }
+-static inline void kvm_disable_steal_time(void)
+-{
+-      return;
+-}
+-
+ static __always_inline bool kvm_handle_async_pf(struct pt_regs *regs, u32 token)
+ {
+       return false;
+--- a/arch/x86/kernel/kvm.c
++++ b/arch/x86/kernel/kvm.c
+@@ -38,6 +38,7 @@
+ #include <asm/tlb.h>
+ #include <asm/cpuidle_haltpoll.h>
+ #include <asm/ptrace.h>
++#include <asm/reboot.h>
+ #include <asm/svm.h>
+ DEFINE_STATIC_KEY_FALSE(kvm_async_pf_enabled);
+@@ -375,6 +376,14 @@ static void kvm_pv_disable_apf(void)
+       pr_info("Unregister pv shared memory for cpu %d\n", smp_processor_id());
+ }
++static void kvm_disable_steal_time(void)
++{
++      if (!has_steal_clock)
++              return;
++
++      wrmsr(MSR_KVM_STEAL_TIME, 0, 0);
++}
++
+ static void kvm_pv_guest_cpu_reboot(void *unused)
+ {
+       /*
+@@ -417,14 +426,6 @@ static u64 kvm_steal_clock(int cpu)
+       return steal;
+ }
+-void kvm_disable_steal_time(void)
+-{
+-      if (!has_steal_clock)
+-              return;
+-
+-      wrmsr(MSR_KVM_STEAL_TIME, 0, 0);
+-}
+-
+ static inline void __set_percpu_decrypted(void *ptr, unsigned long size)
+ {
+       early_set_memory_decrypted((unsigned long) ptr, size);
+@@ -461,13 +462,14 @@ static bool pv_tlb_flush_supported(void)
+ static DEFINE_PER_CPU(cpumask_var_t, __pv_cpu_mask);
+-static void kvm_guest_cpu_offline(void)
++static void kvm_guest_cpu_offline(bool shutdown)
+ {
+       kvm_disable_steal_time();
+       if (kvm_para_has_feature(KVM_FEATURE_PV_EOI))
+               wrmsrl(MSR_KVM_PV_EOI_EN, 0);
+       kvm_pv_disable_apf();
+-      apf_task_wake_all();
++      if (!shutdown)
++              apf_task_wake_all();
+       kvmclock_disable();
+ }
+@@ -613,7 +615,7 @@ static int kvm_cpu_down_prepare(unsigned
+       unsigned long flags;
+       local_irq_save(flags);
+-      kvm_guest_cpu_offline();
++      kvm_guest_cpu_offline(false);
+       local_irq_restore(flags);
+       return 0;
+ }
+@@ -622,7 +624,7 @@ static int kvm_cpu_down_prepare(unsigned
+ static int kvm_suspend(void)
+ {
+-      kvm_guest_cpu_offline();
++      kvm_guest_cpu_offline(false);
+       return 0;
+ }
+@@ -637,6 +639,20 @@ static struct syscore_ops kvm_syscore_op
+       .resume         = kvm_resume,
+ };
++/*
++ * After a PV feature is registered, the host will keep writing to the
++ * registered memory location. If the guest happens to shutdown, this memory
++ * won't be valid. In cases like kexec, in which you install a new kernel, this
++ * means a random memory location will be kept being written.
++ */
++#ifdef CONFIG_KEXEC_CORE
++static void kvm_crash_shutdown(struct pt_regs *regs)
++{
++      kvm_guest_cpu_offline(true);
++      native_machine_crash_shutdown(regs);
++}
++#endif
++
+ static void kvm_flush_tlb_others(const struct cpumask *cpumask,
+                       const struct flush_tlb_info *info)
+ {
+@@ -705,6 +721,10 @@ static void __init kvm_guest_init(void)
+       kvm_guest_cpu_init();
+ #endif
++#ifdef CONFIG_KEXEC_CORE
++      machine_ops.crash_shutdown = kvm_crash_shutdown;
++#endif
++
+       register_syscore_ops(&kvm_syscore_ops);
+       /*
+--- a/arch/x86/kernel/kvmclock.c
++++ b/arch/x86/kernel/kvmclock.c
+@@ -20,7 +20,6 @@
+ #include <asm/hypervisor.h>
+ #include <asm/mem_encrypt.h>
+ #include <asm/x86_init.h>
+-#include <asm/reboot.h>
+ #include <asm/kvmclock.h>
+ static int kvmclock __initdata = 1;
+@@ -203,23 +202,6 @@ static void kvm_setup_secondary_clock(vo
+ }
+ #endif
+-/*
+- * After the clock is registered, the host will keep writing to the
+- * registered memory location. If the guest happens to shutdown, this memory
+- * won't be valid. In cases like kexec, in which you install a new kernel, this
+- * means a random memory location will be kept being written. So before any
+- * kind of shutdown from our side, we unregister the clock by writing anything
+- * that does not have the 'enable' bit set in the msr
+- */
+-#ifdef CONFIG_KEXEC_CORE
+-static void kvm_crash_shutdown(struct pt_regs *regs)
+-{
+-      native_write_msr(msr_kvm_system_time, 0, 0);
+-      kvm_disable_steal_time();
+-      native_machine_crash_shutdown(regs);
+-}
+-#endif
+-
+ void kvmclock_disable(void)
+ {
+       native_write_msr(msr_kvm_system_time, 0, 0);
+@@ -349,9 +331,6 @@ void __init kvmclock_init(void)
+ #endif
+       x86_platform.save_sched_clock_state = kvm_save_sched_clock_state;
+       x86_platform.restore_sched_clock_state = kvm_restore_sched_clock_state;
+-#ifdef CONFIG_KEXEC_CORE
+-      machine_ops.crash_shutdown  = kvm_crash_shutdown;
+-#endif
+       kvm_get_preset_lpj();
+       /*
diff --git a/queue-5.12/x86-kvm-disable-kvmclock-on-all-cpus-on-shutdown.patch b/queue-5.12/x86-kvm-disable-kvmclock-on-all-cpus-on-shutdown.patch
new file mode 100644 (file)
index 0000000..1e16575
--- /dev/null
@@ -0,0 +1,86 @@
+From foo@baz Tue Jun  8 05:43:06 PM CEST 2021
+From: Krzysztof Kozlowski <krzysztof.kozlowski@canonical.com>
+Date: Tue,  1 Jun 2021 09:16:43 +0200
+Subject: x86/kvm: Disable kvmclock on all CPUs on shutdown
+To: stable@vger.kernel.org
+Cc: Andrea Righi <andrea.righi@canonical.com>, Paolo Bonzini <pbonzini@redhat.com>, Vitaly Kuznetsov <vkuznets@redhat.com>, Krzysztof Kozlowski <krzysztof.kozlowski@canonical.com>
+Message-ID: <20210601071644.6055-2-krzysztof.kozlowski@canonical.com>
+
+From: Vitaly Kuznetsov <vkuznets@redhat.com>
+
+commit c02027b5742b5aa804ef08a4a9db433295533046 upstream.
+
+Currenly, we disable kvmclock from machine_shutdown() hook and this
+only happens for boot CPU. We need to disable it for all CPUs to
+guard against memory corruption e.g. on restore from hibernate.
+
+Note, writing '0' to kvmclock MSR doesn't clear memory location, it
+just prevents hypervisor from updating the location so for the short
+while after write and while CPU is still alive, the clock remains usable
+and correct so we don't need to switch to some other clocksource.
+
+Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
+Message-Id: <20210414123544.1060604-4-vkuznets@redhat.com>
+Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
+Signed-off-by: Andrea Righi <andrea.righi@canonical.com>
+Signed-off-by: Krzysztof Kozlowski <krzysztof.kozlowski@canonical.com>
+Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
+---
+ arch/x86/include/asm/kvm_para.h |    4 ++--
+ arch/x86/kernel/kvm.c           |    1 +
+ arch/x86/kernel/kvmclock.c      |    5 +----
+ 3 files changed, 4 insertions(+), 6 deletions(-)
+
+--- a/arch/x86/include/asm/kvm_para.h
++++ b/arch/x86/include/asm/kvm_para.h
+@@ -7,8 +7,6 @@
+ #include <linux/interrupt.h>
+ #include <uapi/asm/kvm_para.h>
+-extern void kvmclock_init(void);
+-
+ #ifdef CONFIG_KVM_GUEST
+ bool kvm_check_and_clear_guest_paused(void);
+ #else
+@@ -86,6 +84,8 @@ static inline long kvm_hypercall4(unsign
+ }
+ #ifdef CONFIG_KVM_GUEST
++void kvmclock_init(void);
++void kvmclock_disable(void);
+ bool kvm_para_available(void);
+ unsigned int kvm_arch_para_features(void);
+ unsigned int kvm_arch_para_hints(void);
+--- a/arch/x86/kernel/kvm.c
++++ b/arch/x86/kernel/kvm.c
+@@ -468,6 +468,7 @@ static void kvm_guest_cpu_offline(void)
+               wrmsrl(MSR_KVM_PV_EOI_EN, 0);
+       kvm_pv_disable_apf();
+       apf_task_wake_all();
++      kvmclock_disable();
+ }
+ static int kvm_cpu_online(unsigned int cpu)
+--- a/arch/x86/kernel/kvmclock.c
++++ b/arch/x86/kernel/kvmclock.c
+@@ -220,11 +220,9 @@ static void kvm_crash_shutdown(struct pt
+ }
+ #endif
+-static void kvm_shutdown(void)
++void kvmclock_disable(void)
+ {
+       native_write_msr(msr_kvm_system_time, 0, 0);
+-      kvm_disable_steal_time();
+-      native_machine_shutdown();
+ }
+ static void __init kvmclock_init_mem(void)
+@@ -351,7 +349,6 @@ void __init kvmclock_init(void)
+ #endif
+       x86_platform.save_sched_clock_state = kvm_save_sched_clock_state;
+       x86_platform.restore_sched_clock_state = kvm_restore_sched_clock_state;
+-      machine_ops.shutdown  = kvm_shutdown;
+ #ifdef CONFIG_KEXEC_CORE
+       machine_ops.crash_shutdown  = kvm_crash_shutdown;
+ #endif
diff --git a/queue-5.12/x86-kvm-teardown-pv-features-on-boot-cpu-as-well.patch b/queue-5.12/x86-kvm-teardown-pv-features-on-boot-cpu-as-well.patch
new file mode 100644 (file)
index 0000000..4f32cad
--- /dev/null
@@ -0,0 +1,139 @@
+From foo@baz Tue Jun  8 05:43:06 PM CEST 2021
+From: Krzysztof Kozlowski <krzysztof.kozlowski@canonical.com>
+Date: Tue,  1 Jun 2021 09:16:42 +0200
+Subject: x86/kvm: Teardown PV features on boot CPU as well
+To: stable@vger.kernel.org
+Cc: Andrea Righi <andrea.righi@canonical.com>, Paolo Bonzini <pbonzini@redhat.com>, Vitaly Kuznetsov <vkuznets@redhat.com>, Krzysztof Kozlowski <krzysztof.kozlowski@canonical.com>
+Message-ID: <20210601071644.6055-1-krzysztof.kozlowski@canonical.com>
+
+From: Vitaly Kuznetsov <vkuznets@redhat.com>
+
+commit 8b79feffeca28c5459458fe78676b081e87c93a4 upstream.
+
+Various PV features (Async PF, PV EOI, steal time) work through memory
+shared with hypervisor and when we restore from hibernation we must
+properly teardown all these features to make sure hypervisor doesn't
+write to stale locations after we jump to the previously hibernated kernel
+(which can try to place anything there). For secondary CPUs the job is
+already done by kvm_cpu_down_prepare(), register syscore ops to do
+the same for boot CPU.
+
+Krzysztof:
+This fixes memory corruption visible after second resume from
+hibernation:
+
+  BUG: Bad page state in process dbus-daemon  pfn:18b01
+  page:ffffea000062c040 refcount:0 mapcount:0 mapping:0000000000000000 index:0x1 compound_mapcount: -30591
+  flags: 0xfffffc0078141(locked|error|workingset|writeback|head|mappedtodisk|reclaim)
+  raw: 000fffffc0078141 dead0000000002d0 dead000000000100 0000000000000000
+  raw: 0000000000000001 0000000000000000 00000000ffffffff 0000000000000000
+  page dumped because: PAGE_FLAGS_CHECK_AT_PREP flag set
+  bad because of flags: 0x78141(locked|error|workingset|writeback|head|mappedtodisk|reclaim)
+
+Signed-off-by: Vitaly Kuznetsov <vkuznets@redhat.com>
+Message-Id: <20210414123544.1060604-3-vkuznets@redhat.com>
+Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
+Signed-off-by: Andrea Righi <andrea.righi@canonical.com>
+[krzysztof: Extend the commit message, adjust for v5.10 context]
+Signed-off-by: Krzysztof Kozlowski <krzysztof.kozlowski@canonical.com>
+Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
+---
+ arch/x86/kernel/kvm.c |   57 +++++++++++++++++++++++++++++++++++---------------
+ 1 file changed, 41 insertions(+), 16 deletions(-)
+
+--- a/arch/x86/kernel/kvm.c
++++ b/arch/x86/kernel/kvm.c
+@@ -26,6 +26,7 @@
+ #include <linux/kprobes.h>
+ #include <linux/nmi.h>
+ #include <linux/swait.h>
++#include <linux/syscore_ops.h>
+ #include <asm/timer.h>
+ #include <asm/cpu.h>
+ #include <asm/traps.h>
+@@ -460,6 +461,25 @@ static bool pv_tlb_flush_supported(void)
+ static DEFINE_PER_CPU(cpumask_var_t, __pv_cpu_mask);
++static void kvm_guest_cpu_offline(void)
++{
++      kvm_disable_steal_time();
++      if (kvm_para_has_feature(KVM_FEATURE_PV_EOI))
++              wrmsrl(MSR_KVM_PV_EOI_EN, 0);
++      kvm_pv_disable_apf();
++      apf_task_wake_all();
++}
++
++static int kvm_cpu_online(unsigned int cpu)
++{
++      unsigned long flags;
++
++      local_irq_save(flags);
++      kvm_guest_cpu_init();
++      local_irq_restore(flags);
++      return 0;
++}
++
+ #ifdef CONFIG_SMP
+ static bool pv_ipi_supported(void)
+@@ -587,31 +607,34 @@ static void __init kvm_smp_prepare_boot_
+       kvm_spinlock_init();
+ }
+-static void kvm_guest_cpu_offline(void)
++static int kvm_cpu_down_prepare(unsigned int cpu)
+ {
+-      kvm_disable_steal_time();
+-      if (kvm_para_has_feature(KVM_FEATURE_PV_EOI))
+-              wrmsrl(MSR_KVM_PV_EOI_EN, 0);
+-      kvm_pv_disable_apf();
+-      apf_task_wake_all();
+-}
++      unsigned long flags;
+-static int kvm_cpu_online(unsigned int cpu)
+-{
+-      local_irq_disable();
+-      kvm_guest_cpu_init();
+-      local_irq_enable();
++      local_irq_save(flags);
++      kvm_guest_cpu_offline();
++      local_irq_restore(flags);
+       return 0;
+ }
+-static int kvm_cpu_down_prepare(unsigned int cpu)
++#endif
++
++static int kvm_suspend(void)
+ {
+-      local_irq_disable();
+       kvm_guest_cpu_offline();
+-      local_irq_enable();
++
+       return 0;
+ }
+-#endif
++
++static void kvm_resume(void)
++{
++      kvm_cpu_online(raw_smp_processor_id());
++}
++
++static struct syscore_ops kvm_syscore_ops = {
++      .suspend        = kvm_suspend,
++      .resume         = kvm_resume,
++};
+ static void kvm_flush_tlb_others(const struct cpumask *cpumask,
+                       const struct flush_tlb_info *info)
+@@ -681,6 +704,8 @@ static void __init kvm_guest_init(void)
+       kvm_guest_cpu_init();
+ #endif
++      register_syscore_ops(&kvm_syscore_ops);
++
+       /*
+        * Hard lockup detection is enabled by default. Disable it, as guests
+        * can get false positives too easily, for example if the host is