]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
irqchip/riscv-imsic: Adjust the number of available guest irq files
authorXu Lu <luxu.kernel@bytedance.com>
Sun, 4 Jan 2026 13:34:57 +0000 (21:34 +0800)
committerAnup Patel <anup@brainfault.org>
Fri, 6 Feb 2026 13:35:34 +0000 (19:05 +0530)
Currently, KVM assumes the minimum of implemented HGEIE bits and
"BIT(gc->guest_index_bits) - 1" as the number of guest files available
across all CPUs. This will not work when CPUs have different number
of guest files because KVM may incorrectly allocate a guest file on a
CPU with fewer guest files.

To address above, during initialization, calculate the number of
available guest interrupt files according to MMIO resources and
constrain the number of guest interrupt files that can be allocated
by KVM.

Signed-off-by: Xu Lu <luxu.kernel@bytedance.com>
Reviewed-by: Nutty Liu <nutty.liu@hotmail.com>
Reviewed-by: Anup Patel <anup@brainfault.org>
Acked-by: Thomas Gleixner <tglx@kernel.org>
Link: https://lore.kernel.org/r/20260104133457.57742-1-luxu.kernel@bytedance.com
Signed-off-by: Anup Patel <anup@brainfault.org>
arch/riscv/kvm/aia.c
drivers/irqchip/irq-riscv-imsic-state.c
include/linux/irqchip/riscv-imsic.h

index dad3181856600f13bd5b1d2bf36f21766da4711c..cac3c2b51d7247cf4be69f4c3b692c72a27b99c9 100644 (file)
@@ -630,7 +630,7 @@ int kvm_riscv_aia_init(void)
         */
        if (gc)
                kvm_riscv_aia_nr_hgei = min((ulong)kvm_riscv_aia_nr_hgei,
-                                           BIT(gc->guest_index_bits) - 1);
+                                           gc->nr_guest_files);
        else
                kvm_riscv_aia_nr_hgei = 0;
 
index b6cebfee9461191b833b077626ac8ac2fd7e5c78..7566c8aa2d487597a7b38fa1993a3984b44bab6f 100644 (file)
@@ -784,7 +784,7 @@ static int __init imsic_parse_fwnode(struct fwnode_handle *fwnode,
 
 int __init imsic_setup_state(struct fwnode_handle *fwnode, void *opaque)
 {
-       u32 i, j, index, nr_parent_irqs, nr_mmios, nr_handlers = 0;
+       u32 i, j, index, nr_parent_irqs, nr_mmios, nr_guest_files, nr_handlers = 0;
        struct imsic_global_config *global;
        struct imsic_local_config *local;
        void __iomem **mmios_va = NULL;
@@ -878,6 +878,7 @@ int __init imsic_setup_state(struct fwnode_handle *fwnode, void *opaque)
        }
 
        /* Configure handlers for target CPUs */
+       global->nr_guest_files = BIT(global->guest_index_bits) - 1;
        for (i = 0; i < nr_parent_irqs; i++) {
                rc = imsic_get_parent_hartid(fwnode, i, &hartid);
                if (rc) {
@@ -918,6 +919,15 @@ int __init imsic_setup_state(struct fwnode_handle *fwnode, void *opaque)
                local->msi_pa = mmios[index].start + reloff;
                local->msi_va = mmios_va[index] + reloff;
 
+               /*
+                * KVM uses global->nr_guest_files to determine the available guest
+                * interrupt files on each CPU. Take the minimum number of guest
+                * interrupt files across all CPUs to avoid KVM incorrectly allocating
+                * an unexisted or unmapped guest interrupt file on some CPUs.
+                */
+               nr_guest_files = (resource_size(&mmios[index]) - reloff) / IMSIC_MMIO_PAGE_SZ - 1;
+               global->nr_guest_files = min(global->nr_guest_files, nr_guest_files);
+
                nr_handlers++;
        }
 
index 7f3ff5c5ea5356a970d528da22a84f68ac8ad457..4b348836de7adaa3790c31968b8536933a66bbfa 100644 (file)
@@ -68,6 +68,9 @@ struct imsic_global_config {
        /* Number of guest interrupt identities */
        u32                                     nr_guest_ids;
 
+       /* Number of guest interrupt files per core */
+       u32                                     nr_guest_files;
+
        /* Per-CPU IMSIC addresses */
        struct imsic_local_config __percpu      *local;
 };