diff options
author | Steven Rostedt <srostedt@redhat.com> | 2008-05-12 15:20:44 -0400 |
---|---|---|
committer | Thomas Gleixner <tglx@linutronix.de> | 2008-05-23 14:40:36 -0400 |
commit | 60a11774b38fef1ab90b18c5353bd1c7c4d311c8 (patch) | |
tree | e2a6fd066b0dba6dcd776d07383e2932055cf66a | |
parent | e1c08bdd9fa73e44096e5a82c0d5928b04ab02c8 (diff) |
ftrace: add self-tests
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
-rw-r--r-- | kernel/trace/Kconfig | 13 | ||||
-rw-r--r-- | kernel/trace/trace.c | 63 | ||||
-rw-r--r-- | kernel/trace/trace.h | 31 | ||||
-rw-r--r-- | kernel/trace/trace_functions.c | 3 | ||||
-rw-r--r-- | kernel/trace/trace_irqsoff.c | 9 | ||||
-rw-r--r-- | kernel/trace/trace_sched_switch.c | 3 | ||||
-rw-r--r-- | kernel/trace/trace_sched_wakeup.c | 3 | ||||
-rw-r--r-- | kernel/trace/trace_selftest.c | 415 |
8 files changed, 538 insertions, 2 deletions
diff --git a/kernel/trace/Kconfig b/kernel/trace/Kconfig index cad9db1dee02..3f73a1710242 100644 --- a/kernel/trace/Kconfig +++ b/kernel/trace/Kconfig | |||
@@ -105,3 +105,16 @@ config DYNAMIC_FTRACE | |||
105 | wakes up once a second and checks to see if any ftrace calls | 105 | wakes up once a second and checks to see if any ftrace calls |
106 | were made. If so, it runs stop_machine (stops all CPUS) | 106 | were made. If so, it runs stop_machine (stops all CPUS) |
107 | and modifies the code to jump over the call to ftrace. | 107 | and modifies the code to jump over the call to ftrace. |
108 | |||
109 | config FTRACE_SELFTEST | ||
110 | bool | ||
111 | |||
112 | config FTRACE_STARTUP_TEST | ||
113 | bool "Perform a startup test on ftrace" | ||
114 | depends on TRACING | ||
115 | select FTRACE_SELFTEST | ||
116 | help | ||
117 | This option performs a series of startup tests on ftrace. On bootup | ||
118 | a series of tests are made to verify that the tracer is | ||
119 | functioning properly. It will do tests on all the configured | ||
120 | tracers of ftrace. | ||
diff --git a/kernel/trace/trace.c b/kernel/trace/trace.c index 9bad2379115a..f6d026f17dbb 100644 --- a/kernel/trace/trace.c +++ b/kernel/trace/trace.c | |||
@@ -32,6 +32,8 @@ | |||
32 | unsigned long __read_mostly tracing_max_latency = (cycle_t)ULONG_MAX; | 32 | unsigned long __read_mostly tracing_max_latency = (cycle_t)ULONG_MAX; |
33 | unsigned long __read_mostly tracing_thresh; | 33 | unsigned long __read_mostly tracing_thresh; |
34 | 34 | ||
35 | static int tracing_disabled = 1; | ||
36 | |||
35 | static long notrace | 37 | static long notrace |
36 | ns2usecs(cycle_t nsec) | 38 | ns2usecs(cycle_t nsec) |
37 | { | 39 | { |
@@ -217,11 +219,48 @@ int register_tracer(struct tracer *type) | |||
217 | } | 219 | } |
218 | } | 220 | } |
219 | 221 | ||
222 | #ifdef CONFIG_FTRACE_STARTUP_TEST | ||
223 | if (type->selftest) { | ||
224 | struct tracer *saved_tracer = current_trace; | ||
225 | struct trace_array_cpu *data; | ||
226 | struct trace_array *tr = &global_trace; | ||
227 | int saved_ctrl = tr->ctrl; | ||
228 | int i; | ||
229 | /* | ||
230 | * Run a selftest on this tracer. | ||
231 | * Here we reset the trace buffer, and set the current | ||
232 | * tracer to be this tracer. The tracer can then run some | ||
233 | * internal tracing to verify that everything is in order. | ||
234 | * If we fail, we do not register this tracer. | ||
235 | */ | ||
236 | for_each_possible_cpu(i) { | ||
237 | if (!data->trace) | ||
238 | continue; | ||
239 | data = tr->data[i]; | ||
240 | tracing_reset(data); | ||
241 | } | ||
242 | current_trace = type; | ||
243 | tr->ctrl = 0; | ||
244 | /* the test is responsible for initializing and enabling */ | ||
245 | pr_info("Testing tracer %s: ", type->name); | ||
246 | ret = type->selftest(type, tr); | ||
247 | /* the test is responsible for resetting too */ | ||
248 | current_trace = saved_tracer; | ||
249 | tr->ctrl = saved_ctrl; | ||
250 | if (ret) { | ||
251 | printk(KERN_CONT "FAILED!\n"); | ||
252 | goto out; | ||
253 | } | ||
254 | printk(KERN_CONT "PASSED\n"); | ||
255 | } | ||
256 | #endif | ||
257 | |||
220 | type->next = trace_types; | 258 | type->next = trace_types; |
221 | trace_types = type; | 259 | trace_types = type; |
222 | len = strlen(type->name); | 260 | len = strlen(type->name); |
223 | if (len > max_tracer_type_len) | 261 | if (len > max_tracer_type_len) |
224 | max_tracer_type_len = len; | 262 | max_tracer_type_len = len; |
263 | |||
225 | out: | 264 | out: |
226 | mutex_unlock(&trace_types_lock); | 265 | mutex_unlock(&trace_types_lock); |
227 | 266 | ||
@@ -985,6 +1024,11 @@ __tracing_open(struct inode *inode, struct file *file, int *ret) | |||
985 | { | 1024 | { |
986 | struct trace_iterator *iter; | 1025 | struct trace_iterator *iter; |
987 | 1026 | ||
1027 | if (tracing_disabled) { | ||
1028 | *ret = -ENODEV; | ||
1029 | return NULL; | ||
1030 | } | ||
1031 | |||
988 | iter = kzalloc(sizeof(*iter), GFP_KERNEL); | 1032 | iter = kzalloc(sizeof(*iter), GFP_KERNEL); |
989 | if (!iter) { | 1033 | if (!iter) { |
990 | *ret = -ENOMEM; | 1034 | *ret = -ENOMEM; |
@@ -1023,6 +1067,9 @@ __tracing_open(struct inode *inode, struct file *file, int *ret) | |||
1023 | 1067 | ||
1024 | int tracing_open_generic(struct inode *inode, struct file *filp) | 1068 | int tracing_open_generic(struct inode *inode, struct file *filp) |
1025 | { | 1069 | { |
1070 | if (tracing_disabled) | ||
1071 | return -ENODEV; | ||
1072 | |||
1026 | filp->private_data = inode->i_private; | 1073 | filp->private_data = inode->i_private; |
1027 | return 0; | 1074 | return 0; |
1028 | } | 1075 | } |
@@ -1128,6 +1175,9 @@ static int show_traces_open(struct inode *inode, struct file *file) | |||
1128 | { | 1175 | { |
1129 | int ret; | 1176 | int ret; |
1130 | 1177 | ||
1178 | if (tracing_disabled) | ||
1179 | return -ENODEV; | ||
1180 | |||
1131 | ret = seq_open(file, &show_traces_seq_ops); | 1181 | ret = seq_open(file, &show_traces_seq_ops); |
1132 | if (!ret) { | 1182 | if (!ret) { |
1133 | struct seq_file *m = file->private_data; | 1183 | struct seq_file *m = file->private_data; |
@@ -1452,6 +1502,11 @@ struct dentry *tracing_init_dentry(void) | |||
1452 | return d_tracer; | 1502 | return d_tracer; |
1453 | } | 1503 | } |
1454 | 1504 | ||
1505 | #ifdef CONFIG_FTRACE_SELFTEST | ||
1506 | /* Let selftest have access to static functions in this file */ | ||
1507 | #include "trace_selftest.c" | ||
1508 | #endif | ||
1509 | |||
1455 | static __init void tracer_init_debugfs(void) | 1510 | static __init void tracer_init_debugfs(void) |
1456 | { | 1511 | { |
1457 | struct dentry *d_tracer; | 1512 | struct dentry *d_tracer; |
@@ -1585,6 +1640,7 @@ __init static int tracer_alloc_buffers(void) | |||
1585 | void *array; | 1640 | void *array; |
1586 | struct page *page; | 1641 | struct page *page; |
1587 | int pages = 0; | 1642 | int pages = 0; |
1643 | int ret = -ENOMEM; | ||
1588 | int i; | 1644 | int i; |
1589 | 1645 | ||
1590 | /* Allocate the first page for all buffers */ | 1646 | /* Allocate the first page for all buffers */ |
@@ -1650,6 +1706,9 @@ __init static int tracer_alloc_buffers(void) | |||
1650 | register_tracer(&no_tracer); | 1706 | register_tracer(&no_tracer); |
1651 | current_trace = &no_tracer; | 1707 | current_trace = &no_tracer; |
1652 | 1708 | ||
1709 | /* All seems OK, enable tracing */ | ||
1710 | tracing_disabled = 0; | ||
1711 | |||
1653 | return 0; | 1712 | return 0; |
1654 | 1713 | ||
1655 | free_buffers: | 1714 | free_buffers: |
@@ -1678,7 +1737,7 @@ __init static int tracer_alloc_buffers(void) | |||
1678 | } | 1737 | } |
1679 | #endif | 1738 | #endif |
1680 | } | 1739 | } |
1681 | return -ENOMEM; | 1740 | return ret; |
1682 | } | 1741 | } |
1683 | 1742 | ||
1684 | device_initcall(tracer_alloc_buffers); | 1743 | fs_initcall(tracer_alloc_buffers); |
diff --git a/kernel/trace/trace.h b/kernel/trace/trace.h index 83e257e38084..88edbf1f6788 100644 --- a/kernel/trace/trace.h +++ b/kernel/trace/trace.h | |||
@@ -99,6 +99,10 @@ struct tracer { | |||
99 | void (*start)(struct trace_iterator *iter); | 99 | void (*start)(struct trace_iterator *iter); |
100 | void (*stop)(struct trace_iterator *iter); | 100 | void (*stop)(struct trace_iterator *iter); |
101 | void (*ctrl_update)(struct trace_array *tr); | 101 | void (*ctrl_update)(struct trace_array *tr); |
102 | #ifdef CONFIG_FTRACE_STARTUP_TEST | ||
103 | int (*selftest)(struct tracer *trace, | ||
104 | struct trace_array *tr); | ||
105 | #endif | ||
102 | struct tracer *next; | 106 | struct tracer *next; |
103 | int print_max; | 107 | int print_max; |
104 | }; | 108 | }; |
@@ -185,4 +189,31 @@ extern int unregister_tracer_switch(struct tracer_switch_ops *ops); | |||
185 | extern unsigned long ftrace_update_tot_cnt; | 189 | extern unsigned long ftrace_update_tot_cnt; |
186 | #endif | 190 | #endif |
187 | 191 | ||
192 | #ifdef CONFIG_FTRACE_STARTUP_TEST | ||
193 | #ifdef CONFIG_FTRACE | ||
194 | extern int trace_selftest_startup_function(struct tracer *trace, | ||
195 | struct trace_array *tr); | ||
196 | #endif | ||
197 | #ifdef CONFIG_IRQSOFF_TRACER | ||
198 | extern int trace_selftest_startup_irqsoff(struct tracer *trace, | ||
199 | struct trace_array *tr); | ||
200 | #endif | ||
201 | #ifdef CONFIG_PREEMPT_TRACER | ||
202 | extern int trace_selftest_startup_preemptoff(struct tracer *trace, | ||
203 | struct trace_array *tr); | ||
204 | #endif | ||
205 | #if defined(CONFIG_IRQSOFF_TRACER) && defined(CONFIG_PREEMPT_TRACER) | ||
206 | extern int trace_selftest_startup_preemptirqsoff(struct tracer *trace, | ||
207 | struct trace_array *tr); | ||
208 | #endif | ||
209 | #ifdef CONFIG_SCHED_TRACER | ||
210 | extern int trace_selftest_startup_wakeup(struct tracer *trace, | ||
211 | struct trace_array *tr); | ||
212 | #endif | ||
213 | #ifdef CONFIG_CONTEXT_SWITCH_TRACER | ||
214 | extern int trace_selftest_startup_sched_switch(struct tracer *trace, | ||
215 | struct trace_array *tr); | ||
216 | #endif | ||
217 | #endif /* CONFIG_FTRACE_STARTUP_TEST */ | ||
218 | |||
188 | #endif /* _LINUX_KERNEL_TRACE_H */ | 219 | #endif /* _LINUX_KERNEL_TRACE_H */ |
diff --git a/kernel/trace/trace_functions.c b/kernel/trace/trace_functions.c index 82988c5336e0..5d8ad7a09605 100644 --- a/kernel/trace/trace_functions.c +++ b/kernel/trace/trace_functions.c | |||
@@ -63,6 +63,9 @@ static struct tracer function_trace __read_mostly = | |||
63 | .init = function_trace_init, | 63 | .init = function_trace_init, |
64 | .reset = function_trace_reset, | 64 | .reset = function_trace_reset, |
65 | .ctrl_update = function_trace_ctrl_update, | 65 | .ctrl_update = function_trace_ctrl_update, |
66 | #ifdef CONFIG_FTRACE_SELFTEST | ||
67 | .selftest = trace_selftest_startup_function, | ||
68 | #endif | ||
66 | }; | 69 | }; |
67 | 70 | ||
68 | static __init int init_function_trace(void) | 71 | static __init int init_function_trace(void) |
diff --git a/kernel/trace/trace_irqsoff.c b/kernel/trace/trace_irqsoff.c index 74165f611f36..14183b8f79c5 100644 --- a/kernel/trace/trace_irqsoff.c +++ b/kernel/trace/trace_irqsoff.c | |||
@@ -432,6 +432,9 @@ static struct tracer irqsoff_tracer __read_mostly = | |||
432 | .close = irqsoff_tracer_close, | 432 | .close = irqsoff_tracer_close, |
433 | .ctrl_update = irqsoff_tracer_ctrl_update, | 433 | .ctrl_update = irqsoff_tracer_ctrl_update, |
434 | .print_max = 1, | 434 | .print_max = 1, |
435 | #ifdef CONFIG_FTRACE_SELFTEST | ||
436 | .selftest = trace_selftest_startup_irqsoff, | ||
437 | #endif | ||
435 | }; | 438 | }; |
436 | # define register_irqsoff(trace) register_tracer(&trace) | 439 | # define register_irqsoff(trace) register_tracer(&trace) |
437 | #else | 440 | #else |
@@ -455,6 +458,9 @@ static struct tracer preemptoff_tracer __read_mostly = | |||
455 | .close = irqsoff_tracer_close, | 458 | .close = irqsoff_tracer_close, |
456 | .ctrl_update = irqsoff_tracer_ctrl_update, | 459 | .ctrl_update = irqsoff_tracer_ctrl_update, |
457 | .print_max = 1, | 460 | .print_max = 1, |
461 | #ifdef CONFIG_FTRACE_SELFTEST | ||
462 | .selftest = trace_selftest_startup_preemptoff, | ||
463 | #endif | ||
458 | }; | 464 | }; |
459 | # define register_preemptoff(trace) register_tracer(&trace) | 465 | # define register_preemptoff(trace) register_tracer(&trace) |
460 | #else | 466 | #else |
@@ -480,6 +486,9 @@ static struct tracer preemptirqsoff_tracer __read_mostly = | |||
480 | .close = irqsoff_tracer_close, | 486 | .close = irqsoff_tracer_close, |
481 | .ctrl_update = irqsoff_tracer_ctrl_update, | 487 | .ctrl_update = irqsoff_tracer_ctrl_update, |
482 | .print_max = 1, | 488 | .print_max = 1, |
489 | #ifdef CONFIG_FTRACE_SELFTEST | ||
490 | .selftest = trace_selftest_startup_preemptirqsoff, | ||
491 | #endif | ||
483 | }; | 492 | }; |
484 | 493 | ||
485 | # define register_preemptirqsoff(trace) register_tracer(&trace) | 494 | # define register_preemptirqsoff(trace) register_tracer(&trace) |
diff --git a/kernel/trace/trace_sched_switch.c b/kernel/trace/trace_sched_switch.c index 2715267be469..6c9284103a62 100644 --- a/kernel/trace/trace_sched_switch.c +++ b/kernel/trace/trace_sched_switch.c | |||
@@ -107,6 +107,9 @@ static struct tracer sched_switch_trace __read_mostly = | |||
107 | .init = sched_switch_trace_init, | 107 | .init = sched_switch_trace_init, |
108 | .reset = sched_switch_trace_reset, | 108 | .reset = sched_switch_trace_reset, |
109 | .ctrl_update = sched_switch_trace_ctrl_update, | 109 | .ctrl_update = sched_switch_trace_ctrl_update, |
110 | #ifdef CONFIG_FTRACE_SELFTEST | ||
111 | .selftest = trace_selftest_startup_sched_switch, | ||
112 | #endif | ||
110 | }; | 113 | }; |
111 | 114 | ||
112 | __init static int init_sched_switch_trace(void) | 115 | __init static int init_sched_switch_trace(void) |
diff --git a/kernel/trace/trace_sched_wakeup.c b/kernel/trace/trace_sched_wakeup.c index 7c3ccefcf4c3..3d10ff01f805 100644 --- a/kernel/trace/trace_sched_wakeup.c +++ b/kernel/trace/trace_sched_wakeup.c | |||
@@ -295,6 +295,9 @@ static struct tracer wakeup_tracer __read_mostly = | |||
295 | .close = wakeup_tracer_close, | 295 | .close = wakeup_tracer_close, |
296 | .ctrl_update = wakeup_tracer_ctrl_update, | 296 | .ctrl_update = wakeup_tracer_ctrl_update, |
297 | .print_max = 1, | 297 | .print_max = 1, |
298 | #ifdef CONFIG_FTRACE_SELFTEST | ||
299 | .selftest = trace_selftest_startup_wakeup, | ||
300 | #endif | ||
298 | }; | 301 | }; |
299 | 302 | ||
300 | __init static int init_wakeup_tracer(void) | 303 | __init static int init_wakeup_tracer(void) |
diff --git a/kernel/trace/trace_selftest.c b/kernel/trace/trace_selftest.c new file mode 100644 index 000000000000..ef4d3cc009f5 --- /dev/null +++ b/kernel/trace/trace_selftest.c | |||
@@ -0,0 +1,415 @@ | |||
1 | /* Include in trace.c */ | ||
2 | |||
3 | #include <linux/kthread.h> | ||
4 | |||
5 | static inline int trace_valid_entry(struct trace_entry *entry) | ||
6 | { | ||
7 | switch (entry->type) { | ||
8 | case TRACE_FN: | ||
9 | case TRACE_CTX: | ||
10 | return 1; | ||
11 | } | ||
12 | return 0; | ||
13 | } | ||
14 | |||
15 | static int | ||
16 | trace_test_buffer_cpu(struct trace_array *tr, struct trace_array_cpu *data) | ||
17 | { | ||
18 | struct page *page; | ||
19 | struct trace_entry *entries; | ||
20 | int idx = 0; | ||
21 | int i; | ||
22 | |||
23 | page = list_entry(data->trace_pages.next, struct page, lru); | ||
24 | entries = page_address(page); | ||
25 | |||
26 | if (data->trace != entries) | ||
27 | goto failed; | ||
28 | |||
29 | /* | ||
30 | * The starting trace buffer always has valid elements, | ||
31 | * if any element exits. | ||
32 | */ | ||
33 | entries = data->trace; | ||
34 | |||
35 | for (i = 0; i < tr->entries; i++) { | ||
36 | |||
37 | if (i < data->trace_idx && | ||
38 | !trace_valid_entry(&entries[idx])) { | ||
39 | printk(KERN_CONT ".. invalid entry %d ", entries[idx].type); | ||
40 | goto failed; | ||
41 | } | ||
42 | |||
43 | idx++; | ||
44 | if (idx >= ENTRIES_PER_PAGE) { | ||
45 | page = virt_to_page(entries); | ||
46 | if (page->lru.next == &data->trace_pages) { | ||
47 | if (i != tr->entries - 1) { | ||
48 | printk(KERN_CONT ".. entries buffer mismatch"); | ||
49 | goto failed; | ||
50 | } | ||
51 | } else { | ||
52 | page = list_entry(page->lru.next, struct page, lru); | ||
53 | entries = page_address(page); | ||
54 | } | ||
55 | idx = 0; | ||
56 | } | ||
57 | } | ||
58 | |||
59 | page = virt_to_page(entries); | ||
60 | if (page->lru.next != &data->trace_pages) { | ||
61 | printk(KERN_CONT ".. too many entries"); | ||
62 | goto failed; | ||
63 | } | ||
64 | |||
65 | return 0; | ||
66 | |||
67 | failed: | ||
68 | printk(KERN_CONT ".. corrupted trace buffer .. "); | ||
69 | return -1; | ||
70 | } | ||
71 | |||
72 | /* | ||
73 | * Test the trace buffer to see if all the elements | ||
74 | * are still sane. | ||
75 | */ | ||
76 | static int trace_test_buffer(struct trace_array *tr, unsigned long *count) | ||
77 | { | ||
78 | unsigned long cnt = 0; | ||
79 | int cpu; | ||
80 | int ret = 0; | ||
81 | |||
82 | for_each_possible_cpu(cpu) { | ||
83 | if (!tr->data[cpu]->trace) | ||
84 | continue; | ||
85 | |||
86 | cnt += tr->data[cpu]->trace_idx; | ||
87 | printk("%d: count = %ld\n", cpu, cnt); | ||
88 | |||
89 | ret = trace_test_buffer_cpu(tr, tr->data[cpu]); | ||
90 | if (ret) | ||
91 | break; | ||
92 | } | ||
93 | |||
94 | if (count) | ||
95 | *count = cnt; | ||
96 | |||
97 | return ret; | ||
98 | } | ||
99 | |||
100 | #ifdef CONFIG_FTRACE | ||
101 | /* | ||
102 | * Simple verification test of ftrace function tracer. | ||
103 | * Enable ftrace, sleep 1/10 second, and then read the trace | ||
104 | * buffer to see if all is in order. | ||
105 | */ | ||
106 | int | ||
107 | trace_selftest_startup_function(struct tracer *trace, struct trace_array *tr) | ||
108 | { | ||
109 | unsigned long count; | ||
110 | int ret; | ||
111 | |||
112 | /* make sure functions have been recorded */ | ||
113 | ret = ftrace_force_update(); | ||
114 | if (ret) { | ||
115 | printk(KERN_CONT ".. ftraced failed .. "); | ||
116 | return ret; | ||
117 | } | ||
118 | |||
119 | /* start the tracing */ | ||
120 | tr->ctrl = 1; | ||
121 | trace->init(tr); | ||
122 | /* Sleep for a 1/10 of a second */ | ||
123 | msleep(100); | ||
124 | /* stop the tracing. */ | ||
125 | tr->ctrl = 0; | ||
126 | trace->ctrl_update(tr); | ||
127 | /* check the trace buffer */ | ||
128 | ret = trace_test_buffer(tr, &count); | ||
129 | trace->reset(tr); | ||
130 | |||
131 | if (!ret && !count) { | ||
132 | printk(KERN_CONT ".. no entries found .."); | ||
133 | ret = -1; | ||
134 | } | ||
135 | |||
136 | return ret; | ||
137 | } | ||
138 | #endif /* CONFIG_FTRACE */ | ||
139 | |||
140 | #ifdef CONFIG_IRQSOFF_TRACER | ||
141 | int | ||
142 | trace_selftest_startup_irqsoff(struct tracer *trace, struct trace_array *tr) | ||
143 | { | ||
144 | unsigned long save_max = tracing_max_latency; | ||
145 | unsigned long count; | ||
146 | int ret; | ||
147 | |||
148 | /* start the tracing */ | ||
149 | tr->ctrl = 1; | ||
150 | trace->init(tr); | ||
151 | /* reset the max latency */ | ||
152 | tracing_max_latency = 0; | ||
153 | /* disable interrupts for a bit */ | ||
154 | local_irq_disable(); | ||
155 | udelay(100); | ||
156 | local_irq_enable(); | ||
157 | /* stop the tracing. */ | ||
158 | tr->ctrl = 0; | ||
159 | trace->ctrl_update(tr); | ||
160 | /* check both trace buffers */ | ||
161 | ret = trace_test_buffer(tr, NULL); | ||
162 | if (!ret) | ||
163 | ret = trace_test_buffer(&max_tr, &count); | ||
164 | trace->reset(tr); | ||
165 | |||
166 | if (!ret && !count) { | ||
167 | printk(KERN_CONT ".. no entries found .."); | ||
168 | ret = -1; | ||
169 | } | ||
170 | |||
171 | tracing_max_latency = save_max; | ||
172 | |||
173 | return ret; | ||
174 | } | ||
175 | #endif /* CONFIG_IRQSOFF_TRACER */ | ||
176 | |||
177 | #ifdef CONFIG_PREEMPT_TRACER | ||
178 | int | ||
179 | trace_selftest_startup_preemptoff(struct tracer *trace, struct trace_array *tr) | ||
180 | { | ||
181 | unsigned long save_max = tracing_max_latency; | ||
182 | unsigned long count; | ||
183 | int ret; | ||
184 | |||
185 | /* start the tracing */ | ||
186 | tr->ctrl = 1; | ||
187 | trace->init(tr); | ||
188 | /* reset the max latency */ | ||
189 | tracing_max_latency = 0; | ||
190 | /* disable preemption for a bit */ | ||
191 | preempt_disable(); | ||
192 | udelay(100); | ||
193 | preempt_enable(); | ||
194 | /* stop the tracing. */ | ||
195 | tr->ctrl = 0; | ||
196 | trace->ctrl_update(tr); | ||
197 | /* check both trace buffers */ | ||
198 | ret = trace_test_buffer(tr, NULL); | ||
199 | if (!ret) | ||
200 | ret = trace_test_buffer(&max_tr, &count); | ||
201 | trace->reset(tr); | ||
202 | |||
203 | if (!ret && !count) { | ||
204 | printk(KERN_CONT ".. no entries found .."); | ||
205 | ret = -1; | ||
206 | } | ||
207 | |||
208 | tracing_max_latency = save_max; | ||
209 | |||
210 | return ret; | ||
211 | } | ||
212 | #endif /* CONFIG_PREEMPT_TRACER */ | ||
213 | |||
214 | #if defined(CONFIG_IRQSOFF_TRACER) && defined(CONFIG_PREEMPT_TRACER) | ||
215 | int | ||
216 | trace_selftest_startup_preemptirqsoff(struct tracer *trace, struct trace_array *tr) | ||
217 | { | ||
218 | unsigned long save_max = tracing_max_latency; | ||
219 | unsigned long count; | ||
220 | int ret; | ||
221 | |||
222 | /* start the tracing */ | ||
223 | tr->ctrl = 1; | ||
224 | trace->init(tr); | ||
225 | |||
226 | /* reset the max latency */ | ||
227 | tracing_max_latency = 0; | ||
228 | |||
229 | /* disable preemption and interrupts for a bit */ | ||
230 | preempt_disable(); | ||
231 | local_irq_disable(); | ||
232 | udelay(100); | ||
233 | preempt_enable(); | ||
234 | /* reverse the order of preempt vs irqs */ | ||
235 | local_irq_enable(); | ||
236 | |||
237 | /* stop the tracing. */ | ||
238 | tr->ctrl = 0; | ||
239 | trace->ctrl_update(tr); | ||
240 | /* check both trace buffers */ | ||
241 | ret = trace_test_buffer(tr, NULL); | ||
242 | if (ret) | ||
243 | goto out; | ||
244 | |||
245 | ret = trace_test_buffer(&max_tr, &count); | ||
246 | if (ret) | ||
247 | goto out; | ||
248 | |||
249 | if (!ret && !count) { | ||
250 | printk(KERN_CONT ".. no entries found .."); | ||
251 | ret = -1; | ||
252 | goto out; | ||
253 | } | ||
254 | |||
255 | /* do the test by disabling interrupts first this time */ | ||
256 | tracing_max_latency = 0; | ||
257 | tr->ctrl = 1; | ||
258 | trace->ctrl_update(tr); | ||
259 | preempt_disable(); | ||
260 | local_irq_disable(); | ||
261 | udelay(100); | ||
262 | preempt_enable(); | ||
263 | /* reverse the order of preempt vs irqs */ | ||
264 | local_irq_enable(); | ||
265 | |||
266 | /* stop the tracing. */ | ||
267 | tr->ctrl = 0; | ||
268 | trace->ctrl_update(tr); | ||
269 | /* check both trace buffers */ | ||
270 | ret = trace_test_buffer(tr, NULL); | ||
271 | if (ret) | ||
272 | goto out; | ||
273 | |||
274 | ret = trace_test_buffer(&max_tr, &count); | ||
275 | |||
276 | if (!ret && !count) { | ||
277 | printk(KERN_CONT ".. no entries found .."); | ||
278 | ret = -1; | ||
279 | goto out; | ||
280 | } | ||
281 | |||
282 | out: | ||
283 | trace->reset(tr); | ||
284 | tracing_max_latency = save_max; | ||
285 | |||
286 | return ret; | ||
287 | } | ||
288 | #endif /* CONFIG_IRQSOFF_TRACER && CONFIG_PREEMPT_TRACER */ | ||
289 | |||
290 | #ifdef CONFIG_SCHED_TRACER | ||
291 | static int trace_wakeup_test_thread(void *data) | ||
292 | { | ||
293 | struct completion *x = data; | ||
294 | |||
295 | /* Make this a RT thread, doesn't need to be too high */ | ||
296 | |||
297 | rt_mutex_setprio(current, MAX_RT_PRIO - 5); | ||
298 | |||
299 | /* Make it know we have a new prio */ | ||
300 | complete(x); | ||
301 | |||
302 | /* now go to sleep and let the test wake us up */ | ||
303 | set_current_state(TASK_INTERRUPTIBLE); | ||
304 | schedule(); | ||
305 | |||
306 | /* we are awake, now wait to disappear */ | ||
307 | while (!kthread_should_stop()) { | ||
308 | /* | ||
309 | * This is an RT task, do short sleeps to let | ||
310 | * others run. | ||
311 | */ | ||
312 | msleep(100); | ||
313 | } | ||
314 | |||
315 | return 0; | ||
316 | } | ||
317 | |||
318 | int | ||
319 | trace_selftest_startup_wakeup(struct tracer *trace, struct trace_array *tr) | ||
320 | { | ||
321 | unsigned long save_max = tracing_max_latency; | ||
322 | struct task_struct *p; | ||
323 | struct completion isrt; | ||
324 | unsigned long count; | ||
325 | int ret; | ||
326 | |||
327 | init_completion(&isrt); | ||
328 | |||
329 | /* create a high prio thread */ | ||
330 | p = kthread_run(trace_wakeup_test_thread, &isrt, "ftrace-test"); | ||
331 | if (!IS_ERR(p)) { | ||
332 | printk(KERN_CONT "Failed to create ftrace wakeup test thread "); | ||
333 | return -1; | ||
334 | } | ||
335 | |||
336 | /* make sure the thread is running at an RT prio */ | ||
337 | wait_for_completion(&isrt); | ||
338 | |||
339 | /* start the tracing */ | ||
340 | tr->ctrl = 1; | ||
341 | trace->init(tr); | ||
342 | /* reset the max latency */ | ||
343 | tracing_max_latency = 0; | ||
344 | |||
345 | /* sleep to let the RT thread sleep too */ | ||
346 | msleep(100); | ||
347 | |||
348 | /* | ||
349 | * Yes this is slightly racy. It is possible that for some | ||
350 | * strange reason that the RT thread we created, did not | ||
351 | * call schedule for 100ms after doing the completion, | ||
352 | * and we do a wakeup on a task that already is awake. | ||
353 | * But that is extremely unlikely, and the worst thing that | ||
354 | * happens in such a case, is that we disable tracing. | ||
355 | * Honestly, if this race does happen something is horrible | ||
356 | * wrong with the system. | ||
357 | */ | ||
358 | |||
359 | wake_up_process(p); | ||
360 | |||
361 | /* stop the tracing. */ | ||
362 | tr->ctrl = 0; | ||
363 | trace->ctrl_update(tr); | ||
364 | /* check both trace buffers */ | ||
365 | ret = trace_test_buffer(tr, NULL); | ||
366 | if (!ret) | ||
367 | ret = trace_test_buffer(&max_tr, &count); | ||
368 | |||
369 | |||
370 | trace->reset(tr); | ||
371 | |||
372 | tracing_max_latency = save_max; | ||
373 | |||
374 | /* kill the thread */ | ||
375 | kthread_stop(p); | ||
376 | |||
377 | if (!ret && !count) { | ||
378 | printk(KERN_CONT ".. no entries found .."); | ||
379 | ret = -1; | ||
380 | } | ||
381 | |||
382 | return ret; | ||
383 | } | ||
384 | #endif /* CONFIG_SCHED_TRACER */ | ||
385 | |||
386 | #ifdef CONFIG_CONTEXT_SWITCH_TRACER | ||
387 | int | ||
388 | trace_selftest_startup_sched_switch(struct tracer *trace, struct trace_array *tr) | ||
389 | { | ||
390 | unsigned long count; | ||
391 | int ret; | ||
392 | |||
393 | /* start the tracing */ | ||
394 | tr->ctrl = 1; | ||
395 | trace->init(tr); | ||
396 | /* Sleep for a 1/10 of a second */ | ||
397 | msleep(100); | ||
398 | /* stop the tracing. */ | ||
399 | tr->ctrl = 0; | ||
400 | trace->ctrl_update(tr); | ||
401 | /* check the trace buffer */ | ||
402 | ret = trace_test_buffer(tr, &count); | ||
403 | trace->reset(tr); | ||
404 | |||
405 | if (!ret && !count) { | ||
406 | printk(KERN_CONT ".. no entries found .."); | ||
407 | ret = -1; | ||
408 | } | ||
409 | |||
410 | return ret; | ||
411 | } | ||
412 | #endif /* CONFIG_CONTEXT_SWITCH_TRACER */ | ||
413 | |||
414 | #ifdef CONFIG_DYNAMIC_FTRACE | ||
415 | #endif /* CONFIG_DYNAMIC_FTRACE */ | ||