/*
 * Copyright (c) 2015, 2020, 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.graalvm.compiler.hotspot;

import static jdk.vm.ci.common.InitTimer.timer;
import static jdk.vm.ci.hotspot.HotSpotJVMCICompilerFactory.CompilationLevelAdjustment.None;
import static jdk.vm.ci.services.Services.IS_BUILDING_NATIVE_IMAGE;
import static jdk.vm.ci.services.Services.IS_IN_NATIVE_IMAGE;
import static org.graalvm.compiler.hotspot.HotSpotGraalOptionValues.GRAAL_OPTION_PROPERTY_PREFIX;

import java.io.PrintStream;

import org.graalvm.compiler.api.runtime.GraalRuntime;
import org.graalvm.compiler.debug.MethodFilter;
import org.graalvm.compiler.options.Option;
import org.graalvm.compiler.options.OptionKey;
import org.graalvm.compiler.options.OptionType;
import org.graalvm.compiler.options.OptionValues;
import org.graalvm.compiler.options.OptionsParser;
import org.graalvm.compiler.phases.tiers.CompilerConfiguration;

import jdk.vm.ci.common.InitTimer;
import jdk.vm.ci.hotspot.HotSpotJVMCICompilerFactory;
import jdk.vm.ci.hotspot.HotSpotJVMCIRuntime;
import jdk.vm.ci.hotspot.HotSpotResolvedJavaMethod;
import jdk.vm.ci.meta.Signature;
import jdk.vm.ci.runtime.JVMCICompilerFactory;
import jdk.vm.ci.runtime.JVMCIRuntime;
import jdk.vm.ci.services.Services;

public final class HotSpotGraalCompilerFactory extends HotSpotJVMCICompilerFactory {

    private static MethodFilter graalCompileOnlyFilter;
    private static boolean compileGraalWithC1Only;

    private IsGraalPredicate isGraalPredicate;

    private final HotSpotGraalJVMCIServiceLocator locator;

    HotSpotGraalCompilerFactory(HotSpotGraalJVMCIServiceLocator locator) {
        this.locator = locator;
    }

    @Override
    public String getCompilerName() {
        return "graal";
    }

    
Initialized when this factory is selected.
/** * Initialized when this factory is {@linkplain #onSelection() selected}. */
private OptionValues options;
An exception thrown when option parsing fails. This will result in a call to HotSpotGraalServices.exit. Ideally, that would be done here but JVMCICompilerFactory.onSelection() does not pass in a HotSpotJVMCIRuntime.
/** * An exception thrown when {@linkplain HotSpotGraalOptionValues#defaultOptions() option * parsing} fails. This will result in a call to {@link HotSpotGraalServices#exit}. Ideally, * that would be done here but {@link JVMCICompilerFactory#onSelection()} does not pass in a * {@link HotSpotJVMCIRuntime}. */
private IllegalArgumentException optionsFailure; @Override public void onSelection() { JVMCIVersionCheck.check(Services.getSavedProperties(), false); assert options == null : "cannot select " + getClass() + " service more than once"; try { options = HotSpotGraalOptionValues.defaultOptions(); } catch (IllegalArgumentException e) { optionsFailure = e; return; } initializeGraalCompilePolicyFields(options); isGraalPredicate = compileGraalWithC1Only ? new IsGraalPredicate() : null; /* * Exercise this code path early to encourage loading now. This doesn't solve problem of * deadlock during class loading but seems to eliminate it in practice. */ if (isGraalPredicate != null && isGraalPredicate.getCompilationLevelAdjustment() != None) { adjustCompilationLevelInternal(Object.class, CompilationLevel.FullOptimization); adjustCompilationLevelInternal(Object.class, CompilationLevel.Simple); } if (IS_BUILDING_NATIVE_IMAGE) { // Triggers initialization of all option descriptors Options.CompileGraalWithC1Only.getName(); } } private static void initializeGraalCompilePolicyFields(OptionValues options) { compileGraalWithC1Only = Options.CompileGraalWithC1Only.getValue(options) && !IS_IN_NATIVE_IMAGE; String optionValue = Options.GraalCompileOnly.getValue(options); if (optionValue != null) { MethodFilter filter = MethodFilter.parse(optionValue); if (filter.matchesNothing()) { filter = null; } graalCompileOnlyFilter = filter; } } @Override public void printProperties(PrintStream out) { out.println("[Graal properties]"); if (optionsFailure != null) { System.err.printf("Error parsing Graal options: %s%n", optionsFailure.getMessage()); return; } options.printHelp(OptionsParser.getOptionsLoader(), out, GRAAL_OPTION_PROPERTY_PREFIX); } static class Options { // @formatter:off @Option(help = "In tiered mode compile Graal and JVMCI using optimized first tier code.", type = OptionType.Expert) public static final OptionKey<Boolean> CompileGraalWithC1Only = new OptionKey<>(true); @Option(help = "A filter applied to a method the VM has selected for compilation by Graal. " + "A method not matching the filter is redirected to a lower tier compiler. " + "The filter format is the same as for the MethodFilter option.", type = OptionType.Expert) public static final OptionKey<String> GraalCompileOnly = new OptionKey<>(null); // @formatter:on } @Override public HotSpotGraalCompiler createCompiler(JVMCIRuntime runtime) { HotSpotJVMCIRuntime hsRuntime = (HotSpotJVMCIRuntime) runtime; if (optionsFailure != null) { System.err.printf("Error parsing Graal options: %s%nError: A fatal exception has occurred. Program will exit.%n", optionsFailure.getMessage()); HotSpotGraalServices.exit(1, hsRuntime); } CompilerConfigurationFactory factory = CompilerConfigurationFactory.selectFactory(null, options, hsRuntime); if (isGraalPredicate != null) { isGraalPredicate.onCompilerConfigurationFactorySelection(hsRuntime, factory); } HotSpotGraalCompiler compiler = createCompiler("VM", runtime, options, factory); // Only the HotSpotGraalRuntime associated with the compiler created via // jdk.vm.ci.runtime.JVMCIRuntime.getCompiler() is registered for receiving // VM events. locator.onCompilerCreation(compiler); return compiler; }
Creates a new HotSpotGraalRuntime object and a new HotSpotGraalCompiler and returns the latter.
Params:
/** * Creates a new {@link HotSpotGraalRuntime} object and a new {@link HotSpotGraalCompiler} and * returns the latter. * * @param runtimeNameQualifier a qualifier to be added to the {@linkplain GraalRuntime#getName() * name} of the {@linkplain HotSpotGraalCompiler#getGraalRuntime() runtime} created * by this method * @param runtime the JVMCI runtime on which the {@link HotSpotGraalRuntime} is built * @param compilerConfigurationFactory factory for the {@link CompilerConfiguration} */
@SuppressWarnings("try") public static HotSpotGraalCompiler createCompiler(String runtimeNameQualifier, JVMCIRuntime runtime, OptionValues options, CompilerConfigurationFactory compilerConfigurationFactory) { HotSpotJVMCIRuntime jvmciRuntime = (HotSpotJVMCIRuntime) runtime; try (InitTimer t = timer("HotSpotGraalRuntime.<init>")) { HotSpotGraalRuntime graalRuntime = new HotSpotGraalRuntime(runtimeNameQualifier, jvmciRuntime, compilerConfigurationFactory, options); return new HotSpotGraalCompiler(jvmciRuntime, graalRuntime, graalRuntime.getOptions()); } } @Override public CompilationLevelAdjustment getCompilationLevelAdjustment() { return isGraalPredicate != null ? isGraalPredicate.getCompilationLevelAdjustment() : None; } @Override public CompilationLevel adjustCompilationLevel(Object declaringClassObject, String name, String signature, boolean isOsr, CompilationLevel level) { Class<?> declaringClass = (Class<?>) declaringClassObject; return adjustCompilationLevelInternal(declaringClass, level); } static { // Fail-fast detection for package renaming to guard use of package // prefixes in adjustCompilationLevelInternal. assert jdk.vm.ci.services.Services.class.getName().equals("jdk.vm.ci.services.Services"); assert HotSpotGraalCompilerFactory.class.getName().equals("org.graalvm.compiler.hotspot.HotSpotGraalCompilerFactory"); } private CompilationLevel adjustCompilationLevelInternal(Class<?> declaringClass, CompilationLevel level) { assert isGraalPredicate != null; if (level.ordinal() > CompilationLevel.Simple.ordinal()) { if (isGraalPredicate.apply(declaringClass)) { return CompilationLevel.Simple; } } return level; } static boolean shouldExclude(HotSpotResolvedJavaMethod method) { if (graalCompileOnlyFilter != null) { String javaClassName = method.getDeclaringClass().toJavaName(); String name = method.getName(); Signature signature = method.getSignature(); if (graalCompileOnlyFilter.matches(javaClassName, name, signature)) { return false; } return true; } return false; } }