From e1c172b4e8c2d4c5bb265096153af7ecb1a643c9 Mon Sep 17 00:00:00 2001
From: Mick Jordan <mick.jordan@oracle.com>
Date: Wed, 1 Feb 2017 14:11:41 -0800
Subject: [PATCH] rffi: refactor BaseRFFI into one node per function

---
 .../truffle/r/library/tools/DirChmod.java     |   8 +-
 .../r/nodes/builtin/base/FileFunctions.java   |  20 +-
 .../truffle/r/nodes/builtin/base/Getwd.java   |   8 +-
 .../truffle/r/nodes/builtin/base/Setwd.java   |  15 +-
 .../r/nodes/builtin/base/SysFunctions.java    |  30 +-
 .../truffle/r/runtime/ffi/jni/JNI_Base.java   | 208 ++++++++-----
 .../oracle/truffle/r/runtime/REnvVars.java    |   4 +-
 .../oracle/truffle/r/runtime/RProfile.java    |   6 +-
 .../com/oracle/truffle/r/runtime/RSrcref.java |   3 +-
 .../truffle/r/runtime/RVersionInfo.java       |   4 +-
 .../truffle/r/runtime/TempPathName.java       |   4 +-
 .../com/oracle/truffle/r/runtime/Utils.java   |   3 +-
 .../truffle/r/runtime/ffi/BaseRFFI.java       | 279 +++++++++++++++---
 .../oracle/truffle/r/runtime/rng/RRNG.java    |   4 +-
 14 files changed, 437 insertions(+), 159 deletions(-)

diff --git a/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/tools/DirChmod.java b/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/tools/DirChmod.java
index 0bee29b13c..cf1869d34a 100644
--- a/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/tools/DirChmod.java
+++ b/com.oracle.truffle.r.library/src/com/oracle/truffle/r/library/tools/DirChmod.java
@@ -24,6 +24,7 @@ import java.util.Iterator;
 import java.util.stream.Stream;
 
 import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
+import com.oracle.truffle.api.dsl.Cached;
 import com.oracle.truffle.api.dsl.Fallback;
 import com.oracle.truffle.api.dsl.Specialization;
 import com.oracle.truffle.r.nodes.builtin.CastBuilder;
@@ -32,7 +33,7 @@ import com.oracle.truffle.r.runtime.RError;
 import com.oracle.truffle.r.runtime.RRuntime;
 import com.oracle.truffle.r.runtime.Utils;
 import com.oracle.truffle.r.runtime.data.RNull;
