tracing/function-return-tracer: add the overrun field
authorFrederic Weisbecker <fweisbec@gmail.com>
Mon, 17 Nov 2008 02:22:41 +0000 (03:22 +0100)
committerIngo Molnar <mingo@elte.hu>
Tue, 18 Nov 2008 10:11:00 +0000 (11:11 +0100)
Impact: help to find the better depth of trace

We decided to arbitrary define the depth of function return trace as
"20". Perhaps this is not enough. To help finding an optimal depth, we
measure now the overrun: the number of functions that have been missed
for the current thread. By default this is not displayed, we have to
do set a particular flag on the return tracer: echo overrun >
/debug/tracing/trace_options And the overrun will be printed on the
right.

As the trace shows below, the current 20 depth is not enough.

update_wall_time+0x37f/0x8c0 -> update_xtime_cache (345 ns) (Overruns: 2838)
update_wall_time+0x384/0x8c0 -> clocksource_get_next (1141 ns) (Overruns: 2838)
do_timer+0x23/0x100 -> update_wall_time (3882 ns) (Overruns: 2838)
tick_do_update_jiffies64+0xbf/0x160 -> do_timer (5339 ns) (Overruns: 2838)
tick_sched_timer+0x6a/0xf0 -> tick_do_update_jiffies64 (7209 ns) (Overruns: 2838)
vgacon_set_cursor_size+0x98/0x120 -> native_io_delay (2613 ns) (Overruns: 274)
vgacon_cursor+0x16e/0x1d0 -> vgacon_set_cursor_size (33151 ns) (Overruns: 274)
set_cursor+0x5f/0x80 -> vgacon_cursor (36432 ns) (Overruns: 274)
con_flush_chars+0x34/0x40 -> set_cursor (38790 ns) (Overruns: 274)
release_console_sem+0x1ec/0x230 -> up (721 ns) (Overruns: 274)
release_console_sem+0x225/0x230 -> wake_up_klogd (316 ns) (Overruns: 274)
con_flush_chars+0x39/0x40 -> release_console_sem (2996 ns) (Overruns: 274)
con_write+0x22/0x30 -> con_flush_chars (46067 ns) (Overruns: 274)
n_tty_write+0x1cc/0x360 -> con_write (292670 ns) (Overruns: 274)
smp_apic_timer_interrupt+0x2a/0x90 -> native_apic_mem_write (330 ns) (Overruns: 274)
irq_enter+0x17/0x70 -> idle_cpu (413 ns) (Overruns: 274)
smp_apic_timer_interrupt+0x2f/0x90 -> irq_enter (1525 ns) (Overruns: 274)
ktime_get_ts+0x40/0x70 -> getnstimeofday (465 ns) (Overruns: 274)
ktime_get_ts+0x60/0x70 -> set_normalized_timespec (436 ns) (Overruns: 274)
ktime_get+0x16/0x30 -> ktime_get_ts (2501 ns) (Overruns: 274)
hrtimer_interrupt+0x77/0x1a0 -> ktime_get (3439 ns) (Overruns: 274)

Signed-off-by: Frederic Weisbecker <fweisbec@gmail.com>
Acked-by: Steven Rostedt <rostedt@goodmis.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
arch/x86/include/asm/thread_info.h
arch/x86/kernel/ftrace.c
include/linux/ftrace.h
include/linux/sched.h
kernel/trace/trace.c
kernel/trace/trace.h
kernel/trace/trace_functions_return.c

index a71158369fd46524cc0551bf3ff3dcc5f16c6b99..e90e81ef6ab9ef7889fb35ac2d14a440db149faf 100644 (file)
@@ -21,6 +21,7 @@ struct task_struct;
 struct exec_domain;
 #include <asm/processor.h>
 #include <asm/ftrace.h>
+#include <asm/atomic.h>
 
 struct thread_info {
        struct task_struct      *task;          /* main task structure */
@@ -45,6 +46,11 @@ struct thread_info {
        int             curr_ret_stack;
        /* Stack of return addresses for return function tracing */
        struct ftrace_ret_stack ret_stack[FTRACE_RET_STACK_SIZE];
+       /*
+        * Number of functions that haven't been traced
+        * because of depth overrun.
+        */
+       atomic_t        trace_overrun;
 #endif
 };
 
@@ -61,6 +67,7 @@ struct thread_info {
                .fn = do_no_restart_syscall,    \
        },                                      \
        .curr_ret_stack = -1,\
+       .trace_overrun  = ATOMIC_INIT(0)        \
 }
 #else
 #define INIT_THREAD_INFO(tsk)                  \
