at v4.0 3.3 kB view raw
1#define _GNU_SOURCE 2#include <errno.h> 3#include <stdio.h> 4#include <stdlib.h> 5#include <string.h> 6#include <stdbool.h> 7#include <sys/vfs.h> 8#include <sys/mount.h> 9#include <linux/kernel.h> 10 11#include "debugfs.h" 12 13char debugfs_mountpoint[PATH_MAX + 1] = "/sys/kernel/debug"; 14 15static const char * const debugfs_known_mountpoints[] = { 16 "/sys/kernel/debug", 17 "/debug", 18 0, 19}; 20 21static bool debugfs_found; 22 23/* find the path to the mounted debugfs */ 24const char *debugfs_find_mountpoint(void) 25{ 26 const char * const *ptr; 27 char type[100]; 28 FILE *fp; 29 30 if (debugfs_found) 31 return (const char *)debugfs_mountpoint; 32 33 ptr = debugfs_known_mountpoints; 34 while (*ptr) { 35 if (debugfs_valid_mountpoint(*ptr) == 0) { 36 debugfs_found = true; 37 strcpy(debugfs_mountpoint, *ptr); 38 return debugfs_mountpoint; 39 } 40 ptr++; 41 } 42 43 /* give up and parse /proc/mounts */ 44 fp = fopen("/proc/mounts", "r"); 45 if (fp == NULL) 46 return NULL; 47 48 while (fscanf(fp, "%*s %" STR(PATH_MAX) "s %99s %*s %*d %*d\n", 49 debugfs_mountpoint, type) == 2) { 50 if (strcmp(type, "debugfs") == 0) 51 break; 52 } 53 fclose(fp); 54 55 if (strcmp(type, "debugfs") != 0) 56 return NULL; 57 58 debugfs_found = true; 59 60 return debugfs_mountpoint; 61} 62 63/* verify that a mountpoint is actually a debugfs instance */ 64 65int debugfs_valid_mountpoint(const char *debugfs) 66{ 67 struct statfs st_fs; 68 69 if (statfs(debugfs, &st_fs) < 0) 70 return -ENOENT; 71 else if ((long)st_fs.f_type != (long)DEBUGFS_MAGIC) 72 return -ENOENT; 73 74 return 0; 75} 76 77/* mount the debugfs somewhere if it's not mounted */ 78char *debugfs_mount(const char *mountpoint) 79{ 80 /* see if it's already mounted */ 81 if (debugfs_find_mountpoint()) 82 goto out; 83 84 /* if not mounted and no argument */ 85 if (mountpoint == NULL) { 86 /* see if environment variable set */ 87 mountpoint = getenv(PERF_DEBUGFS_ENVIRONMENT); 88 /* if no environment variable, use default */ 89 if (mountpoint == NULL) 90 mountpoint = "/sys/kernel/debug"; 91 } 92 93 if (mount(NULL, mountpoint, "debugfs", 0, NULL) < 0) 94 return NULL; 95 96 /* save the mountpoint */ 97 debugfs_found = true; 98 strncpy(debugfs_mountpoint, mountpoint, sizeof(debugfs_mountpoint)); 99out: 100 return debugfs_mountpoint; 101} 102 103int debugfs__strerror_open(int err, char *buf, size_t size, const char *filename) 104{ 105 char sbuf[128]; 106 107 switch (err) { 108 case ENOENT: 109 if (debugfs_found) { 110 snprintf(buf, size, 111 "Error:\tFile %s/%s not found.\n" 112 "Hint:\tPerhaps this kernel misses some CONFIG_ setting to enable this feature?.\n", 113 debugfs_mountpoint, filename); 114 break; 115 } 116 snprintf(buf, size, "%s", 117 "Error:\tUnable to find debugfs\n" 118 "Hint:\tWas your kernel compiled with debugfs support?\n" 119 "Hint:\tIs the debugfs filesystem mounted?\n" 120 "Hint:\tTry 'sudo mount -t debugfs nodev /sys/kernel/debug'"); 121 break; 122 case EACCES: 123 snprintf(buf, size, 124 "Error:\tNo permissions to read %s/%s\n" 125 "Hint:\tTry 'sudo mount -o remount,mode=755 %s'\n", 126 debugfs_mountpoint, filename, debugfs_mountpoint); 127 break; 128 default: 129 snprintf(buf, size, "%s", strerror_r(err, sbuf, sizeof(sbuf))); 130 break; 131 } 132 133 return 0; 134} 135 136int debugfs__strerror_open_tp(int err, char *buf, size_t size, const char *sys, const char *name) 137{ 138 char path[PATH_MAX]; 139 140 snprintf(path, PATH_MAX, "tracing/events/%s/%s", sys, name ?: "*"); 141 142 return debugfs__strerror_open(err, buf, size, path); 143}