-import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 
 public abstract class DirChmod extends RExternalBuiltinNode.Arg2 {
 
@@ -49,7 +50,8 @@ public abstract class DirChmod extends RExternalBuiltinNode.Arg2 {
 
     @Specialization
     @TruffleBoundary
-    protected RNull dirChmod(String pathName, boolean setGroupWrite) {
+    protected RNull dirChmod(String pathName, boolean setGroupWrite,
+                    @Cached("create()") BaseRFFI.ChmodNode chmodNode) {
         Path path = FileSystems.getDefault().getPath(pathName);
         int fileMask = setGroupWrite ? GRPWRITE_FILE_MASK : FILE_MASK;
         int dirMask = setGroupWrite ? GRPWRITE_DIR_MASK : DIR_MASK;
@@ -65,7 +67,7 @@ public abstract class DirChmod extends RExternalBuiltinNode.Arg2 {
                 int elementMode = Utils.intFilePermissions(pfa.permissions());
                 int newMode = Files.isDirectory(element) ? elementMode | dirMask : elementMode | fileMask;
                 // System.out.printf("path %s: old %o, new %o%n", element, elementMode, newMode);
-                RFFIFactory.getRFFI().getBaseRFFI().chmod(element.toString(), newMode);
+                chmodNode.chmod(element.toString(), newMode);
             }
         } catch (IOException ex) {
             // ignore
diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/FileFunctions.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/FileFunctions.java
index 3dd5c7dd8d..47711c0552 100644
--- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/FileFunctions.java
+++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/FileFunctions.java
@@ -56,6 +56,7 @@ import java.util.stream.Stream;
 
 import com.oracle.truffle.api.CompilerDirectives;
 import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
+import com.oracle.truffle.api.dsl.Cached;
 import com.oracle.truffle.api.dsl.Specialization;
 import com.oracle.truffle.r.nodes.attributes.SpecialAttributesFunctions.SetClassAttributeNode;
 import com.oracle.truffle.r.nodes.builtin.CastBuilder;
@@ -80,7 +81,7 @@ import com.oracle.truffle.r.runtime.data.RSymbol;
 import com.oracle.truffle.r.runtime.data.RTypedValue;
 import com.oracle.truffle.r.runtime.data.model.RAbstractContainer;
 import com.oracle.truffle.r.runtime.data.model.RAbstractStringVector;
-import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 
 // Much of this code was influences/transcribed from GnuR src/main/platform.c
 
@@ -1184,7 +1185,8 @@ public class FileFunctions {
 
         @Specialization
         @TruffleBoundary
-        protected byte dirCreate(String pathIn, boolean showWarnings, boolean recursive, int octMode) {
+        protected byte dirCreate(String pathIn, boolean showWarnings, boolean recursive, int octMode,
+                        @Cached("create()") BaseRFFI.MkdirNode mkdirNode) {
             boolean ok;
             if (RRuntime.isNA(pathIn)) {
                 ok = false;
@@ -1192,32 +1194,32 @@ public class FileFunctions {
                 ok = true;
                 String path = Utils.tildeExpand(pathIn);
                 if (recursive) {
-                    ok = mkparentdirs(new File(path).getAbsoluteFile().getParentFile(), showWarnings, octMode);
+                    ok = mkparentdirs(mkdirNode, new File(path).getAbsoluteFile().getParentFile(), showWarnings, octMode);
                 }
                 if (ok) {
-                    ok = mkdir(path, showWarnings, octMode);
+                    ok = mkdir(mkdirNode, path, showWarnings, octMode);
                 }
             }
             return RRuntime.asLogical(ok);
         }
 
-        private boolean mkparentdirs(File file, boolean showWarnings, int mode) {
+        private boolean mkparentdirs(BaseRFFI.MkdirNode mkdirNode, File file, boolean showWarnings, int mode) {
             if (file.isDirectory()) {
                 return true;
             }
             if (file.exists()) {
                 return false;
             }
-            if (mkparentdirs(file.getParentFile(), showWarnings, mode)) {
-                return mkdir(file.getAbsolutePath(), showWarnings, mode);
+            if (mkparentdirs(mkdirNode, file.getParentFile(), showWarnings, mode)) {
+                return mkdir(mkdirNode, file.getAbsolutePath(), showWarnings, mode);
             } else {
                 return false;
             }
         }
 
-        private boolean mkdir(String path, boolean showWarnings, int mode) {
+        private boolean mkdir(BaseRFFI.MkdirNode mkdirNode, String path, boolean showWarnings, int mode) {
             try {
-                RFFIFactory.getRFFI().getBaseRFFI().mkdir(path, mode);
+                mkdirNode.mkdir(path, mode);
                 return true;
             } catch (IOException ex) {
                 if (showWarnings) {
diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Getwd.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Getwd.java
index d083355708..6476105494 100644
--- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Getwd.java
+++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Getwd.java
@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 2014, 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
@@ -30,15 +30,17 @@ import com.oracle.truffle.api.dsl.Specialization;
 import com.oracle.truffle.r.nodes.builtin.RBuiltinNode;
 import com.oracle.truffle.r.runtime.builtins.RBuiltin;
 import com.oracle.truffle.r.runtime.data.RDataFactory;
-import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 
 @RBuiltin(name = "getwd", kind = INTERNAL, parameterNames = {}, behavior = IO)
 public abstract class Getwd extends RBuiltinNode {
 
+    @Child private BaseRFFI.GetwdNode getwdNode = BaseRFFI.GetwdNode.create();
+
     @Specialization
     @TruffleBoundary
     protected Object getwd() {
-        String result = RFFIFactory.getRFFI().getBaseRFFI().getwd();
+        String result = getwdNode.getwd();
         return RDataFactory.createStringVector(result);
     }
 }
diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Setwd.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Setwd.java
index 9180e76850..af7c7482e0 100644
--- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Setwd.java
+++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Setwd.java
@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 2014, 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
@@ -31,13 +31,14 @@ import static com.oracle.truffle.r.runtime.builtins.RBehavior.IO;
 import static com.oracle.truffle.r.runtime.builtins.RBuiltinKind.INTERNAL;
 
 import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
+import com.oracle.truffle.api.dsl.Cached;
 import com.oracle.truffle.api.dsl.Specialization;
 import com.oracle.truffle.r.nodes.builtin.CastBuilder;
 import com.oracle.truffle.r.nodes.builtin.RBuiltinNode;
 import com.oracle.truffle.r.runtime.RError;
 import com.oracle.truffle.r.runtime.Utils;
 import com.oracle.truffle.r.runtime.builtins.RBuiltin;
-import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 
 @RBuiltin(name = "setwd", visibility = OFF, kind = INTERNAL, parameterNames = "path", behavior = IO)
 public abstract class Setwd extends RBuiltinNode {
@@ -49,14 +50,16 @@ public abstract class Setwd extends RBuiltinNode {
 
     @Specialization
     @TruffleBoundary
-    protected Object setwd(String path) {
-        String owd = RFFIFactory.getRFFI().getBaseRFFI().getwd();
+    protected Object setwd(String path,
+                    @Cached("create()") BaseRFFI.GetwdNode getwdNode,
+                    @Cached("create()") BaseRFFI.SetwdNode setwdNode) {
+        String owd = getwdNode.getwd();
         String nwd = Utils.tildeExpand(path);
-        int rc = RFFIFactory.getRFFI().getBaseRFFI().setwd(nwd);
+        int rc = setwdNode.setwd(nwd);
         if (rc != 0) {
             throw RError.error(this, RError.Message.CANNOT_CHANGE_DIRECTORY);
         } else {
-            String nwdAbs = RFFIFactory.getRFFI().getBaseRFFI().getwd();
+            String nwdAbs = getwdNode.getwd();
             Utils.updateCurwd(nwdAbs);
             return owd;
         }
diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/SysFunctions.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/SysFunctions.java
index e62c27d3e4..8796f70527 100644
--- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/SysFunctions.java
+++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/SysFunctions.java
@@ -45,6 +45,7 @@ import java.util.Map;
 import java.util.concurrent.TimeUnit;
 
 import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
+import com.oracle.truffle.api.dsl.Cached;
 import com.oracle.truffle.api.dsl.Specialization;
 import com.oracle.truffle.api.frame.Frame;
 import com.oracle.truffle.api.frame.FrameInstance.FrameAccess;
@@ -68,17 +69,19 @@ import com.oracle.truffle.r.runtime.data.RNull;
 import com.oracle.truffle.r.runtime.data.RStringVector;
 import com.oracle.truffle.r.runtime.data.model.RAbstractIntVector;
 import com.oracle.truffle.r.runtime.data.model.RAbstractStringVector;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 import com.oracle.truffle.r.runtime.ffi.BaseRFFI.UtsName;
-import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
 
 public class SysFunctions {
 
     @RBuiltin(name = "Sys.getpid", kind = INTERNAL, parameterNames = {}, behavior = READS_STATE)
     public abstract static class SysGetpid extends RBuiltinNode {
+        @Child private BaseRFFI.GetpidNode getpidNode = BaseRFFI.GetpidNode.create();
+
         @Specialization
         @TruffleBoundary
         protected Object sysGetPid() {
-            int pid = RFFIFactory.getRFFI().getBaseRFFI().getpid();
+            int pid = getpidNode.getpid();
             return RDataFactory.createIntVectorFromScalar(pid);
         }
     }
@@ -251,7 +254,8 @@ public class SysFunctions {
 
         @Specialization
         @TruffleBoundary
-        protected Object sysReadlink(RAbstractStringVector vector) {
+        protected Object sysReadlink(RAbstractStringVector vector,
+                        @Cached("create()") BaseRFFI.ReadlinkNode readlinkNode) {
             String[] paths = new String[vector.getLength()];
             boolean complete = RDataFactory.COMPLETE_VECTOR;
             for (int i = 0; i < paths.length; i++) {
@@ -259,7 +263,7 @@ public class SysFunctions {
                 if (RRuntime.isNA(path)) {
                     paths[i] = path;
                 } else {
-                    paths[i] = doSysReadLink(path);
+                    paths[i] = doSysReadLink(path, readlinkNode);
                 }
                 if (RRuntime.isNA(paths[i])) {
                     complete = RDataFactory.INCOMPLETE_VECTOR;
@@ -269,10 +273,10 @@ public class SysFunctions {
         }
 
         @TruffleBoundary
-        private static String doSysReadLink(String path) {
+        private static String doSysReadLink(String path, BaseRFFI.ReadlinkNode readlinkNode) {
             String s;
             try {
-                s = RFFIFactory.getRFFI().getBaseRFFI().readlink(path);
+                s = readlinkNode.readlink(path);
                 if (s == null) {
                     s = "";
                 }
@@ -294,14 +298,15 @@ public class SysFunctions {
 
         @Specialization
         @TruffleBoundary
-        protected RLogicalVector sysChmod(RAbstractStringVector pathVec, RAbstractIntVector octmode, @SuppressWarnings("unused") boolean useUmask) {
+        protected RLogicalVector sysChmod(RAbstractStringVector pathVec, RAbstractIntVector octmode, @SuppressWarnings("unused") boolean useUmask,
+                        @Cached("create()") BaseRFFI.ChmodNode chmodNode) {
             byte[] data = new byte[pathVec.getLength()];
             for (int i = 0; i < data.length; i++) {
                 String path = Utils.tildeExpand(pathVec.getDataAt(i));
                 if (path.length() == 0 || RRuntime.isNA(path)) {
                     continue;
                 }
-                int result = RFFIFactory.getRFFI().getBaseRFFI().chmod(path, octmode.getDataAt(i % octmode.getLength()));
+                int result = chmodNode.chmod(path, octmode.getDataAt(i % octmode.getLength()));
                 data[i] = RRuntime.asLogical(result == 0);
             }
             return RDataFactory.createLogicalVector(data, RDataFactory.COMPLETE_VECTOR);
@@ -338,10 +343,12 @@ public class SysFunctions {
         private static final String[] NAMES = new String[]{"sysname", "release", "version", "nodename", "machine", "login", "user", "effective_user"};
         private static final RStringVector NAMES_ATTR = RDataFactory.createStringVector(NAMES, RDataFactory.COMPLETE_VECTOR);
 
+        @Child private BaseRFFI.UnameNode unameNode = BaseRFFI.UnameNode.create();
+
         @Specialization
         @TruffleBoundary
         protected Object sysTime() {
-            UtsName utsname = RFFIFactory.getRFFI().getBaseRFFI().uname();
+            UtsName utsname = unameNode.uname();
             String[] data = new String[NAMES.length];
             data[0] = utsname.sysname();
             data[1] = utsname.release();
@@ -367,7 +374,8 @@ public class SysFunctions {
 
         @Specialization
         @TruffleBoundary
-        protected Object sysGlob(RAbstractStringVector pathVec, @SuppressWarnings("unused") boolean dirMask) {
+        protected Object sysGlob(RAbstractStringVector pathVec, @SuppressWarnings("unused") boolean dirMask,
+                        @Cached("create()") BaseRFFI.GlobNode globNode) {
             ArrayList<String> matches = new ArrayList<>();
             // Sys.glob closure already called path.expand
             for (int i = 0; i < pathVec.getLength(); i++) {
@@ -375,7 +383,7 @@ public class SysFunctions {
                 if (pathPattern.length() == 0 || RRuntime.isNA(pathPattern)) {
                     continue;
                 }
-                ArrayList<String> pathPatternMatches = RFFIFactory.getRFFI().getBaseRFFI().glob(pathPattern);
+                ArrayList<String> pathPatternMatches = globNode.glob(pathPattern);
                 matches.addAll(pathPatternMatches);
             }
             String[] data = new String[matches.size()];
diff --git a/com.oracle.truffle.r.runtime.ffi/src/com/oracle/truffle/r/runtime/ffi/jni/JNI_Base.java b/com.oracle.truffle.r.runtime.ffi/src/com/oracle/truffle/r/runtime/ffi/jni/JNI_Base.java
index 08c7928e00..d96a06215a 100644
--- a/com.oracle.truffle.r.runtime.ffi/src/com/oracle/truffle/r/runtime/ffi/jni/JNI_Base.java
+++ b/com.oracle.truffle.r.runtime.ffi/src/com/oracle/truffle/r/runtime/ffi/jni/JNI_Base.java
@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 2014, 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
@@ -28,98 +28,118 @@ import java.util.ArrayList;
 import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 
 public class JNI_Base implements BaseRFFI {
-    @Override
-    public int getpid() {
-        return native_getpid();
-    }
-
-    @Override
-    public int setwd(String dir) {
-        return native_setwd(dir);
+    public static class JNI_GetpidNode extends GetpidNode {
+        @Override
+        public int getpid() {
+            return native_getpid();
+        }
     }
 
-    @Override
-    public String getwd() {
-        byte[] buf = new byte[4096];
-        int rc = native_getwd(buf, buf.length);
-        if (rc == 0) {
-            return null;
-        } else {
-            int i = 0;
-            while (buf[i] != 0 && i < buf.length) {
-                i++;
+    public static class JNI_GetwdNode extends GetwdNode {
+        @Override
+        public String getwd() {
+            byte[] buf = new byte[4096];
+            int rc = native_getwd(buf, buf.length);
+            if (rc == 0) {
+                return null;
+            } else {
+                int i = 0;
+                while (buf[i] != 0 && i < buf.length) {
+                    i++;
+                }
+                return new String(buf, 0, i);
             }
-            return new String(buf, 0, i);
         }
     }
 
-    private static final int EINVAL = 22;
+    public static class JNI_SetwdNode extends SetwdNode {
+        @Override
+        public int setwd(String dir) {
+            return native_setwd(dir);
+        }
+    }
 
-    @Override
-    public String readlink(String path) throws IOException {
-        int[] errno = new int[]{0};
-        String s = native_readlink(path, errno);
-        if (s == null) {
-            if (errno[0] == EINVAL) {
-                // not a link
-            } else {
-                // some other error
-                throw new IOException("readlink failed: " + errno[0]);
+    public static class JNI_ReadlinkNode extends ReadlinkNode {
+        private static final int EINVAL = 22;
+
+        @Override
+        public String readlink(String path) throws IOException {
+            int[] errno = new int[]{0};
+            String s = native_readlink(path, errno);
+            if (s == null) {
+                if (errno[0] == EINVAL) {
+                    // not a link
+                } else {
+                    // some other error
+                    throw new IOException("readlink failed: " + errno[0]);
+                }
             }
+            return s;
         }
-        return s;
     }
 
-    @Override
-    public String mkdtemp(String template) {
-        /*
-         * Not only must the (C) string end in XXXXXX it must also be null-terminated. Since it is
-         * modified by mkdtemp we must make a copy.
-         */
-        byte[] bytes = template.getBytes();
-        byte[] ztbytes = new byte[bytes.length + 1];
-        System.arraycopy(bytes, 0, ztbytes, 0, bytes.length);
-        ztbytes[bytes.length] = 0;
-        long result = native_mkdtemp(ztbytes);
-        if (result == 0) {
-            return null;
-        } else {
-            return new String(ztbytes, 0, bytes.length);
+    public static class JNI_MkdtempNode extends MkdtempNode {
+        @Override
+        public String mkdtemp(String template) {
+            /*
+             * Not only must the (C) string end in XXXXXX it must also be null-terminated. Since it
+             * is modified by mkdtemp we must make a copy.
+             */
+            byte[] bytes = template.getBytes();
+            byte[] ztbytes = new byte[bytes.length + 1];
+            System.arraycopy(bytes, 0, ztbytes, 0, bytes.length);
+            ztbytes[bytes.length] = 0;
+            long result = native_mkdtemp(ztbytes);
+            if (result == 0) {
+                return null;
+            } else {
+                return new String(ztbytes, 0, bytes.length);
+            }
         }
     }
 
-    @Override
-    public void mkdir(String dir, int mode) throws IOException {
-        int rc = native_mkdir(dir, mode);
-        if (rc != 0) {
-            throw new IOException("mkdir " + dir + " failed");
+    public static class JNI_MkdirNode extends MkdirNode {
+        @Override
+        public void mkdir(String dir, int mode) throws IOException {
+            int rc = native_mkdir(dir, mode);
+            if (rc != 0) {
+                throw new IOException("mkdir " + dir + " failed");
+            }
         }
     }
 
-    @Override
-    public int chmod(String path, int mode) {
-        return native_chmod(path, mode);
+    public static class JNI_ChmodNode extends ChmodNode {
+        @Override
+        public int chmod(String path, int mode) {
+            return native_chmod(path, mode);
+        }
     }
 
-    @Override
-    public long strtol(String s, int base) throws IllegalArgumentException {
-        int[] errno = new int[]{0};
-        long result = native_strtol(s, base, errno);
-        if (errno[0] != 0) {
-            throw new IllegalArgumentException("strtol failure");
-        } else {
-            return result;
+    public static class JNI_StrolNode extends StrolNode {
+        @Override
+        public long strtol(String s, int base) throws IllegalArgumentException {
+            int[] errno = new int[]{0};
+            long result = native_strtol(s, base, errno);
+            if (errno[0] != 0) {
+                throw new IllegalArgumentException("strtol failure");
+            } else {
+                return result;
+            }
         }
     }
 
-    @Override
-    public UtsName uname() {
-        return JNI_UtsName.get();
+    public static class JNI_UnameNode extends UnameNode {
+        @Override
+        public UtsName uname() {
+            return JNI_UtsName.get();
+        }
     }
 
-    @Override
-    public ArrayList<String> glob(String pattern) {
-        return JNI_Glob.glob(pattern);
+    public static class JNI_GlobNode extends GlobNode {
+        @Override
+        public ArrayList<String> glob(String pattern) {
+            return JNI_Glob.glob(pattern);
+        }
     }
 
     // Checkstyle: stop method name
@@ -139,4 +159,54 @@ public class JNI_Base implements BaseRFFI {
     private static native long native_strtol(String s, int base, int[] errno);
 
     private static native String native_readlink(String s, int[] errno);
+
+    @Override
+    public GetpidNode createGetpidNode() {
+        return new JNI_GetpidNode();
+    }
+
+    @Override
+    public GetwdNode createGetwdNode() {
+        return new JNI_GetwdNode();
+    }
+
+    @Override
+    public SetwdNode createSetwdNode() {
+        return new JNI_SetwdNode();
+    }
+
+    @Override
+    public MkdirNode createMkdirNode() {
+        return new JNI_MkdirNode();
+    }
+
+    @Override
+    public ReadlinkNode createReadlinkNode() {
+        return new JNI_ReadlinkNode();
+    }
+
+    @Override
+    public MkdtempNode createMkdtempNode() {
+        return new JNI_MkdtempNode();
+    }
+
+    @Override
+    public ChmodNode createChmodNode() {
+        return new JNI_ChmodNode();
+    }
+
+    @Override
+    public StrolNode createStrolNode() {
+        return new JNI_StrolNode();
+    }
+
+    @Override
+    public UnameNode createUnameNode() {
+        return new JNI_UnameNode();
+    }
+
+    @Override
+    public GlobNode createGlobNode() {
+        return new JNI_GlobNode();
+    }
 }
diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/REnvVars.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/REnvVars.java
index d6045a767b..86e5fd3060 100644
--- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/REnvVars.java
+++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/REnvVars.java
@@ -37,7 +37,7 @@ import java.util.Map;
 import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
 import com.oracle.truffle.api.vm.PolyglotEngine;
 import com.oracle.truffle.r.runtime.context.RContext;
-import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 
 /**
  * Repository for environment variables, including those set by FastR itself, e.g.
@@ -88,7 +88,7 @@ public final class REnvVars implements RContext.ContextState {
             String userFile = envVars.get("R_ENVIRON_USER");
             if (userFile == null) {
                 String dotRenviron = ".Renviron";
-                userFile = fileSystem.getPath(RFFIFactory.getRFFI().getBaseRFFI().getwd(), dotRenviron).toString();
+                userFile = fileSystem.getPath((String) BaseRFFI.GetwdRootNode.create().getCallTarget().call(), dotRenviron).toString();
                 if (!new File(userFile).exists()) {
                     userFile = fileSystem.getPath(System.getProperty("user.home"), dotRenviron).toString();
                 }
diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RProfile.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RProfile.java
index 3af2daa50e..acd5193d57 100644
--- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RProfile.java
+++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RProfile.java
@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 2013, 2016, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 2013, 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
@@ -30,7 +30,7 @@ import java.nio.file.Path;
 
 import com.oracle.truffle.api.source.Source;
 import com.oracle.truffle.r.runtime.context.RContext;
-import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 
 /**
  * Handles the setup of system, site and user profile code. N.B. this class only reads the files and
@@ -68,7 +68,7 @@ public final class RProfile implements RContext.ContextState {
             String userProfilePath = envVars.get("R_PROFILE_USER");
             if (userProfilePath == null) {
                 String dotRenviron = ".Rprofile";
-                userProfilePath = fileSystem.getPath(RFFIFactory.getRFFI().getBaseRFFI().getwd(), dotRenviron).toString();
+                userProfilePath = fileSystem.getPath((String) BaseRFFI.GetwdRootNode.create().getCallTarget().call(), dotRenviron).toString();
                 if (!new File(userProfilePath).exists()) {
                     userProfilePath = fileSystem.getPath(System.getProperty("user.home"), dotRenviron).toString();
                 }
diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RSrcref.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RSrcref.java
index 5de8f818ee..c2b9e89aa7 100644
--- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RSrcref.java
+++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RSrcref.java
@@ -27,6 +27,7 @@ import com.oracle.truffle.r.runtime.data.RNull;
 import com.oracle.truffle.r.runtime.data.RStringVector;
 import com.oracle.truffle.r.runtime.env.REnvironment;
 import com.oracle.truffle.r.runtime.env.REnvironment.PutException;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
 import com.oracle.truffle.r.runtime.nodes.RSyntaxNode;
 
@@ -79,7 +80,7 @@ public class RSrcref {
         env.safePut(SrcrefFields.timestamp.name(), mtime);
         env.safePut(SrcrefFields.filename.name(), path.toString());
         env.safePut(SrcrefFields.isFile.name(), RRuntime.LOGICAL_TRUE);
-        env.safePut(SrcrefFields.wd.name(), RFFIFactory.getRFFI().getBaseRFFI().getwd());
+        env.safePut(SrcrefFields.wd.name(), BaseRFFI.GetwdRootNode.create().getCallTarget().call());
         env.setClassAttr(SRCFILE_ATTR);
         return env;
     }
diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RVersionInfo.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RVersionInfo.java
index 8e64036349..6336fb8835 100644
--- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RVersionInfo.java
+++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/RVersionInfo.java
@@ -24,8 +24,8 @@ package com.oracle.truffle.r.runtime;
 
 import com.oracle.truffle.api.CompilerDirectives;
 import com.oracle.truffle.api.CompilerDirectives.CompilationFinal;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 import com.oracle.truffle.r.runtime.ffi.BaseRFFI.UtsName;
-import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
 
 public enum RVersionInfo {
     // @formatter:off
@@ -82,7 +82,7 @@ public enum RVersionInfo {
             CompilerDirectives.transferToInterpreterAndInvalidate();
             ListValues = new String[VALUES.length];
             ListNames = new String[VALUES.length];
-            UtsName utsname = RFFIFactory.getRFFI().getBaseRFFI().uname();
+            UtsName utsname = (UtsName) BaseRFFI.UnameRootNode.create().getCallTarget().call();
             String osName = toFirstLower(utsname.sysname());
             String vendor = osName.equals("darwin") ? "apple" : "unknown";
             OS.value = osName + utsname.release();
diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/TempPathName.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/TempPathName.java
index f829eb438a..d0783f13cf 100644
--- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/TempPathName.java
+++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/TempPathName.java
@@ -29,7 +29,7 @@ import java.util.Random;
 
 import com.oracle.truffle.api.CompilerDirectives.TruffleBoundary;
 import com.oracle.truffle.r.runtime.context.RContext;
-import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 
 /**
  *
@@ -63,7 +63,7 @@ public class TempPathName implements RContext.ContextState {
         if (!startingTempDirPath.isAbsolute()) {
             startingTempDirPath = startingTempDirPath.toAbsolutePath();
         }
-        String t = RFFIFactory.getRFFI().getBaseRFFI().mkdtemp(startingTempDirPath.toString() + "XXXXXX");
+        String t = (String) BaseRFFI.MkdtempRootNode.create().getCallTarget().call(startingTempDirPath.toString() + "XXXXXX");
         if (t != null) {
             tempDirPath = t;
         } else {
diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/Utils.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/Utils.java
index 639ece39b6..19bb59be13 100644
--- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/Utils.java
+++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/Utils.java
@@ -60,6 +60,7 @@ import com.oracle.truffle.r.runtime.data.RNull;
 import com.oracle.truffle.r.runtime.data.RPairList;
 import com.oracle.truffle.r.runtime.data.RStringVector;
 import com.oracle.truffle.r.runtime.data.model.RAbstractContainer;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
 import com.oracle.truffle.r.runtime.nodes.RSyntaxNode;
 
@@ -284,7 +285,7 @@ public final class Utils {
      */
     public static Path getLogPath(String fileName) {
         String root = RContext.isEmbedded() ? "/tmp" : REnvVars.rHome();
-        int pid = RFFIFactory.getRFFI().getBaseRFFI().getpid();
+        int pid = (int) BaseRFFI.GetpidRootNode.create().getCallTarget().call();
         String baseName = RContext.isEmbedded() ? fileName + "-" + Integer.toString(pid) : fileName;
         return FileSystems.getDefault().getPath(root, baseName);
     }
diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/ffi/BaseRFFI.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/ffi/BaseRFFI.java
index fe950fa2a4..d3124d1a93 100644
--- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/ffi/BaseRFFI.java
+++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/ffi/BaseRFFI.java
@@ -1,5 +1,5 @@
 /*
- * Copyright (c) 2014, 2016, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 2014, 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
@@ -25,55 +25,110 @@ package com.oracle.truffle.r.runtime.ffi;
 import java.io.IOException;
 import java.util.ArrayList;
 
+import com.oracle.truffle.api.Truffle;
+import com.oracle.truffle.api.frame.FrameDescriptor;
+import com.oracle.truffle.api.frame.VirtualFrame;
+import com.oracle.truffle.api.nodes.Node;
+import com.oracle.truffle.api.nodes.RootNode;
+import com.oracle.truffle.r.runtime.context.RContext;
+
 /**
  * A statically typed interface to exactly those native functions required by the R {@code base}
  * package, because the functionality is not provided by the JDK. These methods do not necessarily
  * map 1-1 to a native function, they may involve the invocation of several native functions.
  */
 public interface BaseRFFI {
-    int getpid();
+    abstract class GetpidNode extends Node {
+        public abstract int getpid();
 
-    /**
-     * Returns the current working directory, in the face of calls to {@link #setwd}.
-     */
-    String getwd();
+        public static GetpidNode create() {
+            return RFFIFactory.getRFFI().getBaseRFFI().createGetpidNode();
+        }
+    }
 
-    /**
-     * Sets the current working directory to {@code dir}. (cf. Unix {@code chdir}).
-     *
-     * @return 0 if successful.
-     */
-    int setwd(String dir);
+    abstract class GetwdNode extends Node {
+        /**
+         * Returns the current working directory, in the face of calls to {@code setwd}.
+         */
+        public abstract String getwd();
 
-    /**
-     * Create directory with given mode. Exception is thrown omn error.
-     */
-    void mkdir(String dir, int mode) throws IOException;
-
-    /**
-     * Try to convert a symbolic link to it's target.
-     *
-     * @param path the link path
-     * @return the target if {@code path} is a link else {@code null}
-     * @throws IOException for any other error except "not a link"
-     */
-    String readlink(String path) throws IOException;
+        public static GetwdNode create() {
+            return RFFIFactory.getRFFI().getBaseRFFI().createGetwdNode();
+        }
+    }
 
-    /**
-     * Creates a temporary directory using {@code template} and return the resulting path or
-     * {@code null} if error.
-     */
-    String mkdtemp(String template);
+    abstract class SetwdNode extends Node {
+        /**
+         * Sets the current working directory to {@code dir}. (cf. Unix {@code chdir}).
+         *
+         * @return 0 if successful.
+         */
+        public abstract int setwd(String dir);
 
-    /**
-     * Change the file mode of {@code path}.
-     */
-    int chmod(String path, int mode);
+        public static SetwdNode create() {
+            return RFFIFactory.getRFFI().getBaseRFFI().createSetwdNode();
+        }
+    }
 
-    /**
-     * Convert string to long.
-     */
-    long strtol(String s, int base) throws IllegalArgumentException;
+    abstract class MkdirNode extends Node {
+        /**
+         * Create directory with given mode. Exception is thrown omn error.
+         */
+        public abstract void mkdir(String dir, int mode) throws IOException;
+
+        public static MkdirNode create() {
+            return RFFIFactory.getRFFI().getBaseRFFI().createMkdirNode();
+        }
+    }
+
+    abstract class ReadlinkNode extends Node {
+        /**
+         * Try to convert a symbolic link to it's target.
+         *
+         * @param path the link path
+         * @return the target if {@code path} is a link else {@code null}
+         * @throws IOException for any other error except "not a link"
+         */
+        public abstract String readlink(String path) throws IOException;
+
+        public static ReadlinkNode create() {
+            return RFFIFactory.getRFFI().getBaseRFFI().createReadlinkNode();
+        }
+    }
+
+    abstract class MkdtempNode extends Node {
+        /**
+         * Creates a temporary directory using {@code template} and return the resulting path or
+         * {@code null} if error.
+         */
+        public abstract String mkdtemp(String template);
+
+        public static MkdtempNode create() {
+            return RFFIFactory.getRFFI().getBaseRFFI().createMkdtempNode();
+        }
+    }
+
+    abstract class ChmodNode extends Node {
+        /**
+         * Change the file mode of {@code path}.
+         */
+        public abstract int chmod(String path, int mode);
+
+        public static ChmodNode create() {
+            return RFFIFactory.getRFFI().getBaseRFFI().createChmodNode();
+        }
+    }
+
+    abstract class StrolNode extends Node {
+        /**
+         * Convert string to long.
+         */
+        public abstract long strtol(String s, int base) throws IllegalArgumentException;
+
+        public static StrolNode create() {
+            return RFFIFactory.getRFFI().getBaseRFFI().createStrolNode();
+        }
+    }
 
     public interface UtsName {
         String sysname();
@@ -87,14 +142,148 @@ public interface BaseRFFI {
         String nodename();
     }
 
-    /**
-     * Return {@code utsname} info.
+    abstract class UnameNode extends Node {
+        /**
+         * Return {@code utsname} info.
+         */
+        public abstract UtsName uname();
+
+        public static UnameNode create() {
+            return RFFIFactory.getRFFI().getBaseRFFI().createUnameNode();
+        }
+    }
+
+    abstract class GlobNode extends Node {
+        /**
+         * Returns an array of pathnames that match {@code pattern} using the OS glob function. This
+         * is done in native code because it is very hard to write in Java in the face of
+         * {@code setwd}.
+         */
+        public abstract ArrayList<String> glob(String pattern);
+
+        public static GlobNode create() {
+            return RFFIFactory.getRFFI().getBaseRFFI().createGlobNode();
+        }
+    }
+
+    /*
+     * The RFFI implementation influences exactly what subclass of the above nodes is created. Each
+     * implementation must therefore, implement these methods that are called by the associated
+     * "public static create()" methods above.
      */
-    UtsName uname();
 
-    /**
-     * Returns an array of pathnames that match {@code pattern} using the OS glob function. This is
-     * done in native code because it is very hard to write in Java in the face of {@link #setwd}.
+    GetpidNode createGetpidNode();
+
+    GetwdNode createGetwdNode();
+
+    SetwdNode createSetwdNode();
+
+    MkdirNode createMkdirNode();
+
+    ReadlinkNode createReadlinkNode();
+
+    MkdtempNode createMkdtempNode();
+
+    ChmodNode createChmodNode();
+
+    StrolNode createStrolNode();
+
+    UnameNode createUnameNode();
+
+    GlobNode createGlobNode();
+
+    /*
+     * Some functions are called from non-Truffle contexts, which requires a RootNode
      */
-    ArrayList<String> glob(String pattern);
+
+    abstract class BaseRFFIRootNode<T extends Node> extends RootNode {
+        @Child T baseRFFINode;
+
+        private BaseRFFIRootNode(T baseRFFINode) {
+            super(RContext.getRRuntimeASTAccess().getTruffleRLanguage(), null, new FrameDescriptor());
+            this.baseRFFINode = baseRFFINode;
+            Truffle.getRuntime().createCallTarget(this);
+        }
+    }
+
+    final class GetpidRootNode extends BaseRFFIRootNode<GetpidNode> {
+        private static GetpidRootNode getpidRootNode;
+
+        private GetpidRootNode() {
+            super(RFFIFactory.getRFFI().getBaseRFFI().createGetpidNode());
+        }
+
+        @Override
+        public Object execute(VirtualFrame frame) {
+            return baseRFFINode.getpid();
+        }
+
+        public static GetpidRootNode create() {
+            if (getpidRootNode == null) {
+                getpidRootNode = new GetpidRootNode();
+            }
+            return getpidRootNode;
+        }
+    }
+
+    final class GetwdRootNode extends BaseRFFIRootNode<GetwdNode> {
+        private static GetwdRootNode getwdRootNode;
+
+        private GetwdRootNode() {
+            super(RFFIFactory.getRFFI().getBaseRFFI().createGetwdNode());
+        }
+
+        @Override
+        public Object execute(VirtualFrame frame) {
+            return baseRFFINode.getwd();
+        }
+
+        public static GetwdRootNode create() {
+            if (getwdRootNode == null) {
+                getwdRootNode = new GetwdRootNode();
+            }
+            return getwdRootNode;
+        }
+    }
+
+    final class MkdtempRootNode extends BaseRFFIRootNode<MkdtempNode> {
+        private static MkdtempRootNode mkdtempRootNode;
+
+        private MkdtempRootNode() {
+            super(RFFIFactory.getRFFI().getBaseRFFI().createMkdtempNode());
+        }
+
+        @Override
+        public Object execute(VirtualFrame frame) {
+            Object[] args = frame.getArguments();
+            return baseRFFINode.mkdtemp((String) args[0]);
+        }
+
+        public static MkdtempRootNode create() {
+            if (mkdtempRootNode == null) {
+                mkdtempRootNode = new MkdtempRootNode();
+            }
+            return mkdtempRootNode;
+        }
+    }
+
+    final class UnameRootNode extends BaseRFFIRootNode<UnameNode> {
+        private static UnameRootNode unameRootNode;
+
+        private UnameRootNode() {
+            super(RFFIFactory.getRFFI().getBaseRFFI().createUnameNode());
+        }
+
+        @Override
+        public Object execute(VirtualFrame frame) {
+            return baseRFFINode.uname();
+        }
+
+        public static UnameRootNode create() {
+            if (unameRootNode == null) {
+                unameRootNode = new UnameRootNode();
+            }
+            return unameRootNode;
+        }
+    }
 }
diff --git a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/rng/RRNG.java b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/rng/RRNG.java
index 7ab4c8007b..db3f531e93 100644
--- a/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/rng/RRNG.java
+++ b/com.oracle.truffle.r.runtime/src/com/oracle/truffle/r/runtime/rng/RRNG.java
@@ -25,7 +25,7 @@ import com.oracle.truffle.r.runtime.data.RNull;
 import com.oracle.truffle.r.runtime.data.RPromise;
 import com.oracle.truffle.r.runtime.data.RTypedValue;
 import com.oracle.truffle.r.runtime.env.REnvironment;
-import com.oracle.truffle.r.runtime.ffi.RFFIFactory;
+import com.oracle.truffle.r.runtime.ffi.BaseRFFI;
 import com.oracle.truffle.r.runtime.rng.mm.MarsagliaMulticarry;
 import com.oracle.truffle.r.runtime.rng.mt.MersenneTwister;
 import com.oracle.truffle.r.runtime.rng.user.UserRNG;
@@ -294,7 +294,7 @@ public class RRNG {
      * Create a random integer.
      */
     public static Integer timeToSeed() {
-        int pid = RFFIFactory.getRFFI().getBaseRFFI().getpid();
+        int pid = (int) BaseRFFI.GetpidRootNode.create().getCallTarget().call();
         int millis = (int) (System.currentTimeMillis() & 0xFFFFFFFFL);
         return (millis << 16) ^ pid;
     }
-- 
GitLab