X-Git-Url: http://source.jalview.org/gitweb/?a=blobdiff_plain;f=src%2Fjalview%2Fbin%2FLauncher.java;h=20a466b422338e8d76ca775ede8e91f60cab2630;hb=ed379d3f998a6aea83ce8c9cdaab2ec8234adb6a;hp=aec3acd2eade9c356ec43b3f72653cc82c18a9e1;hpb=f3b1c43221b4602d4253e4357250b57e3be33521;p=jalview.git diff --git a/src/jalview/bin/Launcher.java b/src/jalview/bin/Launcher.java index aec3acd..20a466b 100644 --- a/src/jalview/bin/Launcher.java +++ b/src/jalview/bin/Launcher.java @@ -1,28 +1,177 @@ +/* + * Jalview - A Sequence Alignment Editor and Viewer ($$Version-Rel$$) + * Copyright (C) $$Year-Rel$$ The Jalview Authors + * + * This file is part of Jalview. + * + * Jalview is free software: you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation, either version 3 + * of the License, or (at your option) any later version. + * + * Jalview 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 for more details. + * + * You should have received a copy of the GNU General Public License + * along with Jalview. If not, see . + * The Jalview Authors are detailed in the 'AUTHORS' file. + */ package jalview.bin; import java.io.File; +import java.io.IOException; import java.lang.management.ManagementFactory; -import java.lang.management.OperatingSystemMXBean; import java.util.ArrayList; +import java.util.List; +import java.util.Locale; +import java.util.concurrent.TimeUnit; +import jalview.util.ChannelProperties; +import jalview.util.LaunchUtils; + +/** + * A Launcher class for Jalview. This class is used to launch Jalview from the + * shadowJar when Getdown is not used or available. It attempts to take all the + * command line arguments to pass on to the jalview.bin.Jalview class, but to + * insert a -Xmx memory setting to a sensible default, using the -jvmmempc and + * -jvmmemmax application arguments if specified. If not specified then system + * properties will be looked for by jalview.bin.MemorySetting. If the user has + * provided the JVM with a -Xmx setting directly and not set -jvmmempc or + * -jvmmemmax then this setting will be used and system properties ignored. If + * -Xmx is set as well as -jvmmempc or -jvmmemmax as argument(s) then the -Xmx + * argument will NOT be passed on to the main application launch. + * + * @author bsoares + * + */ public class Launcher { - private final static String startClass = "jalview.bin.Jalview"; - private final static int maxHeapSizePerCent = 95; - - private final static String dockIconPath = "JalviewLogo_Huge.png"; + private final static String headlessProperty = "java.awt.headless"; + /** + * main method for jalview.bin.Launcher. This restarts the same JRE's JVM with + * the same arguments but with memory adjusted based on extracted -jvmmempc + * and -jvmmemmax application arguments. If on a Mac then extra dock:icon and + * dock:name arguments are also set. + * + * @param args + */ public static void main(String[] args) { - final String javaBin = System.getProperty("java.home") + File.separator - + "bin" + File.separator + "java"; + if (!LaunchUtils.checkJavaVersion()) + { + System.err.println("WARNING - The Java version being used (Java " + + LaunchUtils.getJavaVersion() + + ") may lead to problems. This installation of Jalview should be used with Java " + + LaunchUtils.getJavaCompileVersion() + "."); + } - ArrayList command = new ArrayList<>(); + String jvmmempc = null; + String jvmmemmax = null; + boolean debug = false; + boolean wait = true; + boolean quiet = false; + boolean headless = false; + boolean gui = false; + // must set --debug before --launcher... + boolean launcherstop = false; + boolean launcherprint = false; + boolean launcherwait = false; + ArrayList arguments = new ArrayList<>(); + for (String arg : args) + { + if (arg.equals("--debug")) + { + debug = true; + } + if (arg.equals("--quiet")) + { + quiet = true; + } + if (arg.equals("--headless")) + { + headless = true; + } + if (arg.equals("--gui")) + { + gui = true; + } + if (debug && arg.equals("--launcherprint")) + { + launcherprint = true; + } + if (debug && arg.equals("--launcherstop")) + { + launcherstop = true; + } + if (debug && arg.equals("--launcherwait")) + { + launcherwait = true; + } + // this ends the launcher immediately + if (debug && arg.equals("--launchernowait")) + { + wait = false; + } + // Don't add the --launcher... args to Jalview launch + if (arg.startsWith("--launcher")) + { + continue; + } + // jvmmempc and jvmmemmax args used to set memory and are not passed on to + // startClass + if (arg.startsWith( + "-" + MemorySetting.MAX_HEAPSIZE_PERCENT_PROPERTY_NAME + "=")) + { + jvmmempc = arg.substring( + MemorySetting.MAX_HEAPSIZE_PERCENT_PROPERTY_NAME.length() + + 2); + } + else if (arg.startsWith( + "-" + MemorySetting.MAX_HEAPSIZE_PROPERTY_NAME + "=")) + { + jvmmemmax = arg.substring( + MemorySetting.MAX_HEAPSIZE_PROPERTY_NAME.length() + 2); + } + // --doubledash versions + else if (arg.startsWith("--" + + MemorySetting.MAX_HEAPSIZE_PERCENT_PROPERTY_NAME + "=")) + { + jvmmempc = arg.substring( + MemorySetting.MAX_HEAPSIZE_PERCENT_PROPERTY_NAME.length() + + 3); + } + else if (arg.startsWith( + "--" + MemorySetting.MAX_HEAPSIZE_PROPERTY_NAME + "=")) + { + jvmmemmax = arg.substring( + MemorySetting.MAX_HEAPSIZE_PROPERTY_NAME.length() + 3); + } + // retain arg + else + { + arguments.add(arg); + } + } + if (gui) + { + // --gui takes precedence over --headless + headless = false; + } + + final String appName = ChannelProperties.getProperty("app_name"); + + // if we're using jalview.bin.Launcher we always assume a console is in use + final String javaBin = LaunchUtils.findJavaBin(true); + + List command = new ArrayList<>(); command.add(javaBin); - boolean isAMac = System.getProperty("os.name").indexOf("Mac") > -1; + String memSetting = null; for (String jvmArg : ManagementFactory.getRuntimeMXBean() .getInputArguments()) @@ -31,48 +180,100 @@ public class Launcher } command.add("-cp"); command.add(ManagementFactory.getRuntimeMXBean().getClassPath()); - ArrayList arguments = new ArrayList<>(); - for (String arg : args) + + // use saved preferences if no cmdline args + boolean useCustomisedSettings = LaunchUtils + .getBooleanUserPreference(MemorySetting.CUSTOMISED_SETTINGS); + if (useCustomisedSettings) { - arguments.add(arg); + if (jvmmempc == null) + { + jvmmempc = LaunchUtils + .getUserPreference(MemorySetting.MEMORY_JVMMEMPC); + } + if (jvmmemmax == null) + { + jvmmemmax = LaunchUtils + .getUserPreference(MemorySetting.MEMORY_JVMMEMMAX); + } } - // add memory setting if not specified + // add these settings if not already specified boolean memSet = false; boolean dockIcon = false; - ARG: for (int i = 0; i < command.size(); i++) + boolean dockName = false; + boolean headlessProp = false; + for (int i = 0; i < command.size(); i++) { String arg = command.get(i); if (arg.startsWith("-Xmx")) { - memSet = true; + // only use -Xmx if jvmmemmax and jvmmempc have not been set + if (jvmmempc == null && jvmmemmax == null) + { + memSetting = arg; + memSet = true; + } } else if (arg.startsWith("-Xdock:icon")) { dockIcon = true; } + else if (arg.startsWith("-Xdock:name")) + { + dockName = true; + } + else if (arg.startsWith("-D" + headlessProperty + "=")) + { + headlessProp = true; + } } if (!memSet) { - long maxMemLong = -1; - long physicalMem = getPhysicalMemory(); - if (physicalMem > 0) + long maxMemLong = MemorySetting.getMemorySetting(jvmmemmax, jvmmempc); + + if (maxMemLong > 0) { - maxMemLong = physicalMem * maxHeapSizePerCent / 100; + memSetting = "-Xmx" + Long.toString(maxMemLong); + memSet = true; + command.add(memSetting); } - if (maxMemLong > 0) + } + + if (LaunchUtils.isMac) + { + if (!dockIcon) { - command.add("-Xmx" + Long.toString(maxMemLong)); + String dockIconPath = System.getProperty("getdownappdir", ".") + + File.separator + "resource/jalview_logo.png"; + command.add("-Xdock:icon=" + dockIconPath); } + if (!dockName) + { + // -Xdock:name=... doesn't actually work :( + // Leaving it in in case it gets fixed + command.add("-Xdock:name=" + appName); + // this launches WITHOUT an icon in the macOS dock. Could be useful for + // getdown? + // command.add("-Dapple.awt.UIElement=false"); + // This also does not work for the dock + command.add("-Dcom.apple.mrj.application.apple.menu.about.name=" + + appName); + } + } + if (headless && !headlessProp) + { + System.setProperty(headlessProperty, "true"); + command.add("-D" + headlessProperty + "=true"); } - if (!dockIcon && isAMac) + String scalePropertyArg = HiDPISetting.getScalePropertyArg(); + if (scalePropertyArg != null) { - command.add("-Xdock:icon=" + dockIconPath); - // -Xdock:name=... doesn't actually work :( - // Leaving it in in case it gets fixed - command.add("-Xdock:name=" + "Jalview"); + sysout(debug, quiet, "Running " + startClass + " with scale setting " + + scalePropertyArg); + command.add(scalePropertyArg); } command.add(startClass); @@ -80,41 +281,86 @@ public class Launcher final ProcessBuilder builder = new ProcessBuilder(command); - System.out.println("COMMAND: " + String.join(" ", builder.command())); + if ((Boolean.parseBoolean(System.getProperty("launcherprint", "false")) + || launcherprint)) + { + sysout(debug, quiet, + "LAUNCHER COMMAND: " + String.join(" ", builder.command())); + } + sysout(debug, quiet, + "Running " + startClass + " with " + + (memSetting == null ? "no memory setting" + : ("memory setting " + memSetting))); + if (Boolean.parseBoolean(System.getProperty("launcherstop", "false")) + || (debug && launcherstop)) + { + sysout(debug, quiet, + "System property 'launcherstop' is set and not 'false'. Exiting."); + System.exit(0); + } try { builder.inheritIO(); Process process = builder.start(); - process.waitFor(); + if (wait || launcherwait) + { + sysout(debug, quiet, "Launching application process"); + process.waitFor(); + } + else + { + int waitInt = 0; + sysout(debug, quiet, + "Wait time for application process is " + waitInt + "ms"); + process.waitFor(waitInt, TimeUnit.MILLISECONDS); + } + sysout(debug, quiet, "Launcher process ending"); + } catch (IOException e) + { + if (e.getMessage().toLowerCase(Locale.ROOT).contains("memory")) + { + System.err.println("Caught a memory exception: " + e.getMessage()); + // Probably the "Cannot allocate memory" error, try without the memory + // setting + ArrayList commandNoMem = new ArrayList<>(); + for (int i = 0; i < command.size(); i++) + { + if (!command.get(i).startsWith("-Xmx")) + { + commandNoMem.add(command.get(i)); + } + } + final ProcessBuilder builderNoMem = new ProcessBuilder( + commandNoMem); + System.err.println("Command without memory setting: " + + String.join(" ", builderNoMem.command())); + try + { + builderNoMem.inheritIO(); + Process processNoMem = builderNoMem.start(); + processNoMem.waitFor(); + } catch (Exception ex) + { + ex.printStackTrace(); + } + } + else + { + e.printStackTrace(); + } } catch (Exception e) { e.printStackTrace(); } - // System.exit(0); - } - public static long getPhysicalMemory() + private static void sysout(boolean debug, boolean quiet, String message) { - final OperatingSystemMXBean o = ManagementFactory - .getOperatingSystemMXBean(); - - try + if (debug && !quiet) { - if (o instanceof com.sun.management.OperatingSystemMXBean) - { - final com.sun.management.OperatingSystemMXBean osb = (com.sun.management.OperatingSystemMXBean) o; - return osb.getTotalPhysicalMemorySize(); - } - } catch (NoClassDefFoundError e) - { - // com.sun.management.OperatingSystemMXBean doesn't exist in this JVM - System.out.println("No com.sun.management.OperatingSystemMXBean"); + System.out.println("LAUNCHERDEBUG - " + message); } - - // We didn't get a com.sun.management.OperatingSystemMXBean. - return -1; } }