/* * Copyright (c) 2020, 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 * @requires ((os.arch == "amd64" | os.arch == "x86_64") & sun.arch.data.model == "64") | os.arch == "aarch64" * @run testng/othervm --enable-native-access=ALL-UNNAMED -Dgenerator.sample.factor=17 TestVarArgs */ import java.lang.foreign.Arena; import java.lang.foreign.Linker; import java.lang.foreign.FunctionDescriptor; import java.lang.foreign.MemoryLayout; import java.lang.foreign.MemorySegment; import org.testng.annotations.Test; import java.lang.invoke.MethodHandle; import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; import java.lang.invoke.VarHandle; import java.util.ArrayList; import java.util.List; import java.util.function.Consumer; import static java.lang.foreign.MemoryLayout.PathElement.*; public class TestVarArgs extends CallGeneratorHelper { static final VarHandle VH_IntArray = C_INT.arrayElementVarHandle(); static final MethodHandle MH_CHECK; static final Linker LINKER = Linker.nativeLinker(); static { System.loadLibrary("VarArgs"); try { MH_CHECK = MethodHandles.lookup().findStatic(TestVarArgs.class, "check", MethodType.methodType(void.class, int.class, MemorySegment.class, List.class)); } catch (ReflectiveOperationException e) { throw new ExceptionInInitializerError(e); } } static final MemorySegment VARARGS_ADDR = findNativeOrThrow("varargs"); @Test(dataProvider = "functions") public void testVarArgs(int count, String fName, Ret ret, // ignore this stuff List paramTypes, List fields) throws Throwable { List args = makeArgs(paramTypes, fields); try (Arena arena = Arena.openConfined()) { MethodHandle checker = MethodHandles.insertArguments(MH_CHECK, 2, args); MemorySegment writeBack = LINKER.upcallStub(checker, FunctionDescriptor.ofVoid(C_INT, C_POINTER), arena.scope()); MemorySegment callInfo = MemorySegment.allocateNative(CallInfo.LAYOUT, arena.scope());; MemorySegment argIDs = MemorySegment.allocateNative(MemoryLayout.sequenceLayout(args.size(), C_INT), arena.scope());; MemorySegment callInfoPtr = callInfo; CallInfo.writeback(callInfo, writeBack); CallInfo.argIDs(callInfo, argIDs); for (int i = 0; i < args.size(); i++) { VH_IntArray.set(argIDs, (long) i, args.get(i).id.ordinal()); } List argLayouts = new ArrayList<>(); argLayouts.add(C_POINTER); // call info argLayouts.add(C_INT); // size FunctionDescriptor baseDesc = FunctionDescriptor.ofVoid(argLayouts.toArray(MemoryLayout[]::new)); Linker.Option varargIndex = Linker.Option.firstVariadicArg(baseDesc.argumentLayouts().size()); FunctionDescriptor desc = baseDesc.appendArgumentLayouts(args.stream().map(a -> a.layout).toArray(MemoryLayout[]::new)); MethodHandle downcallHandle = LINKER.downcallHandle(VARARGS_ADDR, desc, varargIndex); List argValues = new ArrayList<>(); argValues.add(callInfoPtr); // call info argValues.add(args.size()); // size args.forEach(a -> argValues.add(a.value)); downcallHandle.invokeWithArguments(argValues); // args checked by upcall } } private static List makeArgs(List paramTypes, List fields) throws ReflectiveOperationException { List args = new ArrayList<>(); for (ParamType pType : paramTypes) { MemoryLayout layout = pType.layout(fields); List> checks = new ArrayList<>(); Object arg = makeArg(layout, checks, true); Arg.NativeType type = Arg.NativeType.of(pType.type(fields)); args.add(pType == ParamType.STRUCT ? Arg.structArg(type, layout, arg, checks) : Arg.primitiveArg(type, layout, arg, checks)); } return args; } private static void check(int index, MemorySegment ptr, List args) { Arg varArg = args.get(index); MemoryLayout layout = varArg.layout; MethodHandle getter = varArg.getter; List> checks = varArg.checks; try (Arena arena = Arena.openConfined()) { MemorySegment seg = MemorySegment.ofAddress(ptr.address(), layout.byteSize(), arena.scope()); Object obj = getter.invoke(seg); checks.forEach(check -> check.accept(obj)); } catch (Throwable e) { throw new RuntimeException(e); } } private static class CallInfo { static final MemoryLayout LAYOUT = MemoryLayout.structLayout( C_POINTER.withName("writeback"), // writeback C_POINTER.withName("argIDs")); // arg ids static final VarHandle VH_writeback = LAYOUT.varHandle(groupElement("writeback")); static final VarHandle VH_argIDs = LAYOUT.varHandle(groupElement("argIDs")); static void writeback(MemorySegment seg, MemorySegment addr) { VH_writeback.set(seg, addr); } static void argIDs(MemorySegment seg, MemorySegment addr) { VH_argIDs.set(seg, addr); } } private static final class Arg { final NativeType id; final MemoryLayout layout; final Object value; final MethodHandle getter; final List> checks; private Arg(NativeType id, MemoryLayout layout, Object value, MethodHandle getter, List> checks) { this.id = id; this.layout = layout; this.value = value; this.getter = getter; this.checks = checks; } private static Arg primitiveArg(NativeType id, MemoryLayout layout, Object value, List> checks) { return new Arg(id, layout, value, layout.varHandle().toMethodHandle(VarHandle.AccessMode.GET), checks); } private static Arg structArg(NativeType id, MemoryLayout layout, Object value, List> checks) { return new Arg(id, layout, value, MethodHandles.identity(MemorySegment.class), checks); } enum NativeType { INT, FLOAT, DOUBLE, POINTER, S_I, S_F, S_D, S_P, S_II, S_IF, S_ID, S_IP, S_FI, S_FF, S_FD, S_FP, S_DI, S_DF, S_DD, S_DP, S_PI, S_PF, S_PD, S_PP, S_III, S_IIF, S_IID, S_IIP, S_IFI, S_IFF, S_IFD, S_IFP, S_IDI, S_IDF, S_IDD, S_IDP, S_IPI, S_IPF, S_IPD, S_IPP, S_FII, S_FIF, S_FID, S_FIP, S_FFI, S_FFF, S_FFD, S_FFP, S_FDI, S_FDF, S_FDD, S_FDP, S_FPI, S_FPF, S_FPD, S_FPP, S_DII, S_DIF, S_DID, S_DIP, S_DFI, S_DFF, S_DFD, S_DFP, S_DDI, S_DDF, S_DDD, S_DDP, S_DPI, S_DPF, S_DPD, S_DPP, S_PII, S_PIF, S_PID, S_PIP, S_PFI, S_PFF, S_PFD, S_PFP, S_PDI, S_PDF, S_PDD, S_PDP, S_PPI, S_PPF, S_PPD, S_PPP, ; public static NativeType of(String type) { return NativeType.valueOf(switch (type) { case "int" -> "INT"; case "float" -> "FLOAT"; case "double" -> "DOUBLE"; case "void*" -> "POINTER"; default -> type.substring("struct ".length()); }); } } } }