csky: Add arch_show_interrupts for IPI interrupts
Here is the result: cat /proc/interrupts CPU0 CPU1 CPU2 CPU3 15: 1348 1299 952 1076 C-SKY SMP Intc 15 IPI Interrupt 16: 1203 1825 1598 1307 C-SKY SMP Intc 16 csky_mp_timer 43: 292 0 0 0 C-SKY SMP Intc 43 ttyS0 57: 106 0 0 0 C-SKY SMP Intc 57 virtio0 IPI0: 0 0 0 0 Empty interrupts IPI1: 19 41 45 27 Rescheduling interrupts IPI2: 1330 1259 908 1050 Function call interrupts IPI3: 0 0 0 0 Irq work interrupts Signed-off-by: Guo Ren <guoren@linux.alibaba.com> Cc: Arnd Bergmann <arnd@arndb.de>
This commit is contained in:
parent
2c81b07682
commit
e6169c4b44
|
@ -15,6 +15,7 @@
|
||||||
#include <linux/irq_work.h>
|
#include <linux/irq_work.h>
|
||||||
#include <linux/irqdomain.h>
|
#include <linux/irqdomain.h>
|
||||||
#include <linux/of.h>
|
#include <linux/of.h>
|
||||||
|
#include <linux/seq_file.h>
|
||||||
#include <linux/sched/task_stack.h>
|
#include <linux/sched/task_stack.h>
|
||||||
#include <linux/sched/mm.h>
|
#include <linux/sched/mm.h>
|
||||||
#include <linux/sched/hotplug.h>
|
#include <linux/sched/hotplug.h>
|
||||||
|
@ -27,11 +28,6 @@
|
||||||
#include <abi/fpu.h>
|
#include <abi/fpu.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
struct ipi_data_struct {
|
|
||||||
unsigned long bits ____cacheline_aligned;
|
|
||||||
};
|
|
||||||
static DEFINE_PER_CPU(struct ipi_data_struct, ipi_data);
|
|
||||||
|
|
||||||
enum ipi_message_type {
|
enum ipi_message_type {
|
||||||
IPI_EMPTY,
|
IPI_EMPTY,
|
||||||
IPI_RESCHEDULE,
|
IPI_RESCHEDULE,
|
||||||
|
@ -40,8 +36,16 @@ enum ipi_message_type {
|
||||||
IPI_MAX
|
IPI_MAX
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct ipi_data_struct {
|
||||||
|
unsigned long bits ____cacheline_aligned;
|
||||||
|
unsigned long stats[IPI_MAX] ____cacheline_aligned;
|
||||||
|
};
|
||||||
|
static DEFINE_PER_CPU(struct ipi_data_struct, ipi_data);
|
||||||
|
|
||||||
static irqreturn_t handle_ipi(int irq, void *dev)
|
static irqreturn_t handle_ipi(int irq, void *dev)
|
||||||
{
|
{
|
||||||
|
unsigned long *stats = this_cpu_ptr(&ipi_data)->stats;
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
unsigned long ops;
|
unsigned long ops;
|
||||||
|
|
||||||
|
@ -49,14 +53,20 @@ static irqreturn_t handle_ipi(int irq, void *dev)
|
||||||
if (ops == 0)
|
if (ops == 0)
|
||||||
return IRQ_HANDLED;
|
return IRQ_HANDLED;
|
||||||
|
|
||||||
if (ops & (1 << IPI_RESCHEDULE))
|
if (ops & (1 << IPI_RESCHEDULE)) {
|
||||||
|
stats[IPI_RESCHEDULE]++;
|
||||||
scheduler_ipi();
|
scheduler_ipi();
|
||||||
|
}
|
||||||
|
|
||||||
if (ops & (1 << IPI_CALL_FUNC))
|
if (ops & (1 << IPI_CALL_FUNC)) {
|
||||||
|
stats[IPI_CALL_FUNC]++;
|
||||||
generic_smp_call_function_interrupt();
|
generic_smp_call_function_interrupt();
|
||||||
|
}
|
||||||
|
|
||||||
if (ops & (1 << IPI_IRQ_WORK))
|
if (ops & (1 << IPI_IRQ_WORK)) {
|
||||||
|
stats[IPI_IRQ_WORK]++;
|
||||||
irq_work_run();
|
irq_work_run();
|
||||||
|
}
|
||||||
|
|
||||||
BUG_ON((ops >> IPI_MAX) != 0);
|
BUG_ON((ops >> IPI_MAX) != 0);
|
||||||
}
|
}
|
||||||
|
@ -88,6 +98,29 @@ send_ipi_message(const struct cpumask *to_whom, enum ipi_message_type operation)
|
||||||
send_arch_ipi(to_whom);
|
send_arch_ipi(to_whom);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static const char * const ipi_names[] = {
|
||||||
|
[IPI_EMPTY] = "Empty interrupts",
|
||||||
|
[IPI_RESCHEDULE] = "Rescheduling interrupts",
|
||||||
|
[IPI_CALL_FUNC] = "Function call interrupts",
|
||||||
|
[IPI_IRQ_WORK] = "Irq work interrupts",
|
||||||
|
};
|
||||||
|
|
||||||
|
int arch_show_interrupts(struct seq_file *p, int prec)
|
||||||
|
{
|
||||||
|
unsigned int cpu, i;
|
||||||
|
|
||||||
|
for (i = 0; i < IPI_MAX; i++) {
|
||||||
|
seq_printf(p, "%*s%u:%s", prec - 1, "IPI", i,
|
||||||
|
prec >= 4 ? " " : "");
|
||||||
|
for_each_online_cpu(cpu)
|
||||||
|
seq_printf(p, "%10lu ",
|
||||||
|
per_cpu_ptr(&ipi_data, cpu)->stats[i]);
|
||||||
|
seq_printf(p, " %s\n", ipi_names[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
void arch_send_call_function_ipi_mask(struct cpumask *mask)
|
void arch_send_call_function_ipi_mask(struct cpumask *mask)
|
||||||
{
|
{
|
||||||
send_ipi_message(mask, IPI_CALL_FUNC);
|
send_ipi_message(mask, IPI_CALL_FUNC);
|
||||||
|
|
Loading…
Reference in New Issue