1*d11b16e1SNico Weber // Synthetic benchmark for __tsan_read/write{1,2,4,8}.
2*d11b16e1SNico Weber // As compared to mini_bench_local/shared.cc this benchmark passes through
3*d11b16e1SNico Weber // deduplication logic (ContainsSameAccess).
4*d11b16e1SNico Weber // First argument is access size (1, 2, 4, 8). Second optional arg switches
5*d11b16e1SNico Weber // from writes to reads.
6*d11b16e1SNico Weber
7*d11b16e1SNico Weber #include <pthread.h>
8*d11b16e1SNico Weber #include <stdlib.h>
9*d11b16e1SNico Weber #include <stdio.h>
10*d11b16e1SNico Weber #include <unistd.h>
11*d11b16e1SNico Weber #include <linux/futex.h>
12*d11b16e1SNico Weber #include <sys/syscall.h>
13*d11b16e1SNico Weber #include <sys/time.h>
14*d11b16e1SNico Weber
15*d11b16e1SNico Weber template<typename T, bool write>
thread(void * arg)16*d11b16e1SNico Weber void* thread(void *arg) {
17*d11b16e1SNico Weber const int kSize = 2 << 10;
18*d11b16e1SNico Weber static volatile long data[kSize];
19*d11b16e1SNico Weber static volatile long turn;
20*d11b16e1SNico Weber const int kRepeat = 1 << 17;
21*d11b16e1SNico Weber const int id = !!arg;
22*d11b16e1SNico Weber for (int i = 0; i < kRepeat; i++) {
23*d11b16e1SNico Weber for (;;) {
24*d11b16e1SNico Weber int t = __atomic_load_n(&turn, __ATOMIC_ACQUIRE);
25*d11b16e1SNico Weber if (t == id)
26*d11b16e1SNico Weber break;
27*d11b16e1SNico Weber syscall(SYS_futex, &turn, FUTEX_WAIT, t, 0, 0, 0);
28*d11b16e1SNico Weber }
29*d11b16e1SNico Weber for (int j = 0; j < kSize; j++) {
30*d11b16e1SNico Weber if (write) {
31*d11b16e1SNico Weber ((volatile T*)&data[j])[0] = 1;
32*d11b16e1SNico Weber ((volatile T*)&data[j])[sizeof(T) == 8 ? 0 : 1] = 1;
33*d11b16e1SNico Weber } else {
34*d11b16e1SNico Weber T v0 = ((volatile T*)&data[j])[0];
35*d11b16e1SNico Weber T v1 = ((volatile T*)&data[j])[sizeof(T) == 8 ? 0 : 1];
36*d11b16e1SNico Weber (void)v0;
37*d11b16e1SNico Weber (void)v1;
38*d11b16e1SNico Weber }
39*d11b16e1SNico Weber }
40*d11b16e1SNico Weber __atomic_store_n(&turn, 1 - id, __ATOMIC_RELEASE);
41*d11b16e1SNico Weber syscall(SYS_futex, &turn, FUTEX_WAKE, 0, 0, 0, 0);
42*d11b16e1SNico Weber }
43*d11b16e1SNico Weber return 0;
44*d11b16e1SNico Weber }
45*d11b16e1SNico Weber
46*d11b16e1SNico Weber template<typename T, bool write>
test()47*d11b16e1SNico Weber void test() {
48*d11b16e1SNico Weber pthread_t th;
49*d11b16e1SNico Weber pthread_create(&th, 0, thread<T, write>, (void*)1);
50*d11b16e1SNico Weber thread<T, write>(0);
51*d11b16e1SNico Weber pthread_join(th, 0);
52*d11b16e1SNico Weber }
53*d11b16e1SNico Weber
54*d11b16e1SNico Weber template<bool write>
testw(int size)55*d11b16e1SNico Weber void testw(int size) {
56*d11b16e1SNico Weber switch (size) {
57*d11b16e1SNico Weber case 1: return test<char, write>();
58*d11b16e1SNico Weber case 2: return test<short, write>();
59*d11b16e1SNico Weber case 4: return test<int, write>();
60*d11b16e1SNico Weber case 8: return test<long long, write>();
61*d11b16e1SNico Weber }
62*d11b16e1SNico Weber }
63*d11b16e1SNico Weber
main(int argc,char ** argv)64*d11b16e1SNico Weber int main(int argc, char** argv) {
65*d11b16e1SNico Weber int size = 8;
66*d11b16e1SNico Weber bool write = true;
67*d11b16e1SNico Weber if (argc > 1) {
68*d11b16e1SNico Weber size = atoi(argv[1]);
69*d11b16e1SNico Weber if (size != 1 && size != 2 && size != 4 && size != 8)
70*d11b16e1SNico Weber size = 8;
71*d11b16e1SNico Weber }
72*d11b16e1SNico Weber if (argc > 2)
73*d11b16e1SNico Weber write = false;
74*d11b16e1SNico Weber printf("%s%d\n", write ? "write" : "read", size);
75*d11b16e1SNico Weber if (write)
76*d11b16e1SNico Weber testw<true>(size);
77*d11b16e1SNico Weber else
78*d11b16e1SNico Weber testw<false>(size);
79*d11b16e1SNico Weber return 0;
80*d11b16e1SNico Weber }
81