diff --git a/com.oracle.truffle.r.native/fficall/jni/src/alloc.c b/com.oracle.truffle.r.native/fficall/jni/src/alloc.c index d974d45d9cb5c6acae7a3864eeda99ecad9fd0ad..c46ae28e16eeab94bdfb84067be6fae5ff16a672 100644 --- a/com.oracle.truffle.r.native/fficall/jni/src/alloc.c +++ b/com.oracle.truffle.r.native/fficall/jni/src/alloc.c @@ -12,10 +12,43 @@ #include "rffiutils.h" #include <stdlib.h> +#define T_MEM_TABLE_INITIAL_SIZE 0 +// The table of transient objects that have been allocated dur the current FFI call +static void **tMemTable; +// hwm of tMemTable +static int tMemTableIndex; +static int tMemTableLength; void init_alloc(JNIEnv *env) { + tMemTable = malloc(sizeof(void*) * T_MEM_TABLE_INITIAL_SIZE); + tMemTableLength = T_MEM_TABLE_INITIAL_SIZE; + tMemTableIndex = 0; +} + +// Memory that is auto-reclaimed across FFI calls +char *R_alloc(size_t n, int size) { + void *p = R_chk_alloc(n, size); + if (tMemTableIndex >= tMemTableLength) { + int newLength = 2 * tMemTableLength; + void *newtMemTable = malloc(sizeof(void*) * newLength); + if (newtMemTable == NULL) { + fatalError("malloc failure"); + } + memcpy(newtMemTable, tMemTable, tMemTableLength * sizeof(void*)); + free(tMemTable); + tMemTable = newtMemTable; + tMemTableLength = newLength; + } + tMemTable[tMemTableIndex] = p; + return (char*) p; } +void allocExit() { + int i; + for (i = 0; i < tMemTableIndex; i++) { + free(tMemTable[i]); + } +} void *R_chk_calloc(size_t nelem, size_t elsize) { void *p; diff --git a/com.oracle.truffle.r.native/fficall/jni/src/misc.c b/com.oracle.truffle.r.native/fficall/jni/src/misc.c index 5be911f7d48f0091454bfa966e06afd1ce3774fc..fb9dfbcf8909b17e411a91b68784ba6d2d5d08ce 100644 --- a/com.oracle.truffle.r.native/fficall/jni/src/misc.c +++ b/com.oracle.truffle.r.native/fficall/jni/src/misc.c @@ -45,3 +45,14 @@ const char *R_CHAR(SEXP string) { return copyChars; } +void R_isort(int *x, int n) { + unimplemented("R_isort"); +} + +void R_rsort(double *x, int n) { + unimplemented("R_rsort"); +} + +void R_CheckUserInterrupt() { +// TODO (we don't even do this in the Java code) +} diff --git a/com.oracle.truffle.r.native/fficall/jni/src/rf_functions.c b/com.oracle.truffle.r.native/fficall/jni/src/rf_functions.c index bd385fff8652d8589aaca1fac742bcceac6f1dcc..72017f431d1336d1253209194f98387b1ee9d104 100644 --- a/com.oracle.truffle.r.native/fficall/jni/src/rf_functions.c +++ b/com.oracle.truffle.r.native/fficall/jni/src/rf_functions.c @@ -44,6 +44,8 @@ static jmethodID Rf_isNullMethodID; static jmethodID Rf_warningMethodID; static jmethodID Rf_errorMethodID; static jmethodID Rf_NewHashedEnvMethodID; +static jmethodID Rf_rPsortMethodID; +static jmethodID Rf_iPsortMethodID; void init_rf_functions(JNIEnv *env) { Rf_ScalarIntegerMethodID = checkGetMethodID(env, CallRFFIHelperClass, "Rf_ScalarInteger", "(I)Lcom/oracle/truffle/r/runtime/data/RIntVector;", 1); @@ -64,6 +66,8 @@ void init_rf_functions(JNIEnv *env) { createListMethodID = checkGetMethodID(env, RDataFactoryClass, "createList", "(I)Lcom/oracle/truffle/r/runtime/data/RList;", 1); Rf_duplicateMethodID = checkGetMethodID(env, CallRFFIHelperClass, "Rf_duplicate", "(Ljava/lang/Object;)Ljava/lang/Object;", 1); Rf_NewHashedEnvMethodID = checkGetMethodID(env, RDataFactoryClass, "createNewEnv", "(Lcom/oracle/truffle/r/runtime/env/REnvironment;Ljava/lang/String;ZI)Lcom/oracle/truffle/r/runtime/env/REnvironment;", 1); +// Rf_rPsortMethodID = checkGetMethodID(env, CallRFFIHelperClass, "Rf_rPsort", "(Lcom/oracle/truffle/r/runtime/data/RDoubleVector;II)", 1); +// Rf_iPsortMethodID = checkGetMethodID(env, CallRFFIHelperClass, "Rf_iPsort", "(Lcom/oracle/truffle/r/runtime/data/RIntVector;II)", 1); } SEXP Rf_ScalarInteger(int value) { @@ -210,7 +214,7 @@ void Rf_error(const char *msg, ...) { // and, if it finds any, does not return, but throws a different exception than RError. // We definitely need to exit the FFI call and we certainly cannot return to our caller. // So we call CallRFFIHelper.Rf_error to throw the RError exception. When the pending - // exception (whatever it is) is observed by JNI, he call to Rf_error will return where we do a + // exception (whatever it is) is observed by JNI, the call to Rf_error will return where we do a // non-local transfer of control back to the entry point (which will cleanup). JNIEnv *thisenv = getEnv(); jstring string = (*thisenv)->NewStringUTF(thisenv, msg); @@ -246,3 +250,14 @@ SEXP R_NewHashedEnv(SEXP parent, SEXP size) { SEXP result = (*thisenv)->CallStaticObjectMethod(thisenv, RDataFactoryClass, Rf_NewHashedEnvMethodID, parent, NULL, JNI_TRUE, sizeAsInt); return checkRef(thisenv, result); } + +void Rf_iPsort(int *x, int n, int k) +{ + JNIEnv *thisenv = getEnv(); + unimplemented("Rf_iPsort"); +} + +void Rf_rPsort(double *x, int n, int k) { + JNIEnv *thisenv = getEnv(); + unimplemented("Rf_rPsort"); +} diff --git a/com.oracle.truffle.r.native/fficall/jni/src/rfficall.c b/com.oracle.truffle.r.native/fficall/jni/src/rfficall.c index a7680e164b54df50ef6b837c6c670f8a9d9b1e4d..939c7a7a026266bf85a0017d058d226e2633d60c 100644 --- a/com.oracle.truffle.r.native/fficall/jni/src/rfficall.c +++ b/com.oracle.truffle.r.native/fficall/jni/src/rfficall.c @@ -36,6 +36,8 @@ Java_com_oracle_truffle_r_runtime_ffi_jnr_CallRFFIWithJNI_initialize(JNIEnv *env init_typecoerce(env); init_attrib(env); init_misc(env); + init_rng(env); + init_optim(env); init_vectoraccess(env); init_listaccess(env); } diff --git a/com.oracle.truffle.r.native/fficall/jni/src/rffiutils.h b/com.oracle.truffle.r.native/fficall/jni/src/rffiutils.h index 0e550a5d04545380c669fa4c32e6c72e6748be3a..691a89c7090216d91b71625d8242f1784a274448 100644 --- a/com.oracle.truffle.r.native/fficall/jni/src/rffiutils.h +++ b/com.oracle.truffle.r.native/fficall/jni/src/rffiutils.h @@ -53,6 +53,8 @@ void validateRef(JNIEnv *env, SEXP x, const char *msg); void callEnter(JNIEnv *env, jmp_buf *error_exit); // exiting a top-level JNI call void callExit(JNIEnv *env); +// called by callExit to deallocate transient memory +void allocExit(); jmp_buf *getErrorJmpBuf(); @@ -68,6 +70,8 @@ void init_externalptr(JNIEnv *env); void init_typecoerce(JNIEnv *env); void init_attrib(JNIEnv *env); void init_misc(JNIEnv *env); +void init_rng(JNIEnv *env); +void init_optim(JNIEnv *env); void init_vectoraccess(JNIEnv *env); void init_listaccess(JNIEnv *env); void init_utils(JNIEnv *env); diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/BasePackage.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/BasePackage.java index 1e2d883251e280e898961c1db1babc76046eb090..2bf3ba82f733a4c4d5610cef66858c710505b2aa 100644 --- a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/BasePackage.java +++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/BasePackage.java @@ -502,5 +502,6 @@ public class BasePackage extends RBuiltinPackage { add(WhichFunctions.Which.class, WhichFunctionsFactory.WhichNodeGen::create); add(WhichFunctions.WhichMax.class, WhichFunctionsFactory.WhichMaxNodeGen::create); add(WhichFunctions.WhichMin.class, WhichFunctionsFactory.WhichMinNodeGen::create); + add(Xtfrm.class, XtfrmNodeGen::create); } } 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 8f50865fc7c9d66b2590fdad2c8778a7c19bf40b..8de3ec74ba656feb691a5b4f233c34bd793faaa4 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 @@ -307,15 +307,21 @@ public class FileFunctions { // @formatter:on } + private static void updateComplete(int slot, boolean[] complete, boolean update) { + if (complete[slot]) { + complete[slot] = update; + } + } + private static void setColumnValue(Column column, Object[] data, boolean[] complete, int index, Object value) { int slot = column.ordinal(); // @formatter:off switch(column) { - case size: ((double[]) data[slot])[index] = (double) value; complete[slot] = (double) value != RRuntime.DOUBLE_NA; return; - case isdir: ((byte[]) data[slot])[index] = (byte) value; complete[slot] = (byte) value != RRuntime.LOGICAL_NA; return; + case size: ((double[]) data[slot])[index] = (double) value; updateComplete(slot, complete, (double) value != RRuntime.DOUBLE_NA); return; + case isdir: ((byte[]) data[slot])[index] = (byte) value; updateComplete(slot, complete, (byte) value != RRuntime.LOGICAL_NA); return; case mode: case mtime: case ctime: case atime: - case uid: case gid: ((int[]) data[slot])[index] = (int) value; complete[slot] = (int) value != RRuntime.INT_NA; return; - case uname: case grname: ((String[]) data[slot])[index] = (String) value; complete[slot] = (String) value != RRuntime.STRING_NA; return; + case uid: case gid: ((int[]) data[slot])[index] = (int) value; updateComplete(slot, complete, (int) value != RRuntime.INT_NA); return; + case uname: case grname: ((String[]) data[slot])[index] = (String) value; updateComplete(slot, complete, (String) value != RRuntime.STRING_NA); return; default: throw RInternalError.shouldNotReachHere(); } // @formatter:on diff --git a/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Xtfrm.java b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Xtfrm.java new file mode 100644 index 0000000000000000000000000000000000000000..d35f485be4bfb99b545f75e02db7b0a6af57858d --- /dev/null +++ b/com.oracle.truffle.r.nodes.builtin/src/com/oracle/truffle/r/nodes/builtin/base/Xtfrm.java @@ -0,0 +1,33 @@ +package com.oracle.truffle.r.nodes.builtin.base; + +import static com.oracle.truffle.r.runtime.RBuiltinKind.*; +import static com.oracle.truffle.r.runtime.RDispatch.*; + +import com.oracle.truffle.api.*; +import com.oracle.truffle.api.dsl.*; +import com.oracle.truffle.api.frame.*; +import com.oracle.truffle.r.nodes.builtin.*; +import com.oracle.truffle.r.nodes.builtin.base.GetFunctionsFactory.*; +import com.oracle.truffle.r.runtime.*; +import com.oracle.truffle.r.runtime.data.*; +import com.oracle.truffle.r.runtime.nodes.*; + +@RBuiltin(name = "xtfrm", kind = PRIMITIVE, parameterNames = {"x"}, dispatch = INTERNAL_GENERIC) +public abstract class Xtfrm extends RBuiltinNode { + @Child private GetFunctions.Get getNode; + + @Specialization + protected Object xtfrm(VirtualFrame frame, Object x) { + /* + * Although this is a PRIMITIVE, there is an xtfrm.default that we must call if "x" is not + * of a class that already has an xtfrm.class function defined. We only get here in the + * default case. + */ + if (getNode == null) { + CompilerDirectives.transferToInterpreterAndInvalidate(); + getNode = insert(GetNodeGen.create(new RNode[4], null, null)); + } + RFunction func = (RFunction) getNode.execute(frame, "xtfrm.default", RArguments.getEnvironment(frame), RType.Function.getName(), RRuntime.LOGICAL_TRUE); + return RContext.getEngine().evalFunction(func, x); + } +} diff --git a/com.oracle.truffle.r.runtime.ffi/src/com/oracle/truffle/r/runtime/ffi/jnr/JNR_RFFIFactory.java b/com.oracle.truffle.r.runtime.ffi/src/com/oracle/truffle/r/runtime/ffi/jnr/JNR_RFFIFactory.java index 5ab5cb310179a7c07a00f5b663c3c50a8827eee0..e6f420c416215c781878fd9b187eacb367246552 100644 --- a/com.oracle.truffle.r.runtime.ffi/src/com/oracle/truffle/r/runtime/ffi/jnr/JNR_RFFIFactory.java +++ b/com.oracle.truffle.r.runtime.ffi/src/com/oracle/truffle/r/runtime/ffi/jnr/JNR_RFFIFactory.java @@ -53,6 +53,12 @@ public class JNR_RFFIFactory extends RFFIFactory implements RFFI, BaseRFFI, Stat protected void initialize() { // This must load early as package libraries reference symbols in it. getCallRFFI(); + /* + * Some package C code calls these functions and, therefore, expects the linpack symbols to + * be available, which will not be the case unless one of the functions has already been + * called from R code. So we eagerly load the library to define the symbols. + */ + linpack(); } /**