#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 "linux/string.h" static bool vdso_found; static char vdso_file[] = "/tmp/perf-vdso.so-XXXXXX"; static int find_vdso_map(void **start, void **end) { FILE *maps; char line[128]; int found = 0; maps = fopen("/proc/self/maps", "r"); if (!maps) { pr_err("vdso: cannot open maps\n"); return -1; } while (!found && fgets(line, sizeof(line), maps)) { int m = -1; /* We care only about private r-x mappings. */ if (2 != sscanf(line, "%p-%p r-xp %*x %*x:%*x %*u %n", start, end, &m)) continue; if (m < 0) continue; if (!strncmp(&line[m], VDSO__MAP_NAME, sizeof(VDSO__MAP_NAME) - 1)) found = 1; } fclose(maps); return !found; } static char *get_file(void) { char *vdso = NULL; char *buf = NULL; void *start, *end; size_t size; int fd; if (vdso_found) return vdso_file; if (find_vdso_map(&start, &end)) return NULL; size = end - start; buf = memdup(start, size); if (!buf) return NULL; fd = mkstemp(vdso_file); if (fd < 0) goto out; if (size == (size_t) write(fd, buf, size)) vdso = vdso_file; close(fd); out: free(buf); vdso_found = (vdso != NULL); return vdso; } void vdso__exit(void) { if (vdso_found) unlink(vdso_file); } struct dso *vdso__dso_findnew(struct list_head *head) { struct dso *dso = dsos__find(head, VDSO__MAP_NAME); if (!dso) { char *file; file = get_file(); if (!file) return NULL; dso = dso__new(VDSO__MAP_NAME); if (dso != NULL) { dsos__add(head, dso); dso__set_long_name(dso, file); } } return dso; }