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 }