[foreign] from memory segments to byte buffers

Maurizio Cimadamore maurizio.cimadamore at oracle.com
Mon Mar 22 12:40:18 UTC 2021


Hi,
after putting together the PR for the ResourceScope abstraction [1] it 
occurred to me that, perhaps, support for byte buffer views in the 
memory segment API is more complex than it needs to be.

Currently, when creating a byte buffer out of a memory segment, we need 
to save the segment scope in the buffer, and apply scope checks whenever 
the buffer is accessed. Among the things the scope check will enforce, 
the most important ones are:

* confinement check: if the scope associated with the segment from which 
the buffer has been derived is confined, make sure that the buffer is 
accessed from same thread
* liveness check: if the scope associated with the segment  from which 
the buffer has been derived is closed, access to the buffer is not allowed

Since in the previous API there was no way to disallow a call to 
MemorySegment::close, the only way we had to enforce correctness on the 
buffer was to apply same scope checks we applied on memory segments. 
When combined with the ability of also creating a segment from a byte 
buffer (the dual case), this makes for some convoluted code, as we have 
to ensure that doing segment -> buffer -> segment gives you back the 
same scope you started with.

Now we have, possibly, a new weapon: when creating a byte buffer view 
from a segment, we could acquire the segment. This would mean that the 
segment would remain non-closeable as long as the buffer view (or any 
slices derived from it) are reachable.

This would solve all issues related with liveness - but there would 
still be problems when it comes to confinement. Here we have some options:

* disallow creating a byte buffer view from a confined segment - after 
all, the byte buffer API is not confined (note that for segments created 
using the so called "default scope" this restriction would not apply, 
since default segments are shared)
* still allow creation of BB views from confined segments, but insert 
confinement checks where needed

What do people think about this simplification? How important it is to 
retain deterministic deallocation in the face of byte buffer views? One 
clear advantage of dropping that use case is that no further change 
would be required to the BB API in order e.g. to support async file 
system operation: since a byte buffer now would keep the originating 
scope open, there would be no fear of the memory backing the byte buffer 
disappearing while performing some async IO on it.

The drawback is that if anything inside an application requires some 
conversion from a segment into a BB (e.g. to interop with some legacy 
library), that conversion has a potential of affecting deterministic 
deallocation of the segment, from code which might be oblivious to the 
fact that somewhere some BB view has been created (and maybe the BB has 
already been operated on, so there's no "bug" in the application, but 
the GC/Cleaner might be slow at triggering the ReosurceScope release 
operation).

We can of course keep everything as is (as we have done most of the 
work) I just wanted to make sure we understood the complexity of what 
the impl does today, and make sure this complexity was necessary.

Maurizio



More information about the panama-dev mailing list