]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
LoongArch: KVM: Add valid bit check when set CSR.ESTAT register
authorBibo Mao <maobibo@loongson.cn>
Thu, 11 Jun 2026 12:46:42 +0000 (20:46 +0800)
committerHuacai Chen <chenhuacai@loongson.cn>
Thu, 11 Jun 2026 12:46:42 +0000 (20:46 +0800)
When set CSR.ESTAT register in function _kvm_setcsr(), valid bit check
is added here. Also interrupt CPU_AVEC is checked by msgint feature.

Signed-off-by: Bibo Mao <maobibo@loongson.cn>
Signed-off-by: Huacai Chen <chenhuacai@loongson.cn>
arch/loongarch/kvm/vcpu.c

index cb42f43afcc832fa326d1b13b2a1bb184f272284..89aa49df276a1ba09fee597784a1caabfb18b60e 100644 (file)
@@ -593,7 +593,7 @@ struct kvm_vcpu *kvm_get_vcpu_by_cpuid(struct kvm *kvm, int cpuid)
 
 static int _kvm_getcsr(struct kvm_vcpu *vcpu, unsigned int id, u64 *val)
 {
-       unsigned long gintc;
+       unsigned long estat, gintc;
        struct loongarch_csrs *csr = vcpu->arch.csr;
 
        if (get_gcsr_flag(id) & INVALID_GCSR)
@@ -612,8 +612,9 @@ static int _kvm_getcsr(struct kvm_vcpu *vcpu, unsigned int id, u64 *val)
                preempt_enable();
 
                /* ESTAT IP0~IP7 get from GINTC */
-               gintc = kvm_read_sw_gcsr(csr, LOONGARCH_CSR_GINTC) & 0xff;
-               *val = kvm_read_sw_gcsr(csr, LOONGARCH_CSR_ESTAT) | (gintc << 2);
+               gintc = kvm_read_sw_gcsr(csr, LOONGARCH_CSR_GINTC) & KVM_GINTC_IRQ_MASK;
+               estat = kvm_read_sw_gcsr(csr, LOONGARCH_CSR_ESTAT) & ~KVM_ESTAT_EXTI_MASK;
+               *val = estat | (gintc << VIP_DELTA);
                return 0;
        }
 
@@ -628,7 +629,8 @@ static int _kvm_getcsr(struct kvm_vcpu *vcpu, unsigned int id, u64 *val)
 
 static int _kvm_setcsr(struct kvm_vcpu *vcpu, unsigned int id, u64 val)
 {
-       int ret = 0, gintc;
+       int ret = 0;
+       unsigned long estat, gintc;
        struct loongarch_csrs *csr = vcpu->arch.csr;
 
        if (get_gcsr_flag(id) & INVALID_GCSR)
@@ -639,11 +641,16 @@ static int _kvm_setcsr(struct kvm_vcpu *vcpu, unsigned int id, u64 val)
 
        if (id == LOONGARCH_CSR_ESTAT) {
                /* ESTAT IP0~IP7 inject through GINTC */
-               gintc = (val >> 2) & 0xff;
-               kvm_set_sw_gcsr(csr, LOONGARCH_CSR_GINTC, gintc);
-
-               gintc = val & ~(0xffUL << 2);
-               kvm_set_sw_gcsr(csr, LOONGARCH_CSR_ESTAT, gintc);
+               gintc = (val >> VIP_DELTA) & KVM_GINTC_IRQ_MASK;
+               gintc |= kvm_read_sw_gcsr(csr, LOONGARCH_CSR_GINTC) & ~KVM_GINTC_IRQ_MASK;
+               kvm_write_sw_gcsr(csr, LOONGARCH_CSR_GINTC, gintc);
+
+               /* Only set valid ESTAT bits */
+               estat = val & ~KVM_ESTAT_EXTI_MASK;
+               estat &= CSR_ESTAT_IS | CSR_ESTAT_EXC | CSR_ESTAT_ESUBCODE;
+               if (!kvm_guest_has_msgint(&vcpu->arch))
+                       estat &= ~CPU_AVEC;
+               kvm_write_sw_gcsr(csr, LOONGARCH_CSR_ESTAT, estat);
 
                return ret;
        }