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
use rand::{self, Rng, SeedableRng};

/// The RNG type used by this crate.
///
/// Exists as a type alias to make changing RNG implementation easier.
pub type CrateRng = rand::XorShiftRng;

#[derive(Clone)]
pub struct Seed([u32; 4]);

impl Seed {
    /// Generates a 128-bit seed from a 32-bit seed
    ///
    /// This is done via two steps:
    ///
    /// 1. Create a low quality 128-bit seed (LQS)
    ///
    ///    This is done with simple bit manipulation of the 32-bit seed.
    ///
    /// 2. Create a higher quality 128-bit seed (HQS)
    ///
    ///    This is done by seeding an Rng with the LQS then using the Rng to generate the HQS.
    pub fn from_u32(seed: u32) -> Seed {
        let mut rng = CrateRng::from_seed(
            Seed::from_u32_array([
                seed ^ 0xa5a5_a5a5,
                seed ^ 0x5a5a_5a5a,
                seed ^ 0x5555_5555,
                seed ^ 0xaaaa_aaaa,
            ]).0,
        );

        Seed::from_u32_array([rng.gen(), rng.gen(), rng.gen(), rng.gen()])
    }

    pub fn from_u32_array(x: [u32; 4]) -> Seed {
        Seed(x)
    }

    pub fn to_rng(&self) -> CrateRng {
        CrateRng::from_seed(self.0)
    }
}

impl Default for Seed {
    fn default() -> Seed {
        Seed::from_u32(0)
    }
}