1 /** Extensions to std.datetime_ex.benchmark.
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 	TODO: Use ggplot or similar to visualize results.
8 */
9 module nxt.benchmark_ex;
10 
11 /** Behavior or reservation of space for a specific implicit length/size/count.
12  */
13 enum Reserved
14 {
15 	no,							///< Reservation is not (yet) supported.
16 	yes,						///< Is reserved.
17 	always,						///< Always preserved because of static allocation.
18 }
19 
20 struct Results
21 {
22     import core.time : Duration;
23     @property void toString(Sink)(ref scope Sink sink) const
24 	{
25 		import std.format : formattedWrite;
26 		foreach (memberName; __traits(allMembers, typeof(this)))
27 		{
28 			const member = __traits(getMember, this, memberName);
29 			alias Member = typeof(member);
30 			static if (is(immutable Member == immutable Duration))
31 			{
32 				if (member == Member.init)
33 					continue;
34 				sink.formattedWrite("%s: %s ns/op",
35 									memberName,
36 									cast(double)(member).total!"nsecs" / elementCount);
37 			}
38 		}
39 	}
40 	string typeName;
41 	size_t elementCount;
42 	size_t runCount;
43 	private Duration _insertWithoutGrowthTime;
44 	private Duration _insertWithGrowthTime;
45 	private Duration _removeTime;
46 	private Duration _containsTime;
47 	private Duration _inTime;
48 	private Duration _rehashTime;
49 	private Duration _inAfterRehashTime;
50 	private Duration _indexTime;
51 	private Duration _appendTime;
52 }
53 
54 /// Number of run per benchmark.
55 debug
56     static immutable runCountDefault = 3; // lighter test in debug mode
57 else
58 	static immutable runCountDefault = 10;
59 
60 /// Formatting uses some extra space but should be removed when outputting to plots.
61 static immutable formatNsPerOp = "%6.1f ns/op";
62 
63 /** Benchmark append operation available in type `A` with test source `S`.
64  */
65 Results benchmarkAppendable(A, Sample, Source)(in Source testSource, in size_t runCount = runCountDefault)
66 {
67     import core.time : Duration;
68     import std.datetime : MonoTime;
69     import std.conv : to;
70     import std.algorithm.searching : minElement, maxElement;
71 	import std.stdio : writef, writefln;
72 
73 	auto results = typeof(return)(A.stringof, testSource.length, runCount);
74 
75 	writef("- ");
76 
77 	Reserved wasReserved;
78 	A a = makeWithRequestedCapacity!(A)(results.elementCount, wasReserved);
79 
80 	scope spans = new Duration[runCount];
81 
82 	foreach (const runIx; 0 .. runCount)
83 	{
84 		immutable start = MonoTime.currTime();
85 		foreach (immutable i; testSource)
86 		{
87 			static      if (is(typeof(a ~= i)))
88 				a ~= i;
89 			else static if (is(typeof(a ~= i.to!Sample)))
90 				a ~= i.to!Sample;
91 			else static if (is(typeof(a.put(i))))
92 				a.put(i);
93 			else static if (is(typeof(a.put(i.to!Sample))))
94 				a.put(i.to!Sample);
95 			else
96 				static assert(0, "Cannot append a `" ~ typeof(i).stringof ~ "` to a `" ~ A.stringof ~ "`");
97 		}
98 		spans[runIx] = MonoTime.currTime() - start;
99 	}
100 	results._appendTime = minElement(spans[]);
101 	writef("append (no-growth) (~=): "~formatNsPerOp,
102 		   cast(double)(results._appendTime).total!"nsecs" / results.elementCount);
103 
104 	writefln(` for %s`, A.stringof);
105 
106 	static if (__traits(hasMember, A, `clear`))
107 		a.clear();
108 
109 	return results;
110 }
111 
112 /** Benchmark set (container) type `A` with test source `S`.
113  */
114 Results benchmarkSet(A, Sample, Source)(in Source testSource, in size_t runCount = runCountDefault)
115 // TODO: if (isSet!A)
116 {
117     import core.time : Duration;
118     import std.datetime : MonoTime;
119     import std.conv : to;
120 	import std.stdio : writef, writefln, writeln;
121     import std.algorithm.searching : minElement, maxElement;
122     import nxt.address : AlignedAddress;
123     import nxt.sso_string : SSOString;
124 
125 	alias Address = AlignedAddress!1;
126 
127 	auto results = typeof(return)(A.stringof, testSource.length, runCount);
128 
129 	writef("- ");
130 
131 	A a;
132 
133 	{							// needs scope
134 		immutable start = MonoTime.currTime();
135 		foreach (immutable i; testSource)
136 		{
137 			static if (__traits(hasMember, A, `ElementType`) &&
138 					   is(A.ElementType == ubyte[]))
139 				a.insert(i.toUbytes);
140 			else
141 			{
142 				static if (__traits(hasMember, A, `ElementType`))
143 				{
144 					static if (is(A.ElementType == Address))
145 						const element = A.ElementType(i + 1); ///< Start at 1 instead of 0 because `Address` uses 0 for `nullValue`.
146 					else static if (is(A.ElementType == SSOString) ||
147 									is(A.ElementType == string))
148 						const element = A.ElementType(to!string(i));
149 					else
150 						const element = A.ElementType(i);
151 				}
152 				else
153 					const element = i;
154 				a.insert(element);
155 			}
156 		}
157 		results._insertWithGrowthTime = MonoTime.currTime() - start;
158 		writef("insert (with growth): "~formatNsPerOp,
159 			   cast(double)(results._insertWithGrowthTime).total!"nsecs" / results.elementCount);
160 	}
161 
162 	{							// needs scope
163 		immutable start = MonoTime.currTime();
164 		size_t hitCount = 0;
165 		foreach (immutable i; testSource)
166 		{
167 			static if (__traits(hasMember, A, `ElementType`) &&
168 					   is(A.ElementType == ubyte[]))
169 				hitCount += a.contains(i.toUbytes);
170 			else
171 			{
172 				static if (__traits(hasMember, A, `ElementType`))
173 				{
174 					static if (is(A.ElementType == Address))
175 						const element = A.ElementType(i + 1); ///< Start at 1 instead of 0 because `Address` uses 0 for `nullValue`.
176 					else static if (is(A.ElementType == SSOString) ||
177 									is(A.ElementType == string))
178 						const element = A.ElementType(to!string(i));
179 					else
180 						const element = A.ElementType(i); // wrap in `i` in `Nullable`
181 				}
182 				else
183 					const element = i;
184 				static if (__traits(hasMember, A, "contains"))
185 					hitCount += a.contains(element);
186 				else static if (is(typeof(a.contains(element)) == bool))
187 					hitCount += a.contains(element);
188 				else static if (is(typeof(element in a) == bool))
189 					hitCount += element in a;
190 				else
191 					static assert(0,
192 								  "Cannot check that " ~
193 								  typeof(a).stringof ~
194 								  " contains " ~
195 								  typeof(element).stringof);
196 			}
197 		}
198 		const ok = hitCount == results.elementCount; // for side effect in output
199 		results._containsTime = MonoTime.currTime() - start;
200 		writef(", contains: "~formatNsPerOp~" ns/op (%s)",
201 			   cast(double)(results._containsTime).total!"nsecs" / results.elementCount,
202 			   ok ? "OK" : "ERR");
203 	}
204 
205 	scope spans = new Duration[runCount];
206 
207 	{
208 		Reserved wasReserved;
209 		A b = makeWithRequestedCapacity!(A)(results.elementCount, wasReserved); // TODO: don’t use preallocated capacity in first benchmark here:
210 		if (wasReserved)
211 		{
212 			foreach (const runIx; 0 .. runCount)
213 			{
214 				immutable start = MonoTime.currTime();
215 				foreach (immutable i; testSource)
216 				{
217 					static if (__traits(hasMember, A, `ElementType`) &&
218 							   is(A.ElementType == ubyte[]))
219 						b.insert(i.toUbytes);
220 					else
221 					{
222 						static if (__traits(hasMember, A, `ElementType`))
223 						{
224 							static if (is(A.ElementType == Address))
225 								const element = A.ElementType(i + 1); ///< Start at 1 instead of 0 because `Address` uses 0 for `nullValue`.
226 							else static if (is(A.ElementType == SSOString) ||
227 											is(A.ElementType == string))
228 								const element = A.ElementType(to!string(i));
229 							else
230 								const element = A.ElementType(i); // wrap in `i` in `Nullable`
231 						}
232 						else
233 							const element = i;
234 						b.insert(element);
235 					}
236 				}
237 				spans[runIx] = MonoTime.currTime() - start;
238 			}
239 			results._insertWithoutGrowthTime = minElement(spans[]);
240 			writef(", insert (no growth): "~formatNsPerOp,
241 				   cast(double)(results._insertWithoutGrowthTime).total!"nsecs" / results.elementCount);		}
242 
243 	}
244 
245 	writef(` for %s`, A.stringof);
246 
247 	static if (__traits(hasMember, A, `binCounts`))
248 		writef(" %s", a.binCounts());
249 	static if (__traits(hasMember, A, `smallBinCapacity`))
250 		writef(" smallBinCapacity:%s", A.smallBinCapacity);
251 	static if (__traits(hasMember, A, `averageProbeCount`))
252 		writef(" averageProbeCount:%s", a.averageProbeCount);
253 
254 	writeln();
255 
256 	static if (__traits(hasMember, A, `clear`) &&
257 			   is(typeof(a.clear())))
258 	{
259 		a.clear();          // TODO why does this fail for `RadixTreeMap`?
260 	}
261 
262 	return results;
263 }
264 
265 /** Benchmark map (container) type `A` with test source `S`.
266  */
267 Results benchmarkMap(A, Sample, Source)(in Source testSource, in size_t runCount = runCountDefault)
268 // TODO: if (isMap!A || __traits(isAssociativeArray, A))
269 {
270     import core.time : Duration;
271     import std.datetime : MonoTime;
272     import std.conv : to;
273 	import std.stdio : writef, writefln, writeln;
274     import std.algorithm.searching : minElement, maxElement;
275     import nxt.address : AlignedAddress;
276 
277 	alias Address = AlignedAddress!1;
278 
279 	auto results = typeof(return)(A.stringof, testSource.length, runCount);
280 
281 	Reserved wasReserved;
282 	A a = makeWithRequestedCapacity!(A)(results.elementCount, wasReserved);
283 
284 	writef("- ");
285 
286 	// determine key and value type. TODO: extract to trait `KeyValueType(A)`
287 	static if (is(A : V[K], K, V)) {
288 		alias KeyType = K;
289 		alias ValueType = K;
290     } else {
291 		// determine key type. TODO: extract to trait `KeyType(A)`
292 		static if (is(A.KeyType)) {
293 			alias KeyType = A.KeyType;
294 		} else static if (is(typeof(A.KeyValue.key))) { // StringMap
295 			alias KeyType = typeof(A.KeyValue.key);
296 		} else static if (is(immutable typeof(A.keys()) == immutable K[], K)) { // emsi HashMap
297 			alias KeyType = K;
298 		} else static if (__traits(hasMember, A, "byKey")) { // emsi HashMap
299 			import std.range.primitives : std_ElementType = ElementType;
300 			alias KeyType = std_ElementType!(typeof(A.init.byKey()));
301 		} else {
302 			static assert(0, "Could not determine key type of " ~ A.stringof ~ " " ~ typeof(A.keys()).stringof);
303 		}
304 		// determine value type. TODO: extract to trait `ValueType(A)`
305 		static if (is(A.ValueType)) {
306 			alias ValueType = A.ValueType;
307 		} else static if (is(typeof(A.KeyValue.value))) { // StringMap
308 			alias ValueType = typeof(A.KeyValue.value);
309 		} else static if (__traits(hasMember, A, "byValue")) { // emsi HashMap
310 			import std.range.primitives : std_ElementType = ElementType;
311 			alias ValueType = std_ElementType!(typeof(A.init.byValue()));
312 		} else {
313 			static assert(0, "Could not determine value type of " ~ A.stringof);
314 		}
315 	}
316 
317 	// determine element type. TODO: extract to trait `ElementType(A)` or reuse `std.primitives.ElementType`
318 	static if (is(A.ElementType)) {
319 		alias ElementType = A.ElementType;
320 	} else static if (is(A.KeyValue)) { // StringMap
321 		alias ElementType = A.KeyValue;
322 	} else static if (__traits(hasMember, A, "byKeyValue") || // emsi HashMap
323 					  !is(typeof(A.init.byKeyValue) == void)) {
324 		import std.range.primitives : std_ElementType = ElementType;
325 		alias ElementType = std_ElementType!(typeof(A.init.byKeyValue()));
326 	} else {
327 		static assert(0, "Could not determine element type of " ~ A.stringof);
328 	}
329 
330 	// allocate
331 	const keys = iotaArrayOf!(KeyType)(0, results.elementCount);
332 
333 	scope spans = new Duration[runCount];
334 
335 	// insert/opIndex without preallocation via void capacity(size_t)
336 	{
337 		foreach (const runIx; 0 .. runCount)
338 		{
339 			immutable start = MonoTime.currTime();
340 			foreach (immutable i; testSource)
341 			{
342 				static if (is(typeof(A.init.insert(ElementType.init))))
343 				{
344 					static if (is(KeyType == Address))
345 						const element = ElementType(Address(keys[i] + 1), // avoid `Address.null Value`
346 													ValueType.init);
347 					else
348 						const element = ElementType(keys[i], ValueType.init);
349 					a.insert(element);
350 				}
351 				else
352 					a[keys[i]] = ValueType.init; // AAs
353 			}
354 			spans[runIx] = MonoTime.currTime() - start;
355 		}
356 		results._insertWithGrowthTime = minElement(spans[]);
357 		writef("insert (with growth): "~formatNsPerOp,
358 			   cast(double)(results._insertWithGrowthTime).total!"nsecs" / results.elementCount);
359 	}
360 
361 	// contains
362 	static if (is(typeof(A.init.contains(KeyType.init)) : bool))
363 	{
364 		{
365 			bool okAll = true;
366 			foreach (const runIx; 0 .. runCount)
367 			{
368 				immutable start = MonoTime.currTime();
369 				size_t hitCount = 0;
370 				foreach (immutable i; testSource)
371 				{
372 					static if (is(KeyType == Address))
373 						hitCount += a.contains(Address(keys[i] + 1)); // avoid `Address.nullValue`
374 					else
375 						hitCount += a.contains(keys[i]) ? 1 : 0;
376 				}
377 				const ok = hitCount == results.elementCount; // for side effect in output
378 				if (!ok)
379 					okAll = false;
380 				spans[runIx] = MonoTime.currTime() - start;
381 			}
382 			results._containsTime = minElement(spans[]);
383 			writef(", contains: "~formatNsPerOp~" ns/op (%s)",
384 				   cast(double)(results._containsTime).total!"nsecs" / results.elementCount,
385 				   okAll ? "OK" : "ERR");
386 		}
387 	}
388 
389 	// in
390 	{
391 		bool okAll = true;
392 		foreach (const runIx; 0 .. runCount)
393 		{
394 			immutable start = MonoTime.currTime();
395 			size_t hitCount = 0;
396 			foreach (immutable i; testSource)
397 			{
398 				static if (is(KeyType == Address))
399 					hitCount += cast(bool)(Address(keys[i] + 1) in a); // avoid `Address.nullValue`
400 				else
401 					hitCount += cast(bool)(keys[i] in a);
402 			}
403 			const ok = hitCount == results.elementCount; // for side effect in output
404 			if (!ok)
405 				okAll = false;
406 			spans[runIx] = MonoTime.currTime() - start;
407 		}
408 		results._inTime = minElement(spans[]);
409 		writef(",       in: "~formatNsPerOp~" ns/op (%s)",
410 			   cast(double)(results._inTime).total!"nsecs" / results.elementCount,
411 			   okAll ? "OK" : "ERR");
412 	}
413 
414 	// rehash (AAs) + in
415 	static if (is(typeof(a.rehash())))
416 	{
417 		// rehash
418 		foreach (const runIx; 0 .. runCount)
419 		{
420 			immutable start = MonoTime.currTime();
421 			a.rehash();
422 			spans[runIx] = MonoTime.currTime() - start;
423 		}
424 		results._rehashTime = minElement(spans[]);
425 		writef(", rehash: "~formatNsPerOp,
426 			   cast(double)(results._rehashTime).total!"nsecs" / results.elementCount);
427 		// in
428 		foreach (const runIx; 0 .. runCount)
429 		{
430 			immutable start = MonoTime.currTime();
431 			foreach (immutable i; testSource)
432 				const hit = keys[i] in a;
433 			spans[runIx] = MonoTime.currTime() - start;
434 		}
435 		results._inAfterRehashTime = minElement(spans[]);
436 		writef(", in (after rehash): "~formatNsPerOp,
437 			   cast(double)(results._inAfterRehashTime).total!"nsecs" / results.elementCount);
438 	}
439 
440 	// insert/opIndex with preallocation via void capacity(size_t)
441 	static if (__traits(hasMember, A, "withCapacity") ||
442 			   is(typeof(A.init.reserve(size_t.init))))
443 	{
444 		{
445 			foreach (const runIx; 0 .. runCount)
446 			{
447 				static if (__traits(hasMember, A, "withCapacity"))
448 					A b = A.withCapacity(results.elementCount);
449 				else
450 				{
451 					A b;
452 					static if (is(typeof(A.init.reserve(size_t.init))))
453 						b.reserve(results.elementCount);
454 				}
455 				immutable start = MonoTime.currTime();
456 				foreach (immutable i; testSource)
457 				{
458 					static if (__traits(hasMember, A, "insert"))
459 					{
460 						static if (is(A.KeyType == Address))
461 							b.insert(ElementType(Address(keys[i] + 1), ValueType.init)); // avoid `Address.nullValue`
462 						else
463 							b.insert(ElementType(keys[i], ValueType.init));
464 					}
465 					else
466 						b[keys[i]] = ValueType.init;
467 				}
468 				spans[runIx] = MonoTime.currTime() - start;
469 			}
470 			results._insertWithoutGrowthTime = minElement(spans[]);
471 			writef(", insert (no growth): "~formatNsPerOp,
472 				   cast(double)(results._insertWithoutGrowthTime).total!"nsecs" / results.elementCount);
473 		}
474 	}
475 
476 	writef(` for %s`, A.stringof);
477 
478 	static if (__traits(hasMember, A, `binCounts`))
479 		writef(" %s", a.binCounts());
480 	static if (__traits(hasMember, A, `smallBinCapacity`))
481 		writef(" smallBinCapacity:%s", A.smallBinCapacity);
482 	static if (__traits(hasMember, A, `totalProbeCount`))
483 		writef(" averageProbeCount:%s", cast(double)a.totalProbeCount/a.length);
484 
485 	writeln();
486 
487 	static if (__traits(hasMember, A, `clear`) &&
488 			   is(typeof(a.clear())))
489 	{
490 		a.clear();          // TODO why does this fail for `RadixTreeMap`?
491 	}
492 
493 	return results;
494 }
495 
496 alias benchmarkAssociativeArray = benchmarkMap;
497 
498 /** Make `A` and try setting its capacity to `results.elementCount`.
499  */
500 auto makeWithRequestedCapacity(A)(size_t elementCount, out Reserved wasReserved)
501 {
502     static if (is(typeof(A.init.withCapacity(elementCount))))
503 	{
504 		wasReserved = Reserved.yes;
505         return A.withCapacity(elementCount);
506 	}
507 	else
508 	{
509 		static if (is(A == class))
510 			auto a = new A();
511 		else static if (is(A == struct))
512 			auto a = A();
513 		else
514 			A a;
515 
516 		static if (__traits(hasMember, A, `reserve`))
517 		{
518 			static if (__traits(compiles, { a.reserve(elementCount); }))
519 			{
520 				a.reserve(elementCount);
521 				wasReserved = Reserved.yes;
522 			}
523 			else static if (__traits(compiles, { a.reserve!uint(elementCount); }))
524 			{
525 				a.reserve!uint(elementCount);
526 				wasReserved = Reserved.yes;
527 			}
528 		}
529 		else static if (is(A == U[], U))
530 		{
531 			// this doesn’t work aswell:
532 			// import std.range.primitives : ElementType;
533 			// return new ElementType!A[elementCount];
534 			a.reserve(elementCount); // See_Also: https://dlang.org/library/object/reserve.html
535 			wasReserved = Reserved.yes;
536 			return a;
537 		}
538 		else static if (is(A : V[K], K, V))
539 		{
540 			static if (__traits(compiles, { a.reserve(elementCount); }))
541 			{
542 				a.reserve(elementCount); // builtin AA’s might get this later on
543 				wasReserved = Reserved.yes;
544 			}
545 		}
546 		else static if (is(typeof(a.reserve(elementCount))))
547 		{
548 			a.reserve(elementCount);
549 			wasReserved = Reserved.yes;
550 		}
551 
552 		static if (__traits(isPOD, A))
553 			return a;
554 		else
555 		{
556 			import std.algorithm.mutation : move;
557 			return move(a);
558 		}
559 	}
560 }
561 
562 T[] iotaArrayOf(T)(in size_t begin, in size_t end)
563 {
564     import std.typecons : Nullable;
565 
566     typeof(return) es = new T[end];
567     foreach (immutable i; begin .. end)
568     {
569         static if (is(typeof(T(i)))) // if possible
570             es[i] = T(i);       // try normal construction
571         else
572         {
573             import nxt.sso_string : SSOString; // TODO: make this import optional
574             import std.conv : to;
575             static if (is(T == SSOString))
576                 es[i] = T(i.to!string);
577             else static if (is(typeof(T(i))))
578                 es[i] = T(i);
579 	        else static if (is(typeof(i.to!T)))
580                 es[i] = i.to!T;
581 	        else static if (is(T == Nullable!(uint, uint.max))) // TODO: avoid this
582                 es[i] = T(cast(uint)i);
583     		else
584 				static assert(0, "Cannot convert `" ~ typeof(i).stringof ~ "` to `" ~ T.stringof ~ "`");
585         }
586     }
587     return es;
588 }