]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
KVM: arm64: Make ICC_*SGI*_EL1 undef in the absence of a vGICv3
authorMarc Zyngier <maz@kernel.org>
Tue, 20 Aug 2024 10:03:38 +0000 (11:03 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Thu, 29 Aug 2024 15:36:09 +0000 (17:36 +0200)
commit 3e6245ebe7ef341639e9a7e402b3ade8ad45a19f upstream.

On a system with a GICv3, if a guest hasn't been configured with
GICv3 and that the host is not capable of GICv2 emulation,
a write to any of the ICC_*SGI*_EL1 registers is trapped to EL2.

We therefore try to emulate the SGI access, only to hit a NULL
pointer as no private interrupt is allocated (no GIC, remember?).

The obvious fix is to give the guest what it deserves, in the
shape of a UNDEF exception.

Reported-by: Alexander Potapenko <glider@google.com>
Signed-off-by: Marc Zyngier <maz@kernel.org>
Cc: stable@vger.kernel.org
Link: https://lore.kernel.org/r/20240820100349.3544850-2-maz@kernel.org
Signed-off-by: Oliver Upton <oliver.upton@linux.dev>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
arch/arm64/kvm/sys_regs.c
arch/arm64/kvm/vgic/vgic.h

index 22b45a15d06881ce25ee124f8b69a20491b5df3d..33d866ca8f655114d3bf257141d7728bc8c75b61 100644 (file)
@@ -33,6 +33,7 @@
 #include <trace/events/kvm.h>
 
 #include "sys_regs.h"
+#include "vgic/vgic.h"
 
 #include "trace.h"
 
@@ -428,6 +429,11 @@ static bool access_gic_sgi(struct kvm_vcpu *vcpu,
 {
        bool g1;
 
+       if (!kvm_has_gicv3(vcpu->kvm)) {
+               kvm_inject_undefined(vcpu);
+               return false;
+       }
+
        if (!p->is_write)
                return read_from_write_only(vcpu, p, r);
 
index 03d356a123771fda53c9aff25e475cbb68079bc6..55e3b7108dc9b6c7d57f592743fb31a45108b6f0 100644 (file)
@@ -346,4 +346,11 @@ void vgic_v4_configure_vsgis(struct kvm *kvm);
 void vgic_v4_get_vlpi_state(struct vgic_irq *irq, bool *val);
 int vgic_v4_request_vpe_irq(struct kvm_vcpu *vcpu, int irq);
 
+static inline bool kvm_has_gicv3(struct kvm *kvm)
+{
+       return (static_branch_unlikely(&kvm_vgic_global_state.gicv3_cpuif) &&
+               irqchip_in_kernel(kvm) &&
+               kvm->arch.vgic.vgic_model == KVM_DEV_TYPE_ARM_VGIC_V3);
+}
+
 #endif