selftests/bpf: Allow some tests to be executed in sequence

This patch allows tests to define serial_test_name() instead of
test_name(), and this will make test_progs execute those in sequence
after all other tests finished executing concurrently.

Signed-off-by: Yucong Sun <sunyucong@gmail.com>
Signed-off-by: Andrii Nakryiko <andrii@kernel.org>
Link: https://lore.kernel.org/bpf/20211006185619.364369-3-fallentree@fb.com
This commit is contained in:
Yucong Sun 2021-10-06 11:56:07 -07:00 committed by Andrii Nakryiko
parent 91b2c0afd0
commit 6587ff58ce
1 changed files with 54 additions and 6 deletions

View File

@ -50,6 +50,7 @@ struct prog_test_def {
const char *test_name; const char *test_name;
int test_num; int test_num;
void (*run_test)(void); void (*run_test)(void);
void (*run_serial_test)(void);
bool force_log; bool force_log;
int error_cnt; int error_cnt;
int skip_cnt; int skip_cnt;
@ -457,7 +458,9 @@ static int load_bpf_testmod(void)
} }
/* extern declarations for test funcs */ /* extern declarations for test funcs */
#define DEFINE_TEST(name) extern void test_##name(void); #define DEFINE_TEST(name) \
extern void test_##name(void) __weak; \
extern void serial_test_##name(void) __weak;
#include <prog_tests/tests.h> #include <prog_tests/tests.h>
#undef DEFINE_TEST #undef DEFINE_TEST
@ -465,6 +468,7 @@ static struct prog_test_def prog_test_defs[] = {
#define DEFINE_TEST(name) { \ #define DEFINE_TEST(name) { \
.test_name = #name, \ .test_name = #name, \
.run_test = &test_##name, \ .run_test = &test_##name, \
.run_serial_test = &serial_test_##name, \
}, },
#include <prog_tests/tests.h> #include <prog_tests/tests.h>
#undef DEFINE_TEST #undef DEFINE_TEST
@ -907,7 +911,10 @@ static void run_one_test(int test_num)
env.test = test; env.test = test;
if (test->run_test)
test->run_test(); test->run_test();
else if (test->run_serial_test)
test->run_serial_test();
/* ensure last sub-test is finalized properly */ /* ensure last sub-test is finalized properly */
if (test->subtest_name) if (test->subtest_name)
@ -957,7 +964,7 @@ static void *dispatch_thread(void *ctx)
pthread_mutex_unlock(&current_test_lock); pthread_mutex_unlock(&current_test_lock);
} }
if (!test->should_run) if (!test->should_run || test->run_serial_test)
continue; continue;
/* run test through worker */ /* run test through worker */
@ -1136,6 +1143,40 @@ static int server_main(void)
free(env.worker_current_test); free(env.worker_current_test);
free(data); free(data);
/* run serial tests */
save_netns();
for (int i = 0; i < prog_test_cnt; i++) {
struct prog_test_def *test = &prog_test_defs[i];
struct test_result *result = &test_results[i];
if (!test->should_run || !test->run_serial_test)
continue;
stdio_hijack();
run_one_test(i);
stdio_restore();
if (env.log_buf) {
result->log_cnt = env.log_cnt;
result->log_buf = strdup(env.log_buf);
free(env.log_buf);
env.log_buf = NULL;
env.log_cnt = 0;
}
restore_netns();
fprintf(stdout, "#%d %s:%s\n",
test->test_num, test->test_name,
test->error_cnt ? "FAIL" : (test->skip_cnt ? "SKIP" : "OK"));
result->error_cnt = test->error_cnt;
result->skip_cnt = test->skip_cnt;
result->sub_succ_cnt = test->sub_succ_cnt;
}
/* generate summary */ /* generate summary */
fflush(stderr); fflush(stderr);
fflush(stdout); fflush(stdout);
@ -1333,6 +1374,13 @@ int main(int argc, char **argv)
test->should_run = true; test->should_run = true;
else else
test->should_run = false; test->should_run = false;
if ((test->run_test == NULL && test->run_serial_test == NULL) ||
(test->run_test != NULL && test->run_serial_test != NULL)) {
fprintf(stderr, "Test %d:%s must have either test_%s() or serial_test_%sl() defined.\n",
test->test_num, test->test_name, test->test_name, test->test_name);
exit(EXIT_ERR_SETUP_INFRA);
}
} }
/* ignore workers if we are just listing */ /* ignore workers if we are just listing */