Commit 25a4a015 authored by Bitao Hu's avatar Bitao Hu Committed by Thomas Gleixner

genirq: Avoid summation loops for /proc/interrupts

show_interrupts() unconditionally accumulates the per CPU interrupt
statistics to determine whether an interrupt was ever raised.

This can be avoided for all interrupts which are not strictly per CPU
and not of type NMI because those interrupts provide already an
accumulated counter. The required logic is already implemented in
kstat_irqs().

Split the inner access logic out of kstat_irqs() and use it for
kstat_irqs() and show_interrupts() to avoid the accumulation loop
when possible.
Originally-by: default avatarThomas Gleixner <tglx@linutronix.de>
Signed-off-by: default avatarBitao Hu <yaoma@linux.alibaba.com>
Signed-off-by: default avatarThomas Gleixner <tglx@linutronix.de>
Reviewed-by: default avatarLiu Song <liusong@linux.alibaba.com>
Reviewed-by: default avatarDouglas Anderson <dianders@chromium.org>
Link: https://lore.kernel.org/r/20240411074134.30922-4-yaoma@linux.alibaba.com
parent 99cf63c5
...@@ -98,6 +98,8 @@ extern void mask_irq(struct irq_desc *desc); ...@@ -98,6 +98,8 @@ extern void mask_irq(struct irq_desc *desc);
extern void unmask_irq(struct irq_desc *desc); extern void unmask_irq(struct irq_desc *desc);
extern void unmask_threaded_irq(struct irq_desc *desc); extern void unmask_threaded_irq(struct irq_desc *desc);
extern unsigned int kstat_irqs_desc(struct irq_desc *desc, const struct cpumask *cpumask);
#ifdef CONFIG_SPARSE_IRQ #ifdef CONFIG_SPARSE_IRQ
static inline void irq_mark_irq(unsigned int irq) { } static inline void irq_mark_irq(unsigned int irq) { }
#else #else
......
...@@ -976,24 +976,30 @@ static bool irq_is_nmi(struct irq_desc *desc) ...@@ -976,24 +976,30 @@ static bool irq_is_nmi(struct irq_desc *desc)
return desc->istate & IRQS_NMI; return desc->istate & IRQS_NMI;
} }
static unsigned int kstat_irqs(unsigned int irq) unsigned int kstat_irqs_desc(struct irq_desc *desc, const struct cpumask *cpumask)
{ {
struct irq_desc *desc = irq_to_desc(irq);
unsigned int sum = 0; unsigned int sum = 0;
int cpu; int cpu;
if (!desc || !desc->kstat_irqs)
return 0;
if (!irq_settings_is_per_cpu_devid(desc) && if (!irq_settings_is_per_cpu_devid(desc) &&
!irq_settings_is_per_cpu(desc) && !irq_settings_is_per_cpu(desc) &&
!irq_is_nmi(desc)) !irq_is_nmi(desc))
return data_race(desc->tot_count); return data_race(desc->tot_count);
for_each_possible_cpu(cpu) for_each_cpu(cpu, cpumask)
sum += data_race(per_cpu(desc->kstat_irqs->cnt, cpu)); sum += data_race(per_cpu(desc->kstat_irqs->cnt, cpu));
return sum; return sum;
} }
static unsigned int kstat_irqs(unsigned int irq)
{
struct irq_desc *desc = irq_to_desc(irq);
if (!desc || !desc->kstat_irqs)
return 0;
return kstat_irqs_desc(desc, cpu_possible_mask);
}
#ifdef CONFIG_GENERIC_IRQ_STAT_SNAPSHOT #ifdef CONFIG_GENERIC_IRQ_STAT_SNAPSHOT
void kstat_snapshot_irqs(void) void kstat_snapshot_irqs(void)
......
...@@ -488,10 +488,8 @@ int show_interrupts(struct seq_file *p, void *v) ...@@ -488,10 +488,8 @@ int show_interrupts(struct seq_file *p, void *v)
if (!desc || irq_settings_is_hidden(desc)) if (!desc || irq_settings_is_hidden(desc))
goto outsparse; goto outsparse;
if (desc->kstat_irqs) { if (desc->kstat_irqs)
for_each_online_cpu(j) any_count = kstat_irqs_desc(desc, cpu_online_mask);
any_count |= data_race(per_cpu(desc->kstat_irqs->cnt, j));
}
if ((!desc->action || irq_desc_is_chained(desc)) && !any_count) if ((!desc->action || irq_desc_is_chained(desc)) && !any_count)
goto outsparse; goto outsparse;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment