c5a965cb8243bbb191ebe0620a72bdc50c0e4105
[model-checker-benchmarks.git] / seqlock / seqlock.h
1 #include <stdatomic.h>
2 #include <threads.h>
3
4 typedef struct seqlock {
5         // Sequence for reader consistency check
6         atomic_int _seq;
7         // It needs to be atomic to avoid data races
8         atomic_int _data;
9
10         seqlock() {
11                 atomic_init(&_seq, 0);
12                 atomic_init(&_data, 0);
13         }
14
15         int read() {
16                 while (true) {
17                         int old_seq = _seq.load(memory_order_acquire); // acquire
18                         if (old_seq % 2 == 1) continue;
19
20                         int res = _data.load(memory_order_acquire); 
21                         if (_seq.load(memory_order_relaxed) == old_seq) { // relaxed
22                                 return res;
23                         }
24                 }
25         }
26         
27         void write(int new_data) {
28                 int old_seq = _seq.load(memory_order_acquire); // acquire
29                 while (true) {
30                         // This might be a relaxed too
31                         if (old_seq % 2 == 1)
32                                 continue; // Retry
33
34                         if (_seq.compare_exchange_strong(old_seq, old_seq + 1,
35                                 memory_order_acq_rel, memory_order_acquire)) 
36                                 break;
37                 }
38
39                 // Update the data
40                 _data.store(new_data, memory_order_release); // Can be relaxed
41
42                 _seq.fetch_add(1, memory_order_release); // release
43         }
44
45 } seqlock_t;