xref: /openbmc/linux/tools/testing/selftests/bpf/bench.h (revision da77ae2b27ec73a644624a6d4bffc206e2df6bb8)
18e7c2a02SAndrii Nakryiko /* SPDX-License-Identifier: GPL-2.0 */
28e7c2a02SAndrii Nakryiko #pragma once
38e7c2a02SAndrii Nakryiko #include <stdlib.h>
48e7c2a02SAndrii Nakryiko #include <stdbool.h>
58e7c2a02SAndrii Nakryiko #include <linux/err.h>
68e7c2a02SAndrii Nakryiko #include <errno.h>
78e7c2a02SAndrii Nakryiko #include <unistd.h>
88e7c2a02SAndrii Nakryiko #include <bpf/bpf.h>
98e7c2a02SAndrii Nakryiko #include <bpf/libbpf.h>
108e7c2a02SAndrii Nakryiko #include <math.h>
118e7c2a02SAndrii Nakryiko #include <time.h>
128e7c2a02SAndrii Nakryiko #include <sys/syscall.h>
138e7c2a02SAndrii Nakryiko 
148e7c2a02SAndrii Nakryiko struct cpu_set {
158e7c2a02SAndrii Nakryiko 	bool *cpus;
168e7c2a02SAndrii Nakryiko 	int cpus_len;
178e7c2a02SAndrii Nakryiko 	int next_cpu;
188e7c2a02SAndrii Nakryiko };
198e7c2a02SAndrii Nakryiko 
208e7c2a02SAndrii Nakryiko struct env {
218e7c2a02SAndrii Nakryiko 	char *bench_name;
228e7c2a02SAndrii Nakryiko 	int duration_sec;
238e7c2a02SAndrii Nakryiko 	int warmup_sec;
248e7c2a02SAndrii Nakryiko 	bool verbose;
258e7c2a02SAndrii Nakryiko 	bool list;
268e7c2a02SAndrii Nakryiko 	bool affinity;
2790c22503SAnton Protopopov 	bool quiet;
288e7c2a02SAndrii Nakryiko 	int consumer_cnt;
298e7c2a02SAndrii Nakryiko 	int producer_cnt;
30*da77ae2bSHou Tao 	int nr_cpus;
318e7c2a02SAndrii Nakryiko 	struct cpu_set prod_cpus;
328e7c2a02SAndrii Nakryiko 	struct cpu_set cons_cpus;
338e7c2a02SAndrii Nakryiko };
348e7c2a02SAndrii Nakryiko 
352b4b2621SDave Marchevsky struct basic_stats {
362b4b2621SDave Marchevsky 	double mean;
372b4b2621SDave Marchevsky 	double stddev;
382b4b2621SDave Marchevsky };
392b4b2621SDave Marchevsky 
408e7c2a02SAndrii Nakryiko struct bench_res {
418e7c2a02SAndrii Nakryiko 	long hits;
428e7c2a02SAndrii Nakryiko 	long drops;
4357fd1c63SJoanne Koong 	long false_hits;
4473087489SDave Marchevsky 	long important_hits;
452b4b2621SDave Marchevsky 	unsigned long gp_ns;
462b4b2621SDave Marchevsky 	unsigned long gp_ct;
472b4b2621SDave Marchevsky 	unsigned int stime;
488e7c2a02SAndrii Nakryiko };
498e7c2a02SAndrii Nakryiko 
508e7c2a02SAndrii Nakryiko struct bench {
518e7c2a02SAndrii Nakryiko 	const char *name;
5222ff7aeaSAnton Protopopov 	const struct argp *argp;
539a93bf3fSHou Tao 	void (*validate)(void);
549a93bf3fSHou Tao 	void (*setup)(void);
558e7c2a02SAndrii Nakryiko 	void *(*producer_thread)(void *ctx);
568e7c2a02SAndrii Nakryiko 	void *(*consumer_thread)(void *ctx);
578e7c2a02SAndrii Nakryiko 	void (*measure)(struct bench_res* res);
588e7c2a02SAndrii Nakryiko 	void (*report_progress)(int iter, struct bench_res* res, long delta_ns);
598e7c2a02SAndrii Nakryiko 	void (*report_final)(struct bench_res res[], int res_cnt);
608e7c2a02SAndrii Nakryiko };
618e7c2a02SAndrii Nakryiko 
628e7c2a02SAndrii Nakryiko struct counter {
638e7c2a02SAndrii Nakryiko 	long value;
648e7c2a02SAndrii Nakryiko } __attribute__((aligned(128)));
658e7c2a02SAndrii Nakryiko 
668e7c2a02SAndrii Nakryiko extern struct env env;
678e7c2a02SAndrii Nakryiko extern const struct bench *bench;
688e7c2a02SAndrii Nakryiko 
699a93bf3fSHou Tao void setup_libbpf(void);
708e7c2a02SAndrii Nakryiko void hits_drops_report_progress(int iter, struct bench_res *res, long delta_ns);
718e7c2a02SAndrii Nakryiko void hits_drops_report_final(struct bench_res res[], int res_cnt);
7257fd1c63SJoanne Koong void false_hits_report_progress(int iter, struct bench_res *res, long delta_ns);
7357fd1c63SJoanne Koong void false_hits_report_final(struct bench_res res[], int res_cnt);
74ec151037SJoanne Koong void ops_report_progress(int iter, struct bench_res *res, long delta_ns);
75ec151037SJoanne Koong void ops_report_final(struct bench_res res[], int res_cnt);
7673087489SDave Marchevsky void local_storage_report_progress(int iter, struct bench_res *res,
7773087489SDave Marchevsky 				   long delta_ns);
7873087489SDave Marchevsky void local_storage_report_final(struct bench_res res[], int res_cnt);
792b4b2621SDave Marchevsky void grace_period_latency_basic_stats(struct bench_res res[], int res_cnt,
802b4b2621SDave Marchevsky 				      struct basic_stats *gp_stat);
812b4b2621SDave Marchevsky void grace_period_ticks_basic_stats(struct bench_res res[], int res_cnt,
822b4b2621SDave Marchevsky 				    struct basic_stats *gp_stat);
838e7c2a02SAndrii Nakryiko 
849a93bf3fSHou Tao static inline __u64 get_time_ns(void)
859a93bf3fSHou Tao {
868e7c2a02SAndrii Nakryiko 	struct timespec t;
878e7c2a02SAndrii Nakryiko 
888e7c2a02SAndrii Nakryiko 	clock_gettime(CLOCK_MONOTONIC, &t);
898e7c2a02SAndrii Nakryiko 
908e7c2a02SAndrii Nakryiko 	return (u64)t.tv_sec * 1000000000 + t.tv_nsec;
918e7c2a02SAndrii Nakryiko }
928e7c2a02SAndrii Nakryiko 
938e7c2a02SAndrii Nakryiko static inline void atomic_inc(long *value)
948e7c2a02SAndrii Nakryiko {
958e7c2a02SAndrii Nakryiko 	(void)__atomic_add_fetch(value, 1, __ATOMIC_RELAXED);
968e7c2a02SAndrii Nakryiko }
978e7c2a02SAndrii Nakryiko 
988e7c2a02SAndrii Nakryiko static inline void atomic_add(long *value, long n)
998e7c2a02SAndrii Nakryiko {
1008e7c2a02SAndrii Nakryiko 	(void)__atomic_add_fetch(value, n, __ATOMIC_RELAXED);
1018e7c2a02SAndrii Nakryiko }
1028e7c2a02SAndrii Nakryiko 
1038e7c2a02SAndrii Nakryiko static inline long atomic_swap(long *value, long n)
1048e7c2a02SAndrii Nakryiko {
1058e7c2a02SAndrii Nakryiko 	return __atomic_exchange_n(value, n, __ATOMIC_RELAXED);
1068e7c2a02SAndrii Nakryiko }
107