diff --git a/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/RRuntimeASTAccessImpl.java b/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/RRuntimeASTAccessImpl.java index eeed7e583d8093c46c2f862fee6f3c801759b980..6151ad2bbaccb09068f29d7f2ff6148e3d298a7d 100644 --- a/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/RRuntimeASTAccessImpl.java +++ b/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/RRuntimeASTAccessImpl.java @@ -38,8 +38,7 @@ import com.oracle.truffle.api.instrumentation.StandardTags.RootTag; import com.oracle.truffle.api.instrumentation.StandardTags.StatementTag; import com.oracle.truffle.api.nodes.Node; import com.oracle.truffle.api.nodes.RootNode; -import com.oracle.truffle.r.launcher.RCommand; -import com.oracle.truffle.r.launcher.RscriptCommand; +import com.oracle.truffle.r.launcher.RMain; import com.oracle.truffle.r.nodes.RASTUtils; import com.oracle.truffle.r.nodes.RRootNode; import com.oracle.truffle.r.nodes.access.variables.ReadVariableNode; @@ -353,7 +352,8 @@ class RRuntimeASTAccessImpl implements RRuntimeASTAccess { @Override public Object rcommandMain(String[] args, String[] env, boolean intern) { IORedirect redirect = handleIORedirect(args, intern); - Object result = RCommand.doMain(redirect.args, env, redirect.in, redirect.out, redirect.err); + assert env == null : "re-enable env arguments"; + int result = RMain.runR(redirect.args, redirect.in, redirect.out, redirect.err); return redirect.getInternResult(result); } @@ -362,7 +362,8 @@ class RRuntimeASTAccessImpl implements RRuntimeASTAccess { IORedirect redirect = handleIORedirect(args, intern); // TODO argument parsing can fail with ExitException, which needs to be handled correctly in // nested context - Object result = RscriptCommand.doMain(redirect.args, env, redirect.in, redirect.out, redirect.err); + assert env == null : "re-enable env arguments"; + int result = RMain.runRscript(redirect.args, redirect.in, redirect.out, redirect.err); return redirect.getInternResult(result); } @@ -381,9 +382,8 @@ class RRuntimeASTAccessImpl implements RRuntimeASTAccess { this.intern = intern; } - private Object getInternResult(Object result) { + private Object getInternResult(int result) { if (intern) { - int status = (int) result; ByteArrayOutputStream bos = (ByteArrayOutputStream) out; String s = new String(bos.toByteArray()); RStringVector sresult; @@ -393,8 +393,8 @@ class RRuntimeASTAccessImpl implements RRuntimeASTAccess { String[] lines = s.split("\n"); sresult = RDataFactory.createStringVector(lines, RDataFactory.COMPLETE_VECTOR); } - if (status != 0) { - setResultAttr(status, sresult); + if (result != 0) { + setResultAttr(result, sresult); } return sresult; } else { diff --git a/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/shell/REmbedded.java b/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/shell/REmbedded.java index 3b7eb5535210ed6c49e3009df37ab3e7cebd05c8..2b20fe5aa805ccd146c72bcb69f8669fdbb659c7 100644 --- a/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/shell/REmbedded.java +++ b/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/shell/REmbedded.java @@ -31,7 +31,7 @@ import org.graalvm.polyglot.Source; import com.oracle.truffle.r.launcher.ConsoleHandler; import com.oracle.truffle.r.launcher.RCmdOptions; -import com.oracle.truffle.r.launcher.RCommand; +import com.oracle.truffle.r.launcher.REPL; import com.oracle.truffle.r.launcher.RStartParams; import com.oracle.truffle.r.runtime.RSource.Internal; import com.oracle.truffle.r.runtime.RSuicide; @@ -76,7 +76,7 @@ public class REmbedded { EmbeddedConsoleHandler embeddedConsoleHandler = new EmbeddedConsoleHandler(); - consoleHandler = RCommand.createConsoleHandler(options, embeddedConsoleHandler, System.in, System.out); + consoleHandler = ConsoleHandler.createConsoleHandler(options, embeddedConsoleHandler, System.in, System.out); InputStream input = consoleHandler.createInputStream(); boolean useEmbedded = consoleHandler == embeddedConsoleHandler; OutputStream stdOut = useEmbedded ? embeddedConsoleHandler.createStdOutputStream(System.out) : System.out; @@ -133,7 +133,7 @@ public class REmbedded { RContext ctx = RContext.getInstance(); ctx.completeEmbeddedInitialization(); ctx.getRFFI().initializeEmbedded(ctx); - int status = RCommand.readEvalPrint(context, consoleHandler, false); + int status = REPL.readEvalPrint(context, consoleHandler, false); context.leave(); context.close(); Utils.systemExit(status); diff --git a/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/shell/RMain.java b/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/shell/RMain.java deleted file mode 100644 index 756460ef888d57204945037a9bb2ec535d07e235..0000000000000000000000000000000000000000 --- a/com.oracle.truffle.r.engine/src/com/oracle/truffle/r/engine/shell/RMain.java +++ /dev/null @@ -1,72 +0,0 @@ -/* - * Copyright (c) 2017, 2018, 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 3 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 3 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 - * 3 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.r.engine.shell; - -import com.oracle.truffle.r.launcher.RCommand; -import com.oracle.truffle.r.launcher.RscriptCommand; - -/** - * Convenience class that allows the R/Rscript entry to be chosen by an initial argument. - */ -public class RMain { - - public static void main(String[] args) { - boolean rscript = false; - if (args.length > 0) { - String arg = args[0]; - switch (arg) { - case "R": - case "r": - break; - - case "Rscript": - case "rscript": - rscript = true; - break; - - default: - System.out.println(arg); - usage(); - } - String[] xargs = shiftArgs(args); - if (rscript) { - RscriptCommand.main(xargs); - } else { - RCommand.main(xargs); - } - } else { - usage(); - } - } - - private static void usage() { - System.out.println("usage: [R|Rscript] ..."); - System.exit(1); - } - - private static String[] shiftArgs(String[] args) { - String[] nargs = new String[args.length - 1]; - System.arraycopy(args, 1, nargs, 0, nargs.length); - return nargs; - } -} diff --git a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/ConsoleHandler.java b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/ConsoleHandler.java index 1c8751f429813ce950212c512f21515d182da04f..9147fe22f8851f65924adf1041fd1f3b5d509f6c 100644 --- a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/ConsoleHandler.java +++ b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/ConsoleHandler.java @@ -22,21 +22,81 @@ */ package com.oracle.truffle.r.launcher; +import java.io.File; import java.io.IOException; import java.io.InputStream; +import java.io.OutputStream; import java.nio.charset.StandardCharsets; +import java.nio.file.Files; +import java.util.List; +import java.util.function.Supplier; import org.graalvm.polyglot.Context; import org.graalvm.polyglot.Value; import org.graalvm.polyglot.proxy.ProxyExecutable; import org.graalvm.polyglot.proxy.ProxyObject; +import com.oracle.truffle.r.launcher.RCmdOptions.RCmdOption; + /** * The interface to a source of input/output for the context, which may have different * implementations for different contexts. */ public abstract class ConsoleHandler { + public static ConsoleHandler createConsoleHandler(RCmdOptions options, DelegatingConsoleHandler useDelegatingWrapper, InputStream inStream, OutputStream outStream) { + /* + * Whether the input is from stdin, a file (-f), or an expression on the command line (-e) + * it goes through the console. N.B. -f and -e can't be used together and this is already + * checked. + */ + RStartParams rsp = new RStartParams(options, false); + String fileArgument = rsp.getFileArgument(); + if (fileArgument != null) { + List<String> lines; + try { + /* + * If initial==false, ~ expansion will not have been done and the open will fail. + * It's harmless to always do it. + */ + File file = fileArgument.startsWith("~") ? new File(System.getProperty("user.home") + fileArgument.substring(1)) : new File(fileArgument); + lines = Files.readAllLines(file.toPath()); + } catch (IOException e) { + throw RMain.fatal("cannot open file '%s': No such file or directory", fileArgument); + } + return new StringConsoleHandler(lines, outStream); + } else if (options.getStringList(RCmdOption.EXPR) != null) { + List<String> exprs = options.getStringList(RCmdOption.EXPR); + for (int i = 0; i < exprs.size(); i++) { + exprs.set(i, REPL.unescapeSpace(exprs.get(i))); + } + return new StringConsoleHandler(exprs, outStream); + } else { + boolean isInteractive = options.getBoolean(RCmdOption.INTERACTIVE); + if (!isInteractive && rsp.askForSave()) { + throw RMain.fatal("you must specify '--save', '--no-save' or '--vanilla'"); + } + boolean useReadLine = isInteractive && !rsp.noReadline(); + if (useDelegatingWrapper != null) { + /* + * If we are in embedded mode, the creation of ConsoleReader and the ConsoleHandler + * should be lazy, as these may not be necessary and can cause hangs if stdin has + * been redirected. + */ + Supplier<ConsoleHandler> delegateFactory = useReadLine ? () -> new JLineConsoleHandler(inStream, outStream, rsp.isSlave()) + : () -> new DefaultConsoleHandler(inStream, outStream, isInteractive); + useDelegatingWrapper.setDelegate(delegateFactory); + return useDelegatingWrapper; + } else { + if (useReadLine) { + return new JLineConsoleHandler(inStream, outStream, rsp.isSlave()); + } else { + return new DefaultConsoleHandler(inStream, outStream, isInteractive); + } + } + } + } + /** * Read a line of input, newline is <b>NOT</b> included in result. */ @@ -92,10 +152,10 @@ public abstract class ConsoleHandler { }; } - private static class PolyglotWrapper implements ProxyObject { + private static final class PolyglotWrapper implements ProxyObject { private static final String GET_PROMPT_KEY = "getPrompt"; private static final String SET_PROMPT_KEY = "setPrompt"; - private static final String[] keys = new String[] { GET_PROMPT_KEY, SET_PROMPT_KEY }; + private static final String[] keys = new String[]{GET_PROMPT_KEY, SET_PROMPT_KEY}; private final GetPromptWrapper getPrompt; private final SetPromptWrapper setPrompt; @@ -131,7 +191,7 @@ public abstract class ConsoleHandler { } } - private static class GetPromptWrapper implements ProxyExecutable { + private static final class GetPromptWrapper implements ProxyExecutable { private final ConsoleHandler target; private GetPromptWrapper(ConsoleHandler target) { @@ -144,7 +204,7 @@ public abstract class ConsoleHandler { } } - private static class SetPromptWrapper implements ProxyExecutable { + private static final class SetPromptWrapper implements ProxyExecutable { private final ConsoleHandler target; private SetPromptWrapper(ConsoleHandler target) { diff --git a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/JLineConsoleCompleter.java b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/JLineConsoleCompleter.java index d2bad321f9ef438516d096a184c9bdb58366cf85..234a6e2ac73cf16380723116ebea87e1b5c456b9 100644 --- a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/JLineConsoleCompleter.java +++ b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/JLineConsoleCompleter.java @@ -53,7 +53,7 @@ public class JLineConsoleCompleter implements Completer { if (isTesting) { throw e; } - throw RCommand.fatal(e, "error while determining completion"); + throw RMain.fatal(e, "error while determining completion"); } } diff --git a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/JLineConsoleHandler.java b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/JLineConsoleHandler.java index bf6fc6c9b54242ecae2f001062490f0231322069..85aaaebd722fe6f3c4201979e3361afce422094f 100644 --- a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/JLineConsoleHandler.java +++ b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/JLineConsoleHandler.java @@ -49,7 +49,7 @@ public class JLineConsoleHandler extends ConsoleHandler { console.setHandleUserInterrupt(true); console.setExpandEvents(false); } catch (IOException ex) { - throw RCommand.fatal(ex, "unexpected error opening console reader"); + throw RMain.fatal(ex, "unexpected error opening console reader"); } } diff --git a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RAbstractLauncher.java b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RAbstractLauncher.java deleted file mode 100644 index e3e043fe7ea9ef132bdb0022d8badf2af08aebc2..0000000000000000000000000000000000000000 --- a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RAbstractLauncher.java +++ /dev/null @@ -1,200 +0,0 @@ -/* - * Copyright (c) 2018, 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 3 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 3 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 - * 3 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.r.launcher; - -import java.io.Closeable; -import java.io.InputStream; -import java.io.OutputStream; -import java.util.ArrayList; -import java.util.List; -import java.util.Map; -import java.util.Set; - -import org.graalvm.launcher.AbstractLanguageLauncher; -import org.graalvm.options.OptionCategory; -import org.graalvm.polyglot.Context; -import org.graalvm.polyglot.Context.Builder; - -import com.oracle.truffle.r.launcher.RCmdOptions.Client; -import com.oracle.truffle.r.launcher.RCmdOptions.RCmdOption; -import static com.oracle.truffle.r.launcher.RCommand.fatal; -import java.io.File; -import java.io.IOException; -import org.graalvm.polyglot.Source; - -public abstract class RAbstractLauncher extends AbstractLanguageLauncher implements Closeable { - - private final Client client; - protected final InputStream inStream; - protected final OutputStream outStream; - protected final OutputStream errStream; - protected RCmdOptions options; - private boolean useJVM; - protected ConsoleHandler consoleHandler; - protected Context context; - - RAbstractLauncher(Client client, String[] env, InputStream inStream, OutputStream outStream, OutputStream errStream) { - this.client = client; - assert env == null : "re-enble environment variables"; - this.inStream = inStream; - this.outStream = outStream; - this.errStream = errStream; - } - - @Override - protected List<String> preprocessArguments(List<String> arguments, Map<String, String> polyglotOptions) { - boolean[] recognizedArgsIndices = new boolean[arguments.size()]; - this.options = RCmdOptions.parseArguments(client, arguments.toArray(new String[arguments.size()]), true, recognizedArgsIndices); - List<String> unrecognizedArgs = new ArrayList<>(); - for (int i = 0; i < arguments.size(); i++) { - if ("--jvm.help".equals(arguments.get(i))) { - // This condition should be removed when the Launcher handles --jvm.help - // correctly. - printJvmHelp(); - throw exit(); - } else if ("--jvm".equals(arguments.get(i))) { - useJVM = true; - } else if (!recognizedArgsIndices[i]) { - unrecognizedArgs.add(arguments.get(i)); - } - } - - return unrecognizedArgs; - } - - protected abstract String[] getArguments(); - - @Override - protected void launch(Builder contextBuilder) { - this.consoleHandler = RCommand.createConsoleHandler(options, null, inStream, outStream); - this.context = contextBuilder.allowAllAccess(true).allowHostAccess(useJVM).arguments("R", getArguments()).in(consoleHandler.createInputStream()).out( - outStream).err(errStream).build(); - this.consoleHandler.setContext(context); - try { - Source src = Source.newBuilder("R", ".fastr.set.consoleHandler", "<set-console-handler>").internal(true).build(); - context.eval(src).execute(consoleHandler.getPolyglotWrapper()); - } catch (IOException e) { - throw fatal(e, "error while setting console handler"); - } - } - - @Override - protected String getLanguageId() { - return "R"; - } - - @Override - protected void printHelp(OptionCategory maxCategory) { - RCmdOptions.printHelp(client); - } - - @Override - protected void collectArguments(Set<String> opts) { - for (RCmdOption option : RCmdOption.values()) { - if (option.shortName != null) { - opts.add(option.shortName); - } - if (option.plainName != null) { - opts.add(option.plainName); - } - } - } - - @Override - protected String[] getDefaultLanguages() { - if ("llvm".equals(System.getenv("FASTR_RFFI"))) { - return new String[]{getLanguageId(), "llvm"}; - } - return super.getDefaultLanguages(); - } - - @Override - public void close() { - if (context != null) { - context.close(); - } - } - - // The following code is copied from org.graalvm.launcher.Launcher and it should be removed - // when the Launcher handles --jvm.help correctly. - - private static void printJvmHelp() { - System.out.print("JVM options:"); - printOption("--jvm.classpath <...>", "A " + File.pathSeparator + " separated list of classpath entries that will be added to the JVM's classpath"); - printOption("--jvm.D<name>=<value>", "Set a system property"); - printOption("--jvm.esa", "Enable system assertions"); - printOption("--jvm.ea[:<packagename>...|:<classname>]", "Enable assertions with specified granularity"); - printOption("--jvm.agentlib:<libname>[=<options>]", "Load native agent library <libname>"); - printOption("--jvm.agentpath:<pathname>[=<options>]", "Load native agent library by full pathname"); - printOption("--jvm.javaagent:<jarpath>[=<options>]", "Load Java programming language agent"); - printOption("--jvm.Xbootclasspath/a:<...>", "A " + File.pathSeparator + " separated list of classpath entries that will be added to the JVM's boot classpath"); - printOption("--jvm.Xmx<size>", "Set maximum Java heap size"); - printOption("--jvm.Xms<size>", "Set initial Java heap size"); - printOption("--jvm.Xss<size>", "Set java thread stack size"); - } - - private static void printOption(String option, String description, int indentation) { - String indent = spaces(indentation); - String desc = wrap(description != null ? description : ""); - String nl = System.lineSeparator(); - String[] descLines = desc.split(nl); - int optionWidth = 45; - if (option.length() >= optionWidth && description != null) { - System.out.println(indent + option + nl + indent + spaces(optionWidth) + descLines[0]); - } else { - System.out.println(indent + option + spaces(optionWidth - option.length()) + descLines[0]); - } - for (int i = 1; i < descLines.length; i++) { - System.out.println(indent + spaces(optionWidth) + descLines[i]); - } - } - - static void printOption(String option, String description) { - printOption(option, description, 2); - } - - private static String spaces(int length) { - return new String(new char[length]).replace('\0', ' '); - } - - private static String wrap(String s) { - final int width = 120; - StringBuilder sb = new StringBuilder(s); - int cursor = 0; - while (cursor + width < sb.length()) { - int i = sb.lastIndexOf(" ", cursor + width); - if (i == -1 || i < cursor) { - i = sb.indexOf(" ", cursor + width); - } - if (i != -1) { - sb.replace(i, i + 1, System.lineSeparator()); - cursor = i; - } else { - break; - } - } - return sb.toString(); - } - - // End of copied code -} diff --git a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RCmdOptions.java b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RCmdOptions.java index e9842604ba493260456bbadfb53592923db0d5e2..2ddb715ef2010bc3aa7779c004d5e5b414463d38 100644 --- a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RCmdOptions.java +++ b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RCmdOptions.java @@ -50,7 +50,7 @@ public final class RCmdOptions { EITHER { @Override public String usage() { - throw RCommand.fatal("can't call usage() on Client.EITHER"); + throw RMain.fatal("can't call usage() on Client.EITHER"); } }; @@ -194,6 +194,21 @@ public final class RCmdOptions { } } + public void addInteractive() { + setValue(RCmdOption.INTERACTIVE, true); + if (arguments == null || arguments.length == 0) { + arguments = new String[]{"--interactive"}; + } else { + String[] oldArgs = arguments; + arguments = new String[arguments.length + 1]; + arguments[0] = oldArgs[0]; + arguments[1] = "--interactive"; + for (int i = 1; i < oldArgs.length; i++) { + arguments[i + 1] = oldArgs[i]; + } + } + } + public void setValue(RCmdOption option, boolean value) { setValue(optionValues, option, value); } diff --git a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RCommand.java b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/REPL.java similarity index 62% rename from com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RCommand.java rename to com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/REPL.java index 261cc14678eefddc32b9aca2a248dddf3f6949a5..9be847012a900478db027bcba5734ddfb0eb1d16 100644 --- a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RCommand.java +++ b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/REPL.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2013, 2018, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2018, 2018, 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 @@ -26,181 +26,22 @@ import java.io.EOFException; import java.io.File; import java.io.FileInputStream; import java.io.IOException; -import java.io.InputStream; -import java.io.OutputStream; -import java.nio.file.Files; -import java.nio.file.Path; -import java.util.ArrayList; -import java.util.Arrays; -import java.util.List; import java.util.concurrent.ExecutionException; import java.util.concurrent.ExecutorService; -import java.util.function.Supplier; import org.graalvm.polyglot.Context; -import org.graalvm.polyglot.Context.Builder; import org.graalvm.polyglot.PolyglotException; import org.graalvm.polyglot.Source; - -import com.oracle.truffle.r.launcher.RCmdOptions.Client; -import com.oracle.truffle.r.launcher.RCmdOptions.RCmdOption; +import org.graalvm.polyglot.Value; import jline.console.UserInterruptException; -/* - * TODO: - * - * - create a replacement for "executor" - * - * - fatal needs to be handled more carefully in nested/spawned contexts - */ - /** - * Emulates the (Gnu)R command as precisely as possible. + * Implements the read-eval-print loop. + * + * @see #readEvalPrint(Context, ConsoleHandler, File, boolean) */ -public class RCommand extends RAbstractLauncher { - - RCommand(String[] env, InputStream inStream, OutputStream outStream, OutputStream errStream) { - super(Client.R, env, inStream, outStream, errStream); - } - - @Override - protected String[] getArguments() { - return options.getArguments(); - } - - @Override - protected void launch(Builder contextBuilder) { - super.launch(contextBuilder); - StartupTiming.timestamp("VM Created"); - StartupTiming.printSummary(); - } - - private int execute(String[] args) { - StartupTiming.timestamp("Main Entered"); - ArrayList<String> argsList = new ArrayList<>(Arrays.asList(args)); - if (System.console() != null) { - // add "--interactive" to force interactive mode - boolean addArg = true; - for (String arg : args) { - if ("--interactive".equals(arg)) { - addArg = false; - break; - } - } - if (addArg) { - if (argsList.size() == 0) { - argsList.add("--interactive"); - } else { - argsList.add(1, "--interactive"); - } - } - } - - launch(argsList.toArray(new String[0])); - if (context != null) { - File srcFile = null; - String fileOption = options.getString(RCmdOption.FILE); - if (fileOption != null) { - srcFile = new File(fileOption); - } - - return readEvalPrint(context, consoleHandler, srcFile, true); - } else { - return 0; - } - } - - // CheckStyle: stop system..print check - public static RuntimeException fatal(String message, Object... args) { - System.out.println("FATAL: " + String.format(message, args)); - System.exit(-1); - return new RuntimeException(); - } - - public static RuntimeException fatal(Throwable t, String message, Object... args) { - t.printStackTrace(); - System.out.println("FATAL: " + String.format(message, args)); - System.exit(-1); - return null; - } - - static String[] prependCommand(String[] args) { - String[] result = new String[args.length + 1]; - result[0] = "R"; - System.arraycopy(args, 0, result, 1, args.length); - return result; - } - - public static void main(String[] args) { - try { - System.exit(doMain(prependCommand(args), null, System.in, System.out, System.err)); - // never returns - throw fatal("main should never return"); - } catch (Throwable t) { - throw fatal(t, "error during REPL execution"); - } - } - - public static int doMain(String[] args, String[] env, InputStream inStream, OutputStream outStream, OutputStream errStream) { - try (RCommand rcmd = new RCommand(env, inStream, outStream, errStream)) { - return rcmd.execute(args); - } - } - - public static ConsoleHandler createConsoleHandler(RCmdOptions options, DelegatingConsoleHandler useDelegatingWrapper, InputStream inStream, OutputStream outStream) { - /* - * Whether the input is from stdin, a file (-f), or an expression on the command line (-e) - * it goes through the console. N.B. -f and -e can't be used together and this is already - * checked. - */ - RStartParams rsp = new RStartParams(options, false); - String fileArgument = rsp.getFileArgument(); - if (fileArgument != null) { - List<String> lines; - try { - /* - * If initial==false, ~ expansion will not have been done and the open will fail. - * It's harmless to always do it. - */ - File file = fileArgument.startsWith("~") ? new File(System.getProperty("user.home") + fileArgument.substring(1)) : new File(fileArgument); - lines = Files.readAllLines(file.toPath()); - } catch (IOException e) { - throw fatal("cannot open file '%s': No such file or directory", fileArgument); - } - return new StringConsoleHandler(lines, outStream); - } else if (options.getStringList(RCmdOption.EXPR) != null) { - List<String> exprs = options.getStringList(RCmdOption.EXPR); - for (int i = 0; i < exprs.size(); i++) { - exprs.set(i, unescapeSpace(exprs.get(i))); - } - return new StringConsoleHandler(exprs, outStream); - } else { - boolean isInteractive = options.getBoolean(RCmdOption.INTERACTIVE); - if (!isInteractive && rsp.askForSave()) { - throw fatal("you must specify '--save', '--no-save' or '--vanilla'"); - } - boolean useReadLine = isInteractive && !rsp.noReadline(); - if (useDelegatingWrapper != null) { - /* - * If we are in embedded mode, the creation of ConsoleReader and the ConsoleHandler - * should be lazy, as these may not be necessary and can cause hangs if stdin has - * been redirected. - */ - Supplier<ConsoleHandler> delegateFactory = useReadLine ? () -> new JLineConsoleHandler(inStream, outStream, rsp.isSlave()) - : () -> new DefaultConsoleHandler(inStream, outStream, isInteractive); - useDelegatingWrapper.setDelegate(delegateFactory); - return useDelegatingWrapper; - } else { - if (useReadLine) { - return new JLineConsoleHandler(inStream, outStream, rsp.isSlave()); - } else { - return new DefaultConsoleHandler(inStream, outStream, isInteractive); - } - } - } - } - +public class REPL { /** * The standard R script escapes spaces to "~+~" in "-e" and "-f" commands. */ @@ -275,7 +116,7 @@ public class RCommand extends RAbstractLauncher { context.eval(src); } } catch (InterruptedException ex) { - throw fatal("Unexpected interrup error"); + throw RMain.fatal("Unexpected interrup error"); } catch (PolyglotException e) { if (e.isIncompleteSource()) { if (continuePrompt == null) { @@ -314,7 +155,7 @@ public class RCommand extends RAbstractLauncher { } else if (e2.isCancelled()) { continue; } - throw fatal(e, "error while calling quit"); + throw RMain.fatal(e, "error while calling quit"); } } catch (UserInterruptException e) { // interrupted by ctrl-c @@ -374,7 +215,7 @@ public class RCommand extends RAbstractLauncher { } } - private static boolean doEcho(ExecutorService executor, Context context) throws InterruptedException, ExecutionException { + private static boolean doEcho(ExecutorService executor, Context context) throws InterruptedException { try { if (executor != null) { return executor.submit(() -> doEcho(context)).get(); @@ -382,12 +223,12 @@ public class RCommand extends RAbstractLauncher { return doEcho(context); } } catch (ExecutionException ex) { - throw fatal(ex, "error while retrieving echo"); + throw RMain.fatal(ex, "error while retrieving echo"); } catch (PolyglotException e) { if (e.isExit()) { throw new ExitException(e.getExitStatus()); } - throw fatal(e, "error while retrieving echo"); + throw RMain.fatal(e, "error while retrieving echo"); } } @@ -403,12 +244,12 @@ public class RCommand extends RAbstractLauncher { return getPrompt(context); } } catch (ExecutionException ex) { - throw fatal(ex, "error while retrieving prompt"); + throw RMain.fatal(ex, "error while retrieving prompt"); } catch (PolyglotException e) { if (e.isExit()) { throw new ExitException(e.getExitStatus()); } - throw fatal(e, "error while retrieving prompt"); + throw RMain.fatal(e, "error while retrieving prompt"); } } @@ -424,12 +265,12 @@ public class RCommand extends RAbstractLauncher { return getPrompt(context); } } catch (ExecutionException ex) { - throw fatal(ex, "error while retrieving continue prompt"); + throw RMain.fatal(ex, "error while retrieving continue prompt"); } catch (PolyglotException e) { if (e.isExit()) { throw new ExitException(e.getExitStatus()); } - throw fatal(e, "error while retrieving continue prompt"); + throw RMain.fatal(e, "error while retrieving continue prompt"); } } diff --git a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RMain.java b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RMain.java new file mode 100644 index 0000000000000000000000000000000000000000..2e0af3afcd0e7712bcb00f863a01da3c18bc0084 --- /dev/null +++ b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RMain.java @@ -0,0 +1,410 @@ +/* + * Copyright (c) 2018, 2018, 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 3 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 3 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 + * 3 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.r.launcher; + +import java.io.Closeable; +import java.io.File; +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.util.ArrayList; +import java.util.List; +import java.util.Map; +import java.util.Set; + +import org.graalvm.launcher.AbstractLanguageLauncher; +import org.graalvm.options.OptionCategory; +import org.graalvm.polyglot.Context; +import org.graalvm.polyglot.Context.Builder; +import org.graalvm.polyglot.PolyglotException; +import org.graalvm.polyglot.Source; + +import com.oracle.truffle.r.launcher.RCmdOptions.Client; +import com.oracle.truffle.r.launcher.RCmdOptions.RCmdOption; + +/** + * Main entry point for the R engine. The first argument must be either {@code 'R'} or + * {@code 'Rscript'} and decides which of the two R commands will be run. + */ +public final class RMain extends AbstractLanguageLauncher implements Closeable { + + public static void main(String[] args) { + new RMain().launch(args); + } + + public static int runR(String[] args, InputStream inStream, OutputStream outStream, OutputStream errStream) { + return runROrRScript("R", args, inStream, outStream, errStream); + } + + public static int runRscript(String[] args, InputStream inStream, OutputStream outStream, OutputStream errStream) { + return runROrRScript("Rscript", args, inStream, outStream, errStream); + } + + private static int runROrRScript(String command, String[] args, InputStream inStream, OutputStream outStream, OutputStream errStream) { + String[] newArgs = new String[args.length + 1]; + System.arraycopy(args, 0, newArgs, 1, args.length); + newArgs[0] = command; + try (RMain cmd = new RMain(false, inStream, outStream, errStream)) { + cmd.launch(newArgs); + return cmd.execute(); + } + } + + /** + * Tells this R launcher to not process the {@code --jvm} and {@code --jvm.help}. Normally such + * arguments are processed by the native launcher, but if there is no native launcher, we need + * to explicitly process them in this class since the Truffle launcher does not count on this + * eventuality. + */ + private static final boolean ignoreJvmArguments = "true".equals(System.getProperty("fastr.internal.ignorejvmargs")); + + protected final InputStream inStream; + protected final OutputStream outStream; + protected final OutputStream errStream; + + /** + * In launcher mode {@link #launch(String[])} runs the command and uses {@link System#exit(int)} + * to terminate and return the status. In non-launcher mode, {@link #launch(String[])} only + * prepares the {@link Context} and then {@link #execute()} executes the command itself and + * returns the status. + */ + protected final boolean launcherMode; + + private Client client; + private RCmdOptions options; + private ConsoleHandler consoleHandler; + private String[] rArguments; + private boolean useJVM; + private Context preparedContext; // to transfer between launch and execute when !launcherMode + + private RMain(boolean launcherMode, InputStream inStream, OutputStream outStream, OutputStream errStream) { + this.launcherMode = launcherMode; + this.inStream = inStream; + this.outStream = outStream; + this.errStream = errStream; + } + + private RMain() { + this.launcherMode = true; + this.inStream = System.in; + this.outStream = System.out; + this.errStream = System.err; + } + + @Override + protected List<String> preprocessArguments(List<String> arguments, Map<String, String> polyglotOptions) { + String clientName = arguments.size() > 0 ? arguments.get(0).trim() : null; + if ("R".equals(clientName)) { + client = Client.R; + } else if ("Rscript".equals(clientName)) { + client = Client.RSCRIPT; + } else { + System.err.printf("RMain: the first argument must be either 'R' or 'Rscript'. Given was '%s'.\n", clientName); + System.err.println("If you did not run RMain class explicitly, then this is a bug in launcher script, please report it at http://github.com/oracle/fastr."); + if (launcherMode) { + System.exit(1); + } + return arguments; + } + boolean[] recognizedArgsIndices = new boolean[arguments.size()]; + options = RCmdOptions.parseArguments(client, arguments.toArray(new String[arguments.size()]), true, recognizedArgsIndices); + if (System.console() != null && client == Client.R) { + options.addInteractive(); + } + List<String> unrecognizedArgs = new ArrayList<>(); + for (int i = 0; i < arguments.size(); i++) { + if (!ignoreJvmArguments && "--jvm.help".equals(arguments.get(i))) { + // This condition should be removed when FastR always ships with native launcher + // that handles this option for us + printJvmHelp(); + throw exit(); + } else if (!ignoreJvmArguments && "--jvm".equals(arguments.get(i))) { + useJVM = true; + } else + + if (!recognizedArgsIndices[i]) { + unrecognizedArgs.add(arguments.get(i)); + } + } + return unrecognizedArgs; + } + + @Override + protected void validateArguments(Map<String, String> polyglotOptions) { + if (options == null) { + // preprocessArguments did not set the value + return; + } + if (client == Client.RSCRIPT) { + try { + rArguments = preprocessRScriptOptions(options); + } catch (PrintHelp e) { + printHelp(OptionCategory.USER); + } + } else { + rArguments = this.options.getArguments(); + } + } + + @Override + protected void launch(Builder contextBuilder) { + assert client != null; + if (rArguments == null) { + // validateArguments did not set the value + return; + } + this.consoleHandler = ConsoleHandler.createConsoleHandler(options, null, inStream, outStream); + Builder contextBuilderAllowAll = contextBuilder.allowAllAccess(true); + if (ignoreJvmArguments) { + contextBuilderAllowAll = contextBuilderAllowAll.allowHostAccess(useJVM); + } + Context context = preparedContext = contextBuilderAllowAll.arguments("R", rArguments).in(consoleHandler.createInputStream()).out(outStream).err(errStream).build(); + this.consoleHandler.setContext(context); + try { + Source src = Source.newBuilder("R", ".fastr.set.consoleHandler", "<set-console-handler>").internal(true).build(); + context.eval(src).execute(consoleHandler.getPolyglotWrapper()); + } catch (IOException e) { + throw fatal(e, "error while setting console handler"); + } + if (launcherMode) { + try { + System.exit(execute(context)); + } finally { + context.close(); + } + } + } + + protected int execute() { + if (preparedContext == null) { + // launch did not set the value + return 1; + } + return execute(preparedContext); + } + + protected int execute(Context context) { + String fileOption = options.getString(RCmdOption.FILE); + File srcFile = null; + if (fileOption != null) { + if (client == Client.RSCRIPT) { + return executeFile(context, fileOption); + } + srcFile = new File(fileOption); + } + return REPL.readEvalPrint(context, consoleHandler, srcFile, true); + } + + @Override + protected String getLanguageId() { + return "R"; + } + + @Override + protected void printHelp(OptionCategory maxCategory) { + assert client != null; + RCmdOptions.printHelp(client); + } + + @Override + protected void collectArguments(Set<String> opts) { + for (RCmdOption option : RCmdOption.values()) { + if (option.shortName != null) { + opts.add(option.shortName); + } + if (option.plainName != null) { + opts.add(option.plainName); + } + } + } + + @Override + protected String[] getDefaultLanguages() { + if ("llvm".equals(System.getenv("FASTR_RFFI"))) { + return new String[]{getLanguageId(), "llvm"}; + } + return super.getDefaultLanguages(); + } + + @Override + public void close() { + if (preparedContext != null) { + preparedContext.close(); + } + } + + private static int executeFile(Context context, String fileOption) { + Source src; + try { + src = Source.newBuilder("R", new File(fileOption)).interactive(false).build(); + } catch (IOException ex) { + System.err.printf("IO error while reading the source file '%s'.\nDetails: '%s'.", fileOption, ex.getLocalizedMessage()); + return 1; + } + try { + context.eval(src); + return 0; + } catch (Throwable ex) { + if (ex instanceof PolyglotException && ((PolyglotException) ex).isExit()) { + return ((PolyglotException) ex).getExitStatus(); + } + // Internal exceptions are reported by the engine already + return 1; + } + } + + // CheckStyle: stop system..print check + public static RuntimeException fatal(String message, Object... args) { + System.out.println("FATAL: " + String.format(message, args)); + System.exit(-1); + return new RuntimeException(); + } + + public static RuntimeException fatal(Throwable t, String message, Object... args) { + t.printStackTrace(); + System.out.println("FATAL: " + String.format(message, args)); + System.exit(-1); + return null; + } + + // CheckStyle: stop system..print check + + private static String[] preprocessRScriptOptions(RCmdOptions options) throws PrintHelp { + String[] arguments = options.getArguments(); + int resultArgsLength = arguments.length; + int firstNonOptionArgIndex = options.getFirstNonOptionArgIndex(); + // Now reformat the args, setting --slave and --no-restore as per the spec + ArrayList<String> adjArgs = new ArrayList<>(resultArgsLength + 1); + adjArgs.add(arguments[0]); + adjArgs.add("--slave"); + options.setValue(RCmdOption.SLAVE, true); + adjArgs.add("--no-restore"); + options.setValue(RCmdOption.NO_RESTORE, true); + // Either -e options are set or first non-option arg is a file + if (options.getStringList(RCmdOption.EXPR) == null) { + if (firstNonOptionArgIndex == resultArgsLength) { + throw new PrintHelp(); + } else { + options.setValue(RCmdOption.FILE, arguments[firstNonOptionArgIndex]); + } + } + String defaultPackagesArg = options.getString(RCmdOption.DEFAULT_PACKAGES); + String defaultPackagesEnv = System.getenv("R_DEFAULT_PACKAGES"); + if (defaultPackagesArg == null && defaultPackagesEnv == null) { + defaultPackagesArg = "datasets,utils,grDevices,graphics,stats"; + } + if (defaultPackagesEnv == null) { + options.setValue(RCmdOption.DEFAULT_PACKAGES, defaultPackagesArg); + } + // copy up to non-option args + int rx = 1; + while (rx < firstNonOptionArgIndex) { + adjArgs.add(arguments[rx]); + rx++; + } + if (options.getString(RCmdOption.FILE) != null) { + adjArgs.add("--file=" + options.getString(RCmdOption.FILE)); + rx++; // skip over file arg + firstNonOptionArgIndex++; + } + + if (firstNonOptionArgIndex < resultArgsLength) { + adjArgs.add("--args"); + while (rx < resultArgsLength) { + adjArgs.add(arguments[rx++]); + } + } + return adjArgs.toArray(new String[adjArgs.size()]); + } + + @SuppressWarnings("serial") + static class PrintHelp extends Exception { + @Override + public synchronized Throwable fillInStackTrace() { + return this; + } + } + + // The following code is copied from org.graalvm.launcher.Launcher and it should be removed + // when the R launcher always ships native version that handles --jvm.help for us. + + private static void printJvmHelp() { + System.out.println("JVM options:"); + printOption("--jvm.classpath <...>", "A " + File.pathSeparator + " separated list of classpath entries that will be added to the JVM's classpath"); + printOption("--jvm.D<name>=<value>", "Set a system property"); + printOption("--jvm.esa", "Enable system assertions"); + printOption("--jvm.ea[:<packagename>...|:<classname>]", "Enable assertions with specified granularity"); + printOption("--jvm.agentlib:<libname>[=<options>]", "Load native agent library <libname>"); + printOption("--jvm.agentpath:<pathname>[=<options>]", "Load native agent library by full pathname"); + printOption("--jvm.javaagent:<jarpath>[=<options>]", "Load Java programming language agent"); + printOption("--jvm.Xbootclasspath/a:<...>", "A " + File.pathSeparator + " separated list of classpath entries that will be added to the JVM's boot classpath"); + printOption("--jvm.Xmx<size>", "Set maximum Java heap size"); + printOption("--jvm.Xms<size>", "Set initial Java heap size"); + printOption("--jvm.Xss<size>", "Set java thread stack size"); + } + + private static void printOption(String option, String description, int indentation) { + String indent = spaces(indentation); + String desc = wrap(description != null ? description : ""); + String nl = System.lineSeparator(); + String[] descLines = desc.split(nl); + int optionWidth = 45; + if (option.length() >= optionWidth && description != null) { + System.out.println(indent + option + nl + indent + spaces(optionWidth) + descLines[0]); + } else { + System.out.println(indent + option + spaces(optionWidth - option.length()) + descLines[0]); + } + for (int i = 1; i < descLines.length; i++) { + System.out.println(indent + spaces(optionWidth) + descLines[i]); + } + } + + static void printOption(String option, String description) { + printOption(option, description, 2); + } + + private static String spaces(int length) { + return new String(new char[length]).replace('\0', ' '); + } + + private static String wrap(String s) { + final int width = 120; + StringBuilder sb = new StringBuilder(s); + int cursor = 0; + while (cursor + width < sb.length()) { + int i = sb.lastIndexOf(" ", cursor + width); + if (i == -1 || i < cursor) { + i = sb.indexOf(" ", cursor + width); + } + if (i != -1) { + sb.replace(i, i + 1, System.lineSeparator()); + cursor = i; + } else { + break; + } + } + return sb.toString(); + } + + // End of copied code +} diff --git a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RStartParams.java b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RStartParams.java index fba99e9eff339554c6369c391cd9873efc78dbd2..20e36f864816a3a6996386e0eea19315c3092dda 100644 --- a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RStartParams.java +++ b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RStartParams.java @@ -97,7 +97,7 @@ public class RStartParams { List<String> expressions = options.getStringList(EXPR); if (file != null) { if (expressions != null) { - throw RCommand.fatal("cannot use -e with -f or --file"); + throw RMain.fatal("cannot use -e with -f or --file"); } this.askForSave = false; this.save = false; @@ -105,7 +105,7 @@ public class RStartParams { // means stdin, but still implies NO_SAVE file = null; } else { - file = RCommand.unescapeSpace(file); + file = REPL.unescapeSpace(file); } this.interactive = false; } else if (expressions != null) { diff --git a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RscriptCommand.java b/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RscriptCommand.java deleted file mode 100644 index 2f3915b32c31bc311a92ee1c3dd7ee9fc1b8c05c..0000000000000000000000000000000000000000 --- a/com.oracle.truffle.r.launcher/src/com/oracle/truffle/r/launcher/RscriptCommand.java +++ /dev/null @@ -1,165 +0,0 @@ -/* - * Copyright (c) 2013, 2018, 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 3 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 3 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 - * 3 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.r.launcher; - -import java.io.File; -import java.io.IOException; -import java.io.InputStream; -import java.io.OutputStream; -import java.util.ArrayList; -import java.util.Map; - -import org.graalvm.options.OptionCategory; -import org.graalvm.polyglot.PolyglotException; -import org.graalvm.polyglot.Source; - -import com.oracle.truffle.r.launcher.RCmdOptions.Client; -import com.oracle.truffle.r.launcher.RCmdOptions.RCmdOption; - -/** - * Emulates the (Gnu)Rscript command as precisely as possible. in GnuR, Rscript is a genuine wrapper - * to R, as evidenced by the script {@code print(commandArgs())}. We don't implement it quite that - * way but the effect is similar. - * - */ -public final class RscriptCommand extends RAbstractLauncher { - - private String[] rScriptArguments; - - RscriptCommand(String[] env, InputStream inStream, OutputStream outStream, OutputStream errStream) { - super(Client.RSCRIPT, env, inStream, outStream, errStream); - } - - @Override - protected void validateArguments(Map<String, String> polyglotOptions) { - try { - this.rScriptArguments = preprocessRScriptOptions(options); - } catch (PrintHelp e) { - printHelp(OptionCategory.USER); - } - } - - @Override - protected String[] getArguments() { - return rScriptArguments; - } - - protected int execute(String[] args) { - launch(args); - if (context != null) { - String fileOption = options.getString(RCmdOption.FILE); - if (fileOption != null) { - return executeFile(fileOption); - } else { - return RCommand.readEvalPrint(context, consoleHandler, null, true); - } - } else { - return 0; - } - } - - private int executeFile(String fileOption) { - Source src; - try { - src = Source.newBuilder("R", new File(fileOption)).interactive(false).build(); - } catch (IOException ex) { - System.err.printf("IO error while reading the source file '%s'.\nDetails: '%s'.", fileOption, ex.getLocalizedMessage()); - return 1; - } - try { - context.eval(src); - return 0; - } catch (Throwable ex) { - if (ex instanceof PolyglotException && ((PolyglotException) ex).isExit()) { - return ((PolyglotException) ex).getExitStatus(); - } - // Internal exceptions are reported by the engine already - return 1; - } - } - - // CheckStyle: stop system..print check - - private static String[] preprocessRScriptOptions(RCmdOptions options) throws PrintHelp { - String[] arguments = options.getArguments(); - int resultArgsLength = arguments.length; - int firstNonOptionArgIndex = options.getFirstNonOptionArgIndex(); - // Now reformat the args, setting --slave and --no-restore as per the spec - ArrayList<String> adjArgs = new ArrayList<>(resultArgsLength + 1); - adjArgs.add(arguments[0]); - adjArgs.add("--slave"); - options.setValue(RCmdOption.SLAVE, true); - adjArgs.add("--no-restore"); - options.setValue(RCmdOption.NO_RESTORE, true); - // Either -e options are set or first non-option arg is a file - if (options.getStringList(RCmdOption.EXPR) == null) { - if (firstNonOptionArgIndex == resultArgsLength) { - throw new PrintHelp(); - } else { - options.setValue(RCmdOption.FILE, arguments[firstNonOptionArgIndex]); - } - } - String defaultPackagesArg = options.getString(RCmdOption.DEFAULT_PACKAGES); - String defaultPackagesEnv = System.getenv("R_DEFAULT_PACKAGES"); - if (defaultPackagesArg == null && defaultPackagesEnv == null) { - defaultPackagesArg = "datasets,utils,grDevices,graphics,stats"; - } - if (defaultPackagesEnv == null) { - options.setValue(RCmdOption.DEFAULT_PACKAGES, defaultPackagesArg); - } - // copy up to non-option args - int rx = 1; - while (rx < firstNonOptionArgIndex) { - adjArgs.add(arguments[rx]); - rx++; - } - if (options.getString(RCmdOption.FILE) != null) { - adjArgs.add("--file=" + options.getString(RCmdOption.FILE)); - rx++; // skip over file arg - firstNonOptionArgIndex++; - } - - if (firstNonOptionArgIndex < resultArgsLength) { - adjArgs.add("--args"); - while (rx < resultArgsLength) { - adjArgs.add(arguments[rx++]); - } - } - return adjArgs.toArray(new String[adjArgs.size()]); - } - - @SuppressWarnings("serial") - static class PrintHelp extends Exception { - } - - public static void main(String[] args) { - System.exit(doMain(RCommand.prependCommand(args), null, System.in, System.out, System.err)); - throw RCommand.fatal("should not reach here"); - } - - public static int doMain(String[] args, String[] env, InputStream inStream, OutputStream outStream, OutputStream errStream) { - try (RscriptCommand rcmd = new RscriptCommand(env, inStream, outStream, errStream)) { - return rcmd.execute(args); - } - } -} diff --git a/com.oracle.truffle.r.native/run/Makefile b/com.oracle.truffle.r.native/run/Makefile index 6c95f70f2be506512687fe560f317ef3c63a7b49..5ee2435d2478265e312c3c2ff61985036fa8fffc 100644 --- a/com.oracle.truffle.r.native/run/Makefile +++ b/com.oracle.truffle.r.native/run/Makefile @@ -100,6 +100,8 @@ ifeq ($(FASTR_RFFI),llvm) endif cp Makeconf.etc $(FASTR_ETC_DIR)/Makeconf + ## deploy helper script that can build native-image of FastR and update the launchers in GraalVM + cp install_r_native_image $(FASTR_BIN_DIR) ## prepare the configuration artefacts for the manual FastR configuration in GraalVM cp configure_fastr $(FASTR_BIN_DIR) chmod +x $(FASTR_BIN_DIR)/configure_fastr diff --git a/com.oracle.truffle.r.native/run/install_r_native_image b/com.oracle.truffle.r.native/run/install_r_native_image new file mode 100755 index 0000000000000000000000000000000000000000..a9faede2aaea8b984dd852329917ee2a781b9a18 --- /dev/null +++ b/com.oracle.truffle.r.native/run/install_r_native_image @@ -0,0 +1,61 @@ +#!/usr/bin/env bash + +# This script is deployed as <FASTR>/bin/install_r_native_image +set -e + +source="${BASH_SOURCE[0]}" +# "un-link" the source path +while [ -h "$source" ] ; do + prev_source="$source" + source="$(readlink "$source")"; + if [[ "$source" != /* ]]; then + # if the link was relative, it was relative to where it came from + dir="$( cd -P "$( dirname "$prev_source" )" && pwd )" + source="$dir/$source" + fi +done +fastr_home="$(dirname "$source")/.." + +silent=0 +uninstall=0 +for arg in "$@"; do + if [[ $arg == "--silent" ]]; then + silent=1 + elif [[ $arg == "uninstall" ]]; then + uninstall=1 + elif [[ $arg == "--help" ]]; then + echo "Usage: install_r_native_image [uninstall] [--silent]" + echo "When 'uninstall' argument is not present: builds and installs native image of the R runtime." + echo "When 'uninstall' argument is present: uninstalls previously installed native image of the R runtime." + echo "Use the --silent option to turn off the confirmation when installing." + fi +done + +if [[ $uninstall -eq 1 ]]; then + echo "Uninstalling native image of R runtime..." + mv "$fastr_home/bin/exec_R.backup" "$fastr_home/bin/exec/R" + mv "$fastr_home/bin/Rscript.backup" "$fastr_home/bin/Rscript" + rm -f "$fastr_home/bin/RorRscriptDispatcher" + echo "Native image of R runtime uninstalled" + exit 0 +fi + +if [[ $silent -eq 0 ]]; then + echo "This script is going to build a native image of the R runtime and update the R launchers to use that image as the default, i.e., when '--jvm' option is not used." + echo "The build takes several minutes and needs a minimum of 6GB of RAM and 150MB of free disk space. The computer may lag during the build." + read -p "Are you sure you want to build and install the native image of the R runtime? (Yy/Nn) " -n 1 -r + echo + if [[ ! $REPLY =~ ^[Yy]$ ]]; then + echo "Installation cancelled..." + exit 2 + fi +fi + +cd "$fastr_home/bin" +../../../../bin/native-image --no-server --language:R +cp "exec/R" "exec_R.backup" +cp "Rscript" "Rscript.backup" +sed -e '/^## REMOVE FOR NATIVE IMAGE: BEGIN/,/^## REMOVE FOR NATIVE IMAGE: END/d;' "exec_R.backup" | \ + sed -e 's|^exec "$JAVA_HOME/bin/java" .*|exec "$R_HOME/bin/RMain" R "${FASTR_INTERNAL_ARGS[@]}" "$@"|' > "exec/R" +sed -e '/^## REMOVE FOR NATIVE IMAGE: BEGIN/,/^## REMOVE FOR NATIVE IMAGE: END/d;' "Rscript.backup" | \ + sed -e 's|^exec "$JAVA_HOME/bin/java" .*|exec "$R_HOME/bin/RMain" Rscript "${FASTR_INTERNAL_ARGS[@]}" "$@"|' > "Rscript" diff --git a/com.oracle.truffle.r.release/src/R_legacy b/com.oracle.truffle.r.release/src/R_legacy index ebb220d571adfc974108e1a5e998100b1cd59595..b2640a971ea01b1ace066af0cfe708aa8f3a48f3 100755 --- a/com.oracle.truffle.r.release/src/R_legacy +++ b/com.oracle.truffle.r.release/src/R_legacy @@ -23,6 +23,7 @@ for i in $(seq 1 ${dir_count}); do JRE="$( dirname "$JRE" )" done +## REMOVE FOR NATIVE IMAGE: BEGIN absolute_cp=() jvm_args=() launcher_args=() @@ -67,6 +68,7 @@ JAVA_HOME="$( dirname "$JRE" )" cp="$(IFS=: ; echo "${absolute_cp[*]}")" cp="$JAVA_HOME/jre/languages/R/legacy/fastr-legacy-launcher.jar:$cp" +## REMOVE FOR NATIVE IMAGE: END # internal variable used to pass args to child R processes if [ -z "$FASTR_INTERNAL_ARGS" ]; then FASTR_INTERNAL_ARGS=() @@ -74,4 +76,4 @@ fi # we can invoke FastR directly, but we do have to set R_HOME export R_HOME="$JRE/languages/R" -exec "$JAVA_HOME/bin/java" -cp "$cp" -noverify -Dgraal.TruffleCompilationThreshold=10000 -Dgraal.TruffleCompilerThreads=2 -Xmx8g "${jvm_args[@]}" com.oracle.truffle.r.legacylauncher.LegacyLauncher com.oracle.truffle.r.launcher.RCommand "${FASTR_INTERNAL_ARGS[@]}" "${launcher_args[@]}" +exec "$JAVA_HOME/bin/java" -cp "$cp" -noverify -Dgraal.TruffleCompilationThreshold=10000 -Dgraal.TruffleCompilerThreads=2 -Xmx8g "${jvm_args[@]}" com.oracle.truffle.r.legacylauncher.LegacyLauncher com.oracle.truffle.r.launcher.RMain R "${FASTR_INTERNAL_ARGS[@]}" "${launcher_args[@]}" diff --git a/com.oracle.truffle.r.release/src/Rscript_legacy b/com.oracle.truffle.r.release/src/Rscript_legacy index f34da5ce8299f912f9434d1be900a7d6672593f2..88ab22b81b430c516974e0fbb780c3ba6a50e7c8 100755 --- a/com.oracle.truffle.r.release/src/Rscript_legacy +++ b/com.oracle.truffle.r.release/src/Rscript_legacy @@ -23,6 +23,7 @@ for i in $(seq 1 ${dir_count}); do JRE="$( dirname "$JRE" )" done +## REMOVE FOR NATIVE IMAGE: BEGIN absolute_cp=() jvm_args=() launcher_args=() @@ -67,6 +68,7 @@ JAVA_HOME="$( dirname "$JRE" )" cp="$(IFS=: ; echo "${absolute_cp[*]}")" cp="$JAVA_HOME/jre/languages/R/legacy/fastr-legacy-launcher.jar:$cp" +## REMOVE FOR NATIVE IMAGE: END # internal variable used to pass args to child R processes if [ -z "$FASTR_INTERNAL_ARGS" ]; then FASTR_INTERNAL_ARGS=() @@ -74,4 +76,4 @@ fi # we can invoke FastR directly, but we do have to set R_HOME export R_HOME="$JRE/languages/R" -exec "$JAVA_HOME/bin/java" -cp "$cp" -noverify -Dgraal.TruffleCompilationThreshold=10000 -Dgraal.TruffleCompilerThreads=2 -Xmx4g "${jvm_args[@]}" com.oracle.truffle.r.legacylauncher.LegacyLauncher com.oracle.truffle.r.launcher.RscriptCommand "${FASTR_INTERNAL_ARGS[@]}" "${launcher_args[@]}" +exec "$JAVA_HOME/bin/java" -cp "$cp" -noverify -Dgraal.TruffleCompilationThreshold=10000 -Dgraal.TruffleCompilerThreads=2 -Xmx4g "${jvm_args[@]}" com.oracle.truffle.r.legacylauncher.LegacyLauncher com.oracle.truffle.r.launcher.RMain Rscript "${FASTR_INTERNAL_ARGS[@]}" "${launcher_args[@]}" diff --git a/mx.fastr/mx_fastr.py b/mx.fastr/mx_fastr.py index 6907e4a77125a2db3166400a66467efaf5cd3ccc..c8cac58a4d60a62365b206c387e1f056187a33cb 100644 --- a/mx.fastr/mx_fastr.py +++ b/mx.fastr/mx_fastr.py @@ -45,10 +45,10 @@ was passed as an mx global option. _fastr_suite = mx.suite('fastr') -_command_class_dict = {'r': "com.oracle.truffle.r.launcher.RCommand", - 'rscript': "com.oracle.truffle.r.launcher.RscriptCommand", - 'rrepl': "com.oracle.truffle.tools.debug.shell.client.SimpleREPLClient", - 'rembed': "com.oracle.truffle.r.engine.shell.REmbedded", +_command_class_dict = {'r': ["com.oracle.truffle.r.launcher.RMain", "R"], + 'rscript': ["com.oracle.truffle.r.launcher.RMain", "Rscript"], + 'rrepl': ["com.oracle.truffle.tools.debug.shell.client.SimpleREPLClient"], + 'rembed': ["com.oracle.truffle.r.engine.shell.REmbedded"], } # benchmarking support def r_path(): @@ -104,7 +104,7 @@ def do_run_r(args, command, extraVmArgs=None, jdk=None, **kwargs): vmArgs = _sanitize_vmArgs(jdk, vmArgs) if command: - vmArgs.append(_command_class_dict[command.lower()]) + vmArgs.extend(_command_class_dict[command.lower()]) return mx.run_java(vmArgs + args, jdk=jdk, **kwargs) def r_classpath(args): diff --git a/mx.fastr/mx_fastr_dists.py b/mx.fastr/mx_fastr_dists.py index e18e321abc008c086c04250fd03a70f35d0c8a51..ee304ad2356c0554e79e9de0b7a9e701556becc8 100644 --- a/mx.fastr/mx_fastr_dists.py +++ b/mx.fastr/mx_fastr_dists.py @@ -268,6 +268,7 @@ def mx_register_dynamic_suite_constituents(register_project, register_distributi "bin/fastr_jars/graal-sdk*", ], }, + "dependency:fastr:FASTR_LAUNCHER", ], "LICENSE_FASTR" : "file:LICENSE", "3rd_party_licenses_fastr.txt" : "file:3rd_party_licenses.txt", diff --git a/mx.fastr/native-image.properties b/mx.fastr/native-image.properties index 48ad9b11c82930294f222d907b0acacefeb2043f..b0cc90d91ab5aedb578d11286463aca565819be9 100644 --- a/mx.fastr/native-image.properties +++ b/mx.fastr/native-image.properties @@ -1,16 +1,22 @@ # This file contains native-image arguments needed to fastr # -ImageName = R +ImageName = RMain Requires = tool:truffle JavaArgs = \ -Dfastr.resource.factory.class=com.oracle.truffle.r.nodes.builtin.EagerResourceHandlerFactory \ -Dfastr.internal.grid.awt.support=false \ + -Dfastr.internal.usemxbeans=false \ + -Dfastr.internal.usenativeeventloop=false \ + -Dfastr.internal.defaultdownloadmethod=wget \ + -Dfastr.internal.ignorejvmargs=true \ -Xmx6G -Args = -H:Class=com.oracle.truffle.r.launcher.RCommand \ - -H:MaxRuntimeCompileMethods=8000 \ +LauncherClass = com.oracle.truffle.r.launcher.RMain +LauncherClassPath = lib/graalvm/launcher-common.jar:languages/R/fastr-launcher.jar + +Args = -H:MaxRuntimeCompileMethods=8000 \ -H:-TruffleCheckFrameImplementation \ -H:+TruffleCheckNeverPartOfCompilation diff --git a/mx.fastr/suite.py b/mx.fastr/suite.py index 63d619f77549af7038a38bd5689e18fc32dc449c..6eaff94f25cbb4f111ef35d084a6eca7bcf657fb 100644 --- a/mx.fastr/suite.py +++ b/mx.fastr/suite.py @@ -366,6 +366,14 @@ suite = { ], }, + "FASTR_LAUNCHER" : { + "description" : "launcher for the GraalVM (at the moment used only when native image is installed)", + "dependencies" : ["com.oracle.truffle.r.launcher"], + "distDependencies" : [ + "sdk:GRAAL_SDK" + ], + }, + "FASTR" : { "description" : "class files for compiling against FastR in a separate suite", "dependencies" : [