What should happen here?
bauss
jj_1337 at live.dk
Tue Sep 21 12:11:38 UTC 2021
On Tuesday, 21 September 2021 at 12:02:05 UTC, Steven
Schveighoffer wrote:
> On 9/21/21 6:58 AM, Johan wrote:
>> On Monday, 20 September 2021 at 18:49:12 UTC, Steven
>> Schveighoffer wrote:
>>>
>>> I feel like this might not necessarily be an issue, because
>>> technically, you aren't using `c` any more, so it can be
>>> deallocated immediately. But right in our documentation
>>> [here](https://dlang.org/spec/interfaceToC.html#storage_allocation) it lists ways to alleviate this:
>>>
>>> ```
>>> If pointers to D garbage collector allocated memory are
>>> passed to C functions, it's critical to ensure that the
>>> memory will not be collected by the garbage collector before
>>> the C function is done with it. This is accomplished by:
>>>
>>> * Making a copy of the data using core.stdc.stdlib.malloc()
>>> and passing the copy instead.
>>> * Leaving a pointer to it on the stack (as a parameter or
>>> automatic variable), as the garbage collector will scan the
>>> stack.
>>> * Leaving a pointer to it in the static data segment, as the
>>> garbage collector will scan the static data segment.
>>> * Registering the pointer with the garbage collector with the
>>> std.gc.addRoot() or std.gc.addRange() calls.
>>> ```
>>>
>>> This to me seems like "leaving a pointer to it on the stack".
>>> I'm not sure how else I would do that specifically? Plus,
>>> this option is the only "free" one -- the others all require
>>> much more complication. Adding a pointer to the stack is
>>> free. It's just, I don't know how to tell the compiler to do
>>> that besides declaring it.
>>
>> First: the use of "stack" here is wrong and confusing. It
>> should be "local storage" (notorious error throughout the
>> spec). Indeed, what is done in your example is putting the
>> pointer in local storage. The scope of that local storage is
>> until the end of function scope (in your example). I don't
>> think (in LDC) that we track the lifetime of variables in that
>> way, so what is done is that the optimizer just looks at last
>> point of use. This is similar to how Java behaves:
>> https://stackoverflow.com/questions/39285108/can-java-garbage-collect-variables-before-end-of-scope
>
> Yikes, that's quite aggressive. It says even a method can be in
> progress on the thing and it's collected early.
>
>> As per language spec, the D compilers are non-compliant on
>> this point. So a decision is needed to either change the
>> language spec, or to complain with the D compilers to fix it.
>
> I would say if you can somehow find a way to trigger the
> optimizer not to avoid that stack push, in all compilers, we
> should do that. IMO, the cost of a stack pointer is minimal
> compared to the surprising result that we currently see. But I
> don't know enough about compilers implementation to know
> whether this is a reasonable ask.
>
> Regardless of whether it's spec or implementation, something
> needs to change. This is why I asked the question without any
> context first, to have everyone think about what they *expect*
> to happen before finding out what actually happens. I'm
> surprised so many expected the current behavior, I did not.
>
> I just thought of a possible easy and effective way to ensure
> the thing isn't collected early:
>
> ```d
> struct Pin(T)
> {
> T t;
> @nogc nothrow pure @safe ~this() {}
> alias t this;
> }
>
> ...
> // usage
> auto c = Pin!C(new C); // now it needs to be held until the
> scope ends
> ```
>
> This seems to work on LDC with -O3 to prevent the early
> collection, so maybe it is sound? If this is a valid mechanism
> to ensure it's saved, maybe it can be added to Phobos and the
> spec updated to recommend that.
>
> -Steve
What about a scope variable that holds an instance of a class? It
could be made to mean the same thing as in (must not be collected
until the end of the scope)
More information about the Digitalmars-d
mailing list