package jdk.internal.foreign.abi.x64.sysv;
import jdk.incubator.foreign.Addressable;
import jdk.incubator.foreign.FunctionDescriptor;
import jdk.incubator.foreign.MemoryAddress;
import jdk.incubator.foreign.MemoryLayout;
import jdk.incubator.foreign.MemorySegment;
import jdk.incubator.foreign.CLinker;
import jdk.internal.foreign.abi.SharedUtils;
import jdk.internal.foreign.abi.UpcallStubs;
import java.lang.invoke.MethodHandle;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodType;
import java.util.Objects;
import java.util.Optional;
import java.util.function.Consumer;
import static jdk.internal.foreign.PlatformLayouts.*;
public class SysVx64Linker implements CLinker {
public static final int MAX_INTEGER_ARGUMENT_REGISTERS = 6;
public static final int MAX_INTEGER_RETURN_REGISTERS = 2;
public static final int MAX_VECTOR_ARGUMENT_REGISTERS = 8;
public static final int MAX_VECTOR_RETURN_REGISTERS = 2;
public static final int MAX_X87_RETURN_REGISTERS = 2;
private static SysVx64Linker instance;
static final long ADDRESS_SIZE = 64;
private static final MethodHandle MH_unboxVaList;
private static final MethodHandle MH_boxVaList;
static {
try {
MethodHandles.Lookup lookup = MethodHandles.lookup();
MH_unboxVaList = lookup.findVirtual(VaList.class, "address",
MethodType.methodType(MemoryAddress.class));
MH_boxVaList = lookup.findStatic(SysVx64Linker.class, "newVaListOfAddress",
MethodType.methodType(VaList.class, MemoryAddress.class));
} catch (ReflectiveOperationException e) {
throw new ExceptionInInitializerError(e);
}
}
public static SysVx64Linker getInstance() {
if (instance == null) {
instance = new SysVx64Linker();
}
return instance;
}
public static VaList newVaList(Consumer<VaList.Builder> actions, SharedUtils.Allocator allocator) {
SysVVaList.Builder builder = SysVVaList.builder(allocator);
actions.accept(builder);
return builder.build();
}
@Override
public MethodHandle downcallHandle(Addressable symbol, MethodType type, FunctionDescriptor function) {
Objects.requireNonNull(symbol);
Objects.requireNonNull(type);
Objects.requireNonNull(function);
MethodType llMt = SharedUtils.convertVaListCarriers(type, SysVVaList.CARRIER);
MethodHandle handle = CallArranger.arrangeDowncall(symbol, llMt, function);
handle = SharedUtils.unboxVaLists(type, handle, MH_unboxVaList);
return handle;
}
@Override
public MemorySegment upcallStub(MethodHandle target, FunctionDescriptor function) {
Objects.requireNonNull(target);
Objects.requireNonNull(function);
target = SharedUtils.boxVaLists(target, MH_boxVaList);
return UpcallStubs.upcallAddress(CallArranger.arrangeUpcall(target, target.type(), function));
}
public static VaList newVaListOfAddress(MemoryAddress ma) {
return SysVVaList.ofAddress(ma);
}
public static VaList emptyVaList() {
return SysVVaList.empty();
}
}