diff options
Diffstat (limited to 'tools/perf/builtin-report.c')
| -rw-r--r-- | tools/perf/builtin-report.c | 63 |
1 files changed, 16 insertions, 47 deletions
diff --git a/tools/perf/builtin-report.c b/tools/perf/builtin-report.c index 3ed3baf96ffb..0e83ffcbe55a 100644 --- a/tools/perf/builtin-report.c +++ b/tools/perf/builtin-report.c | |||
| @@ -407,9 +407,9 @@ static int call__match(struct symbol *sym) | |||
| 407 | return 0; | 407 | return 0; |
| 408 | } | 408 | } |
| 409 | 409 | ||
| 410 | static struct symbol ** | 410 | static struct symbol **resolve_callchain(struct thread *thread, struct map *map, |
| 411 | resolve_callchain(struct thread *thread, struct map *map, | 411 | struct ip_callchain *chain, |
| 412 | struct ip_callchain *chain, struct hist_entry *entry) | 412 | struct symbol **parent) |
| 413 | { | 413 | { |
| 414 | u64 context = PERF_CONTEXT_MAX; | 414 | u64 context = PERF_CONTEXT_MAX; |
| 415 | struct symbol **syms = NULL; | 415 | struct symbol **syms = NULL; |
| @@ -444,9 +444,8 @@ resolve_callchain(struct thread *thread, struct map *map, | |||
| 444 | } | 444 | } |
| 445 | 445 | ||
| 446 | if (sym) { | 446 | if (sym) { |
| 447 | if (sort__has_parent && call__match(sym) && | 447 | if (sort__has_parent && !*parent && call__match(sym)) |
| 448 | !entry->parent) | 448 | *parent = sym; |
| 449 | entry->parent = sym; | ||
| 450 | if (!callchain) | 449 | if (!callchain) |
| 451 | break; | 450 | break; |
| 452 | syms[i] = sym; | 451 | syms[i] = sym; |
| @@ -465,57 +464,27 @@ hist_entry__add(struct thread *thread, struct map *map, | |||
| 465 | struct symbol *sym, u64 ip, struct ip_callchain *chain, | 464 | struct symbol *sym, u64 ip, struct ip_callchain *chain, |
| 466 | char level, u64 count) | 465 | char level, u64 count) |
| 467 | { | 466 | { |
| 468 | struct rb_node **p = &hist.rb_node; | 467 | struct symbol **syms = NULL, *parent = NULL; |
| 469 | struct rb_node *parent = NULL; | 468 | bool hit; |
| 470 | struct hist_entry *he; | 469 | struct hist_entry *he; |
| 471 | struct symbol **syms = NULL; | ||
| 472 | struct hist_entry entry = { | ||
| 473 | .thread = thread, | ||
| 474 | .map = map, | ||
| 475 | .sym = sym, | ||
| 476 | .ip = ip, | ||
| 477 | .level = level, | ||
| 478 | .count = count, | ||
| 479 | .parent = NULL, | ||
| 480 | .sorted_chain = RB_ROOT | ||
| 481 | }; | ||
| 482 | int cmp; | ||
| 483 | 470 | ||
| 484 | if ((sort__has_parent || callchain) && chain) | 471 | if ((sort__has_parent || callchain) && chain) |
| 485 | syms = resolve_callchain(thread, map, chain, &entry); | 472 | syms = resolve_callchain(thread, map, chain, &parent); |
| 486 | |||
| 487 | while (*p != NULL) { | ||
| 488 | parent = *p; | ||
| 489 | he = rb_entry(parent, struct hist_entry, rb_node); | ||
| 490 | |||
| 491 | cmp = hist_entry__cmp(&entry, he); | ||
| 492 | 473 | ||
| 493 | if (!cmp) { | 474 | he = __hist_entry__add(thread, map, sym, parent, |
| 494 | he->count += count; | 475 | ip, count, level, &hit); |
| 495 | if (callchain) { | 476 | if (he == NULL) |
| 496 | append_chain(&he->callchain, chain, syms); | 477 | return -ENOMEM; |
| 497 | free(syms); | ||
| 498 | } | ||
| 499 | return 0; | ||
| 500 | } | ||
| 501 | 478 | ||
| 502 | if (cmp < 0) | 479 | if (hit) |
| 503 | p = &(*p)->rb_left; | 480 | he->count += count; |
| 504 | else | ||
| 505 | p = &(*p)->rb_right; | ||
| 506 | } | ||
| 507 | 481 | ||
| 508 | he = malloc(sizeof(*he)); | ||
| 509 | if (!he) | ||
| 510 | return -ENOMEM; | ||
| 511 | *he = entry; | ||
| 512 | if (callchain) { | 482 | if (callchain) { |
| 513 | callchain_init(&he->callchain); | 483 | if (!hit) |
| 484 | callchain_init(&he->callchain); | ||
| 514 | append_chain(&he->callchain, chain, syms); | 485 | append_chain(&he->callchain, chain, syms); |
| 515 | free(syms); | 486 | free(syms); |
| 516 | } | 487 | } |
| 517 | rb_link_node(&he->rb_node, parent, p); | ||
| 518 | rb_insert_color(&he->rb_node, &hist); | ||
| 519 | 488 | ||
| 520 | return 0; | 489 | return 0; |
| 521 | } | 490 | } |
