 ebf294bf4f
			
		
	
	
	ebf294bf4f
	
	
	
		
			
			We don't need to have two PATH_MAX char sized arrays holding it, just one in util/debugfs.c will do. Also rename debugfs_path to tracing_events_path, as it is not the path to debugfs, that is debugfs_mountpoint. Both are now accessible. This will allow accessing this code in the perf python binding without having to drag in perf.c and util/parse-events.c. The defaults for these variables are the canonical "/sys/kernel/debug" and "/sys/kernel/debug/tracing/events/", removing the need for simple tools to call debugfs_mount(NULL). Cc: David Ahern <dsahern@gmail.com> Cc: Frederic Weisbecker <fweisbec@gmail.com> Cc: Mike Galbraith <efault@gmx.de> Cc: Paul Mackerras <paulus@samba.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Stephane Eranian <eranian@google.com> Link: http://lkml.kernel.org/n/tip-ug9jvtjrsqbluuhqqxpvg30f@git.kernel.org Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
		
			
				
	
	
		
			255 lines
		
	
	
	
		
			4.9 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			255 lines
		
	
	
	
		
			4.9 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
| #include "util.h"
 | |
| #include "debugfs.h"
 | |
| #include "cache.h"
 | |
| 
 | |
| #include <linux/kernel.h>
 | |
| #include <sys/mount.h>
 | |
| 
 | |
| static int debugfs_premounted;
 | |
| char debugfs_mountpoint[PATH_MAX + 1] = "/sys/kernel/debug";
 | |
| char tracing_events_path[PATH_MAX + 1] = "/sys/kernel/debug/tracing/events";
 | |
| 
 | |
| static const char *debugfs_known_mountpoints[] = {
 | |
| 	"/sys/kernel/debug/",
 | |
| 	"/debug/",
 | |
| 	0,
 | |
| };
 | |
| 
 | |
| /* use this to force a umount */
 | |
| void debugfs_force_cleanup(void)
 | |
| {
 | |
| 	debugfs_find_mountpoint();
 | |
| 	debugfs_premounted = 0;
 | |
| 	debugfs_umount();
 | |
| }
 | |
| 
 | |
| /* construct a full path to a debugfs element */
 | |
| int debugfs_make_path(const char *element, char *buffer, int size)
 | |