index 924153edd97331c1b244ec63446f0978e43f5fb1..356bb1eb6e9a5271d91224a872856a868fc59780 100644 (file)
@@ -353,8 +353,10 @@ static int push_return_trace(unsigned long ret, unsigned long long time,
        struct thread_info *ti = current_thread_info();
 
        /* The return trace stack is full */
-       if (ti->curr_ret_stack == FTRACE_RET_STACK_SIZE - 1)
+       if (ti->curr_ret_stack == FTRACE_RET_STACK_SIZE - 1) {
+               atomic_inc(&ti->trace_overrun);
                return -EBUSY;
+       }
 
        index = ++ti->curr_ret_stack;
        barrier();
@@ -367,7 +369,7 @@ static int push_return_trace(unsigned long ret, unsigned long long time,
 
 /* Retrieve a function return address to the trace stack on thread info.*/
 static void pop_return_trace(unsigned long *ret, unsigned long long *time,
-                               unsigned long *func)
+                               unsigned long *func, unsigned long *overrun)
 {
        int index;
 
@@ -376,6 +378,7 @@ static void pop_return_trace(unsigned long *ret, unsigned long long *time,
        *ret = ti->ret_stack[index].ret;
        *func = ti->ret_stack[index].func;
        *time = ti->ret_stack[index].calltime;
+       *overrun = atomic_read(&ti->trace_overrun);
        ti->curr_ret_stack--;
 }
 
@@ -386,7 +389,8 @@ static void pop_return_trace(unsigned long *ret, unsigned long long *time,
 unsigned long ftrace_return_to_handler(void)
 {
        struct ftrace_retfunc trace;
-       pop_return_trace(&trace.ret, &trace.calltime, &trace.func);
+       pop_return_trace(&trace.ret, &trace.calltime, &trace.func,
+                       &trace.overrun);
        trace.rettime = cpu_clock(raw_smp_processor_id());
        ftrace_function_return(&trace);
 
index f1af1aab00e6ffaa3e463ffe3522df63918953d8..f7ba4ea5e128dca40746569fb580965cd12f9faf 100644 (file)
@@ -318,6 +318,8 @@ struct ftrace_retfunc {
        unsigned long func; /* Current function */
        unsigned long long calltime;
        unsigned long long rettime;
+       /* Number of functions that overran the depth limit for current task */
+       unsigned long overrun;
 };
 
 #ifdef CONFIG_FUNCTION_RET_TRACER
index 61c8cc36028ae37d5c5c1c9bc9ac2292d3493f57..c8e0db46420674184abd181666034fa6da1cbefc 100644 (file)
@@ -2016,6 +2016,7 @@ static inline void setup_thread_stack(struct task_struct *p, struct task_struct
         * used.
         */
        task_thread_info(p)->curr_ret_stack = -1;
+       atomic_set(&task_thread_info(p)->trace_overrun, 0);
 #endif
 }
 
index 9531fddcfb8dba67dc01755dc7f3b3082077ffdf..e97c29a6e7b09cb6850dfb8cb1caaab9c24f65bc 100644 (file)
@@ -853,6 +853,7 @@ static void __trace_function_return(struct trace_array *tr,
        entry->parent_ip        = trace->ret;
        entry->rettime          = trace->rettime;
        entry->calltime         = trace->calltime;
+       entry->overrun          = trace->overrun;
        ring_buffer_unlock_commit(global_trace.buffer, event, irq_flags);
 }
 #endif
index 9d22618bf99f4ab72ef95a2bf154b4cc614d2288..2cb12fd98f6b395dbdc1083d01a96b527633f858 100644 (file)
@@ -60,6 +60,7 @@ struct ftrace_ret_entry {
        unsigned long           parent_ip;
        unsigned long long      calltime;
        unsigned long long      rettime;
+       unsigned long           overrun;
 };
 extern struct tracer boot_tracer;
 
index a68564af022bff3dafc11d189a8eccfce4c026bd..e00d64509c9c42bc2ef92fca9aa7fcbc9187c702 100644 (file)
 #include "trace.h"
 
 
+#define TRACE_RETURN_PRINT_OVERRUN     0x1
+static struct tracer_opt trace_opts[] = {
+       /* Display overruns or not */
+       { TRACER_OPT(overrun, TRACE_RETURN_PRINT_OVERRUN) },
+       { } /* Empty entry */
+};
+
+static struct tracer_flags tracer_flags = {
+       .val = 0, /* Don't display overruns by default */
+       .opts = trace_opts
+};
+
+
 static int return_trace_init(struct trace_array *tr)
 {
        int cpu;
@@ -42,26 +55,39 @@ print_return_function(struct trace_iterator *iter)
                ret = trace_seq_printf(s, "%pF -> ", (void *)field->parent_ip);
                if (!ret)
                        return TRACE_TYPE_PARTIAL_LINE;
+
                ret = seq_print_ip_sym(s, field->ip,
                                        trace_flags & TRACE_ITER_SYM_MASK);
                if (!ret)
                        return TRACE_TYPE_PARTIAL_LINE;
-               ret = trace_seq_printf(s, " (%llu ns)\n",
+
+               ret = trace_seq_printf(s, " (%llu ns)",
                                        field->rettime - field->calltime);
                if (!ret)
                        return TRACE_TYPE_PARTIAL_LINE;
-               else
-                       return TRACE_TYPE_HANDLED;
+
+               if (tracer_flags.val & TRACE_RETURN_PRINT_OVERRUN) {
+                       ret = trace_seq_printf(s, " (Overruns: %lu)",
+                                               field->overrun);
+                       if (!ret)
+                               return TRACE_TYPE_PARTIAL_LINE;
+               }
+
+               ret = trace_seq_printf(s, "\n");
+               if (!ret)
+                       return TRACE_TYPE_PARTIAL_LINE;
+
+               return TRACE_TYPE_HANDLED;
        }
        return TRACE_TYPE_UNHANDLED;
 }
 
-static struct tracer return_trace __read_mostly =
-{
+static struct tracer return_trace __read_mostly = {
        .name        = "return",
        .init        = return_trace_init,
        .reset       = return_trace_reset,
-       .print_line = print_return_function
+       .print_line = print_return_function,
+       .flags          = &tracer_flags,
 };
 
 static __init int init_return_trace(void)