-
Notifications
You must be signed in to change notification settings - Fork 0
/
rng.hpp
85 lines (77 loc) · 1.4 KB
/
rng.hpp
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
#pragma once
#include "qlog.hpp"
#include <cstdint>
#define A_Default (18145460002477866997ull)
static uint64_t __lcg64_r = 0;
inline uint64_t lcg64(void)
{
uint64_t * const r = &__lcg64_r;
const uint64_t a = A_Default;
const uint64_t b = 1;
(*r) = ((*r)*a + b);
return (*r);
}
inline void lcg64(uint64_t seed)
{
__lcg64_r = seed;
}
inline double drand(void)
{
uint64_t M = ~0ull;
return((double)lcg64()/M);
}
template <typename T>
void shuffle(T* x, uint32_t len)
{
T t;
for(uint32_t i=0;i<len-1;i++)
{
uint32_t j = i+lcg64()%(len-1-i);
// swap x[i] & x[j]
t = x[i];
x[i] = x[j];
x[j] = t;
}
}
/**
* Input:
* Probability (without normalization) in p
* length of p and cum: len
* Output:
* return in [0,len-1]
* Cumulative Prob in (without normalization) in cum
*/
template <typename T>
inline uint32_t rmultinorm(T* p, T* cum, uint32_t len, bool cal_cum=true)
// Sequential search is faster for small size prob array
#define SEQ_SEARCH
{
if(cal_cum)
{
cum[0] = p[0];
for(uint32_t i=1;i<len;i++)
cum[i] = cum[i-1] + p[i];
}
double r = drand()*cum[len-1];
#ifndef SEQ_SEARCH
uint32_t med = len/2;
uint32_t low=0;
uint32_t high=len-1;
do {
if(cum[med]>r)
high = med;
else
low = med;
med = (high+low)/2;
} while(high>1+low);
if(r<cum[low])
return(low);
else
return(high);
#else
for(uint32_t i=0;i<len;i++)
if(cum[i]>=r)
return(i);
#endif
return(~0u);
}