/* * Copyright (c) 2021, 2022, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ /* * @test * @enablePreview * @run testng/othervm --enable-native-access=ALL-UNNAMED TestMemoryAccessInstance */ import java.lang.foreign.MemoryAddress; import java.lang.foreign.MemorySegment; import java.lang.foreign.MemorySession; import java.lang.foreign.ValueLayout; import java.nio.ByteBuffer; import java.nio.ByteOrder; import java.util.function.Function; import org.testng.annotations.*; import org.testng.SkipException; import static org.testng.Assert.*; public class TestMemoryAccessInstance { static class Accessor { interface SegmentGetter { X get(T buffer, L layout, long offset); } interface SegmentSetter { void set(T buffer, L layout, long offset, X o); } interface BufferGetter { X get(ByteBuffer segment, int offset); } interface BufferSetter { void set(ByteBuffer buffer, int offset, X o); } final X value; final L layout; final Function transform; final SegmentGetter segmentGetter; final SegmentSetter segmentSetter; final BufferGetter bufferGetter; final BufferSetter bufferSetter; Accessor(Function transform, L layout, X value, SegmentGetter segmentGetter, SegmentSetter segmentSetter, BufferGetter bufferGetter, BufferSetter bufferSetter) { this.transform = transform; this.layout = layout; this.value = value; this.segmentGetter = segmentGetter; this.segmentSetter = segmentSetter; this.bufferGetter = bufferGetter; this.bufferSetter = bufferSetter; } void test() { try (MemorySession session = MemorySession.openConfined()) { MemorySegment segment = MemorySegment.allocateNative(128, session); ByteBuffer buffer = segment.asByteBuffer(); T t = transform.apply(segment); segmentSetter.set(t, layout, 8, value); assertEquals(bufferGetter.get(buffer, 8), value); bufferSetter.set(buffer, 8, value); assertEquals(value, segmentGetter.get(t, layout, 8)); } } @SuppressWarnings("unchecked") void testHyperAligned() { try (MemorySession session = MemorySession.openConfined()) { MemorySegment segment = MemorySegment.allocateNative(64, session); T t = transform.apply(segment); L alignedLayout = (L)layout.withBitAlignment(layout.byteSize() * 8 * 2); try { segmentSetter.set(t, alignedLayout, 0, value); fail(); } catch (IllegalArgumentException exception) { assertTrue(exception.getMessage().contains("greater")); } try { segmentGetter.get(t, alignedLayout, 0); fail(); } catch (IllegalArgumentException exception) { assertTrue(exception.getMessage().contains("greater")); } } } static Accessor ofSegment(L layout, X value, SegmentGetter segmentGetter, SegmentSetter segmentSetter, BufferGetter bufferGetter, BufferSetter bufferSetter) { return new Accessor<>(Function.identity(), layout, value, segmentGetter, segmentSetter, bufferGetter, bufferSetter); } static Accessor ofAddress(L layout, X value, SegmentGetter segmentGetter, SegmentSetter segmentSetter, BufferGetter bufferGetter, BufferSetter bufferSetter) { return new Accessor<>(MemorySegment::address, layout, value, segmentGetter, segmentSetter, bufferGetter, bufferSetter); } } @Test(dataProvider = "segmentAccessors") public void testSegmentAccess(String testName, Accessor accessor) { accessor.test(); } @Test(dataProvider = "addressAccessors") public void testAddressAccess(String testName, Accessor accessor) { accessor.test(); } @Test(dataProvider = "segmentAccessors") public void testSegmentAccessHyper(String testName, Accessor accessor) { if (testName.contains("index")) { accessor.testHyperAligned(); } else { throw new SkipException("Skipping"); } } @Test(dataProvider = "addressAccessors") public void testAddressAccessHyper(String testName, Accessor accessor) { if (testName.contains("index")) { accessor.testHyperAligned(); } else { throw new SkipException("Skipping"); } } static final ByteOrder NE = ByteOrder.nativeOrder(); @DataProvider(name = "segmentAccessors") static Object[][] segmentAccessors() { return new Object[][]{ {"byte", Accessor.ofSegment(ValueLayout.JAVA_BYTE, (byte) 42, MemorySegment::get, MemorySegment::set, ByteBuffer::get, ByteBuffer::put) }, {"bool", Accessor.ofSegment(ValueLayout.JAVA_BOOLEAN, false, MemorySegment::get, MemorySegment::set, (bb, pos) -> bb.get(pos) != 0, (bb, pos, v) -> bb.put(pos, v ? (byte)1 : (byte)0)) }, {"char", Accessor.ofSegment(ValueLayout.JAVA_CHAR, (char) 42, MemorySegment::get, MemorySegment::set, (bb, pos) -> bb.order(NE).getChar(pos), (bb, pos, v) -> bb.order(NE).putChar(pos, v)) }, {"int", Accessor.ofSegment(ValueLayout.JAVA_INT, 42, MemorySegment::get, MemorySegment::set, (bb, pos) -> bb.order(NE).getInt(pos), (bb, pos, v) -> bb.order(NE).putInt(pos, v)) }, {"float", Accessor.ofSegment(ValueLayout.JAVA_FLOAT, 42f, MemorySegment::get, MemorySegment::set, (bb, pos) -> bb.order(NE).getFloat(pos), (bb, pos, v) -> bb.order(NE).putFloat(pos, v)) }, {"long", Accessor.ofSegment(ValueLayout.JAVA_LONG, 42L, MemorySegment::get, MemorySegment::set, (bb, pos) -> bb.order(NE).getLong(pos), (bb, pos, v) -> bb.order(NE).putLong(pos, v)) }, {"double", Accessor.ofSegment(ValueLayout.JAVA_DOUBLE, 42d, MemorySegment::get, MemorySegment::set, (bb, pos) -> bb.order(NE).getDouble(pos), (bb, pos, v) -> bb.order(NE).putDouble(pos, v)) }, { "address", Accessor.ofSegment(ValueLayout.ADDRESS, MemoryAddress.ofLong(42), MemorySegment::get, MemorySegment::set, (bb, pos) -> { ByteBuffer nb = bb.order(NE); long addr = ValueLayout.ADDRESS.byteSize() == 8 ? nb.getLong(pos) : nb.getInt(pos); return MemoryAddress.ofLong(addr); }, (bb, pos, v) -> { ByteBuffer nb = bb.order(NE); if (ValueLayout.ADDRESS.byteSize() == 8) { nb.putLong(pos, v.toRawLongValue()); } else { nb.putInt(pos, (int)v.toRawLongValue()); } }) }, {"char/index", Accessor.ofSegment(ValueLayout.JAVA_CHAR, (char) 42, MemorySegment::getAtIndex, MemorySegment::setAtIndex, (bb, pos) -> bb.order(NE).getChar(pos * 2), (bb, pos, v) -> bb.order(NE).putChar(pos * 2, v)) }, {"int/index", Accessor.ofSegment(ValueLayout.JAVA_INT, 42, MemorySegment::getAtIndex, MemorySegment::setAtIndex, (bb, pos) -> bb.order(NE).getInt(pos * 4), (bb, pos, v) -> bb.order(NE).putInt(pos * 4, v)) }, {"float/index", Accessor.ofSegment(ValueLayout.JAVA_FLOAT, 42f, MemorySegment::getAtIndex, MemorySegment::setAtIndex, (bb, pos) -> bb.order(NE).getFloat(pos * 4), (bb, pos, v) -> bb.order(NE).putFloat(pos * 4, v)) }, {"long/index", Accessor.ofSegment(ValueLayout.JAVA_LONG, 42L, MemorySegment::getAtIndex, MemorySegment::setAtIndex, (bb, pos) -> bb.order(NE).getLong(pos * 8), (bb, pos, v) -> bb.order(NE).putLong(pos * 8, v)) }, {"double/index", Accessor.ofSegment(ValueLayout.JAVA_DOUBLE, 42d, MemorySegment::getAtIndex, MemorySegment::setAtIndex, (bb, pos) -> bb.order(NE).getDouble(pos * 8), (bb, pos, v) -> bb.order(NE).putDouble(pos * 8, v)) }, { "address/index", Accessor.ofSegment(ValueLayout.ADDRESS, MemoryAddress.ofLong(42), MemorySegment::getAtIndex, MemorySegment::setAtIndex, (bb, pos) -> { ByteBuffer nb = bb.order(NE); long addr = ValueLayout.ADDRESS.byteSize() == 8 ? nb.getLong(pos * 8) : nb.getInt(pos * 4); return MemoryAddress.ofLong(addr); }, (bb, pos, v) -> { ByteBuffer nb = bb.order(NE); if (ValueLayout.ADDRESS.byteSize() == 8) { nb.putLong(pos * 8, v.toRawLongValue()); } else { nb.putInt(pos * 4, (int)v.toRawLongValue()); } }) }, }; } @DataProvider(name = "addressAccessors") static Object[][] addressAccessors() { return new Object[][]{ {"byte", Accessor.ofAddress(ValueLayout.JAVA_BYTE, (byte) 42, MemoryAddress::get, MemoryAddress::set, ByteBuffer::get, ByteBuffer::put) }, {"bool", Accessor.ofAddress(ValueLayout.JAVA_BOOLEAN, false, MemoryAddress::get, MemoryAddress::set, (bb, pos) -> bb.get(pos) != 0, (bb, pos, v) -> bb.put(pos, v ? (byte)1 : (byte)0)) }, {"char", Accessor.ofAddress(ValueLayout.JAVA_CHAR, (char) 42, MemoryAddress::get, MemoryAddress::set, (bb, pos) -> bb.order(NE).getChar(pos), (bb, pos, v) -> bb.order(NE).putChar(pos, v)) }, {"int", Accessor.ofAddress(ValueLayout.JAVA_INT, 42, MemoryAddress::get, MemoryAddress::set, (bb, pos) -> bb.order(NE).getInt(pos), (bb, pos, v) -> bb.order(NE).putInt(pos, v)) }, {"float", Accessor.ofAddress(ValueLayout.JAVA_FLOAT, 42f, MemoryAddress::get, MemoryAddress::set, (bb, pos) -> bb.order(NE).getFloat(pos), (bb, pos, v) -> bb.order(NE).putFloat(pos, v)) }, {"long", Accessor.ofAddress(ValueLayout.JAVA_LONG, 42L, MemoryAddress::get, MemoryAddress::set, (bb, pos) -> bb.order(NE).getLong(pos), (bb, pos, v) -> bb.order(NE).putLong(pos, v)) }, {"double", Accessor.ofAddress(ValueLayout.JAVA_DOUBLE, 42d, MemoryAddress::get, MemoryAddress::set, (bb, pos) -> bb.order(NE).getDouble(pos), (bb, pos, v) -> bb.order(NE).putDouble(pos, v)) }, { "address", Accessor.ofAddress(ValueLayout.ADDRESS, MemoryAddress.ofLong(42), MemoryAddress::get, MemoryAddress::set, (bb, pos) -> { ByteBuffer nb = bb.order(NE); long addr = ValueLayout.ADDRESS.byteSize() == 8 ? nb.getLong(pos) : nb.getInt(pos); return MemoryAddress.ofLong(addr); }, (bb, pos, v) -> { ByteBuffer nb = bb.order(NE); if (ValueLayout.ADDRESS.byteSize() == 8) { nb.putLong(pos, v.toRawLongValue()); } else { nb.putInt(pos, (int)v.toRawLongValue()); } }) }, {"char/index", Accessor.ofAddress(ValueLayout.JAVA_CHAR, (char) 42, MemoryAddress::getAtIndex, MemoryAddress::setAtIndex, (bb, pos) -> bb.order(NE).getChar(pos * 2), (bb, pos, v) -> bb.order(NE).putChar(pos * 2, v)) }, {"int/index", Accessor.ofAddress(ValueLayout.JAVA_INT, 42, MemoryAddress::getAtIndex, MemoryAddress::setAtIndex, (bb, pos) -> bb.order(NE).getInt(pos * 4), (bb, pos, v) -> bb.order(NE).putInt(pos * 4, v)) }, {"float/index", Accessor.ofAddress(ValueLayout.JAVA_FLOAT, 42f, MemoryAddress::getAtIndex, MemoryAddress::setAtIndex, (bb, pos) -> bb.order(NE).getFloat(pos * 4), (bb, pos, v) -> bb.order(NE).putFloat(pos * 4, v)) }, {"long/index", Accessor.ofAddress(ValueLayout.JAVA_LONG, 42L, MemoryAddress::getAtIndex, MemoryAddress::setAtIndex, (bb, pos) -> bb.order(NE).getLong(pos * 8), (bb, pos, v) -> bb.order(NE).putLong(pos * 8, v)) }, {"double/index", Accessor.ofAddress(ValueLayout.JAVA_DOUBLE, 42d, MemoryAddress::getAtIndex, MemoryAddress::setAtIndex, (bb, pos) -> bb.order(NE).getDouble(pos * 8), (bb, pos, v) -> bb.order(NE).putDouble(pos * 8, v)) }, { "address/index", Accessor.ofAddress(ValueLayout.ADDRESS, MemoryAddress.ofLong(42), MemoryAddress::getAtIndex, MemoryAddress::setAtIndex, (bb, pos) -> { ByteBuffer nb = bb.order(NE); long addr = ValueLayout.ADDRESS.byteSize() == 8 ? nb.getLong(pos * 8) : nb.getInt(pos * 4); return MemoryAddress.ofLong(addr); }, (bb, pos, v) -> { ByteBuffer nb = bb.order(NE); if (ValueLayout.ADDRESS.byteSize() == 8) { nb.putLong(pos * 8, v.toRawLongValue()); } else { nb.putInt(pos * 4, (int)v.toRawLongValue()); } }) } }; } }