aboutsummaryrefslogtreecommitdiffstats
path: root/kernel/trace/trace_functions_graph.c
diff options
context:
space:
mode:
Diffstat (limited to 'kernel/trace/trace_functions_graph.c')
-rw-r--r--kernel/trace/trace_functions_graph.c298
1 files changed, 195 insertions, 103 deletions
diff --git a/kernel/trace/trace_functions_graph.c b/kernel/trace/trace_functions_graph.c
index 3c545984816f..66fc7b806a38 100644
--- a/kernel/trace/trace_functions_graph.c
+++ b/kernel/trace/trace_functions_graph.c
@@ -1,7 +1,7 @@
1/* 1/*
2 * 2 *
3 * Function graph tracer. 3 * Function graph tracer.
4 * Copyright (c) 2008 Frederic Weisbecker <fweisbec@gmail.com> 4 * Copyright (c) 2008-2009 Frederic Weisbecker <fweisbec@gmail.com>
5 * Mostly borrowed from function tracer which 5 * Mostly borrowed from function tracer which
6 * is Copyright (c) Steven Rostedt <srostedt@redhat.com> 6 * is Copyright (c) Steven Rostedt <srostedt@redhat.com>
7 * 7 *
@@ -21,9 +21,11 @@
21#define TRACE_GRAPH_PRINT_CPU 0x2 21#define TRACE_GRAPH_PRINT_CPU 0x2
22#define TRACE_GRAPH_PRINT_OVERHEAD 0x4 22#define TRACE_GRAPH_PRINT_OVERHEAD 0x4
23#define TRACE_GRAPH_PRINT_PROC 0x8 23#define TRACE_GRAPH_PRINT_PROC 0x8
24#define TRACE_GRAPH_PRINT_DURATION 0x10
25#define TRACE_GRAPH_PRINT_ABS_TIME 0X20
24 26
25static struct tracer_opt trace_opts[] = { 27static struct tracer_opt trace_opts[] = {
26 /* Display overruns ? */ 28 /* Display overruns? (for self-debug purpose) */
27 { TRACER_OPT(funcgraph-overrun, TRACE_GRAPH_PRINT_OVERRUN) }, 29 { TRACER_OPT(funcgraph-overrun, TRACE_GRAPH_PRINT_OVERRUN) },
28 /* Display CPU ? */ 30 /* Display CPU ? */
29 { TRACER_OPT(funcgraph-cpu, TRACE_GRAPH_PRINT_CPU) }, 31 { TRACER_OPT(funcgraph-cpu, TRACE_GRAPH_PRINT_CPU) },
@@ -31,17 +33,22 @@ static struct tracer_opt trace_opts[] = {
31 { TRACER_OPT(funcgraph-overhead, TRACE_GRAPH_PRINT_OVERHEAD) }, 33 { TRACER_OPT(funcgraph-overhead, TRACE_GRAPH_PRINT_OVERHEAD) },
32 /* Display proc name/pid */ 34 /* Display proc name/pid */
33 { TRACER_OPT(funcgraph-proc, TRACE_GRAPH_PRINT_PROC) }, 35 { TRACER_OPT(funcgraph-proc, TRACE_GRAPH_PRINT_PROC) },
36 /* Display duration of execution */
37 { TRACER_OPT(funcgraph-duration, TRACE_GRAPH_PRINT_DURATION) },
38 /* Display absolute time of an entry */
39 { TRACER_OPT(funcgraph-abstime, TRACE_GRAPH_PRINT_ABS_TIME) },
34 { } /* Empty entry */ 40 { } /* Empty entry */
35}; 41};
36 42
37static struct tracer_flags tracer_flags = { 43static struct tracer_flags tracer_flags = {
38 /* Don't display overruns and proc by default */ 44 /* Don't display overruns and proc by default */
39 .val = TRACE_GRAPH_PRINT_CPU | TRACE_GRAPH_PRINT_OVERHEAD, 45 .val = TRACE_GRAPH_PRINT_CPU | TRACE_GRAPH_PRINT_OVERHEAD |
46 TRACE_GRAPH_PRINT_DURATION,
40 .opts = trace_opts 47 .opts = trace_opts
41}; 48};
42 49
43/* pid on the last trace processed */ 50/* pid on the last trace processed */
44static pid_t last_pid[NR_CPUS] = { [0 ... NR_CPUS-1] = -1 }; 51
45 52
46static int graph_trace_init(struct trace_array *tr) 53static int graph_trace_init(struct trace_array *tr)
47{ 54{
@@ -154,17 +161,25 @@ print_graph_proc(struct trace_seq *s, pid_t pid)
154 161
155/* If the pid changed since the last trace, output this event */ 162/* If the pid changed since the last trace, output this event */
156static enum print_line_t 163static enum print_line_t
157verif_pid(struct trace_seq *s, pid_t pid, int cpu) 164verif_pid(struct trace_seq *s, pid_t pid, int cpu, pid_t *last_pids_cpu)
158{ 165{
159 pid_t prev_pid; 166 pid_t prev_pid;
167 pid_t *last_pid;
160 int ret; 168 int ret;
161 169
162 if (last_pid[cpu] != -1 && last_pid[cpu] == pid) 170 if (!last_pids_cpu)
171 return TRACE_TYPE_HANDLED;
172
173 last_pid = per_cpu_ptr(last_pids_cpu, cpu);
174
175 if (*last_pid == pid)
163 return TRACE_TYPE_HANDLED; 176 return TRACE_TYPE_HANDLED;
164 177
165 prev_pid = last_pid[cpu]; 178 prev_pid = *last_pid;
166 last_pid[cpu] = pid; 179 *last_pid = pid;
167 180
181 if (prev_pid == -1)
182 return TRACE_TYPE_HANDLED;
168/* 183/*
169 * Context-switch trace line: 184 * Context-switch trace line:
170 185
@@ -232,9 +247,34 @@ trace_branch_is_leaf(struct trace_iterator *iter,
232 return true; 247 return true;
233} 248}
234 249
250/* Signal a overhead of time execution to the output */
251static int
252print_graph_overhead(unsigned long long duration, struct trace_seq *s)
253{
254 /* If duration disappear, we don't need anything */
255 if (!(tracer_flags.val & TRACE_GRAPH_PRINT_DURATION))
256 return 1;
257
258 /* Non nested entry or return */
259 if (duration == -1)
260 return trace_seq_printf(s, " ");
261
262 if (tracer_flags.val & TRACE_GRAPH_PRINT_OVERHEAD) {
263 /* Duration exceeded 100 msecs */
264 if (duration > 100000ULL)
265 return trace_seq_printf(s, "! ");
266
267 /* Duration exceeded 10 msecs */
268 if (duration > 10000ULL)
269 return trace_seq_printf(s, "+ ");
270 }
271
272 return trace_seq_printf(s, " ");
273}
274
235static enum print_line_t 275static enum print_line_t
236print_graph_irq(struct trace_seq *s, unsigned long addr, 276print_graph_irq(struct trace_seq *s, unsigned long addr,
237 enum trace_type type, int cpu, pid_t pid) 277 enum trace_type type, int cpu, pid_t pid)
238{ 278{
239 int ret; 279 int ret;
240 280
@@ -242,35 +282,40 @@ print_graph_irq(struct trace_seq *s, unsigned long addr,
242 addr >= (unsigned long)__irqentry_text_end) 282 addr >= (unsigned long)__irqentry_text_end)
243 return TRACE_TYPE_UNHANDLED; 283 return TRACE_TYPE_UNHANDLED;
244 284
245 if (type == TRACE_GRAPH_ENT) { 285 /* Cpu */
246 ret = trace_seq_printf(s, "==========> | "); 286 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU) {
247 } else { 287 ret = print_graph_cpu(s, cpu);
248 /* Cpu */ 288 if (ret == TRACE_TYPE_PARTIAL_LINE)
249 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU) { 289 return TRACE_TYPE_PARTIAL_LINE;
250 ret = print_graph_cpu(s, cpu); 290 }
251 if (ret == TRACE_TYPE_PARTIAL_LINE) 291 /* Proc */
252 return TRACE_TYPE_PARTIAL_LINE; 292 if (tracer_flags.val & TRACE_GRAPH_PRINT_PROC) {
253 } 293 ret = print_graph_proc(s, pid);
254 /* Proc */ 294 if (ret == TRACE_TYPE_PARTIAL_LINE)
255 if (tracer_flags.val & TRACE_GRAPH_PRINT_PROC) { 295 return TRACE_TYPE_PARTIAL_LINE;
256 ret = print_graph_proc(s, pid); 296 ret = trace_seq_printf(s, " | ");
257 if (ret == TRACE_TYPE_PARTIAL_LINE) 297 if (!ret)
258 return TRACE_TYPE_PARTIAL_LINE; 298 return TRACE_TYPE_PARTIAL_LINE;
299 }
259 300
260 ret = trace_seq_printf(s, " | "); 301 /* No overhead */
261 if (!ret) 302 ret = print_graph_overhead(-1, s);
262 return TRACE_TYPE_PARTIAL_LINE; 303 if (!ret)
263 } 304 return TRACE_TYPE_PARTIAL_LINE;
264 305
265 /* No overhead */ 306 if (type == TRACE_GRAPH_ENT)
266 if (tracer_flags.val & TRACE_GRAPH_PRINT_OVERHEAD) { 307 ret = trace_seq_printf(s, "==========>");
267 ret = trace_seq_printf(s, " "); 308 else
268 if (!ret) 309 ret = trace_seq_printf(s, "<==========");
269 return TRACE_TYPE_PARTIAL_LINE; 310
270 } 311 if (!ret)
312 return TRACE_TYPE_PARTIAL_LINE;
313
314 /* Don't close the duration column if haven't one */
315 if (tracer_flags.val & TRACE_GRAPH_PRINT_DURATION)
316 trace_seq_printf(s, " |");
317 ret = trace_seq_printf(s, "\n");
271 318
272 ret = trace_seq_printf(s, "<========== |\n");
273 }
274 if (!ret) 319 if (!ret)
275 return TRACE_TYPE_PARTIAL_LINE; 320 return TRACE_TYPE_PARTIAL_LINE;
276 return TRACE_TYPE_HANDLED; 321 return TRACE_TYPE_HANDLED;
@@ -289,7 +334,7 @@ print_graph_duration(unsigned long long duration, struct trace_seq *s)
289 sprintf(msecs_str, "%lu", (unsigned long) duration); 334 sprintf(msecs_str, "%lu", (unsigned long) duration);
290 335
291 /* Print msecs */ 336 /* Print msecs */
292 ret = trace_seq_printf(s, msecs_str); 337 ret = trace_seq_printf(s, "%s", msecs_str);
293 if (!ret) 338 if (!ret)
294 return TRACE_TYPE_PARTIAL_LINE; 339 return TRACE_TYPE_PARTIAL_LINE;
295 340
@@ -322,19 +367,15 @@ print_graph_duration(unsigned long long duration, struct trace_seq *s)
322 367
323} 368}
324 369
325/* Signal a overhead of time execution to the output */ 370static int print_graph_abs_time(u64 t, struct trace_seq *s)
326static int
327print_graph_overhead(unsigned long long duration, struct trace_seq *s)
328{ 371{
329 /* Duration exceeded 100 msecs */ 372 unsigned long usecs_rem;
330 if (duration > 100000ULL)
331 return trace_seq_printf(s, "! ");
332 373
333 /* Duration exceeded 10 msecs */ 374 usecs_rem = do_div(t, 1000000000);
334 if (duration > 10000ULL) 375 usecs_rem /= 1000;
335 return trace_seq_printf(s, "+ ");
336 376
337 return trace_seq_printf(s, " "); 377 return trace_seq_printf(s, "%5lu.%06lu | ",
378 (unsigned long)t, usecs_rem);
338} 379}
339 380
340/* Case of a leaf function on its call entry */ 381/* Case of a leaf function on its call entry */
@@ -357,16 +398,16 @@ print_graph_entry_leaf(struct trace_iterator *iter,
357 duration = graph_ret->rettime - graph_ret->calltime; 398 duration = graph_ret->rettime - graph_ret->calltime;
358 399
359 /* Overhead */ 400 /* Overhead */
360 if (tracer_flags.val & TRACE_GRAPH_PRINT_OVERHEAD) { 401 ret = print_graph_overhead(duration, s);
361 ret = print_graph_overhead(duration, s); 402 if (!ret)
362 if (!ret) 403 return TRACE_TYPE_PARTIAL_LINE;
363 return TRACE_TYPE_PARTIAL_LINE;
364 }
365 404
366 /* Duration */ 405 /* Duration */
367 ret = print_graph_duration(duration, s); 406 if (tracer_flags.val & TRACE_GRAPH_PRINT_DURATION) {
368 if (ret == TRACE_TYPE_PARTIAL_LINE) 407 ret = print_graph_duration(duration, s);
369 return TRACE_TYPE_PARTIAL_LINE; 408 if (ret == TRACE_TYPE_PARTIAL_LINE)
409 return TRACE_TYPE_PARTIAL_LINE;
410 }
370 411
371 /* Function */ 412 /* Function */
372 for (i = 0; i < call->depth * TRACE_GRAPH_INDENT; i++) { 413 for (i = 0; i < call->depth * TRACE_GRAPH_INDENT; i++) {
@@ -395,25 +436,17 @@ print_graph_entry_nested(struct ftrace_graph_ent_entry *entry,
395 struct ftrace_graph_ent *call = &entry->graph_ent; 436 struct ftrace_graph_ent *call = &entry->graph_ent;
396 437
397 /* No overhead */ 438 /* No overhead */
398 if (tracer_flags.val & TRACE_GRAPH_PRINT_OVERHEAD) { 439 ret = print_graph_overhead(-1, s);
399 ret = trace_seq_printf(s, " "); 440 if (!ret)
400 if (!ret) 441 return TRACE_TYPE_PARTIAL_LINE;
401 return TRACE_TYPE_PARTIAL_LINE;
402 }
403 442
404 /* Interrupt */ 443 /* No time */
405 ret = print_graph_irq(s, call->func, TRACE_GRAPH_ENT, cpu, pid); 444 if (tracer_flags.val & TRACE_GRAPH_PRINT_DURATION) {
406 if (ret == TRACE_TYPE_UNHANDLED) {
407 /* No time */
408 ret = trace_seq_printf(s, " | "); 445 ret = trace_seq_printf(s, " | ");
409 if (!ret) 446 if (!ret)
410 return TRACE_TYPE_PARTIAL_LINE; 447 return TRACE_TYPE_PARTIAL_LINE;
411 } else {
412 if (ret == TRACE_TYPE_PARTIAL_LINE)
413 return TRACE_TYPE_PARTIAL_LINE;
414 } 448 }
415 449
416
417 /* Function */ 450 /* Function */
418 for (i = 0; i < call->depth * TRACE_GRAPH_INDENT; i++) { 451 for (i = 0; i < call->depth * TRACE_GRAPH_INDENT; i++) {
419 ret = trace_seq_printf(s, " "); 452 ret = trace_seq_printf(s, " ");
@@ -434,15 +467,30 @@ print_graph_entry_nested(struct ftrace_graph_ent_entry *entry,
434 467
435static enum print_line_t 468static enum print_line_t
436print_graph_entry(struct ftrace_graph_ent_entry *field, struct trace_seq *s, 469print_graph_entry(struct ftrace_graph_ent_entry *field, struct trace_seq *s,
437 struct trace_iterator *iter, int cpu) 470 struct trace_iterator *iter)
438{ 471{
439 int ret; 472 int ret;
473 int cpu = iter->cpu;
474 pid_t *last_entry = iter->private;
440 struct trace_entry *ent = iter->ent; 475 struct trace_entry *ent = iter->ent;
476 struct ftrace_graph_ent *call = &field->graph_ent;
441 477
442 /* Pid */ 478 /* Pid */
443 if (verif_pid(s, ent->pid, cpu) == TRACE_TYPE_PARTIAL_LINE) 479 if (verif_pid(s, ent->pid, cpu, last_entry) == TRACE_TYPE_PARTIAL_LINE)
480 return TRACE_TYPE_PARTIAL_LINE;
481
482 /* Interrupt */
483 ret = print_graph_irq(s, call->func, TRACE_GRAPH_ENT, cpu, ent->pid);
484 if (ret == TRACE_TYPE_PARTIAL_LINE)
444 return TRACE_TYPE_PARTIAL_LINE; 485 return TRACE_TYPE_PARTIAL_LINE;
445 486
487 /* Absolute time */
488 if (tracer_flags.val & TRACE_GRAPH_PRINT_ABS_TIME) {
489 ret = print_graph_abs_time(iter->ts, s);
490 if (!ret)
491 return TRACE_TYPE_PARTIAL_LINE;
492 }
493
446 /* Cpu */ 494 /* Cpu */
447 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU) { 495 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU) {
448 ret = print_graph_cpu(s, cpu); 496 ret = print_graph_cpu(s, cpu);
@@ -470,16 +518,25 @@ print_graph_entry(struct ftrace_graph_ent_entry *field, struct trace_seq *s,
470 518
471static enum print_line_t 519static enum print_line_t
472print_graph_return(struct ftrace_graph_ret *trace, struct trace_seq *s, 520print_graph_return(struct ftrace_graph_ret *trace, struct trace_seq *s,
473 struct trace_entry *ent, int cpu) 521 struct trace_entry *ent, struct trace_iterator *iter)
474{ 522{
475 int i; 523 int i;
476 int ret; 524 int ret;
525 int cpu = iter->cpu;
526 pid_t *last_pid = iter->private;
477 unsigned long long duration = trace->rettime - trace->calltime; 527 unsigned long long duration = trace->rettime - trace->calltime;
478 528
479 /* Pid */ 529 /* Pid */
480 if (verif_pid(s, ent->pid, cpu) == TRACE_TYPE_PARTIAL_LINE) 530 if (verif_pid(s, ent->pid, cpu, last_pid) == TRACE_TYPE_PARTIAL_LINE)
481 return TRACE_TYPE_PARTIAL_LINE; 531 return TRACE_TYPE_PARTIAL_LINE;
482 532
533 /* Absolute time */
534 if (tracer_flags.val & TRACE_GRAPH_PRINT_ABS_TIME) {
535 ret = print_graph_abs_time(iter->ts, s);
536 if (!ret)
537 return TRACE_TYPE_PARTIAL_LINE;
538 }
539
483 /* Cpu */ 540 /* Cpu */
484 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU) { 541 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU) {
485 ret = print_graph_cpu(s, cpu); 542 ret = print_graph_cpu(s, cpu);
@@ -499,16 +556,16 @@ print_graph_return(struct ftrace_graph_ret *trace, struct trace_seq *s,
499 } 556 }
500 557
501 /* Overhead */ 558 /* Overhead */
502 if (tracer_flags.val & TRACE_GRAPH_PRINT_OVERHEAD) { 559 ret = print_graph_overhead(duration, s);
503 ret = print_graph_overhead(duration, s); 560 if (!ret)
504 if (!ret) 561 return TRACE_TYPE_PARTIAL_LINE;
505 return TRACE_TYPE_PARTIAL_LINE;
506 }
507 562
508 /* Duration */ 563 /* Duration */
509 ret = print_graph_duration(duration, s); 564 if (tracer_flags.val & TRACE_GRAPH_PRINT_DURATION) {
510 if (ret == TRACE_TYPE_PARTIAL_LINE) 565 ret = print_graph_duration(duration, s);
511 return TRACE_TYPE_PARTIAL_LINE; 566 if (ret == TRACE_TYPE_PARTIAL_LINE)
567 return TRACE_TYPE_PARTIAL_LINE;
568 }
512 569
513 /* Closing brace */ 570 /* Closing brace */
514 for (i = 0; i < trace->depth * TRACE_GRAPH_INDENT; i++) { 571 for (i = 0; i < trace->depth * TRACE_GRAPH_INDENT; i++) {
@@ -542,14 +599,23 @@ print_graph_comment(struct print_entry *trace, struct trace_seq *s,
542{ 599{
543 int i; 600 int i;
544 int ret; 601 int ret;
602 int cpu = iter->cpu;
603 pid_t *last_pid = iter->private;
604
605 /* Absolute time */
606 if (tracer_flags.val & TRACE_GRAPH_PRINT_ABS_TIME) {
607 ret = print_graph_abs_time(iter->ts, s);
608 if (!ret)
609 return TRACE_TYPE_PARTIAL_LINE;
610 }
545 611
546 /* Pid */ 612 /* Pid */
547 if (verif_pid(s, ent->pid, iter->cpu) == TRACE_TYPE_PARTIAL_LINE) 613 if (verif_pid(s, ent->pid, cpu, last_pid) == TRACE_TYPE_PARTIAL_LINE)
548 return TRACE_TYPE_PARTIAL_LINE; 614 return TRACE_TYPE_PARTIAL_LINE;
549 615
550 /* Cpu */ 616 /* Cpu */
551 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU) { 617 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU) {
552 ret = print_graph_cpu(s, iter->cpu); 618 ret = print_graph_cpu(s, cpu);
553 if (ret == TRACE_TYPE_PARTIAL_LINE) 619 if (ret == TRACE_TYPE_PARTIAL_LINE)
554 return TRACE_TYPE_PARTIAL_LINE; 620 return TRACE_TYPE_PARTIAL_LINE;
555 } 621 }
@@ -566,17 +632,17 @@ print_graph_comment(struct print_entry *trace, struct trace_seq *s,
566 } 632 }
567 633
568 /* No overhead */ 634 /* No overhead */
569 if (tracer_flags.val & TRACE_GRAPH_PRINT_OVERHEAD) { 635 ret = print_graph_overhead(-1, s);
570 ret = trace_seq_printf(s, " "); 636 if (!ret)
637 return TRACE_TYPE_PARTIAL_LINE;
638
639 /* No time */
640 if (tracer_flags.val & TRACE_GRAPH_PRINT_DURATION) {
641 ret = trace_seq_printf(s, " | ");
571 if (!ret) 642 if (!ret)
572 return TRACE_TYPE_PARTIAL_LINE; 643 return TRACE_TYPE_PARTIAL_LINE;
573 } 644 }
574 645
575 /* No time */
576 ret = trace_seq_printf(s, " | ");
577 if (!ret)
578 return TRACE_TYPE_PARTIAL_LINE;
579
580 /* Indentation */ 646 /* Indentation */
581 if (trace->depth > 0) 647 if (trace->depth > 0)
582 for (i = 0; i < (trace->depth + 1) * TRACE_GRAPH_INDENT; i++) { 648 for (i = 0; i < (trace->depth + 1) * TRACE_GRAPH_INDENT; i++) {
@@ -614,13 +680,12 @@ print_graph_function(struct trace_iterator *iter)
614 case TRACE_GRAPH_ENT: { 680 case TRACE_GRAPH_ENT: {
615 struct ftrace_graph_ent_entry *field; 681 struct ftrace_graph_ent_entry *field;
616 trace_assign_type(field, entry); 682 trace_assign_type(field, entry);
617 return print_graph_entry(field, s, iter, 683 return print_graph_entry(field, s, iter);
618 iter->cpu);
619 } 684 }
620 case TRACE_GRAPH_RET: { 685 case TRACE_GRAPH_RET: {
621 struct ftrace_graph_ret_entry *field; 686 struct ftrace_graph_ret_entry *field;
622 trace_assign_type(field, entry); 687 trace_assign_type(field, entry);
623 return print_graph_return(&field->ret, s, entry, iter->cpu); 688 return print_graph_return(&field->ret, s, entry, iter);
624 } 689 }
625 case TRACE_PRINT: { 690 case TRACE_PRINT: {
626 struct print_entry *field; 691 struct print_entry *field;
@@ -636,28 +701,55 @@ static void print_graph_headers(struct seq_file *s)
636{ 701{
637 /* 1st line */ 702 /* 1st line */
638 seq_printf(s, "# "); 703 seq_printf(s, "# ");
704 if (tracer_flags.val & TRACE_GRAPH_PRINT_ABS_TIME)
705 seq_printf(s, " TIME ");
639 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU) 706 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU)
640 seq_printf(s, "CPU "); 707 seq_printf(s, "CPU");
641 if (tracer_flags.val & TRACE_GRAPH_PRINT_PROC) 708 if (tracer_flags.val & TRACE_GRAPH_PRINT_PROC)
642 seq_printf(s, "TASK/PID "); 709 seq_printf(s, " TASK/PID ");
643 if (tracer_flags.val & TRACE_GRAPH_PRINT_OVERHEAD) 710 if (tracer_flags.val & TRACE_GRAPH_PRINT_DURATION)
644 seq_printf(s, "OVERHEAD/"); 711 seq_printf(s, " DURATION ");
645 seq_printf(s, "DURATION FUNCTION CALLS\n"); 712 seq_printf(s, " FUNCTION CALLS\n");
646 713
647 /* 2nd line */ 714 /* 2nd line */
648 seq_printf(s, "# "); 715 seq_printf(s, "# ");
716 if (tracer_flags.val & TRACE_GRAPH_PRINT_ABS_TIME)
717 seq_printf(s, " | ");
649 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU) 718 if (tracer_flags.val & TRACE_GRAPH_PRINT_CPU)
650 seq_printf(s, "| "); 719 seq_printf(s, "| ");
651 if (tracer_flags.val & TRACE_GRAPH_PRINT_PROC) 720 if (tracer_flags.val & TRACE_GRAPH_PRINT_PROC)
652 seq_printf(s, "| | "); 721 seq_printf(s, " | | ");
653 if (tracer_flags.val & TRACE_GRAPH_PRINT_OVERHEAD) { 722 if (tracer_flags.val & TRACE_GRAPH_PRINT_DURATION)
654 seq_printf(s, "| "); 723 seq_printf(s, " | | ");
655 seq_printf(s, "| | | | |\n"); 724 seq_printf(s, " | | | |\n");
656 } else
657 seq_printf(s, " | | | | |\n");
658} 725}
726
727static void graph_trace_open(struct trace_iterator *iter)
728{
729 /* pid on the last trace processed */
730 pid_t *last_pid = alloc_percpu(pid_t);
731 int cpu;
732
733 if (!last_pid)
734 pr_warning("function graph tracer: not enough memory\n");
735 else
736 for_each_possible_cpu(cpu) {
737 pid_t *pid = per_cpu_ptr(last_pid, cpu);
738 *pid = -1;
739 }
740
741 iter->private = last_pid;
742}
743
744static void graph_trace_close(struct trace_iterator *iter)
745{
746 percpu_free(iter->private);
747}
748
659static struct tracer graph_trace __read_mostly = { 749static struct tracer graph_trace __read_mostly = {
660 .name = "function_graph", 750 .name = "function_graph",
751 .open = graph_trace_open,
752 .close = graph_trace_close,
661 .init = graph_trace_init, 753 .init = graph_trace_init,
662 .reset = graph_trace_reset, 754 .reset = graph_trace_reset,
663 .print_line = print_graph_function, 755 .print_line = print_graph_function,