master
1//! PCG32 - http://www.pcg-random.org/
2//!
3//! PRNG
4
5const std = @import("std");
6const Pcg = @This();
7
8const default_multiplier = 6364136223846793005;
9
10s: u64,
11i: u64,
12
13pub fn init(init_s: u64) Pcg {
14 var pcg = Pcg{
15 .s = undefined,
16 .i = undefined,
17 };
18
19 pcg.seed(init_s);
20 return pcg;
21}
22
23pub fn random(self: *Pcg) std.Random {
24 return std.Random.init(self, fill);
25}
26
27fn next(self: *Pcg) u32 {
28 const l = self.s;
29 self.s = l *% default_multiplier +% (self.i | 1);
30
31 const xor_s: u32 = @truncate(((l >> 18) ^ l) >> 27);
32 const rot: u32 = @intCast(l >> 59);
33
34 return (xor_s >> @as(u5, @intCast(rot))) | (xor_s << @as(u5, @intCast((0 -% rot) & 31)));
35}
36
37fn seed(self: *Pcg, init_s: u64) void {
38 // Pcg requires 128-bits of seed.
39 var gen = std.Random.SplitMix64.init(init_s);
40 self.seedTwo(gen.next(), gen.next());
41}
42
43fn seedTwo(self: *Pcg, init_s: u64, init_i: u64) void {
44 self.s = 0;
45 self.i = (init_s << 1) | 1;
46 self.s = self.s *% default_multiplier +% self.i;
47 self.s +%= init_i;
48 self.s = self.s *% default_multiplier +% self.i;
49}
50
51pub fn fill(self: *Pcg, buf: []u8) void {
52 var i: usize = 0;
53 const aligned_len = buf.len - (buf.len & 3);
54
55 // Complete 4 byte segments.
56 while (i < aligned_len) : (i += 4) {
57 var n = self.next();
58 comptime var j: usize = 0;
59 inline while (j < 4) : (j += 1) {
60 buf[i + j] = @as(u8, @truncate(n));
61 n >>= 8;
62 }
63 }
64
65 // Remaining. (cuts the stream)
66 if (i != buf.len) {
67 var n = self.next();
68 while (i < buf.len) : (i += 1) {
69 buf[i] = @as(u8, @truncate(n));
70 n >>= 8;
71 }
72 }
73}
74
75test "sequence" {
76 var r = Pcg.init(0);
77 const s0: u64 = 0x9394bf54ce5d79de;
78 const s1: u64 = 0x84e9c579ef59bbf7;
79 r.seedTwo(s0, s1);
80
81 const seq = [_]u32{
82 2881561918,
83 3063928540,
84 1199791034,
85 2487695858,
86 1479648952,
87 3247963454,
88 };
89
90 for (seq) |s| {
91 try std.testing.expect(s == r.next());
92 }
93}
94
95test fill {
96 var r = Pcg.init(0);
97 const s0: u64 = 0x9394bf54ce5d79de;
98 const s1: u64 = 0x84e9c579ef59bbf7;
99 r.seedTwo(s0, s1);
100
101 const seq = [_]u32{
102 2881561918,
103 3063928540,
104 1199791034,
105 2487695858,
106 1479648952,
107 3247963454,
108 };
109
110 var i: u32 = 0;
111 while (i < seq.len) : (i += 2) {
112 var buf0: [8]u8 = undefined;
113 std.mem.writeInt(u32, buf0[0..4], seq[i], .little);
114 std.mem.writeInt(u32, buf0[4..8], seq[i + 1], .little);
115
116 var buf1: [7]u8 = undefined;
117 r.fill(&buf1);
118
119 try std.testing.expect(std.mem.eql(u8, buf0[0..7], buf1[0..]));
120 }
121}