blob: 9eb014aad300816dc89f8b8ea1d517d29dd9f512 [file] [log] [blame]
/**
This module contains support for controlling dynamic arrays' appending
Copyright: Copyright Digital Mars 2000 - 2019.
License: Distributed under the
$(LINK2 http://www.boost.org/LICENSE_1_0.txt, Boost Software License 1.0).
(See accompanying file LICENSE)
Source: $(DRUNTIMESRC core/_internal/_array/_appending.d)
*/
module core.internal.array.appending;
private extern (C)
{
bool gc_expandArrayUsed(void[] slice, size_t newUsed, bool atomic) pure nothrow;
bool gc_shrinkArrayUsed(void[] slice, size_t existingUsed, bool atomic) pure nothrow;
}
private enum isCopyingNothrow(T) = __traits(compiles, (ref T rhs) nothrow { T lhs = rhs; });
/**
* Extend an array `px` by `n` elements.
* Caller must initialize those elements.
* Params:
* px = the array that will be extended, taken as a reference
* n = how many new elements to extend it with
* Returns:
* The new value of `px`
* Bugs:
* This function template was ported from a much older runtime hook that bypassed safety,
* purity, and throwabilty checks. To prevent breaking existing code, this function template
* is temporarily declared `@trusted` until the implementation can be brought up to modern D expectations.
*/
ref Tarr _d_arrayappendcTX(Tarr : T[], T)(return ref scope Tarr px, size_t n) @trusted
{
import core.internal.traits: Unqual;
alias Unqual_T = Unqual!T;
alias Unqual_Tarr = Unqual_T[];
enum isshared = is(T == shared);
auto unqual_px = cast(Unqual_Tarr) px;
// Ignoring additional attributes allows reusing the same generated code
px = cast(Tarr)_d_arrayappendcTX_(unqual_px, n, isshared);
return px;
}
private ref Tarr _d_arrayappendcTX_(Tarr : T[], T)(return ref scope Tarr px, size_t n, bool isshared) @trusted
{
version (DigitalMars) pragma(inline, false);
version (D_TypeInfo)
{
// Short circuit if no data is being appended.
if (n == 0)
return px;
import core.stdc.string : memcpy, memset;
import core.internal.lifetime : __doPostblit;
import core.internal.array.utils: __arrayAlloc, newCapacity, __typeAttrs;
import core.internal.gc.blockmeta : PAGESIZE;
import core.exception: onOutOfMemoryError;
import core.memory: GC;
alias BlkAttr = GC.BlkAttr;
enum sizeelem = T.sizeof;
auto length = px.length;
auto newlength = length + n;
auto newsize = newlength * sizeelem;
auto size = length * sizeelem;
if (!gc_expandArrayUsed(px, newsize, isshared))
{
// could not set the size, we must reallocate.
auto newcap = newCapacity(newlength, sizeelem);
auto attrs = __typeAttrs!T(cast(void*)px.ptr) | BlkAttr.APPENDABLE;
T* ptr = cast(T*)GC.malloc(newcap, attrs, typeid(T));
if (ptr is null)
{
onOutOfMemoryError();
assert(0);
}
if (newsize != newcap)
{
// For small blocks that are always fully scanned, if we allocated more
// capacity than was requested, we are responsible for zeroing that
// memory.
// TODO: should let the GC figure this out, as this property may
// not always hold.
if (!(attrs & BlkAttr.NO_SCAN) && newcap < PAGESIZE)
memset(ptr + newlength, 0, newcap - newsize);
gc_shrinkArrayUsed(ptr[0 .. newlength], newcap, isshared);
}
memcpy(ptr, px.ptr, size);
// do potsblit processing.
__doPostblit!T(ptr[0 .. length]);
px = ptr[0 .. newlength];
return px;
}
// we were able to expand in place, just update the length
px = px.ptr[0 .. newlength];
return px;
}
else
assert(0, "Cannot append to array if compiling without support for runtime type information!");
}
version (D_ProfileGC)
{
/**
* TraceGC wrapper around _d_arrayappendcTX.
*/
ref Tarr _d_arrayappendcTXTrace(Tarr : T[], T)(return ref scope Tarr px, size_t n,
string file = __FILE__, int line = __LINE__, string funcname = __FUNCTION__) @trusted
{
version (D_TypeInfo)
{
import core.internal.array.utils: TraceHook, gcStatsPure, accumulatePure;
mixin(TraceHook!("Tarr", "_d_arrayappendcTX"));
return _d_arrayappendcTX(px, n);
}
else
static assert(0, "Cannot append to array if compiling without support for runtime type information!");
}
}
/// Implementation of `_d_arrayappendT`
ref Tarr _d_arrayappendT(Tarr : T[], T)(return ref scope Tarr x, scope Tarr y) @trusted
{
version (DigitalMars) pragma(inline, false);
import core.stdc.string : memcpy;
import core.internal.traits : Unqual;
auto length = x.length;
_d_arrayappendcTX(x, y.length);
// Only call `copyEmplace` if `T` has a copy ctor and no postblit.
static if (__traits(hasCopyConstructor, T))
{
import core.lifetime : copyEmplace;
size_t i;
try
{
for (i = 0; i < y.length; ++i)
copyEmplace(y[i], x[length + i]);
}
catch (Exception o)
{
/* Destroy, in reverse order, what we've constructed so far
*/
while (i--)
{
auto elem = cast(Unqual!T*) &x[length + i];
destroy(*elem);
}
throw o;
}
}
else
{
if (y.length)
{
// blit all elements at once
memcpy(cast(void*)&x[length], cast(void*)&y[0], y.length * T.sizeof);
// call postblits if they exist
static if (__traits(hasPostblit, T))
{
import core.internal.lifetime : __doPostblit;
size_t i = 0;
try __doPostblit(x[length .. $], i);
catch (Exception o)
{
// Destroy, in reverse order, what we've constructed so far
while (i--)
{
auto elem = cast(Unqual!T*) &x[length + i];
destroy(*elem);
}
throw o;
}
}}
}
return x;
}
version (D_ProfileGC)
{
/**
* TraceGC wrapper around $(REF _d_arrayappendT, core,internal,array,appending).
*/
ref Tarr _d_arrayappendTTrace(Tarr : T[], T)(return ref scope Tarr x, scope Tarr y, string file = __FILE__, int line = __LINE__, string funcname = __FUNCTION__) @trusted
{
version (D_TypeInfo)
{
import core.internal.array.utils: TraceHook, gcStatsPure, accumulatePure;
mixin(TraceHook!("Tarr", "_d_arrayappendT"));
return _d_arrayappendT(x, y);
}
else
static assert(0, "Cannot append to array if compiling without support for runtime type information!");
}
}
@safe unittest
{
double[] arr1;
foreach (i; 0 .. 4)
_d_arrayappendT(arr1, [cast(double)i]);
assert(arr1 == [0.0, 1.0, 2.0, 3.0]);
}
@safe unittest
{
int blitted;
struct Item
{
this(this)
{
blitted++;
}
}
Item[] arr1 = [Item(), Item()];
Item[] arr2 = [Item(), Item()];
Item[] arr1_org = [Item(), Item()];
arr1_org ~= arr2;
_d_arrayappendT(arr1, arr2);
// postblit should have triggered on at least the items in arr2
assert(blitted >= arr2.length);
}
@safe nothrow unittest
{
int blitted;
struct Item
{
this(this) nothrow
{
blitted++;
}
}
Item[][] arr1 = [[Item()]];
Item[][] arr2 = [[Item()]];
_d_arrayappendT(arr1, arr2);
// no postblit should have happened because arr{1,2} contain dynamic arrays
assert(blitted == 0);
}
@safe nothrow unittest
{
int copied;
struct Item
{
this(const scope ref Item) nothrow
{
copied++;
}
}
Item[1][] arr1 = [[Item()]];
Item[1][] arr2 = [[Item()]];
_d_arrayappendT(arr1, arr2);
// copy constructor should have been invoked because arr{1,2} contain static arrays
assert(copied >= arr2.length);
}
@safe nothrow unittest
{
string str;
_d_arrayappendT(str, "a");
_d_arrayappendT(str, "b");
_d_arrayappendT(str, "c");
assert(str == "abc");
}
@safe nothrow unittest
{
static class FailedPostblitException : Exception { this() nothrow @safe { super(null); } }
static size_t inner_postblit_cnt = 0;
static size_t inner_dtor_cnt = 0;
static size_t outer_postblit_cnt = 0;
static size_t outer_dtor_cnt = 0;
static struct Inner
{
char id;
@safe:
this(this)
{
++inner_postblit_cnt;
if (id == '2')
throw new FailedPostblitException();
}
~this() nothrow
{
++inner_dtor_cnt;
}
}
static struct Outer
{
Inner inner1, inner2, inner3;
nothrow @safe:
this(char first, char second, char third)
{
inner1 = Inner(first);
inner2 = Inner(second);
inner3 = Inner(third);
}
this(this)
{
++outer_postblit_cnt;
}
~this()
{
++outer_dtor_cnt;
}
}
Outer[3] arr = [Outer('1', '1', '1'), Outer('1', '2', '3'), Outer('3', '3', '3')];
try {
Outer[] arrApp;
arrApp ~= arr;
}
catch (FailedPostblitException) {}
catch (Exception) assert(false);
assert(inner_postblit_cnt == 5);
assert(inner_dtor_cnt == 4);
assert(outer_postblit_cnt == 1);
assert(outer_dtor_cnt == 1);
}