ResourceScope attachment
Maurizio Cimadamore
maurizio.cimadamore at oracle.com
Mon Mar 29 10:12:22 UTC 2021
This is a somewhat obscure corner of the API. Of course this is another
thing that can be pruned, but let's talk about where it comes from first.
The main use case for this is native global variables.
A global variable is just an address. You can convert that address
(unsafely) into a segment.
```
LibraryLookup lookup = ....
MemoryAddress address = lookup.lookup("errno").get().address();
MemorySegment errno = address.asSegmentRestricted(8, ...);
...
//dereference errno?
```
Now, LibraryLookup are auto-cleaned using GC. Meaning that when the last
LibraryLookup pointing to a given library L goes unreachable, the
library is unloaded. We could add support for scopes here, but we didn't
do so (yet), so library unloading is implicit.
This means that a strong reference must be kept on that LibraryLookup:
now, the javadoc says that the MemoryAddress generated by the library
keep a strong reference to the lookup they come from.
But in the above we use the address to create a new segment, so that
invisible link is gone. Creating a segment with a lookup attachment
makes sure that the lookup is not unreachable prematurely, as the
segment points to it.
I can imagine other similar circumstances where stuff like this might be
handy.
As for native image, what you say is true, but if that was the case,
we'd start running afoul of what reachability means. For instance, the
ByteBuffer API implementation also uses attachment objects all over the
place, so I'd have to assume that there has to be a way to say "I want
to keep that field even if nobody uses it".
In other words, I think you touch on a valid question here, although I
don't agree on the "why this is wrong" explanation you give.
What I could understand one of either:
1) the problem you are seeing with LibraryLookup is caused by the fact
that, again, an implicit choice is made on scoping. Give users control
of that scope parameter, and now you can know _exactly_ when the lookup
is closed.
2) the problem is that the segment obtained by an address (obtained from
a LibraryLookup) doesn't point automatically back to the lookup (e.g.
users has to add reachability link on their own). This is error prone.
The reason I didn't go for (1), at least now, is that the library
loading mechanism is quite complex - in the sense that it is possible to
load the same library multiple times (e.g. using different
classloaders). This is a difference from JNI, for instance.
This means we can only close a library when _all_ the library lookups
associated with it have been closed (as a library can be used by
multiple clients at the same time). So it's not sufficient to use a
_single_ ResourceScope here, we need many.
Now, with ResourceScope we can probably set up a cleanup action which
checks some atomic counter (per library) and unloads the library when
the counter reaches zero. But it's not a slamdunk.
Maurizio
On 27/03/2021 12:16, Remi Forax wrote:
> I see ways to add an attachment when creating a scope
> (the ref Object in the MemoryScope implementation)
> but no way to get that object from the scope.
>
> I'm also wondering if theoretically a VM like native-image (the Substrate VM) can just not create the field "ref" since it's not used as an optimization ?
> You also have to be sure that nobody access this field by reflection but this is exactly the kind of information native-image is asking.
>
> Rémi
More information about the panama-dev
mailing list