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