Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support for --jvm option in Enso runner #10374

Merged
merged 29 commits into from
Jul 6, 2024
Merged
Show file tree
Hide file tree
Changes from 17 commits
Commits
Show all changes
29 commits
Select commit Hold shift + click to select a range
43929db
Support for --jvm option in Enso launcher
JaroslavTulach Jun 26, 2024
7d31eab
Iterate installed GraalVM runtimes
JaroslavTulach Jun 26, 2024
67f1632
feat: engineDistributionRoot parameter
4e6 Jun 26, 2024
e67cc87
Generate NI executable as ./built-distribution/enso-engine-0.0.0-dev-…
JaroslavTulach Jun 26, 2024
8f18b2f
Ignore --jvm when already in the right JVM
JaroslavTulach Jun 28, 2024
54bb7d8
Recognize JAVA_OPTS environment variable and pass it to the JVM
JaroslavTulach Jun 28, 2024
ec51515
Let EditionManager work with any EditionProvider, not just updating one
JaroslavTulach Jun 28, 2024
f1ad21b
Don't use UpdatingEditionManager in regular execution
JaroslavTulach Jun 28, 2024
cfedcb6
Derive languageHome from code location
JaroslavTulach Jun 29, 2024
2f3e832
Search for the parser in component subdirectory
JaroslavTulach Jun 29, 2024
406291e
Setup logging before processing --jvm option
JaroslavTulach Jun 29, 2024
cc48ec3
Providing missing argument to makeEditionProvider
JaroslavTulach Jun 29, 2024
9c81fe5
Adding update flag to FakeEditionProvider
JaroslavTulach Jun 29, 2024
eafa192
Change log entry
JaroslavTulach Jun 29, 2024
5ea0814
Benchmarks need to be executed in --jvm mode
JaroslavTulach Jun 29, 2024
8c393bf
Merge remote-tracking branch 'origin/develop' into wip/jtulach/JvmOpt…
JaroslavTulach Jul 1, 2024
fa7a62d
Merge remote-tracking branch 'origin/develop' into wip/jtulach/JvmOpt…
JaroslavTulach Jul 3, 2024
13cb3d4
native-image needs location without .exe on Windows
JaroslavTulach Jul 3, 2024
3fb6a1b
There is no runner anymore
JaroslavTulach Jul 4, 2024
8e4834e
Use .exe in native runner tests
JaroslavTulach Jul 5, 2024
59f3087
run_benchmarks without bash script wrapper
JaroslavTulach Jul 5, 2024
83f31be
Supressing question mark operator is useless here warning
JaroslavTulach Jul 5, 2024
1ec943a
Updating documentation to new location of runner
JaroslavTulach Jul 6, 2024
13c9518
Just return benchmarks
JaroslavTulach Jul 6, 2024
ffb2fb8
No need to disable linter
JaroslavTulach Jul 6, 2024
60595ba
Use with_executable_extension
JaroslavTulach Jul 6, 2024
49700cd
Use with_executable_extension II
JaroslavTulach Jul 6, 2024
2bd08b1
cargo fmt
JaroslavTulach Jul 6, 2024
bbbbf25
Using longer local variable name
JaroslavTulach Jul 6, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
6 changes: 6 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,12 @@
[10352]: https://github.com/enso-org/enso/pull/10352
[10353]: https://github.com/enso-org/enso/pull/10353

#### Enso Language & Runtime

- Support for [explicit --jvm option][10374] when launching `enso` CLI

[10374]: https://github.com/enso-org/enso/pull/10374

#### Enso Standard Library

