2022-01-04 12:21:08 +03:00
|
|
|
// Copyright (c) 2019-2022 Alexander Medvednikov. All rights reserved.
|
2019-09-17 22:03:54 +03:00
|
|
|
// Use of this source code is governed by an MIT license
|
|
|
|
// that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
module rand
|
|
|
|
|
2020-04-26 12:56:27 +03:00
|
|
|
import math.bits
|
|
|
|
import encoding.binary
|
2019-09-17 22:03:54 +03:00
|
|
|
|
2021-02-05 21:26:34 +03:00
|
|
|
// int_u64 returns a random unsigned 64-bit integer `u64` read from a real OS source of entropy.
|
2019-09-23 23:17:06 +03:00
|
|
|
pub fn int_u64(max u64) ?u64 {
|
2020-02-09 11:25:27 +03:00
|
|
|
bitlen := bits.len_64(max)
|
2019-09-17 22:03:54 +03:00
|
|
|
if bitlen == 0 {
|
|
|
|
return u64(0)
|
|
|
|
}
|
|
|
|
k := (bitlen + 7) / 8
|
|
|
|
mut b := u64(bitlen % 8)
|
|
|
|
if b == u64(0) {
|
|
|
|
b = u64(8)
|
|
|
|
}
|
|
|
|
mut n := u64(0)
|
|
|
|
for {
|
2021-02-05 21:26:34 +03:00
|
|
|
mut bytes := read(k) ?
|
2022-04-15 14:58:56 +03:00
|
|
|
bytes[0] &= u8(int(u64(1) << b) - 1)
|
2019-09-17 22:03:54 +03:00
|
|
|
x := bytes_to_u64(bytes)
|
|
|
|
n = x[0]
|
2019-09-18 16:12:16 +03:00
|
|
|
// NOTE: maybe until we have bigint could do it another way?
|
|
|
|
// if x.len > 1 {
|
|
|
|
// n = u64(u32(x[1])<<u32(32)) | n
|
|
|
|
// }
|
2019-09-17 22:03:54 +03:00
|
|
|
if n < max {
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return n
|
|
|
|
}
|
|
|
|
|
2022-04-15 15:35:35 +03:00
|
|
|
fn bytes_to_u64(b []u8) []u64 {
|
2021-02-05 21:26:34 +03:00
|
|
|
ws := 64 / 8
|
|
|
|
mut z := []u64{len: ((b.len + ws - 1) / ws)}
|
2019-09-17 22:03:54 +03:00
|
|
|
mut i := b.len
|
2019-09-18 16:12:16 +03:00
|
|
|
for k := 0; i >= ws; k++ {
|
2021-02-05 21:26:34 +03:00
|
|
|
z[k] = binary.big_endian_u64(b[i - ws..i])
|
2019-09-18 16:12:16 +03:00
|
|
|
i -= ws
|
2019-09-17 22:03:54 +03:00
|
|
|
}
|
|
|
|
if i > 0 {
|
|
|
|
mut d := u64(0)
|
|
|
|
for s := u64(0); i > 0; s += u64(8) {
|
2021-02-05 21:26:34 +03:00
|
|
|
d |= u64(b[i - 1]) << s
|
2019-09-17 22:03:54 +03:00
|
|
|
i--
|
|
|
|
}
|
2021-02-05 21:26:34 +03:00
|
|
|
z[z.len - 1] = d
|
2019-09-17 22:03:54 +03:00
|
|
|
}
|
|
|
|
return z
|
|
|
|
}
|