diff options
author | Bjoern Brandenburg <bbb@mpi-sws.org> | 2015-08-09 07:18:45 -0400 |
---|---|---|
committer | Bjoern Brandenburg <bbb@mpi-sws.org> | 2017-05-26 17:12:22 -0400 |
commit | 3adfeae8db15fd5aa4027a906f7f4b1b609cb2f9 (patch) | |
tree | 70418df1ce83d270704f750afc1a74e7079bd0df | |
parent | 49e9c1480b1bb8cd89c9caf7850d37d33584c104 (diff) |
Integrate ft_irq_fired() with Linux
This patch hooks up Feather-Trace's ft_irq_fired() handler with
Linux's interrupt handling infrastructure.
IRQ tracing: ft_irq_fired() only if irq_enter() was skipped
On x86, the rescheduling IPI path already calls irq_enter(), which
calls ft_irq_fired(), so we don't have to do it again.
IRQ tracing: don't count softirqs
Only triggered by irq_exit(), which implies that we called
irq_enter(), which means that we already traced the current
hard IRQ.
-rw-r--r-- | include/linux/hardirq.h | 3 | ||||
-rw-r--r-- | include/litmus/trace_irq.h | 14 | ||||
-rw-r--r-- | kernel/sched/core.c | 7 |
3 files changed, 24 insertions, 0 deletions
diff --git a/include/linux/hardirq.h b/include/linux/hardirq.h index c683996110b1..7eec96ace9c7 100644 --- a/include/linux/hardirq.h +++ b/include/linux/hardirq.h | |||
@@ -7,6 +7,7 @@ | |||
7 | #include <linux/vtime.h> | 7 | #include <linux/vtime.h> |
8 | #include <asm/hardirq.h> | 8 | #include <asm/hardirq.h> |
9 | 9 | ||
10 | #include <litmus/trace_irq.h> | ||
10 | 11 | ||
11 | extern void synchronize_irq(unsigned int irq); | 12 | extern void synchronize_irq(unsigned int irq); |
12 | extern bool synchronize_hardirq(unsigned int irq); | 13 | extern bool synchronize_hardirq(unsigned int irq); |
@@ -37,6 +38,7 @@ extern void rcu_nmi_exit(void); | |||
37 | account_irq_enter_time(current); \ | 38 | account_irq_enter_time(current); \ |
38 | preempt_count_add(HARDIRQ_OFFSET); \ | 39 | preempt_count_add(HARDIRQ_OFFSET); \ |
39 | trace_hardirq_enter(); \ | 40 | trace_hardirq_enter(); \ |
41 | ft_irq_fired(); \ | ||
40 | } while (0) | 42 | } while (0) |
41 | 43 | ||
42 | /* | 44 | /* |
@@ -68,6 +70,7 @@ extern void irq_exit(void); | |||
68 | preempt_count_add(NMI_OFFSET + HARDIRQ_OFFSET); \ | 70 | preempt_count_add(NMI_OFFSET + HARDIRQ_OFFSET); \ |
69 | rcu_nmi_enter(); \ | 71 | rcu_nmi_enter(); \ |
70 | trace_hardirq_enter(); \ | 72 | trace_hardirq_enter(); \ |
73 | ft_irq_fired(); \ | ||
71 | } while (0) | 74 | } while (0) |
72 | 75 | ||
73 | #define nmi_exit() \ | 76 | #define nmi_exit() \ |
diff --git a/include/litmus/trace_irq.h b/include/litmus/trace_irq.h new file mode 100644 index 000000000000..0d0c042ba9c3 --- /dev/null +++ b/include/litmus/trace_irq.h | |||
@@ -0,0 +1,14 @@ | |||
1 | #ifndef _LITMUS_TRACE_IRQ_H_ | ||
2 | #define _LITMUS_TRACE_IRQ_H_ | ||
3 | |||
4 | #ifdef CONFIG_SCHED_OVERHEAD_TRACE | ||
5 | |||
6 | void ft_irq_fired(void); | ||
7 | |||
8 | #else | ||
9 | |||
10 | #define ft_irq_fired() /* nothing to do */ | ||
11 | |||
12 | #endif | ||
13 | |||
14 | #endif | ||
diff --git a/kernel/sched/core.c b/kernel/sched/core.c index 154fd689fe02..6440fe53d1a8 100644 --- a/kernel/sched/core.c +++ b/kernel/sched/core.c | |||
@@ -1806,7 +1806,14 @@ void scheduler_ipi(void) | |||
1806 | preempt_fold_need_resched(); | 1806 | preempt_fold_need_resched(); |
1807 | 1807 | ||
1808 | if (llist_empty(&this_rq()->wake_list) && !got_nohz_idle_kick()) | 1808 | if (llist_empty(&this_rq()->wake_list) && !got_nohz_idle_kick()) |
1809 | { | ||
1810 | #ifndef CONFIG_ARCH_CALLS_IRQ_ENTER_ON_RESCHED_IPI | ||
1811 | /* If we don't call irq_enter(), we need to triggger the IRQ | ||
1812 | * tracing manually. */ | ||
1813 | ft_irq_fired(); | ||
1814 | #endif | ||
1809 | return; | 1815 | return; |
1816 | } | ||
1810 | 1817 | ||
1811 | /* | 1818 | /* |
1812 | * Not all reschedule IPI handlers call irq_enter/irq_exit, since | 1819 | * Not all reschedule IPI handlers call irq_enter/irq_exit, since |