Commit 943b043a authored by Andrii Nakryiko's avatar Andrii Nakryiko Committed by Daniel Borkmann

selftests/bpf: Fix bench runner SIGSEGV

Some benchmarks don't have either "consumer" or "producer" sides. For
example, trig-tp and other BPF triggering benchmarks don't have
consumers, as they only do "producing" by calling into syscall or
predefined uproes. As such it's valid for some benchmarks to have zero
consumers or producers. So allows to specify `-c0` explicitly.

This triggers another problem. If benchmark doesn't support either
consumer or producer side, consumer_thread/producer_thread callback will
be NULL, but benchmark runner will attempt to use those NULL callback to
create threads anyways. So instead of crashing with SIGSEGV in case of
misconfigured benchmark, detect the condition and report error.
Signed-off-by: default avatarAndrii Nakryiko <andrii@kernel.org>
Signed-off-by: default avatarDaniel Borkmann <daniel@iogearbox.net>
Acked-by: default avatarEduard Zingerman <eddyz87@gmail.com>
Link: https://lore.kernel.org/bpf/20240201172027.604869-6-andrii@kernel.org
parent b9551da8
...@@ -323,14 +323,14 @@ static error_t parse_arg(int key, char *arg, struct argp_state *state) ...@@ -323,14 +323,14 @@ static error_t parse_arg(int key, char *arg, struct argp_state *state)
break; break;
case 'p': case 'p':
env.producer_cnt = strtol(arg, NULL, 10); env.producer_cnt = strtol(arg, NULL, 10);
if (env.producer_cnt <= 0) { if (env.producer_cnt < 0) {
fprintf(stderr, "Invalid producer count: %s\n", arg); fprintf(stderr, "Invalid producer count: %s\n", arg);
argp_usage(state); argp_usage(state);
} }
break; break;
case 'c': case 'c':
env.consumer_cnt = strtol(arg, NULL, 10); env.consumer_cnt = strtol(arg, NULL, 10);
if (env.consumer_cnt <= 0) { if (env.consumer_cnt < 0) {
fprintf(stderr, "Invalid consumer count: %s\n", arg); fprintf(stderr, "Invalid consumer count: %s\n", arg);
argp_usage(state); argp_usage(state);
} }
...@@ -607,6 +607,10 @@ static void setup_benchmark(void) ...@@ -607,6 +607,10 @@ static void setup_benchmark(void)
bench->setup(); bench->setup();
for (i = 0; i < env.consumer_cnt; i++) { for (i = 0; i < env.consumer_cnt; i++) {
if (!bench->consumer_thread) {
fprintf(stderr, "benchmark doesn't support consumers!\n");
exit(1);
}
err = pthread_create(&state.consumers[i], NULL, err = pthread_create(&state.consumers[i], NULL,
bench->consumer_thread, (void *)(long)i); bench->consumer_thread, (void *)(long)i);
if (err) { if (err) {
...@@ -626,6 +630,10 @@ static void setup_benchmark(void) ...@@ -626,6 +630,10 @@ static void setup_benchmark(void)
env.prod_cpus.next_cpu = env.cons_cpus.next_cpu; env.prod_cpus.next_cpu = env.cons_cpus.next_cpu;
for (i = 0; i < env.producer_cnt; i++) { for (i = 0; i < env.producer_cnt; i++) {
if (!bench->producer_thread) {
fprintf(stderr, "benchmark doesn't support producers!\n");
exit(1);
}
err = pthread_create(&state.producers[i], NULL, err = pthread_create(&state.producers[i], NULL,
bench->producer_thread, (void *)(long)i); bench->producer_thread, (void *)(long)i);
if (err) { if (err) {
......
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