cregit-Linux how code gets into the kernel

Release 4.10 tools/perf/util/vdso.c

Directory: tools/perf/util
#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 "util.h"
#include "symbol.h"
#include "machine.h"
#include "thread.h"
#include "linux/string.h"
#include "debug.h"

/*
 * Include definition of find_vdso_map() also used in perf-read-vdso.c for
 * building perf-read-vdso32 and perf-read-vdsox32.
 */
#include "find-vdso-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)); }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter9197.85%480.00%
jiri olsajiri olsa22.15%120.00%
Total93100.00%5100.00%


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_vdso_map(&start, &end)) 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; }

Contributors

PersonTokensPropCommitsCommitProp
jiri olsajiri olsa13482.21%150.00%
adrian hunteradrian hunter2917.79%150.00%
Total163100.00%2100.00%


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); }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter7885.71%360.00%
jiri olsajiri olsa1213.19%120.00%
arnaldo carvalho de meloarnaldo carvalho de melo11.10%120.00%
Total91100.00%5100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter6295.38%133.33%
arnaldo carvalho de meloarnaldo carvalho de melo34.62%266.67%
Total65100.00%3100.00%


static enum dso_type machine__thread_dso_type(struct machine *machine, struct thread *thread) { enum dso_type dso_type = DSO__TYPE_UNKNOWN; struct map *map; struct dso *dso; map = map_groups__first(thread->mg, MAP__FUNCTION); for (; map ; map = map_groups__next(map)) { dso = map->dso; if (!dso || dso->long_name[0] != '/') continue; dso_type = dso__type(dso, machine); if (dso_type != DSO__TYPE_UNKNOWN) break; } return dso_type; }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter8384.69%466.67%
jiri olsajiri olsa1414.29%116.67%
waiman longwaiman long11.02%116.67%
Total98100.00%6100.00%

#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; }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter7079.55%266.67%
jiri olsajiri olsa1820.45%133.33%
Total88100.00%3100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter65100.00%2100.00%
Total65100.00%2100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter63100.00%1100.00%
Total63100.00%1100.00%


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; }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter86100.00%1100.00%
Total86100.00%1100.00%


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) goto out; file_name = vdso__get_compat_file(vdso_file); if (!file_name) goto out; dso = __machine__addnew_vdso(machine, vdso_file->dso_name, file_name); out: return dso; }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter7083.33%250.00%
arnaldo carvalho de meloarnaldo carvalho de melo1416.67%250.00%
Total84100.00%4100.00%


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; } }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter11997.54%266.67%
arnaldo carvalho de meloarnaldo carvalho de melo32.46%133.33%
Total122100.00%3100.00%

#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 = 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; }

Contributors

PersonTokensPropCommitsCommitProp
he kuanghe kuang143100.00%1100.00%
Total143100.00%1100.00%


struct dso *machine__findnew_vdso(struct machine *machine, struct thread *thread) { struct vdso_info *vdso_info; struct dso *dso = NULL; pthread_rwlock_wrlock(&machine->dsos.lock); if (!machine->vdso_info) machine->vdso_info = vdso_info__new(); vdso_info = machine->vdso_info; if (!vdso_info) goto out_unlock; dso = machine__find_vdso(machine, thread); if (dso) goto out_unlock; #if BITS_PER_LONG == 64 if (__machine__findnew_vdso_compat(machine, thread, vdso_info, &dso)) goto out_unlock; #endif dso = __dsos__find(&machine->dsos, DSO__NAME_VDSO, true); if (!dso) { char *file; file = get_file(&vdso_info->vdso); if (file) dso = __machine__addnew_vdso(machine, DSO__NAME_VDSO, file); } out_unlock: dso__get(dso); pthread_rwlock_unlock(&machine->dsos.lock); return dso; }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter12168.36%116.67%
arnaldo carvalho de meloarnaldo carvalho de melo4022.60%466.67%
he kuanghe kuang169.04%116.67%
Total177100.00%6100.00%


bool dso__is_vdso(struct dso *dso) { return !strcmp(dso->short_name, DSO__NAME_VDSO) || !strcmp(dso->short_name, DSO__NAME_VDSO32) || !strcmp(dso->short_name, DSO__NAME_VDSOX32); }

Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter41100.00%2100.00%
Total41100.00%2100.00%


Overall Contributors

PersonTokensPropCommitsCommitProp
adrian hunteradrian hunter104870.19%1055.56%
jiri olsajiri olsa21914.67%211.11%
he kuanghe kuang16410.98%15.56%
arnaldo carvalho de meloarnaldo carvalho de melo614.09%422.22%
waiman longwaiman long10.07%15.56%
Total1493100.00%18100.00%
Directory: tools/perf/util
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.