#define _GNU_SOURCE #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "../../include/linux/perf_counter.h" #include "util/list.h" #include "util/rbtree.h" #define SHOW_KERNEL 1 #define SHOW_USER 2 #define SHOW_HV 4 static char const *input_name = "output.perf"; static int input; static int show_mask = SHOW_KERNEL | SHOW_USER | SHOW_HV; static unsigned long page_size; static unsigned long mmap_window = 32; static const char *perf_event_names[] = { [PERF_EVENT_MMAP] = " PERF_EVENT_MMAP", [PERF_EVENT_MUNMAP] = " PERF_EVENT_MUNMAP", [PERF_EVENT_COMM] = " PERF_EVENT_COMM", }; struct ip_event { struct perf_event_header header; __u64 ip; __u32 pid, tid; }; struct mmap_event { struct perf_event_header header; __u32 pid, tid; __u64 start; __u64 len; __u64 pgoff; char filename[PATH_MAX]; }; struct comm_event { struct perf_event_header header; __u32 pid,tid; char comm[16]; }; typedef union event_union { struct perf_event_header header; struct ip_event ip; struct mmap_event mmap; struct comm_event comm; } event_t; struct section { struct list_head node; uint64_t start; uint64_t end; uint64_t offset; char name[0]; }; static struct section *section__new(uint64_t start, uint64_t size, uint64_t offset, char *name) { struct section *self = malloc(sizeof(*self) + strlen(name) + 1); if (self != NULL) { self->start = start; self->end = start + size; self->offset = offset; strcpy(self->name, name); } return self; } static void section__delete(struct section *self) { free(self); } struct symbol { struct rb_node rb_node; uint64_t start; uint64_t end; char name[0]; }; static struct symbol *symbol__new(uint64_t start, uint64_t len, const char *name) { struct symbol *self = malloc(sizeof(*self) + strlen(name) + 1); if (self != NULL) { self->start = start; self->end = start + len; strcpy(self->name, name); } return self; } static void symbol__delete(struct symbol *self) { free(self); } static size_t symbol__fprintf(struct symbol *self, FILE *fp) { return fprintf(fp, " %lx-%lx %s\n", self->start, self->end, self->name); } struct dso { struct list_head node; struct list_head sections; struct rb_root syms; char name[0]; }; static struct dso *dso__new(const char *name) { struct dso *self = malloc(sizeof(*self) + strlen(name) + 1); if (self != NULL) { strcpy(self->name, name); INIT_LIST_HEAD(&self->sections); self->syms = RB_ROOT; } return self; } static void dso__delete_sections(struct dso *self) { struct section *pos, *n; list_for_each_entry_safe(pos, n, &self->sections, node) section__delete(pos); } static void dso__delete_symbols(struct dso *self) { struct symbol *pos; struct rb_node *next = rb_first(&self->syms); while (next) { pos = rb_entry(next, struct symbol, rb_node); next = rb_next(&pos->rb_node); symbol__delete(pos); } } static void dso__delete(struct dso *self) { dso__delete_sections(self); dso__delete_symbols(self); free(self); } static void dso__insert_symbol(struct dso *self, struct symbol *sym) { struct rb_node **p = &self->syms.rb_node; struct rb_node *parent = NULL; const uint64_t ip = sym->start; struct symbol *s; while (*p != NULL) { parent = *p; s = rb_entry(parent, struct symbol, rb_node); if (ip < s->start) p = &(*p)->rb_left; else p = &(*p)->rb_right; } rb_link_node(&sym->rb_node, parent, p); rb_insert_color(&sym->rb_node, &self->syms); } static struct symbol *dso__find_symbol(struct dso *self, uint64_t ip) { if (self == NULL) return NULL; struct rb_node *n = self->syms.rb_node; while (n) { struct symbol *s = rb_entry(n, struct symbol, rb_node); if (ip < s->start) n = n->rb_left; else if (ip > s->end) n = n->rb_right; else return s; } return NULL; } /** * elf_symtab__for_each_symbol - iterate thru all the symbols * * @self: struct elf_symtab instance to iterate * @index: uint32_t index * @sym: GElf_Sym iterator */ #define elf_symtab__for_each_symbol(syms, nr_syms, index, sym) \ for (index = 0, gelf_getsym(syms, index, &sym);\ index < nr_syms; \ index++, gelf_getsym(syms, index, &sym)) static inline uint8_t elf_sym__type(const GElf_Sym *sym) { return GELF_ST_TYPE(sym->st_info); } static inline bool elf_sym__is_function(const GElf_Sym *sym) { return elf_sym__type(sym) == STT_FUNC && sym->st_name != 0 && sym->st_shndx != SHN_UNDEF; } static inline const char *elf_sym__name(const GElf_Sym *sym, const Elf_Data *symstrs) { return symstrs->d_buf + sym->st_name; } static Elf_Scn *elf_section_by_name(Elf *elf, GElf_Ehdr *ep, GElf_Shdr *shp, const char *name, size_t *index) { Elf_Scn *sec = NULL; size_t cnt = 1; while ((sec = elf_nextscn(elf, sec)) != NULL) { char *str; gelf_getshdr(sec, shp); str = elf_strptr(elf, ep->e_shstrndx, shp->sh_name); if (!strcmp(name, str)) { if (index) *index = cnt; break; } ++cnt; } return sec; } static int dso__load(struct dso *self) { int fd = open(self->name, O_RDONLY), err = -1; if (fd == -1) return -1; Elf *elf = elf_begin(fd, ELF_C_READ_MMAP, NULL); if (elf == NULL) { fprintf(stderr, "%s: cannot read %s ELF file.\n", __func__, self->name); goto out_close; } GElf_Ehdr ehdr; if (gelf_getehdr(elf, &ehdr) == NULL) { fprintf(stderr, "%s: cannot get elf header.\n", __func__); goto out_elf_end; } GElf_Shdr shdr; Elf_Scn *sec = elf_section_by_name(elf, &ehdr, &shdr, ".symtab", NULL); if (sec == NULL) sec = elf_section_by_name(elf, &ehdr, &shdr, ".dynsym", NULL); if (sec == NULL) goto out_elf_end; if (gelf_getshdr(sec, &shdr) == NULL) goto out_elf_end; Elf_Data *syms = elf_getdata(sec, NULL); if (syms == NULL) goto out_elf_end; sec = elf_getscn(elf, shdr.sh_link); if (sec == NULL) goto out_elf_end; Elf_Data *symstrs = elf_getdata(sec, NULL); if (symstrs == NULL) goto out_elf_end; const uint32_t nr_syms = shdr.sh_size / shdr.sh_entsize; GElf_Sym sym; uint32_t index; elf_symtab__for_each_symbol(syms, nr_syms, index, sym) { if (!elf_sym__is_function(&sym)) continue; struct symbol *f = symbol__new(sym.st_value, sym.st_size, elf_sym__name(&sym, symstrs)); if (f == NULL) goto out_elf_end; dso__insert_symbol(self, f); } err = 0; out_elf_end: elf_end(elf); out_close: close(fd); return err; } static size_t dso__fprintf(struct dso *self, FILE *fp) { size_t ret = fprintf(fp, "dso: %s\n", self->name); struct rb_node *nd; for (nd = rb_first(&self->syms); nd; nd = rb_next(nd)) { struct symbol *pos = rb_entry(nd, struct symbol, rb_node); ret += symbol__fprintf(pos, fp); } return ret; } static LIST_HEAD(dsos); static struct dso *kernel_dso; static void dsos__add(struct dso *dso) { list_add_tail(&dso->node, &dsos); } static struct dso *dsos__find(const char *name) { struct dso *pos; list_for_each_entry(pos, &dsos, node) if (strcmp(pos->name, name) == 0) return pos; return NULL; } static struct dso *dsos__findnew(const char *name) { struct dso *dso = dsos__find(name); if (dso == NULL) { dso = dso__new(name); if (dso != NULL && dso__load(dso) < 0) goto out_delete_dso; dsos__add(dso); } return dso; out_delete_dso: dso__delete(dso); return NULL; } static void dsos__fprintf(FILE *fp) { struct dso *pos; list_for_each_entry(pos, &dsos, node) dso__fprintf(pos, fp); } static int load_kallsyms(void) { kernel_dso = dso__new("[kernel]"); if (kernel_dso == NULL) return -1; FILE *file = fopen("/proc/kallsyms", "r"); if (file == NULL) goto out_delete_dso; char *line = NULL; size_t n; while (!feof(file)) { unsigned long long start; char c, symbf[4096]; if (getline(&line, &n, file) < 0) break; if (!line) goto out_delete_dso; if (sscanf(line, "%llx %c %s", &start, &c, symbf) == 3) { struct symbol *sym = symbol__new(start, 0x1000000, symbf); if (sym == NULL) goto out_delete_dso; dso__insert_symbol(kernel_dso, sym); } } dsos__add(kernel_dso); free(line); fclose(file); return 0; out_delete_dso: dso__delete(kernel_dso); return -1; } struct map { struct list_head node; uint64_t start; uint64_t end; uint64_t pgoff; struct dso *dso; }; static struct map *map__new(struct mmap_event *event) { struct map *self = malloc(sizeof(*self)); if (self != NULL) { self->start = event->start; self->end = event->start + event->len; self->pgoff = event->pgoff; self->dso = dsos__findnew(event->filename); if (self->dso == NULL) goto out_delete; } return self; out_delete: free(self); return NULL; } static size_t map__fprintf(struct map *self, FILE *fp) { return fprintf(fp, " %lx-%lx %lx %s\n", self->start, self->end, self->pgoff, self->dso->name); } struct symhist { struct list_head node; struct dso *dso; struct symbol *sym; uint32_t count; char level; }; static struct symhist *symhist__new(struct symbol *sym, struct dso *dso, char level) { struct symhist *self = malloc(sizeof(*self)); if (self != NULL) { self->sym = sym; self->dso = dso; self->level = level; self->count = 0; } return self; } static void symhist__delete(struct symhist *self) { free(self); } static bool symhist__equal(struct symhist *self, struct symbol *sym, struct dso *dso, char level) { return self->level == level && self->sym == sym && self->dso == dso; } static void symhist__inc(struct symhist *self) { ++self->count; } static size_t symhist__fprintf(struct symhist *self, FILE *fp) { size_t ret = fprintf(fp, "[%c] ", self->level); if (self->level != '.') ret += fprintf(fp, "%s", self->sym->name); else ret += fprintf(fp, "%s: %s", self->dso ? self->dso->name : "sym ? self->sym->name : ""); return ret + fprintf(fp, ": %u\n", self->count); } struct thread { struct list_head node; struct list_head maps; struct list_head symhists; pid_t pid; char *comm; }; static struct thread *thread__new(pid_t pid) { struct thread *self = malloc(sizeof(*self)); if (self != NULL) { self->pid = pid; self->comm = NULL; INIT_LIST_HEAD(&self->maps); INIT_LIST_HEAD(&self->symhists); } return self; } static void thread__insert_symhist(struct thread *self, struct symhist *symhist) { list_add_tail(&symhist->node, &self->symhists); } static struct symhist *thread__symhists_find(struct thread *self, struct symbol *sym, struct dso *dso, char level) { struct symhist *pos; list_for_each_entry(pos, &self->symhists, node) if (symhist__equal(pos, sym, dso, level)) return pos; return NULL; } static int thread__symbol_incnew(struct thread *self, struct symbol *sym, struct dso *dso, char level) { struct symhist *symhist = thread__symhists_find(self, sym, dso, level); if (symhist == NULL) { symhist = symhist__new(sym, dso, level); if (symhist == NULL) goto out_error; thread__insert_symhist(self, symhist); } symhist__inc(symhist); return 0; out_error: return -ENOMEM; } static int thread__set_comm(struct thread *self, const char *comm) { self->comm = strdup(comm); return self->comm ? 0 : -ENOMEM; } static size_t thread__maps_fprintf(struct thread *self, FILE *fp) { struct map *pos; size_t ret = 0; list_for_each_entry(pos, &self->maps, node) ret += map__fprintf(pos, fp); return ret; } static size_t thread__fprintf(struct thread *self, FILE *fp) { struct symhist *pos; int ret = fprintf(fp, "thread: %d %s\n", self->pid, self->comm); list_for_each_entry(pos, &self->symhists, node) ret += symhist__fprintf(pos, fp); return ret; } static LIST_HEAD(threads); static void threads__add(struct thread *thread) { list_add_tail(&thread->node, &threads); } static struct thread *threads__find(pid_t pid) { struct thread *pos; list_for_each_entry(pos, &threads, node) if (pos->pid == pid) return pos; return NULL; } static struct thread *threads__findnew(pid_t pid) { struct thread *thread = threads__find(pid); if (thread == NULL) { thread = thread__new(pid); if (thread != NULL) threads__add(thread); } return thread; } static void thread__insert_map(struct thread *self, struct map *map) { list_add_tail(&map->node, &self->maps); } static struct map *thread__find_map(struct thread *self, uint64_t ip) { if (self == NULL) return NULL; struct map *pos; list_for_each_entry(pos, &self->maps, node) if (ip >= pos->start && ip <= pos->end) return pos; return NULL; } static void threads__fprintf(FILE *fp) { struct thread *pos; list_for_each_entry(pos, &threads, node) thread__fprintf(pos, fp); } #if 0 static std::string resolve_user_symbol(int pid, uint64_t ip) { std::string sym = ""; maps_t &m = maps[pid]; maps_t::const_iterator mi = m.upper_bound(map(ip)); if (mi == m.end()) return sym; ip -= mi->start + mi->pgoff; symbols_t &s = dsos[mi->dso].syms; symbols_t::const_iterator si = s.upper_bound(symbol(ip)); sym = mi->dso + ": "; if (si == s.begin()) return sym; si--; if (si->start <= ip && ip < si->end) sym = mi->dso + ": " + si->name; #if 0 else if (si->start <= ip) sym = mi->dso + ": ?" + si->name; #endif return sym; } #endif static void display_help(void) { printf( "Usage: perf-report []\n" " -i file --input= # input file\n" ); exit(0); } static void process_options(int argc, char *argv[]) { int error = 0; for (;;) { int option_index = 0; /** Options for getopt */ static struct option long_options[] = { {"input", required_argument, NULL, 'i'}, {"no-user", no_argument, NULL, 'u'}, {"no-kernel", no_argument, NULL, 'k'}, {"no-hv", no_argument, NULL, 'h'}, {NULL, 0, NULL, 0 } }; int c = getopt_long(argc, argv, "+:i:kuh", long_options, &option_index); if (c == -1) break; switch (c) { case 'i': input_name = strdup(optarg); break; case 'k': show_mask &= ~SHOW_KERNEL; break; case 'u': show_mask &= ~SHOW_USER; break; case 'h': show_mask &= ~SHOW_HV; break; default: error = 1; break; } } if (error) display_help(); } int cmd_report(int argc, char **argv) { unsigned long offset = 0; unsigned long head = 0; struct stat stat; char *buf; event_t *event; int ret, rc = EXIT_FAILURE; unsigned long total = 0; elf_version(EV_CURRENT); page_size = getpagesize(); process_options(argc, argv); input = open(input_name, O_RDONLY); if (input < 0) { perror("failed to open file"); exit(-1); } ret = fstat(input, &stat); if (ret < 0) { perror("failed to stat file"); exit(-1); } if (!stat.st_size) { fprintf(stderr, "zero-sized file, nothing to do!\n"); exit(0); } if (load_kallsyms() < 0) { perror("failed to open kallsyms"); return EXIT_FAILURE; } remap: buf = (char *)mmap(NULL, page_size * mmap_window, PROT_READ, MAP_SHARED, input, offset); if (buf == MAP_FAILED) { perror("failed to mmap file"); exit(-1); } more: event = (event_t *)(buf + head); if (head + event->header.size >= page_size * mmap_window) { unsigned long shift = page_size * (head / page_size); int ret; ret = munmap(buf, page_size * mmap_window); assert(ret == 0); offset += shift; head -= shift; goto remap; } if (!event->header.size) { fprintf(stderr, "zero-sized event at file offset %ld\n", offset + head); fprintf(stderr, "skipping %ld bytes of events.\n", stat.st_size - offset - head); goto done; } head += event->header.size; if (event->header.misc & PERF_EVENT_MISC_OVERFLOW) { char level; int show = 0; struct dso *dso = NULL; struct thread *thread = threads__findnew(event->ip.pid); if (thread == NULL) goto done; if (event->header.misc & PERF_EVENT_MISC_KERNEL) { show = SHOW_KERNEL; level = 'k'; dso = kernel_dso; } else if (event->header.misc & PERF_EVENT_MISC_USER) { show = SHOW_USER; level = '.'; struct map *map = thread__find_map(thread, event->ip.ip); if (map != NULL) dso = map->dso; } else { show = SHOW_HV; level = 'H'; } if (show & show_mask) { struct symbol *sym = dso__find_symbol(dso, event->ip.ip); if (thread__symbol_incnew(thread, sym, dso, level)) goto done; } total++; } else switch (event->header.type) { case PERF_EVENT_MMAP: { struct thread *thread = threads__findnew(event->mmap.pid); struct map *map = map__new(&event->mmap); if (thread == NULL || map == NULL ) goto done; thread__insert_map(thread, map); break; } case PERF_EVENT_COMM: { struct thread *thread = threads__findnew(event->comm.pid); if (thread == NULL || thread__set_comm(thread, event->comm.comm)) goto done; break; } } if (offset + head < stat.st_size) goto more; rc = EXIT_SUCCESS; done: close(input); //dsos__fprintf(stdout); threads__fprintf(stdout); #if 0 std::map::iterator hi = hist.begin(); while (hi != hist.end()) { rev_hist.insert(std::pair(hi->second, hi->first)); hist.erase(hi++); } std::multimap::const_iterator ri = rev_hist.begin(); while (ri != rev_hist.end()) { printf(" %5.2f %s\n", (100.0 * ri->first)/total, ri->second.c_str()); ri++; } #endif return rc; }