| {
 | |
| 	int len;
 | |
| 
 | |
| 	if (strlen(debugfs_mountpoint) == 0) {
 | |
| 		buffer[0] = '\0';
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	len = strlen(debugfs_mountpoint) + strlen(element) + 1;
 | |
| 	if (len >= size)
 | |
| 		return len+1;
 | |
| 
 | |
| 	snprintf(buffer, size-1, "%s/%s", debugfs_mountpoint, element);
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int debugfs_found;
 | |
| 
 | |
| /* find the path to the mounted debugfs */
 | |
| const char *debugfs_find_mountpoint(void)
 | |
| {
 | |
| 	const char **ptr;
 | |
| 	char type[100];
 | |
| 	FILE *fp;
 | |
| 
 | |
| 	if (debugfs_found)
 | |
| 		return (const char *) debugfs_mountpoint;
 | |
| 
 | |
| 	ptr = debugfs_known_mountpoints;
 | |
| 	while (*ptr) {
 | |
| 		if (debugfs_valid_mountpoint(*ptr) == 0) {
 | |
| 			debugfs_found = 1;
 | |
| 			strcpy(debugfs_mountpoint, *ptr);
 | |
| 			return debugfs_mountpoint;
 | |
| 		}
 | |
| 		ptr++;
 | |
| 	}
 | |
| 
 | |
| 	/* give up and parse /proc/mounts */
 | |
| 	fp = fopen("/proc/mounts", "r");
 | |
| 	if (fp == NULL)
 | |
| 		return NULL;
 | |
| 
 | |
| 	while (fscanf(fp, "%*s %" STR(PATH_MAX) "s %99s %*s %*d %*d\n",
 | |
| 		      debugfs_mountpoint, type) == 2) {
 | |
| 		if (strcmp(type, "debugfs") == 0)
 | |
| 			break;
 | |
| 	}
 | |
| 	fclose(fp);
 | |
| 
 | |
| 	if (strcmp(type, "debugfs") != 0)
 | |
| 		return NULL;
 | |
| 
 | |
| 	debugfs_found = 1;
 | |
| 
 | |
| 	return debugfs_mountpoint;
 | |
| }
 | |
| 
 | |
| /* verify that a mountpoint is actually a debugfs instance */
 | |
| 
 | |
| int debugfs_valid_mountpoint(const char *debugfs)
 | |
| {
 | |
| 	struct statfs st_fs;
 | |
| 
 | |
| 	if (statfs(debugfs, &st_fs) < 0)
 | |
| 		return -ENOENT;
 | |
| 	else if (st_fs.f_type != (long) DEBUGFS_MAGIC)
 | |
| 		return -ENOENT;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int debugfs_valid_entry(const char *path)
 | |
| {
 | |
| 	struct stat st;
 | |
| 
 | |
| 	if (stat(path, &st))
 | |
| 		return -errno;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static void debugfs_set_tracing_events_path(const char *mountpoint)
 | |
| {
 | |
| 	snprintf(tracing_events_path, sizeof(tracing_events_path), "%s/%s",
 | |
| 		 mountpoint, "tracing/events");
 | |
| }
 | |
| 
 | |
| /* mount the debugfs somewhere if it's not mounted */
 | |
| 
 | |
| char *debugfs_mount(const char *mountpoint)
 | |
| {
 | |
| 	/* see if it's already mounted */
 | |
| 	if (debugfs_find_mountpoint()) {
 | |
| 		debugfs_premounted = 1;
 | |
| 		goto out;
 | |
| 	}
 | |
| 
 | |
| 	/* if not mounted and no argument */
 | |
| 	if (mountpoint == NULL) {
 | |
| 		/* see if environment variable set */
 | |
| 		mountpoint = getenv(PERF_DEBUGFS_ENVIRONMENT);
 | |
| 		/* if no environment variable, use default */
 | |
| 		if (mountpoint == NULL)
 | |
| 			mountpoint = "/sys/kernel/debug";
 | |
| 	}
 | |
| 
 | |
| 	if (mount(NULL, mountpoint, "debugfs", 0, NULL) < 0)
 | |
| 		return NULL;
 | |
| 
 | |
| 	/* save the mountpoint */
 | |
| 	debugfs_found = 1;
 | |
| 	strncpy(debugfs_mountpoint, mountpoint, sizeof(debugfs_mountpoint));
 | |
| out:
 | |
| 	debugfs_set_tracing_events_path(debugfs_mountpoint);
 | |
| 	return debugfs_mountpoint;
 | |
| }
 | |
| 
 | |
| void debugfs_set_path(const char *mountpoint)
 | |
| {
 | |
| 	snprintf(debugfs_mountpoint, sizeof(debugfs_mountpoint), "%s", mountpoint);
 | |
| 	debugfs_set_tracing_events_path(mountpoint);
 | |
| }
 | |
| 
 | |
| /* umount the debugfs */
 | |
| 
 | |
| int debugfs_umount(void)
 | |
| {
 | |
| 	char umountcmd[128];
 | |
| 	int ret;
 | |
| 
 | |
| 	/* if it was already mounted, leave it */
 | |
| 	if (debugfs_premounted)
 | |
| 		return 0;
 | |
| 
 | |
| 	/* make sure it's a valid mount point */
 | |
| 	ret = debugfs_valid_mountpoint(debugfs_mountpoint);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	snprintf(umountcmd, sizeof(umountcmd),
 | |
| 		 "/bin/umount %s", debugfs_mountpoint);
 | |
| 	return system(umountcmd);
 | |
| }
 | |
| 
 | |
| int debugfs_write(const char *entry, const char *value)
 | |
| {
 | |
| 	char path[PATH_MAX + 1];
 | |
| 	int ret, count;
 | |
| 	int fd;
 | |
| 
 | |
| 	/* construct the path */
 | |
| 	snprintf(path, sizeof(path), "%s/%s", debugfs_mountpoint, entry);
 | |
| 
 | |
| 	/* verify that it exists */
 | |
| 	ret = debugfs_valid_entry(path);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	/* get how many chars we're going to write */
 | |
| 	count = strlen(value);
 | |
| 
 | |
| 	/* open the debugfs entry */
 | |
| 	fd = open(path, O_RDWR);
 | |
| 	if (fd < 0)
 | |
| 		return -errno;
 | |
| 
 | |
| 	while (count > 0) {
 | |
| 		/* write it */
 | |
| 		ret = write(fd, value, count);
 | |
| 		if (ret <= 0) {
 | |
| 			if (ret == EAGAIN)
 | |
| 				continue;
 | |
| 			close(fd);
 | |
| 			return -errno;
 | |
| 		}
 | |
| 		count -= ret;
 | |
| 	}
 | |
| 
 | |
| 	/* close it */
 | |
| 	close(fd);
 | |
| 
 | |
| 	/* return success */
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * read a debugfs entry
 | |
|  * returns the number of chars read or a negative errno
 | |
|  */
 | |
| int debugfs_read(const char *entry, char *buffer, size_t size)
 | |
| {
 | |
| 	char path[PATH_MAX + 1];
 | |
| 	int ret;
 | |
| 	int fd;
 | |
| 
 | |
| 	/* construct the path */
 | |
| 	snprintf(path, sizeof(path), "%s/%s", debugfs_mountpoint, entry);
 | |
| 
 | |
| 	/* verify that it exists */
 | |
| 	ret = debugfs_valid_entry(path);
 | |
| 	if (ret)
 | |
| 		return ret;
 | |
| 
 | |
| 	/* open the debugfs entry */
 | |
| 	fd = open(path, O_RDONLY);
 | |
| 	if (fd < 0)
 | |
| 		return -errno;
 | |
| 
 | |
| 	do {
 | |
| 		/* read it */
 | |
| 		ret = read(fd, buffer, size);
 | |
| 		if (ret == 0) {
 | |
| 			close(fd);
 | |
| 			return EOF;
 | |
| 		}
 | |
| 	} while (ret < 0 && errno == EAGAIN);
 | |
| 
 | |
| 	/* close it */
 | |
| 	close(fd);
 | |
| 
 | |
| 	/* make *sure* there's a null character at the end */
 | |
| 	buffer[ret] = '\0';
 | |
| 
 | |
| 	/* return the number of chars read */
 | |
| 	return ret;
 | |
| }
 |