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 }