v2 / vlib / sync / stdatomic / atomic_test.v
278 lines · 250 sloc · 8.19 KB · a9f4a942fb7d30e187593e3ba0e7865137e3680b
Raw
1import sync
2import sync.stdatomic
3
4const iterations_per_cycle = 100_000
5
6struct Counter {
7mut:
8 counter u64
9}
10
11// without proper synchronization this would fail
12fn test_count_10_times_1_cycle_should_result_10_cycles_with_sync() {
13 desired_iterations := 10 * iterations_per_cycle
14 mut wg := sync.new_waitgroup()
15 mut counter := &Counter{}
16 wg.add(10)
17 for i := 0; i < 10; i++ {
18 spawn count_one_cycle(mut counter, mut wg)
19 }
20 wg.wait()
21 assert counter.counter == u64(desired_iterations)
22 eprintln(' with synchronization the counter is: ${counter.counter:10} , expectedly == ${desired_iterations:10}')
23}
24
25// This test just to make sure that we have an anti-test to prove it works
26fn test_count_10_times_1_cycle_should_not_be_10_cycles_without_sync() {
27 desired_iterations := 10 * iterations_per_cycle
28 mut wg := sync.new_waitgroup()
29 mut counter := &Counter{}
30 wg.add(10)
31 for i := 0; i < 10; i++ {
32 spawn count_one_cycle_without_sync(mut counter, mut wg)
33 }
34 wg.wait()
35 // Note: we do not assert here, just print, because sometimes by chance counter.counter may be == desired_iterations
36 eprintln('without synchronization the counter is: ${counter.counter:10} , expectedly != ${desired_iterations:10}')
37}
38
39fn test_atomic_count_plus_one_u64() {
40 mut c := u64(0)
41 assert stdatomic.add_u64(&c, 1) == 1
42}
43
44fn test_atomic_count_plus_one_i64() {
45 mut c := i64(0)
46 assert stdatomic.add_i64(&c, 1) == 1
47}
48
49fn test_atomic_count_plus_greater_than_one_u64() {
50 mut c := u64(0)
51 assert stdatomic.add_u64(&c, 10) == 10
52}
53
54fn test_atomic_count_plus_greater_than_one_i64() {
55 mut c := i64(0)
56 assert stdatomic.add_i64(&c, 10) == 10
57}
58
59fn test_atomic_count_minus_one_u64() {
60 mut c := u64(1)
61 assert stdatomic.sub_u64(&c, 1) == 0
62}
63
64fn test_atomic_count_minus_one_i64() {
65 mut c := i64(0)
66 assert stdatomic.sub_i64(&c, 1) == -1
67}
68
69fn test_atomic_count_minus_greater_than_one_u64() {
70 mut c := u64(0)
71 stdatomic.store_u64(&c, 10)
72 assert stdatomic.sub_u64(&c, 10) == 0
73}
74
75fn test_atomic_count_minus_greater_than_one_i64() {
76 mut c := i64(0)
77 stdatomic.store_i64(&c, 10)
78 assert stdatomic.sub_i64(&c, 20) == -10
79}
80
81// count_one_cycle counts the common counter iterations_per_cycle times in thread-safe way
82fn count_one_cycle(mut counter Counter, mut group sync.WaitGroup) {
83 for i := 0; i < iterations_per_cycle; i++ {
84 stdatomic.add_u64(&counter.counter, 1)
85 }
86 group.done()
87}
88
89// count_one_cycle_without_sync counts the common counter iterations_per_cycle times in none thread-safe way
90fn count_one_cycle_without_sync(mut counter Counter, mut group sync.WaitGroup) {
91 for i := 0; i < iterations_per_cycle; i++ {
92 counter.counter++
93 }
94 group.done()
95}
96
97fn test_atomic_vals() {
98 mut v_bool := stdatomic.new_atomic(false)
99 v_bool.store(true)
100 assert v_bool.load() == true
101 v_bool.store(false)
102 assert v_bool.load() == false
103 assert v_bool.swap(true) == false
104 assert v_bool.swap(false) == true
105 assert v_bool.compare_and_swap(false, true) == true
106 assert v_bool.load() == true
107
108 mut v_i8 := stdatomic.new_atomic(i8(-33))
109 v_i8.store(-34)
110 assert v_i8.load() == -34
111 assert v_i8.add(10) == -34
112 assert v_i8.load() == -24
113 assert v_i8.sub(7) == -24
114 assert v_i8.load() == -31
115 mut new_i8 := i8(-20)
116 assert v_i8.swap(new_i8) == -31
117 assert v_i8.swap(-31) == new_i8
118 assert v_i8.compare_and_swap(-31, new_i8) == true
119 assert v_i8.compare_and_swap(new_i8, -32) == true
120 assert v_i8.load() == -32
121
122 mut v_u8 := stdatomic.new_atomic(u8(33))
123 v_u8.store(34)
124 assert v_u8.load() == 34
125 assert v_u8.add(10) == 34
126 assert v_u8.load() == 44
127 assert v_u8.sub(7) == 44
128 assert v_u8.load() == 37
129 mut new_u8 := u8(20)
130 assert v_u8.swap(new_u8) == 37
131 assert v_u8.swap(37) == new_u8
132 assert v_u8.compare_and_swap(37, new_u8) == true
133 assert v_u8.compare_and_swap(new_u8, 38) == true
134 assert v_u8.load() == 38
135
136 mut v_i16 := stdatomic.new_atomic(i16(-333))
137 v_i16.store(-334)
138 assert v_i16.load() == -334
139 assert v_i16.add(10) == -334
140 assert v_i16.load() == -324
141 assert v_i16.sub(7) == -324
142 assert v_i16.load() == -331
143 mut new_i16 := i16(-200)
144 assert v_i16.swap(new_i16) == -331
145 assert v_i16.swap(-331) == new_i16
146 assert v_i16.compare_and_swap(-331, new_i16) == true
147 assert v_i16.compare_and_swap(new_i16, -332) == true
148 assert v_i16.load() == -332
149
150 mut v_u16 := stdatomic.new_atomic(u16(333))
151 v_u16.store(334)
152 assert v_u16.load() == 334
153 assert v_u16.add(10) == 334
154 assert v_u16.load() == 344
155 assert v_u16.sub(7) == 344
156 assert v_u16.load() == 337
157 mut new_u16 := u16(200)
158 assert v_u16.swap(new_u16) == 337
159 assert v_u16.swap(337) == new_u16
160 assert v_u16.compare_and_swap(337, new_u16) == true
161 assert v_u16.compare_and_swap(new_u16, 332) == true
162 assert v_u16.load() == 332
163
164 mut v_i32 := stdatomic.new_atomic(i32(-3333))
165 v_i32.store(-3334)
166 assert v_i32.load() == -3334
167 assert v_i32.add(10) == -3334
168 assert v_i32.load() == -3324
169 assert v_i32.sub(7) == -3324
170 assert v_i32.load() == -3331
171 mut new_i32 := i32(-2000)
172 assert v_i32.swap(new_i32) == -3331
173 assert v_i32.swap(-3331) == new_i32
174 assert v_i32.compare_and_swap(-3331, new_i32) == true
175 assert v_i32.compare_and_swap(new_i32, -3332) == true
176 assert v_i32.load() == -3332
177
178 mut v_u32 := stdatomic.new_atomic(u32(3333))
179 v_u32.store(3334)
180 assert v_u32.load() == 3334
181 assert v_u32.add(10) == 3334
182 assert v_u32.load() == 3344
183 assert v_u32.sub(7) == 3344
184 assert v_u32.load() == 3337
185 mut new_u32 := u32(2000)
186 assert v_u32.swap(new_u32) == 3337
187 assert v_u32.swap(3337) == new_u32
188 assert v_u32.compare_and_swap(3337, new_u32) == true
189 assert v_u32.compare_and_swap(new_u32, 3338) == true
190 assert v_u32.load() == 3338
191
192 mut v_i64 := stdatomic.new_atomic(i64(-33333))
193 v_i64.store(-33334)
194 assert v_i64.load() == -33334
195 assert v_i64.add(10) == -33334
196 assert v_i64.load() == -33324
197 assert v_i64.sub(7) == -33324
198 assert v_i64.load() == -33331
199 mut new_i64 := i64(-20000)
200 assert v_i64.swap(new_i64) == -33331
201 assert v_i64.swap(-33331) == new_i64
202 assert v_i64.compare_and_swap(-33331, new_i64) == true
203 assert v_i64.compare_and_swap(new_i64, -33332) == true
204 assert v_i64.load() == -33332
205
206 mut v_u64 := stdatomic.new_atomic(u64(33333))
207 v_u64.store(33334)
208 assert v_u64.load() == 33334
209 assert v_u64.add(10) == 33334
210 assert v_u64.load() == 33344
211 assert v_u64.sub(7) == 33344
212 assert v_u64.load() == 33337
213 mut new_u64 := u64(20000)
214 assert v_u64.swap(new_u64) == 33337
215 assert v_u64.swap(33337) == new_u64
216 assert v_u64.compare_and_swap(33337, new_u64) == true
217 assert v_u64.compare_and_swap(new_u64, 33338) == true
218 assert v_u64.load() == 33338
219
220 mut v_int := stdatomic.new_atomic(int(-44))
221 v_int.store(-45)
222 assert v_int.load() == -45
223 assert v_int.add(10) == -45
224 assert v_int.load() == -35
225 assert v_int.sub(7) == -35
226 assert v_int.load() == -42
227 mut new_int := int(-40000)
228 assert v_int.swap(new_int) == -42
229 assert v_int.swap(-42) == new_int
230 assert v_int.compare_and_swap(-42, new_int) == true
231 assert v_int.compare_and_swap(new_int, -40001) == true
232 assert v_int.load() == -40001
233
234 mut v_isize := stdatomic.new_atomic(isize(-55))
235 v_isize.store(-56)
236 assert v_isize.load() == -56
237 assert v_isize.add(10) == -56
238 assert v_isize.load() == -46
239 assert v_isize.sub(7) == -46
240 assert v_isize.load() == -53
241 mut new_isize := isize(-50000)
242 assert v_isize.swap(new_isize) == -53
243 assert v_isize.swap(-53) == new_isize
244 assert v_isize.compare_and_swap(-53, new_isize) == true
245 assert v_isize.compare_and_swap(new_isize, -50001) == true
246 assert v_isize.load() == -50001
247
248 mut v_usize := stdatomic.new_atomic(usize(55))
249 v_usize.store(56)
250 assert v_usize.load() == 56
251 assert v_usize.add(10) == 56
252 assert v_usize.load() == 66
253 assert v_usize.sub(7) == 66
254 assert v_usize.load() == 59
255 mut new_usize := usize(50000)
256 assert v_usize.swap(new_usize) == 59
257 assert v_usize.swap(59) == new_usize
258 assert v_usize.compare_and_swap(59, new_usize) == true
259 assert v_usize.compare_and_swap(new_usize, 60) == true
260 assert v_usize.load() == 60
261
262 mut val_1 := int(100)
263 mut ptr_1 := voidptr(&val_1)
264 mut val_2 := int(200)
265 mut ptr_2 := voidptr(&val_2)
266 mut v_voidptr := stdatomic.new_atomic(ptr_1)
267 assert v_voidptr.load() == ptr_1
268 v_voidptr.store(ptr_2)
269 assert v_voidptr.load() == ptr_2
270 assert v_voidptr.swap(ptr_1) == ptr_2
271 assert v_voidptr.swap(ptr_2) == ptr_1
272 assert v_voidptr.compare_and_swap(ptr_2, ptr_1) == true
273 assert v_voidptr.load() == ptr_1
274
275 // just for compile
276 C.atomic_thread_fence(C.memory_order_relaxed)
277 C.cpu_relax()
278}
279