diff options
Diffstat (limited to 'Documentation/perf_counter/builtin-report.c')
-rw-r--r-- | Documentation/perf_counter/builtin-report.c | 103 |
1 files changed, 51 insertions, 52 deletions
diff --git a/Documentation/perf_counter/builtin-report.c b/Documentation/perf_counter/builtin-report.c index 2d65d9c12aad..7f1255dcd222 100644 --- a/Documentation/perf_counter/builtin-report.c +++ b/Documentation/perf_counter/builtin-report.c | |||
@@ -1,4 +1,5 @@ | |||
1 | #include "util/util.h" | 1 | #include "util/util.h" |
2 | #include "builtin.h" | ||
2 | 3 | ||
3 | #include <libelf.h> | 4 | #include <libelf.h> |
4 | #include <gelf.h> | 5 | #include <gelf.h> |
@@ -22,7 +23,7 @@ static int input; | |||
22 | static int show_mask = SHOW_KERNEL | SHOW_USER | SHOW_HV; | 23 | static int show_mask = SHOW_KERNEL | SHOW_USER | SHOW_HV; |
23 | 24 | ||
24 | static int dump_trace = 0; | 25 | static int dump_trace = 0; |
25 | static int verbose; | 26 | static int verbose; |
26 | 27 | ||
27 | static unsigned long page_size; | 28 | static unsigned long page_size; |
28 | static unsigned long mmap_window = 32; | 29 | static unsigned long mmap_window = 32; |
@@ -60,10 +61,10 @@ typedef union event_union { | |||
60 | } event_t; | 61 | } event_t; |
61 | 62 | ||
62 | struct symbol { | 63 | struct symbol { |
63 | struct rb_node rb_node; | 64 | struct rb_node rb_node; |
64 | uint64_t start; | 65 | __u64 start; |
65 | uint64_t end; | 66 | __u64 end; |
66 | char name[0]; | 67 | char name[0]; |
67 | }; | 68 | }; |
68 | 69 | ||
69 | static struct symbol *symbol__new(uint64_t start, uint64_t len, const char *name) | 70 | static struct symbol *symbol__new(uint64_t start, uint64_t len, const char *name) |
@@ -86,7 +87,7 @@ static void symbol__delete(struct symbol *self) | |||
86 | 87 | ||
87 | static size_t symbol__fprintf(struct symbol *self, FILE *fp) | 88 | static size_t symbol__fprintf(struct symbol *self, FILE *fp) |
88 | { | 89 | { |
89 | return fprintf(fp, " %lx-%lx %s\n", | 90 | return fprintf(fp, " %llx-%llx %s\n", |
90 | self->start, self->end, self->name); | 91 | self->start, self->end, self->name); |
91 | } | 92 | } |
92 | 93 | ||
@@ -147,10 +148,12 @@ static void dso__insert_symbol(struct dso *self, struct symbol *sym) | |||
147 | 148 | ||
148 | static struct symbol *dso__find_symbol(struct dso *self, uint64_t ip) | 149 | static struct symbol *dso__find_symbol(struct dso *self, uint64_t ip) |
149 | { | 150 | { |
151 | struct rb_node *n; | ||
152 | |||
150 | if (self == NULL) | 153 | if (self == NULL) |
151 | return NULL; | 154 | return NULL; |
152 | 155 | ||
153 | struct rb_node *n = self->syms.rb_node; | 156 | n = self->syms.rb_node; |
154 | 157 | ||
155 | while (n) { | 158 | while (n) { |
156 | struct symbol *s = rb_entry(n, struct symbol, rb_node); | 159 | struct symbol *s = rb_entry(n, struct symbol, rb_node); |
@@ -221,33 +224,42 @@ static Elf_Scn *elf_section_by_name(Elf *elf, GElf_Ehdr *ep, | |||
221 | 224 | ||
222 | static int dso__load(struct dso *self) | 225 | static int dso__load(struct dso *self) |
223 | { | 226 | { |
224 | int fd = open(self->name, O_RDONLY), err = -1; | 227 | Elf_Data *symstrs; |
228 | uint32_t nr_syms; | ||
229 | int fd, err = -1; | ||
230 | uint32_t index; | ||
231 | GElf_Ehdr ehdr; | ||
232 | GElf_Shdr shdr; | ||
233 | Elf_Data *syms; | ||
234 | GElf_Sym sym; | ||
235 | Elf_Scn *sec; | ||
236 | Elf *elf; | ||
225 | 237 | ||
238 | |||
239 | fd = open(self->name, O_RDONLY); | ||
226 | if (fd == -1) | 240 | if (fd == -1) |
227 | return -1; | 241 | return -1; |
228 | 242 | ||
229 | Elf *elf = elf_begin(fd, ELF_C_READ_MMAP, NULL); | 243 | elf = elf_begin(fd, ELF_C_READ_MMAP, NULL); |
230 | if (elf == NULL) { | 244 | if (elf == NULL) { |
231 | fprintf(stderr, "%s: cannot read %s ELF file.\n", | 245 | fprintf(stderr, "%s: cannot read %s ELF file.\n", |
232 | __func__, self->name); | 246 | __func__, self->name); |
233 | goto out_close; | 247 | goto out_close; |
234 | } | 248 | } |
235 | 249 | ||
236 | GElf_Ehdr ehdr; | ||
237 | if (gelf_getehdr(elf, &ehdr) == NULL) { | 250 | if (gelf_getehdr(elf, &ehdr) == NULL) { |
238 | fprintf(stderr, "%s: cannot get elf header.\n", __func__); | 251 | fprintf(stderr, "%s: cannot get elf header.\n", __func__); |
239 | goto out_elf_end; | 252 | goto out_elf_end; |
240 | } | 253 | } |
241 | 254 | ||
242 | GElf_Shdr shdr; | 255 | sec = elf_section_by_name(elf, &ehdr, &shdr, ".symtab", NULL); |
243 | Elf_Scn *sec = elf_section_by_name(elf, &ehdr, &shdr, ".symtab", NULL); | ||
244 | if (sec == NULL) | 256 | if (sec == NULL) |
245 | sec = elf_section_by_name(elf, &ehdr, &shdr, ".dynsym", NULL); | 257 | sec = elf_section_by_name(elf, &ehdr, &shdr, ".dynsym", NULL); |
246 | 258 | ||
247 | if (sec == NULL) | 259 | if (sec == NULL) |
248 | goto out_elf_end; | 260 | goto out_elf_end; |
249 | 261 | ||
250 | Elf_Data *syms = elf_getdata(sec, NULL); | 262 | syms = elf_getdata(sec, NULL); |
251 | if (syms == NULL) | 263 | if (syms == NULL) |
252 | goto out_elf_end; | 264 | goto out_elf_end; |
253 | 265 | ||
@@ -255,14 +267,12 @@ static int dso__load(struct dso *self) | |||
255 | if (sec == NULL) | 267 | if (sec == NULL) |
256 | goto out_elf_end; | 268 | goto out_elf_end; |
257 | 269 | ||
258 | Elf_Data *symstrs = elf_getdata(sec, NULL); | 270 | symstrs = elf_getdata(sec, NULL); |
259 | if (symstrs == NULL) | 271 | if (symstrs == NULL) |
260 | goto out_elf_end; | 272 | goto out_elf_end; |
261 | 273 | ||
262 | const uint32_t nr_syms = shdr.sh_size / shdr.sh_entsize; | 274 | nr_syms = shdr.sh_size / shdr.sh_entsize; |
263 | 275 | ||
264 | GElf_Sym sym; | ||
265 | uint32_t index; | ||
266 | elf_symtab__for_each_symbol(syms, nr_syms, index, sym) { | 276 | elf_symtab__for_each_symbol(syms, nr_syms, index, sym) { |
267 | struct symbol *f; | 277 | struct symbol *f; |
268 | 278 | ||
@@ -342,7 +352,7 @@ out_delete_dso: | |||
342 | return NULL; | 352 | return NULL; |
343 | } | 353 | } |
344 | 354 | ||
345 | void dsos__fprintf(FILE *fp) | 355 | static void dsos__fprintf(FILE *fp) |
346 | { | 356 | { |
347 | struct dso *pos; | 357 | struct dso *pos; |
348 | 358 | ||
@@ -365,7 +375,7 @@ static int hex(char ch) | |||
365 | * While we find nice hex chars, build a long_val. | 375 | * While we find nice hex chars, build a long_val. |
366 | * Return number of chars processed. | 376 | * Return number of chars processed. |
367 | */ | 377 | */ |
368 | int hex2long(char *ptr, unsigned long *long_val) | 378 | static int hex2long(char *ptr, unsigned long *long_val) |
369 | { | 379 | { |
370 | const char *p = ptr; | 380 | const char *p = ptr; |
371 | *long_val = 0; | 381 | *long_val = 0; |
@@ -493,12 +503,6 @@ out_delete: | |||
493 | return NULL; | 503 | return NULL; |
494 | } | 504 | } |
495 | 505 | ||
496 | static size_t map__fprintf(struct map *self, FILE *fp) | ||
497 | { | ||
498 | return fprintf(fp, " %lx-%lx %lx %s\n", | ||
499 | self->start, self->end, self->pgoff, self->dso->name); | ||
500 | } | ||
501 | |||
502 | struct thread; | 506 | struct thread; |
503 | 507 | ||
504 | static const char *thread__name(struct thread *self, char *bf, size_t size); | 508 | static const char *thread__name(struct thread *self, char *bf, size_t size); |
@@ -531,11 +535,6 @@ static struct symhist *symhist__new(struct symbol *sym, uint64_t ip, | |||
531 | return self; | 535 | return self; |
532 | } | 536 | } |
533 | 537 | ||
534 | void symhist__delete(struct symhist *self) | ||
535 | { | ||
536 | free(self); | ||
537 | } | ||
538 | |||
539 | static void symhist__inc(struct symhist *self) | 538 | static void symhist__inc(struct symhist *self) |
540 | { | 539 | { |
541 | ++self->count; | 540 | ++self->count; |
@@ -608,6 +607,8 @@ static int thread__symbol_incnew(struct thread *self, struct symbol *sym, | |||
608 | struct symhist *sh; | 607 | struct symhist *sh; |
609 | 608 | ||
610 | while (*p != NULL) { | 609 | while (*p != NULL) { |
610 | uint64_t start; | ||
611 | |||
611 | parent = *p; | 612 | parent = *p; |
612 | sh = rb_entry(parent, struct symhist, rb_node); | 613 | sh = rb_entry(parent, struct symhist, rb_node); |
613 | 614 | ||
@@ -617,7 +618,7 @@ static int thread__symbol_incnew(struct thread *self, struct symbol *sym, | |||
617 | } | 618 | } |
618 | 619 | ||
619 | /* Handle unresolved symbols too */ | 620 | /* Handle unresolved symbols too */ |
620 | const uint64_t start = !sh->sym ? sh->ip : sh->sym->start; | 621 | start = !sh->sym ? sh->ip : sh->sym->start; |
621 | 622 | ||
622 | if (ip < start) | 623 | if (ip < start) |
623 | p = &(*p)->rb_left; | 624 | p = &(*p)->rb_left; |
@@ -639,17 +640,6 @@ static int thread__set_comm(struct thread *self, const char *comm) | |||
639 | return self->comm ? 0 : -ENOMEM; | 640 | return self->comm ? 0 : -ENOMEM; |
640 | } | 641 | } |
641 | 642 | ||
642 | size_t thread__maps_fprintf(struct thread *self, FILE *fp) | ||
643 | { | ||
644 | struct map *pos; | ||
645 | size_t ret = 0; | ||
646 | |||
647 | list_for_each_entry(pos, &self->maps, node) | ||
648 | ret += map__fprintf(pos, fp); | ||
649 | |||
650 | return ret; | ||
651 | } | ||
652 | |||
653 | static size_t thread__fprintf(struct thread *self, FILE *fp) | 643 | static size_t thread__fprintf(struct thread *self, FILE *fp) |
654 | { | 644 | { |
655 | int ret = fprintf(fp, "thread: %d %s\n", self->pid, self->comm); | 645 | int ret = fprintf(fp, "thread: %d %s\n", self->pid, self->comm); |
@@ -657,13 +647,14 @@ static size_t thread__fprintf(struct thread *self, FILE *fp) | |||
657 | 647 | ||
658 | for (nd = rb_first(&self->symhists); nd; nd = rb_next(nd)) { | 648 | for (nd = rb_first(&self->symhists); nd; nd = rb_next(nd)) { |
659 | struct symhist *pos = rb_entry(nd, struct symhist, rb_node); | 649 | struct symhist *pos = rb_entry(nd, struct symhist, rb_node); |
650 | |||
660 | ret += symhist__fprintf(pos, 0, fp); | 651 | ret += symhist__fprintf(pos, 0, fp); |
661 | } | 652 | } |
662 | 653 | ||
663 | return ret; | 654 | return ret; |
664 | } | 655 | } |
665 | 656 | ||
666 | static struct rb_root threads = RB_ROOT; | 657 | static struct rb_root threads; |
667 | 658 | ||
668 | static struct thread *threads__findnew(pid_t pid) | 659 | static struct thread *threads__findnew(pid_t pid) |
669 | { | 660 | { |
@@ -699,11 +690,11 @@ static void thread__insert_map(struct thread *self, struct map *map) | |||
699 | 690 | ||
700 | static struct map *thread__find_map(struct thread *self, uint64_t ip) | 691 | static struct map *thread__find_map(struct thread *self, uint64_t ip) |
701 | { | 692 | { |
693 | struct map *pos; | ||
694 | |||
702 | if (self == NULL) | 695 | if (self == NULL) |
703 | return NULL; | 696 | return NULL; |
704 | 697 | ||
705 | struct map *pos; | ||
706 | |||
707 | list_for_each_entry(pos, &self->maps, node) | 698 | list_for_each_entry(pos, &self->maps, node) |
708 | if (ip >= pos->start && ip <= pos->end) | 699 | if (ip >= pos->start && ip <= pos->end) |
709 | return pos; | 700 | return pos; |
@@ -711,7 +702,7 @@ static struct map *thread__find_map(struct thread *self, uint64_t ip) | |||
711 | return NULL; | 702 | return NULL; |
712 | } | 703 | } |
713 | 704 | ||
714 | void threads__fprintf(FILE *fp) | 705 | static void threads__fprintf(FILE *fp) |
715 | { | 706 | { |
716 | struct rb_node *nd; | 707 | struct rb_node *nd; |
717 | for (nd = rb_first(&threads); nd; nd = rb_next(nd)) { | 708 | for (nd = rb_first(&threads); nd; nd = rb_next(nd)) { |
@@ -720,7 +711,7 @@ void threads__fprintf(FILE *fp) | |||
720 | } | 711 | } |
721 | } | 712 | } |
722 | 713 | ||
723 | static struct rb_root global_symhists = RB_ROOT; | 714 | static struct rb_root global_symhists; |
724 | 715 | ||
725 | static void threads__insert_symhist(struct symhist *sh) | 716 | static void threads__insert_symhist(struct symhist *sh) |
726 | { | 717 | { |
@@ -852,7 +843,7 @@ more: | |||
852 | (void *)(long)(event->header.size), | 843 | (void *)(long)(event->header.size), |
853 | event->header.misc, | 844 | event->header.misc, |
854 | event->ip.pid, | 845 | event->ip.pid, |
855 | (void *)event->ip.ip); | 846 | (void *)(long)ip); |
856 | } | 847 | } |
857 | 848 | ||
858 | if (thread == NULL) { | 849 | if (thread == NULL) { |
@@ -866,9 +857,12 @@ more: | |||
866 | level = 'k'; | 857 | level = 'k'; |
867 | dso = kernel_dso; | 858 | dso = kernel_dso; |
868 | } else if (event->header.misc & PERF_EVENT_MISC_USER) { | 859 | } else if (event->header.misc & PERF_EVENT_MISC_USER) { |
860 | struct map *map; | ||
861 | |||
869 | show = SHOW_USER; | 862 | show = SHOW_USER; |
870 | level = '.'; | 863 | level = '.'; |
871 | struct map *map = thread__find_map(thread, ip); | 864 | |
865 | map = thread__find_map(thread, ip); | ||
872 | if (map != NULL) { | 866 | if (map != NULL) { |
873 | dso = map->dso; | 867 | dso = map->dso; |
874 | ip -= map->start + map->pgoff; | 868 | ip -= map->start + map->pgoff; |
@@ -896,9 +890,9 @@ more: | |||
896 | fprintf(stderr, "%p [%p]: PERF_EVENT_MMAP: [%p(%p) @ %p]: %s\n", | 890 | fprintf(stderr, "%p [%p]: PERF_EVENT_MMAP: [%p(%p) @ %p]: %s\n", |
897 | (void *)(offset + head), | 891 | (void *)(offset + head), |
898 | (void *)(long)(event->header.size), | 892 | (void *)(long)(event->header.size), |
899 | (void *)event->mmap.start, | 893 | (void *)(long)event->mmap.start, |
900 | (void *)event->mmap.len, | 894 | (void *)(long)event->mmap.len, |
901 | (void *)event->mmap.pgoff, | 895 | (void *)(long)event->mmap.pgoff, |
902 | event->mmap.filename); | 896 | event->mmap.filename); |
903 | } | 897 | } |
904 | if (thread == NULL || map == NULL) { | 898 | if (thread == NULL || map == NULL) { |
@@ -964,6 +958,11 @@ done: | |||
964 | return 0; | 958 | return 0; |
965 | } | 959 | } |
966 | 960 | ||
961 | if (verbose >= 2) { | ||
962 | dsos__fprintf(stdout); | ||
963 | threads__fprintf(stdout); | ||
964 | } | ||
965 | |||
967 | threads__sort_symhists(); | 966 | threads__sort_symhists(); |
968 | threads__symhists_fprintf(total, stdout); | 967 | threads__symhists_fprintf(total, stdout); |
969 | 968 | ||