Commit a137401d authored by Daniel Borkmann's avatar Daniel Borkmann

Merge branch 'bpf-bpftool-mount-tracefs'

Quentin Monnet says:

====================
This series focus on mounting (or not mounting) tracefs with bpftool.

First patch makes bpftool attempt to mount tracefs if tracefs is not
found when running "bpftool prog tracelog".

Second patch adds an option to bpftool to prevent it from attempting
to mount any file system (tracefs or bpffs), in case this behaviour
is undesirable for some users.
====================
Signed-off-by: default avatarDaniel Borkmann <daniel@iogearbox.net>
parents 0d7410ea 33221307
...@@ -128,6 +128,10 @@ OPTIONS ...@@ -128,6 +128,10 @@ OPTIONS
-f, --bpffs -f, --bpffs
Show file names of pinned maps. Show file names of pinned maps.
-n, --nomount
Do not automatically attempt to mount any virtual file system
(such as tracefs or BPF virtual file system) when necessary.
EXAMPLES EXAMPLES
======== ========
**# bpftool map show** **# bpftool map show**
......
...@@ -161,6 +161,10 @@ OPTIONS ...@@ -161,6 +161,10 @@ OPTIONS
-m, --mapcompat -m, --mapcompat
Allow loading maps with unknown map definitions. Allow loading maps with unknown map definitions.
-n, --nomount
Do not automatically attempt to mount any virtual file system
(such as tracefs or BPF virtual file system) when necessary.
EXAMPLES EXAMPLES
======== ========
**# bpftool prog show** **# bpftool prog show**
......
...@@ -60,6 +60,10 @@ OPTIONS ...@@ -60,6 +60,10 @@ OPTIONS
-m, --mapcompat -m, --mapcompat
Allow loading maps with unknown map definitions. Allow loading maps with unknown map definitions.
-n, --nomount
Do not automatically attempt to mount any virtual file system
(such as tracefs or BPF virtual file system) when necessary.
SEE ALSO SEE ALSO
======== ========
......
...@@ -76,7 +76,8 @@ void set_max_rlimit(void) ...@@ -76,7 +76,8 @@ void set_max_rlimit(void)
setrlimit(RLIMIT_MEMLOCK, &rinf); setrlimit(RLIMIT_MEMLOCK, &rinf);
} }
static int mnt_bpffs(const char *target, char *buff, size_t bufflen) static int
mnt_fs(const char *target, const char *type, char *buff, size_t bufflen)
{ {
bool bind_done = false; bool bind_done = false;
...@@ -98,15 +99,29 @@ static int mnt_bpffs(const char *target, char *buff, size_t bufflen) ...@@ -98,15 +99,29 @@ static int mnt_bpffs(const char *target, char *buff, size_t bufflen)
bind_done = true; bind_done = true;
} }
if (mount("bpf", target, "bpf", 0, "mode=0700")) { if (mount(type, target, type, 0, "mode=0700")) {
snprintf(buff, bufflen, "mount -t bpf bpf %s failed: %s", snprintf(buff, bufflen, "mount -t %s %s %s failed: %s",
target, strerror(errno)); type, type, target, strerror(errno));
return -1; return -1;
} }
return 0; return 0;
} }
int mount_tracefs(const char *target)
{
char err_str[ERR_MAX_LEN];
int err;
err = mnt_fs(target, "tracefs", err_str, ERR_MAX_LEN);
if (err) {
err_str[ERR_MAX_LEN - 1] = '\0';
p_err("can't mount tracefs: %s", err_str);
}
return err;
}
int open_obj_pinned(char *path, bool quiet) int open_obj_pinned(char *path, bool quiet)
{ {
int fd; int fd;
...@@ -162,7 +177,13 @@ int mount_bpffs_for_pin(const char *name) ...@@ -162,7 +177,13 @@ int mount_bpffs_for_pin(const char *name)
/* nothing to do if already mounted */ /* nothing to do if already mounted */
goto out_free; goto out_free;
err = mnt_bpffs(dir, err_str, ERR_MAX_LEN); if (block_mount) {
p_err("no BPF file system found, not mounting it due to --nomount option");
err = -1;
goto out_free;
}
err = mnt_fs(dir, "bpf", err_str, ERR_MAX_LEN);
if (err) { if (err) {
err_str[ERR_MAX_LEN - 1] = '\0'; err_str[ERR_MAX_LEN - 1] = '\0';
p_err("can't mount BPF file system to pin the object (%s): %s", p_err("can't mount BPF file system to pin the object (%s): %s",
......
...@@ -24,6 +24,7 @@ json_writer_t *json_wtr; ...@@ -24,6 +24,7 @@ json_writer_t *json_wtr;
bool pretty_output; bool pretty_output;
bool json_output; bool json_output;
bool show_pinned; bool show_pinned;
bool block_mount;
int bpf_flags; int bpf_flags;
struct pinned_obj_table prog_table; struct pinned_obj_table prog_table;
struct pinned_obj_table map_table; struct pinned_obj_table map_table;
...@@ -313,6 +314,7 @@ int main(int argc, char **argv) ...@@ -313,6 +314,7 @@ int main(int argc, char **argv)
{ "version", no_argument, NULL, 'V' }, { "version", no_argument, NULL, 'V' },
{ "bpffs", no_argument, NULL, 'f' }, { "bpffs", no_argument, NULL, 'f' },
{ "mapcompat", no_argument, NULL, 'm' }, { "mapcompat", no_argument, NULL, 'm' },
{ "nomount", no_argument, NULL, 'n' },
{ 0 } { 0 }
}; };
int opt, ret; int opt, ret;
...@@ -321,13 +323,14 @@ int main(int argc, char **argv) ...@@ -321,13 +323,14 @@ int main(int argc, char **argv)
pretty_output = false; pretty_output = false;
json_output = false; json_output = false;
show_pinned = false; show_pinned = false;
block_mount = false;
bin_name = argv[0]; bin_name = argv[0];
hash_init(prog_table.table); hash_init(prog_table.table);
hash_init(map_table.table); hash_init(map_table.table);
opterr = 0; opterr = 0;
while ((opt = getopt_long(argc, argv, "Vhpjfm", while ((opt = getopt_long(argc, argv, "Vhpjfmn",
options, NULL)) >= 0) { options, NULL)) >= 0) {
switch (opt) { switch (opt) {
case 'V': case 'V':
...@@ -354,6 +357,9 @@ int main(int argc, char **argv) ...@@ -354,6 +357,9 @@ int main(int argc, char **argv)
case 'm': case 'm':
bpf_flags = MAPS_RELAX_COMPAT; bpf_flags = MAPS_RELAX_COMPAT;
break; break;
case 'n':
block_mount = true;
break;
default: default:
p_err("unrecognized option '%s'", argv[optind - 1]); p_err("unrecognized option '%s'", argv[optind - 1]);
if (json_output) if (json_output)
......
...@@ -44,7 +44,8 @@ ...@@ -44,7 +44,8 @@
#define HELP_SPEC_PROGRAM \ #define HELP_SPEC_PROGRAM \
"PROG := { id PROG_ID | pinned FILE | tag PROG_TAG }" "PROG := { id PROG_ID | pinned FILE | tag PROG_TAG }"
#define HELP_SPEC_OPTIONS \ #define HELP_SPEC_OPTIONS \
"OPTIONS := { {-j|--json} [{-p|--pretty}] | {-f|--bpffs} | {-m|--mapcompat}" "OPTIONS := { {-j|--json} [{-p|--pretty}] | {-f|--bpffs} |\n" \
"\t {-m|--mapcompat} | {-n|--nomount} }"
#define HELP_SPEC_MAP \ #define HELP_SPEC_MAP \
"MAP := { id MAP_ID | pinned FILE }" "MAP := { id MAP_ID | pinned FILE }"
...@@ -85,6 +86,7 @@ extern const char *bin_name; ...@@ -85,6 +86,7 @@ extern const char *bin_name;
extern json_writer_t *json_wtr; extern json_writer_t *json_wtr;
extern bool json_output; extern bool json_output;
extern bool show_pinned; extern bool show_pinned;
extern bool block_mount;
extern int bpf_flags; extern int bpf_flags;
extern struct pinned_obj_table prog_table; extern struct pinned_obj_table prog_table;
extern struct pinned_obj_table map_table; extern struct pinned_obj_table map_table;
...@@ -98,6 +100,8 @@ void usage(void) __noreturn; ...@@ -98,6 +100,8 @@ void usage(void) __noreturn;
void set_max_rlimit(void); void set_max_rlimit(void);
int mount_tracefs(const char *target);
struct pinned_obj_table { struct pinned_obj_table {
DECLARE_HASHTABLE(table, 16); DECLARE_HASHTABLE(table, 16);
}; };
......
...@@ -54,7 +54,7 @@ find_tracefs_mnt_single(unsigned long magic, char *mnt, const char *mntpt) ...@@ -54,7 +54,7 @@ find_tracefs_mnt_single(unsigned long magic, char *mnt, const char *mntpt)
return true; return true;
} }
static bool find_tracefs_pipe(char *mnt) static bool get_tracefs_pipe(char *mnt)
{ {
static const char * const known_mnts[] = { static const char * const known_mnts[] = {
"/sys/kernel/debug/tracing", "/sys/kernel/debug/tracing",
...@@ -88,7 +88,20 @@ static bool find_tracefs_pipe(char *mnt) ...@@ -88,7 +88,20 @@ static bool find_tracefs_pipe(char *mnt)
fclose(fp); fclose(fp);
/* The string from fscanf() might be truncated, check mnt is valid */ /* The string from fscanf() might be truncated, check mnt is valid */
if (!found || validate_tracefs_mnt(mnt, TRACEFS_MAGIC)) if (found && validate_tracefs_mnt(mnt, TRACEFS_MAGIC))
goto exit_found;
if (block_mount)
return false;
p_info("could not find tracefs, attempting to mount it now");
/* Most of the time, tracefs is automatically mounted by debugfs at
* /sys/kernel/debug/tracing when we try to access it. If we could not
* find it, it is likely that debugfs is not mounted. Let's give one
* attempt at mounting just tracefs at /sys/kernel/tracing.
*/
strcpy(mnt, known_mnts[1]);
if (mount_tracefs(mnt))
return false; return false;
exit_found: exit_found:
...@@ -115,17 +128,13 @@ int do_tracelog(int argc, char **argv) ...@@ -115,17 +128,13 @@ int do_tracelog(int argc, char **argv)
.sa_handler = exit_tracelog .sa_handler = exit_tracelog
}; };
char trace_pipe[PATH_MAX]; char trace_pipe[PATH_MAX];
bool found_trace_pipe;
size_t buff_len = 0; size_t buff_len = 0;
if (json_output) if (json_output)
jsonw_start_array(json_wtr); jsonw_start_array(json_wtr);
found_trace_pipe = find_tracefs_pipe(trace_pipe); if (!get_tracefs_pipe(trace_pipe))
if (!found_trace_pipe) {
p_err("could not find trace pipe, tracefs not mounted?");
return -1; return -1;
}
trace_pipe_fd = fopen(trace_pipe, "r"); trace_pipe_fd = fopen(trace_pipe, "r");
if (!trace_pipe_fd) { if (!trace_pipe_fd) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment