* [PATCH v2] s390/traps: Add exception statistics
@ 2026-06-17 12:33 Heiko Carstens
2026-06-17 14:37 ` Christian Borntraeger
0 siblings, 1 reply; 2+ messages in thread
From: Heiko Carstens @ 2026-06-17 12:33 UTC (permalink / raw)
To: Alexander Gordeev, Sven Schnelle, Vasily Gorbik,
Christian Borntraeger
Cc: linux-s390
From: Sven Schnelle <svens@linux.ibm.com>
Add a new debugfs file which displays the number of exceptions (program
checks) per CPU. This is helpful for debugging purposes.
The statistics are typically available at
/sys/kernel/debug/s390/exceptions.
[ hca@linux.ibm.com: Forward ported code, changed file location ]
Suggested-by: Christian Borntraeger <borntraeger@linux.ibm.com>
Signed-off-by: Sven Schnelle <svens@linux.ibm.com>
Signed-off-by: Heiko Carstens <hca@linux.ibm.com>
---
Notes:
v2:
- Fix / remove broken iterator handling as reported by Sashiko:
https://sashiko.dev/#/patchset/20260617113726.2079324-1-hca@linux.ibm.com?part=1
arch/s390/kernel/traps.c | 41 +++++++++++++++++++++++++++++++++++++++-
1 file changed, 40 insertions(+), 1 deletion(-)
diff --git a/arch/s390/kernel/traps.c b/arch/s390/kernel/traps.c
index 564403496a7c..fcd8c1122ef5 100644
--- a/arch/s390/kernel/traps.c
+++ b/arch/s390/kernel/traps.c
@@ -9,7 +9,9 @@
* Copyright (C) 1991, 1992 Linus Torvalds
*/
+#include <linux/capability.h>
#include <linux/cpufeature.h>
+#include <linux/debugfs.h>
#include <linux/kprobes.h>
#include <linux/kdebug.h>
#include <linux/randomize_kstack.h>
@@ -33,6 +35,12 @@
#include <asm/fault.h>
#include "entry.h"
+struct pgm_stat {
+ unsigned int count[128];
+};
+
+static DEFINE_PER_CPU_SHARED_ALIGNED(struct pgm_stat, pgm_stat);
+
static inline void __user *get_trap_ip(struct pt_regs *regs)
{
unsigned long address;
@@ -332,6 +340,7 @@ void noinstr __do_pgm_check(struct pt_regs *regs)
struct lowcore *lc = get_lowcore();
bool percpu_needs_fixup;
irqentry_state_t state;
+ struct pgm_stat *stat;
unsigned int trapnr;
union teid teid;
@@ -339,6 +348,10 @@ void noinstr __do_pgm_check(struct pt_regs *regs)
regs->int_code = lc->pgm_int_code;
regs->int_parm_long = teid.val;
regs->monitor_code = lc->monitor_code;
+
+ trapnr = regs->int_code & PGM_INT_CODE_MASK;
+ stat = this_cpu_ptr(&pgm_stat);
+ stat->count[trapnr]++;
/*
* In case of a guest fault, short-circuit the fault handler and return.
* This way the sie64a() function will return 0; fault address and
@@ -383,7 +396,6 @@ void noinstr __do_pgm_check(struct pt_regs *regs)
if (!irqs_disabled_flags(regs->psw.mask))
trace_hardirqs_on();
__arch_local_irq_ssm(regs->psw.mask & ~PSW_MASK_PER);
- trapnr = regs->int_code & PGM_INT_CODE_MASK;
if (trapnr)
pgm_check_table[trapnr](regs);
out:
@@ -393,6 +405,33 @@ void noinstr __do_pgm_check(struct pt_regs *regs)
percpu_exit(regs, percpu_needs_fixup);
}
+static int pgm_check_stat_show(struct seq_file *p, void *v)
+{
+ int i, cpu;
+
+ cpus_read_lock();
+ seq_puts(p, " ");
+ for_each_online_cpu(cpu)
+ seq_printf(p, "CPU%-8d", cpu);
+ seq_putc(p, '\n');
+ for (i = 0; i < 128; i++) {
+ seq_printf(p, "%02x: ", i);
+ for_each_online_cpu(cpu)
+ seq_printf(p, "%10u ", per_cpu(pgm_stat, cpu).count[i]);
+ seq_putc(p, '\n');
+ }
+ cpus_read_unlock();
+ return 0;
+}
+DEFINE_SHOW_ATTRIBUTE(pgm_check_stat);
+
+static int __init debugfs_pgm_check_init(void)
+{
+ debugfs_create_file("exceptions", 0400, arch_debugfs_dir, NULL, &pgm_check_stat_fops);
+ return 0;
+}
+late_initcall(debugfs_pgm_check_init);
+
/*
* The program check table contains exactly 128 (0x00-0x7f) entries. Each
* line defines the function to be called corresponding to the program check
--
2.53.0
^ permalink raw reply related [flat|nested] 2+ messages in thread
* Re: [PATCH v2] s390/traps: Add exception statistics
2026-06-17 12:33 [PATCH v2] s390/traps: Add exception statistics Heiko Carstens
@ 2026-06-17 14:37 ` Christian Borntraeger
0 siblings, 0 replies; 2+ messages in thread
From: Christian Borntraeger @ 2026-06-17 14:37 UTC (permalink / raw)
To: Heiko Carstens, Alexander Gordeev, Sven Schnelle, Vasily Gorbik
Cc: linux-s390
Am 17.06.26 um 14:33 schrieb Heiko Carstens:
> From: Sven Schnelle <svens@linux.ibm.com>
>
> Add a new debugfs file which displays the number of exceptions (program
> checks) per CPU. This is helpful for debugging purposes.
>
> The statistics are typically available at
> /sys/kernel/debug/s390/exceptions.
>
> [ hca@linux.ibm.com: Forward ported code, changed file location ]
>
> Suggested-by: Christian Borntraeger <borntraeger@linux.ibm.com>
> Signed-off-by: Sven Schnelle <svens@linux.ibm.com>
> Signed-off-by: Heiko Carstens <hca@linux.ibm.com>
Looks good to me
Tested-by: Christian Borntraeger <borntraeger@linux.ibm.com>
> ---
>
> Notes:
> v2:
> - Fix / remove broken iterator handling as reported by Sashiko:
> https://sashiko.dev/#/patchset/20260617113726.2079324-1-hca@linux.ibm.com?part=1
>
> arch/s390/kernel/traps.c | 41 +++++++++++++++++++++++++++++++++++++++-
> 1 file changed, 40 insertions(+), 1 deletion(-)
>
> diff --git a/arch/s390/kernel/traps.c b/arch/s390/kernel/traps.c
> index 564403496a7c..fcd8c1122ef5 100644
> --- a/arch/s390/kernel/traps.c
> +++ b/arch/s390/kernel/traps.c
> @@ -9,7 +9,9 @@
> * Copyright (C) 1991, 1992 Linus Torvalds
> */
>
> +#include <linux/capability.h>
> #include <linux/cpufeature.h>
> +#include <linux/debugfs.h>
> #include <linux/kprobes.h>
> #include <linux/kdebug.h>
> #include <linux/randomize_kstack.h>
> @@ -33,6 +35,12 @@
> #include <asm/fault.h>
> #include "entry.h"
>
> +struct pgm_stat {
> + unsigned int count[128];
> +};
> +
> +static DEFINE_PER_CPU_SHARED_ALIGNED(struct pgm_stat, pgm_stat);
> +
> static inline void __user *get_trap_ip(struct pt_regs *regs)
> {
> unsigned long address;
> @@ -332,6 +340,7 @@ void noinstr __do_pgm_check(struct pt_regs *regs)
> struct lowcore *lc = get_lowcore();
> bool percpu_needs_fixup;
> irqentry_state_t state;
> + struct pgm_stat *stat;
> unsigned int trapnr;
> union teid teid;
>
> @@ -339,6 +348,10 @@ void noinstr __do_pgm_check(struct pt_regs *regs)
> regs->int_code = lc->pgm_int_code;
> regs->int_parm_long = teid.val;
> regs->monitor_code = lc->monitor_code;
> +
> + trapnr = regs->int_code & PGM_INT_CODE_MASK;
> + stat = this_cpu_ptr(&pgm_stat);
> + stat->count[trapnr]++;
> /*
> * In case of a guest fault, short-circuit the fault handler and return.
> * This way the sie64a() function will return 0; fault address and
> @@ -383,7 +396,6 @@ void noinstr __do_pgm_check(struct pt_regs *regs)
> if (!irqs_disabled_flags(regs->psw.mask))
> trace_hardirqs_on();
> __arch_local_irq_ssm(regs->psw.mask & ~PSW_MASK_PER);
> - trapnr = regs->int_code & PGM_INT_CODE_MASK;
> if (trapnr)
> pgm_check_table[trapnr](regs);
> out:
> @@ -393,6 +405,33 @@ void noinstr __do_pgm_check(struct pt_regs *regs)
> percpu_exit(regs, percpu_needs_fixup);
> }
>
> +static int pgm_check_stat_show(struct seq_file *p, void *v)
> +{
> + int i, cpu;
> +
> + cpus_read_lock();
> + seq_puts(p, " ");
> + for_each_online_cpu(cpu)
> + seq_printf(p, "CPU%-8d", cpu);
> + seq_putc(p, '\n');
> + for (i = 0; i < 128; i++) {
> + seq_printf(p, "%02x: ", i);
> + for_each_online_cpu(cpu)
> + seq_printf(p, "%10u ", per_cpu(pgm_stat, cpu).count[i]);
> + seq_putc(p, '\n');
> + }
> + cpus_read_unlock();
> + return 0;
> +}
> +DEFINE_SHOW_ATTRIBUTE(pgm_check_stat);
> +
> +static int __init debugfs_pgm_check_init(void)
> +{
> + debugfs_create_file("exceptions", 0400, arch_debugfs_dir, NULL, &pgm_check_stat_fops);
> + return 0;
> +}
> +late_initcall(debugfs_pgm_check_init);
> +
> /*
> * The program check table contains exactly 128 (0x00-0x7f) entries. Each
> * line defines the function to be called corresponding to the program check
^ permalink raw reply [flat|nested] 2+ messages in thread
end of thread, other threads:[~2026-06-17 14:38 UTC | newest]
Thread overview: 2+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2026-06-17 12:33 [PATCH v2] s390/traps: Add exception statistics Heiko Carstens
2026-06-17 14:37 ` Christian Borntraeger
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.