From c282903d142b08e0d3fcc7c86e47516687e29c8a Mon Sep 17 00:00:00 2001 From: Andrei Alexandrescu Date: Fri, 2 Mar 2018 11:39:35 -0500 Subject: [PATCH 1/2] Eliminate obsolete workaround for bugzilla 14746 --- src/object.d | 8 ++------ 1 file changed, 2 insertions(+), 6 deletions(-) diff --git a/src/object.d b/src/object.d index c34d26e6fa..04d35d4363 100644 --- a/src/object.d +++ b/src/object.d @@ -2623,9 +2623,7 @@ unittest private void _destructRecurse(S)(ref S s) if (is(S == struct)) { - static if (__traits(hasMember, S, "__xdtor") && - // Bugzilla 14746: Check that it's the exact member of S. - __traits(isSame, S, __traits(parent, s.__xdtor))) + static if (__traits(hasMember, S, "__xdtor")) s.__xdtor(); } @@ -2644,9 +2642,7 @@ private void _destructRecurse(E, size_t n)(ref E[n] arr) void _postblitRecurse(S)(ref S s) if (is(S == struct)) { - static if (__traits(hasMember, S, "__xpostblit") && - // Bugzilla 14746: Check that it's the exact member of S. - __traits(isSame, S, __traits(parent, s.__xpostblit))) + static if (__traits(hasMember, S, "__xpostblit")) s.__xpostblit(); } From 3d6a92e1a29da2bc39b68543b10a716db2714833 Mon Sep 17 00:00:00 2001 From: Andrei Alexandrescu Date: Fri, 2 Mar 2018 13:07:43 -0500 Subject: [PATCH 2/2] Implement enhanced behavior of destroy --- src/object.d | 151 ++++++++++++++++++++++++++++++++++++++++----------- 1 file changed, 120 insertions(+), 31 deletions(-) diff --git a/src/object.d b/src/object.d index 04d35d4363..0620af4238 100644 --- a/src/object.d +++ b/src/object.d @@ -2918,21 +2918,53 @@ unittest assert(postblitRecurseOrder == order); } +// destroy /******** -Destroys the given object and sets it back to its initial state. It's used to -_destroy an object, calling its destructor or finalizer so it no longer -references any other objects. It does $(I not) initiate a GC cycle or free -any GC memory. +Destroys an object by means of calling its destructor, if present. For +types `T` that define a destructor, `destroy` then also fills the +object with `T.init` to ensure the object's destructor can be called +again (e.g. when the object goes out of scope). Unlike $(LREF +__delete), `destroy` does not release memory or other resources by +itself (though it is possible that destructors it calls do so). + +Note: A `struct` type may have a destructor even if it does not +directly define `~this()`, if at least one field has a destructor +(transitively). A `class` type may have a destructor even if it does +not directly define `~this()`, if at least one field has a destructor +(transitively), or if the destructor is inherited. + +Detailed behavior of `destroy(obj)` depends on the type `T` as +follows: + +$(UL $(LI If `T` is a primitive type, a `struct` types that does not +have a destructor, or a `class` type that does not have a destructor, +the call to has no effect.) + +$(LI For `class` and `interface` types, the call `destroy(obj)` has no +effect if `obj` is `null`.) + +$(LI For `struct` and `class` types that have a destructor (either +defined explicitly or synthesized by the compiler), the destructor is +called. Then the object is initialized using +`T.init`. Reinitialization occurs even if the type disables default +construction with `@disable this()`.) + +$(LI For `interface` types, the dynamic `class` object is fetched from +the interface object, and then `destroy` is applied to it.) + +$(LI For static and dynamic arrays, `destroy` is applied right-to-left +to all elements. This means multidimensional arrays are destroyed in +depth, transitively (behavior different from $(LREF __delete), which +is by necessity shallow). If one or more destructors throw an +exception, the calls are completed and then the first exception +encountered is rethrown.)) */ -void destroy(T)(T obj) if (is(T == class)) +void destroy(T)(T obj) if (is(T == class) || (is(T == interface))) { - rt_finalize(cast(void*)obj); -} - -/// ditto -void destroy(T)(T obj) if (is(T == interface)) -{ - destroy(cast(Object)obj); + static if (is(T == interface)) + destroy(cast(Object) obj); + else + rt_finalize(cast(void*) obj); } /// Reference type demonstration @@ -2964,7 +2996,7 @@ unittest i = 1; assert(i == 1); // `i` changed to `1` destroy(i); - assert(i == 0); // `i` is back to its initial state `0` + assert(i == 1); // `i` stays unchanged } unittest @@ -3022,18 +3054,35 @@ unittest } } +// Does calling destroy against an object of type T do anything? +private template hasElaborateDestroy(T) +{ + // Note: __xdtor includes fields, __dtor doesn't. + static if (__traits(hasMember, T, "__xdtor")) + enum bool hasElaborateDestroy = true; + else static if (is(T : U[], U)) + enum bool hasElaborateDestroy = hasElaborateDestroy!U; + else static if (is(T : U[n], U, size_t n)) + enum bool hasElaborateDestroy = hasElaborateDestroy!U && n > 0; + else + enum bool hasElaborateDestroy = false; +} + /// ditto void destroy(T)(ref T obj) if (is(T == struct)) { - _destructRecurse(obj); - () @trusted { - auto buf = (cast(ubyte*) &obj)[0 .. T.sizeof]; - auto init = cast(ubyte[])typeid(T).initializer(); - if (init.ptr is null) // null ptr means initialize to 0s - buf[] = 0; - else - buf[] = init[]; - } (); + static if (hasElaborateDestroy!T) + { + obj.__xdtor(); + () @trusted { + auto buf = (cast(ubyte*) &obj)[0 .. T.sizeof]; + auto init = cast(ubyte[])typeid(T).initializer(); + if (init.ptr is null) // null ptr means initialize to 0s + buf[] = 0; + else + buf[] = init[]; + } (); + } } nothrow @safe @nogc unittest @@ -3043,7 +3092,7 @@ nothrow @safe @nogc unittest A a; a.s = "asd"; destroy(a); - assert(a.s == "A"); + assert(a.s == "asd"); } { static int destroyed = 0; @@ -3076,10 +3125,51 @@ nothrow @safe @nogc unittest } /// ditto -void destroy(T : U[n], U, size_t n)(ref T obj) if (!is(T == struct)) +void destroy(T : U[n], U, size_t n)(ref T obj) +{ + destroy(obj[]); +} + +/// ditto +void destroy(T : U[], U)(T obj) { - foreach_reverse (ref e; obj[]) - destroy(e); + static if (hasElaborateDestroy!T) + { + enum isNothrow = is(typeof(() nothrow { obj[0].__xdtor; })); + static if (isNothrow) + { + foreach_reverse (ref e; obj[]) + destroy(e); + } + else + { + if (!obj.length) + return; + + Exception response; + // Double-loop so we don't inefficiently use try in a loop. + bigloop: + for (size_t i = obj.length; ; ) + { + try + { + for (;;) + { + if (i == 0) break bigloop; + destroy(obj[--i]); + } + } + catch (Exception e) + { + if (!response) response = e; + if (i == 0) break; + --i; // skip the failed object + } + } + // Only exit point from the function + if (response) throw response; + } + } } unittest @@ -3088,7 +3178,7 @@ unittest a[0] = 1; a[1] = 2; destroy(a); - assert(a == [ 0, 0 ]); + assert(a == [ 1, 2 ]); } unittest @@ -3141,9 +3231,8 @@ unittest /// ditto void destroy(T)(ref T obj) - if (!is(T == struct) && !is(T == interface) && !is(T == class) && !_isStaticArray!T) +if (!is(T == struct) && !is(T == interface) && !is(T == class) && !_isStaticArray!T) { - obj = T.init; } template _isStaticArray(T : U[N], U, size_t N) @@ -3161,12 +3250,12 @@ unittest { int a = 42; destroy(a); - assert(a == 0); + assert(a == 42); } { float a = 42; destroy(a); - assert(isnan(a)); + assert(a == 42); } }