Re: [PATCH v6 05/29] x86/apic/vector: Do not allocate vectors for NMIs

From: Thomas Gleixner
Date: Fri May 06 2022 - 17:12:33 EST


On Thu, May 05 2022 at 16:59, Ricardo Neri wrote:
> Vectors are meaningless when allocating IRQs with NMI as the delivery
> mode.

Vectors are not meaningless. NMI has a fixed vector.

The point is that for a fixed vector there is no vector management
required.

Can you spot the difference?

> In such case, skip the reservation of IRQ vectors. Do it in the lowest-
> level functions where the actual IRQ reservation takes place.
>
> Since NMIs target specific CPUs, keep the functionality to find the best
> CPU.

Again. What for?

> + if (apicd->hw_irq_cfg.delivery_mode == APIC_DELIVERY_MODE_NMI) {
> + cpu = irq_matrix_find_best_cpu(vector_matrix, dest);
> + apicd->cpu = cpu;
> + vector = 0;
> + goto no_vector;
> + }

Why would a NMI ever end up in this code? There is no vector management
required and this find cpu exercise is pointless.

> vector = irq_matrix_alloc(vector_matrix, dest, resvd, &cpu);
> trace_vector_alloc(irqd->irq, vector, resvd, vector);
> if (vector < 0)
> return vector;
> apic_update_vector(irqd, vector, cpu);
> +
> +no_vector:
> apic_update_irq_cfg(irqd, vector, cpu);
>
> return 0;
> @@ -321,12 +330,22 @@ assign_managed_vector(struct irq_data *irqd, const struct cpumask *dest)
> /* set_affinity might call here for nothing */
> if (apicd->vector && cpumask_test_cpu(apicd->cpu, vector_searchmask))
> return 0;
> +
> + if (apicd->hw_irq_cfg.delivery_mode == APIC_DELIVERY_MODE_NMI) {
> + cpu = irq_matrix_find_best_cpu_managed(vector_matrix, dest);
> + apicd->cpu = cpu;
> + vector = 0;
> + goto no_vector;
> + }

This code can never be reached for a NMI delivery. If so, then it's a
bug.

This all is special purpose for that particular HPET NMI watchdog use
case and we are not exposing this to anything else at all.

So why are you sprinkling this NMI nonsense all over the place? Just
because? There are well defined entry points to all of this where this
can be fenced off.

If at some day the hardware people get their act together and provide a
proper vector space for NMIs then we have to revisit that, but then
there will be a separate NMI vector management too.

What you want is the below which also covers the next patch. Please keep
an eye on the comments I added/modified.

Thanks,

tglx
---
--- a/arch/x86/kernel/apic/vector.c
+++ b/arch/x86/kernel/apic/vector.c
@@ -42,6 +42,7 @@ EXPORT_SYMBOL_GPL(x86_vector_domain);
static DEFINE_RAW_SPINLOCK(vector_lock);
static cpumask_var_t vector_searchmask;
static struct irq_chip lapic_controller;
+static struct irq_chip lapic_nmi_controller;
static struct irq_matrix *vector_matrix;
#ifdef CONFIG_SMP
static DEFINE_PER_CPU(struct hlist_head, cleanup_list);
@@ -451,6 +452,10 @@ static int x86_vector_activate(struct ir
trace_vector_activate(irqd->irq, apicd->is_managed,
apicd->can_reserve, reserve);

+ /* NMI has a fixed vector. No vector management required */
+ if (apicd->hw_irq_cfg.delivery_mode == APIC_DELIVERY_MODE_NMI)
+ return 0;
+
raw_spin_lock_irqsave(&vector_lock, flags);
if (!apicd->can_reserve && !apicd->is_managed)
assign_irq_vector_any_locked(irqd);
@@ -472,6 +477,10 @@ static void vector_free_reserved_and_man
trace_vector_teardown(irqd->irq, apicd->is_managed,
apicd->has_reserved);

+ /* NMI has a fixed vector. No vector management required */
+ if (apicd->hw_irq_cfg.delivery_mode == APIC_DELIVERY_MODE_NMI)
+ return;
+
if (apicd->has_reserved)
irq_matrix_remove_reserved(vector_matrix);
if (apicd->is_managed)
@@ -568,6 +577,24 @@ static int x86_vector_alloc_irqs(struct
irqd->hwirq = virq + i;
irqd_set_single_target(irqd);

+ if (info->flags & X86_IRQ_ALLOC_AS_NMI) {
+ /*
+ * NMIs have a fixed vector and need their own
+ * interrupt chip so nothing can end up in the
+ * regular local APIC management code except the
+ * MSI message composing callback.
+ */
+ irqd->chip = &lapic_nmi_controller;
+ /*
+ * Don't allow affinity setting attempts for NMIs.
+ * This cannot work with the regular affinity
+ * mechanisms and for the intended HPET NMI
+ * watchdog use case it's not required.
+ */
+ irqd_set_no_balance(irqd);
+ continue;
+ }
+
/*
* Prevent that any of these interrupts is invoked in
* non interrupt context via e.g. generic_handle_irq()
@@ -921,6 +948,11 @@ static struct irq_chip lapic_controller
.irq_retrigger = apic_retrigger_irq,
};

+static struct irq_chip lapic_nmi_controller = {
+ .name = "APIC-NMI",
+ .irq_compose_msi_msg = x86_vector_msi_compose_msg,
+};
+
#ifdef CONFIG_SMP

static void free_moved_vector(struct apic_chip_data *apicd)
--- a/include/linux/irq.h
+++ b/include/linux/irq.h
@@ -261,6 +261,11 @@ static inline bool irqd_is_per_cpu(struc
return __irqd_to_state(d) & IRQD_PER_CPU;
}

+static inline void irqd_set_no_balance(struct irq_data *d)
+{
+ __irqd_to_state(d) |= IRQD_NO_BALANCING;
+}
+
static inline bool irqd_can_balance(struct irq_data *d)
{
return !(__irqd_to_state(d) & (IRQD_PER_CPU | IRQD_NO_BALANCING));