1 module nxt.cyclic_array; 2 3 import core.internal.traits : hasElaborateDestructor; 4 import std.algorithm : max; 5 import std.container.array : Array; // TODO: Use nxt.dynamic_array 6 import std.range; 7 import std.traits : isMutable; 8 import nxt.container_traits : mustAddGCRange; 9 10 struct CyclicArray(T, size_t len = max(8, 4096 / T.sizeof)) 11 { 12 static if (len == 0) 13 { 14 private void reserve(size_t length) @nogc @system nothrow 15 { 16 assert(length > 0); 17 array.length = length; 18 } 19 } 20 21 private: 22 static if (len == 0) 23 Array!T array; 24 else 25 T[len] array; 26 size_t start; 27 size_t size; 28 29 public: 30 static if (len == 0) 31 { 32 invariant 33 { 34 assert(array.length > 0); 35 } 36 37 // @disable this(); // TODO: this triggers bug in dmd: https://issues.dlang.org/show_bug.cgi?id=20934 38 39 this(Array!T array) @nogc 40 { 41 assert(array.length > 0); 42 this.array = array; 43 } 44 45 this(size_t length) @nogc 46 { 47 assert(length > 0); 48 array = Array!T(); 49 reserve(length); 50 } 51 52 this(size_t n)(T[n] val) 53 { 54 array = Array!T(); 55 reserve(max(8, 4096 / T.sizeof)); 56 static if (len != 0) 57 static assert(n <= len); 58 foreach (ref v; val) 59 { 60 put(v); 61 } 62 } 63 64 this(Range)(Range val) 65 if (__traits(compiles, ElementType!Range) && is(ElementType!Range : T)) 66 { 67 array = Array!T(); 68 reserve(max(8, 4096 / T.sizeof)); 69 foreach (ref v; val) 70 { 71 put(v); 72 } 73 } 74 75 this(Args...)(Args val) 76 { 77 array = Array!T(); 78 reserve(max(8, 4096 / T.sizeof)); 79 foreach (ref v; val) 80 { 81 put(v); 82 } 83 } 84 } 85 else 86 { 87 this(size_t n)(T[n] val) 88 { 89 static if (len != 0) 90 { 91 static assert(n <= len); 92 } 93 foreach (ref v; val) 94 { 95 put(v); 96 } 97 } 98 99 this(Range)(Range val) 100 if (__traits(compiles, ElementType!Range) && is(ElementType!Range : T)) 101 { 102 foreach (ref v; val) 103 { 104 put(v); 105 } 106 } 107 108 this(Args...)(Args val) 109 { 110 foreach (ref v; val) 111 { 112 put(v); 113 } 114 } 115 } 116 117 mixin CyclicRangePrimitives!(T, q{ 118 static if (len == 0) 119 auto copy = typeof(cast() this)(array.length); 120 else 121 typeof(cast() this) copy; 122 }); 123 124 static if (len != 0) 125 CyclicRange!T byRef() @property @nogc @safe 126 { 127 return CyclicRange!T(array[], start, size); 128 } 129 130 size_t length() const @property @nogc @safe 131 { 132 return size; 133 } 134 135 size_t length(size_t val) @property 136 { 137 if (size > array.length) 138 assert(false); 139 if (val == 0) 140 { 141 clear(); 142 return size; 143 } 144 if (val > size) 145 { 146 foreach (const i; size .. val) 147 array[(start + i) % array.length] = T.init; 148 } 149 else if (val < size) 150 { 151 static if (hasElaborateDestructor!T) 152 foreach (const i; val .. size) 153 destroy(array[(start + i) % array.length]); 154 else static if (mustAddGCRange!T) 155 foreach (const i; val .. size) 156 array[(start + i) % array.length] = T.init; // avoid GC mark-phase dereference 157 } 158 return size = val; 159 } 160 161 void clear() 162 { 163 static if (hasElaborateDestructor!T) 164 foreach (const i; 0 .. size) 165 destroy(array[(start + i) % array.length]); 166 start = 0; // optimize clears 167 size = 0; 168 } 169 170 auto opBinary(string op : "~")(T rhs) const 171 { 172 auto copy = this.dup; 173 copy.put(rhs); 174 return copy; 175 } 176 177 auto opBinary(string op : "~", size_t n)(T[n] rhs) const 178 { 179 auto copy = this.dup; 180 copy ~= rhs; 181 return copy; 182 } 183 184 auto opBinary(string op : "~", Range)(Range rhs) const 185 if (isInputRange!Range && is(ElementType!Range : T)) 186 { 187 auto copy = this.dup; 188 copy ~= rhs; 189 return copy; 190 } 191 192 void opOpAssign(string op : "~")(T rhs) 193 { 194 put(rhs); 195 } 196 197 void opOpAssign(string op : "~", size_t n)(T[n] rhs) 198 { 199 foreach (c; rhs) 200 { 201 put(c); 202 } 203 } 204 205 void opOpAssign(string op : "~", size_t n)(CyclicArray!(T, n) rhs) 206 { 207 for (int i = 0; i < rhs.size; i++) 208 put(rhs.array[(rhs.start + i) % rhs.array.length]); 209 } 210 211 void opOpAssign(string op : "~", Range)(Range rhs) 212 if (__traits(compiles, ElementType!Range) && is(ElementType!Range : T)) 213 { 214 foreach (c; rhs) 215 put(c); 216 } 217 218 static if (len == 0) 219 { 220 CyclicArray!(T, len) dup() const 221 { 222 auto ret = CyclicArray!(T, len)(array); 223 ret.start = start; 224 ret.size = size; 225 return ret; 226 } 227 } 228 else 229 { 230 CyclicArray!(T, len) dup() const 231 { 232 CyclicArray!(T, len) ret; 233 ret.array = cast(typeof(ret.array)) array; 234 ret.start = start; 235 ret.size = size; 236 return ret; 237 } 238 } 239 240 static if (len != 0) 241 { 242 int opApply(int delegate(ref T) @nogc dg) @nogc 243 { 244 int result = 0; 245 246 for (size_t i = 0; i < size; i++) 247 { 248 result = dg(array[(i + start) % array.length]); 249 if (result) 250 break; 251 } 252 253 return result; 254 } 255 256 int opApply(int delegate(size_t, ref T) @nogc dg) @nogc 257 { 258 int result = 0; 259 260 for (size_t i = 0; i < size; i++) 261 { 262 result = dg(i, array[(i + start) % array.length]); 263 if (result) 264 break; 265 } 266 267 return result; 268 } 269 } 270 271 bool opEquals(in CyclicArray!(T, len) b) 272 { 273 if (size != b.size) 274 return false; 275 for (int i = 0; i < size; i++) 276 if (array[(i + start) % array.length] != 277 b.array[(i + b.start) % b.array.length]) 278 return false; 279 return true; 280 } 281 282 bool opEquals(size_t n)(T[n] b) 283 { 284 if (size != n) 285 return false; 286 for (int i = 0; i < size; i++) 287 if (array[(i + start) % array.length] != b[i]) 288 return false; 289 return true; 290 } 291 292 bool opEquals(Range)(Range b) if (hasLength!Range && is(ElementType!Range : T)) 293 { 294 if (size != b.length) 295 return false; 296 auto r = b.save; 297 for (int i = 0; i < size; i++) 298 { 299 if (array[(i + start) % array.length] != r.front) 300 return false; 301 r.popFront(); 302 } 303 return true; 304 } 305 } 306 307 struct CyclicRange(T) 308 { 309 T[] array; 310 size_t start, size; 311 312 mixin CyclicRangePrimitives!T; 313 314 CyclicRange!T dup() const 315 { 316 return cast(CyclicRange!T) this; 317 } 318 } 319 320 private mixin template CyclicRangePrimitives(T, string makeCopy = "typeof(cast() this) copy;") 321 { 322 size_t capacity() const @property @nogc @safe 323 { 324 return array.length; 325 } 326 327 size_t length() const @property @nogc @safe 328 { 329 return size; 330 } 331 332 void put()(auto ref T val) 333 { 334 if (size >= array.length) 335 throw staticError!CyclicRangeError("array capacity overflow", __FILE__, __LINE__); 336 array[(start + size) % array.length] = val; 337 size++; 338 } 339 340 void put()(const auto ref T val) 341 { 342 if (size >= array.length) 343 throw staticError!CyclicRangeError("array capacity overflow", __FILE__, __LINE__); 344 array[(start + size) % array.length] = val; 345 size++; 346 } 347 348 void insertBack(T rhs) 349 { 350 put(rhs); 351 } 352 353 void insertBack(size_t n)(T[n] rhs) 354 { 355 foreach (c; rhs) 356 { 357 put(c); 358 } 359 } 360 361 void insertBack(Range)(Range rhs) 362 if (__traits(compiles, ElementType!Range) && 363 is(ElementType!Range : T)) 364 { 365 foreach (c; rhs) 366 { 367 put(c); 368 } 369 } 370 371 alias stableInsertBack = insertBack; 372 alias insert = insertBack; 373 alias stableInsert = insertBack; 374 alias linearInsert = insertBack; 375 alias stableLinearInsert = insertBack; 376 377 void insertFront()(auto ref T val) 378 { 379 if (size >= array.length) 380 throw staticError!CyclicRangeError("array capacity overflow", __FILE__, __LINE__); 381 start = (start + array.length - 1) % array.length; 382 array[start] = val; 383 size++; 384 } 385 386 void popFront() 387 { 388 if (size == 0) 389 throw staticError!CyclicRangeError("trying to pop an empty array", __FILE__, __LINE__); 390 static if (hasElaborateDestructor!T) 391 destroy(array[start]); 392 else static if (mustAddGCRange!T) 393 array[start] = T.init; // avoid GC mark-phase dereference 394 start = (start + 1) % array.length; 395 size--; 396 } 397 398 auto save() 399 { 400 return this; 401 } 402 403 bool empty() @nogc @safe @property const 404 { 405 return size == 0; 406 } 407 408 ref auto front() @nogc @safe @property inout 409 { 410 if (size == 0) 411 throw staticError!CyclicRangeError("trying to call front on empty array", 412 __FILE__, __LINE__); 413 return array[start]; 414 } 415 416 void popBack() 417 { 418 if (size == 0) 419 throw staticError!CyclicRangeError("trying to pop an empty array", __FILE__, __LINE__); 420 size--; 421 static if (hasElaborateDestructor!T) 422 destroy(array[(start + size) % array.length]); 423 else static if (mustAddGCRange!T) 424 array[(start + size) % array.length] = T.init; // avoid GC mark-phase dereference 425 } 426 427 ref auto back() @property 428 { 429 if (size == 0) 430 throw staticError!CyclicRangeError("trying to call back on empty array", 431 __FILE__, __LINE__); 432 return array[(start + size - 1) % array.length]; 433 } 434 435 auto back() @property const 436 { 437 if (size == 0) 438 throw staticError!CyclicRangeError("trying to call back on empty array", 439 __FILE__, __LINE__); 440 return array[(start + size - 1) % array.length]; 441 } 442 443 size_t opDollar() @nogc @safe const 444 { 445 return length; 446 } 447 448 ref inout(T) opIndex(size_t v) inout 449 { 450 if (v >= size) 451 throw staticError!CyclicRangeError("out of range", __FILE__, __LINE__); 452 else 453 return array[(v + start) % array.length]; 454 } 455 456 auto opIndex() const 457 { 458 return this.dup(); 459 } 460 461 private void validateRange(size_t[2] range) 462 { 463 immutable size_t from = range[0]; 464 immutable size_t to = range[1]; 465 if (to > size) 466 throw staticError!CyclicRangeError("trying to slice over array size", 467 __FILE__, __LINE__); 468 if (from > to) 469 throw staticError!CyclicRangeError("trying to negative slice", __FILE__, __LINE__); 470 if (from != to && from >= size || to - from > size) 471 throw staticError!CyclicRangeError("trying to slice over array bounds", 472 __FILE__, __LINE__); 473 } 474 475 auto opIndex(size_t[2] range) 476 { 477 immutable size_t from = range[0]; 478 immutable size_t to = range[1]; 479 validateRange(range); 480 481 mixin(makeCopy); 482 copy.start = 0; 483 copy.size = to - from; 484 485 foreach (const i; 0 .. copy.size) 486 { 487 copy.array[i] = array[(i + start + from) % array.length]; 488 } 489 return copy; 490 } 491 492 static if (isMutable!T) 493 { 494 void opIndexUnary(string op)() 495 { 496 foreach (const i; 0 .. size) 497 { 498 mixin(op ~ "array[(i + start) % array.length];"); 499 } 500 } 501 502 auto opIndexUnary(string op)(size_t i) 503 { 504 return mixin(op ~ "array[(i + start) % array.length]"); 505 } 506 507 void opIndexUnary(string op)(size_t[2] range) 508 { 509 immutable size_t from = range[0]; 510 immutable size_t to = range[1]; 511 validateRange(range); 512 513 foreach (const i; 0 .. to - from) 514 { 515 mixin(op ~ "array[(i + start + from) % array.length];"); 516 } 517 } 518 519 void opIndexAssign(U)(U val) 520 { 521 foreach (const i; 0 .. size) 522 { 523 array[(i + start) % array.length] = val; 524 } 525 } 526 527 void opIndexAssign(U)(U val, size_t i) 528 { 529 opIndex(i) = val; 530 } 531 532 void opIndexAssign(U)(U val, size_t[2] range) 533 { 534 immutable size_t from = range[0]; 535 immutable size_t to = range[1]; 536 validateRange(range); 537 538 foreach (const i; 0 .. to - from) 539 { 540 array[(i + start + from) % array.length] = val; 541 } 542 } 543 544 void opIndexOpAssign(string op, U)(U val) 545 { 546 foreach (const i; 0 .. size) 547 { 548 mixin("array[(i + start) % array.length] " ~ op ~ "= val;"); 549 } 550 } 551 552 void opIndexOpAssign(string op, U)(U val, size_t i) 553 { 554 mixin("array[(i + start) % array.length] " ~ op ~ "= val;"); 555 } 556 557 void opIndexOpAssign(string op, U)(U val, size_t[2] range) 558 { 559 immutable size_t from = range[0]; 560 immutable size_t to = range[1]; 561 validateRange(range); 562 563 foreach (const i; 0 .. to - from) 564 { 565 mixin("array[(i + start + from) % array.length] " ~ op ~ "= val;"); 566 } 567 } 568 } 569 570 static if (isMutable!T) 571 { 572 import std.algorithm.mutation : move; 573 574 T moveFront() 575 { 576 if (size == 0) 577 throw staticError!CyclicRangeError("trying to move in empty array", 578 __FILE__, __LINE__); 579 return move(array[0]); 580 } 581 582 T moveBack() 583 { 584 if (size == 0) 585 throw staticError!CyclicRangeError("trying to move in empty array", 586 __FILE__, __LINE__); 587 return move(array[(start + size - 1) % array.length]); 588 } 589 590 T moveAt(size_t i) 591 { 592 if (i >= size) 593 throw staticError!CyclicRangeError("trying to move outside range", 594 __FILE__, __LINE__); 595 return move(array[(start + i) % array.length]); 596 } 597 } 598 599 size_t[2] opSlice(size_t i : 0)() const 600 { 601 return [0, size]; 602 } 603 604 size_t[2] opSlice(size_t i : 0)(size_t from, size_t to) const 605 { 606 return [from, to]; 607 } 608 609 /** 610 * Removes the last element from the array and returns it. 611 * Both stable and non-stable versions behave the same and guarantee 612 * that ranges iterating over the array are never invalidated. 613 * 614 * Precondition: `empty == false` 615 * 616 * Returns: The element removed. 617 * 618 * Complexity: $(BIGOH 1). 619 * 620 * Throws: `Exception` if the array is empty. 621 */ 622 T removeAny() 623 { 624 auto result = back; 625 removeBack(); 626 return result; 627 } 628 629 alias stableRemoveAny = removeAny; 630 631 /** 632 * Removes the value from the back of the array. Both stable and non-stable 633 * versions behave the same and guarantee that ranges iterating over the 634 * array are never invalidated. 635 * 636 * Precondition: `empty == false` 637 * 638 * Complexity: $(BIGOH 1). 639 * 640 * Throws: `Exception` if the array is empty. 641 */ 642 void removeBack() 643 { 644 popBack(); 645 } 646 647 /// ditto 648 alias stableRemoveBack = removeBack; 649 650 void removeBack(int howMany) 651 { 652 foreach (const i; 0 .. howMany) 653 { 654 popBack(); 655 } 656 } 657 } 658 659 /// @nogc array without memory management using a cyclic array internally 660 /// Should be treated like a static array when no len is set (copies on assignment) 661 /// The maximum capacity is static and by default so many elements that it fills at most 4KiB, but at least 8 elements (even if that is more than 4KiB). 662 /// Set length to 0 to make it a std.container.Array based array 663 /// Bugs: foreach with ref value requires @nogc body to make this @nogc compatible 664 /// 665 @nogc @safe unittest 666 { 667 CyclicArray!int array; 668 assert(array.length == 0); 669 array ~= 5; 670 assert(!array.empty); 671 assert(array.front == 5); 672 assert(array[0] == 5); 673 array ~= [4, 3]; 674 675 assert(array == [5, 4, 3]); 676 677 // same efficiency as insertBack/put/concat 678 array.insertFront(5); 679 } 680 681 @nogc @system unittest 682 { 683 // heap array using std.container.Array with 16 elements 684 auto heapArray = CyclicArray!(int, 0)(16); 685 686 // custom memory using Array 687 auto myArray = Array!int(0, 1, 2, 3, 4, 5, 6, 7, 8, 9); 688 auto customArray = CyclicArray!(int, 0)(myArray[0 .. 6]); 689 } 690 691 class CyclicRangeError : Error 692 { 693 @nogc @safe pure nothrow this(string msg, string file = __FILE__, 694 size_t line = __LINE__, Throwable next = null) 695 { 696 super(msg, file, line, next); 697 } 698 } 699 700 // TLS storage shared for all errors, chaining might create circular reference 701 private void[128] _store; 702 703 // only Errors for now as those are rarely chained 704 private T staticError(T, Args...)(auto ref Args args) @trusted if (is(T : Error)) 705 { 706 // pure hack, what we actually need is @noreturn and allow to call that in pure functions 707 static T get() 708 { 709 static assert(__traits(classInstanceSize, T) <= _store.length, 710 T.stringof ~ " is too large for staticError()"); 711 712 _store[0 .. __traits(classInstanceSize, T)] = typeid(T).initializer[]; 713 return cast(T) _store.ptr; 714 } 715 716 auto res = (cast(T function() @trusted pure nothrow @nogc)&get)(); 717 res.__ctor(args); 718 return res; 719 } 720 721 // lots of unittests taken from std.container.array 722 723 // Give the Range object some testing. 724 @system unittest 725 { 726 import std.algorithm.comparison : equal; 727 import std.range : retro; 728 729 auto a = CyclicArray!int(0, 1, 2, 3, 4, 5, 6)[]; 730 assert(equal(a, [0, 1, 2, 3, 4, 5, 6])); 731 assert(equal(a[], [0, 1, 2, 3, 4, 5, 6])); 732 assert(equal(a[0 .. $], [0, 1, 2, 3, 4, 5, 6])); 733 auto b = CyclicArray!int(6, 5, 4, 3, 2, 1, 0)[]; 734 alias A = typeof(a); 735 alias ARef = typeof(a.byRef); 736 737 static assert(isRandomAccessRange!A); 738 static assert(hasSlicing!A); 739 static assert(hasAssignableElements!A); 740 static assert(hasMobileElements!A); 741 742 static assert(isRandomAccessRange!ARef); 743 static assert(hasSlicing!ARef); 744 static assert(hasAssignableElements!ARef); 745 static assert(hasMobileElements!ARef); 746 747 assert(equal(retro(b), a)); 748 assert(a.length == 7); 749 assert(equal(a[1 .. 4], [1, 2, 3])); 750 } 751 752 @system unittest 753 { 754 alias S = structBug5920; 755 uint dMask; 756 757 auto arr = CyclicArray!S(cast(S[])[]); 758 foreach (const i; 0 .. 8) 759 { 760 arr.insertBack(S(i, &dMask)); 761 } 762 // don't check dMask now as S may be copied multiple times (it's ok?) 763 { 764 assert(arr.length == 8); 765 dMask = 0; 766 arr.length = 6; 767 assert(arr.length == 6); // make sure shrinking calls the d'tor 768 assert(dMask == 0b1100_0000); 769 arr.removeBack(); 770 assert(arr.length == 5); // make sure removeBack() calls the d'tor 771 assert(dMask == 0b1110_0000); 772 arr.removeBack(3); 773 assert(arr.length == 2); // ditto 774 assert(dMask == 0b1111_1100); 775 arr.clear(); 776 assert(arr.length == 0); // make sure clear() calls the d'tor 777 assert(dMask == 0b1111_1111); 778 } 779 assert(dMask == 0b1111_1111); // make sure the d'tor is called once only. 780 } 781 782 @system unittest 783 { 784 auto a = CyclicArray!(int[])([[1, 2], [3, 4]]); 785 assert(a.length == 2); 786 assert(a[0] == [1, 2]); 787 assert(a[1] == [3, 4]); 788 } 789 790 @system unittest 791 { 792 import std.algorithm.comparison : equal; 793 794 //Test "array-wide" operations 795 auto a = CyclicArray!int([0, 1, 2]); //CyclicArray 796 a[] += 5; 797 assert(a[].equal([5, 6, 7])); 798 ++a[]; 799 assert(a[].equal([6, 7, 8])); 800 import std.stdio; 801 802 a[1 .. 3] *= 5; 803 assert(a[].equal([6, 35, 40])); 804 a[0 .. 2] = 0; 805 assert(a[].equal([0, 0, 40])); 806 807 //Test empty array 808 auto a2 = CyclicArray!int.init; 809 ++a2[]; 810 ++a2[0 .. 0]; 811 a2[] = 0; 812 a2[0 .. 0] = 0; 813 a2[] += 0; 814 a2[0 .. 0] += 0; 815 816 //Test "range-wide" operations 817 auto r = CyclicArray!int([0, 1, 2])[]; //CyclicArray.Range 818 r[] += 5; 819 assert(r.equal([5, 6, 7])); 820 ++r[]; 821 assert(r.equal([6, 7, 8])); 822 r[1 .. 3] *= 5; 823 assert(r.equal([6, 35, 40])); 824 r[0 .. 2] = 0; 825 assert(r.equal([0, 0, 40])); 826 827 //Test empty Range 828 auto r2 = CyclicArray!int.init[]; 829 ++r2[]; 830 ++r2[0 .. 0]; 831 r2[] = 0; 832 r2[0 .. 0] = 0; 833 r2[] += 0; 834 r2[0 .. 0] += 0; 835 } 836 837 // Test issue 838 @system unittest 839 { 840 static struct S 841 { 842 int i = 1337; 843 void* p; 844 this(this) 845 { 846 assert(i == 1337); 847 } 848 849 ~this() @nogc 850 { 851 assert(i == 1337); 852 } 853 } 854 855 CyclicArray!S arr; 856 S s; 857 arr ~= s; 858 arr ~= s; 859 } 860 861 @system unittest 862 { 863 import std.algorithm.iteration : filter; 864 865 auto a = CyclicArray!int([1, 2, 2].filter!"true"()); 866 } 867 868 @safe unittest 869 { 870 auto arr = new CyclicArray!int; 871 } 872 873 @system unittest //6998 874 { 875 static int i = 0; 876 class C 877 { 878 int dummy = 1; 879 this() 880 { 881 ++i; 882 } 883 884 ~this() @nogc 885 { 886 --i; 887 } 888 } 889 890 assert(i == 0); 891 auto c = new C(); 892 assert(i == 1); 893 894 //scope 895 { 896 auto arr = CyclicArray!C(c); 897 assert(i == 1); 898 } 899 //CyclicArray should not have destroyed the class instance 900 assert(i == 1); 901 902 //Just to make sure the GC doesn't collect before the above test. 903 assert(c.dummy == 1); 904 } 905 906 @system unittest //6998-2 907 { 908 static class C 909 { 910 int i; 911 } 912 913 auto c = new C; 914 c.i = 42; 915 CyclicArray!C a; 916 a ~= c; 917 a.clear; 918 assert(c.i == 42); //fails 919 } 920 921 @nogc: 922 unittest 923 { 924 alias IntArray = CyclicArray!int; 925 alias IntRange = CyclicRange!int; 926 927 static assert(isInputRange!IntArray); 928 static assert(isOutputRange!(IntArray, int)); 929 static assert(isForwardRange!IntArray); 930 static assert(isBidirectionalRange!IntArray); 931 static assert(isRandomAccessRange!IntArray); 932 static assert(hasMobileElements!IntArray); 933 static assert(is(ElementType!IntArray == int)); 934 static assert(hasSwappableElements!IntArray); 935 static assert(hasAssignableElements!IntArray); 936 static assert(hasLvalueElements!IntArray); 937 static assert(hasLength!IntArray); 938 static assert(hasSlicing!IntArray); 939 940 static assert(isInputRange!IntRange); 941 static assert(isOutputRange!(IntRange, int)); 942 static assert(isForwardRange!IntRange); 943 static assert(isBidirectionalRange!IntRange); 944 static assert(isRandomAccessRange!IntRange); 945 static assert(hasMobileElements!IntRange); 946 static assert(is(ElementType!IntRange == int)); 947 static assert(hasSwappableElements!IntRange); 948 static assert(hasAssignableElements!IntRange); 949 static assert(hasLvalueElements!IntRange); 950 static assert(hasLength!IntRange); 951 static assert(hasSlicing!IntRange); 952 953 IntArray array; 954 assert(array.length == 0); 955 assert(array.empty); 956 array ~= 5; 957 assert(!array.empty); 958 assert(array.length == 1); 959 assert(array.front == 5); 960 assert(array[0] == 5); 961 assert(array[0 .. 1].front == 5); 962 assert(array[0 .. 0].empty); 963 array ~= cast(int[2])[4, 3]; 964 assert(array.length == 3); 965 assert(array[1] == 4); 966 assert(array[2] == 3); 967 968 for (int i = 0; i < 50000; i++) 969 { 970 array ~= i; 971 array.popFront(); 972 } 973 974 assert(array.length == 3); 975 976 array ~= array; 977 assert(array.length == 6); 978 assert((array ~ array).length == 12); 979 assert(array.length == 6); 980 981 array[5] = 1; 982 assert(array[5] == 1); 983 auto copy = array; 984 copy[5] = 2; 985 assert(array[5] == 1); 986 array[][5] = 1; 987 assert(array[5] == 1); 988 989 foreach (ref v; array.byRef) 990 { 991 v = 42; 992 } 993 994 assert(array[5] == 42); 995 } 996 997 unittest 998 { 999 alias IntArray = CyclicArray!(int, 0); 1000 1001 static assert(isInputRange!IntArray); 1002 static assert(isOutputRange!(IntArray, int)); 1003 static assert(isForwardRange!IntArray); 1004 static assert(isBidirectionalRange!IntArray); 1005 static assert(isRandomAccessRange!IntArray); 1006 static assert(hasMobileElements!IntArray); 1007 static assert(is(ElementType!IntArray == int)); 1008 static assert(hasSwappableElements!IntArray); 1009 static assert(hasAssignableElements!IntArray); 1010 static assert(hasLvalueElements!IntArray); 1011 static assert(hasLength!IntArray); 1012 static assert(hasSlicing!IntArray); 1013 1014 auto array = IntArray(1024); 1015 assert(array.length == 0); 1016 assert(array.empty); 1017 array ~= 5; 1018 assert(!array.empty); 1019 assert(array.length == 1); 1020 assert(array.front == 5); 1021 assert(array[0] == 5); 1022 assert(array[0 .. 1].front == 5); 1023 assert(array[0 .. 0].empty); 1024 array ~= cast(int[2])[4, 3]; 1025 assert(array.length == 3); 1026 assert(array[1] == 4); 1027 assert(array[2] == 3); 1028 1029 for (int i = 0; i < 50000; i++) 1030 { 1031 array ~= i; 1032 array.popFront(); 1033 } 1034 1035 assert(array.length == 3); 1036 1037 array ~= array; 1038 assert(array.length == 6); 1039 assert((array ~ array).length == 12); 1040 assert(array.length == 6); 1041 1042 array[5] = 1; 1043 assert(array[5] == 1); 1044 auto copy = array[]; 1045 copy[5] = 2; 1046 assert(array[5] == 1); 1047 array[][5] = 1; 1048 assert(array[5] == 1); 1049 } 1050 1051 @system unittest 1052 { 1053 CyclicArray!int a; 1054 assert(a.empty); 1055 } 1056 1057 @system unittest 1058 { 1059 CyclicArray!int a; 1060 a.length = 10; 1061 assert(a.length == 10); 1062 assert(a.capacity >= a.length); 1063 } 1064 1065 @system unittest 1066 { 1067 struct Dumb 1068 { 1069 int x = 5; 1070 } 1071 1072 CyclicArray!Dumb a; 1073 a.length = 10; 1074 assert(a.length == 10); 1075 assert(a.capacity >= a.length); 1076 immutable cap = a.capacity; 1077 foreach (ref e; a) 1078 { 1079 e.x = 10; 1080 } 1081 a.length = 5; 1082 assert(a.length == 5); 1083 foreach (ref e; a) 1084 { 1085 assert(e.x == 10); 1086 } 1087 1088 a.length = 8; 1089 assert(a.length == 8); 1090 assert(Dumb.init.x == 5); 1091 foreach (const i; 0 .. 5) 1092 { 1093 assert(a[i].x == 10); 1094 } 1095 foreach (const i; 5 .. a.length) 1096 { 1097 assert(a[i].x == Dumb.init.x); 1098 } 1099 1100 a[] = Dumb(1); 1101 a.length = 20; 1102 foreach (const i; 0 .. 8) 1103 { 1104 assert(a[i].x == 1); 1105 } 1106 foreach (const i; 8 .. a.length) 1107 { 1108 assert(a[i].x == Dumb.init.x); 1109 } 1110 1111 // check if overlapping elements properly initialized 1112 a.length = 1; 1113 a.length = 20; 1114 assert(a[0].x == 1); 1115 foreach (e; a[1 .. $]) 1116 { 1117 assert(e.x == Dumb.init.x); 1118 } 1119 } 1120 1121 @system unittest 1122 { 1123 CyclicArray!int a = CyclicArray!int(1, 2, 3); 1124 //a._data._refCountedDebug = true; 1125 auto b = a.dup; 1126 assert(b == CyclicArray!int(1, 2, 3)); 1127 b.front = 42; 1128 assert(b == CyclicArray!int(42, 2, 3)); 1129 assert(a == CyclicArray!int(1, 2, 3)); 1130 } 1131 1132 @system unittest 1133 { 1134 auto a = CyclicArray!int(1, 2, 3); 1135 assert(a.length == 3); 1136 } 1137 1138 @system unittest 1139 { 1140 const CyclicArray!int a = [1, 2]; 1141 1142 assert(a[0] == 1); 1143 assert(a.front == 1); 1144 assert(a.back == 2); 1145 1146 static assert(!__traits(compiles, { a[0] = 1; })); 1147 static assert(!__traits(compiles, { a.front = 1; })); 1148 static assert(!__traits(compiles, { a.back = 1; })); 1149 1150 auto r = a[]; 1151 size_t i; 1152 foreach (e; r) 1153 { 1154 assert(e == i + 1); 1155 i++; 1156 } 1157 } 1158 1159 @system unittest 1160 { 1161 auto a = CyclicArray!int(1, 2, 3); 1162 a[1] *= 42; 1163 assert(a[1] == 84); 1164 } 1165 1166 @system unittest 1167 { 1168 auto a = CyclicArray!int(1, 2, 3); 1169 auto b = CyclicArray!int(11, 12, 13); 1170 auto c = a ~ b; 1171 assert(c == CyclicArray!int(1, 2, 3, 11, 12, 13)); 1172 assert(a ~ b[] == CyclicArray!int(1, 2, 3, 11, 12, 13)); 1173 assert(a ~ [4, 5] == CyclicArray!int(1, 2, 3, 4, 5)); 1174 } 1175 1176 @system unittest 1177 { 1178 auto a = CyclicArray!int(1, 2, 3); 1179 auto b = CyclicArray!int(11, 12, 13); 1180 a ~= b; 1181 assert(a == CyclicArray!int(1, 2, 3, 11, 12, 13)); 1182 } 1183 1184 @system unittest 1185 { 1186 auto a = CyclicArray!int(1, 2, 3, 4); 1187 assert(a.removeAny() == 4); 1188 assert(a == CyclicArray!int(1, 2, 3)); 1189 } 1190 1191 private struct structBug5920 1192 { 1193 int order; 1194 uint* pDestructionMask; 1195 ~this() @nogc 1196 { 1197 if (pDestructionMask) 1198 *pDestructionMask += 1 << order; 1199 } 1200 } 1201 1202 @system unittest 1203 { 1204 auto a = CyclicArray!int([1, 1]); 1205 a[1] = 0; //Check CyclicArray.opIndexAssign 1206 assert(a[1] == 0); 1207 a[1] += 1; //Check CyclicArray.opIndexOpAssign 1208 assert(a[1] == 1); 1209 1210 //Check CyclicArray.opIndexUnary 1211 ++a[0]; 1212 //a[0]++ //op++ doesn't return, so this shouldn't work, even with 5044 fixed 1213 assert(a[0] == 2); 1214 assert(+a[0] == +2); 1215 assert(-a[0] == -2); 1216 assert(~a[0] == ~2); 1217 1218 auto r = a[]; 1219 r[1] = 0; //Check CyclicArray.Range.opIndexAssign 1220 assert(r[1] == 0); 1221 r[1] += 1; //Check CyclicArray.Range.opIndexOpAssign 1222 assert(r[1] == 1); 1223 1224 //Check CyclicArray.Range.opIndexUnary 1225 ++r[0]; 1226 //r[0]++ //op++ doesn't return, so this shouldn't work, even with 5044 fixed 1227 assert(r[0] == 3); 1228 assert(+r[0] == +3); 1229 assert(-r[0] == -3); 1230 assert(~r[0] == ~3); 1231 } 1232 1233 @safe unittest 1234 { 1235 static struct S 1236 { 1237 bool b; 1238 alias b this; 1239 } 1240 1241 alias A = CyclicArray!S; 1242 alias B = CyclicArray!(shared bool); 1243 } 1244 1245 @system unittest 1246 { 1247 CyclicArray!int ai; 1248 ai ~= 1; 1249 assert(ai.front == 1); 1250 1251 static const arr = [1, 2, 3]; 1252 ai.insertBack(arr); 1253 }