1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0 286470930SIngo Molnar /* 386470930SIngo Molnar * builtin-annotate.c 486470930SIngo Molnar * 586470930SIngo Molnar * Builtin annotate command: Analyze the perf.data input file, 686470930SIngo Molnar * look up and read DSOs and symbol information and display 786470930SIngo Molnar * a histogram of results, along various sorting keys. 886470930SIngo Molnar */ 986470930SIngo Molnar #include "builtin.h" 1086470930SIngo Molnar 1186470930SIngo Molnar #include "util/util.h" 1286470930SIngo Molnar #include "util/color.h" 135da50258SArnaldo Carvalho de Melo #include <linux/list.h> 1486470930SIngo Molnar #include "util/cache.h" 1543cbcd8aSArnaldo Carvalho de Melo #include <linux/rbtree.h> 1686470930SIngo Molnar #include "util/symbol.h" 1786470930SIngo Molnar 1886470930SIngo Molnar #include "perf.h" 198f28827aSFrederic Weisbecker #include "util/debug.h" 2086470930SIngo Molnar 21e248de33SArnaldo Carvalho de Melo #include "util/evlist.h" 22e248de33SArnaldo Carvalho de Melo #include "util/evsel.h" 2378f7defeSArnaldo Carvalho de Melo #include "util/annotate.h" 2462daacb5SArnaldo Carvalho de Melo #include "util/event.h" 254b6ab94eSJosh Poimboeuf #include <subcmd/parse-options.h> 2686470930SIngo Molnar #include "util/parse-events.h" 276baa0a5aSFrederic Weisbecker #include "util/thread.h" 28dd68ada2SJohn Kacur #include "util/sort.h" 293d1d07ecSJohn Kacur #include "util/hist.h" 3094c744b6SArnaldo Carvalho de Melo #include "util/session.h" 3145694aa7SArnaldo Carvalho de Melo #include "util/tool.h" 32f5fc1412SJiri Olsa #include "util/data.h" 3368e94f4eSIrina Tirdea #include "arch/common.h" 3470fbe057SPeter Zijlstra #include "util/block-range.h" 3586470930SIngo Molnar 36fc67297bSNamhyung Kim #include <dlfcn.h> 37a43783aeSArnaldo Carvalho de Melo #include <errno.h> 385d67be97SAnton Blanchard #include <linux/bitmap.h> 395d67be97SAnton Blanchard 40d20deb64SArnaldo Carvalho de Melo struct perf_annotate { 4145694aa7SArnaldo Carvalho de Melo struct perf_tool tool; 42fa10f316SNamhyung Kim struct perf_session *session; 43befd2a38SArnaldo Carvalho de Melo bool use_tui, use_stdio, use_stdio2, use_gtk; 447009cc34SArnaldo Carvalho de Melo bool full_paths; 457009cc34SArnaldo Carvalho de Melo bool print_line; 4618c9e5c5SNamhyung Kim bool skip_missing; 47bb848c14SJin Yao bool has_br_stack; 487009cc34SArnaldo Carvalho de Melo const char *sym_hist_filter; 497009cc34SArnaldo Carvalho de Melo const char *cpu_list; 507009cc34SArnaldo Carvalho de Melo DECLARE_BITMAP(cpu_bitmap, MAX_NR_CPUS); 51d20deb64SArnaldo Carvalho de Melo }; 525d67be97SAnton Blanchard 5370fbe057SPeter Zijlstra /* 5470fbe057SPeter Zijlstra * Given one basic block: 5570fbe057SPeter Zijlstra * 5670fbe057SPeter Zijlstra * from to branch_i 5770fbe057SPeter Zijlstra * * ----> * 5870fbe057SPeter Zijlstra * | 5970fbe057SPeter Zijlstra * | block 6070fbe057SPeter Zijlstra * v 6170fbe057SPeter Zijlstra * * ----> * 6270fbe057SPeter Zijlstra * from to branch_i+1 6370fbe057SPeter Zijlstra * 6470fbe057SPeter Zijlstra * where the horizontal are the branches and the vertical is the executed 6570fbe057SPeter Zijlstra * block of instructions. 6670fbe057SPeter Zijlstra * 6770fbe057SPeter Zijlstra * We count, for each 'instruction', the number of blocks that covered it as 6870fbe057SPeter Zijlstra * well as count the ratio each branch is taken. 6970fbe057SPeter Zijlstra * 7070fbe057SPeter Zijlstra * We can do this without knowing the actual instruction stream by keeping 7170fbe057SPeter Zijlstra * track of the address ranges. We break down ranges such that there is no 7270fbe057SPeter Zijlstra * overlap and iterate from the start until the end. 7370fbe057SPeter Zijlstra * 7470fbe057SPeter Zijlstra * @acme: once we parse the objdump output _before_ processing the samples, 7570fbe057SPeter Zijlstra * we can easily fold the branch.cycles IPC bits in. 7670fbe057SPeter Zijlstra */ 7770fbe057SPeter Zijlstra static void process_basic_block(struct addr_map_symbol *start, 7870fbe057SPeter Zijlstra struct addr_map_symbol *end, 7970fbe057SPeter Zijlstra struct branch_flags *flags) 8070fbe057SPeter Zijlstra { 8170fbe057SPeter Zijlstra struct symbol *sym = start->sym; 8270fbe057SPeter Zijlstra struct annotation *notes = sym ? symbol__annotation(sym) : NULL; 8370fbe057SPeter Zijlstra struct block_range_iter iter; 8470fbe057SPeter Zijlstra struct block_range *entry; 8570fbe057SPeter Zijlstra 8670fbe057SPeter Zijlstra /* 8770fbe057SPeter Zijlstra * Sanity; NULL isn't executable and the CPU cannot execute backwards 8870fbe057SPeter Zijlstra */ 8970fbe057SPeter Zijlstra if (!start->addr || start->addr > end->addr) 9070fbe057SPeter Zijlstra return; 9170fbe057SPeter Zijlstra 9270fbe057SPeter Zijlstra iter = block_range__create(start->addr, end->addr); 9370fbe057SPeter Zijlstra if (!block_range_iter__valid(&iter)) 9470fbe057SPeter Zijlstra return; 9570fbe057SPeter Zijlstra 9670fbe057SPeter Zijlstra /* 9770fbe057SPeter Zijlstra * First block in range is a branch target. 9870fbe057SPeter Zijlstra */ 9970fbe057SPeter Zijlstra entry = block_range_iter(&iter); 10070fbe057SPeter Zijlstra assert(entry->is_target); 10170fbe057SPeter Zijlstra entry->entry++; 10270fbe057SPeter Zijlstra 10370fbe057SPeter Zijlstra do { 10470fbe057SPeter Zijlstra entry = block_range_iter(&iter); 10570fbe057SPeter Zijlstra 10670fbe057SPeter Zijlstra entry->coverage++; 10770fbe057SPeter Zijlstra entry->sym = sym; 10870fbe057SPeter Zijlstra 10970fbe057SPeter Zijlstra if (notes) 11070fbe057SPeter Zijlstra notes->max_coverage = max(notes->max_coverage, entry->coverage); 11170fbe057SPeter Zijlstra 11270fbe057SPeter Zijlstra } while (block_range_iter__next(&iter)); 11370fbe057SPeter Zijlstra 11470fbe057SPeter Zijlstra /* 11570fbe057SPeter Zijlstra * Last block in rage is a branch. 11670fbe057SPeter Zijlstra */ 11770fbe057SPeter Zijlstra entry = block_range_iter(&iter); 11870fbe057SPeter Zijlstra assert(entry->is_branch); 11970fbe057SPeter Zijlstra entry->taken++; 12070fbe057SPeter Zijlstra if (flags->predicted) 12170fbe057SPeter Zijlstra entry->pred++; 12270fbe057SPeter Zijlstra } 12370fbe057SPeter Zijlstra 12470fbe057SPeter Zijlstra static void process_branch_stack(struct branch_stack *bs, struct addr_location *al, 12570fbe057SPeter Zijlstra struct perf_sample *sample) 12670fbe057SPeter Zijlstra { 12770fbe057SPeter Zijlstra struct addr_map_symbol *prev = NULL; 12870fbe057SPeter Zijlstra struct branch_info *bi; 12970fbe057SPeter Zijlstra int i; 13070fbe057SPeter Zijlstra 13170fbe057SPeter Zijlstra if (!bs || !bs->nr) 13270fbe057SPeter Zijlstra return; 13370fbe057SPeter Zijlstra 13470fbe057SPeter Zijlstra bi = sample__resolve_bstack(sample, al); 13570fbe057SPeter Zijlstra if (!bi) 13670fbe057SPeter Zijlstra return; 13770fbe057SPeter Zijlstra 13870fbe057SPeter Zijlstra for (i = bs->nr - 1; i >= 0; i--) { 13970fbe057SPeter Zijlstra /* 14070fbe057SPeter Zijlstra * XXX filter against symbol 14170fbe057SPeter Zijlstra */ 14270fbe057SPeter Zijlstra if (prev) 14370fbe057SPeter Zijlstra process_basic_block(prev, &bi[i].from, &bi[i].flags); 14470fbe057SPeter Zijlstra prev = &bi[i].to; 14570fbe057SPeter Zijlstra } 14670fbe057SPeter Zijlstra 14770fbe057SPeter Zijlstra free(bi); 14870fbe057SPeter Zijlstra } 14970fbe057SPeter Zijlstra 150bb848c14SJin Yao static int hist_iter__branch_callback(struct hist_entry_iter *iter, 151bb848c14SJin Yao struct addr_location *al __maybe_unused, 152bb848c14SJin Yao bool single __maybe_unused, 153bb848c14SJin Yao void *arg __maybe_unused) 154bb848c14SJin Yao { 155bb848c14SJin Yao struct hist_entry *he = iter->he; 156bb848c14SJin Yao struct branch_info *bi; 157bb848c14SJin Yao struct perf_sample *sample = iter->sample; 158bb848c14SJin Yao struct perf_evsel *evsel = iter->evsel; 159bb848c14SJin Yao int err; 160bb848c14SJin Yao 161bb848c14SJin Yao hist__account_cycles(sample->branch_stack, al, sample, false); 162bb848c14SJin Yao 163bb848c14SJin Yao bi = he->branch_info; 164bb848c14SJin Yao err = addr_map_symbol__inc_samples(&bi->from, sample, evsel->idx); 165bb848c14SJin Yao 166bb848c14SJin Yao if (err) 167bb848c14SJin Yao goto out; 168bb848c14SJin Yao 169bb848c14SJin Yao err = addr_map_symbol__inc_samples(&bi->to, sample, evsel->idx); 170bb848c14SJin Yao 171bb848c14SJin Yao out: 172bb848c14SJin Yao return err; 173bb848c14SJin Yao } 174bb848c14SJin Yao 175bb848c14SJin Yao static int process_branch_callback(struct perf_evsel *evsel, 176bb848c14SJin Yao struct perf_sample *sample, 177bb848c14SJin Yao struct addr_location *al __maybe_unused, 178bb848c14SJin Yao struct perf_annotate *ann, 179bb848c14SJin Yao struct machine *machine) 180bb848c14SJin Yao { 181bb848c14SJin Yao struct hist_entry_iter iter = { 182bb848c14SJin Yao .evsel = evsel, 183bb848c14SJin Yao .sample = sample, 184bb848c14SJin Yao .add_entry_cb = hist_iter__branch_callback, 185bb848c14SJin Yao .hide_unresolved = symbol_conf.hide_unresolved, 186bb848c14SJin Yao .ops = &hist_iter_branch, 187bb848c14SJin Yao }; 188bb848c14SJin Yao 189bb848c14SJin Yao struct addr_location a; 190bb848c14SJin Yao int ret; 191bb848c14SJin Yao 192bb848c14SJin Yao if (machine__resolve(machine, &a, sample) < 0) 193bb848c14SJin Yao return -1; 194bb848c14SJin Yao 195bb848c14SJin Yao if (a.sym == NULL) 196bb848c14SJin Yao return 0; 197bb848c14SJin Yao 198bb848c14SJin Yao if (a.map != NULL) 199bb848c14SJin Yao a.map->dso->hit = 1; 200bb848c14SJin Yao 201bb848c14SJin Yao ret = hist_entry_iter__add(&iter, &a, PERF_MAX_STACK_DEPTH, ann); 202bb848c14SJin Yao return ret; 203bb848c14SJin Yao } 204bb848c14SJin Yao 205befd2a38SArnaldo Carvalho de Melo static bool has_annotation(struct perf_annotate *ann) 206befd2a38SArnaldo Carvalho de Melo { 207befd2a38SArnaldo Carvalho de Melo return ui__has_annotation() || ann->use_stdio2; 208befd2a38SArnaldo Carvalho de Melo } 209befd2a38SArnaldo Carvalho de Melo 210d04b35f8SArnaldo Carvalho de Melo static int perf_evsel__add_sample(struct perf_evsel *evsel, 211fd36f3ddSNamhyung Kim struct perf_sample *sample, 212d20deb64SArnaldo Carvalho de Melo struct addr_location *al, 213bb848c14SJin Yao struct perf_annotate *ann, 214bb848c14SJin Yao struct machine *machine) 21586470930SIngo Molnar { 2164ea062edSArnaldo Carvalho de Melo struct hists *hists = evsel__hists(evsel); 217628ada0cSArnaldo Carvalho de Melo struct hist_entry *he; 218e248de33SArnaldo Carvalho de Melo int ret; 219628ada0cSArnaldo Carvalho de Melo 220befd2a38SArnaldo Carvalho de Melo if ((!ann->has_br_stack || !has_annotation(ann)) && 221bb848c14SJin Yao ann->sym_hist_filter != NULL && 2227009cc34SArnaldo Carvalho de Melo (al->sym == NULL || 2237009cc34SArnaldo Carvalho de Melo strcmp(ann->sym_hist_filter, al->sym->name) != 0)) { 224628ada0cSArnaldo Carvalho de Melo /* We're only interested in a symbol named sym_hist_filter */ 225facf3f06SArnaldo Carvalho de Melo /* 226facf3f06SArnaldo Carvalho de Melo * FIXME: why isn't this done in the symbol_filter when loading 227facf3f06SArnaldo Carvalho de Melo * the DSO? 228facf3f06SArnaldo Carvalho de Melo */ 229628ada0cSArnaldo Carvalho de Melo if (al->sym != NULL) { 230628ada0cSArnaldo Carvalho de Melo rb_erase(&al->sym->rb_node, 231628ada0cSArnaldo Carvalho de Melo &al->map->dso->symbols[al->map->type]); 232628ada0cSArnaldo Carvalho de Melo symbol__delete(al->sym); 233c0b4dffbSArnaldo Carvalho de Melo dso__reset_find_symbol_cache(al->map->dso); 234628ada0cSArnaldo Carvalho de Melo } 235628ada0cSArnaldo Carvalho de Melo return 0; 236628ada0cSArnaldo Carvalho de Melo } 237628ada0cSArnaldo Carvalho de Melo 23870fbe057SPeter Zijlstra /* 23970fbe057SPeter Zijlstra * XXX filtered samples can still have branch entires pointing into our 24070fbe057SPeter Zijlstra * symbol and are missed. 24170fbe057SPeter Zijlstra */ 24270fbe057SPeter Zijlstra process_branch_stack(sample->branch_stack, al, sample); 24370fbe057SPeter Zijlstra 244befd2a38SArnaldo Carvalho de Melo if (ann->has_br_stack && has_annotation(ann)) 245bb848c14SJin Yao return process_branch_callback(evsel, sample, al, ann, machine); 246bb848c14SJin Yao 2470102ef3eSJiri Olsa he = hists__add_entry(hists, al, NULL, NULL, NULL, sample, true); 2489735abf1SArnaldo Carvalho de Melo if (he == NULL) 24986470930SIngo Molnar return -ENOMEM; 250628ada0cSArnaldo Carvalho de Melo 251bab89f6aSTaeung Song ret = hist_entry__inc_addr_samples(he, sample, evsel->idx, al->addr); 2524ea062edSArnaldo Carvalho de Melo hists__inc_nr_samples(hists, true); 253e248de33SArnaldo Carvalho de Melo return ret; 25486470930SIngo Molnar } 25586470930SIngo Molnar 25645694aa7SArnaldo Carvalho de Melo static int process_sample_event(struct perf_tool *tool, 257d20deb64SArnaldo Carvalho de Melo union perf_event *event, 2588115d60cSArnaldo Carvalho de Melo struct perf_sample *sample, 2599e69c210SArnaldo Carvalho de Melo struct perf_evsel *evsel, 260743eb868SArnaldo Carvalho de Melo struct machine *machine) 26186470930SIngo Molnar { 26245694aa7SArnaldo Carvalho de Melo struct perf_annotate *ann = container_of(tool, struct perf_annotate, tool); 2631ed091c4SArnaldo Carvalho de Melo struct addr_location al; 264b91fc39fSArnaldo Carvalho de Melo int ret = 0; 2656baa0a5aSFrederic Weisbecker 266bb3eb566SArnaldo Carvalho de Melo if (machine__resolve(machine, &al, sample) < 0) { 26729a9f66dSArnaldo Carvalho de Melo pr_warning("problem processing %d event, skipping it.\n", 26886470930SIngo Molnar event->header.type); 26986470930SIngo Molnar return -1; 27086470930SIngo Molnar } 27186470930SIngo Molnar 2727009cc34SArnaldo Carvalho de Melo if (ann->cpu_list && !test_bit(sample->cpu, ann->cpu_bitmap)) 273b91fc39fSArnaldo Carvalho de Melo goto out_put; 2745d67be97SAnton Blanchard 275bb848c14SJin Yao if (!al.filtered && 276bb848c14SJin Yao perf_evsel__add_sample(evsel, sample, &al, ann, machine)) { 27729a9f66dSArnaldo Carvalho de Melo pr_warning("problem incrementing symbol count, " 278ec218fc4SArnaldo Carvalho de Melo "skipping event\n"); 279b91fc39fSArnaldo Carvalho de Melo ret = -1; 28086470930SIngo Molnar } 281b91fc39fSArnaldo Carvalho de Melo out_put: 282b91fc39fSArnaldo Carvalho de Melo addr_location__put(&al); 283b91fc39fSArnaldo Carvalho de Melo return ret; 28486470930SIngo Molnar } 28586470930SIngo Molnar 286db8fd07aSNamhyung Kim static int hist_entry__tty_annotate(struct hist_entry *he, 287db8fd07aSNamhyung Kim struct perf_evsel *evsel, 288d20deb64SArnaldo Carvalho de Melo struct perf_annotate *ann) 28986470930SIngo Molnar { 290befd2a38SArnaldo Carvalho de Melo if (!ann->use_stdio2) 291db8fd07aSNamhyung Kim return symbol__tty_annotate(he->ms.sym, he->ms.map, evsel, 2927009cc34SArnaldo Carvalho de Melo ann->print_line, ann->full_paths, 0, 0); 293befd2a38SArnaldo Carvalho de Melo return symbol__tty_annotate2(he->ms.sym, he->ms.map, evsel, 294befd2a38SArnaldo Carvalho de Melo ann->print_line, ann->full_paths); 29586470930SIngo Molnar } 29686470930SIngo Molnar 297c824c433SArnaldo Carvalho de Melo static void hists__find_annotations(struct hists *hists, 298db8fd07aSNamhyung Kim struct perf_evsel *evsel, 299d20deb64SArnaldo Carvalho de Melo struct perf_annotate *ann) 30086470930SIngo Molnar { 301c824c433SArnaldo Carvalho de Melo struct rb_node *nd = rb_first(&hists->entries), *next; 302cf958003SArnaldo Carvalho de Melo int key = K_RIGHT; 30386470930SIngo Molnar 30446e3e055SArnaldo Carvalho de Melo while (nd) { 305ed52ce2eSArnaldo Carvalho de Melo struct hist_entry *he = rb_entry(nd, struct hist_entry, rb_node); 30678f7defeSArnaldo Carvalho de Melo struct annotation *notes; 30786470930SIngo Molnar 30846e3e055SArnaldo Carvalho de Melo if (he->ms.sym == NULL || he->ms.map->dso->annotate_warned) 30946e3e055SArnaldo Carvalho de Melo goto find_next; 310e4204992SArnaldo Carvalho de Melo 311bb848c14SJin Yao if (ann->sym_hist_filter && 312bb848c14SJin Yao (strcmp(he->ms.sym->name, ann->sym_hist_filter) != 0)) 313bb848c14SJin Yao goto find_next; 314bb848c14SJin Yao 31578f7defeSArnaldo Carvalho de Melo notes = symbol__annotation(he->ms.sym); 316ce6f4fabSArnaldo Carvalho de Melo if (notes->src == NULL) { 31746e3e055SArnaldo Carvalho de Melo find_next: 318cf958003SArnaldo Carvalho de Melo if (key == K_LEFT) 31946e3e055SArnaldo Carvalho de Melo nd = rb_prev(nd); 32046e3e055SArnaldo Carvalho de Melo else 32146e3e055SArnaldo Carvalho de Melo nd = rb_next(nd); 322e4204992SArnaldo Carvalho de Melo continue; 32346e3e055SArnaldo Carvalho de Melo } 324e4204992SArnaldo Carvalho de Melo 3252b676bf0SNamhyung Kim if (use_browser == 2) { 32618c9e5c5SNamhyung Kim int ret; 327fc67297bSNamhyung Kim int (*annotate)(struct hist_entry *he, 328fc67297bSNamhyung Kim struct perf_evsel *evsel, 329fc67297bSNamhyung Kim struct hist_browser_timer *hbt); 33018c9e5c5SNamhyung Kim 331fc67297bSNamhyung Kim annotate = dlsym(perf_gtk_handle, 332fc67297bSNamhyung Kim "hist_entry__gtk_annotate"); 333fc67297bSNamhyung Kim if (annotate == NULL) { 334fc67297bSNamhyung Kim ui__error("GTK browser not found!\n"); 335fc67297bSNamhyung Kim return; 336fc67297bSNamhyung Kim } 337fc67297bSNamhyung Kim 338fc67297bSNamhyung Kim ret = annotate(he, evsel, NULL); 33918c9e5c5SNamhyung Kim if (!ret || !ann->skip_missing) 3402b676bf0SNamhyung Kim return; 34118c9e5c5SNamhyung Kim 34218c9e5c5SNamhyung Kim /* skip missing symbols */ 34318c9e5c5SNamhyung Kim nd = rb_next(nd); 3442b676bf0SNamhyung Kim } else if (use_browser == 1) { 345db8fd07aSNamhyung Kim key = hist_entry__tui_annotate(he, evsel, NULL); 346bb848c14SJin Yao 34746e3e055SArnaldo Carvalho de Melo switch (key) { 34818c9e5c5SNamhyung Kim case -1: 34918c9e5c5SNamhyung Kim if (!ann->skip_missing) 35018c9e5c5SNamhyung Kim return; 35118c9e5c5SNamhyung Kim /* fall through */ 352cf958003SArnaldo Carvalho de Melo case K_RIGHT: 353b50e003dSArnaldo Carvalho de Melo next = rb_next(nd); 35446e3e055SArnaldo Carvalho de Melo break; 355cf958003SArnaldo Carvalho de Melo case K_LEFT: 356b50e003dSArnaldo Carvalho de Melo next = rb_prev(nd); 35746e3e055SArnaldo Carvalho de Melo break; 358b50e003dSArnaldo Carvalho de Melo default: 359b50e003dSArnaldo Carvalho de Melo return; 36046e3e055SArnaldo Carvalho de Melo } 361b50e003dSArnaldo Carvalho de Melo 362b50e003dSArnaldo Carvalho de Melo if (next != NULL) 363b50e003dSArnaldo Carvalho de Melo nd = next; 36446e3e055SArnaldo Carvalho de Melo } else { 365db8fd07aSNamhyung Kim hist_entry__tty_annotate(he, evsel, ann); 36646e3e055SArnaldo Carvalho de Melo nd = rb_next(nd); 367ed52ce2eSArnaldo Carvalho de Melo /* 36846e3e055SArnaldo Carvalho de Melo * Since we have a hist_entry per IP for the same 369ce6f4fabSArnaldo Carvalho de Melo * symbol, free he->ms.sym->src to signal we already 37046e3e055SArnaldo Carvalho de Melo * processed this symbol. 371ed52ce2eSArnaldo Carvalho de Melo */ 372d4957633SAndi Kleen zfree(¬es->src->cycles_hist); 37304662523SArnaldo Carvalho de Melo zfree(¬es->src); 37486470930SIngo Molnar } 37586470930SIngo Molnar } 37646e3e055SArnaldo Carvalho de Melo } 37786470930SIngo Molnar 378d20deb64SArnaldo Carvalho de Melo static int __cmd_annotate(struct perf_annotate *ann) 37986470930SIngo Molnar { 380bab81b62SLi Zefan int ret; 381fa10f316SNamhyung Kim struct perf_session *session = ann->session; 382e248de33SArnaldo Carvalho de Melo struct perf_evsel *pos; 383e248de33SArnaldo Carvalho de Melo u64 total_nr_samples; 38494c744b6SArnaldo Carvalho de Melo 3857009cc34SArnaldo Carvalho de Melo if (ann->cpu_list) { 3867009cc34SArnaldo Carvalho de Melo ret = perf_session__cpu_bitmap(session, ann->cpu_list, 3877009cc34SArnaldo Carvalho de Melo ann->cpu_bitmap); 3885d67be97SAnton Blanchard if (ret) 389fa10f316SNamhyung Kim goto out; 3905d67be97SAnton Blanchard } 3915d67be97SAnton Blanchard 39268e94f4eSIrina Tirdea if (!objdump_path) { 393eebd0bfcSArnaldo Carvalho de Melo ret = perf_env__lookup_objdump(&session->header.env); 39468e94f4eSIrina Tirdea if (ret) 395fa10f316SNamhyung Kim goto out; 39668e94f4eSIrina Tirdea } 39768e94f4eSIrina Tirdea 398b7b61cbeSArnaldo Carvalho de Melo ret = perf_session__process_events(session); 399bab81b62SLi Zefan if (ret) 400fa10f316SNamhyung Kim goto out; 40186470930SIngo Molnar 40262daacb5SArnaldo Carvalho de Melo if (dump_trace) { 403c8446b9bSArnaldo Carvalho de Melo perf_session__fprintf_nr_events(session, stdout); 4042a1731fbSArnaldo Carvalho de Melo perf_evlist__fprintf_nr_events(session->evlist, stdout); 405fa10f316SNamhyung Kim goto out; 40662daacb5SArnaldo Carvalho de Melo } 40786470930SIngo Molnar 408da21d1b5SArnaldo Carvalho de Melo if (verbose > 3) 409b3165f41SArnaldo Carvalho de Melo perf_session__fprintf(session, stdout); 41086470930SIngo Molnar 411da21d1b5SArnaldo Carvalho de Melo if (verbose > 2) 412cbf69680SArnaldo Carvalho de Melo perf_session__fprintf_dsos(session, stdout); 41386470930SIngo Molnar 414e248de33SArnaldo Carvalho de Melo total_nr_samples = 0; 415e5cadb93SArnaldo Carvalho de Melo evlist__for_each_entry(session->evlist, pos) { 4164ea062edSArnaldo Carvalho de Melo struct hists *hists = evsel__hists(pos); 417e248de33SArnaldo Carvalho de Melo u32 nr_samples = hists->stats.nr_events[PERF_RECORD_SAMPLE]; 41886470930SIngo Molnar 419e248de33SArnaldo Carvalho de Melo if (nr_samples > 0) { 420e248de33SArnaldo Carvalho de Melo total_nr_samples += nr_samples; 421c1fb5651SNamhyung Kim hists__collapse_resort(hists, NULL); 422f9db0d0fSKan Liang /* Don't sort callchain */ 423f9db0d0fSKan Liang perf_evsel__reset_sample_bit(pos, CALLCHAIN); 424452ce03bSJiri Olsa perf_evsel__output_resort(pos, NULL); 425b1dd4432SNamhyung Kim 426b1dd4432SNamhyung Kim if (symbol_conf.event_group && 427b1dd4432SNamhyung Kim !perf_evsel__is_group_leader(pos)) 428b1dd4432SNamhyung Kim continue; 429b1dd4432SNamhyung Kim 430db8fd07aSNamhyung Kim hists__find_annotations(hists, pos, ann); 431e248de33SArnaldo Carvalho de Melo } 432e248de33SArnaldo Carvalho de Melo } 433e248de33SArnaldo Carvalho de Melo 434e248de33SArnaldo Carvalho de Melo if (total_nr_samples == 0) { 435eae8ad80SJiri Olsa ui__error("The %s file has no samples!\n", session->data->file.path); 436fa10f316SNamhyung Kim goto out; 437e248de33SArnaldo Carvalho de Melo } 4387a60ba94SNamhyung Kim 439fc67297bSNamhyung Kim if (use_browser == 2) { 440fc67297bSNamhyung Kim void (*show_annotations)(void); 441fc67297bSNamhyung Kim 442fc67297bSNamhyung Kim show_annotations = dlsym(perf_gtk_handle, 443fc67297bSNamhyung Kim "perf_gtk__show_annotations"); 444fc67297bSNamhyung Kim if (show_annotations == NULL) { 445fc67297bSNamhyung Kim ui__error("GTK browser not found!\n"); 446fa10f316SNamhyung Kim goto out; 447fc67297bSNamhyung Kim } 448fc67297bSNamhyung Kim show_annotations(); 449fc67297bSNamhyung Kim } 4507a60ba94SNamhyung Kim 451fa10f316SNamhyung Kim out: 452bab81b62SLi Zefan return ret; 45386470930SIngo Molnar } 45486470930SIngo Molnar 45586470930SIngo Molnar static const char * const annotate_usage[] = { 45699345254SNamhyung Kim "perf annotate [<options>]", 45786470930SIngo Molnar NULL 45886470930SIngo Molnar }; 45986470930SIngo Molnar 460b0ad8ea6SArnaldo Carvalho de Melo int cmd_annotate(int argc, const char **argv) 461d20deb64SArnaldo Carvalho de Melo { 462d20deb64SArnaldo Carvalho de Melo struct perf_annotate annotate = { 46345694aa7SArnaldo Carvalho de Melo .tool = { 464d20deb64SArnaldo Carvalho de Melo .sample = process_sample_event, 465d20deb64SArnaldo Carvalho de Melo .mmap = perf_event__process_mmap, 4665c5e854bSStephane Eranian .mmap2 = perf_event__process_mmap2, 467d20deb64SArnaldo Carvalho de Melo .comm = perf_event__process_comm, 468ec4622f5SArnaldo Carvalho de Melo .exit = perf_event__process_exit, 469f62d3f0fSArnaldo Carvalho de Melo .fork = perf_event__process_fork, 470f3b3614aSHari Bathini .namespaces = perf_event__process_namespaces, 4716ab11f3aSDavid Carrillo-Cisneros .attr = perf_event__process_attr, 4726ab11f3aSDavid Carrillo-Cisneros .build_id = perf_event__process_build_id, 473f4849599SDavid Carrillo-Cisneros .tracing_data = perf_event__process_tracing_data, 474e9def1b2SDavid Carrillo-Cisneros .feature = perf_event__process_feature, 4750a8cb85cSJiri Olsa .ordered_events = true, 476d20deb64SArnaldo Carvalho de Melo .ordering_requires_timestamps = true, 477d20deb64SArnaldo Carvalho de Melo }, 478d20deb64SArnaldo Carvalho de Melo }; 4798ceb41d7SJiri Olsa struct perf_data data = { 480fa10f316SNamhyung Kim .mode = PERF_DATA_MODE_READ, 481fa10f316SNamhyung Kim }; 4821ac39372STaeung Song struct option options[] = { 48370cb4e96SFeng Tang OPT_STRING('i', "input", &input_name, "file", 48486470930SIngo Molnar "input file name"), 485ac73c5a9SArnaldo Carvalho de Melo OPT_STRING('d', "dsos", &symbol_conf.dso_list_str, "dso[,dso...]", 486ac73c5a9SArnaldo Carvalho de Melo "only consider symbols in these dsos"), 4877009cc34SArnaldo Carvalho de Melo OPT_STRING('s', "symbol", &annotate.sym_hist_filter, "symbol", 48886470930SIngo Molnar "symbol to annotate"), 4898ceb41d7SJiri Olsa OPT_BOOLEAN('f', "force", &data.force, "don't complain, do it"), 490c0555642SIan Munsie OPT_INCR('v', "verbose", &verbose, 49186470930SIngo Molnar "be more verbose (show symbol address, etc)"), 492eddaef88SNamhyung Kim OPT_BOOLEAN('q', "quiet", &quiet, "do now show any message"), 49386470930SIngo Molnar OPT_BOOLEAN('D', "dump-raw-trace", &dump_trace, 49486470930SIngo Molnar "dump raw trace in ASCII"), 4952b676bf0SNamhyung Kim OPT_BOOLEAN(0, "gtk", &annotate.use_gtk, "Use the GTK interface"), 4967009cc34SArnaldo Carvalho de Melo OPT_BOOLEAN(0, "tui", &annotate.use_tui, "Use the TUI interface"), 4977009cc34SArnaldo Carvalho de Melo OPT_BOOLEAN(0, "stdio", &annotate.use_stdio, "Use the stdio interface"), 498befd2a38SArnaldo Carvalho de Melo OPT_BOOLEAN(0, "stdio2", &annotate.use_stdio2, "Use the stdio interface"), 499b32d133aSArnaldo Carvalho de Melo OPT_STRING('k', "vmlinux", &symbol_conf.vmlinux_name, 500b32d133aSArnaldo Carvalho de Melo "file", "vmlinux pathname"), 501b32d133aSArnaldo Carvalho de Melo OPT_BOOLEAN('m', "modules", &symbol_conf.use_modules, 50242976487SMike Galbraith "load module symbols - WARNING: use only with -k and LIVE kernel"), 5037009cc34SArnaldo Carvalho de Melo OPT_BOOLEAN('l', "print-line", &annotate.print_line, 504301406b9SFrederic Weisbecker "print matching source lines (may be slow)"), 5057009cc34SArnaldo Carvalho de Melo OPT_BOOLEAN('P', "full-paths", &annotate.full_paths, 50642976487SMike Galbraith "Don't shorten the displayed pathnames"), 50718c9e5c5SNamhyung Kim OPT_BOOLEAN(0, "skip-missing", &annotate.skip_missing, 50818c9e5c5SNamhyung Kim "Skip symbols that cannot be annotated"), 509c8e66720SDavid Ahern OPT_STRING('C', "cpu", &annotate.cpu_list, "cpu", "list of cpus to profile"), 510a7066709SHe Kuang OPT_CALLBACK(0, "symfs", NULL, "directory", 511a7066709SHe Kuang "Look for files with symbols relative to this directory", 512a7066709SHe Kuang symbol__config_symfs), 51364c6f0c7SArnaldo Carvalho de Melo OPT_BOOLEAN(0, "source", &symbol_conf.annotate_src, 5143e6a2a7fSStephane Eranian "Interleave source code with assembly code (default)"), 51564c6f0c7SArnaldo Carvalho de Melo OPT_BOOLEAN(0, "asm-raw", &symbol_conf.annotate_asm_raw, 5163e6a2a7fSStephane Eranian "Display raw encoding of assembly instructions (default)"), 517f69b64f7SAndi Kleen OPT_STRING('M', "disassembler-style", &disassembler_style, "disassembler style", 518f69b64f7SAndi Kleen "Specify disassembler style (e.g. -M intel for intel syntax)"), 5197a4ec938SMaciek Borzecki OPT_STRING(0, "objdump", &objdump_path, "path", 5207a4ec938SMaciek Borzecki "objdump binary to use for disassembly and annotations"), 521b1dd4432SNamhyung Kim OPT_BOOLEAN(0, "group", &symbol_conf.event_group, 522b1dd4432SNamhyung Kim "Show event group information together"), 5230c4a5bceSMartin Liška OPT_BOOLEAN(0, "show-total-period", &symbol_conf.show_total_period, 5240c4a5bceSMartin Liška "Show a column with the sum of periods"), 5251ac39372STaeung Song OPT_BOOLEAN('n', "show-nr-samples", &symbol_conf.show_nr_samples, 5261ac39372STaeung Song "Show a column with the number of samples"), 52753fe4ba1SArnaldo Carvalho de Melo OPT_CALLBACK_DEFAULT(0, "stdio-color", NULL, "mode", 52853fe4ba1SArnaldo Carvalho de Melo "'always' (default), 'never' or 'auto' only applicable to --stdio mode", 52953fe4ba1SArnaldo Carvalho de Melo stdio__config_color, "always"), 53086470930SIngo Molnar OPT_END() 53186470930SIngo Molnar }; 5321ac39372STaeung Song int ret; 533a635fc51SArnaldo Carvalho de Melo 5341ac39372STaeung Song set_option_flag(options, 0, "show-total-period", PARSE_OPT_EXCLUSIVE); 5351ac39372STaeung Song set_option_flag(options, 0, "show-nr-samples", PARSE_OPT_EXCLUSIVE); 5361ac39372STaeung Song 5371ac39372STaeung Song 5381ac39372STaeung Song ret = hists__init(); 539a635fc51SArnaldo Carvalho de Melo if (ret < 0) 540a635fc51SArnaldo Carvalho de Melo return ret; 54186470930SIngo Molnar 542655000e7SArnaldo Carvalho de Melo argc = parse_options(argc, argv, options, annotate_usage, 0); 54350e19ef9SNamhyung Kim if (argc) { 54450e19ef9SNamhyung Kim /* 54550e19ef9SNamhyung Kim * Special case: if there's an argument left then assume that 54650e19ef9SNamhyung Kim * it's a symbol filter: 54750e19ef9SNamhyung Kim */ 54850e19ef9SNamhyung Kim if (argc > 1) 54950e19ef9SNamhyung Kim usage_with_options(annotate_usage, options); 55050e19ef9SNamhyung Kim 55150e19ef9SNamhyung Kim annotate.sym_hist_filter = argv[0]; 55250e19ef9SNamhyung Kim } 553655000e7SArnaldo Carvalho de Melo 5549cef4b0bSTaeung Song if (symbol_conf.show_nr_samples && annotate.use_gtk) { 5559cef4b0bSTaeung Song pr_err("--show-nr-samples is not available in --gtk mode at this time\n"); 5561ac39372STaeung Song return ret; 5571ac39372STaeung Song } 5581ac39372STaeung Song 559eddaef88SNamhyung Kim if (quiet) 560eddaef88SNamhyung Kim perf_quiet_option(); 561eddaef88SNamhyung Kim 562eae8ad80SJiri Olsa data.file.path = input_name; 56344848cdbSMartin Liška 5648ceb41d7SJiri Olsa annotate.session = perf_session__new(&data, false, &annotate.tool); 565fa10f316SNamhyung Kim if (annotate.session == NULL) 56652e02834STaeung Song return -1; 567fa10f316SNamhyung Kim 568bb848c14SJin Yao annotate.has_br_stack = perf_header__has_feat(&annotate.session->header, 569bb848c14SJin Yao HEADER_BRANCH_STACK); 570bb848c14SJin Yao 571b01141f4SArnaldo Carvalho de Melo ret = symbol__annotation_init(); 572b01141f4SArnaldo Carvalho de Melo if (ret < 0) 573b01141f4SArnaldo Carvalho de Melo goto out_delete; 574b01141f4SArnaldo Carvalho de Melo 5757f0b6fdeSArnaldo Carvalho de Melo annotation_config__init(); 5767f0b6fdeSArnaldo Carvalho de Melo 57775be6cf4SArnaldo Carvalho de Melo symbol_conf.try_vmlinux_path = true; 57875be6cf4SArnaldo Carvalho de Melo 5790a7e6d1bSNamhyung Kim ret = symbol__init(&annotate.session->header.env); 580fa10f316SNamhyung Kim if (ret < 0) 581fa10f316SNamhyung Kim goto out_delete; 58286470930SIngo Molnar 583befd2a38SArnaldo Carvalho de Melo if (annotate.use_stdio || annotate.use_stdio2) 5843df668e7SNamhyung Kim use_browser = 0; 5853df668e7SNamhyung Kim else if (annotate.use_tui) 5863df668e7SNamhyung Kim use_browser = 1; 5873df668e7SNamhyung Kim else if (annotate.use_gtk) 5883df668e7SNamhyung Kim use_browser = 2; 5893df668e7SNamhyung Kim 5903df668e7SNamhyung Kim setup_browser(true); 5913df668e7SNamhyung Kim 592befd2a38SArnaldo Carvalho de Melo if ((use_browser == 1 || annotate.use_stdio2) && annotate.has_br_stack) { 593bb848c14SJin Yao sort__mode = SORT_MODE__BRANCH; 594bb848c14SJin Yao if (setup_sorting(annotate.session->evlist) < 0) 595bb848c14SJin Yao usage_with_options(annotate_usage, options); 596bb848c14SJin Yao } else { 597bb848c14SJin Yao if (setup_sorting(NULL) < 0) 598bb848c14SJin Yao usage_with_options(annotate_usage, options); 599bb848c14SJin Yao } 600bb848c14SJin Yao 601fa10f316SNamhyung Kim ret = __cmd_annotate(&annotate); 602fa10f316SNamhyung Kim 603fa10f316SNamhyung Kim out_delete: 604fa10f316SNamhyung Kim /* 605fa10f316SNamhyung Kim * Speed up the exit process, for large files this can 606fa10f316SNamhyung Kim * take quite a while. 607fa10f316SNamhyung Kim * 608fa10f316SNamhyung Kim * XXX Enable this when using valgrind or if we ever 609fa10f316SNamhyung Kim * librarize this command. 610fa10f316SNamhyung Kim * 611fa10f316SNamhyung Kim * Also experiment with obstacks to see how much speed 612fa10f316SNamhyung Kim * up we'll get here. 613fa10f316SNamhyung Kim * 614fa10f316SNamhyung Kim * perf_session__delete(session); 615fa10f316SNamhyung Kim */ 616fa10f316SNamhyung Kim return ret; 61786470930SIngo Molnar } 618