jdk-24/test/jdk/java/foreign/valist/VaListTest.java

808 lines
40 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2020, 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
* @modules jdk.incubator.foreign/jdk.internal.foreign
* jdk.incubator.foreign/jdk.internal.foreign.abi
* jdk.incubator.foreign/jdk.internal.foreign.abi.aarch64
* jdk.incubator.foreign/jdk.internal.foreign.abi.x64.windows
* jdk.incubator.foreign/jdk.internal.foreign.abi.x64.sysv
* @run testng/othervm -Dforeign.restricted=permit VaListTest
*/
import jdk.incubator.foreign.*;
import jdk.incubator.foreign.CLinker.VaList;
import jdk.internal.foreign.abi.SharedUtils;
import jdk.internal.foreign.abi.aarch64.AArch64Linker;
import jdk.internal.foreign.abi.x64.sysv.SysVx64Linker;
import jdk.internal.foreign.abi.x64.windows.Windowsx64Linker;
import org.testng.annotations.DataProvider;
import org.testng.annotations.Test;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandleProxies;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;
import java.lang.invoke.VarHandle;
import java.util.function.BiFunction;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.stream.DoubleStream;
import java.util.stream.IntStream;
import static jdk.incubator.foreign.CLinker.C_DOUBLE;
import static jdk.incubator.foreign.CLinker.C_FLOAT;
import static jdk.incubator.foreign.CLinker.C_INT;
import static jdk.incubator.foreign.CLinker.C_LONG_LONG;
import static jdk.incubator.foreign.CLinker.C_POINTER;
import static jdk.incubator.foreign.CLinker.C_VA_LIST;
import static jdk.incubator.foreign.MemoryLayout.PathElement.groupElement;
import static jdk.incubator.foreign.MemoryLayouts.JAVA_INT;
import static jdk.internal.foreign.PlatformLayouts.*;
import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertFalse;
import static org.testng.Assert.assertTrue;
public class VaListTest {
private static final CLinker abi = CLinker.getInstance();
private static final LibraryLookup lookup = LibraryLookup.ofLibrary("VaList");
private static final MethodHandle MH_sumInts = link("sumInts",
MethodType.methodType(int.class, int.class, VaList.class),
FunctionDescriptor.of(C_INT, C_INT, C_VA_LIST));
private static final MethodHandle MH_sumDoubles = link("sumDoubles",
MethodType.methodType(double.class, int.class, VaList.class),
FunctionDescriptor.of(C_DOUBLE, C_INT, C_VA_LIST));
private static final MethodHandle MH_getInt = link("getInt",
MethodType.methodType(int.class, VaList.class),
FunctionDescriptor.of(C_INT, C_VA_LIST));
private static final MethodHandle MH_sumStruct = link("sumStruct",
MethodType.methodType(int.class, VaList.class),
FunctionDescriptor.of(C_INT, C_VA_LIST));
private static final MethodHandle MH_sumBigStruct = link("sumBigStruct",
MethodType.methodType(long.class, VaList.class),
FunctionDescriptor.of(C_LONG_LONG, C_VA_LIST));
private static final MethodHandle MH_sumHugeStruct = link("sumHugeStruct",
MethodType.methodType(long.class, VaList.class),
FunctionDescriptor.of(C_LONG_LONG, C_VA_LIST));
private static final MethodHandle MH_sumFloatStruct = link("sumFloatStruct",
MethodType.methodType(float.class, VaList.class),
FunctionDescriptor.of(C_FLOAT, C_VA_LIST));
private static final MethodHandle MH_sumStack = link("sumStack",
MethodType.methodType(void.class, MemoryAddress.class, MemoryAddress.class, VaList.class),
FunctionDescriptor.ofVoid(C_POINTER, C_POINTER, C_VA_LIST));
private static MethodHandle link(String symbol, MethodType mt, FunctionDescriptor fd) {
return abi.downcallHandle(lookup.lookup(symbol).get(), mt, fd);
}
private static MethodHandle linkVaListCB(String symbol) {
return link(symbol,
MethodType.methodType(void.class, MemoryAddress.class),
FunctionDescriptor.ofVoid(C_POINTER));
}
private static final Function<Consumer<VaList.Builder>, VaList> winVaListFactory
= actions -> Windowsx64Linker.newVaList(actions, MemorySegment::allocateNative);
private static final Function<Consumer<VaList.Builder>, VaList> sysvVaListFactory
= actions -> SysVx64Linker.newVaList(actions, MemorySegment::allocateNative);
private static final Function<Consumer<VaList.Builder>, VaList> aarch64VaListFactory
= actions -> AArch64Linker.newVaList(actions, MemorySegment::allocateNative);
private static final Function<Consumer<VaList.Builder>, VaList> platformVaListFactory
= VaList::make;
private static final BiFunction<Consumer<VaList.Builder>, NativeScope, VaList> winVaListScopedFactory
= (actions, scope) -> Windowsx64Linker.newVaList(actions, SharedUtils.Allocator.ofScope(scope));
private static final BiFunction<Consumer<VaList.Builder>, NativeScope, VaList> sysvVaListScopedFactory
= (actions, scope) -> SysVx64Linker.newVaList(actions, SharedUtils.Allocator.ofScope(scope));
private static final BiFunction<Consumer<VaList.Builder>, NativeScope, VaList> aarch64VaListScopedFactory
= (actions, scope) -> AArch64Linker.newVaList(actions, SharedUtils.Allocator.ofScope(scope));
private static final BiFunction<Consumer<VaList.Builder>, NativeScope, VaList> platformVaListScopedFactory
= VaList::make;
@DataProvider
@SuppressWarnings("unchecked")
public static Object[][] sumInts() {
Function<MemoryLayout, BiFunction<Integer, VaList, Integer>> sumIntsJavaFact = layout ->
(num, list) -> IntStream.generate(() -> list.vargAsInt(layout)).limit(num).sum();
BiFunction<Integer, VaList, Integer> sumIntsNative
= MethodHandleProxies.asInterfaceInstance(BiFunction.class, MH_sumInts);
return new Object[][]{
{ winVaListFactory, sumIntsJavaFact.apply(Win64.C_INT), Win64.C_INT },
{ sysvVaListFactory, sumIntsJavaFact.apply(SysV.C_INT), SysV.C_INT },
{ aarch64VaListFactory, sumIntsJavaFact.apply(AArch64.C_INT), AArch64.C_INT },
{ platformVaListFactory, sumIntsNative, C_INT },
};
}
@Test(dataProvider = "sumInts")
public void testIntSum(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
BiFunction<Integer, VaList, Integer> sumInts,
ValueLayout intLayout) {
try (VaList vaList = vaListFactory.apply(b ->
b.vargFromInt(intLayout, 10)
.vargFromInt(intLayout, 15)
.vargFromInt(intLayout, 20))) {
int x = sumInts.apply(3, vaList);
assertEquals(x, 45);
}
}
@DataProvider
@SuppressWarnings("unchecked")
public static Object[][] sumDoubles() {
Function<MemoryLayout, BiFunction<Integer, VaList, Double>> sumDoublesJavaFact = layout ->
(num, list) -> DoubleStream.generate(() -> list.vargAsDouble(layout)).limit(num).sum();
BiFunction<Integer, VaList, Double> sumDoublesNative
= MethodHandleProxies.asInterfaceInstance(BiFunction.class, MH_sumDoubles);
return new Object[][]{
{ winVaListFactory, sumDoublesJavaFact.apply(Win64.C_DOUBLE), Win64.C_DOUBLE },
{ sysvVaListFactory, sumDoublesJavaFact.apply(SysV.C_DOUBLE), SysV.C_DOUBLE },
{ aarch64VaListFactory, sumDoublesJavaFact.apply(AArch64.C_DOUBLE), AArch64.C_DOUBLE },
{ platformVaListFactory, sumDoublesNative, C_DOUBLE },
};
}
@Test(dataProvider = "sumDoubles")
public void testDoubleSum(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
BiFunction<Integer, VaList, Double> sumDoubles,
ValueLayout doubleLayout) {
try (VaList vaList = vaListFactory.apply(b ->
b.vargFromDouble(doubleLayout, 3.0D)
.vargFromDouble(doubleLayout, 4.0D)
.vargFromDouble(doubleLayout, 5.0D))) {
double x = sumDoubles.apply(3, vaList);
assertEquals(x, 12.0D);
}
}
@DataProvider
@SuppressWarnings("unchecked")
public static Object[][] pointers() {
Function<MemoryLayout, Function<VaList, Integer>> getIntJavaFact = layout ->
list -> {
MemoryAddress ma = list.vargAsAddress(layout);
return MemoryAccess.getIntAtOffset(MemorySegment.ofNativeRestricted(), ma.toRawLongValue());
};
Function<VaList, Integer> getIntNative = MethodHandleProxies.asInterfaceInstance(Function.class, MH_getInt);
return new Object[][]{
{ winVaListFactory, getIntJavaFact.apply(Win64.C_POINTER), Win64.C_POINTER },
{ sysvVaListFactory, getIntJavaFact.apply(SysV.C_POINTER), SysV.C_POINTER },
{ aarch64VaListFactory, getIntJavaFact.apply(AArch64.C_POINTER), AArch64.C_POINTER },
{ platformVaListFactory, getIntNative, C_POINTER },
};
}
@Test(dataProvider = "pointers")
public void testVaListMemoryAddress(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
Function<VaList, Integer> getFromPointer,
ValueLayout pointerLayout) {
try (MemorySegment msInt = MemorySegment.allocateNative(JAVA_INT)) {
MemoryAccess.setInt(msInt, 10);
try (VaList vaList = vaListFactory.apply(b -> b.vargFromAddress(pointerLayout, msInt.address()))) {
int x = getFromPointer.apply(vaList);
assertEquals(x, 10);
}
}
}
interface TriFunction<S, T, U, R> {
R apply(S s, T t, U u);
}
@DataProvider
@SuppressWarnings("unchecked")
public static Object[][] structs() {
TriFunction<MemoryLayout, VarHandle, VarHandle, Function<VaList, Integer>> sumStructJavaFact
= (pointLayout, VH_Point_x, VH_Point_y) ->
list -> {
MemorySegment struct = list.vargAsSegment(pointLayout);
int x = (int) VH_Point_x.get(struct);
int y = (int) VH_Point_y.get(struct);
return x + y;
};
TriFunction<MemoryLayout, VarHandle, VarHandle, Function<VaList, Integer>> sumStructNativeFact
= (pointLayout, VH_Point_x, VH_Point_y) ->
MethodHandleProxies.asInterfaceInstance(Function.class, MH_sumStruct);
TriFunction<Function<Consumer<VaList.Builder>, VaList>, MemoryLayout,
TriFunction<MemoryLayout, VarHandle, VarHandle, Function<VaList, Integer>>, Object[]> argsFact
= (vaListFact, intLayout, sumStructFact) -> {
GroupLayout pointLayout = MemoryLayout.ofStruct(
intLayout.withName("x"),
intLayout.withName("y")
);
VarHandle VH_Point_x = pointLayout.varHandle(int.class, groupElement("x"));
VarHandle VH_Point_y = pointLayout.varHandle(int.class, groupElement("y"));
return new Object[] { vaListFact, sumStructFact.apply(pointLayout, VH_Point_x, VH_Point_y),
pointLayout, VH_Point_x, VH_Point_y };
};
return new Object[][]{
argsFact.apply(winVaListFactory, Win64.C_INT, sumStructJavaFact),
argsFact.apply(sysvVaListFactory, SysV.C_INT, sumStructJavaFact),
argsFact.apply(aarch64VaListFactory, AArch64.C_INT, sumStructJavaFact),
argsFact.apply(platformVaListFactory, C_INT, sumStructNativeFact),
};
}
@Test(dataProvider = "structs")
public void testStruct(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
Function<VaList, Integer> sumStruct,
GroupLayout Point_LAYOUT, VarHandle VH_Point_x, VarHandle VH_Point_y) {
try (MemorySegment struct = MemorySegment.allocateNative(Point_LAYOUT)) {
VH_Point_x.set(struct, 5);
VH_Point_y.set(struct, 10);
try (VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(Point_LAYOUT, struct))) {
int sum = sumStruct.apply(vaList);
assertEquals(sum, 15);
}
}
}
@DataProvider
@SuppressWarnings("unchecked")
public static Object[][] bigStructs() {
TriFunction<MemoryLayout, VarHandle, VarHandle, Function<VaList, Long>> sumStructJavaFact
= (BigPoint_LAYOUT, VH_BigPoint_x, VH_BigPoint_y) ->
list -> {
MemorySegment struct = list.vargAsSegment(BigPoint_LAYOUT);
long x = (long) VH_BigPoint_x.get(struct);
long y = (long) VH_BigPoint_y.get(struct);
return x + y;
};
TriFunction<MemoryLayout, VarHandle, VarHandle, Function<VaList, Long>> sumStructNativeFact
= (pointLayout, VH_BigPoint_x, VH_BigPoint_y) ->
MethodHandleProxies.asInterfaceInstance(Function.class, MH_sumBigStruct);
TriFunction<Function<Consumer<VaList.Builder>, VaList>, MemoryLayout,
TriFunction<MemoryLayout, VarHandle, VarHandle, Function<VaList, Long>>, Object[]> argsFact
= (vaListFact, longLongLayout, sumBigStructFact) -> {
GroupLayout BigPoint_LAYOUT = MemoryLayout.ofStruct(
longLongLayout.withName("x"),
longLongLayout.withName("y")
);
VarHandle VH_BigPoint_x = BigPoint_LAYOUT.varHandle(long.class, groupElement("x"));
VarHandle VH_BigPoint_y = BigPoint_LAYOUT.varHandle(long.class, groupElement("y"));
return new Object[] { vaListFact, sumBigStructFact.apply(BigPoint_LAYOUT, VH_BigPoint_x, VH_BigPoint_y),
BigPoint_LAYOUT, VH_BigPoint_x, VH_BigPoint_y };
};
return new Object[][]{
argsFact.apply(winVaListFactory, Win64.C_LONG_LONG, sumStructJavaFact),
argsFact.apply(sysvVaListFactory, SysV.C_LONG_LONG, sumStructJavaFact),
argsFact.apply(aarch64VaListFactory, AArch64.C_LONG_LONG, sumStructJavaFact),
argsFact.apply(platformVaListFactory, C_LONG_LONG, sumStructNativeFact),
};
}
@Test(dataProvider = "bigStructs")
public void testBigStruct(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
Function<VaList, Long> sumBigStruct,
GroupLayout BigPoint_LAYOUT, VarHandle VH_BigPoint_x, VarHandle VH_BigPoint_y) {
try (MemorySegment struct = MemorySegment.allocateNative(BigPoint_LAYOUT)) {
VH_BigPoint_x.set(struct, 5);
VH_BigPoint_y.set(struct, 10);
try (VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(BigPoint_LAYOUT, struct))) {
long sum = sumBigStruct.apply(vaList);
assertEquals(sum, 15);
}
}
}
@DataProvider
@SuppressWarnings("unchecked")
public static Object[][] floatStructs() {
TriFunction<MemoryLayout, VarHandle, VarHandle, Function<VaList, Float>> sumStructJavaFact
= (FloatPoint_LAYOUT, VH_FloatPoint_x, VH_FloatPoint_y) ->
list -> {
MemorySegment struct = list.vargAsSegment(FloatPoint_LAYOUT);
float x = (float) VH_FloatPoint_x.get(struct);
float y = (float) VH_FloatPoint_y.get(struct);
return x + y;
};
TriFunction<MemoryLayout, VarHandle, VarHandle, Function<VaList, Float>> sumStructNativeFact
= (pointLayout, VH_FloatPoint_x, VH_FloatPoint_y) ->
MethodHandleProxies.asInterfaceInstance(Function.class, MH_sumFloatStruct);
TriFunction<Function<Consumer<VaList.Builder>, VaList>, MemoryLayout,
TriFunction<MemoryLayout, VarHandle, VarHandle, Function<VaList, Float>>, Object[]> argsFact
= (vaListFact, floatLayout, sumFloatStructFact) -> {
GroupLayout FloatPoint_LAYOUT = MemoryLayout.ofStruct(
floatLayout.withName("x"),
floatLayout.withName("y")
);
VarHandle VH_FloatPoint_x = FloatPoint_LAYOUT.varHandle(float.class, groupElement("x"));
VarHandle VH_FloatPoint_y = FloatPoint_LAYOUT.varHandle(float.class, groupElement("y"));
return new Object[] { vaListFact, sumFloatStructFact.apply(FloatPoint_LAYOUT, VH_FloatPoint_x, VH_FloatPoint_y),
FloatPoint_LAYOUT, VH_FloatPoint_x, VH_FloatPoint_y };
};
return new Object[][]{
argsFact.apply(winVaListFactory, Win64.C_FLOAT, sumStructJavaFact),
argsFact.apply(sysvVaListFactory, SysV.C_FLOAT, sumStructJavaFact),
argsFact.apply(aarch64VaListFactory, AArch64.C_FLOAT, sumStructJavaFact),
argsFact.apply(platformVaListFactory, C_FLOAT, sumStructNativeFact),
};
}
@Test(dataProvider = "floatStructs")
public void testFloatStruct(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
Function<VaList, Float> sumFloatStruct,
GroupLayout FloatPoint_LAYOUT,
VarHandle VH_FloatPoint_x, VarHandle VH_FloatPoint_y) {
try (MemorySegment struct = MemorySegment.allocateNative(FloatPoint_LAYOUT)) {
VH_FloatPoint_x.set(struct, 1.234f);
VH_FloatPoint_y.set(struct, 3.142f);
try (VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(FloatPoint_LAYOUT, struct))) {
float sum = sumFloatStruct.apply(vaList);
assertEquals(sum, 4.376f, 0.00001f);
}
}
}
interface QuadFunc<T0, T1, T2, T3, R> {
R apply(T0 t0, T1 t1, T2 t2, T3 t3);
}
@DataProvider
@SuppressWarnings("unchecked")
public static Object[][] hugeStructs() {
QuadFunc<MemoryLayout, VarHandle, VarHandle, VarHandle, Function<VaList, Long>> sumStructJavaFact
= (HugePoint_LAYOUT, VH_HugePoint_x, VH_HugePoint_y, VH_HugePoint_z) ->
list -> {
MemorySegment struct = list.vargAsSegment(HugePoint_LAYOUT);
long x = (long) VH_HugePoint_x.get(struct);
long y = (long) VH_HugePoint_y.get(struct);
long z = (long) VH_HugePoint_z.get(struct);
return x + y + z;
};
QuadFunc<MemoryLayout, VarHandle, VarHandle, VarHandle, Function<VaList, Long>> sumStructNativeFact
= (pointLayout, VH_HugePoint_x, VH_HugePoint_y, VH_HugePoint_z) ->
MethodHandleProxies.asInterfaceInstance(Function.class, MH_sumHugeStruct);
TriFunction<Function<Consumer<VaList.Builder>, VaList>, MemoryLayout,
QuadFunc<MemoryLayout, VarHandle, VarHandle, VarHandle, Function<VaList, Long>>, Object[]> argsFact
= (vaListFact, longLongLayout, sumBigStructFact) -> {
GroupLayout HugePoint_LAYOUT = MemoryLayout.ofStruct(
longLongLayout.withName("x"),
longLongLayout.withName("y"),
longLongLayout.withName("z")
);
VarHandle VH_HugePoint_x = HugePoint_LAYOUT.varHandle(long.class, groupElement("x"));
VarHandle VH_HugePoint_y = HugePoint_LAYOUT.varHandle(long.class, groupElement("y"));
VarHandle VH_HugePoint_z = HugePoint_LAYOUT.varHandle(long.class, groupElement("z"));
return new Object[] { vaListFact,
sumBigStructFact.apply(HugePoint_LAYOUT, VH_HugePoint_x, VH_HugePoint_y, VH_HugePoint_z),
HugePoint_LAYOUT, VH_HugePoint_x, VH_HugePoint_y, VH_HugePoint_z };
};
return new Object[][]{
argsFact.apply(winVaListFactory, Win64.C_LONG_LONG, sumStructJavaFact),
argsFact.apply(sysvVaListFactory, SysV.C_LONG_LONG, sumStructJavaFact),
argsFact.apply(aarch64VaListFactory, AArch64.C_LONG_LONG, sumStructJavaFact),
argsFact.apply(platformVaListFactory, C_LONG_LONG, sumStructNativeFact),
};
}
@Test(dataProvider = "hugeStructs")
public void testHugeStruct(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
Function<VaList, Long> sumHugeStruct,
GroupLayout HugePoint_LAYOUT,
VarHandle VH_HugePoint_x, VarHandle VH_HugePoint_y, VarHandle VH_HugePoint_z) {
// On AArch64 a struct needs to be larger than 16 bytes to be
// passed by reference.
try (MemorySegment struct = MemorySegment.allocateNative(HugePoint_LAYOUT)) {
VH_HugePoint_x.set(struct, 1);
VH_HugePoint_y.set(struct, 2);
VH_HugePoint_z.set(struct, 3);
try (VaList vaList = vaListFactory.apply(b -> b.vargFromSegment(HugePoint_LAYOUT, struct))) {
long sum = sumHugeStruct.apply(vaList);
assertEquals(sum, 6);
}
}
}
public interface SumStackFunc {
void invoke(MemorySegment longSum, MemorySegment doubleSum, VaList list);
}
@DataProvider
public static Object[][] sumStack() {
BiFunction<MemoryLayout, MemoryLayout, SumStackFunc> sumStackJavaFact = (longLayout, doubleLayout) ->
(longSum, doubleSum, list) -> {
long lSum = 0L;
for (int i = 0; i < 16; i++) {
lSum += list.vargAsLong(longLayout);
}
MemoryAccess.setLong(longSum, lSum);
double dSum = 0D;
for (int i = 0; i < 16; i++) {
dSum += list.vargAsDouble(doubleLayout);
}
MemoryAccess.setDouble(doubleSum, dSum);
};
SumStackFunc sumStackNative = (longSum, doubleSum, list) -> {
try {
MH_sumStack.invokeExact(longSum.address(), doubleSum.address(), list);
} catch (Throwable ex) {
throw new AssertionError(ex);
}
};
return new Object[][]{
{ winVaListFactory, sumStackJavaFact.apply(Win64.C_LONG_LONG, Win64.C_DOUBLE), Win64.C_LONG_LONG, Win64.C_DOUBLE },
{ sysvVaListFactory, sumStackJavaFact.apply(SysV.C_LONG_LONG, SysV.C_DOUBLE), SysV.C_LONG_LONG, SysV.C_DOUBLE },
{ aarch64VaListFactory, sumStackJavaFact.apply(AArch64.C_LONG_LONG, AArch64.C_DOUBLE), AArch64.C_LONG_LONG, AArch64.C_DOUBLE },
{ platformVaListFactory, sumStackNative, C_LONG_LONG, C_DOUBLE },
};
}
@Test(dataProvider = "sumStack")
public void testStack(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
SumStackFunc sumStack,
ValueLayout longLayout,
ValueLayout doubleLayout) {
try (MemorySegment longSum = MemorySegment.allocateNative(longLayout);
MemorySegment doubleSum = MemorySegment.allocateNative(doubleLayout)) {
MemoryAccess.setLong(longSum, 0L);
MemoryAccess.setDouble(doubleSum, 0D);
VaList list = vaListFactory.apply(b -> {
for (long l = 1; l <= 16L; l++) {
b.vargFromLong(longLayout, l);
}
for (double d = 1; d <= 16D; d++) {
b.vargFromDouble(doubleLayout, d);
}
});
try (list) {
sumStack.invoke(longSum, doubleSum, list);
}
long lSum = MemoryAccess.getLong(longSum);
double dSum = MemoryAccess.getDouble(doubleSum);
assertEquals(lSum, 136L);
assertEquals(dSum, 136D);
}
}
@Test(dataProvider = "upcalls")
public void testUpcall(MethodHandle target, MethodHandle callback) throws Throwable {
FunctionDescriptor desc = FunctionDescriptor.ofVoid(C_VA_LIST);
try (MemorySegment stub = abi.upcallStub(callback, desc)) {
target.invokeExact(stub.address());
}
}
@DataProvider
public Object[][] emptyVaLists() {
return new Object[][] {
{ Windowsx64Linker.emptyVaList() },
{ winVaListFactory.apply(b -> {}) },
{ SysVx64Linker.emptyVaList() },
{ sysvVaListFactory.apply(b -> {}) },
{ AArch64Linker.emptyVaList() },
{ aarch64VaListFactory.apply(b -> {}) },
};
}
@Test(expectedExceptions = UnsupportedOperationException.class,
expectedExceptionsMessageRegExp = ".*Empty VaList.*",
dataProvider = "emptyVaLists")
public void testEmptyNotCloseable(VaList emptyList) {
emptyList.close();
}
@DataProvider
@SuppressWarnings("unchecked")
public static Object[][] sumIntsScoped() {
Function<MemoryLayout, BiFunction<Integer, VaList, Integer>> sumIntsJavaFact = layout ->
(num, list) -> IntStream.generate(() -> list.vargAsInt(layout)).limit(num).sum();
BiFunction<Integer, VaList, Integer> sumIntsNative
= MethodHandleProxies.asInterfaceInstance(BiFunction.class, MH_sumInts);
return new Object[][]{
{ winVaListScopedFactory, sumIntsJavaFact.apply(Win64.C_INT), Win64.C_INT },
{ sysvVaListScopedFactory, sumIntsJavaFact.apply(SysV.C_INT), SysV.C_INT },
{ aarch64VaListScopedFactory, sumIntsJavaFact.apply(AArch64.C_INT), AArch64.C_INT },
{ platformVaListScopedFactory, sumIntsNative, C_INT },
};
}
@Test(dataProvider = "sumIntsScoped")
public void testScopedVaList(BiFunction<Consumer<VaList.Builder>, NativeScope, VaList> vaListFactory,
BiFunction<Integer, VaList, Integer> sumInts,
ValueLayout intLayout) {
VaList listLeaked;
try (NativeScope scope = NativeScope.unboundedScope()) {
VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4)
.vargFromInt(intLayout, 8),
scope);
int x = sumInts.apply(2, list);
assertEquals(x, 12);
listLeaked = list;
}
assertFalse(listLeaked.isAlive());
}
@Test(dataProvider = "structs")
public void testScopeMSRead(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
Function<VaList, Integer> sumStruct, // ignored
GroupLayout Point_LAYOUT, VarHandle VH_Point_x, VarHandle VH_Point_y) {
MemorySegment pointOut;
try (NativeScope scope = NativeScope.unboundedScope()) {
try (MemorySegment pointIn = MemorySegment.allocateNative(Point_LAYOUT)) {
VH_Point_x.set(pointIn, 3);
VH_Point_y.set(pointIn, 6);
try (VaList list = vaListFactory.apply(b -> b.vargFromSegment(Point_LAYOUT, pointIn))) {
pointOut = list.vargAsSegment(Point_LAYOUT, scope);
assertEquals((int) VH_Point_x.get(pointOut), 3);
assertEquals((int) VH_Point_y.get(pointOut), 6);
}
assertTrue(pointOut.isAlive()); // after VaList freed
}
assertTrue(pointOut.isAlive()); // after input MS freed
}
assertFalse(pointOut.isAlive()); // after scope freed
}
@DataProvider
public Object[][] copy() {
return new Object[][] {
{ winVaListFactory, Win64.C_INT },
{ sysvVaListFactory, SysV.C_INT },
{ aarch64VaListFactory, AArch64.C_INT },
};
}
@Test(dataProvider = "copy")
public void testCopy(Function<Consumer<VaList.Builder>, VaList> vaListFactory, ValueLayout intLayout) {
try (VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4)
.vargFromInt(intLayout, 8))) {
VaList copy = list.copy();
assertEquals(copy.vargAsInt(intLayout), 4);
assertEquals(copy.vargAsInt(intLayout), 8);
copy.close();
assertFalse(copy.isAlive());
assertEquals(list.vargAsInt(intLayout), 4);
assertEquals(list.vargAsInt(intLayout), 8);
}
}
@Test(dataProvider = "copy")
public void testScopedCopy(Function<Consumer<VaList.Builder>, VaList> vaListFactory, ValueLayout intLayout) {
try (VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4)
.vargFromInt(intLayout, 8))) {
VaList copy;
try (NativeScope scope = NativeScope.unboundedScope()) {
copy = list.copy(scope);
assertEquals(copy.vargAsInt(intLayout), 4);
assertEquals(copy.vargAsInt(intLayout), 8);
}
assertFalse(copy.isAlive());
assertEquals(list.vargAsInt(intLayout), 4);
assertEquals(list.vargAsInt(intLayout), 8);
}
}
@Test(dataProvider = "copy",
expectedExceptions = IllegalStateException.class)
public void testCopyUnusableAfterOriginalClosed(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
ValueLayout intLayout) {
VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4)
.vargFromInt(intLayout, 8));
try (VaList copy = list.copy()) {
list.close();
copy.vargAsInt(intLayout); // should throw
}
}
@Test(dataProvider = "copy",
expectedExceptions = IllegalStateException.class)
public void testCopyUnusableAfterOriginalClosedScope(Function<Consumer<VaList.Builder>, VaList> vaListFactory,
ValueLayout intLayout) {
VaList list = vaListFactory.apply(b -> b.vargFromInt(intLayout, 4)
.vargFromInt(intLayout, 8));
try (NativeScope scope = NativeScope.unboundedScope()) {
VaList copy = list.copy(scope);
list.close();
copy.vargAsInt(intLayout); // should throw
}
}
@DataProvider
public static Object[][] upcalls() {
GroupLayout BigPoint_LAYOUT = MemoryLayout.ofStruct(
C_LONG_LONG.withName("x"),
C_LONG_LONG.withName("y")
);
VarHandle VH_BigPoint_x = BigPoint_LAYOUT.varHandle(long.class, groupElement("x"));
VarHandle VH_BigPoint_y = BigPoint_LAYOUT.varHandle(long.class, groupElement("y"));
GroupLayout Point_LAYOUT = MemoryLayout.ofStruct(
C_INT.withName("x"),
C_INT.withName("y")
);
VarHandle VH_Point_x = Point_LAYOUT.varHandle(int.class, groupElement("x"));
VarHandle VH_Point_y = Point_LAYOUT.varHandle(int.class, groupElement("y"));
GroupLayout FloatPoint_LAYOUT = MemoryLayout.ofStruct(
C_FLOAT.withName("x"),
C_FLOAT.withName("y")
);
VarHandle VH_FloatPoint_x = FloatPoint_LAYOUT.varHandle(float.class, groupElement("x"));
VarHandle VH_FloatPoint_y = FloatPoint_LAYOUT.varHandle(float.class, groupElement("y"));
GroupLayout HugePoint_LAYOUT = MemoryLayout.ofStruct(
C_LONG_LONG.withName("x"),
C_LONG_LONG.withName("y"),
C_LONG_LONG.withName("z")
);
VarHandle VH_HugePoint_x = HugePoint_LAYOUT.varHandle(long.class, groupElement("x"));
VarHandle VH_HugePoint_y = HugePoint_LAYOUT.varHandle(long.class, groupElement("y"));
VarHandle VH_HugePoint_z = HugePoint_LAYOUT.varHandle(long.class, groupElement("z"));
return new Object[][]{
{ linkVaListCB("upcallBigStruct"), VaListConsumer.mh(vaList -> {
try (MemorySegment struct = vaList.vargAsSegment(BigPoint_LAYOUT)) {
assertEquals((long) VH_BigPoint_x.get(struct), 8);
assertEquals((long) VH_BigPoint_y.get(struct), 16);
}
})},
{ linkVaListCB("upcallBigStruct"), VaListConsumer.mh(vaList -> {
VaList copy = vaList.copy();
try (MemorySegment struct = vaList.vargAsSegment(BigPoint_LAYOUT)) {
assertEquals((long) VH_BigPoint_x.get(struct), 8);
assertEquals((long) VH_BigPoint_y.get(struct), 16);
VH_BigPoint_x.set(struct, 0);
VH_BigPoint_y.set(struct, 0);
}
// should be independent
try (MemorySegment struct = copy.vargAsSegment(BigPoint_LAYOUT)) {
assertEquals((long) VH_BigPoint_x.get(struct), 8);
assertEquals((long) VH_BigPoint_y.get(struct), 16);
}
})},
{ linkVaListCB("upcallStruct"), VaListConsumer.mh(vaList -> {
try (MemorySegment struct = vaList.vargAsSegment(Point_LAYOUT)) {
assertEquals((int) VH_Point_x.get(struct), 5);
assertEquals((int) VH_Point_y.get(struct), 10);
}
})},
{ linkVaListCB("upcallHugeStruct"), VaListConsumer.mh(vaList -> {
try (MemorySegment struct = vaList.vargAsSegment(HugePoint_LAYOUT)) {
assertEquals((long) VH_HugePoint_x.get(struct), 1);
assertEquals((long) VH_HugePoint_y.get(struct), 2);
assertEquals((long) VH_HugePoint_z.get(struct), 3);
}
})},
{ linkVaListCB("upcallFloatStruct"), VaListConsumer.mh(vaList -> {
try (MemorySegment struct = vaList.vargAsSegment(FloatPoint_LAYOUT)) {
assertEquals((float) VH_FloatPoint_x.get(struct), 1.0f);
assertEquals((float) VH_FloatPoint_y.get(struct), 2.0f);
}
})},
{ linkVaListCB("upcallMemoryAddress"), VaListConsumer.mh(vaList -> {
MemoryAddress intPtr = vaList.vargAsAddress(C_POINTER);
MemorySegment ms = intPtr.asSegmentRestricted(C_INT.byteSize());
int x = MemoryAccess.getInt(ms);
assertEquals(x, 10);
})},
{ linkVaListCB("upcallDoubles"), VaListConsumer.mh(vaList -> {
assertEquals(vaList.vargAsDouble(C_DOUBLE), 3.0);
assertEquals(vaList.vargAsDouble(C_DOUBLE), 4.0);
assertEquals(vaList.vargAsDouble(C_DOUBLE), 5.0);
})},
{ linkVaListCB("upcallInts"), VaListConsumer.mh(vaList -> {
assertEquals(vaList.vargAsInt(C_INT), 10);
assertEquals(vaList.vargAsInt(C_INT), 15);
assertEquals(vaList.vargAsInt(C_INT), 20);
})},
{ linkVaListCB("upcallStack"), VaListConsumer.mh(vaList -> {
// skip all registers
for (long l = 1; l <= 16; l++) {
assertEquals(vaList.vargAsLong(C_LONG_LONG), l);
}
for (double d = 1; d <= 16; d++) {
assertEquals(vaList.vargAsDouble(C_DOUBLE), d);
}
// test some arbitrary values on the stack
assertEquals((byte) vaList.vargAsInt(C_INT), (byte) 1);
assertEquals((char) vaList.vargAsInt(C_INT), 'a');
assertEquals((short) vaList.vargAsInt(C_INT), (short) 3);
assertEquals(vaList.vargAsInt(C_INT), 4);
assertEquals(vaList.vargAsLong(C_LONG_LONG), 5L);
assertEquals((float) vaList.vargAsDouble(C_DOUBLE), 6.0F);
assertEquals(vaList.vargAsDouble(C_DOUBLE), 7.0D);
assertEquals((byte) vaList.vargAsInt(C_INT), (byte) 8);
assertEquals((char) vaList.vargAsInt(C_INT), 'b');
assertEquals((short) vaList.vargAsInt(C_INT), (short) 10);
assertEquals(vaList.vargAsInt(C_INT), 11);
assertEquals(vaList.vargAsLong(C_LONG_LONG), 12L);
assertEquals((float) vaList.vargAsDouble(C_DOUBLE), 13.0F);
assertEquals(vaList.vargAsDouble(C_DOUBLE), 14.0D);
try (MemorySegment point = vaList.vargAsSegment(Point_LAYOUT)) {
assertEquals((int) VH_Point_x.get(point), 5);
assertEquals((int) VH_Point_y.get(point), 10);
}
VaList copy = vaList.copy();
try (MemorySegment bigPoint = vaList.vargAsSegment(BigPoint_LAYOUT)) {
assertEquals((long) VH_BigPoint_x.get(bigPoint), 15);
assertEquals((long) VH_BigPoint_y.get(bigPoint), 20);
VH_BigPoint_x.set(bigPoint, 0);
VH_BigPoint_y.set(bigPoint, 0);
}
// should be independent
try (MemorySegment struct = copy.vargAsSegment(BigPoint_LAYOUT)) {
assertEquals((long) VH_BigPoint_x.get(struct), 15);
assertEquals((long) VH_BigPoint_y.get(struct), 20);
}
})},
// test skip
{ linkVaListCB("upcallStack"), VaListConsumer.mh(vaList -> {
vaList.skip(C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG);
assertEquals(vaList.vargAsLong(C_LONG_LONG), 5L);
vaList.skip(C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG);
assertEquals(vaList.vargAsLong(C_LONG_LONG), 10L);
vaList.skip(C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG, C_LONG_LONG);
assertEquals(vaList.vargAsDouble(C_DOUBLE), 1.0D);
vaList.skip(C_DOUBLE, C_DOUBLE, C_DOUBLE, C_DOUBLE);
assertEquals(vaList.vargAsDouble(C_DOUBLE), 6.0D);
})},
};
}
interface VaListConsumer {
void accept(VaList list);
static MethodHandle mh(VaListConsumer instance) {
try {
return MethodHandles.lookup().findVirtual(VaListConsumer.class, "accept",
MethodType.methodType(void.class, VaList.class)).bindTo(instance);
} catch (ReflectiveOperationException e) {
throw new InternalError(e);
}
}
}
}