Re: [patch V4 06/15] x86/irq: Move IOAPIC misrouted and PIC/APIC error counts into irq_stats
From: Radu Rendec
Date: Tue Mar 31 2026 - 17:46:52 EST
On Tue, 2026-03-31 at 09:25 +0200, Thomas Gleixner wrote:
> From: Thomas Gleixner <tglx@xxxxxxxxxx>
>
> The special treatment of these counts is just adding extra code for no real
> value. The irq_stats mechanism allows to suppress output of counters, which
> should never happen by default and provides a mechanism to enable them for
> the rare case that they occur.
>
> Move the IOAPIC misrouted and the PIC/APIC error counts into irq_stats,
> mark them suppressed by default and update the sites which increment them.
>
> This changes the output format of 'ERR' and 'MIS' in case there are events
> to the regular per CPU display format and otherwise suppresses them
> completely.
>
> As a side effect this removes the arch_cpu_stat() mechanism from proc/stat
> which was only there to account for the error interrupts on x86 and missed
> to take the misrouted ones into account.
>
> Signed-off-by: Thomas Gleixner <tglx@xxxxxxxxxx>
> ---
> V3: New patch
> ---
> arch/x86/include/asm/hardirq.h | 7 ++++---
> arch/x86/include/asm/hw_irq.h | 4 ----
> arch/x86/kernel/apic/apic.c | 2 +-
> arch/x86/kernel/apic/io_apic.c | 4 +---
> arch/x86/kernel/i8259.c | 2 +-
> arch/x86/kernel/irq.c | 16 ++++------------
> fs/proc/stat.c | 4 ----
> 7 files changed, 11 insertions(+), 28 deletions(-)
> --- a/arch/x86/include/asm/hardirq.h
> +++ b/arch/x86/include/asm/hardirq.h
> @@ -50,6 +50,10 @@ enum irq_stat_counts {
> #ifdef CONFIG_X86_POSTED_MSI
> IRQ_COUNT_POSTED_MSI_NOTIFICATION,
> #endif
> + IRQ_COUNT_PIC_APIC_ERROR,
> +#ifdef CONFIG_X86_IO_APIC
> + IRQ_COUNT_IOAPIC_MISROUTED,
> +#endif
> IRQ_COUNT_MAX,
> };
>
> @@ -81,9 +85,6 @@ extern void ack_bad_irq(unsigned int irq
> #ifdef CONFIG_PROC_FS
> extern u64 arch_irq_stat_cpu(unsigned int cpu);
> #define arch_irq_stat_cpu arch_irq_stat_cpu
> -
> -extern u64 arch_irq_stat(void);
> -#define arch_irq_stat arch_irq_stat
> #endif
>
> DECLARE_PER_CPU_CACHE_HOT(u16, __softirq_pending);
> --- a/arch/x86/include/asm/hw_irq.h
> +++ b/arch/x86/include/asm/hw_irq.h
> @@ -110,10 +110,6 @@ static inline void lock_vector_lock(void
> static inline void unlock_vector_lock(void) {}
> #endif
>
> -/* Statistics */
> -extern atomic_t irq_err_count;
> -extern atomic_t irq_mis_count;
> -
> extern void elcr_set_level_irq(unsigned int irq);
>
> extern char irq_entries_start[];
> --- a/arch/x86/kernel/apic/apic.c
> +++ b/arch/x86/kernel/apic/apic.c
> @@ -2180,7 +2180,7 @@ DEFINE_IDTENTRY_SYSVEC(sysvec_error_inte
> apic_write(APIC_ESR, 0);
> v = apic_read(APIC_ESR);
> apic_eoi();
> - atomic_inc(&irq_err_count);
> + irq_stat_inc_and_enable(IRQ_COUNT_PIC_APIC_ERROR);
>
> apic_pr_debug("APIC error on CPU%d: %02x", smp_processor_id(), v);
>
> --- a/arch/x86/kernel/apic/io_apic.c
> +++ b/arch/x86/kernel/apic/io_apic.c
> @@ -1575,8 +1575,6 @@ static unsigned int startup_ioapic_irq(s
> return was_pending;
> }
>
> -atomic_t irq_mis_count;
> -
> #ifdef CONFIG_GENERIC_PENDING_IRQ
> static bool io_apic_level_ack_pending(struct mp_chip_data *data)
> {
> @@ -1713,7 +1711,7 @@ static void ioapic_ack_level(struct irq_
> * at the cpu.
> */
> if (!(v & (1 << (i & 0x1f)))) {
> - atomic_inc(&irq_mis_count);
> + irq_stat_inc_and_enable(IRQ_COUNT_IOAPIC_MISROUTED);
> eoi_ioapic_pin(cfg->vector, irq_data->chip_data);
> }
>
> --- a/arch/x86/kernel/i8259.c
> +++ b/arch/x86/kernel/i8259.c
> @@ -214,7 +214,7 @@ static void mask_and_ack_8259A(struct ir
> "spurious 8259A interrupt: IRQ%d.\n", irq);
> spurious_irq_mask |= irqmask;
> }
> - atomic_inc(&irq_err_count);
> + irq_stat_inc_and_enable(IRQ_COUNT_PIC_APIC_ERROR);
> /*
> * Theoretically we do not have to handle this IRQ,
> * but in Linux this does not cause problems and is
> --- a/arch/x86/kernel/irq.c
> +++ b/arch/x86/kernel/irq.c
> @@ -39,8 +39,6 @@ EXPORT_PER_CPU_SYMBOL(__softirq_pending)
>
> DEFINE_PER_CPU_CACHE_HOT(struct irq_stack *, hardirq_stack_ptr);
>
> -atomic_t irq_err_count;
> -
> /*
> * 'what should we do if we get a hw irq event on an illegal vector'.
> * each architecture has to answer this themselves.
> @@ -124,6 +122,10 @@ static const struct irq_stat_info irq_st
> #ifdef CONFIG_X86_POSTED_MSI
> ISS(POSTED_MSI_NOTIFICATION, "PMN", " Posted MSI notification event\n"),
> #endif
> + IDS(PIC_APIC_ERROR, "ERR", " PIC/APIC error interrupts\n"),
> +#ifdef CONFIG_X86_IO_APIC
> + IDS(IOAPIC_MISROUTED, "MIS", " Misrouted IO/APIC interrupts\n"),
> +#endif
> };
>
> static DECLARE_BITMAP(irq_stat_count_show, IRQ_COUNT_MAX) __read_mostly;
> @@ -183,10 +185,6 @@ int arch_show_interrupts(struct seq_file
> irq_proc_emit_counts(p, &irq_stat.counts[i]);
> seq_puts(p, info->text);
> }
> -
> - seq_printf(p, "%*s: %10u\n", prec, "ERR", atomic_read(&irq_err_count));
> - if (IS_ENABLED(CONFIG_X86_IO_APIC))
> - seq_printf(p, "%*s: %10u\n", prec, "MIS", atomic_read(&irq_mis_count));
> return 0;
> }
>
> @@ -202,12 +200,6 @@ u64 arch_irq_stat_cpu(unsigned int cpu)
> sum += p->counts[i];
> return sum;
> }
> -
> -u64 arch_irq_stat(void)
> -{
> - u64 sum = atomic_read(&irq_err_count);
> - return sum;
> -}
> #endif /* CONFIG_PROC_FS */
>
> static __always_inline void handle_irq(struct irq_desc *desc,
> --- a/fs/proc/stat.c
> +++ b/fs/proc/stat.c
> @@ -18,9 +18,6 @@
> #ifndef arch_irq_stat_cpu
> #define arch_irq_stat_cpu(cpu) 0
> #endif
> -#ifndef arch_irq_stat
> -#define arch_irq_stat() 0
> -#endif
>
> u64 get_idle_time(struct kernel_cpustat *kcs, int cpu)
> {
> @@ -122,7 +119,6 @@ static int show_stat(struct seq_file *p,
> sum_softirq += softirq_stat;
> }
> }
> - sum += arch_irq_stat();
>
> seq_put_decimal_ull(p, "cpu ", nsec_to_clock_t(user));
> seq_put_decimal_ull(p, " ", nsec_to_clock_t(nice));
Reviewed-by: Radu Rendec <radu@xxxxxxxxxx>