#include "util.h" #include "debugfs.h" #include "cache.h" #include #include 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; }