test/hotspot/jtreg/compiler/whitebox/CompilerWhiteBoxTest.java
changeset 47216 71c04702a3d5
parent 42650 1f304d0c888b
child 47687 fb290fd1f9d4
equal deleted inserted replaced
47215:4ebc2e2fb97c 47216:71c04702a3d5
       
     1 /*
       
     2  * Copyright (c) 2013, 2016, Oracle and/or its affiliates. All rights reserved.
       
     3  * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
       
     4  *
       
     5  * This code is free software; you can redistribute it and/or modify it
       
     6  * under the terms of the GNU General Public License version 2 only, as
       
     7  * published by the Free Software Foundation.
       
     8  *
       
     9  * This code is distributed in the hope that it will be useful, but WITHOUT
       
    10  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
       
    11  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
       
    12  * version 2 for more details (a copy is included in the LICENSE file that
       
    13  * accompanied this code).
       
    14  *
       
    15  * You should have received a copy of the GNU General Public License version
       
    16  * 2 along with this work; if not, write to the Free Software Foundation,
       
    17  * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
       
    18  *
       
    19  * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
       
    20  * or visit www.oracle.com if you need additional information or have any
       
    21  * questions.
       
    22  */
       
    23 
       
    24 package compiler.whitebox;
       
    25 
       
    26 import jdk.test.lib.Platform;
       
    27 import sun.hotspot.WhiteBox;
       
    28 import sun.hotspot.code.NMethod;
       
    29 
       
    30 import java.lang.reflect.Executable;
       
    31 import java.util.Objects;
       
    32 import java.util.concurrent.Callable;
       
    33 import java.util.function.Function;
       
    34 
       
    35 /**
       
    36  * Abstract class for WhiteBox testing of JIT.
       
    37  * Depends on jdk.test.lib.Platform from testlibrary.
       
    38  *
       
    39  * @author igor.ignatyev@oracle.com
       
    40  */
       
    41 public abstract class CompilerWhiteBoxTest {
       
    42     /** {@code CompLevel::CompLevel_none} -- Interpreter */
       
    43     public static final int COMP_LEVEL_NONE = 0;
       
    44     /** {@code CompLevel::CompLevel_any}, {@code CompLevel::CompLevel_all} */
       
    45     public static final int COMP_LEVEL_ANY = -2;
       
    46     /** {@code CompLevel::CompLevel_aot} -- AOT */
       
    47     public static final int COMP_LEVEL_AOT = -1;
       
    48     /** {@code CompLevel::CompLevel_simple} -- C1 */
       
    49     public static final int COMP_LEVEL_SIMPLE = 1;
       
    50     /** {@code CompLevel::CompLevel_limited_profile} -- C1, invocation & backedge counters */
       
    51     public static final int COMP_LEVEL_LIMITED_PROFILE = 2;
       
    52     /** {@code CompLevel::CompLevel_full_profile} -- C1, invocation & backedge counters + mdo */
       
    53     public static final int COMP_LEVEL_FULL_PROFILE = 3;
       
    54     /** {@code CompLevel::CompLevel_full_optimization} -- C2 or Shark */
       
    55     public static final int COMP_LEVEL_FULL_OPTIMIZATION = 4;
       
    56     /** Maximal value for CompLevel */
       
    57     public static final int COMP_LEVEL_MAX = COMP_LEVEL_FULL_OPTIMIZATION;
       
    58 
       
    59     /** Instance of WhiteBox */
       
    60     protected static final WhiteBox WHITE_BOX = WhiteBox.getWhiteBox();
       
    61     /** Value of {@code -XX:CompileThreshold} */
       
    62     protected static final int COMPILE_THRESHOLD
       
    63             = Integer.parseInt(getVMOption("CompileThreshold", "10000"));
       
    64     /** Value of {@code -XX:BackgroundCompilation} */
       
    65     protected static final boolean BACKGROUND_COMPILATION
       
    66             = Boolean.valueOf(getVMOption("BackgroundCompilation", "true"));
       
    67     protected static final boolean USE_COUNTER_DECAY
       
    68             = Boolean.valueOf(getVMOption("UseCounterDecay", "true"));
       
    69     /** Value of {@code -XX:TieredCompilation} */
       
    70     protected static final boolean TIERED_COMPILATION
       
    71             = Boolean.valueOf(getVMOption("TieredCompilation", "false"));
       
    72     /** Value of {@code -XX:TieredStopAtLevel} */
       
    73     protected static final int TIERED_STOP_AT_LEVEL
       
    74             = Integer.parseInt(getVMOption("TieredStopAtLevel", "0"));
       
    75     /** Flag for verbose output, true if {@code -Dverbose} specified */
       
    76     protected static final boolean IS_VERBOSE
       
    77             = System.getProperty("verbose") != null;
       
    78     /** invocation count to trigger compilation */
       
    79     public static final int THRESHOLD;
       
    80     /** invocation count to trigger OSR compilation */
       
    81     protected static final long BACKEDGE_THRESHOLD;
       
    82 
       
    83     static {
       
    84         if (TIERED_COMPILATION) {
       
    85             BACKEDGE_THRESHOLD = THRESHOLD = 150000;
       
    86         } else {
       
    87             THRESHOLD = COMPILE_THRESHOLD;
       
    88             BACKEDGE_THRESHOLD = COMPILE_THRESHOLD * Long.parseLong(getVMOption(
       
    89                     "OnStackReplacePercentage"));
       
    90         }
       
    91     }
       
    92 
       
    93     /**
       
    94      * Returns value of VM option.
       
    95      *
       
    96      * @param name option's name
       
    97      * @return value of option or {@code null}, if option doesn't exist
       
    98      * @throws NullPointerException if name is null
       
    99      */
       
   100     protected static String getVMOption(String name) {
       
   101         Objects.requireNonNull(name);
       
   102         return Objects.toString(WHITE_BOX.getVMFlag(name), null);
       
   103     }
       
   104 
       
   105     /**
       
   106      * Returns value of VM option or default value.
       
   107      *
       
   108      * @param name         option's name
       
   109      * @param defaultValue default value
       
   110      * @return value of option or {@code defaultValue}, if option doesn't exist
       
   111      * @throws NullPointerException if name is null
       
   112      * @see #getVMOption(String)
       
   113      */
       
   114     protected static String getVMOption(String name, String defaultValue) {
       
   115         String result = getVMOption(name);
       
   116         return result == null ? defaultValue : result;
       
   117     }
       
   118 
       
   119     /** copy of is_c1_compile(int) from utilities/globalDefinitions.hpp */
       
   120     protected static boolean isC1Compile(int compLevel) {
       
   121         return (compLevel > COMP_LEVEL_NONE)
       
   122                 && (compLevel < COMP_LEVEL_FULL_OPTIMIZATION);
       
   123     }
       
   124 
       
   125     /** copy of is_c2_compile(int) from utilities/globalDefinitions.hpp */
       
   126     protected static boolean isC2Compile(int compLevel) {
       
   127         return compLevel == COMP_LEVEL_FULL_OPTIMIZATION;
       
   128     }
       
   129 
       
   130     protected static void main(
       
   131             Function<TestCase, CompilerWhiteBoxTest> constructor,
       
   132             String[] args) {
       
   133         if (args.length == 0) {
       
   134             for (TestCase test : SimpleTestCase.values()) {
       
   135                 constructor.apply(test).runTest();
       
   136             }
       
   137         } else {
       
   138             for (String name : args) {
       
   139                 constructor.apply(SimpleTestCase.valueOf(name)).runTest();
       
   140             }
       
   141         }
       
   142     }
       
   143 
       
   144     /** tested method */
       
   145     protected final Executable method;
       
   146     protected final TestCase testCase;
       
   147 
       
   148     /**
       
   149      * Constructor.
       
   150      *
       
   151      * @param testCase object, that contains tested method and way to invoke it.
       
   152      */
       
   153     protected CompilerWhiteBoxTest(TestCase testCase) {
       
   154         Objects.requireNonNull(testCase);
       
   155         System.out.println("TEST CASE:" + testCase.name());
       
   156         method = testCase.getExecutable();
       
   157         this.testCase = testCase;
       
   158     }
       
   159 
       
   160     /**
       
   161      * Template method for testing. Prints tested method's info before
       
   162      * {@linkplain #test()} and after {@linkplain #test()} or on thrown
       
   163      * exception.
       
   164      *
       
   165      * @throws RuntimeException if method {@linkplain #test()} throws any
       
   166      *                          exception
       
   167      * @see #test()
       
   168      */
       
   169     protected final void runTest() {
       
   170         if (Platform.isInt()) {
       
   171             throw new Error("TESTBUG: test can not be run in interpreter");
       
   172         }
       
   173         System.out.println("at test's start:");
       
   174         printInfo();
       
   175         try {
       
   176             test();
       
   177         } catch (Exception e) {
       
   178             System.out.printf("on exception '%s':", e.getMessage());
       
   179             printInfo();
       
   180             e.printStackTrace();
       
   181             if (e instanceof RuntimeException) {
       
   182                 throw (RuntimeException) e;
       
   183             }
       
   184             throw new RuntimeException(e);
       
   185         }
       
   186         System.out.println("at test's end:");
       
   187         printInfo();
       
   188     }
       
   189 
       
   190     /**
       
   191      * Checks, that {@linkplain #method} is not compiled at the given compilation
       
   192      * level or above.
       
   193      *
       
   194      * @param compLevel
       
   195      *
       
   196      * @throws RuntimeException if {@linkplain #method} is in compiler queue or
       
   197      *                          is compiled, or if {@linkplain #method} has zero
       
   198      *                          compilation level.
       
   199      */
       
   200     protected final void checkNotCompiled(int compLevel) {
       
   201         if (WHITE_BOX.isMethodQueuedForCompilation(method)) {
       
   202             throw new RuntimeException(method + " must not be in queue");
       
   203         }
       
   204         if (WHITE_BOX.getMethodCompilationLevel(method, false) >= compLevel) {
       
   205             throw new RuntimeException(method + " comp_level must be >= maxCompLevel");
       
   206         }
       
   207         if (WHITE_BOX.getMethodCompilationLevel(method, true) >= compLevel) {
       
   208             throw new RuntimeException(method + " osr_comp_level must be >= maxCompLevel");
       
   209         }
       
   210     }
       
   211 
       
   212     /**
       
   213      * Checks, that {@linkplain #method} is not compiled.
       
   214      *
       
   215      * @throws RuntimeException if {@linkplain #method} is in compiler queue or
       
   216      *                          is compiled, or if {@linkplain #method} has zero
       
   217      *                          compilation level.
       
   218      */
       
   219     protected final void checkNotCompiled() {
       
   220         waitBackgroundCompilation();
       
   221         checkNotCompiled(true);
       
   222         checkNotCompiled(false);
       
   223     }
       
   224 
       
   225     /**
       
   226      * Checks, that {@linkplain #method} is not (OSR-)compiled.
       
   227      *
       
   228      * @param isOsr Check for OSR compilation if true
       
   229      * @throws RuntimeException if {@linkplain #method} is in compiler queue or
       
   230      *                          is compiled, or if {@linkplain #method} has zero
       
   231      *                          compilation level.
       
   232      */
       
   233     protected final void checkNotCompiled(boolean isOsr) {
       
   234         if (WHITE_BOX.isMethodQueuedForCompilation(method)) {
       
   235             throw new RuntimeException(method + " must not be in queue");
       
   236         }
       
   237         if (WHITE_BOX.isMethodCompiled(method, isOsr)) {
       
   238             throw new RuntimeException(method + " must not be " +
       
   239                                        (isOsr ? "osr_" : "") + "compiled");
       
   240         }
       
   241         if (WHITE_BOX.getMethodCompilationLevel(method, isOsr) != 0) {
       
   242             throw new RuntimeException(method + (isOsr ? " osr_" : " ") +
       
   243                                        "comp_level must be == 0");
       
   244         }
       
   245     }
       
   246 
       
   247     /**
       
   248      * Checks, that {@linkplain #method} is compiled.
       
   249      *
       
   250      * @throws RuntimeException if {@linkplain #method} isn't in compiler queue
       
   251      *                          and isn't compiled, or if {@linkplain #method}
       
   252      *                          has nonzero compilation level
       
   253      */
       
   254     protected final void checkCompiled() {
       
   255         final long start = System.currentTimeMillis();
       
   256         waitBackgroundCompilation();
       
   257         if (WHITE_BOX.isMethodQueuedForCompilation(method)) {
       
   258             System.err.printf("Warning: %s is still in queue after %dms%n",
       
   259                     method, System.currentTimeMillis() - start);
       
   260             return;
       
   261         }
       
   262         if (!WHITE_BOX.isMethodCompiled(method, testCase.isOsr())) {
       
   263             throw new RuntimeException(method + " must be "
       
   264                     + (testCase.isOsr() ? "osr_" : "") + "compiled");
       
   265         }
       
   266         if (WHITE_BOX.getMethodCompilationLevel(method, testCase.isOsr())
       
   267                 == 0) {
       
   268             throw new RuntimeException(method
       
   269                     + (testCase.isOsr() ? " osr_" : " ")
       
   270                     + "comp_level must be != 0");
       
   271         }
       
   272     }
       
   273 
       
   274     protected final void deoptimize() {
       
   275         WHITE_BOX.deoptimizeMethod(method, testCase.isOsr());
       
   276         if (testCase.isOsr()) {
       
   277             WHITE_BOX.deoptimizeMethod(method, false);
       
   278         }
       
   279     }
       
   280 
       
   281     protected final int getCompLevel() {
       
   282         NMethod nm = NMethod.get(method, testCase.isOsr());
       
   283         return nm == null ? COMP_LEVEL_NONE : nm.comp_level;
       
   284     }
       
   285 
       
   286     protected final boolean isCompilable() {
       
   287         return WHITE_BOX.isMethodCompilable(method, COMP_LEVEL_ANY,
       
   288                 testCase.isOsr());
       
   289     }
       
   290 
       
   291     protected final boolean isCompilable(int compLevel) {
       
   292         return WHITE_BOX
       
   293                 .isMethodCompilable(method, compLevel, testCase.isOsr());
       
   294     }
       
   295 
       
   296     protected final void makeNotCompilable() {
       
   297         WHITE_BOX.makeMethodNotCompilable(method, COMP_LEVEL_ANY,
       
   298                 testCase.isOsr());
       
   299     }
       
   300 
       
   301     protected final void makeNotCompilable(int compLevel) {
       
   302         WHITE_BOX.makeMethodNotCompilable(method, compLevel, testCase.isOsr());
       
   303     }
       
   304 
       
   305     /**
       
   306      * Waits for completion of background compilation of {@linkplain #method}.
       
   307      */
       
   308     protected final void waitBackgroundCompilation() {
       
   309         waitBackgroundCompilation(method);
       
   310     }
       
   311 
       
   312     /**
       
   313      * Waits for completion of background compilation of the given executable.
       
   314      *
       
   315      * @param executable Executable
       
   316      */
       
   317     public static final void waitBackgroundCompilation(Executable executable) {
       
   318         if (!BACKGROUND_COMPILATION) {
       
   319             return;
       
   320         }
       
   321         final Object obj = new Object();
       
   322         for (int i = 0; i < 100
       
   323                 && WHITE_BOX.isMethodQueuedForCompilation(executable); ++i) {
       
   324             synchronized (obj) {
       
   325                 try {
       
   326                     obj.wait(100);
       
   327                 } catch (InterruptedException e) {
       
   328                     Thread.currentThread().interrupt();
       
   329                 }
       
   330             }
       
   331         }
       
   332     }
       
   333 
       
   334     /**
       
   335      * Prints information about {@linkplain #method}.
       
   336      */
       
   337     protected final void printInfo() {
       
   338         System.out.printf("%n%s:%n", method);
       
   339         System.out.printf("\tcompilable:\t%b%n",
       
   340                 WHITE_BOX.isMethodCompilable(method, COMP_LEVEL_ANY, false));
       
   341         boolean isCompiled = WHITE_BOX.isMethodCompiled(method, false);
       
   342         System.out.printf("\tcompiled:\t%b%n", isCompiled);
       
   343         if (isCompiled) {
       
   344             System.out.printf("\tcompile_id:\t%d%n",
       
   345                     NMethod.get(method, false).compile_id);
       
   346         }
       
   347         System.out.printf("\tcomp_level:\t%d%n",
       
   348                 WHITE_BOX.getMethodCompilationLevel(method, false));
       
   349         System.out.printf("\tosr_compilable:\t%b%n",
       
   350                 WHITE_BOX.isMethodCompilable(method, COMP_LEVEL_ANY, true));
       
   351         isCompiled = WHITE_BOX.isMethodCompiled(method, true);
       
   352         System.out.printf("\tosr_compiled:\t%b%n", isCompiled);
       
   353         if (isCompiled) {
       
   354             System.out.printf("\tosr_compile_id:\t%d%n",
       
   355                     NMethod.get(method, true).compile_id);
       
   356         }
       
   357         System.out.printf("\tosr_comp_level:\t%d%n",
       
   358                 WHITE_BOX.getMethodCompilationLevel(method, true));
       
   359         System.out.printf("\tin_queue:\t%b%n",
       
   360                 WHITE_BOX.isMethodQueuedForCompilation(method));
       
   361         System.out.printf("compile_queues_size:\t%d%n%n",
       
   362                 WHITE_BOX.getCompileQueuesSize());
       
   363     }
       
   364 
       
   365     /**
       
   366      * Executes testing.
       
   367      */
       
   368     protected abstract void test() throws Exception;
       
   369 
       
   370     /**
       
   371      * Tries to trigger compilation of {@linkplain #method} by call
       
   372      * {@linkplain TestCase#getCallable()} enough times.
       
   373      *
       
   374      * @return accumulated result
       
   375      * @see #compile(int)
       
   376      */
       
   377     protected final int compile() throws Exception {
       
   378         if (USE_COUNTER_DECAY) {
       
   379             throw new Exception("Tests using compile method must turn off counter decay for reliability");
       
   380         }
       
   381         if (testCase.isOsr()) {
       
   382             return compile(1);
       
   383         } else {
       
   384             return compile(THRESHOLD);
       
   385         }
       
   386     }
       
   387 
       
   388     /**
       
   389      * Tries to trigger compilation of {@linkplain #method} by call
       
   390      * {@linkplain TestCase#getCallable()} specified times.
       
   391      *
       
   392      * @param count invocation count
       
   393      * @return accumulated result
       
   394      */
       
   395     protected final int compile(int count) {
       
   396         int result = 0;
       
   397         Integer tmp;
       
   398         for (int i = 0; i < count; ++i) {
       
   399             try {
       
   400                 tmp = testCase.getCallable().call();
       
   401             } catch (Exception e) {
       
   402                 tmp = null;
       
   403             }
       
   404             result += tmp == null ? 0 : tmp;
       
   405         }
       
   406         if (IS_VERBOSE) {
       
   407             System.out.println("method was invoked " + count + " times");
       
   408         }
       
   409         return result;
       
   410     }
       
   411 
       
   412     /**
       
   413      * Utility interface provides tested method and object to invoke it.
       
   414      */
       
   415     public interface TestCase {
       
   416         /** the name of test case */
       
   417         String name();
       
   418 
       
   419         /** tested method */
       
   420         Executable getExecutable();
       
   421 
       
   422         /** object to invoke {@linkplain #getExecutable()} */
       
   423         Callable<Integer> getCallable();
       
   424 
       
   425         /** flag for OSR test case */
       
   426         boolean isOsr();
       
   427     }
       
   428 
       
   429     /**
       
   430      * @return {@code true} if the current test case is OSR and the mode is
       
   431      *          Xcomp, otherwise {@code false}
       
   432      */
       
   433     protected boolean skipXcompOSR() {
       
   434         boolean result = testCase.isOsr() && Platform.isComp();
       
   435         if (result && IS_VERBOSE) {
       
   436             System.err.printf("Warning: %s is not applicable in %s%n",
       
   437                     testCase.name(), Platform.vmInfo);
       
   438         }
       
   439         return result;
       
   440     }
       
   441 
       
   442     /**
       
   443      * Skip the test for the specified value of Tiered Compilation
       
   444      * @param value of TieredCompilation the test should not run with
       
   445      * @return {@code true} if the test should be skipped,
       
   446      *         {@code false} otherwise
       
   447      */
       
   448     public static boolean skipOnTieredCompilation(boolean value) {
       
   449         if (value == CompilerWhiteBoxTest.TIERED_COMPILATION) {
       
   450             System.err.println("Test isn't applicable w/ "
       
   451                     + (value ? "enabled" : "disabled")
       
   452                     + "TieredCompilation. Skip test.");
       
   453             return true;
       
   454         }
       
   455         return false;
       
   456     }
       
   457 }
       
   458