Graphviz 13.0.0~dev.20250402.0402
Loading...
Searching...
No Matches
random.c
Go to the documentation of this file.
1
3
4#include <assert.h>
5#include <limits.h>
6#include <stdint.h>
7#include <stdlib.h>
8#include <string.h>
9#include <util/alloc.h>
10#include <util/gv_math.h>
11#include <util/random.h>
12
13int *gv_permutation(int bound) {
14 if (bound <= 0) {
15 return NULL;
16 }
17
18 // initialize a sequence `{0, 1, …, bound - 1}`
19 int *const p = gv_calloc((size_t)bound, sizeof(int));
20 for (int i = 0; i < bound; i++) {
21 p[i] = i;
22 }
23
24 // perform a Fisher-Yates shuffle
25 for (int i = bound - 1; i > 0; --i) {
26 const int j = gv_random(i + 1);
27 SWAP(&p[i], &p[j]);
28 }
29
30 return p;
31}
32
34static int random_small(int bound) {
35 assert(bound > 0);
36 assert(bound <= RAND_MAX);
37
38 // The interval `[0, RAND_MAX]` is not necessarily neatly divided into
39 // `bound`-sized chunks. E.g. using a bound of 3 with a `RAND_MAX` of 7:
40 // ┌───┬───┬───┬───┬───┬───┬───┬───┐
41 // │ 0 │ 1 │ 2 │ 3 │ 4 │ 5 │ 6 │ 7 │
42 // └───┴───┴───┴───┴───┴───┴───┴───┘
43 // ◄─────────► ◄─────────► ◄─────►
44 // 3 values 3 values 2 values
45 // To guarantee a uniform distribution, derive the upper bound of the last
46 // complete chunk (5 in the example above), above which we discard and
47 // resample to avoid pulling from the partial trailing chunk.
48 const int discard_threshold =
49 RAND_MAX - (int)(((unsigned)RAND_MAX + 1) % (unsigned)bound);
50
51 int r;
52 do {
53 r = rand();
54 } while (r > discard_threshold);
55
56 return r % bound;
57}
58
60static int random_big(int bound) {
61 assert(bound > 0);
62
63 // see comment in `random_small`, but note that our maximum generated value
64 // here will be `INT_MAX` instead of `RAND_MAX`
65 const int discard_threshold =
66 INT_MAX - (int)(((unsigned)INT_MAX + 1) % (unsigned)bound);
67
68 int r;
69 do {
70 // generate a random `sizeof(int) * CHAR_BIT`-bit wide value
71 unsigned raw = 0;
72 for (size_t i = 0; i < sizeof(int); ++i) {
73 // `RAND_MAX ≥ 32767` is guaranteed, so `random_small(256)` is safe
74 const uint8_t byte = (uint8_t)random_small((int)UINT8_MAX + 1);
75 memcpy((char *)&raw + i, &byte, sizeof(byte));
76 }
77
78 // Shift out the sign bit to force a non-negative value. Assumes two’s
79 // complement representation.
80 const unsigned natural = raw << 1 >> 1;
81
82 r = (int)natural;
83
84 } while (r > discard_threshold);
85
86 return r % bound;
87}
88
89int gv_random(int bound) {
90 assert(bound > 0);
91
92 if (bound > RAND_MAX) {
93 return random_big(bound);
94 }
95 return random_small(bound);
96}
Memory allocation wrappers that exit on failure.
static void * gv_calloc(size_t nmemb, size_t size)
Definition alloc.h:26
#define UINT8_MAX
Definition gmlscan.c:337
node NULL
Definition grammar.y:163
Arithmetic helper functions.
#define SWAP(a, b)
Definition gv_math.h:130
static int random_small(int bound)
handle random number generation, bound ≤ RAND_MAX
Definition random.c:34
static int random_big(int bound)
handle random number generation, bound > RAND_MAX
Definition random.c:60
int gv_random(int bound)
Definition random.c:89
int * gv_permutation(int bound)
Definition random.c:13
random number generation