]> www.infradead.org Git - users/hch/configfs.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)
committerOliver Upton <oliver.upton@linux.dev>
Thu, 22 Aug 2024 08:08:37 +0000 (08:08 +0000)
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>
arch/arm64/kvm/sys_regs.c
arch/arm64/kvm/vgic/vgic.h

index c90324060436b272fd0f7dde8e023a7911edcc7b..31e49da867ffc32ecb66bf30bd0e667ef441170f 100644 (file)
@@ -33,6 +33,7 @@
 #include <trace/events/kvm.h>
 
 #include "sys_regs.h"
+#include "vgic/vgic.h"
 
 #include "trace.h"
 
@@ -435,6 +436,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 ba8f790431bd3a8b34745f5679cbd587a2666440..8532bfe3fed40c6fcfb05515bf0709e59278d07b 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