aboutsummaryrefslogtreecommitdiff
path: root/src/stdlib/random.h
blob: 5bd4cc18c40a744d5f91a27ab39ed211442197e4 (plain)
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
// This file defines some wrapper logic around different systems' random functions

#include <assert.h>
#include <stdint.h>

#if defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__NetBSD__) || defined(__APPLE__)
#include <stdlib.h>
static ssize_t getrandom(void *buf, size_t buflen, unsigned int flags) {
    (void)flags;
    arc4random_buf(buf, buflen);
    return buflen;
}
#elif defined(__linux__)
// Use getrandom()
#include <sys/random.h>
#else
#error "Unsupported platform for secure random number generation"
#endif

static int64_t random_range(int64_t low, int64_t high) {
    uint64_t range = (uint64_t)high - (uint64_t)low + 1;
    uint64_t min_r = -range % range;
    uint64_t r;
    do {
        assert(getrandom(&r, sizeof(r), 0) == sizeof(r));
    } while (r < min_r);
    return (int64_t)((uint64_t)low + (r % range));
}