/* * Copyright (c) 2017, 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. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * 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. */ package com.oracle.truffle.nfi; import com.oracle.truffle.api.CallTarget; import com.oracle.truffle.api.CompilerDirectives; import com.oracle.truffle.api.Truffle; import com.oracle.truffle.api.dsl.Cached; import com.oracle.truffle.api.dsl.Specialization; import com.oracle.truffle.api.frame.VirtualFrame; import com.oracle.truffle.api.interop.ForeignAccess; import com.oracle.truffle.api.interop.InteropException; import com.oracle.truffle.api.interop.Message; import com.oracle.truffle.api.interop.TruffleObject; import com.oracle.truffle.api.interop.UnsupportedMessageException; import com.oracle.truffle.api.interop.UnsupportedTypeException; import com.oracle.truffle.api.nodes.ExplodeLoop; import com.oracle.truffle.api.nodes.Node; import com.oracle.truffle.api.nodes.RootNode; import com.oracle.truffle.nfi.LibFFIClosureFactory.UnboxStringNodeGen; import java.nio.ByteBuffer; final class LibFFIClosure { final ClosureNativePointer nativePointer; static LibFFIClosure create(LibFFISignature signature, TruffleObject executable) { LibFFIClosure ret = new LibFFIClosure(signature, executable); NativeAllocation.registerNativeAllocation(ret, ret.nativePointer); return ret; } private LibFFIClosure(LibFFISignature signature, TruffleObject executable) { Message message = Message.createExecute(signature.getArgTypes().length); LibFFIType retType = signature.getRetType(); if (retType instanceof LibFFIType.ObjectType) { // shortcut for simple object return values CallTarget executeCallTarget = Truffle.getRuntime().createCallTarget(new ObjectRetClosureRootNode(signature, executable, message)); this.nativePointer = allocateClosureObjectRet(signature, executeCallTarget); } else if (retType instanceof LibFFIType.StringType) { // shortcut for simple string return values CallTarget executeCallTarget = Truffle.getRuntime().createCallTarget(new StringRetClosureRootNode(signature, executable, message)); this.nativePointer = allocateClosureStringRet(signature, executeCallTarget); } else if (retType instanceof LibFFIType.VoidType) { // special handling for no return value CallTarget executeCallTarget = Truffle.getRuntime().createCallTarget(new ObjectRetClosureRootNode(signature, executable, message)); this.nativePointer = allocateClosureVoidRet(signature, executeCallTarget); } else { // generic case: last argument is the return buffer CallTarget executeCallTarget = Truffle.getRuntime().createCallTarget(new BufferRetClosureRootNode(signature, executable, message)); this.nativePointer = allocateClosureBufferRet(signature, executeCallTarget); } } static final class RetPatches { final int count; final int[] patches; final Object[] objects; RetPatches(int count, int[] patches, Object[] objects) { this.count = count; this.patches = patches; this.objects = objects; } } private static final class CallClosureNode extends Node { private final TruffleObject receiver; @Child Node messageNode; @Children final ClosureArgumentNode[] argNodes; private CallClosureNode(LibFFISignature signature, TruffleObject receiver, Message message) { this.receiver = receiver; this.messageNode = message.createNode(); LibFFIType[] args = signature.getArgTypes(); argNodes = new ClosureArgumentNode[args.length]; for (int i = 0; i < args.length; i++) { argNodes[i] = args[i].createClosureArgumentNode(); } } @ExplodeLoop Object execute(Object[] argBuffers) { Object[] args = new Object[argNodes.length]; for (int i = 0; i < argNodes.length; i++) { args[i] = argNodes[i].execute(argBuffers[i]); } try { return ForeignAccess.send(messageNode, receiver, args); } catch (InteropException ex) { CompilerDirectives.transferToInterpreter(); throw new IllegalStateException(ex); } } } private static final class EncodeRetNode extends Node { @Child SerializeArgumentNode retNode; private final int retObjectCount; private EncodeRetNode(LibFFIType retType) { retNode = retType.createClosureReturnNode(); retObjectCount = retType.objectCount; } RetPatches execute(Object ret, ByteBuffer retBuffer) { NativeArgumentBuffer nativeRetBuffer = new NativeArgumentBuffer.Direct(retBuffer, retObjectCount); retNode.execute(nativeRetBuffer, ret); if (nativeRetBuffer.getPatchCount() > 0) { return new RetPatches(nativeRetBuffer.getPatchCount(), nativeRetBuffer.patches, nativeRetBuffer.objects); } else { return null; } } } private static final class BufferRetClosureRootNode extends RootNode { @Child CallClosureNode callClosure; @Child EncodeRetNode encodeRet; private BufferRetClosureRootNode(LibFFISignature signature, TruffleObject receiver, Message message) { super(null); callClosure = new CallClosureNode(signature, receiver, message); encodeRet = new EncodeRetNode(signature.getRetType()); } @Override public Object execute(VirtualFrame frame) { ByteBuffer retBuffer = (ByteBuffer) frame.getArguments()[frame.getArguments().length - 1]; Object ret = callClosure.execute(frame.getArguments()); return encodeRet.execute(ret, retBuffer); } } private static final class ObjectRetClosureRootNode extends RootNode { @Child CallClosureNode callClosure; private ObjectRetClosureRootNode(LibFFISignature signature, TruffleObject receiver, Message message) { super(null); callClosure = new CallClosureNode(signature, receiver, message); } @Override public Object execute(VirtualFrame frame) { return callClosure.execute(frame.getArguments()); } } abstract static class UnboxStringNode extends Node { protected abstract Object execute(Object obj); @Specialization protected Object nativeString(NativeString str) { return str; } @SuppressWarnings("unused") @Specialization(guards = "checkIsNull(isNull, str)") protected Object unboxNull(TruffleObject str, @Cached("createIsNull()") Node isNull) { return null; } @Specialization protected Object javaString(String obj) { return obj; } @Specialization(guards = "checkNeedUnbox(str)") protected Object unboxBoxed(TruffleObject str, @Cached("createUnbox()") Node unbox, @Cached("createRecursive()") UnboxStringNode recursive) { try { Object unboxed = ForeignAccess.sendUnbox(unbox, str); return recursive.execute(unboxed); } catch (UnsupportedMessageException ex) { CompilerDirectives.transferToInterpreter(); throw UnsupportedTypeException.raise(ex, new Object[]{str}); } } protected static Node createIsNull() { return Message.IS_NULL.createNode(); } protected static boolean checkIsNull(Node isNullNode, TruffleObject obj) { return ForeignAccess.sendIsNull(isNullNode, obj); } protected static boolean checkNeedUnbox(TruffleObject obj) { return !(obj instanceof NativeString); } protected static Node createUnbox() { return Message.UNBOX.createNode(); } protected static UnboxStringNode createRecursive() { return UnboxStringNodeGen.create(); } } private static final class StringRetClosureRootNode extends RootNode { @Child CallClosureNode callClosure; @Child UnboxStringNode unboxString; private StringRetClosureRootNode(LibFFISignature signature, TruffleObject receiver, Message message) { super(null); callClosure = new CallClosureNode(signature, receiver, message); unboxString = UnboxStringNodeGen.create(); } @Override public Object execute(VirtualFrame frame) { Object ret = callClosure.execute(frame.getArguments()); return unboxString.execute(ret); } } private static native ClosureNativePointer allocateClosureObjectRet(LibFFISignature signature, CallTarget callTarget); private static native ClosureNativePointer allocateClosureStringRet(LibFFISignature signature, CallTarget callTarget); private static native ClosureNativePointer allocateClosureBufferRet(LibFFISignature signature, CallTarget callTarget); private static native ClosureNativePointer allocateClosureVoidRet(LibFFISignature signature, CallTarget callTarget); }