Commit a8fdaad5 authored by Andrii Nakryiko's avatar Andrii Nakryiko Committed by Alexei Starovoitov

selftests/bpf: Integrate verbose verifier log into test_progs

Add exra level of verboseness, activated by -vvv argument. When -vv is
specified, verbose libbpf and verifier log (level 1) is output, even for
successful tests. With -vvv, verifier log goes to level 2.

This is extremely useful to debug verifier failures, as well as just see the
state and flow of verification. Before this, you'd have to go and modify
load_program()'s source code inside libbpf to specify extra log_level flags,
which is suboptimal to say the least.

Currently -vv and -vvv triggering verifier output is integrated into
test_stub's bpf_prog_load as well as bpf_verif_scale.c tests.
Signed-off-by: default avatarAndrii Nakryiko <andriin@fb.com>
Signed-off-by: default avatarAlexei Starovoitov <ast@kernel.org>
Link: https://lore.kernel.org/bpf/20191120003548.4159797-1-andriin@fb.com
parent 5940c5bf
...@@ -15,6 +15,8 @@ static int libbpf_debug_print(enum libbpf_print_level level, ...@@ -15,6 +15,8 @@ static int libbpf_debug_print(enum libbpf_print_level level,
return 0; return 0;
} }
extern int extra_prog_load_log_flags;
static int check_load(const char *file, enum bpf_prog_type type) static int check_load(const char *file, enum bpf_prog_type type)
{ {
struct bpf_prog_load_attr attr; struct bpf_prog_load_attr attr;
...@@ -24,7 +26,7 @@ static int check_load(const char *file, enum bpf_prog_type type) ...@@ -24,7 +26,7 @@ static int check_load(const char *file, enum bpf_prog_type type)
memset(&attr, 0, sizeof(struct bpf_prog_load_attr)); memset(&attr, 0, sizeof(struct bpf_prog_load_attr));
attr.file = file; attr.file = file;
attr.prog_type = type; attr.prog_type = type;
attr.log_level = 4; attr.log_level = 4 | extra_prog_load_log_flags;
attr.prog_flags = BPF_F_TEST_RND_HI32; attr.prog_flags = BPF_F_TEST_RND_HI32;
err = bpf_prog_load_xattr(&attr, &obj, &prog_fd); err = bpf_prog_load_xattr(&attr, &obj, &prog_fd);
bpf_object__close(obj); bpf_object__close(obj);
......
...@@ -45,7 +45,7 @@ static void dump_test_log(const struct prog_test_def *test, bool failed) ...@@ -45,7 +45,7 @@ static void dump_test_log(const struct prog_test_def *test, bool failed)
fflush(stdout); /* exports env.log_buf & env.log_cnt */ fflush(stdout); /* exports env.log_buf & env.log_cnt */
if (env.verbose || test->force_log || failed) { if (env.verbosity > VERBOSE_NONE || test->force_log || failed) {
if (env.log_cnt) { if (env.log_cnt) {
env.log_buf[env.log_cnt] = '\0'; env.log_buf[env.log_cnt] = '\0';
fprintf(env.stdout, "%s", env.log_buf); fprintf(env.stdout, "%s", env.log_buf);
...@@ -346,14 +346,14 @@ static const struct argp_option opts[] = { ...@@ -346,14 +346,14 @@ static const struct argp_option opts[] = {
{ "verifier-stats", ARG_VERIFIER_STATS, NULL, 0, { "verifier-stats", ARG_VERIFIER_STATS, NULL, 0,
"Output verifier statistics", }, "Output verifier statistics", },
{ "verbose", ARG_VERBOSE, "LEVEL", OPTION_ARG_OPTIONAL, { "verbose", ARG_VERBOSE, "LEVEL", OPTION_ARG_OPTIONAL,
"Verbose output (use -vv for extra verbose output)" }, "Verbose output (use -vv or -vvv for progressively verbose output)" },
{}, {},
}; };
static int libbpf_print_fn(enum libbpf_print_level level, static int libbpf_print_fn(enum libbpf_print_level level,
const char *format, va_list args) const char *format, va_list args)
{ {
if (!env.very_verbose && level == LIBBPF_DEBUG) if (env.verbosity < VERBOSE_VERY && level == LIBBPF_DEBUG)
return 0; return 0;
vprintf(format, args); vprintf(format, args);
return 0; return 0;
...@@ -419,6 +419,8 @@ int parse_num_list(const char *s, struct test_selector *sel) ...@@ -419,6 +419,8 @@ int parse_num_list(const char *s, struct test_selector *sel)
return 0; return 0;
} }
extern int extra_prog_load_log_flags;
static error_t parse_arg(int key, char *arg, struct argp_state *state) static error_t parse_arg(int key, char *arg, struct argp_state *state)
{ {
struct test_env *env = state->input; struct test_env *env = state->input;
...@@ -460,9 +462,14 @@ static error_t parse_arg(int key, char *arg, struct argp_state *state) ...@@ -460,9 +462,14 @@ static error_t parse_arg(int key, char *arg, struct argp_state *state)
env->verifier_stats = true; env->verifier_stats = true;
break; break;
case ARG_VERBOSE: case ARG_VERBOSE:
env->verbosity = VERBOSE_NORMAL;
if (arg) { if (arg) {
if (strcmp(arg, "v") == 0) { if (strcmp(arg, "v") == 0) {
env->very_verbose = true; env->verbosity = VERBOSE_VERY;
extra_prog_load_log_flags = 1;
} else if (strcmp(arg, "vv") == 0) {
env->verbosity = VERBOSE_SUPER;
extra_prog_load_log_flags = 2;
} else { } else {
fprintf(stderr, fprintf(stderr,
"Unrecognized verbosity setting ('%s'), only -v and -vv are supported\n", "Unrecognized verbosity setting ('%s'), only -v and -vv are supported\n",
...@@ -470,7 +477,6 @@ static error_t parse_arg(int key, char *arg, struct argp_state *state) ...@@ -470,7 +477,6 @@ static error_t parse_arg(int key, char *arg, struct argp_state *state)
return -EINVAL; return -EINVAL;
} }
} }
env->verbose = true;
break; break;
case ARGP_KEY_ARG: case ARGP_KEY_ARG:
argp_usage(state); argp_usage(state);
...@@ -489,7 +495,7 @@ static void stdio_hijack(void) ...@@ -489,7 +495,7 @@ static void stdio_hijack(void)
env.stdout = stdout; env.stdout = stdout;
env.stderr = stderr; env.stderr = stderr;
if (env.verbose) { if (env.verbosity > VERBOSE_NONE) {
/* nothing to do, output to stdout by default */ /* nothing to do, output to stdout by default */
return; return;
} }
......
...@@ -39,6 +39,13 @@ typedef __u16 __sum16; ...@@ -39,6 +39,13 @@ typedef __u16 __sum16;
#include "trace_helpers.h" #include "trace_helpers.h"
#include "flow_dissector_load.h" #include "flow_dissector_load.h"
enum verbosity {
VERBOSE_NONE,
VERBOSE_NORMAL,
VERBOSE_VERY,
VERBOSE_SUPER,
};
struct test_selector { struct test_selector {
const char *name; const char *name;
bool *num_set; bool *num_set;
...@@ -49,8 +56,7 @@ struct test_env { ...@@ -49,8 +56,7 @@ struct test_env {
struct test_selector test_selector; struct test_selector test_selector;
struct test_selector subtest_selector; struct test_selector subtest_selector;
bool verifier_stats; bool verifier_stats;
bool verbose; enum verbosity verbosity;
bool very_verbose;
bool jit_enabled; bool jit_enabled;
......
...@@ -5,6 +5,8 @@ ...@@ -5,6 +5,8 @@
#include <bpf/libbpf.h> #include <bpf/libbpf.h>
#include <string.h> #include <string.h>
int extra_prog_load_log_flags = 0;
int bpf_prog_test_load(const char *file, enum bpf_prog_type type, int bpf_prog_test_load(const char *file, enum bpf_prog_type type,
struct bpf_object **pobj, int *prog_fd) struct bpf_object **pobj, int *prog_fd)
{ {
...@@ -15,6 +17,7 @@ int bpf_prog_test_load(const char *file, enum bpf_prog_type type, ...@@ -15,6 +17,7 @@ int bpf_prog_test_load(const char *file, enum bpf_prog_type type,
attr.prog_type = type; attr.prog_type = type;
attr.expected_attach_type = 0; attr.expected_attach_type = 0;
attr.prog_flags = BPF_F_TEST_RND_HI32; attr.prog_flags = BPF_F_TEST_RND_HI32;
attr.log_level = extra_prog_load_log_flags;
return bpf_prog_load_xattr(&attr, pobj, prog_fd); return bpf_prog_load_xattr(&attr, pobj, prog_fd);
} }
...@@ -35,6 +38,7 @@ int bpf_test_load_program(enum bpf_prog_type type, const struct bpf_insn *insns, ...@@ -35,6 +38,7 @@ int bpf_test_load_program(enum bpf_prog_type type, const struct bpf_insn *insns,
load_attr.license = license; load_attr.license = license;
load_attr.kern_version = kern_version; load_attr.kern_version = kern_version;
load_attr.prog_flags = BPF_F_TEST_RND_HI32; load_attr.prog_flags = BPF_F_TEST_RND_HI32;
load_attr.log_level = extra_prog_load_log_flags;
return bpf_load_program_xattr(&load_attr, log_buf, log_buf_sz); return bpf_load_program_xattr(&load_attr, log_buf, log_buf_sz);
} }
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