forked from andbof/yastg
-
Notifications
You must be signed in to change notification settings - Fork 0
/
mtrandom.c
100 lines (84 loc) · 1.78 KB
/
mtrandom.c
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
91
92
93
94
95
96
97
98
99
100
#include <stdio.h>
#include <time.h>
#include <assert.h>
#include <math.h>
#include <limits.h>
#include <stdint.h>
#include "mtrandom.h"
#include "mt19937ar-cok.h"
#define RANDOM_DEV "/dev/urandom"
void mtrandom_init()
{
FILE *f;
uint32_t seed;
unsigned int n;
if (!(f = fopen(RANDOM_DEV, "r"))) {
printf("No %s, initializing PRNG from system time\n", RANDOM_DEV);
seed = (unsigned long)time(0);
} else {
printf("%s detected, will use for PRNG initialization\n", RANDOM_DEV);
n = 0;
while (n < sizeof(seed))
n += fread(&seed + n, 1, sizeof(seed) - n, f);
fclose(f);
}
mt_init_genrand(seed);
}
int64_t mtrandom_int64(int64_t range)
{
int64_t r = 0;
if (range == 0)
return range;
/*
* The 31 bit shift guarantees the number is positive.
* We only return a negative number if the range is negative.
*/
r |= mt_genrand_int32();
r <<= 31;
r |= mt_genrand_int32();
if (range > 0)
return r % range;
else
return -(r % -range);
}
uint64_t mtrandom_uint64(uint64_t range)
{
uint64_t r = 0;
if (range == 0)
return range;
r |= mt_genrand_int32();
r <<= 32;
r |= mt_genrand_int32();
return r % range;
}
unsigned int mtrandom_uint(unsigned int range)
{
if (range == 0)
return range;
return mt_genrand_int32() % range;
}
int mtrandom_int(int range)
{
if (range == 0)
return range;
return mt_genrand_int32() % range;
}
unsigned long mtrandom_ulong(unsigned long range)
{
return mtrandom_uint64(range);
}
long mtrandom_long(long range)
{
return mtrandom_int64(range);
}
double mtrandom_double(double range)
{
/* This method will actually generate a numerical distribution error
of range * 2^(-32), but it is good enough for our purposes. */
double r = floor(mt_genrand_real2() * range);
return r;
}
int mtrandom_bool()
{
return mt_genrand_int32() & 0x1;
}