CompilerHints.java

/*
 * Copyright (c) 2005, 2013, 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 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * 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 2 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
 * 2 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 org.openjdk.jmh.runner;

import org.openjdk.jmh.util.FileUtils;
import org.openjdk.jmh.util.Utils;

import java.io.*;
import java.util.*;

public class CompilerHints extends AbstractResourceReader {

    public static final String LIST = "/META-INF/CompilerHints";

    // All OpenJDK/HotSpot VMs are supported
    static final String[] HINT_COMPATIBLE_JVMS = { "OpenJDK", "HotSpot", "GraalVM" };
    // Zing is only compatible from post 5.10.*.* releases
    static final String JVM_ZING = "Zing";

    private static volatile CompilerHints defaultList;
    private static volatile String hintsFile;

    private final Set<String> hints;

    static final String XX_COMPILE_COMMAND_FILE = "-XX:CompileCommandFile=";

    static final String BLACKHOLE_MODE_NAME       = "jmh.blackhole.mode";
    static final String BLACKHOLE_AUTODETECT_NAME = "jmh.blackhole.autoDetect";
    static final String BLACKHOLE_DEBUG_NAME      = "jmh.blackhole.debug";
    static final String COMPILER_HINTS_MODE       = "jmh.compilerhints.mode";

    static final boolean BLACKHOLE_MODE_AUTODETECT =
            Boolean.parseBoolean(System.getProperty(BLACKHOLE_AUTODETECT_NAME, "true"));
    static final boolean BLACKHOLE_MODE_DEBUG =
            Boolean.parseBoolean(System.getProperty(BLACKHOLE_DEBUG_NAME, "false"));

    public static CompilerHints defaultList() {
        if (defaultList == null) {
            defaultList = fromResource(LIST);
        }
        return defaultList;
    }

    public static String hintsFile() {
        if (hintsFile == null) {
            try {
                final Set<String> defaultHints = defaultList().get();
                List<String> hints = new ArrayList<>(defaultHints.size() + 2);
                hints.add("quiet");

                // Set up Blackholes
                BlackholeMode bhMode = blackholeMode();
                hints.add("inline,org/openjdk/jmh/infra/Blackhole.consume");
                hints.add("dontinline,org/openjdk/jmh/infra/Blackhole.consumeCPU");
                if (bhMode.shouldBlackhole()) {
                    hints.add("blackhole,org/openjdk/jmh/infra/Blackhole.consumeCompiler");
                }
                if (bhMode.shouldNotInline()) {
                    hints.add("dontinline,org/openjdk/jmh/infra/Blackhole.consumeFull");
                }
                hints.add("dontinline,org/openjdk/jmh/infra/Blackhole.clearBox");
                hints.addAll(defaultHints);
                hintsFile = FileUtils.createTempFileWithLines("compilecommand", hints);
            } catch (IOException e) {
                throw new IllegalStateException("Error creating compiler hints file", e);
            }
        }
        return hintsFile;
    }

    public static CompilerHints fromResource(String resource) {
        return new CompilerHints(null, resource);
    }

    public static CompilerHints fromFile(String file) {
        return new CompilerHints(file, null);
    }

    private CompilerHints(String file, String resource) {
        super(file, resource, null);
        hints = Collections.unmodifiableSet(read());
    }

    public enum CompilerHintsSelect {
        FORCE_ON("Forced on", false, true),
        FORCE_OFF("Forced off", false, false),
        AUTO_ON("Automatically enabled", true, true),
        AUTO_OFF("Automatically disabled", true, false);

        private final String desc;
        private final boolean auto;
        private final boolean enabled;

        CompilerHintsSelect(String desc, boolean auto, boolean enabled) {
            this.desc = desc;
            this.auto = auto;
            this.enabled = enabled;
        }

        public String desc() {
            return desc;
        }

        public boolean isAuto() {
            return auto;
        }

        public boolean isEnabled() {
            return enabled;
        }
    }

    static CompilerHintsSelect compilerHintsSelect;

    public static CompilerHintsSelect compilerHintsSelect() {
        if (compilerHintsSelect == null) {
            compilerHintsSelect = checkCompilerHintsState();
        }
        return compilerHintsSelect;
    }

    static void resetCompilerHintsSelect() {
        // only used by tests
        compilerHintsSelect = null;
    }

    private static boolean compilerHintsEnabled() {
        return compilerHintsSelect().isEnabled();
    }

    /**
     * FIXME (low priority): check if supplied JVM is hint compatible. This test is applied to the Runner VM,
     * not the Forked and may therefore be wrong if the forked VM is not the same JVM
     */
    private static CompilerHintsSelect checkCompilerHintsState() {
        String propMode = System.getProperty(COMPILER_HINTS_MODE);
        if (propMode != null) {
            CompilerHintsSelect forced;
            try {
                forced = CompilerHintsSelect.valueOf(propMode);
            } catch (IllegalArgumentException e) {
                throw new IllegalArgumentException(COMPILER_HINTS_MODE + " should be set to FORCE_ON or FORCE_OFF", e);
            }
            if (forced.isAuto()) {
                throw new IllegalArgumentException(COMPILER_HINTS_MODE + " should be set to FORCE_ON or FORCE_OFF");
            }
            return forced;
        }
        String name = System.getProperty("java.vm.name");
        for (String vmName : HINT_COMPATIBLE_JVMS) {
            if (name.contains(vmName)) {
                return CompilerHintsSelect.AUTO_ON;
            }
        }
        if (name.contains(JVM_ZING)) {
            // 1.*.0-zing_*.*.*.*
            String version = System.getProperty("java.version");
            try {
                // get the version digits
                String[] versionDigits = version.substring(version.indexOf('_') + 1).split("\\.");
                if (Integer.parseInt(versionDigits[0]) > 5) {
                    return CompilerHintsSelect.AUTO_ON;
                } else if (Integer.parseInt(versionDigits[0]) == 5 && Integer.parseInt(versionDigits[1]) >= 10) {
                    return CompilerHintsSelect.AUTO_ON;
                }
            } catch (NumberFormatException | ArrayIndexOutOfBoundsException e) {
                // unknown Zing version format
                System.err.println("ERROR: Zing version format does not match 1.*.0-zing_*.*.*.*");
            }
        }
        return CompilerHintsSelect.AUTO_OFF;
    }

    public Set<String> get() {
        return hints;
    }

    private Set<String> read() {
        Set<String> result = new TreeSet<>();

        try {
            for (Reader r : getReaders()) {
                try (BufferedReader reader = new BufferedReader(r)) {
                    for (String line = reader.readLine(); line != null; line = reader.readLine()) {
                        if (line.startsWith("#")) {
                            continue;
                        }

                        if (line.trim().isEmpty()) {
                            continue;
                        }

                        result.add(line);
                    }
                }
            }
        } catch (IOException ex) {
            throw new RuntimeException("Error reading compiler hints", ex);
        }

        return result;
    }

    /**
     * @param command command arguments list
     * @return the compiler hint files specified by the command
     */
    public static List<String> getCompileCommandFiles(List<String> command){
        List<String> compileCommandFiles = new ArrayList<>();
        for (String cmdLineWord : command) {
            if (cmdLineWord.startsWith(XX_COMPILE_COMMAND_FILE)) {
                compileCommandFiles.add(cmdLineWord.substring(XX_COMPILE_COMMAND_FILE.length()));
            }
        }
        return compileCommandFiles;
    }

    /**
     * We need to generate a compiler hints file such that it includes:
     * <ul>
     * <li> No compile command files are specified and no .hotspotrc file is available, then do JMH hints only
     * <li> No compile command files are specified and .hotspotrc file is available, then do JMH hints + .hotspotrc
     * <li> 1 to N compile command files are specified, then do JMH hints + all specified hints in files
     * </ul>
     * <p>This is a departure from default JVM behavior as the JVM would normally just take the last hints file and ignore
     * the rest.
     *
     * @param command all -XX:CompileCommandLine args will be removed and a merged file will be set
     */
    public static void addCompilerHints(List<String> command) {
        if (compilerHintsSelect() == CompilerHintsSelect.AUTO_OFF) {
            System.err.println("WARNING: Not a HotSpot compiler command compatible VM (\""
                    + System.getProperty("java.vm.name") + "-" + System.getProperty("java.version")
                    + "\"), compiler hints are disabled.");
        }
        if (!compilerHintsEnabled()) {
            return;
        }

        List<String> hintFiles = new ArrayList<>();
        hintFiles.add(hintsFile());
        removeCompileCommandFiles(command, hintFiles);
        if (hintFiles.size() == 1) {
            File hotspotCompilerFile = new File(".hotspot_compiler");
            if (hotspotCompilerFile.exists()) {
                hintFiles.add(hotspotCompilerFile.getAbsolutePath());
            }
        }

        if (blackholeMode() == BlackholeMode.COMPILER) {
            command.add("-XX:+UnlockDiagnosticVMOptions");
            command.add("-XX:+UnlockExperimentalVMOptions");
            command.add("-DcompilerBlackholesEnabled=true");
        }

        command.add(CompilerHints.XX_COMPILE_COMMAND_FILE + mergeHintFiles(hintFiles));
    }

    /**
     * @param command the compile command file options will be removed from this command
     * @param compileCommandFiles the compiler hint files specified by the command will be added to this list
     */
    private static void removeCompileCommandFiles(List<String> command, List<String> compileCommandFiles){
        Iterator<String> iterator = command.iterator();
        while (iterator.hasNext()) {
            String cmdLineWord = iterator.next();
            if(cmdLineWord.startsWith(XX_COMPILE_COMMAND_FILE)) {
                compileCommandFiles.add(cmdLineWord.substring(XX_COMPILE_COMMAND_FILE.length()));
                iterator.remove();
            }
        }
    }

    private static String mergeHintFiles(List<String> compileCommandFiles) {
        if (compileCommandFiles.size() == 1) {
            return compileCommandFiles.get(0);
        }
        try {
            Set<String> hints = new TreeSet<>();
            for(String file : compileCommandFiles) {
                hints.addAll(fromFile(file).get());
            }
            return FileUtils.createTempFileWithLines("compilecommand", hints);
        } catch (IOException e) {
            throw new IllegalStateException("Error merging compiler hints files", e);
        }
    }

    private static BlackholeMode blackholeMode;
    private static BlackholeSelect blackholeSelect;

    private static BlackholeMode blackholeMode() {
        if (blackholeMode != null) {
            return blackholeMode;
        }

        // Forced mode takes precedence.
        String propMode = System.getProperty(BLACKHOLE_MODE_NAME);
        if (propMode != null) {
            try {
                blackholeMode = BlackholeMode.valueOf(propMode);
                blackholeSelect = BlackholeSelect.FORCED;

                // Extra safety: If user requested compiler blackholes, check
                // if they are available and fail otherwise.
                if (blackholeMode.shouldBlackhole() && !compilerBlackholesAvailable()) {
                    throw new IllegalStateException("Compiler Blackholes are not available in current VM");
                }

                return blackholeMode;
            } catch (IllegalArgumentException iae) {
                throw new IllegalStateException("Unknown Blackhole mode: " + propMode);
            }
        }

        // Try to autodetect blackhole mode, fail if not available
        if (BLACKHOLE_MODE_AUTODETECT) {
            if (compilerBlackholesAvailable()) {
                blackholeMode = BlackholeMode.COMPILER;
            } else {
                blackholeMode = BlackholeMode.FULL_DONTINLINE;
            }
            blackholeSelect = BlackholeSelect.AUTO;
            return blackholeMode;
        }

        // Not forced, not auto-detected, fallback
        blackholeMode = BlackholeMode.FULL_DONTINLINE;
        blackholeSelect = BlackholeSelect.FALLBACK;
        return blackholeMode;
    }

    private static BlackholeSelect blackholeSelect() {
        blackholeMode();
        return blackholeSelect;
    }

    private enum BlackholeMode {
        COMPILER(true, false, "compiler"),
        FULL_DONTINLINE(false, true, "full + dont-inline hint"),
        FULL(false, false, "full"),
        ;

        private final boolean shouldBlackhole;
        private final boolean shouldNotInline;
        private final String desc;

        BlackholeMode(boolean shouldBlackhole, boolean shouldNotInline, String desc) {
            this.shouldBlackhole = shouldBlackhole;
            this.shouldNotInline = shouldNotInline;
            this.desc = desc;
        }

        public boolean shouldBlackhole() {
            return shouldBlackhole;
        }

        public boolean shouldNotInline() {
            return shouldNotInline;
        }

        public String desc() { return desc; }
    }

    private enum BlackholeSelect {
        AUTO("auto-detected, use -D" + BLACKHOLE_AUTODETECT_NAME + "=false to disable"),
        FALLBACK("fallback, use -D" + BLACKHOLE_MODE_NAME + " to force"),
        FORCED("forced"),
        ;

        final String desc;

        BlackholeSelect(String desc) {
            this.desc = desc;
        }

        public String desc() {
            return desc;
        }
    }

    private static boolean compilerBlackholesAvailable() {
        // Step 1. See if there were any error messages from CompilerOracle
        {
            List<String> cmd = new ArrayList<>();
            cmd.add(Utils.getCurrentJvm());
            cmd.add("-XX:+UnlockExperimentalVMOptions");
            cmd.add("-XX:CompileCommand=quiet");
            cmd.add("-XX:CompileCommand=blackhole,some/fake/Class.method");
            cmd.add("-version");

            debug("Blackhole command errors test:");

            Collection<String> log = Utils.runWith(cmd);
            for (String l : log) {
                debug(l);
                if (l.contains("CompilerOracle") || l.contains("CompileCommand")) {
                    debug("Found the suspected error line, no compiler blackholes.");
                    return false;
                }
            }
        }

        // Step 2. See that CompilerOracle accepted the command explicitly
        {
            List<String> cmd = new ArrayList<>();
            cmd.add(Utils.getCurrentJvm());
            cmd.add("-XX:+UnlockExperimentalVMOptions");
            cmd.add("-XX:CompileCommand=blackhole,some/fake/Class.method");
            cmd.add("-version");

            debug("Blackhole command acceptance test:");

            Collection<String> log = Utils.runWith(cmd);
            for (String l : log) {
                debug(l);
                if (l.contains("CompilerOracle") || l.contains("CompileCommand")) {
                    debug("Found the acceptance line, compiler blackholes are available.");
                    return true;
                }
            }
        }

        // Err on the side of the caution: compiler blackholes are not available.
        debug("Compiler blackholes are not available.");
        return false;
    }

    private static void debug(String msg) {
        if (BLACKHOLE_MODE_DEBUG) {
            System.out.println(msg);
        }
    }

    public static void printHints(PrintStream out) {
        if (!compilerHintsSelect().isAuto()) {
            out.print("# Compiler hints: " + (compilerHintsEnabled() ? "enabled" : "disabled") + " (" + compilerHintsSelect().desc() + ")");
            out.println();
        }
        BlackholeMode mode = blackholeMode();
        out.print("# Blackhole mode: " + mode.desc() + " (" + blackholeSelect().desc() + ")");
        out.println();
    }

    public static void printWarnings(PrintStream out) {
        if (blackholeMode() == BlackholeMode.COMPILER) {
            out.println("NOTE: Current JVM experimentally supports Compiler Blackholes, and they are in use. Please exercise");
            out.println("extra caution when trusting the results, look into the generated code to check the benchmark still");
            out.println("works, and factor in a small probability of new VM bugs. Additionally, while comparisons between");
            out.println("different JVMs are already problematic, the performance difference caused by different Blackhole");
            out.println("modes can be very significant. Please make sure you use the consistent Blackhole mode for comparisons.");
            out.println();
        }
    }

}