/** * Array container for internal usage. * * Copyright: Copyright Martin Nowak 2013. * License: $(WEB www.boost.org/LICENSE_1_0.txt, Boost License 1.0). * Authors: Martin Nowak */ module rt.util.container.array; static import common = rt.util.container.common; import core.exception : onOutOfMemoryErrorNoGC; struct Array(T) { nothrow: @disable this(this); ~this() { reset(); } void reset() { length = 0; } @property size_t length() const { return _length; } @property void length(size_t nlength) { import core.checkedint : mulu; bool overflow = false; size_t reqsize = mulu(T.sizeof, nlength, overflow); if (!overflow) { if (nlength < _length) foreach (ref val; _ptr[nlength .. _length]) common.destroy(val); _ptr = cast(T*)common.xrealloc(_ptr, reqsize); if (nlength > _length) foreach (ref val; _ptr[_length .. nlength]) common.initialize(val); _length = nlength; } else onOutOfMemoryErrorNoGC(); } @property bool empty() const { return !length; } @property ref inout(T) front() inout in { assert(!empty); } body { return _ptr[0]; } @property ref inout(T) back() inout in { assert(!empty); } body { return _ptr[_length - 1]; } ref inout(T) opIndex(size_t idx) inout in { assert(idx < length); } body { return _ptr[idx]; } inout(T)[] opSlice() inout { return _ptr[0 .. _length]; } inout(T)[] opSlice(size_t a, size_t b) inout in { assert(a < b && b <= length); } body { return _ptr[a .. b]; } alias length opDollar; void insertBack()(auto ref T val) { import core.checkedint : addu; bool overflow = false; size_t newlength = addu(length, 1, overflow); if (!overflow) { length = newlength; back = val; } else onOutOfMemoryErrorNoGC(); } void popBack() { length = length - 1; } void remove(size_t idx) in { assert(idx < length); } body { foreach (i; idx .. length - 1) _ptr[i] = _ptr[i+1]; popBack(); } void swap(ref Array other) { auto ptr = _ptr; _ptr = other._ptr; other._ptr = ptr; immutable len = _length; _length = other._length; other._length = len; } invariant { assert(!_ptr == !_length); } private: T* _ptr; size_t _length; } unittest { Array!size_t ary; assert(ary[] == []); ary.insertBack(5); assert(ary[] == [5]); assert(ary[$-1] == 5); ary.popBack(); assert(ary[] == []); ary.insertBack(0); ary.insertBack(1); assert(ary[] == [0, 1]); assert(ary[0 .. 1] == [0]); assert(ary[1 .. 2] == [1]); assert(ary[$ - 2 .. $] == [0, 1]); size_t idx; foreach (val; ary) assert(idx++ == val); foreach_reverse (val; ary) assert(--idx == val); foreach (i, val; ary) assert(i == val); foreach_reverse (i, val; ary) assert(i == val); ary.insertBack(2); ary.remove(1); assert(ary[] == [0, 2]); assert(!ary.empty); ary.reset(); assert(ary.empty); ary.insertBack(0); assert(!ary.empty); destroy(ary); assert(ary.empty); // not copyable static assert(!__traits(compiles, { Array!size_t ary2 = ary; })); Array!size_t ary2; static assert(!__traits(compiles, ary = ary2)); static void foo(Array!size_t copy) {} static assert(!__traits(compiles, foo(ary))); ary2.insertBack(0); assert(ary.empty); assert(ary2[] == [0]); ary.swap(ary2); assert(ary[] == [0]); assert(ary2.empty); } unittest { alias RC = common.RC!(); Array!RC ary; size_t cnt; assert(cnt == 0); ary.insertBack(RC(&cnt)); assert(cnt == 1); ary.insertBack(RC(&cnt)); assert(cnt == 2); ary.back = ary.front; assert(cnt == 2); ary.popBack(); assert(cnt == 1); ary.popBack(); assert(cnt == 0); } unittest { import core.exception; try { // Overflow ary.length. auto ary = Array!size_t(cast(size_t*)0xdeadbeef, -1); ary.insertBack(0); } catch (OutOfMemoryError) { } try { // Overflow requested memory size for common.xrealloc(). auto ary = Array!size_t(cast(size_t*)0xdeadbeef, -2); ary.insertBack(0); } catch (OutOfMemoryError) { } }