[foreign-jextract] RFR: 8264187: Add a method for creating a slicing MethodHandle

Maurizio Cimadamore mcimadamore at openjdk.java.net
Fri Mar 26 14:32:38 UTC 2021


On Fri, 26 Mar 2021 12:20:02 GMT, Jorn Vernee <jvernee at openjdk.org> wrote:

> Hi,
> 
> This patch adds a `sliceHandle(PathElement...)` method to MemoryLayout, which can be used to create a method handle that, when given a segment, will return a slice of that segment for the layout element selected by the earlier given layout path.
> 
> This for instance makes it easier to iterate over an array of structs by first creating a slice for each element, and then accessing struct fields from there, without having to resolve to manual offset computation. For example:
> 
> SequenceLayout seq = MemoryLayout.ofSequence(5,
>     MemoryLayout.ofStruct(
>        MemoryLayout.ofValueBits(32, ByteOrder.nativeOrder()).withName("x"),
>        MemoryLayout.ofValueBits(32, ByteOrder.nativeOrder()).withName("y")
>     ));
> 
> MethodHandle sliceHandle = seq.sliceHandle(sequenceElement());
> 
> MemoryLayout structLayout = seq.select(sequenceElement());
> VarHandle xHandle = structLayout.varHandle(int.class, groupElement("x"));
> VarHandle yHandle = structLayout.varHandle(int.class, groupElement("y"));
> 
> try (ResourceScope scope = ResourceScope.ofConfined()) {
>     MemorySegment segment = MemorySegment.allocateNative(seq, scope);
>     long count = seq.elementCount().orElseThrow();
>     for (int i = 0; i < count; i++) {
>         // create view segment of single struct element
>         MemorySegment struct = (MemorySegment) sliceHandle.invokeExact(segment, (long) i);
>         xHandle.set(struct, 42);
>         yHandle.set(struct, 84);
>     }
> }
> 
> Thanks,
> Jorn

Looks good - I've left some comments

src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayout.java line 497:

> 495: 
> 496:     /**
> 497:      * Creates a method handle which can be used to create a slice of the layout selected by a given layout path,

I'd say "which, given a memory segment, returns a slice (hyperlink to MemorySegment::asSlice) corresponding to the layout selected by a given layout path"

src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayout.java line 516:

> 514:      * where {@code x_1}, {@code x_2}, ... {@code x_n} are <em>dynamic</em> values provided as {@code long}
> 515:      * arguments, whereas {@code c_1}, {@code c_2}, ... {@code c_m} and {@code s_0}, {@code s_1}, ... {@code s_n} are
> 516:      * <em>static</em> stride constants which are derived from the layout path.

There is an issue here, which is probably not introduced by you, but also present in other methods: c_1 and s_0 are not all strides; c_0 are offsets, while s_0 are strides.

src/jdk.incubator.foreign/share/classes/jdk/incubator/foreign/MemoryLayout.java line 523:

> 521:      * }</pre></blockquote>
> 522:      *
> 523:      * where {@code segment} is the segment to be sliced, and where {@code layout} is the layout selected by the given

Maybe we can say "as per MemoryLayout::select(path)" ?

src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LayoutPath.java line 76:

> 74:             MH_ADD_SCALED_OFFSET = lookup.findStatic(LayoutPath.class, "addScaledOffset",
> 75:                     MethodType.methodType(long.class, long.class, long.class, long.class));
> 76:             MH_SLICE = lookup.findStatic(LayoutPath.class, "slice",

Why not just pointing to MS::asSlice?

src/jdk.incubator.foreign/share/classes/jdk/internal/foreign/LayoutPath.java line 211:

> 209: 
> 210:     public MethodHandle sliceHandle() {
> 211:         MethodHandle offsetHandle = offsetHandle(); // bit offset

Nice composition of existing bits!

test/jdk/java/foreign/TestLayoutPaths.java line 533:

> 531:         MethodHandle sliceHandle;
> 532:         try {
> 533:             sliceHandle = layout.sliceHandle(groupElement("y")); // should work

Question: why is the failure dynamic? We know exactly, at construction time that the offset is gonna be odd?

-------------

PR: https://git.openjdk.java.net/panama-foreign/pull/477


More information about the panama-dev mailing list