Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
hrydgard
GitHub Repository: hrydgard/ppsspp
Path: blob/master/Common/Data/Random/Rng.h
3188 views
1
#pragma once
2
3
#include <cstdint>
4
#include "Common/Swap.h"
5
6
// George Marsaglia-style random number generator.
7
class GMRng {
8
public:
9
GMRng() {
10
m_w = 0x23E866ED;
11
m_z = 0x80FD5AF2;
12
}
13
void Init(int seed) {
14
m_w = seed ^ (seed << 16);
15
if (!m_w) m_w = 1337;
16
m_z = ~seed;
17
if (!m_z) m_z = 31337;
18
}
19
uint32_t R32() {
20
m_z = 36969 * (m_z & 65535) + (m_z >> 16);
21
m_w = 18000 * (m_w & 65535) + (m_w >> 16);
22
return (m_z << 16) + m_w;
23
}
24
float F() {
25
return (float)R32() / (float)(0xFFFFFFFF);
26
}
27
28
// public for easy save/load. Yes a bit ugly but better than moving DoState into native.
29
uint32_t m_w;
30
uint32_t m_z;
31
};
32
33
// Data must consist only of the index and the twister array. This matches the PSP
34
// MT context exactly.
35
class MersenneTwister {
36
public:
37
MersenneTwister(uint32_t seed) : index_(0) {
38
mt_[0] = seed;
39
for (uint32_t i = 1; i < MT_SIZE; i++)
40
mt_[i] = (1812433253UL * (mt_[i - 1] ^ (mt_[i - 1] >> 30)) + i);
41
}
42
43
uint32_t R32() {
44
if (index_ == 0)
45
gen();
46
uint32_t y = mt_[index_];
47
y ^= y >> 11;
48
y ^= (y << 7) & 2636928640UL;
49
y ^= (y << 15) & 4022730752UL;
50
y ^= y >> 18;
51
index_ = (index_ + 1) % MT_SIZE;
52
return y;
53
}
54
55
private:
56
enum {
57
MT_SIZE = 624,
58
};
59
60
u32_le index_;
61
u32_le mt_[MT_SIZE];
62
63
void gen() {
64
for(uint32_t i = 0; i < MT_SIZE; i++){
65
uint32_t y = (mt_[i] & 0x80000000) + (mt_[(i + 1) % MT_SIZE] & 0x80000000);
66
mt_[i] = mt_[(i + 397) % MT_SIZE] ^ (y >> 1);
67
if (y % 2) mt_[i] ^= 2567483615UL;
68
}
69
return;
70
}
71
};
72
73