186470930SIngo Molnar /* 286470930SIngo Molnar * builtin-annotate.c 386470930SIngo Molnar * 486470930SIngo Molnar * Builtin annotate command: Analyze the perf.data input file, 586470930SIngo Molnar * look up and read DSOs and symbol information and display 686470930SIngo Molnar * a histogram of results, along various sorting keys. 786470930SIngo Molnar */ 886470930SIngo Molnar #include "builtin.h" 986470930SIngo Molnar 1086470930SIngo Molnar #include "util/util.h" 1186470930SIngo Molnar #include "util/color.h" 125da50258SArnaldo Carvalho de Melo #include <linux/list.h> 1386470930SIngo Molnar #include "util/cache.h" 1443cbcd8aSArnaldo Carvalho de Melo #include <linux/rbtree.h> 1586470930SIngo Molnar #include "util/symbol.h" 1686470930SIngo Molnar 1786470930SIngo Molnar #include "perf.h" 188f28827aSFrederic Weisbecker #include "util/debug.h" 1986470930SIngo Molnar 20e248de33SArnaldo Carvalho de Melo #include "util/evlist.h" 21e248de33SArnaldo Carvalho de Melo #include "util/evsel.h" 2278f7defeSArnaldo Carvalho de Melo #include "util/annotate.h" 2362daacb5SArnaldo Carvalho de Melo #include "util/event.h" 244b6ab94eSJosh Poimboeuf #include <subcmd/parse-options.h> 2586470930SIngo Molnar #include "util/parse-events.h" 266baa0a5aSFrederic Weisbecker #include "util/thread.h" 27dd68ada2SJohn Kacur #include "util/sort.h" 283d1d07ecSJohn Kacur #include "util/hist.h" 2994c744b6SArnaldo Carvalho de Melo #include "util/session.h" 3045694aa7SArnaldo Carvalho de Melo #include "util/tool.h" 31f5fc1412SJiri Olsa #include "util/data.h" 3268e94f4eSIrina Tirdea #include "arch/common.h" 3370fbe057SPeter Zijlstra #include "util/block-range.h" 3486470930SIngo Molnar 35fc67297bSNamhyung Kim #include <dlfcn.h> 36a43783aeSArnaldo Carvalho de Melo #include <errno.h> 375d67be97SAnton Blanchard #include <linux/bitmap.h> 385d67be97SAnton Blanchard 39d20deb64SArnaldo Carvalho de Melo struct perf_annotate { 4045694aa7SArnaldo Carvalho de Melo struct perf_tool tool; 41fa10f316SNamhyung Kim struct perf_session *session; 42fa10f316SNamhyung Kim bool use_tui, use_stdio, use_gtk; 437009cc34SArnaldo Carvalho de Melo bool full_paths; 447009cc34SArnaldo Carvalho de Melo bool print_line; 4518c9e5c5SNamhyung Kim bool skip_missing; 467009cc34SArnaldo Carvalho de Melo const char *sym_hist_filter; 477009cc34SArnaldo Carvalho de Melo const char *cpu_list; 487009cc34SArnaldo Carvalho de Melo DECLARE_BITMAP(cpu_bitmap, MAX_NR_CPUS); 49d20deb64SArnaldo Carvalho de Melo }; 505d67be97SAnton Blanchard 5170fbe057SPeter Zijlstra /* 5270fbe057SPeter Zijlstra * Given one basic block: 5370fbe057SPeter Zijlstra * 5470fbe057SPeter Zijlstra * from to branch_i 5570fbe057SPeter Zijlstra * * ----> * 5670fbe057SPeter Zijlstra * | 5770fbe057SPeter Zijlstra * | block 5870fbe057SPeter Zijlstra * v 5970fbe057SPeter Zijlstra * * ----> * 6070fbe057SPeter Zijlstra * from to branch_i+1 6170fbe057SPeter Zijlstra * 6270fbe057SPeter Zijlstra * where the horizontal are the branches and the vertical is the executed 6370fbe057SPeter Zijlstra * block of instructions. 6470fbe057SPeter Zijlstra * 6570fbe057SPeter Zijlstra * We count, for each 'instruction', the number of blocks that covered it as 6670fbe057SPeter Zijlstra * well as count the ratio each branch is taken. 6770fbe057SPeter Zijlstra * 6870fbe057SPeter Zijlstra * We can do this without knowing the actual instruction stream by keeping 6970fbe057SPeter Zijlstra * track of the address ranges. We break down ranges such that there is no 7070fbe057SPeter Zijlstra * overlap and iterate from the start until the end. 7170fbe057SPeter Zijlstra * 7270fbe057SPeter Zijlstra * @acme: once we parse the objdump output _before_ processing the samples, 7370fbe057SPeter Zijlstra * we can easily fold the branch.cycles IPC bits in. 7470fbe057SPeter Zijlstra */ 7570fbe057SPeter Zijlstra static void process_basic_block(struct addr_map_symbol *start, 7670fbe057SPeter Zijlstra struct addr_map_symbol *end, 7770fbe057SPeter Zijlstra struct branch_flags *flags) 7870fbe057SPeter Zijlstra { 7970fbe057SPeter Zijlstra struct symbol *sym = start->sym; 8070fbe057SPeter Zijlstra struct annotation *notes = sym ? symbol__annotation(sym) : NULL; 8170fbe057SPeter Zijlstra struct block_range_iter iter; 8270fbe057SPeter Zijlstra struct block_range *entry; 8370fbe057SPeter Zijlstra 8470fbe057SPeter Zijlstra /* 8570fbe057SPeter Zijlstra * Sanity; NULL isn't executable and the CPU cannot execute backwards 8670fbe057SPeter Zijlstra */ 8770fbe057SPeter Zijlstra if (!start->addr || start->addr > end->addr) 8870fbe057SPeter Zijlstra return; 8970fbe057SPeter Zijlstra 9070fbe057SPeter Zijlstra iter = block_range__create(start->addr, end->addr); 9170fbe057SPeter Zijlstra if (!block_range_iter__valid(&iter)) 9270fbe057SPeter Zijlstra return; 9370fbe057SPeter Zijlstra 9470fbe057SPeter Zijlstra /* 9570fbe057SPeter Zijlstra * First block in range is a branch target. 9670fbe057SPeter Zijlstra */ 9770fbe057SPeter Zijlstra entry = block_range_iter(&iter); 9870fbe057SPeter Zijlstra assert(entry->is_target); 9970fbe057SPeter Zijlstra entry->entry++; 10070fbe057SPeter Zijlstra 10170fbe057SPeter Zijlstra do { 10270fbe057SPeter Zijlstra entry = block_range_iter(&iter); 10370fbe057SPeter Zijlstra 10470fbe057SPeter Zijlstra entry->coverage++; 10570fbe057SPeter Zijlstra entry->sym = sym; 10670fbe057SPeter Zijlstra 10770fbe057SPeter Zijlstra if (notes) 10870fbe057SPeter Zijlstra notes->max_coverage = max(notes->max_coverage, entry->coverage); 10970fbe057SPeter Zijlstra 11070fbe057SPeter Zijlstra } while (block_range_iter__next(&iter)); 11170fbe057SPeter Zijlstra 11270fbe057SPeter Zijlstra /* 11370fbe057SPeter Zijlstra * Last block in rage is a branch. 11470fbe057SPeter Zijlstra */ 11570fbe057SPeter Zijlstra entry = block_range_iter(&iter); 11670fbe057SPeter Zijlstra assert(entry->is_branch); 11770fbe057SPeter Zijlstra entry->taken++; 11870fbe057SPeter Zijlstra if (flags->predicted) 11970fbe057SPeter Zijlstra entry->pred++; 12070fbe057SPeter Zijlstra } 12170fbe057SPeter Zijlstra 12270fbe057SPeter Zijlstra static void process_branch_stack(struct branch_stack *bs, struct addr_location *al, 12370fbe057SPeter Zijlstra struct perf_sample *sample) 12470fbe057SPeter Zijlstra { 12570fbe057SPeter Zijlstra struct addr_map_symbol *prev = NULL; 12670fbe057SPeter Zijlstra struct branch_info *bi; 12770fbe057SPeter Zijlstra int i; 12870fbe057SPeter Zijlstra 12970fbe057SPeter Zijlstra if (!bs || !bs->nr) 13070fbe057SPeter Zijlstra return; 13170fbe057SPeter Zijlstra 13270fbe057SPeter Zijlstra bi = sample__resolve_bstack(sample, al); 13370fbe057SPeter Zijlstra if (!bi) 13470fbe057SPeter Zijlstra return; 13570fbe057SPeter Zijlstra 13670fbe057SPeter Zijlstra for (i = bs->nr - 1; i >= 0; i--) { 13770fbe057SPeter Zijlstra /* 13870fbe057SPeter Zijlstra * XXX filter against symbol 13970fbe057SPeter Zijlstra */ 14070fbe057SPeter Zijlstra if (prev) 14170fbe057SPeter Zijlstra process_basic_block(prev, &bi[i].from, &bi[i].flags); 14270fbe057SPeter Zijlstra prev = &bi[i].to; 14370fbe057SPeter Zijlstra } 14470fbe057SPeter Zijlstra 14570fbe057SPeter Zijlstra free(bi); 14670fbe057SPeter Zijlstra } 14770fbe057SPeter Zijlstra 148d04b35f8SArnaldo Carvalho de Melo static int perf_evsel__add_sample(struct perf_evsel *evsel, 149fd36f3ddSNamhyung Kim struct perf_sample *sample, 150d20deb64SArnaldo Carvalho de Melo struct addr_location *al, 151d20deb64SArnaldo Carvalho de Melo struct perf_annotate *ann) 15286470930SIngo Molnar { 1534ea062edSArnaldo Carvalho de Melo struct hists *hists = evsel__hists(evsel); 154628ada0cSArnaldo Carvalho de Melo struct hist_entry *he; 155e248de33SArnaldo Carvalho de Melo int ret; 156628ada0cSArnaldo Carvalho de Melo 1577009cc34SArnaldo Carvalho de Melo if (ann->sym_hist_filter != NULL && 1587009cc34SArnaldo Carvalho de Melo (al->sym == NULL || 1597009cc34SArnaldo Carvalho de Melo strcmp(ann->sym_hist_filter, al->sym->name) != 0)) { 160628ada0cSArnaldo Carvalho de Melo /* We're only interested in a symbol named sym_hist_filter */ 161facf3f06SArnaldo Carvalho de Melo /* 162facf3f06SArnaldo Carvalho de Melo * FIXME: why isn't this done in the symbol_filter when loading 163facf3f06SArnaldo Carvalho de Melo * the DSO? 164facf3f06SArnaldo Carvalho de Melo */ 165628ada0cSArnaldo Carvalho de Melo if (al->sym != NULL) { 166628ada0cSArnaldo Carvalho de Melo rb_erase(&al->sym->rb_node, 167628ada0cSArnaldo Carvalho de Melo &al->map->dso->symbols[al->map->type]); 168628ada0cSArnaldo Carvalho de Melo symbol__delete(al->sym); 169c0b4dffbSArnaldo Carvalho de Melo dso__reset_find_symbol_cache(al->map->dso); 170628ada0cSArnaldo Carvalho de Melo } 171628ada0cSArnaldo Carvalho de Melo return 0; 172628ada0cSArnaldo Carvalho de Melo } 173628ada0cSArnaldo Carvalho de Melo 17470fbe057SPeter Zijlstra /* 17570fbe057SPeter Zijlstra * XXX filtered samples can still have branch entires pointing into our 17670fbe057SPeter Zijlstra * symbol and are missed. 17770fbe057SPeter Zijlstra */ 17870fbe057SPeter Zijlstra process_branch_stack(sample->branch_stack, al, sample); 17970fbe057SPeter Zijlstra 180fd36f3ddSNamhyung Kim sample->period = 1; 181fd36f3ddSNamhyung Kim sample->weight = 1; 182fd36f3ddSNamhyung Kim 1830102ef3eSJiri Olsa he = hists__add_entry(hists, al, NULL, NULL, NULL, sample, true); 1849735abf1SArnaldo Carvalho de Melo if (he == NULL) 18586470930SIngo Molnar return -ENOMEM; 186628ada0cSArnaldo Carvalho de Melo 187bab89f6aSTaeung Song ret = hist_entry__inc_addr_samples(he, sample, evsel->idx, al->addr); 1884ea062edSArnaldo Carvalho de Melo hists__inc_nr_samples(hists, true); 189e248de33SArnaldo Carvalho de Melo return ret; 19086470930SIngo Molnar } 19186470930SIngo Molnar 19245694aa7SArnaldo Carvalho de Melo static int process_sample_event(struct perf_tool *tool, 193d20deb64SArnaldo Carvalho de Melo union perf_event *event, 1948115d60cSArnaldo Carvalho de Melo struct perf_sample *sample, 1959e69c210SArnaldo Carvalho de Melo struct perf_evsel *evsel, 196743eb868SArnaldo Carvalho de Melo struct machine *machine) 19786470930SIngo Molnar { 19845694aa7SArnaldo Carvalho de Melo struct perf_annotate *ann = container_of(tool, struct perf_annotate, tool); 1991ed091c4SArnaldo Carvalho de Melo struct addr_location al; 200b91fc39fSArnaldo Carvalho de Melo int ret = 0; 2016baa0a5aSFrederic Weisbecker 202bb3eb566SArnaldo Carvalho de Melo if (machine__resolve(machine, &al, sample) < 0) { 20329a9f66dSArnaldo Carvalho de Melo pr_warning("problem processing %d event, skipping it.\n", 20486470930SIngo Molnar event->header.type); 20586470930SIngo Molnar return -1; 20686470930SIngo Molnar } 20786470930SIngo Molnar 2087009cc34SArnaldo Carvalho de Melo if (ann->cpu_list && !test_bit(sample->cpu, ann->cpu_bitmap)) 209b91fc39fSArnaldo Carvalho de Melo goto out_put; 2105d67be97SAnton Blanchard 211d20deb64SArnaldo Carvalho de Melo if (!al.filtered && perf_evsel__add_sample(evsel, sample, &al, ann)) { 21229a9f66dSArnaldo Carvalho de Melo pr_warning("problem incrementing symbol count, " 213ec218fc4SArnaldo Carvalho de Melo "skipping event\n"); 214b91fc39fSArnaldo Carvalho de Melo ret = -1; 21586470930SIngo Molnar } 216b91fc39fSArnaldo Carvalho de Melo out_put: 217b91fc39fSArnaldo Carvalho de Melo addr_location__put(&al); 218b91fc39fSArnaldo Carvalho de Melo return ret; 21986470930SIngo Molnar } 22086470930SIngo Molnar 221db8fd07aSNamhyung Kim static int hist_entry__tty_annotate(struct hist_entry *he, 222db8fd07aSNamhyung Kim struct perf_evsel *evsel, 223d20deb64SArnaldo Carvalho de Melo struct perf_annotate *ann) 22486470930SIngo Molnar { 225db8fd07aSNamhyung Kim return symbol__tty_annotate(he->ms.sym, he->ms.map, evsel, 2267009cc34SArnaldo Carvalho de Melo ann->print_line, ann->full_paths, 0, 0); 22786470930SIngo Molnar } 22886470930SIngo Molnar 229c824c433SArnaldo Carvalho de Melo static void hists__find_annotations(struct hists *hists, 230db8fd07aSNamhyung Kim struct perf_evsel *evsel, 231d20deb64SArnaldo Carvalho de Melo struct perf_annotate *ann) 23286470930SIngo Molnar { 233c824c433SArnaldo Carvalho de Melo struct rb_node *nd = rb_first(&hists->entries), *next; 234cf958003SArnaldo Carvalho de Melo int key = K_RIGHT; 23586470930SIngo Molnar 23646e3e055SArnaldo Carvalho de Melo while (nd) { 237ed52ce2eSArnaldo Carvalho de Melo struct hist_entry *he = rb_entry(nd, struct hist_entry, rb_node); 23878f7defeSArnaldo Carvalho de Melo struct annotation *notes; 23986470930SIngo Molnar 24046e3e055SArnaldo Carvalho de Melo if (he->ms.sym == NULL || he->ms.map->dso->annotate_warned) 24146e3e055SArnaldo Carvalho de Melo goto find_next; 242e4204992SArnaldo Carvalho de Melo 24378f7defeSArnaldo Carvalho de Melo notes = symbol__annotation(he->ms.sym); 244ce6f4fabSArnaldo Carvalho de Melo if (notes->src == NULL) { 24546e3e055SArnaldo Carvalho de Melo find_next: 246cf958003SArnaldo Carvalho de Melo if (key == K_LEFT) 24746e3e055SArnaldo Carvalho de Melo nd = rb_prev(nd); 24846e3e055SArnaldo Carvalho de Melo else 24946e3e055SArnaldo Carvalho de Melo nd = rb_next(nd); 250e4204992SArnaldo Carvalho de Melo continue; 25146e3e055SArnaldo Carvalho de Melo } 252e4204992SArnaldo Carvalho de Melo 2532b676bf0SNamhyung Kim if (use_browser == 2) { 25418c9e5c5SNamhyung Kim int ret; 255fc67297bSNamhyung Kim int (*annotate)(struct hist_entry *he, 256fc67297bSNamhyung Kim struct perf_evsel *evsel, 257fc67297bSNamhyung Kim struct hist_browser_timer *hbt); 25818c9e5c5SNamhyung Kim 259fc67297bSNamhyung Kim annotate = dlsym(perf_gtk_handle, 260fc67297bSNamhyung Kim "hist_entry__gtk_annotate"); 261fc67297bSNamhyung Kim if (annotate == NULL) { 262fc67297bSNamhyung Kim ui__error("GTK browser not found!\n"); 263fc67297bSNamhyung Kim return; 264fc67297bSNamhyung Kim } 265fc67297bSNamhyung Kim 266fc67297bSNamhyung Kim ret = annotate(he, evsel, NULL); 26718c9e5c5SNamhyung Kim if (!ret || !ann->skip_missing) 2682b676bf0SNamhyung Kim return; 26918c9e5c5SNamhyung Kim 27018c9e5c5SNamhyung Kim /* skip missing symbols */ 27118c9e5c5SNamhyung Kim nd = rb_next(nd); 2722b676bf0SNamhyung Kim } else if (use_browser == 1) { 273db8fd07aSNamhyung Kim key = hist_entry__tui_annotate(he, evsel, NULL); 27446e3e055SArnaldo Carvalho de Melo switch (key) { 27518c9e5c5SNamhyung Kim case -1: 27618c9e5c5SNamhyung Kim if (!ann->skip_missing) 27718c9e5c5SNamhyung Kim return; 27818c9e5c5SNamhyung Kim /* fall through */ 279cf958003SArnaldo Carvalho de Melo case K_RIGHT: 280b50e003dSArnaldo Carvalho de Melo next = rb_next(nd); 28146e3e055SArnaldo Carvalho de Melo break; 282cf958003SArnaldo Carvalho de Melo case K_LEFT: 283b50e003dSArnaldo Carvalho de Melo next = rb_prev(nd); 28446e3e055SArnaldo Carvalho de Melo break; 285b50e003dSArnaldo Carvalho de Melo default: 286b50e003dSArnaldo Carvalho de Melo return; 28746e3e055SArnaldo Carvalho de Melo } 288b50e003dSArnaldo Carvalho de Melo 289b50e003dSArnaldo Carvalho de Melo if (next != NULL) 290b50e003dSArnaldo Carvalho de Melo nd = next; 29146e3e055SArnaldo Carvalho de Melo } else { 292db8fd07aSNamhyung Kim hist_entry__tty_annotate(he, evsel, ann); 29346e3e055SArnaldo Carvalho de Melo nd = rb_next(nd); 294ed52ce2eSArnaldo Carvalho de Melo /* 29546e3e055SArnaldo Carvalho de Melo * Since we have a hist_entry per IP for the same 296ce6f4fabSArnaldo Carvalho de Melo * symbol, free he->ms.sym->src to signal we already 29746e3e055SArnaldo Carvalho de Melo * processed this symbol. 298ed52ce2eSArnaldo Carvalho de Melo */ 299d4957633SAndi Kleen zfree(¬es->src->cycles_hist); 30004662523SArnaldo Carvalho de Melo zfree(¬es->src); 30186470930SIngo Molnar } 30286470930SIngo Molnar } 30346e3e055SArnaldo Carvalho de Melo } 30486470930SIngo Molnar 305d20deb64SArnaldo Carvalho de Melo static int __cmd_annotate(struct perf_annotate *ann) 30686470930SIngo Molnar { 307bab81b62SLi Zefan int ret; 308fa10f316SNamhyung Kim struct perf_session *session = ann->session; 309e248de33SArnaldo Carvalho de Melo struct perf_evsel *pos; 310e248de33SArnaldo Carvalho de Melo u64 total_nr_samples; 31194c744b6SArnaldo Carvalho de Melo 3127009cc34SArnaldo Carvalho de Melo if (ann->cpu_list) { 3137009cc34SArnaldo Carvalho de Melo ret = perf_session__cpu_bitmap(session, ann->cpu_list, 3147009cc34SArnaldo Carvalho de Melo ann->cpu_bitmap); 3155d67be97SAnton Blanchard if (ret) 316fa10f316SNamhyung Kim goto out; 3175d67be97SAnton Blanchard } 3185d67be97SAnton Blanchard 31968e94f4eSIrina Tirdea if (!objdump_path) { 320eebd0bfcSArnaldo Carvalho de Melo ret = perf_env__lookup_objdump(&session->header.env); 32168e94f4eSIrina Tirdea if (ret) 322fa10f316SNamhyung Kim goto out; 32368e94f4eSIrina Tirdea } 32468e94f4eSIrina Tirdea 325b7b61cbeSArnaldo Carvalho de Melo ret = perf_session__process_events(session); 326bab81b62SLi Zefan if (ret) 327fa10f316SNamhyung Kim goto out; 32886470930SIngo Molnar 32962daacb5SArnaldo Carvalho de Melo if (dump_trace) { 330c8446b9bSArnaldo Carvalho de Melo perf_session__fprintf_nr_events(session, stdout); 3312a1731fbSArnaldo Carvalho de Melo perf_evlist__fprintf_nr_events(session->evlist, stdout); 332fa10f316SNamhyung Kim goto out; 33362daacb5SArnaldo Carvalho de Melo } 33486470930SIngo Molnar 335da21d1b5SArnaldo Carvalho de Melo if (verbose > 3) 336b3165f41SArnaldo Carvalho de Melo perf_session__fprintf(session, stdout); 33786470930SIngo Molnar 338da21d1b5SArnaldo Carvalho de Melo if (verbose > 2) 339cbf69680SArnaldo Carvalho de Melo perf_session__fprintf_dsos(session, stdout); 34086470930SIngo Molnar 341e248de33SArnaldo Carvalho de Melo total_nr_samples = 0; 342e5cadb93SArnaldo Carvalho de Melo evlist__for_each_entry(session->evlist, pos) { 3434ea062edSArnaldo Carvalho de Melo struct hists *hists = evsel__hists(pos); 344e248de33SArnaldo Carvalho de Melo u32 nr_samples = hists->stats.nr_events[PERF_RECORD_SAMPLE]; 34586470930SIngo Molnar 346e248de33SArnaldo Carvalho de Melo if (nr_samples > 0) { 347e248de33SArnaldo Carvalho de Melo total_nr_samples += nr_samples; 348c1fb5651SNamhyung Kim hists__collapse_resort(hists, NULL); 349f9db0d0fSKan Liang /* Don't sort callchain */ 350f9db0d0fSKan Liang perf_evsel__reset_sample_bit(pos, CALLCHAIN); 351452ce03bSJiri Olsa perf_evsel__output_resort(pos, NULL); 352b1dd4432SNamhyung Kim 353b1dd4432SNamhyung Kim if (symbol_conf.event_group && 354b1dd4432SNamhyung Kim !perf_evsel__is_group_leader(pos)) 355b1dd4432SNamhyung Kim continue; 356b1dd4432SNamhyung Kim 357db8fd07aSNamhyung Kim hists__find_annotations(hists, pos, ann); 358e248de33SArnaldo Carvalho de Melo } 359e248de33SArnaldo Carvalho de Melo } 360e248de33SArnaldo Carvalho de Melo 361e248de33SArnaldo Carvalho de Melo if (total_nr_samples == 0) { 362fa10f316SNamhyung Kim ui__error("The %s file has no samples!\n", session->file->path); 363fa10f316SNamhyung Kim goto out; 364e248de33SArnaldo Carvalho de Melo } 3657a60ba94SNamhyung Kim 366fc67297bSNamhyung Kim if (use_browser == 2) { 367fc67297bSNamhyung Kim void (*show_annotations)(void); 368fc67297bSNamhyung Kim 369fc67297bSNamhyung Kim show_annotations = dlsym(perf_gtk_handle, 370fc67297bSNamhyung Kim "perf_gtk__show_annotations"); 371fc67297bSNamhyung Kim if (show_annotations == NULL) { 372fc67297bSNamhyung Kim ui__error("GTK browser not found!\n"); 373fa10f316SNamhyung Kim goto out; 374fc67297bSNamhyung Kim } 375fc67297bSNamhyung Kim show_annotations(); 376fc67297bSNamhyung Kim } 3777a60ba94SNamhyung Kim 378fa10f316SNamhyung Kim out: 379bab81b62SLi Zefan return ret; 38086470930SIngo Molnar } 38186470930SIngo Molnar 38286470930SIngo Molnar static const char * const annotate_usage[] = { 38399345254SNamhyung Kim "perf annotate [<options>]", 38486470930SIngo Molnar NULL 38586470930SIngo Molnar }; 38686470930SIngo Molnar 387b0ad8ea6SArnaldo Carvalho de Melo int cmd_annotate(int argc, const char **argv) 388d20deb64SArnaldo Carvalho de Melo { 389d20deb64SArnaldo Carvalho de Melo struct perf_annotate annotate = { 39045694aa7SArnaldo Carvalho de Melo .tool = { 391d20deb64SArnaldo Carvalho de Melo .sample = process_sample_event, 392d20deb64SArnaldo Carvalho de Melo .mmap = perf_event__process_mmap, 3935c5e854bSStephane Eranian .mmap2 = perf_event__process_mmap2, 394d20deb64SArnaldo Carvalho de Melo .comm = perf_event__process_comm, 395ec4622f5SArnaldo Carvalho de Melo .exit = perf_event__process_exit, 396f62d3f0fSArnaldo Carvalho de Melo .fork = perf_event__process_fork, 397f3b3614aSHari Bathini .namespaces = perf_event__process_namespaces, 3986ab11f3aSDavid Carrillo-Cisneros .attr = perf_event__process_attr, 3996ab11f3aSDavid Carrillo-Cisneros .build_id = perf_event__process_build_id, 400e9def1b2SDavid Carrillo-Cisneros .feature = perf_event__process_feature, 4010a8cb85cSJiri Olsa .ordered_events = true, 402d20deb64SArnaldo Carvalho de Melo .ordering_requires_timestamps = true, 403d20deb64SArnaldo Carvalho de Melo }, 404d20deb64SArnaldo Carvalho de Melo }; 405fa10f316SNamhyung Kim struct perf_data_file file = { 406fa10f316SNamhyung Kim .mode = PERF_DATA_MODE_READ, 407fa10f316SNamhyung Kim }; 408d20deb64SArnaldo Carvalho de Melo const struct option options[] = { 40970cb4e96SFeng Tang OPT_STRING('i', "input", &input_name, "file", 41086470930SIngo Molnar "input file name"), 411ac73c5a9SArnaldo Carvalho de Melo OPT_STRING('d', "dsos", &symbol_conf.dso_list_str, "dso[,dso...]", 412ac73c5a9SArnaldo Carvalho de Melo "only consider symbols in these dsos"), 4137009cc34SArnaldo Carvalho de Melo OPT_STRING('s', "symbol", &annotate.sym_hist_filter, "symbol", 41486470930SIngo Molnar "symbol to annotate"), 415fa10f316SNamhyung Kim OPT_BOOLEAN('f', "force", &file.force, "don't complain, do it"), 416c0555642SIan Munsie OPT_INCR('v', "verbose", &verbose, 41786470930SIngo Molnar "be more verbose (show symbol address, etc)"), 418eddaef88SNamhyung Kim OPT_BOOLEAN('q', "quiet", &quiet, "do now show any message"), 41986470930SIngo Molnar OPT_BOOLEAN('D', "dump-raw-trace", &dump_trace, 42086470930SIngo Molnar "dump raw trace in ASCII"), 4212b676bf0SNamhyung Kim OPT_BOOLEAN(0, "gtk", &annotate.use_gtk, "Use the GTK interface"), 4227009cc34SArnaldo Carvalho de Melo OPT_BOOLEAN(0, "tui", &annotate.use_tui, "Use the TUI interface"), 4237009cc34SArnaldo Carvalho de Melo OPT_BOOLEAN(0, "stdio", &annotate.use_stdio, "Use the stdio interface"), 424b32d133aSArnaldo Carvalho de Melo OPT_STRING('k', "vmlinux", &symbol_conf.vmlinux_name, 425b32d133aSArnaldo Carvalho de Melo "file", "vmlinux pathname"), 426b32d133aSArnaldo Carvalho de Melo OPT_BOOLEAN('m', "modules", &symbol_conf.use_modules, 42742976487SMike Galbraith "load module symbols - WARNING: use only with -k and LIVE kernel"), 4287009cc34SArnaldo Carvalho de Melo OPT_BOOLEAN('l', "print-line", &annotate.print_line, 429301406b9SFrederic Weisbecker "print matching source lines (may be slow)"), 4307009cc34SArnaldo Carvalho de Melo OPT_BOOLEAN('P', "full-paths", &annotate.full_paths, 43142976487SMike Galbraith "Don't shorten the displayed pathnames"), 43218c9e5c5SNamhyung Kim OPT_BOOLEAN(0, "skip-missing", &annotate.skip_missing, 43318c9e5c5SNamhyung Kim "Skip symbols that cannot be annotated"), 434c8e66720SDavid Ahern OPT_STRING('C', "cpu", &annotate.cpu_list, "cpu", "list of cpus to profile"), 435a7066709SHe Kuang OPT_CALLBACK(0, "symfs", NULL, "directory", 436a7066709SHe Kuang "Look for files with symbols relative to this directory", 437a7066709SHe Kuang symbol__config_symfs), 43864c6f0c7SArnaldo Carvalho de Melo OPT_BOOLEAN(0, "source", &symbol_conf.annotate_src, 4393e6a2a7fSStephane Eranian "Interleave source code with assembly code (default)"), 44064c6f0c7SArnaldo Carvalho de Melo OPT_BOOLEAN(0, "asm-raw", &symbol_conf.annotate_asm_raw, 4413e6a2a7fSStephane Eranian "Display raw encoding of assembly instructions (default)"), 442f69b64f7SAndi Kleen OPT_STRING('M', "disassembler-style", &disassembler_style, "disassembler style", 443f69b64f7SAndi Kleen "Specify disassembler style (e.g. -M intel for intel syntax)"), 4447a4ec938SMaciek Borzecki OPT_STRING(0, "objdump", &objdump_path, "path", 4457a4ec938SMaciek Borzecki "objdump binary to use for disassembly and annotations"), 446b1dd4432SNamhyung Kim OPT_BOOLEAN(0, "group", &symbol_conf.event_group, 447b1dd4432SNamhyung Kim "Show event group information together"), 4480c4a5bceSMartin Liška OPT_BOOLEAN(0, "show-total-period", &symbol_conf.show_total_period, 4490c4a5bceSMartin Liška "Show a column with the sum of periods"), 45053fe4ba1SArnaldo Carvalho de Melo OPT_CALLBACK_DEFAULT(0, "stdio-color", NULL, "mode", 45153fe4ba1SArnaldo Carvalho de Melo "'always' (default), 'never' or 'auto' only applicable to --stdio mode", 45253fe4ba1SArnaldo Carvalho de Melo stdio__config_color, "always"), 45386470930SIngo Molnar OPT_END() 45486470930SIngo Molnar }; 455a635fc51SArnaldo Carvalho de Melo int ret = hists__init(); 456a635fc51SArnaldo Carvalho de Melo 457a635fc51SArnaldo Carvalho de Melo if (ret < 0) 458a635fc51SArnaldo Carvalho de Melo return ret; 45986470930SIngo Molnar 460655000e7SArnaldo Carvalho de Melo argc = parse_options(argc, argv, options, annotate_usage, 0); 46150e19ef9SNamhyung Kim if (argc) { 46250e19ef9SNamhyung Kim /* 46350e19ef9SNamhyung Kim * Special case: if there's an argument left then assume that 46450e19ef9SNamhyung Kim * it's a symbol filter: 46550e19ef9SNamhyung Kim */ 46650e19ef9SNamhyung Kim if (argc > 1) 46750e19ef9SNamhyung Kim usage_with_options(annotate_usage, options); 46850e19ef9SNamhyung Kim 46950e19ef9SNamhyung Kim annotate.sym_hist_filter = argv[0]; 47050e19ef9SNamhyung Kim } 471655000e7SArnaldo Carvalho de Melo 472eddaef88SNamhyung Kim if (quiet) 473eddaef88SNamhyung Kim perf_quiet_option(); 474eddaef88SNamhyung Kim 47544848cdbSMartin Liška file.path = input_name; 47644848cdbSMartin Liška 477fa10f316SNamhyung Kim annotate.session = perf_session__new(&file, false, &annotate.tool); 478fa10f316SNamhyung Kim if (annotate.session == NULL) 47952e02834STaeung Song return -1; 480fa10f316SNamhyung Kim 481b01141f4SArnaldo Carvalho de Melo ret = symbol__annotation_init(); 482b01141f4SArnaldo Carvalho de Melo if (ret < 0) 483b01141f4SArnaldo Carvalho de Melo goto out_delete; 484b01141f4SArnaldo Carvalho de Melo 48575be6cf4SArnaldo Carvalho de Melo symbol_conf.try_vmlinux_path = true; 48675be6cf4SArnaldo Carvalho de Melo 4870a7e6d1bSNamhyung Kim ret = symbol__init(&annotate.session->header.env); 488fa10f316SNamhyung Kim if (ret < 0) 489fa10f316SNamhyung Kim goto out_delete; 49086470930SIngo Molnar 49140184c46SNamhyung Kim if (setup_sorting(NULL) < 0) 49255309985SNamhyung Kim usage_with_options(annotate_usage, options); 49386470930SIngo Molnar 4943df668e7SNamhyung Kim if (annotate.use_stdio) 4953df668e7SNamhyung Kim use_browser = 0; 4963df668e7SNamhyung Kim else if (annotate.use_tui) 4973df668e7SNamhyung Kim use_browser = 1; 4983df668e7SNamhyung Kim else if (annotate.use_gtk) 4993df668e7SNamhyung Kim use_browser = 2; 5003df668e7SNamhyung Kim 5013df668e7SNamhyung Kim setup_browser(true); 5023df668e7SNamhyung Kim 503fa10f316SNamhyung Kim ret = __cmd_annotate(&annotate); 504fa10f316SNamhyung Kim 505fa10f316SNamhyung Kim out_delete: 506fa10f316SNamhyung Kim /* 507fa10f316SNamhyung Kim * Speed up the exit process, for large files this can 508fa10f316SNamhyung Kim * take quite a while. 509fa10f316SNamhyung Kim * 510fa10f316SNamhyung Kim * XXX Enable this when using valgrind or if we ever 511fa10f316SNamhyung Kim * librarize this command. 512fa10f316SNamhyung Kim * 513fa10f316SNamhyung Kim * Also experiment with obstacks to see how much speed 514fa10f316SNamhyung Kim * up we'll get here. 515fa10f316SNamhyung Kim * 516fa10f316SNamhyung Kim * perf_session__delete(session); 517fa10f316SNamhyung Kim */ 518fa10f316SNamhyung Kim return ret; 51986470930SIngo Molnar } 520