xref: /aosp_15_r20/external/compiler-rt/test/tsan/bench.h (revision 7c3d14c8b49c529e04be81a3ce6f5cc23712e4c6)
1*7c3d14c8STreehugger Robot #include <pthread.h>
2*7c3d14c8STreehugger Robot #include <stdlib.h>
3*7c3d14c8STreehugger Robot #include <stddef.h>
4*7c3d14c8STreehugger Robot #include <unistd.h>
5*7c3d14c8STreehugger Robot #include <stdio.h>
6*7c3d14c8STreehugger Robot #include <time.h>
7*7c3d14c8STreehugger Robot 
8*7c3d14c8STreehugger Robot int bench_nthread;
9*7c3d14c8STreehugger Robot int bench_niter;
10*7c3d14c8STreehugger Robot int grow_clock_var;
11*7c3d14c8STreehugger Robot pthread_barrier_t glow_clock_barrier;
12*7c3d14c8STreehugger Robot 
13*7c3d14c8STreehugger Robot void bench();  // defined by user
14*7c3d14c8STreehugger Robot void start_thread_group(int nth, void(*f)(int tid));
15*7c3d14c8STreehugger Robot void grow_clock_worker(int tid);
16*7c3d14c8STreehugger Robot 
main(int argc,char ** argv)17*7c3d14c8STreehugger Robot int main(int argc, char **argv) {
18*7c3d14c8STreehugger Robot   bench_nthread = 2;
19*7c3d14c8STreehugger Robot   if (argc > 1)
20*7c3d14c8STreehugger Robot     bench_nthread = atoi(argv[1]);
21*7c3d14c8STreehugger Robot   bench_niter = 100;
22*7c3d14c8STreehugger Robot   if (argc > 2)
23*7c3d14c8STreehugger Robot     bench_niter = atoi(argv[2]);
24*7c3d14c8STreehugger Robot 
25*7c3d14c8STreehugger Robot   // Grow thread's clock.
26*7c3d14c8STreehugger Robot   int clock_size = 10;
27*7c3d14c8STreehugger Robot   if (argc > 1)
28*7c3d14c8STreehugger Robot     clock_size = 1000;
29*7c3d14c8STreehugger Robot   pthread_barrier_init(&glow_clock_barrier, 0, clock_size);
30*7c3d14c8STreehugger Robot   start_thread_group(clock_size, grow_clock_worker);
31*7c3d14c8STreehugger Robot   pthread_barrier_destroy(&glow_clock_barrier);
32*7c3d14c8STreehugger Robot   __atomic_load_n(&grow_clock_var, __ATOMIC_ACQUIRE);
33*7c3d14c8STreehugger Robot 
34*7c3d14c8STreehugger Robot   timespec tp0;
35*7c3d14c8STreehugger Robot   clock_gettime(CLOCK_MONOTONIC, &tp0);
36*7c3d14c8STreehugger Robot   bench();
37*7c3d14c8STreehugger Robot   timespec tp1;
38*7c3d14c8STreehugger Robot   clock_gettime(CLOCK_MONOTONIC, &tp1);
39*7c3d14c8STreehugger Robot   unsigned long long t =
40*7c3d14c8STreehugger Robot       (tp1.tv_sec * 1000000000ULL + tp1.tv_nsec) -
41*7c3d14c8STreehugger Robot       (tp0.tv_sec * 1000000000ULL + tp0.tv_nsec);
42*7c3d14c8STreehugger Robot   fprintf(stderr, "%llu ns/iter\n", t / bench_niter);
43*7c3d14c8STreehugger Robot   fprintf(stderr, "DONE\n");
44*7c3d14c8STreehugger Robot }
45*7c3d14c8STreehugger Robot 
start_thread_group(int nth,void (* f)(int tid))46*7c3d14c8STreehugger Robot void start_thread_group(int nth, void(*f)(int tid)) {
47*7c3d14c8STreehugger Robot   pthread_t *th = (pthread_t*)malloc(nth * sizeof(pthread_t));
48*7c3d14c8STreehugger Robot   for (int i = 0; i < nth; i++)
49*7c3d14c8STreehugger Robot     pthread_create(&th[i], 0, (void*(*)(void*))f, (void*)(long)i);
50*7c3d14c8STreehugger Robot   for (int i = 0; i < nth; i++)
51*7c3d14c8STreehugger Robot     pthread_join(th[i], 0);
52*7c3d14c8STreehugger Robot }
53*7c3d14c8STreehugger Robot 
grow_clock_worker(int tid)54*7c3d14c8STreehugger Robot void grow_clock_worker(int tid) {
55*7c3d14c8STreehugger Robot   int res = pthread_barrier_wait(&glow_clock_barrier);
56*7c3d14c8STreehugger Robot   if (res == PTHREAD_BARRIER_SERIAL_THREAD)
57*7c3d14c8STreehugger Robot     __atomic_store_n(&grow_clock_var, 0, __ATOMIC_RELEASE);
58*7c3d14c8STreehugger Robot }
59*7c3d14c8STreehugger Robot 
60