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