8f0eb5d401
Reviewed-by: mcimadamore, ngasson
281 lines
9.3 KiB
Java
281 lines
9.3 KiB
Java
/*
|
|
* 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.Addressable;
|
|
import java.lang.foreign.Linker;
|
|
import java.lang.foreign.FunctionDescriptor;
|
|
import java.lang.foreign.MemoryAddress;
|
|
import java.lang.foreign.MemoryLayout;
|
|
import java.lang.foreign.MemorySegment;
|
|
import java.lang.foreign.MemorySession;
|
|
|
|
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, MemoryAddress.class, List.class));
|
|
} catch (ReflectiveOperationException e) {
|
|
throw new ExceptionInInitializerError(e);
|
|
}
|
|
}
|
|
|
|
static final Addressable VARARGS_ADDR = findNativeOrThrow("varargs");
|
|
|
|
@Test(dataProvider = "functions")
|
|
public void testVarArgs(int count, String fName, Ret ret, // ignore this stuff
|
|
List<ParamType> paramTypes, List<StructFieldType> fields) throws Throwable {
|
|
List<Arg> args = makeArgs(paramTypes, fields);
|
|
|
|
try (MemorySession session = MemorySession.openConfined()) {
|
|
MethodHandle checker = MethodHandles.insertArguments(MH_CHECK, 2, args);
|
|
MemorySegment writeBack = LINKER.upcallStub(checker, FunctionDescriptor.ofVoid(C_INT, C_POINTER), session);
|
|
MemorySegment callInfo = MemorySegment.allocateNative(CallInfo.LAYOUT, session);
|
|
MemorySegment argIDs = MemorySegment.allocateNative(MemoryLayout.sequenceLayout(args.size(), C_INT), session);
|
|
|
|
MemoryAddress callInfoPtr = callInfo.address();
|
|
|
|
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<MemoryLayout> argLayouts = new ArrayList<>();
|
|
argLayouts.add(C_POINTER); // call info
|
|
argLayouts.add(C_INT); // size
|
|
|
|
FunctionDescriptor desc = FunctionDescriptor.ofVoid(argLayouts.toArray(MemoryLayout[]::new))
|
|
.asVariadic(args.stream().map(a -> a.layout).toArray(MemoryLayout[]::new));
|
|
|
|
MethodHandle downcallHandle = LINKER.downcallHandle(VARARGS_ADDR, desc);
|
|
|
|
List<Object> 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<Arg> makeArgs(List<ParamType> paramTypes, List<StructFieldType> fields) throws ReflectiveOperationException {
|
|
List<Arg> args = new ArrayList<>();
|
|
for (ParamType pType : paramTypes) {
|
|
MemoryLayout layout = pType.layout(fields);
|
|
List<Consumer<Object>> 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, MemoryAddress ptr, List<Arg> args) {
|
|
Arg varArg = args.get(index);
|
|
MemoryLayout layout = varArg.layout;
|
|
MethodHandle getter = varArg.getter;
|
|
List<Consumer<Object>> checks = varArg.checks;
|
|
try (MemorySession session = MemorySession.openConfined()) {
|
|
MemorySegment seg = MemorySegment.ofAddress(ptr, layout.byteSize(), session);
|
|
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, Addressable addr) {
|
|
VH_writeback.set(seg, addr.address());
|
|
}
|
|
static void argIDs(MemorySegment seg, Addressable addr) {
|
|
VH_argIDs.set(seg, addr.address());
|
|
}
|
|
}
|
|
|
|
private static final class Arg {
|
|
final NativeType id;
|
|
final MemoryLayout layout;
|
|
final Object value;
|
|
final MethodHandle getter;
|
|
final List<Consumer<Object>> checks;
|
|
|
|
private Arg(NativeType id, MemoryLayout layout, Object value, MethodHandle getter, List<Consumer<Object>> 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<Consumer<Object>> 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<Consumer<Object>> 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());
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|