1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
use rand_core;
use rand_core::{RngCore, SeedableRng};
use byteorder::{LittleEndian, ByteOrder};
#[allow(missing_copy_implementations)]
#[derive(Debug, Clone)]
pub struct SplitMix64 {
x: u64,
}
impl SplitMix64 {
pub fn new_unseeded() -> SplitMix64 {
SplitMix64 {
x: 0,
}
}
pub fn from_seed_u64(seed: u64) -> SplitMix64 {
let mut x = [0; 8];
LittleEndian::write_u64(&mut x, seed);
SplitMix64::from_seed(x)
}
}
impl RngCore for SplitMix64 {
#[inline]
fn next_u32(&mut self) -> u32 {
self.next_u64() as u32
}
#[inline]
fn next_u64(&mut self) -> u64 {
self.x = self.x.wrapping_add(0x9e3779b97f4a7c15);
let mut z = self.x;
z = (z ^ (z >> 30)).wrapping_mul(0xbf58476d1ce4e5b9);
z = (z ^ (z >> 27)).wrapping_mul(0x94d049bb133111eb);
return z ^ (z >> 31);
}
#[inline]
fn fill_bytes(&mut self, dest: &mut [u8]) {
for mut chunk in dest.chunks_mut(8) {
if chunk.len() == 8 {
LittleEndian::write_u64(&mut chunk, self.next_u64());
} else {
debug_assert!(chunk.len() < 8);
let r = self.next_u64();
let mut i = 0;
for v in chunk.iter_mut() {
*v = (r >> 8*i) as u8;
i += 1;
}
}
}
}
#[inline]
fn try_fill_bytes(&mut self, dest: &mut [u8]) -> Result<(), rand_core::Error> {
self.fill_bytes(dest);
Ok(())
}
}
impl SeedableRng for SplitMix64 {
type Seed = [u8; 8];
fn from_seed(seed: [u8; 8]) -> SplitMix64 {
SplitMix64 {
x: LittleEndian::read_u64(&seed),
}
}
}