2019-12-12 22:59:57 +00:00
|
|
|
/*
|
2023-08-07 10:58:11 +00:00
|
|
|
* Copyright (c) 2019, 2023, Oracle and/or its affiliates. All rights reserved.
|
|
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
2019-12-12 22:59:57 +00:00
|
|
|
*
|
2023-08-07 10:58:11 +00:00
|
|
|
* 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.
|
2019-12-12 22:59:57 +00:00
|
|
|
*
|
2023-08-07 10:58:11 +00:00
|
|
|
* 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).
|
2019-12-12 22:59:57 +00:00
|
|
|
*
|
2023-08-07 10:58:11 +00:00
|
|
|
* 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.
|
2019-12-12 22:59:57 +00:00
|
|
|
*
|
2023-08-07 10:58:11 +00:00
|
|
|
* 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.
|
2019-12-12 22:59:57 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
|
|
|
* @run testng TestLayouts
|
|
|
|
*/
|
|
|
|
|
2022-05-12 16:17:45 +00:00
|
|
|
import java.lang.foreign.*;
|
2019-12-12 22:59:57 +00:00
|
|
|
|
|
|
|
import java.lang.invoke.VarHandle;
|
|
|
|
import java.nio.ByteOrder;
|
2023-05-05 15:59:13 +00:00
|
|
|
import java.util.ArrayList;
|
2023-04-27 09:00:58 +00:00
|
|
|
import java.util.List;
|
2023-11-15 13:52:06 +00:00
|
|
|
import java.util.Locale;
|
2019-12-12 22:59:57 +00:00
|
|
|
import java.util.function.LongFunction;
|
2020-05-25 09:54:39 +00:00
|
|
|
import java.util.stream.Stream;
|
2019-12-12 22:59:57 +00:00
|
|
|
|
|
|
|
import org.testng.annotations.*;
|
2021-11-24 11:51:16 +00:00
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
import static java.lang.foreign.ValueLayout.*;
|
2019-12-12 22:59:57 +00:00
|
|
|
import static org.testng.Assert.*;
|
|
|
|
|
|
|
|
public class TestLayouts {
|
|
|
|
|
|
|
|
@Test(dataProvider = "badAlignments", expectedExceptions = IllegalArgumentException.class)
|
|
|
|
public void testBadLayoutAlignment(MemoryLayout layout, long alignment) {
|
2023-05-22 14:57:00 +00:00
|
|
|
layout.withByteAlignment(alignment);
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
@Test(dataProvider = "basicLayoutsAndAddressAndGroups")
|
|
|
|
public void testEqualities(MemoryLayout layout) {
|
|
|
|
|
|
|
|
// Use another Type
|
2023-05-22 14:57:00 +00:00
|
|
|
MemoryLayout differentType = MemoryLayout.paddingLayout(1);
|
2023-04-27 09:00:58 +00:00
|
|
|
assertFalse(layout.equals(differentType));
|
|
|
|
|
|
|
|
// Use another name
|
|
|
|
MemoryLayout differentName = layout.withName("CustomName");
|
|
|
|
assertFalse(layout.equals(differentName));
|
|
|
|
|
|
|
|
// Use another alignment
|
2023-05-22 14:57:00 +00:00
|
|
|
MemoryLayout differentAlignment = layout.withByteAlignment(layout.byteAlignment() * 2);
|
2023-04-27 09:00:58 +00:00
|
|
|
assertFalse(layout.equals(differentAlignment));
|
|
|
|
|
|
|
|
// Swap endian
|
|
|
|
MemoryLayout differentOrder = JAVA_INT.withOrder(JAVA_INT.order() == ByteOrder.BIG_ENDIAN ? ByteOrder.LITTLE_ENDIAN : ByteOrder.BIG_ENDIAN);
|
|
|
|
assertFalse(layout.equals(differentOrder));
|
|
|
|
|
|
|
|
// Something totally different
|
|
|
|
assertFalse(layout.equals("A"));
|
|
|
|
|
|
|
|
// Null
|
|
|
|
assertFalse(layout.equals(null));
|
|
|
|
|
|
|
|
// Identity
|
|
|
|
assertTrue(layout.equals(layout));
|
|
|
|
|
|
|
|
assertFalse(layout.equals(MemoryLayout.sequenceLayout(13, JAVA_LONG)));
|
|
|
|
|
2023-05-22 14:57:00 +00:00
|
|
|
MemoryLayout other = layout.withByteAlignment(16).withByteAlignment(layout.byteAlignment());
|
2023-04-27 09:00:58 +00:00
|
|
|
assertTrue(layout.equals(other));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
public void testTargetLayoutEquals() {
|
|
|
|
MemoryLayout differentTargetLayout = ADDRESS.withTargetLayout(JAVA_CHAR);
|
|
|
|
assertFalse(ADDRESS.equals(differentTargetLayout));
|
|
|
|
var equalButNotSame = ADDRESS.withTargetLayout(JAVA_INT).withTargetLayout(JAVA_CHAR);
|
|
|
|
assertTrue(differentTargetLayout.equals(equalButNotSame));
|
|
|
|
}
|
|
|
|
|
2019-12-12 22:59:57 +00:00
|
|
|
@Test
|
|
|
|
public void testIndexedSequencePath() {
|
2021-11-24 11:51:16 +00:00
|
|
|
MemoryLayout seq = MemoryLayout.sequenceLayout(10, ValueLayout.JAVA_INT);
|
2023-04-27 09:00:58 +00:00
|
|
|
try (Arena arena = Arena.ofConfined()) {
|
|
|
|
MemorySegment segment = arena.allocate(seq);;
|
2021-11-24 11:51:16 +00:00
|
|
|
VarHandle indexHandle = seq.varHandle(MemoryLayout.PathElement.sequenceElement());
|
2019-12-12 22:59:57 +00:00
|
|
|
// init segment
|
|
|
|
for (int i = 0 ; i < 10 ; i++) {
|
2023-10-12 19:50:08 +00:00
|
|
|
indexHandle.set(segment, 0L, (long)i, i);
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
|
|
|
//check statically indexed handles
|
|
|
|
for (int i = 0 ; i < 10 ; i++) {
|
2021-11-24 11:51:16 +00:00
|
|
|
VarHandle preindexHandle = seq.varHandle(MemoryLayout.PathElement.sequenceElement(i));
|
2023-10-12 19:50:08 +00:00
|
|
|
int expected = (int)indexHandle.get(segment, 0L, (long)i);
|
|
|
|
int found = (int)preindexHandle.get(segment, 0L);
|
2019-12-12 22:59:57 +00:00
|
|
|
assertEquals(expected, found);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-15 11:09:00 +00:00
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
|
|
public void testBadBoundSequenceLayoutResize() {
|
2021-11-24 11:51:16 +00:00
|
|
|
SequenceLayout seq = MemoryLayout.sequenceLayout(10, ValueLayout.JAVA_INT);
|
2020-01-15 11:09:00 +00:00
|
|
|
seq.withElementCount(-1);
|
|
|
|
}
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
|
|
public void testReshape() {
|
|
|
|
SequenceLayout layout = MemoryLayout.sequenceLayout(10, JAVA_INT);
|
|
|
|
layout.reshape();
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider = "basicLayoutsAndAddressAndGroups", expectedExceptions = IllegalArgumentException.class)
|
|
|
|
public void testGroupIllegalAlignmentNotPowerOfTwo(MemoryLayout layout) {
|
2023-05-22 14:57:00 +00:00
|
|
|
layout.withByteAlignment(9);
|
2023-04-27 09:00:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider = "basicLayoutsAndAddressAndGroups", expectedExceptions = IllegalArgumentException.class)
|
2023-05-22 14:57:00 +00:00
|
|
|
public void testGroupIllegalAlignmentNotGreaterOrEqualTo1(MemoryLayout layout) {
|
|
|
|
layout.withByteAlignment(0);
|
2023-04-27 09:00:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testEqualsPadding() {
|
2023-05-22 14:57:00 +00:00
|
|
|
PaddingLayout paddingLayout = MemoryLayout.paddingLayout(2);
|
2023-04-27 09:00:58 +00:00
|
|
|
testEqualities(paddingLayout);
|
2023-05-22 14:57:00 +00:00
|
|
|
PaddingLayout paddingLayout2 = MemoryLayout.paddingLayout(4);
|
2023-04-27 09:00:58 +00:00
|
|
|
assertNotEquals(paddingLayout, paddingLayout2);
|
|
|
|
}
|
|
|
|
|
2019-12-12 22:59:57 +00:00
|
|
|
@Test
|
|
|
|
public void testEmptyGroup() {
|
2021-06-02 10:53:06 +00:00
|
|
|
MemoryLayout struct = MemoryLayout.structLayout();
|
2023-05-22 14:57:00 +00:00
|
|
|
assertEquals(struct.byteSize(), 0);
|
|
|
|
assertEquals(struct.byteAlignment(), 1);
|
2019-12-12 22:59:57 +00:00
|
|
|
|
2021-06-02 10:53:06 +00:00
|
|
|
MemoryLayout union = MemoryLayout.unionLayout();
|
2023-05-22 14:57:00 +00:00
|
|
|
assertEquals(union.byteSize(), 0);
|
|
|
|
assertEquals(union.byteAlignment(), 1);
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testStructSizeAndAlign() {
|
2021-06-02 10:53:06 +00:00
|
|
|
MemoryLayout struct = MemoryLayout.structLayout(
|
2023-05-22 14:57:00 +00:00
|
|
|
MemoryLayout.paddingLayout(1),
|
2021-11-24 11:51:16 +00:00
|
|
|
ValueLayout.JAVA_BYTE,
|
|
|
|
ValueLayout.JAVA_CHAR,
|
|
|
|
ValueLayout.JAVA_INT,
|
|
|
|
ValueLayout.JAVA_LONG
|
2019-12-12 22:59:57 +00:00
|
|
|
);
|
|
|
|
assertEquals(struct.byteSize(), 1 + 1 + 2 + 4 + 8);
|
2024-03-05 13:32:24 +00:00
|
|
|
assertEquals(struct.byteAlignment(), 8);
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
|
|
|
|
2020-05-25 09:54:39 +00:00
|
|
|
@Test(dataProvider="basicLayouts")
|
|
|
|
public void testPaddingNoAlign(MemoryLayout layout) {
|
2023-05-22 14:57:00 +00:00
|
|
|
assertEquals(MemoryLayout.paddingLayout(layout.byteSize()).byteAlignment(), 1);
|
2020-05-25 09:54:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider="basicLayouts")
|
|
|
|
public void testStructPaddingAndAlign(MemoryLayout layout) {
|
2021-06-02 10:53:06 +00:00
|
|
|
MemoryLayout struct = MemoryLayout.structLayout(
|
2023-05-22 14:57:00 +00:00
|
|
|
layout, MemoryLayout.paddingLayout(16 - layout.byteSize()));
|
|
|
|
assertEquals(struct.byteAlignment(), layout.byteAlignment());
|
2020-05-25 09:54:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider="basicLayouts")
|
|
|
|
public void testUnionPaddingAndAlign(MemoryLayout layout) {
|
2021-06-02 10:53:06 +00:00
|
|
|
MemoryLayout struct = MemoryLayout.unionLayout(
|
2023-05-22 14:57:00 +00:00
|
|
|
layout, MemoryLayout.paddingLayout(16 - layout.byteSize()));
|
|
|
|
assertEquals(struct.byteAlignment(), layout.byteAlignment());
|
2020-05-25 09:54:39 +00:00
|
|
|
}
|
|
|
|
|
2019-12-12 22:59:57 +00:00
|
|
|
@Test
|
|
|
|
public void testUnionSizeAndAlign() {
|
2021-06-02 10:53:06 +00:00
|
|
|
MemoryLayout struct = MemoryLayout.unionLayout(
|
2021-11-24 11:51:16 +00:00
|
|
|
ValueLayout.JAVA_BYTE,
|
|
|
|
ValueLayout.JAVA_CHAR,
|
|
|
|
ValueLayout.JAVA_INT,
|
|
|
|
ValueLayout.JAVA_LONG
|
2019-12-12 22:59:57 +00:00
|
|
|
);
|
|
|
|
assertEquals(struct.byteSize(), 8);
|
2024-03-05 13:32:24 +00:00
|
|
|
assertEquals(struct.byteAlignment(), 8);
|
2022-05-12 16:17:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testSequenceBadCount() {
|
|
|
|
assertThrows(IllegalArgumentException.class, // negative
|
|
|
|
() -> MemoryLayout.sequenceLayout(-2, JAVA_SHORT));
|
|
|
|
}
|
|
|
|
|
2023-10-12 19:50:08 +00:00
|
|
|
@Test
|
2022-12-05 13:49:53 +00:00
|
|
|
public void testSequenceNegativeElementCount() {
|
|
|
|
assertThrows(IllegalArgumentException.class, // negative
|
|
|
|
() -> MemoryLayout.sequenceLayout(-1, JAVA_SHORT));
|
|
|
|
}
|
|
|
|
|
2022-05-12 16:17:45 +00:00
|
|
|
@Test
|
|
|
|
public void testSequenceOverflow() {
|
|
|
|
assertThrows(IllegalArgumentException.class, // negative
|
|
|
|
() -> MemoryLayout.sequenceLayout(Long.MAX_VALUE, JAVA_SHORT));
|
|
|
|
assertThrows(IllegalArgumentException.class, // flip back to positive
|
|
|
|
() -> MemoryLayout.sequenceLayout(Long.MAX_VALUE/3, JAVA_LONG));
|
2023-06-12 15:55:49 +00:00
|
|
|
assertThrows(IllegalArgumentException.class, // flip back to positive
|
|
|
|
() -> MemoryLayout.sequenceLayout(0, JAVA_LONG).withElementCount(Long.MAX_VALUE));
|
2022-05-12 16:17:45 +00:00
|
|
|
}
|
|
|
|
|
2023-11-10 14:24:22 +00:00
|
|
|
@Test
|
|
|
|
public void testSequenceLayoutWithZeroLength() {
|
|
|
|
SequenceLayout layout = MemoryLayout.sequenceLayout(0, JAVA_INT);
|
2023-11-15 13:52:06 +00:00
|
|
|
assertEquals(layout.toString().toLowerCase(Locale.ROOT), "[0:i4]");
|
2023-11-10 14:24:22 +00:00
|
|
|
|
|
|
|
SequenceLayout nested = MemoryLayout.sequenceLayout(0, layout);
|
2023-11-15 13:52:06 +00:00
|
|
|
assertEquals(nested.toString().toLowerCase(Locale.ROOT), "[0:[0:i4]]");
|
2023-11-10 14:24:22 +00:00
|
|
|
|
|
|
|
SequenceLayout layout2 = MemoryLayout.sequenceLayout(0, JAVA_INT);
|
|
|
|
assertEquals(layout, layout2);
|
|
|
|
|
|
|
|
SequenceLayout nested2 = MemoryLayout.sequenceLayout(0, layout2);
|
|
|
|
assertEquals(nested, nested2);
|
|
|
|
}
|
|
|
|
|
2022-05-12 16:17:45 +00:00
|
|
|
@Test
|
|
|
|
public void testStructOverflow() {
|
|
|
|
assertThrows(IllegalArgumentException.class, // negative
|
|
|
|
() -> MemoryLayout.structLayout(MemoryLayout.sequenceLayout(Long.MAX_VALUE, JAVA_BYTE),
|
|
|
|
MemoryLayout.sequenceLayout(Long.MAX_VALUE, JAVA_BYTE)));
|
|
|
|
assertThrows(IllegalArgumentException.class, // flip back to positive
|
|
|
|
() -> MemoryLayout.structLayout(MemoryLayout.sequenceLayout(Long.MAX_VALUE, JAVA_BYTE),
|
|
|
|
MemoryLayout.sequenceLayout(Long.MAX_VALUE, JAVA_BYTE),
|
|
|
|
MemoryLayout.sequenceLayout(Long.MAX_VALUE, JAVA_BYTE)));
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
@Test
|
|
|
|
public void testPadding() {
|
2023-05-22 14:57:00 +00:00
|
|
|
var padding = MemoryLayout.paddingLayout(1);
|
2023-04-27 09:00:58 +00:00
|
|
|
assertEquals(padding.byteAlignment(), 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testPaddingInStruct() {
|
2023-05-22 14:57:00 +00:00
|
|
|
var padding = MemoryLayout.paddingLayout(1);
|
2023-04-27 09:00:58 +00:00
|
|
|
var struct = MemoryLayout.structLayout(padding);
|
|
|
|
assertEquals(struct.byteAlignment(), 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
2023-05-22 14:57:00 +00:00
|
|
|
public void testPaddingIllegalByteSize() {
|
|
|
|
for (long byteSize : List.of(-1L, 0L)) {
|
2023-04-27 09:00:58 +00:00
|
|
|
try {
|
2023-05-22 14:57:00 +00:00
|
|
|
MemoryLayout.paddingLayout(byteSize);
|
|
|
|
fail("byte size cannot be " + byteSize);
|
2023-04-27 09:00:58 +00:00
|
|
|
} catch (IllegalArgumentException ignore) {
|
|
|
|
// Happy path
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void testStructToString() {
|
2023-07-24 15:00:47 +00:00
|
|
|
for (ByteOrder order : List.of(ByteOrder.LITTLE_ENDIAN, ByteOrder.BIG_ENDIAN)) {
|
|
|
|
String intRepresentation = (order == ByteOrder.LITTLE_ENDIAN ? "i" : "I");
|
|
|
|
StructLayout padding = MemoryLayout.structLayout(JAVA_INT.withOrder(order)).withName("struct");
|
|
|
|
assertEquals(padding.toString(), "[" + intRepresentation + "4](struct)");
|
|
|
|
var toStringUnaligned = padding.withByteAlignment(8).toString();
|
|
|
|
assertEquals(toStringUnaligned, "8%[" + intRepresentation + "4](struct)");
|
|
|
|
}
|
2023-04-27 09:00:58 +00:00
|
|
|
}
|
|
|
|
|
2020-05-25 09:54:39 +00:00
|
|
|
@Test(dataProvider = "layoutKinds")
|
|
|
|
public void testPadding(LayoutKind kind) {
|
2022-12-05 13:49:53 +00:00
|
|
|
assertEquals(kind == LayoutKind.PADDING, kind.layout instanceof PaddingLayout);
|
2020-05-25 09:54:39 +00:00
|
|
|
}
|
|
|
|
|
2019-12-12 22:59:57 +00:00
|
|
|
@Test(dataProvider="layoutsAndAlignments")
|
2023-05-22 14:57:00 +00:00
|
|
|
public void testAlignmentString(MemoryLayout layout, long byteAlign) {
|
|
|
|
long[] alignments = { 1, 2, 4, 8, 16 };
|
2019-12-12 22:59:57 +00:00
|
|
|
for (long a : alignments) {
|
2023-05-22 14:57:00 +00:00
|
|
|
if (layout.byteAlignment() == byteAlign) {
|
2020-11-12 16:37:23 +00:00
|
|
|
assertFalse(layout.toString().contains("%"));
|
2023-05-22 14:57:00 +00:00
|
|
|
if (a >= layout.byteAlignment()) {
|
|
|
|
assertEquals(layout.withByteAlignment(a).toString().contains("%"), a != byteAlign);
|
2023-04-27 09:00:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider="layoutsAndAlignments")
|
2023-05-22 14:57:00 +00:00
|
|
|
public void testBadByteAlignment(MemoryLayout layout, long byteAlign) {
|
|
|
|
long[] alignments = { 1, 2, 4, 8, 16 };
|
2023-04-27 09:00:58 +00:00
|
|
|
for (long a : alignments) {
|
2023-05-22 14:57:00 +00:00
|
|
|
if (a < byteAlign && !(layout instanceof ValueLayout)) {
|
|
|
|
assertThrows(IllegalArgumentException.class, () -> layout.withByteAlignment(a));
|
2020-11-12 16:37:23 +00:00
|
|
|
}
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
@Test(dataProvider="layoutsAndAlignments", expectedExceptions = IllegalArgumentException.class)
|
2023-05-22 14:57:00 +00:00
|
|
|
public void testBadSequenceElementAlignmentTooBig(MemoryLayout layout, long byteAlign) {
|
|
|
|
layout = layout.withByteAlignment(layout.byteSize() * 2); // hyper-align
|
2023-10-12 19:50:08 +00:00
|
|
|
MemoryLayout.sequenceLayout(1, layout);
|
2023-04-27 09:00:58 +00:00
|
|
|
}
|
|
|
|
|
2023-05-05 15:59:13 +00:00
|
|
|
@Test(dataProvider="layoutsAndAlignments")
|
2023-05-22 14:57:00 +00:00
|
|
|
public void testBadSequenceElementSizeNotMultipleOfAlignment(MemoryLayout layout, long byteAlign) {
|
2023-05-05 15:59:13 +00:00
|
|
|
boolean shouldFail = layout.byteSize() % layout.byteAlignment() != 0;
|
|
|
|
try {
|
2023-10-12 19:50:08 +00:00
|
|
|
MemoryLayout.sequenceLayout(1, layout);
|
2023-05-05 15:59:13 +00:00
|
|
|
assertFalse(shouldFail);
|
|
|
|
} catch (IllegalArgumentException ex) {
|
|
|
|
assertTrue(shouldFail);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider="layoutsAndAlignments")
|
2023-05-22 14:57:00 +00:00
|
|
|
public void testBadSpliteratorElementSizeNotMultipleOfAlignment(MemoryLayout layout, long byteAlign) {
|
2023-05-05 15:59:13 +00:00
|
|
|
boolean shouldFail = layout.byteSize() % layout.byteAlignment() != 0;
|
|
|
|
try (Arena arena = Arena.ofConfined()) {
|
|
|
|
MemorySegment segment = arena.allocate(layout);
|
|
|
|
segment.spliterator(layout);
|
|
|
|
assertFalse(shouldFail);
|
|
|
|
} catch (IllegalArgumentException ex) {
|
|
|
|
assertTrue(shouldFail);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(dataProvider="layoutsAndAlignments")
|
2023-05-22 14:57:00 +00:00
|
|
|
public void testBadElementsElementSizeNotMultipleOfAlignment(MemoryLayout layout, long byteAlign) {
|
2023-05-05 15:59:13 +00:00
|
|
|
boolean shouldFail = layout.byteSize() % layout.byteAlignment() != 0;
|
|
|
|
try (Arena arena = Arena.ofConfined()) {
|
|
|
|
MemorySegment segment = arena.allocate(layout);
|
|
|
|
segment.elements(layout);
|
|
|
|
assertFalse(shouldFail);
|
|
|
|
} catch (IllegalArgumentException ex) {
|
|
|
|
assertTrue(shouldFail);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
@Test(dataProvider="layoutsAndAlignments", expectedExceptions = IllegalArgumentException.class)
|
2023-05-22 14:57:00 +00:00
|
|
|
public void testBadStruct(MemoryLayout layout, long byteAlign) {
|
|
|
|
layout = layout.withByteAlignment(layout.byteSize() * 2); // hyper-align
|
2023-04-27 09:00:58 +00:00
|
|
|
MemoryLayout.structLayout(layout, layout);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(expectedExceptions = IllegalArgumentException.class)
|
|
|
|
public void testSequenceElement() {
|
|
|
|
SequenceLayout layout = MemoryLayout.sequenceLayout(10, JAVA_INT);
|
|
|
|
// Step must be != 0
|
|
|
|
PathElement.sequenceElement(3, 0);
|
|
|
|
}
|
|
|
|
|
2023-10-12 19:50:08 +00:00
|
|
|
@Test
|
|
|
|
public void testVarHandleCaching() {
|
|
|
|
assertSame(JAVA_INT.varHandle(), JAVA_INT.varHandle());
|
|
|
|
assertSame(JAVA_INT.withName("foo").varHandle(), JAVA_INT.varHandle());
|
|
|
|
|
|
|
|
assertNotSame(JAVA_INT_UNALIGNED.varHandle(), JAVA_INT.varHandle());
|
|
|
|
assertNotSame(ADDRESS.withTargetLayout(JAVA_INT).varHandle(), ADDRESS.varHandle());
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(expectedExceptions=IllegalArgumentException.class,
|
2024-01-08 08:20:07 +00:00
|
|
|
expectedExceptionsMessageRegExp=".*offset is negative.*")
|
2023-10-12 19:50:08 +00:00
|
|
|
public void testScaleNegativeOffset() {
|
|
|
|
JAVA_INT.scale(-1, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(expectedExceptions=IllegalArgumentException.class,
|
2024-01-08 08:20:07 +00:00
|
|
|
expectedExceptionsMessageRegExp=".*index is negative.*")
|
2023-10-12 19:50:08 +00:00
|
|
|
public void testScaleNegativeIndex() {
|
|
|
|
JAVA_INT.scale(0, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(expectedExceptions=ArithmeticException.class)
|
|
|
|
public void testScaleAddOverflow() {
|
|
|
|
JAVA_INT.scale(Long.MAX_VALUE, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test(expectedExceptions=ArithmeticException.class)
|
|
|
|
public void testScaleMultiplyOverflow() {
|
|
|
|
JAVA_INT.scale(0, Long.MAX_VALUE);
|
|
|
|
}
|
|
|
|
|
2019-12-12 22:59:57 +00:00
|
|
|
@DataProvider(name = "badAlignments")
|
|
|
|
public Object[][] layoutsAndBadAlignments() {
|
|
|
|
LayoutKind[] layoutKinds = LayoutKind.values();
|
|
|
|
Object[][] values = new Object[layoutKinds.length * 2][2];
|
|
|
|
for (int i = 0; i < layoutKinds.length ; i++) {
|
2023-05-22 14:57:00 +00:00
|
|
|
values[i * 2] = new Object[] { layoutKinds[i].layout, 0 }; // smaller than 1
|
|
|
|
values[(i * 2) + 1] = new Object[] { layoutKinds[i].layout, 5 }; // not a power of 2
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
|
|
|
return values;
|
|
|
|
}
|
|
|
|
|
2020-05-25 09:54:39 +00:00
|
|
|
@DataProvider(name = "layoutKinds")
|
|
|
|
public Object[][] layoutsKinds() {
|
|
|
|
return Stream.of(LayoutKind.values())
|
|
|
|
.map(lk -> new Object[] { lk })
|
|
|
|
.toArray(Object[][]::new);
|
|
|
|
}
|
|
|
|
|
2019-12-12 22:59:57 +00:00
|
|
|
enum SizedLayoutFactory {
|
2021-11-24 11:51:16 +00:00
|
|
|
VALUE_LE(size -> valueLayoutForSize((int)size).withOrder(ByteOrder.LITTLE_ENDIAN)),
|
|
|
|
VALUE_BE(size -> valueLayoutForSize((int)size).withOrder(ByteOrder.BIG_ENDIAN)),
|
2021-06-02 10:53:06 +00:00
|
|
|
PADDING(MemoryLayout::paddingLayout),
|
2023-05-22 14:57:00 +00:00
|
|
|
SEQUENCE(size -> MemoryLayout.sequenceLayout(size, MemoryLayout.paddingLayout(1)));
|
2019-12-12 22:59:57 +00:00
|
|
|
|
|
|
|
private final LongFunction<MemoryLayout> factory;
|
|
|
|
|
|
|
|
SizedLayoutFactory(LongFunction<MemoryLayout> factory) {
|
|
|
|
this.factory = factory;
|
|
|
|
}
|
|
|
|
|
|
|
|
MemoryLayout make(long size) {
|
|
|
|
return factory.apply(size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-11-24 11:51:16 +00:00
|
|
|
static ValueLayout valueLayoutForSize(int size) {
|
|
|
|
return switch (size) {
|
|
|
|
case 1 -> JAVA_BYTE;
|
|
|
|
case 2 -> JAVA_SHORT;
|
|
|
|
case 4 -> JAVA_INT;
|
|
|
|
case 8 -> JAVA_LONG;
|
|
|
|
default -> throw new UnsupportedOperationException();
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-12-12 22:59:57 +00:00
|
|
|
enum LayoutKind {
|
2021-11-24 11:51:16 +00:00
|
|
|
VALUE(ValueLayout.JAVA_BYTE),
|
2023-05-22 14:57:00 +00:00
|
|
|
PADDING(MemoryLayout.paddingLayout(1)),
|
|
|
|
SEQUENCE(MemoryLayout.sequenceLayout(1, MemoryLayout.paddingLayout(1))),
|
|
|
|
STRUCT(MemoryLayout.structLayout(MemoryLayout.paddingLayout(1), MemoryLayout.paddingLayout(1))),
|
|
|
|
UNION(MemoryLayout.unionLayout(MemoryLayout.paddingLayout(1), MemoryLayout.paddingLayout(1)));
|
2019-12-12 22:59:57 +00:00
|
|
|
|
|
|
|
final MemoryLayout layout;
|
|
|
|
|
|
|
|
LayoutKind(MemoryLayout layout) {
|
|
|
|
this.layout = layout;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-25 09:54:39 +00:00
|
|
|
@DataProvider(name = "basicLayouts")
|
|
|
|
public Object[][] basicLayouts() {
|
|
|
|
return Stream.of(basicLayouts)
|
|
|
|
.map(l -> new Object[] { l })
|
|
|
|
.toArray(Object[][]::new);
|
|
|
|
}
|
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
@DataProvider(name = "basicLayoutsAndAddress")
|
|
|
|
public Object[][] basicLayoutsAndAddress() {
|
|
|
|
return Stream.concat(Stream.of(basicLayouts), Stream.of(ADDRESS))
|
|
|
|
.map(l -> new Object[] { l })
|
|
|
|
.toArray(Object[][]::new);
|
|
|
|
}
|
|
|
|
|
|
|
|
@DataProvider(name = "basicLayoutsAndAddressAndGroups")
|
|
|
|
public Object[][] basicLayoutsAndAddressAndGroups() {
|
|
|
|
return Stream.concat(Stream.concat(Stream.of(basicLayouts), Stream.of(ADDRESS)), groupLayoutStream())
|
|
|
|
.map(l -> new Object[] { l })
|
|
|
|
.toArray(Object[][]::new);
|
|
|
|
}
|
|
|
|
|
2019-12-12 22:59:57 +00:00
|
|
|
@DataProvider(name = "layoutsAndAlignments")
|
|
|
|
public Object[][] layoutsAndAlignments() {
|
2023-05-05 15:59:13 +00:00
|
|
|
List<Object[]> layoutsAndAlignments = new ArrayList<>();
|
2019-12-12 22:59:57 +00:00
|
|
|
int i = 0;
|
|
|
|
//add basic layouts
|
2024-03-05 13:32:24 +00:00
|
|
|
for (MemoryLayout l : basicLayouts) {
|
2023-05-22 14:57:00 +00:00
|
|
|
layoutsAndAlignments.add(new Object[] { l, l.byteAlignment() });
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
|
|
|
//add basic layouts wrapped in a sequence with given size
|
2024-03-05 13:32:24 +00:00
|
|
|
for (MemoryLayout l : basicLayouts) {
|
2023-05-22 14:57:00 +00:00
|
|
|
layoutsAndAlignments.add(new Object[] { MemoryLayout.sequenceLayout(4, l), l.byteAlignment() });
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
|
|
|
//add basic layouts wrapped in a struct
|
2024-03-05 13:32:24 +00:00
|
|
|
for (MemoryLayout l1 : basicLayouts) {
|
|
|
|
for (MemoryLayout l2 : basicLayouts) {
|
2023-05-05 15:59:13 +00:00
|
|
|
if (l1.byteSize() % l2.byteAlignment() != 0) continue; // second element is not aligned, skip
|
2023-05-22 14:57:00 +00:00
|
|
|
long align = Math.max(l1.byteAlignment(), l2.byteAlignment());
|
2023-05-05 15:59:13 +00:00
|
|
|
layoutsAndAlignments.add(new Object[]{MemoryLayout.structLayout(l1, l2), align});
|
|
|
|
}
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
|
|
|
//add basic layouts wrapped in a union
|
2024-03-05 13:32:24 +00:00
|
|
|
for (MemoryLayout l1 : basicLayouts) {
|
|
|
|
for (MemoryLayout l2 : basicLayouts) {
|
2023-05-22 14:57:00 +00:00
|
|
|
long align = Math.max(l1.byteAlignment(), l2.byteAlignment());
|
2023-05-05 15:59:13 +00:00
|
|
|
layoutsAndAlignments.add(new Object[]{MemoryLayout.unionLayout(l1, l2), align});
|
|
|
|
}
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
2023-05-05 15:59:13 +00:00
|
|
|
return layoutsAndAlignments.toArray(Object[][]::new);
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|
2020-05-25 09:54:39 +00:00
|
|
|
|
2023-04-27 09:00:58 +00:00
|
|
|
@DataProvider(name = "groupLayouts")
|
|
|
|
public Object[][] groupLayouts() {
|
|
|
|
return groupLayoutStream()
|
|
|
|
.map(l -> new Object[] { l })
|
|
|
|
.toArray(Object[][]::new);
|
|
|
|
}
|
|
|
|
|
|
|
|
@DataProvider(name = "validCarriers")
|
|
|
|
public Object[][] validCarriers() {
|
|
|
|
return Stream.of(
|
|
|
|
boolean.class,
|
|
|
|
byte.class,
|
|
|
|
char.class,
|
|
|
|
short.class,
|
|
|
|
int.class,
|
|
|
|
long.class,
|
|
|
|
float.class,
|
|
|
|
double.class,
|
|
|
|
MemorySegment.class
|
|
|
|
)
|
|
|
|
.map(l -> new Object[]{l})
|
|
|
|
.toArray(Object[][]::new);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Stream<MemoryLayout> groupLayoutStream() {
|
|
|
|
return Stream.of(
|
|
|
|
MemoryLayout.sequenceLayout(10, JAVA_INT),
|
2023-05-22 14:57:00 +00:00
|
|
|
MemoryLayout.structLayout(JAVA_INT, MemoryLayout.paddingLayout(4), JAVA_LONG),
|
2023-04-27 09:00:58 +00:00
|
|
|
MemoryLayout.unionLayout(JAVA_LONG, JAVA_DOUBLE)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-05-19 14:28:14 +00:00
|
|
|
static ValueLayout[] basicLayouts = {
|
2021-11-24 11:51:16 +00:00
|
|
|
ValueLayout.JAVA_BYTE,
|
|
|
|
ValueLayout.JAVA_CHAR,
|
|
|
|
ValueLayout.JAVA_SHORT,
|
|
|
|
ValueLayout.JAVA_INT,
|
|
|
|
ValueLayout.JAVA_FLOAT,
|
|
|
|
ValueLayout.JAVA_LONG,
|
|
|
|
ValueLayout.JAVA_DOUBLE,
|
2020-05-25 09:54:39 +00:00
|
|
|
};
|
2019-12-12 22:59:57 +00:00
|
|
|
}
|