2 * ring buffer based initcalls tracer
4 * Copyright (C) 2008 Frederic Weisbecker <fweisbec@gmail.com>
8 #include <linux/init.h>
9 #include <linux/debugfs.h>
10 #include <linux/ftrace.h>
11 #include <linux/kallsyms.h>
12 #include <linux/time.h>
15 #include "trace_output.h"
17 static struct trace_array
*boot_trace
;
18 static bool pre_initcalls_finished
;
20 /* Tells the boot tracer that the pre_smp_initcalls are finished.
22 * It doesn't enable sched events tracing however.
23 * You have to call enable_boot_trace to do so.
25 void start_boot_trace(void)
27 pre_initcalls_finished
= true;
30 void enable_boot_trace(void)
32 if (boot_trace
&& pre_initcalls_finished
)
33 tracing_start_sched_switch_record();
36 void disable_boot_trace(void)
38 if (boot_trace
&& pre_initcalls_finished
)
39 tracing_stop_sched_switch_record();
42 static int boot_trace_init(struct trace_array
*tr
)
49 tracing_reset_online_cpus(tr
);
51 tracing_sched_switch_assign_trace(tr
);
55 static enum print_line_t
56 initcall_call_print_line(struct trace_iterator
*iter
)
58 struct trace_entry
*entry
= iter
->ent
;
59 struct trace_seq
*s
= &iter
->seq
;
60 struct trace_boot_call
*field
;
61 struct boot_trace_call
*call
;
63 unsigned long nsec_rem
;
66 trace_assign_type(field
, entry
);
67 call
= &field
->boot_call
;
69 nsec_rem
= do_div(ts
, NSEC_PER_SEC
);
71 ret
= trace_seq_printf(s
, "[%5ld.%09ld] calling %s @ %i\n",
72 (unsigned long)ts
, nsec_rem
, call
->func
, call
->caller
);
75 return TRACE_TYPE_PARTIAL_LINE
;
77 return TRACE_TYPE_HANDLED
;
80 static enum print_line_t
81 initcall_ret_print_line(struct trace_iterator
*iter
)
83 struct trace_entry
*entry
= iter
->ent
;
84 struct trace_seq
*s
= &iter
->seq
;
85 struct trace_boot_ret
*field
;
86 struct boot_trace_ret
*init_ret
;
88 unsigned long nsec_rem
;
91 trace_assign_type(field
, entry
);
92 init_ret
= &field
->boot_ret
;
94 nsec_rem
= do_div(ts
, NSEC_PER_SEC
);
96 ret
= trace_seq_printf(s
, "[%5ld.%09ld] initcall %s "
97 "returned %d after %llu msecs\n",
100 init_ret
->func
, init_ret
->result
, init_ret
->duration
);
103 return TRACE_TYPE_PARTIAL_LINE
;
105 return TRACE_TYPE_HANDLED
;
108 static enum print_line_t
initcall_print_line(struct trace_iterator
*iter
)
110 struct trace_entry
*entry
= iter
->ent
;
112 switch (entry
->type
) {
113 case TRACE_BOOT_CALL
:
114 return initcall_call_print_line(iter
);
116 return initcall_ret_print_line(iter
);
118 return TRACE_TYPE_UNHANDLED
;
122 struct tracer boot_tracer __read_mostly
=
125 .init
= boot_trace_init
,
126 .reset
= tracing_reset_online_cpus
,
127 .print_line
= initcall_print_line
,
130 void trace_boot_call(struct boot_trace_call
*bt
, initcall_t fn
)
132 struct ring_buffer_event
*event
;
133 struct ring_buffer
*buffer
;
134 struct trace_boot_call
*entry
;
135 struct trace_array
*tr
= boot_trace
;
137 if (!tr
|| !pre_initcalls_finished
)
140 /* Get its name now since this function could
141 * disappear because it is in the .init section.
143 sprint_symbol(bt
->func
, (unsigned long)fn
);
147 event
= trace_buffer_lock_reserve(buffer
, TRACE_BOOT_CALL
,
148 sizeof(*entry
), 0, 0);
151 entry
= ring_buffer_event_data(event
);
152 entry
->boot_call
= *bt
;
153 trace_buffer_unlock_commit(buffer
, event
, 0, 0);
158 void trace_boot_ret(struct boot_trace_ret
*bt
, initcall_t fn
)
160 struct ring_buffer_event
*event
;
161 struct ring_buffer
*buffer
;
162 struct trace_boot_ret
*entry
;
163 struct trace_array
*tr
= boot_trace
;
165 if (!tr
|| !pre_initcalls_finished
)
168 sprint_symbol(bt
->func
, (unsigned long)fn
);
172 event
= trace_buffer_lock_reserve(buffer
, TRACE_BOOT_RET
,
173 sizeof(*entry
), 0, 0);
176 entry
= ring_buffer_event_data(event
);
177 entry
->boot_ret
= *bt
;
178 trace_buffer_unlock_commit(buffer
, event
, 0, 0);