Author | Tokens | Token Proportion | Commits | Commit Proportion |
---|---|---|---|---|
Adrian Hunter | 958 | 62.70% | 9 | 23.08% |
Jiri Olsa | 290 | 18.98% | 4 | 10.26% |
Ian Rogers | 126 | 8.25% | 5 | 12.82% |
He Kuang | 63 | 4.12% | 1 | 2.56% |
Arnaldo Carvalho de Melo | 42 | 2.75% | 12 | 30.77% |
Waiman Long | 12 | 0.79% | 1 | 2.56% |
Peter Zijlstra | 11 | 0.72% | 2 | 5.13% |
Stéphane Eranian | 9 | 0.59% | 1 | 2.56% |
Ingo Molnar | 8 | 0.52% | 1 | 2.56% |
Florian Fainelli | 5 | 0.33% | 1 | 2.56% |
Frédéric Weisbecker | 3 | 0.20% | 1 | 2.56% |
Greg Kroah-Hartman | 1 | 0.07% | 1 | 2.56% |
Total | 1528 | 39 |
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369
// SPDX-License-Identifier: GPL-2.0 #include <errno.h> #include <unistd.h> #include <stdio.h> #include <string.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <stdlib.h> #include <linux/kernel.h> #include "vdso.h" #include "dso.h" #include <internal/lib.h> #include "map.h" #include "symbol.h" #include "machine.h" #include "thread.h" #include "linux/string.h" #include <linux/zalloc.h> #include "debug.h" /* * Include definition of find_map() also used in perf-read-vdso.c for * building perf-read-vdso32 and perf-read-vdsox32. */ #include "find-map.c" #define VDSO__TEMP_FILE_NAME "/tmp/perf-vdso.so-XXXXXX" struct vdso_file { bool found; bool error; char temp_file_name[sizeof(VDSO__TEMP_FILE_NAME)]; const char *dso_name; const char *read_prog; }; struct vdso_info { struct vdso_file vdso; #if BITS_PER_LONG == 64 struct vdso_file vdso32; struct vdso_file vdsox32; #endif }; static struct vdso_info *vdso_info__new(void) { static const struct vdso_info vdso_info_init = { .vdso = { .temp_file_name = VDSO__TEMP_FILE_NAME, .dso_name = DSO__NAME_VDSO, }, #if BITS_PER_LONG == 64 .vdso32 = { .temp_file_name = VDSO__TEMP_FILE_NAME, .dso_name = DSO__NAME_VDSO32, .read_prog = "perf-read-vdso32", }, .vdsox32 = { .temp_file_name = VDSO__TEMP_FILE_NAME, .dso_name = DSO__NAME_VDSOX32, .read_prog = "perf-read-vdsox32", }, #endif }; return memdup(&vdso_info_init, sizeof(vdso_info_init)); } static char *get_file(struct vdso_file *vdso_file) { char *vdso = NULL; char *buf = NULL; void *start, *end; size_t size; int fd; if (vdso_file->found) return vdso_file->temp_file_name; if (vdso_file->error || find_map(&start, &end, VDSO__MAP_NAME)) return NULL; size = end - start; buf = memdup(start, size); if (!buf) return NULL; fd = mkstemp(vdso_file->temp_file_name); if (fd < 0) goto out; if (size == (size_t) write(fd, buf, size)) vdso = vdso_file->temp_file_name; close(fd); out: free(buf); vdso_file->found = (vdso != NULL); vdso_file->error = !vdso_file->found; return vdso; } void machine__exit_vdso(struct machine *machine) { struct vdso_info *vdso_info = machine->vdso_info; if (!vdso_info) return; if (vdso_info->vdso.found) unlink(vdso_info->vdso.temp_file_name); #if BITS_PER_LONG == 64 if (vdso_info->vdso32.found) unlink(vdso_info->vdso32.temp_file_name); if (vdso_info->vdsox32.found) unlink(vdso_info->vdsox32.temp_file_name); #endif zfree(&machine->vdso_info); } static struct dso *__machine__addnew_vdso(struct machine *machine, const char *short_name, const char *long_name) { struct dso *dso; dso = dso__new(short_name); if (dso != NULL) { __dsos__add(&machine->dsos, dso); dso__set_long_name(dso, long_name, false); } return dso; } struct machine__thread_dso_type_maps_cb_args { struct machine *machine; enum dso_type dso_type; }; static int machine__thread_dso_type_maps_cb(struct map *map, void *data) { struct machine__thread_dso_type_maps_cb_args *args = data; struct dso *dso = map__dso(map); if (!dso || dso__long_name(dso)[0] != '/') return 0; args->dso_type = dso__type(dso, args->machine); return (args->dso_type != DSO__TYPE_UNKNOWN) ? 1 : 0; } static enum dso_type machine__thread_dso_type(struct machine *machine, struct thread *thread) { struct machine__thread_dso_type_maps_cb_args args = { .machine = machine, .dso_type = DSO__TYPE_UNKNOWN, }; maps__for_each_map(thread__maps(thread), machine__thread_dso_type_maps_cb, &args); return args.dso_type; } #if BITS_PER_LONG == 64 static int vdso__do_copy_compat(FILE *f, int fd) { char buf[4096]; size_t count; while (1) { count = fread(buf, 1, sizeof(buf), f); if (ferror(f)) return -errno; if (feof(f)) break; if (count && writen(fd, buf, count) != (ssize_t)count) return -errno; } return 0; } static int vdso__copy_compat(const char *prog, int fd) { FILE *f; int err; f = popen(prog, "r"); if (!f) return -errno; err = vdso__do_copy_compat(f, fd); if (pclose(f) == -1) return -errno; return err; } static int vdso__create_compat_file(const char *prog, char *temp_name) { int fd, err; fd = mkstemp(temp_name); if (fd < 0) return -errno; err = vdso__copy_compat(prog, fd); if (close(fd) == -1) return -errno; return err; } static const char *vdso__get_compat_file(struct vdso_file *vdso_file) { int err; if (vdso_file->found) return vdso_file->temp_file_name; if (vdso_file->error) return NULL; err = vdso__create_compat_file(vdso_file->read_prog, vdso_file->temp_file_name); if (err) { pr_err("%s failed, error %d\n", vdso_file->read_prog, err); vdso_file->error = true; return NULL; } vdso_file->found = true; return vdso_file->temp_file_name; } static struct dso *__machine__findnew_compat(struct machine *machine, struct vdso_file *vdso_file) { const char *file_name; struct dso *dso; dso = dsos__find(&machine->dsos, vdso_file->dso_name, true); if (dso) return dso; file_name = vdso__get_compat_file(vdso_file); if (!file_name) return NULL; return __machine__addnew_vdso(machine, vdso_file->dso_name, file_name); } static int __machine__findnew_vdso_compat(struct machine *machine, struct thread *thread, struct vdso_info *vdso_info, struct dso **dso) { enum dso_type dso_type; dso_type = machine__thread_dso_type(machine, thread); #ifndef HAVE_PERF_READ_VDSO32 if (dso_type == DSO__TYPE_32BIT) return 0; #endif #ifndef HAVE_PERF_READ_VDSOX32 if (dso_type == DSO__TYPE_X32BIT) return 0; #endif switch (dso_type) { case DSO__TYPE_32BIT: *dso = __machine__findnew_compat(machine, &vdso_info->vdso32); return 1; case DSO__TYPE_X32BIT: *dso = __machine__findnew_compat(machine, &vdso_info->vdsox32); return 1; case DSO__TYPE_UNKNOWN: case DSO__TYPE_64BIT: default: return 0; } } #endif static struct dso *machine__find_vdso(struct machine *machine, struct thread *thread) { struct dso *dso = NULL; enum dso_type dso_type; dso_type = machine__thread_dso_type(machine, thread); switch (dso_type) { case DSO__TYPE_32BIT: dso = dsos__find(&machine->dsos, DSO__NAME_VDSO32, true); if (!dso) { dso = dsos__find(&machine->dsos, DSO__NAME_VDSO, true); if (dso && dso_type != dso__type(dso, machine)) { dso__put(dso); dso = NULL; } } break; case DSO__TYPE_X32BIT: dso = dsos__find(&machine->dsos, DSO__NAME_VDSOX32, true); break; case DSO__TYPE_64BIT: case DSO__TYPE_UNKNOWN: default: dso = dsos__find(&machine->dsos, DSO__NAME_VDSO, true); break; } return dso; } struct dso *machine__findnew_vdso(struct machine *machine, struct thread *thread) { struct vdso_info *vdso_info; struct dso *dso = NULL; char *file; if (!machine->vdso_info) machine->vdso_info = vdso_info__new(); vdso_info = machine->vdso_info; if (!vdso_info) return NULL; dso = machine__find_vdso(machine, thread); if (dso) return dso; #if BITS_PER_LONG == 64 if (__machine__findnew_vdso_compat(machine, thread, vdso_info, &dso)) return dso; #endif dso = dsos__find(&machine->dsos, DSO__NAME_VDSO, true); if (dso) return dso; file = get_file(&vdso_info->vdso); if (!file) return NULL; return __machine__addnew_vdso(machine, DSO__NAME_VDSO, file); } bool dso__is_vdso(struct dso *dso) { return !strcmp(dso__short_name(dso), DSO__NAME_VDSO) || !strcmp(dso__short_name(dso), DSO__NAME_VDSO32) || !strcmp(dso__short_name(dso), DSO__NAME_VDSOX32); }
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with Cregit http://github.com/cregit/cregit
Version 2.0-RC1