1 /** Randomize existing instances and generate randomized instances of a given type.
2 
3     Copyright: Per Nordlöw 2018-.
4     License: $(WEB boost.org/LICENSE_1_0.txt, Boost License 1.0).
5     Authors: $(WEB Per Nordlöw)
6 
7     See_Also: http://forum.dlang.org/thread/byonwfghdqgcirdjyboh@forum.dlang.org
8 
9     TODO: Can these be tagged with @nogc? Currently std.random.uniform may allocate.
10 
11     TODO: Tags as nothrow when std.random gets there.
12 
13     TODO: How to handle possibly null reference (class, dynamic types) types?
14     Answer relates to how to randomize empty/null variable length structures
15     (arrays, strings, etc).
16 
17     - Maybe some kind of length randomization?
18  */
19 module nxt.random_ex;
20 
21 import std.traits: isIntegral, isFloatingPoint, isNumeric, isIterable, isStaticArray, isArray, hasIndirections, isSomeString, isScalarType, isBoolean;
22 import std.range: ElementType, hasAssignableElements;
23 import std.random: uniform;
24 
25 version(unittest) private enum testLength = 64;
26 
27 /** Randomize value $(D x). */
28 ref E randInPlace(E)(return ref E x)
29 if (isBoolean!E)
30 {
31     return x = cast(bool)uniform(0, 2);
32 }
33 
34 /** Randomize value $(D x), optionally in range [$(D low), $(D high)]. */
35 ref E randInPlace(E)(return ref E x)
36 if (isIntegral!E)
37 {
38     return x = uniform(E.min, E.max);    // BUG: Never assigns the value E.max
39 }
40 
41 /** Randomize value $(D x), optional in range [$(D low), $(D high)]. */
42 ref E randInPlace(E)(return ref E x)
43 if (isFloatingPoint!E)
44 {
45     return x = uniform(cast(E)0, cast(E)1);
46 }
47 
48 /** Randomize value $(D x), optionally in range [$(D low), $(D high)]. */
49 ref E randInPlaceWithRange(E)(return ref E x,
50                               E low,
51                               E high)
52 if (isIntegral!E)
53 {
54     return x = uniform(low, high);    // BUG: Never assigns the value E.max
55 }
56 
57 /** Randomize value of $(D x), optional in range [$(D low), $(D high)]. */
58 ref E randInPlaceWithRange(E)(return ref E x,
59                               E low /* E.min_normal */,
60                               E high /* E.max */)
61 if (isFloatingPoint!E)
62 {
63     return x = uniform(low, high);
64 }
65 
66 version(unittest)
67 {
68     import nxt.rational: Rational, rational;
69 }
70 
71 /** Randomize value of $(D x). */
72 ref Rational!E randInPlace(Rational, E)(return ref Rational!E x) @trusted
73 if (isIntegral!E)
74 {
75     return x = rational(uniform(E.min, E.max),
76                         uniform(1, E.max));
77 }
78 
79 @safe unittest
80 {
81     Rational!int x;
82     x.randInPlace();
83 }
84 
85 /** Generate random value of $(D x).
86     See_Also: http://forum.dlang.org/thread/emlgflxpgecxsqweauhc@forum.dlang.org
87  */
88 ref dchar randInPlace(return ref dchar x) @trusted
89 {
90     auto ui = uniform(0,
91                       0xD800 +
92                       (0x110000 - 0xE000) - 2 // minus two for U+FFFE and U+FFFF
93         );
94     if (ui < 0xD800)
95     {
96         return x = ui;
97     }
98     else
99     {
100         ui -= 0xD800;
101         ui += 0xE000;
102 
103         // skip undefined
104         if (ui < 0xFFFE)
105             return x = ui;
106         else
107             ui += 2;
108 
109         assert(ui < 0x110000);
110         return x = ui;
111     }
112 }
113 
114 unittest
115 {
116     auto x = randomized!dchar;
117     dstring d = "alphaalphaalphaalphaalphaalphaalphaalphaalphaalpha";
118     auto r = d.randomize; // TODO: Use Phobos function to check if string is legally coded.
119 }
120 
121 /** Randomize value of $(D x). */
122 dstring randInPlace(dstring x) @trusted
123 {
124     typeof(x) y;
125     foreach (ix; 0 .. x.length)
126         y ~= randomized!dchar; // TODO: How to do this in a better way?
127     x = y;
128     return x;
129 }
130 
131 /** Randomize value of $(D x).
132  */
133 R randInPlace(R)(R x)
134 if (isIterable!R &&
135     hasAssignableElements!R)
136 {
137     import core.lifetime : move;
138     foreach (ref e; x)
139     {
140         e.randInPlace();
141     }
142     return move(x);             // TODO: remove when compiler does this for us
143 }
144 
145 /** Randomize all elements of $(D x).
146     Each element is randomized within range `[elementLow, elementHigh]`.
147  */
148 R randInPlaceWithElementRange(R, E)(R x,
149                                     E elementLow,
150                                     E elementHigh)
151 if (isIterable!R &&
152     hasAssignableElements!R &&
153     is(ElementType!R == E))
154 {
155     import core.lifetime : move;
156     foreach (ref e; x)
157     {
158         e.randInPlaceWithRange(elementLow, elementHigh);
159     }
160     return move(x);
161 }
162 
163 @safe unittest
164 {
165     void testDynamic(T)()
166     {
167         auto x = new T[testLength];
168         auto y = x.dup;
169         x.randInPlace();
170         y.randInPlace();
171         assert(y != x);
172     }
173     testDynamic!int;
174     testDynamic!float;
175     testDynamic!bool;
176 }
177 
178 /** Randomize elements of $(D x).
179  */
180 ref T randInPlace(T)(return ref T x)
181 if (isStaticArray!T)
182 {
183     foreach (ref e; x)
184     {
185         e.randInPlace();
186     }
187     return x;
188 }
189 
190 @safe unittest
191 {
192     void testStatic(T)()
193     {
194         T[testLength] x;
195         auto y = x;
196         x.randInPlace();
197         y.randInPlace();
198         assert(y != x);
199     }
200     testStatic!bool;
201     testStatic!int;
202     testStatic!real;
203     enum E { a, b, c, d, e, f, g, h,
204              i, j, k, l, m, n, o, p }
205     testStatic!E;
206 }
207 
208 /** Blockwise-randomize elements of $(D x) of array type $(D A).
209     Randomizes in array blocks of type $(D B).
210  */
211 ref A randInPlaceBlockwise(B = size_t, A)(ref A x)
212 if (isArray!A &&
213     isIntegral!(ElementType!A))
214 {
215     alias E = ElementType!A;
216     static assert(E.sizeof < B.sizeof);
217     enum mult = B.sizeof / E.sizeof; // block multiplicity
218 
219     immutable n = x.length;
220 
221     // beginning unaligned bytes
222     auto p = cast(size_t)x.ptr;
223     immutable size_t mask = B.sizeof - 1;
224     immutable r = (p & mask) / E.sizeof; // element-offset from B-aligned address before x
225     size_t k = 0; // E-index to first B-block
226     if (r)
227     {
228         import std.algorithm.comparison : min;
229         k = min(n, mult - r); // at first aligned B-block
230         foreach (i, ref e; x[0 .. k])
231         {
232             e.randInPlace();
233         }
234     }
235 
236     // mid blocks of type B
237     auto bp = cast(B*)(x.ptr + k); // block pointer
238     immutable nB = (n - k) / mult; // number of B-blocks
239     foreach (ref b; 0 .. nB) // for each block index
240     {
241         bp[b].randInPlace();
242     }
243 
244     // ending unaligned bytes
245     foreach (i, ref e; x[k + nB*mult .. $])
246     {
247         e.randInPlace();
248     }
249 
250     return x;
251 }
252 
253 unittest
254 {
255     static void test(B = size_t, T)()
256     {
257         enum n = 1024;
258 
259         // dynamic array
260         for (size_t i = 0; i < n; i++)
261         {
262             T[] da = new T[i];
263             da.randInPlaceBlockwise!B;
264             size_t j = randomInstanceOf!(typeof(i))(0, n/2);
265             da.randInPlaceBlockwise!B;
266         }
267 
268         // static array
269         T[n] sa;
270         auto sa2 = sa[1 .. $];
271         sa2.randInPlaceBlockwise!B;
272     }
273 
274     import std.meta : AliasSeq;
275     foreach (T; AliasSeq!(byte, short, int, ubyte, ushort, uint))
276     {
277         test!(size_t, T);
278     }
279 }
280 
281 /** Randomize members of $(D x).
282  */
283 auto ref randInPlace(T)(return ref T x)
284 if (is(T == struct))
285 {
286     foreach (ref e; x.tupleof)
287     {
288         e.randInPlace();
289     }
290     return x;
291 }
292 
293 @safe unittest
294 {
295     struct T { ubyte a, b, c, d; }
296     T[testLength] x;
297     auto y = x;
298     x.randInPlace();
299     y.randInPlace();
300     assert(y != x);
301 }
302 
303 /** Randomize members of $(D x).
304  */
305 auto ref randInPlace(T)(T x)
306 if (is(T == class))
307 {
308     foreach (ref e; x.tupleof)
309     {
310         e.randInPlace();
311     }
312     return x;
313 }
314 
315 alias randomize = randInPlace;
316 
317 unittest
318 {
319     void testClass(E)()
320     {
321         class T { E a, b; }
322         auto x = new T;
323         auto y = new T;
324         x.randInPlace();
325         y.randInPlace();
326         assert(y != x);
327     }
328     testClass!bool;
329     testClass!int;
330     testClass!float;
331 }
332 
333 /** Returns: randomized instance of type $(D T).
334  */
335 T randomInstanceOf(T)()
336 {
337     /* TODO: recursively only void-initialize parts of T that are POD, not
338      reference types */
339     static if (hasIndirections!T)
340         T x;
341     else
342         /* don't init - randInPlace below fills in everything safely */
343         T x = void;
344     x.randInPlace();
345     return x;
346 }
347 
348 /** Returns: randomized instance of type $(D T).
349  */
350 T randomInstanceOf(T)(T low = T.min,
351                       T high = T.max)
352 if (isNumeric!T)
353 {
354     /* TODO: recursively only void-initialize parts of T that are POD, not
355        reference types */
356     static if (hasIndirections!T)
357         T x;
358     else
359         /* don't init - `randInPlace()` below fills in everything safely */
360         T x = void;
361     return x.randInPlaceWithRange(low, high);
362 }
363 
364 alias randomized = randomInstanceOf;
365 
366 /** Random number generator xoroshiro128+
367 
368     See_Also: http://xoroshiro.di.unimi.it/
369     See_Also: http://forum.dlang.org/post/kdobdorqztlsomweftmi@forum.dlang.org
370     See_Also: https://www.reddit.com/r/programming/comments/4gtlfz/xoroshiro128_the_fastest_fullperiod_pseudorandom/
371  */
372 struct Xoroshiro128plus
373 {
374 @safe pure nothrow @nogc:
375     public:
376 
377     enum ulong min = ulong.min;
378     enum ulong max = ulong.max;
379 
380     enum bool isUniformRandom = true;
381 
382     /// Range primitives
383     enum bool empty = false;
384 
385     /// ditto
386     ulong front() @property
387     {
388         return s[0] + s[1];
389     }
390 
391     /// ditto
392     void popFront()
393     {
394         immutable ulong s1 = s[1] ^ s[0];
395         s[0] = rotateLeft(s[0], 55) ^ s1 ^ (s1 << 14);
396         s[1] = rotateLeft(s1, 36);
397     }
398 
399     void seed(ulong s0, ulong s1)
400     in
401     {
402         // seeds are not both 0
403         assert(!(!s0 && !s1));
404     }
405     do
406     {
407         s[0] = s0;
408         s[1] = s1;
409     }
410 
411     void seed(ulong[2] s01)
412     in
413     {
414         // seeds are not both 0
415         assert(!(!s01[0] && !s01[1]));
416     }
417     do
418     {
419         s[] = s01[];
420     }
421 
422 private:
423     ulong[2] s;
424 
425     static ulong rotateLeft(ulong x, int k)
426     in
427     {
428         assert(k <= 64);
429     }
430     do
431     {
432         return (x << k) | (x >> (64 - k));
433     }
434 }
435 
436 @safe pure nothrow unittest
437 {
438     Xoroshiro128plus gen;
439     gen.seed(150078950, 1313143614);
440     import std.random : uniform;
441     import std.range : generate, take;
442     auto x = generate!(() => uniform!int(gen)).take(103);
443 }