- [Added Statistic.Product][10122]
Expand Down
103 changes: 56 additions & 47 deletions build.sbt
Original file line number Diff line number Diff line change
Expand Up @@ -2614,55 +2614,60 @@ lazy val `engine-runner` = project
assembly := assembly
.dependsOn(`runtime-fat-jar` / assembly)
.value,
rebuildNativeImage :=
NativeImage
.buildNativeImage(
"runner",
staticOnLinux = false,
additionalOptions = Seq(
"-Dorg.apache.commons.logging.Log=org.apache.commons.logging.impl.NoOpLog",
"-H:IncludeResources=.*Main.enso$",
"-H:+AddAllCharsets",
"-H:+IncludeAllLocales",
"-ea",
// useful perf & debug switches:
// "-g",
// "-H:+SourceLevelDebug",
// "-H:-DeleteLocalSymbols",
// you may need to set smallJdk := None to use following flags:
// "--trace-class-initialization=org.enso.syntax2.Parser",
"-Dnic=nic"
),
mainClass = Some("org.enso.runner.Main"),
initializeAtRuntime = Seq(
"org.jline.nativ.JLineLibrary",
"org.jline.terminal.impl.jna",
"io.methvin.watchservice.jna.CarbonAPI",
"zio.internal.ZScheduler$$anon$4",
"org.enso.runner.Main$",
"sun.awt",
"sun.java2d",
"sun.font",
"java.awt",
"com.sun.imageio",
"com.sun.jna.internal.Cleaner",
"com.sun.jna.Structure$FFIType",
"akka.http"
rebuildNativeImage := Def
.taskDyn {
NativeImage
.buildNativeImage(
"enso",
JaroslavTulach marked this conversation as resolved.
Show resolved Hide resolved
targetDir = engineDistributionRoot.value / "bin",
staticOnLinux = false,
additionalOptions = Seq(
"-Dorg.apache.commons.logging.Log=org.apache.commons.logging.impl.NoOpLog",
"-H:IncludeResources=.*Main.enso$",
"-H:+AddAllCharsets",
"-H:+IncludeAllLocales",
"-ea",
// useful perf & debug switches:
// "-g",
// "-H:+SourceLevelDebug",
// "-H:-DeleteLocalSymbols",
// you may need to set smallJdk := None to use following flags:
// "--trace-class-initialization=org.enso.syntax2.Parser",
"-Dnic=nic"
),
mainClass = Some("org.enso.runner.Main"),
initializeAtRuntime = Seq(
"org.jline.nativ.JLineLibrary",
"org.jline.terminal.impl.jna",
"io.methvin.watchservice.jna.CarbonAPI",
"zio.internal.ZScheduler$$anon$4",
"org.enso.runner.Main$",
"sun.awt",
"sun.java2d",
"sun.font",
"java.awt",
"com.sun.imageio",
"com.sun.jna.internal.Cleaner",
"com.sun.jna.Structure$FFIType",
"akka.http"
)
)
)
.dependsOn(NativeImage.additionalCp)
.dependsOn(NativeImage.smallJdk)
.dependsOn(assembly)
.dependsOn(
buildEngineDistribution
)
.value,
buildNativeImage := NativeImage
.incrementalNativeImageBuild(
rebuildNativeImage,
"runner"
}
.dependsOn(NativeImage.additionalCp)
.dependsOn(NativeImage.smallJdk)
.dependsOn(assembly)
.dependsOn(
buildEngineDistribution
)
.value
.value,
buildNativeImage := Def.taskDyn {
NativeImage
.incrementalNativeImageBuild(
rebuildNativeImage,
"enso",
targetDir = engineDistributionRoot.value / "bin"
)
}.value
)
.dependsOn(`version-output`)
.dependsOn(yaml)
Expand Down Expand Up @@ -3559,6 +3564,10 @@ ThisBuild / buildEngineDistribution := {
buildEngineDistribution.result.value
}

ThisBuild / engineDistributionRoot := {
engineDistributionRoot.value
}

lazy val buildEngineDistributionNoIndex =
taskKey[Unit]("Builds the engine distribution without generating indexes")
buildEngineDistributionNoIndex := {
Expand Down
2 changes: 1 addition & 1 deletion build/build/src/enso.rs
Original file line number Diff line number Diff line change
Expand Up @@ -74,7 +74,7 @@ impl BuiltEnso {

pub async fn run_benchmarks(&self, opt: BenchmarkOptions) -> Result {
self.cmd()?
.with_args(["--run", self.paths.repo_root.test.benchmarks.as_str()])
.with_args(["--jvm", "--run", self.paths.repo_root.test.benchmarks.as_str()])
JaroslavTulach marked this conversation as resolved.
Show resolved Hide resolved
.set_env(ENSO_BENCHMARK_TEST_DRY_RUN, &Boolean::from(opt.dry_run))?
.run_ok()
.await
Expand Down
2 changes: 1 addition & 1 deletion distribution/bin/enso
Original file line number Diff line number Diff line change
Expand Up @@ -8,5 +8,5 @@ for opt in "$@"; do
done

JAVA_OPTS="--add-opens=java.base/java.nio=ALL-UNNAMED $JAVA_OPTS"
exec java --module-path $COMP_PATH -Dorg.graalvm.language.enso.home=$COMP_PATH $EXTRA_OPTS $JAVA_OPTS -m org.enso.runtime/org.enso.EngineRunnerBootLoader "$@"
exec java --module-path $COMP_PATH $EXTRA_OPTS $JAVA_OPTS -m org.enso.runtime/org.enso.EngineRunnerBootLoader "$@"
exit
2 changes: 1 addition & 1 deletion distribution/bin/enso.bat
Original file line number Diff line number Diff line change
Expand Up @@ -7,5 +7,5 @@ set EXTRA_OPTS=%EXTRA_OPTS% -Dgraal.Dump=Truffle:1
)
)
set JAVA_OPTS=%JAVA_OPTS% --add-opens=java.base/java.nio=ALL-UNNAMED
java --module-path %comp-dir% -Dorg.graalvm.language.enso.home=%comp-dir% -Dpolyglot.compiler.IterativePartialEscape=true %EXTRA_OPTS% %JAVA_OPTS% -m org.enso.runtime/org.enso.EngineRunnerBootLoader %*
java --module-path %comp-dir% -Dpolyglot.compiler.IterativePartialEscape=true %EXTRA_OPTS% %JAVA_OPTS% -m org.enso.runtime/org.enso.EngineRunnerBootLoader %*
exit /B %errorlevel%
126 changes: 113 additions & 13 deletions engine/runner/src/main/java/org/enso/runner/Main.java
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
import java.nio.file.InvalidPathException;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
Expand Down Expand Up @@ -53,6 +54,7 @@

/** The main CLI entry point class. */
public final class Main {
private static final String JVM_OPTION = "jvm";
private static final String RUN_OPTION = "run";
private static final String INSPECT_OPTION = "inspect";
private static final String DUMP_GRAPHS_OPTION = "dump-graphs";
Expand Down Expand Up @@ -127,6 +129,15 @@ private static Options buildOptions() {
.longOpt(RUN_OPTION)
.desc("Runs a specified Enso file.")
.build();
var jvm =
cliOptionBuilder()
.hasArg(true)
.numberOfArgs(1)
.optionalArg(true)
.argName("jvm")
.longOpt(JVM_OPTION)
.desc("Specifies whether to run JVM mode and optionally selects a JVM to run with.")
.build();
var inspect =
cliOptionBuilder()
.longOpt(INSPECT_OPTION)
Expand Down Expand Up @@ -451,6 +462,7 @@ private static Options buildOptions() {
options
.addOption(help)
.addOption(repl)
.addOption(jvm)
.addOption(run)
.addOption(inspect)
.addOption(dumpGraphs)
Expand Down Expand Up @@ -508,17 +520,17 @@ private static void printHelp(Options options) {
}

/** Terminates the process with a failure exit code. */
private RuntimeException exitFail() {
private static RuntimeException exitFail() {
return doExit(1);
}

/** Terminates the process with a success exit code. */
private RuntimeException exitSuccess() {
private static RuntimeException exitSuccess() {
return doExit(0);
}

/** Shuts down the logging service and terminates the process. */
private RuntimeException doExit(int exitCode) {
private static RuntimeException doExit(int exitCode) {
RunnerLogging.tearDown();
System.exit(exitCode);
return null;
Expand Down Expand Up @@ -932,7 +944,7 @@ private void displayVersion(boolean useJson) {
}

/** Parses the log level option. */
private Level parseLogLevel(String levelOption) {
private static Level parseLogLevel(String levelOption) {
var name = levelOption.toLowerCase();
var found =
Stream.of(Level.values()).filter(x -> name.equals(x.name().toLowerCase())).findFirst();
Expand All @@ -949,7 +961,7 @@ private Level parseLogLevel(String levelOption) {
}

/** Parses an URI that specifies the logging service connection. */
private URI parseUri(String string) {
private static URI parseUri(String string) {
try {
return new URI(string);
} catch (URISyntaxException ex) {
Expand All @@ -966,7 +978,7 @@ private URI parseUri(String string) {
*
* @param args the command line arguments
*/
public static void main(String[] args) throws IOException {
public static void main(String[] args) throws Exception {
new Main().launch(args);
}

Expand Down Expand Up @@ -1294,10 +1306,95 @@ private void println(String msg) {
System.out.println(msg);
}

private void launch(String[] args) {
private void launch(String[] args) throws IOException, InterruptedException, URISyntaxException {
var options = buildOptions();
var line = preprocessArguments(options, args);
launch(options, line);

var logMasking = new boolean[1];
var logLevel = setupLogging(options, line, logMasking);

if (line.hasOption(JVM_OPTION)) {
var jvm = line.getOptionValue(JVM_OPTION);
var current = System.getProperty("java.home");
if (jvm == null) {
jvm = current;
}
if (current == null || !current.equals(jvm)) {
var loc = Main.class.getProtectionDomain().getCodeSource().getLocation();
var commandAndArgs = new ArrayList<String>();
JVM_FOUND:
if (jvm == null) {
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The JVM_OPTION takes zero or one argument. If the argument is present, it should be a path to the GraalVM to use to execute Enso process. The code below uses ProcessBuilder to construct such process - more or less the code is a copy of what is in bin/enso and bin/enso.bat files.

When the --jvm option is used without additional argument, then we should somehow locate the GraalVM in the installation. CCing @4e6 and @radeusgd (as author of enso launcher which deals with similar issues) to guide this part of the implementation.

var env = new Environment() {};
var dm = new DistributionManager(env);
var paths = dm.paths();
var files = paths.runtimes().toFile().listFiles();
JaroslavTulach marked this conversation as resolved.
Show resolved Hide resolved
if (files != null) {
for (var d : files) {
var java = new File(new File(d, "bin"), "java").getAbsoluteFile();
if (java.exists()) {
commandAndArgs.add(java.getPath());
break JVM_FOUND;
}
}
}
commandAndArgs.add("java");
} else {
commandAndArgs.add(new File(new File(new File(jvm), "bin"), "java").getAbsolutePath());
}
var jvmOptions = System.getenv("JAVA_OPTS");
if (jvmOptions != null) {
for (var op : jvmOptions.split(" ")) {
if (op.isEmpty()) {
continue;
}
commandAndArgs.add(op);
}
}

commandAndArgs.add("--add-opens=java.base/java.nio=ALL-UNNAMED");
commandAndArgs.add("--module-path");
var component = new File(loc.toURI().resolve("..")).getAbsoluteFile();
if (!component.getName().equals("component")) {
component = new File(component, "component");
}
if (!component.isDirectory()) {
throw new IOException("Cannot find " + component + " directory");
}
commandAndArgs.add(component.getPath());
commandAndArgs.add("-m");
commandAndArgs.add("org.enso.runtime/org.enso.EngineRunnerBootLoader");
var it = line.iterator();
while (it.hasNext()) {
var op = it.next();
if (JVM_OPTION.equals(op.getLongOpt())) {
continue;
}
var longName = op.getLongOpt();
if (longName != null) {
commandAndArgs.add("--" + longName);
} else {
commandAndArgs.add("-" + op.getOpt());
}
var values = op.getValuesList();
if (values != null) {
commandAndArgs.addAll(values);
}
}
commandAndArgs.addAll(line.getArgList());
var pb = new ProcessBuilder();
pb.inheritIO();
pb.command(commandAndArgs);
var p = pb.start();
var exitCode = p.waitFor();
if (exitCode == 0) {
throw exitSuccess();
} else {
throw doExit(exitCode);
}
}
}

launch(options, line, logLevel, logMasking[0]);
}

protected CommandLine preprocessArguments(Options options, String[] args) {
Expand All @@ -1311,15 +1408,18 @@ protected CommandLine preprocessArguments(Options options, String[] args) {
}
}

protected void launch(Options options, CommandLine line) {
private static Level setupLogging(Options options, CommandLine line, boolean[] logMasking) {
var logLevel =
scala.Option.apply(line.getOptionValue(LOG_LEVEL))
.map(this::parseLogLevel)
.map(Main::parseLogLevel)
.getOrElse(() -> defaultLogLevel);
var connectionUri = scala.Option.apply(line.getOptionValue(LOGGER_CONNECT)).map(this::parseUri);
var logMasking = !line.hasOption(NO_LOG_MASKING);
RunnerLogging.setup(connectionUri, logLevel, logMasking);
var connectionUri = scala.Option.apply(line.getOptionValue(LOGGER_CONNECT)).map(Main::parseUri);
logMasking[0] = !line.hasOption(NO_LOG_MASKING);
RunnerLogging.setup(connectionUri, logLevel, logMasking[0]);
return logLevel;
}

private void launch(Options options, CommandLine line, Level logLevel, boolean logMasking) {
if (line.hasOption(LANGUAGE_SERVER_OPTION)) {
runLanguageServer(line, logLevel);
} else {
Expand Down
Loading
Loading