v / vlib / math / bits / bits_test.v
403 lines · 360 sloc · 8.63 KB · 9dce7713cfb9380d0d7d8e1eed73bc2c280ccca2
Raw
1// test suite for bits and bits math functions
2module bits
3
4fn test_leading_zeros() {
5 mut i := 0
6
7 // 8 bit
8 i = 1
9 for x in 0 .. 8 {
10 assert leading_zeros_8(u8(u8(i) << x)) == 7 - x
11 }
12 assert leading_zeros_8(0) == 8
13
14 // 16 bit
15 i = 1
16 for x in 0 .. 16 {
17 assert leading_zeros_16(u16(i) << x) == 15 - x
18 }
19 assert leading_zeros_16(0) == 16
20
21 // 32 bit
22 i = 1
23 for x in 0 .. 32 {
24 assert leading_zeros_32(u32(i) << x) == 31 - x
25 }
26 assert leading_zeros_32(0) == 32
27
28 // 64 bit
29 i = 1
30 for x in 0 .. 64 {
31 assert leading_zeros_64(u64(i) << x) == 63 - x
32 }
33 assert leading_zeros_64(0) == 64
34}
35
36fn test_trailing_zeros() {
37 mut i := 0
38 // 8 bit
39 i = 1
40 for x in 0 .. 8 {
41 assert trailing_zeros_8(u8(u8(i) << x)) == x
42 }
43 assert trailing_zeros_8(0) == 8
44
45 // 16 bit
46 i = 1
47 for x in 0 .. 16 {
48 assert trailing_zeros_16(u16(i) << x) == x
49 }
50 assert trailing_zeros_16(0) == 16
51
52 // 32 bit
53 i = 1
54 for x in 0 .. 32 {
55 assert trailing_zeros_32(u32(i) << x) == x
56 }
57 assert trailing_zeros_32(0) == 32
58
59 // 64 bit
60 i = 1
61 for x in 0 .. 64 {
62 assert trailing_zeros_64(u64(i) << x) == x
63 }
64 assert trailing_zeros_64(0) == 64
65}
66
67fn test_ones_count() {
68 mut i := 0
69 mut i1 := u64(0)
70 // 8 bit
71 i = 0
72 for x in 0 .. 9 {
73 assert ones_count_8(u8(i)) == x
74 i = int(u32(i) << 1) + 1
75 }
76 assert ones_count_8(0) == 0
77 assert ones_count_8(0xFF) == 8
78
79 // 16 bit
80 i = 0
81 for x in 0 .. 17 {
82 assert ones_count_16(u16(i)) == x
83 i = int(u32(i) << 1) + 1
84 }
85 assert ones_count_16(0) == 0
86 assert ones_count_16(0xFFFF) == 16
87
88 // 32 bit
89 i = 0
90 for x in 0 .. 33 {
91 assert ones_count_32(u32(i)) == x
92 i = int(u32(i) << 1) + 1
93 }
94 assert ones_count_32(0) == 0
95 assert ones_count_32(0xFFFF_FFFF) == 32
96
97 // 64 bit
98 i1 = 0
99 for x in 0 .. 65 {
100 assert ones_count_64(i1) == x
101 i1 = (i1 << 1) + 1
102 }
103 assert ones_count_64(0) == 0
104 assert ones_count_64(0xFFFF_FFFF_FFFF_FFFF) == 64
105}
106
107fn test_rotate_left_right() {
108 assert rotate_left_8(0x12, 4) == 0x21
109 assert rotate_left_16(0x1234, 8) == 0x3412
110 assert rotate_left_32(0x12345678, 16) == 0x56781234
111 assert rotate_left_64(0x1234567887654321, 32) == 0x8765432112345678
112}
113
114fn test_reverse() {
115 mut i := 0
116 mut i1 := u64(0)
117 // 8 bit
118 i = 0
119 for _ in 0 .. 9 {
120 mut rv := u8(0)
121 mut bc := 0
122 mut n := i
123 for bc < 8 {
124 rv = (rv << 1) | (u8(n) & 0x01)
125 bc++
126 n = n >> 1
127 }
128 assert reverse_8(u8(i)) == rv
129 i = int(u32(i) << 1) + 1
130 }
131
132 // 16 bit
133 i = 0
134 for _ in 0 .. 17 {
135 mut rv := u16(0)
136 mut bc := 0
137 mut n := i
138 for bc < 16 {
139 rv = (rv << 1) | (u16(n) & 0x01)
140 bc++
141 n = n >> 1
142 }
143 assert reverse_16(u16(i)) == rv
144 i = int(u32(i) << 1) + 1
145 }
146
147 // 32 bit
148 i = 0
149 for _ in 0 .. 33 {
150 mut rv := u32(0)
151 mut bc := 0
152 mut n := i
153 for bc < 32 {
154 rv = (rv << 1) | (u32(n) & 0x01)
155 bc++
156 n = n >> 1
157 }
158 assert reverse_32(u32(i)) == rv
159 i = int(u32(i) << 1) + 1
160 }
161
162 // 64 bit
163 i1 = 0
164 for _ in 0 .. 64 {
165 mut rv := u64(0)
166 mut bc := 0
167 mut n := i1
168 for bc < 64 {
169 rv = (rv << 1) | (n & 0x01)
170 bc++
171 n = n >> 1
172 }
173 assert reverse_64(i1) == rv
174 i1 = (i1 << 1) + 1
175 }
176}
177
178fn test_add() {
179 mut i := 0
180 // 32 bit
181 i = 1
182 for x in 0 .. 32 {
183 v := u32(i) << x
184 sum, carry := add_32(v, v, u32(0))
185 assert ((u64(carry) << 32) | u64(sum)) == u64(v) + u64(v)
186 }
187 mut sum_32t, mut carry_32t := add_32(0x8000_0000, 0x8000_0000, u32(0))
188 assert sum_32t == u32(0)
189 assert carry_32t == u32(1)
190
191 sum_32t, carry_32t = add_32(0xFFFF_FFFF, 0xFFFF_FFFF, u32(1))
192 assert sum_32t == 0xFFFF_FFFF
193 assert carry_32t == u32(1)
194
195 // 64 bit
196 i = 1
197 for x in 0 .. 63 {
198 v := u64(i) << x
199 sum, carry := add_64(v, v, u64(0))
200 expected_sum := v + v
201 expected_carry := u64(expected_sum < v)
202 assert sum == expected_sum
203 assert carry == expected_carry
204 }
205 mut sum_64t, mut carry_64t := add_64(0x8000_0000_0000_0000, 0x8000_0000_0000_0000, u64(0))
206 assert sum_64t == u64(0)
207 assert carry_64t == u64(1)
208
209 sum_64t, carry_64t = add_64(0xFFFF_FFFF_FFFF_FFFF, 0xFFFF_FFFF_FFFF_FFFF, u64(1))
210 assert sum_64t == 0xFFFF_FFFF_FFFF_FFFF
211 assert carry_64t == u64(1)
212}
213
214fn test_sub() {
215 mut i := 0
216 // 32 bit
217 i = 1
218 for x in 1 .. 32 {
219 v0 := u32(i) << x
220 v1 := v0 >> 1
221 mut diff, mut borrow_out := sub_32(v0, v1, u32(0))
222 assert diff == v1
223
224 diff, borrow_out = sub_32(v0, v1, u32(1))
225 assert diff == (v1 - 1)
226 assert borrow_out == u32(0)
227
228 diff, borrow_out = sub_32(v1, v0, u32(1))
229 assert borrow_out == u32(1)
230 }
231
232 // 64 bit
233 i = 1
234 for x in 1 .. 64 {
235 v0 := u64(i) << x
236 v1 := v0 >> 1
237 mut diff, mut borrow_out := sub_64(v0, v1, u64(0))
238 assert diff == v1
239
240 diff, borrow_out = sub_64(v0, v1, u64(1))
241 assert diff == (v1 - 1)
242 assert borrow_out == u64(0)
243
244 diff, borrow_out = sub_64(v1, v0, u64(1))
245 assert borrow_out == u64(1)
246 }
247}
248
249fn test_mul() {
250 mut i := 0
251 // 32 bit
252 i = 1
253 for x in 0 .. 32 {
254 v0 := u32(i) << x
255 v1 := v0 - 1
256 hi, lo := mul_32(v0, v1)
257 assert (u64(hi) << 32) | (u64(lo)) == u64(v0) * u64(v1)
258 v2 := u32(x)
259 h, l := mul_add_32(v0, v1, v2)
260 assert (u64(h) << 32) | (u64(l)) == u64(v0) * u64(v1) + u64(v2)
261 }
262
263 // 64 bit
264 i = 1
265 for x in 0 .. 64 {
266 v0 := u64(i) << x
267 v1 := v0 - 1
268 hi, lo := mul_64(v0, v1)
269 exp_hi, exp_lo := mul_64_default(v0, v1)
270 assert hi == exp_hi
271 assert lo == exp_lo
272 v2 := u64(x)
273 h, l := mul_add_64(v0, v1, v2)
274 exp_h, exp_l := mul_add_64_default(v0, v1, v2)
275 assert h == exp_h
276 assert l == exp_l
277 }
278}
279
280fn test_div() {
281 mut i := 0
282 // 32 bit
283 i = 1
284 for x in 0 .. 31 {
285 hi := u32(i) << x
286 lo := hi - 1
287 y := u32(3) << x
288 quo, rem := div_32(hi, lo, y)
289 tst := ((u64(hi) << 32) | u64(lo))
290 assert quo == (tst / u64(y))
291 assert rem == (tst % u64(y))
292 assert rem == rem_32(hi, lo, y)
293 }
294
295 // 64 bit
296 i = 1
297 for x in 0 .. 62 {
298 hi := u64(i) << x
299 lo := u64(2) // hi - 1
300 y := u64(0x4000_0000_0000_0000)
301 quo, rem := div_64(hi, lo, y)
302 assert quo == u64(2) << (x + 1)
303 _, rem1 := div_64(hi % y, lo, y)
304 assert rem == rem1
305 assert rem == rem_64(hi, lo, y)
306 }
307}
308
309fn test_div_64_edge_cases() {
310 qq, rr := div_64(10, 12, 11)
311 assert qq == 16769767339735956015
312 assert rr == 7
313 q, r := div_64(0, 23, 10000000000000000000)
314 assert q == 0
315 assert r == 23
316}
317
318fn test_randomized_arithmetic_properties() {
319 mut state := u64(0x9e3779b97f4a7c15)
320 for _ in 0 .. 2000 {
321 state = next_u64(state)
322 a64 := state
323 state = next_u64(state)
324 b64 := state
325 state = next_u64(state)
326 carry_in64 := state & 1
327 sum64, carry_out64 := add_64(a64, b64, carry_in64)
328 tmp := a64 + b64
329 expected_sum64 := tmp + carry_in64
330 expected_carry64 := u64(tmp < a64) | u64(expected_sum64 < tmp)
331 assert sum64 == expected_sum64
332 assert carry_out64 == expected_carry64
333
334 diff64, borrow_out64 := sub_64(a64, b64, carry_in64)
335 tmp_diff := a64 - b64
336 expected_diff64 := tmp_diff - carry_in64
337 expected_borrow64 := u64(a64 < b64) | u64(tmp_diff < carry_in64)
338 assert diff64 == expected_diff64
339 assert borrow_out64 == expected_borrow64
340
341 mul_hi, mul_lo := mul_64(a64, b64)
342 exp_mul_hi, exp_mul_lo := mul_64_default(a64, b64)
343 assert mul_hi == exp_mul_hi
344 assert mul_lo == exp_mul_lo
345
346 state = next_u64(state)
347 z64 := state
348 mul_add_hi, mul_add_lo := mul_add_64(a64, b64, z64)
349 exp_mul_add_hi, exp_mul_add_lo := mul_add_64_default(a64, b64, z64)
350 assert mul_add_hi == exp_mul_add_hi
351 assert mul_add_lo == exp_mul_add_lo
352
353 state = next_u64(state)
354 mut y64 := state | 1
355 state = next_u64(state)
356 mut hi64 := state
357 hi64 %= y64
358 state = next_u64(state)
359 lo64 := state
360 quo64, rem64 := div_64(hi64, lo64, y64)
361 exp_quo64, exp_rem64 := div_64_default(hi64, lo64, y64)
362 assert quo64 == exp_quo64
363 assert rem64 == exp_rem64
364 assert rem64 == rem_64(hi64, lo64, y64)
365
366 a32 := u32(a64)
367 b32 := u32(b64)
368 carry_in32 := u32(carry_in64)
369 sum32, carry_out32 := add_32(a32, b32, carry_in32)
370 expected32 := u64(a32) + u64(b32) + u64(carry_in32)
371 assert sum32 == u32(expected32)
372 assert carry_out32 == u32(expected32 >> 32)
373
374 diff32, borrow_out32 := sub_32(a32, b32, carry_in32)
375 expected_diff32 := a32 - b32 - carry_in32
376 expected_borrow32 := u32((~a32 & b32) | (~(a32 ^ b32) & expected_diff32)) >> 31
377 assert diff32 == expected_diff32
378 assert borrow_out32 == expected_borrow32
379
380 mut y32 := u32(y64)
381 if y32 == 0 {
382 y32 = 1
383 }
384 state = next_u64(state)
385 hi32 := u32(state % u64(y32))
386 state = next_u64(state)
387 lo32 := u32(state)
388 quo32, rem32 := div_32(hi32, lo32, y32)
389 numerator32 := (u64(hi32) << 32) | u64(lo32)
390 assert quo32 == u32(numerator32 / u64(y32))
391 assert rem32 == u32(numerator32 % u64(y32))
392 assert rem32 == rem_32(hi32, lo32, y32)
393 }
394}
395
396// rem_32 and rem_64 panic when y == 0 (division by zero). This behavior is tested
397// through the randomized property test which guards against y==0, and through manual
398// verification. Direct panic tests are avoided to prevent test suite crashes.
399
400@[inline]
401fn next_u64(state u64) u64 {
402 return state * u64(6364136223846793005) + u64(1442695040888963407)
403}
404