8254129: IR Test Framework to support regex-based matching on the IR in JTreg compiler tests

Co-authored-by: Christian Hagedorn <chagedorn@openjdk.org>
Co-authored-by: Tobias Hartmann <thartmann@openjdk.org>
Reviewed-by: iignatyev
This commit is contained in:
Christian Hagedorn 2021-06-07 14:11:50 +00:00
parent 270ec975b6
commit 3396b69fc9
67 changed files with 13454 additions and 0 deletions

View File

@ -0,0 +1,121 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.shared.TestRunException;
import compiler.lib.ir_framework.test.TestVM;
import jdk.test.lib.Utils;
import java.lang.reflect.Method;
import java.util.Arrays;
import java.util.Random;
import java.util.stream.Collectors;
/**
* Base info class which provides some useful utility methods and information about a test.
* <p>
* <b>Base tests</b> and <b>checked tests</b> use {@link TestInfo} while <b>custom run tests</b> use {@link RunInfo}.
*
* @see Test
* @see Check
* @see Run
*/
abstract public class AbstractInfo {
private static final Random RANDOM = Utils.getRandomInstance();
protected final Class<?> testClass;
private boolean onWarmUp = true;
AbstractInfo(Class<?> testClass) {
this.testClass = testClass;
}
/**
* Get the initialized {@link Random} object.
*
* @return the random object.
*/
public Random getRandom() {
return RANDOM;
}
/**
* Returns a boolean indicating if the framework is currently warming up the associated test.
*
* @return the warm-up status of the associated test.
*
* @see Warmup
*/
public boolean isWarmUp() {
return onWarmUp;
}
/**
* Get the method object of the method {@code name} of class {@code c} with arguments {@code args}.
*
* @param c the class containing the method.
* @param name the name of the method.
* @param args the arguments of the method, leave empty if no arguments.
*
* @return the method object of the requested method.
*/
public Method getMethod(Class<?> c, String name, Class<?>... args) {
try {
return c.getMethod(name, args);
} catch (NoSuchMethodException e) {
String parameters = args == null || args.length == 0 ? "" :
" with arguments [" + Arrays.stream(args).map(Class::getName).collect(Collectors.joining(",")) + "]";
throw new TestRunException("Could not find method " + name + " in " + c + parameters, e);
}
}
/**
* Get the method object of the method {@code name} of the test class with arguments {@code args}.
*
* @param name the name of the method in the test class.
* @param args the arguments of the method, leave empty if no arguments.
*
* @return the method object of the requested method in the test class.
*/
public Method getTestClassMethod(String name, Class<?>... args) {
return getMethod(testClass, name, args);
}
/**
* Returns a boolean indicating if the test VM runs with flags that allow C2 compilations.
*
* @return {@code true} if C2 compilations are allowed;
* {@code false} otherwise (run with {@code -XX:TieredStopAtLevel={1,2,3}, -XX:-UseCompiler}).
*/
public boolean isC2CompilationEnabled() {
return TestVM.USE_COMPILER && !TestVM.TEST_C1;
}
/**
* Called by {@link TestFramework} when the warm-up is finished. Should not be called by user code.
*/
public void setWarmUpFinished() {
onWarmUp = false;
}
}

View File

@ -0,0 +1,81 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
/**
* Well-defined argument values that can be used in the {@link Arguments} annotation at a {@link Test} method for a
* <b>base test</b> or a <b>checked test</b>.
*
* @see Arguments
* @see Test
* @see Check
*/
public enum Argument {
/**
* Provides the default value for any kind of primitive type and object type if the class provides a default constructor.
*/
DEFAULT,
/**
* Provides the number 42 for any primitive number type.
*/
NUMBER_42,
/**
* Provides the number -42 for any primitive number type.
*/
NUMBER_MINUS_42,
/**
* Provides the minimum value of the specified primitive number type.
*/
MIN,
/**
* Provides the maximum value of the specified primitive number type.
*/
MAX,
/**
* Provides the boolean value false.
*/
FALSE,
/**
* Provides the boolean value true.
*/
TRUE,
/**
* Provides a different boolean value on each test invocation, starting with false.
*/
BOOLEAN_TOGGLE_FIRST_FALSE,
/**
* Provides a different boolean value on each test invocation, starting with true.
*/
BOOLEAN_TOGGLE_FIRST_TRUE,
/**
* Provides a random primitive value on the first test invocation and reuses the same value for all invocation of the test.
* Float and double values are restricted to the range [-10000,10000].
*/
RANDOM_ONCE,
/**
* Provides a different random primitive value on each test invocation.
* Float and double values are restricted to the range [-10000,10000].
*/
RANDOM_EACH,
}

View File

@ -0,0 +1,43 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* This annotation is used to specify well-defined {@link Argument} values for test methods (specifying {@link Test}) when
* used as part of a <b>base test</b> or <b>checked test</b>.
*
* @see Argument
* @see Test
* @see Check
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface Arguments {
/**
* Get the argument value.
*/
Argument[] value();
}

View File

@ -0,0 +1,114 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.shared.TestFormatException;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* Annotation for a check method of a <b>checked test</b>.
*
* <p>
* Let {@code t} be a test method specifying the {@link Test @Test} annotation and {@code c} be a check method specifying
* the {@code @Check(test = "t")} annotation. These two methods represent a so-called <i>checked test</i>. The only
* difference to a <i>base test</i> (see {@link Test}) is that the framework will invoke the check method {@code c}
* directly after the invocation of the test method {@code t} which allows to do some additional verification,
* including the return value of {@code t}. The framework does the following, similar as for <i>base tests</i>:
* <ol>
* <li><p>The framework warms {@code t} up by invoking it for a predefined number of iterations (default: 2000)
* or any number specified by an additional {@link Warmup @Warmup} annotation at {@code t} or by using
* {@link TestFramework#setDefaultWarmup(int)} (could also be 0 which skips the warm-up completely which is
* similar to simulating {@code -Xcomp}). After each invocation of {@code t}, the framework also invokes
* {@code c} if the {@code @Check} annotation specifies {@link CheckAt#EACH_INVOCATION} at {@link #when()}.
* More information about the warm-up in general can be found at {@link Warmup}</li>
* <li><p>After the warm-up, the framework compiles {@code t} at the specified compilation level set by
* {@link Test#compLevel()} (default {@link CompLevel#ANY} will pick the highest available level which is
* usually {@link CompLevel#C2}).</li>
* <li><p>The framework invokes {@code t} one more time to run the compilation. Afterwards, the framework will
* always invoke {@code c} again to be able perform additional checks after the compilation of {@code t}.</li>
* <li><p>The framework checks any specified {@link IR @IR} constraints at the test method {@code t}.
* More information about IR matching can be found at {@link IR}.</li>
* </ol>
*
* <p>
* The test method {@code t} has the same properties and follows the same constraints as stated in {@link Test}.
* <p>
* The following additional constraints must be met for the test method {@code t} and check method {@code c}:
* <ul>
* <li><p>{@code c} must specify the method name {@code t} as property in {@code @Check(test = "t")}
* (see {@link #test()}. Specifying a non-{@code @Test} annotated method or a {@code @Test} method that
* has already been used by another {@code @Check} or {@link Run @Run} method results in a {@link TestFormatException}.
* <li><p>{@code c} can specify the following method parameter combinations:
* <ul>
* <li><p>void</li>
* <li><p>One parameter: {@link TestInfo} which provides some information about {@code t} and utility methods.</li>
* <li><p>One parameter: the <i>exact</i> same type as the return value of {@code t}. When {@code c} is
* invoked by the framework, this parameter contains the return value of {@code t}.</li>
* <li><p>1st parameter: {@link TestInfo}; 2nd parameter: the <i>exact</i> same type as the return value of
* {@code t} (see above)</li>
* <li><p> Any other combination will result in a {@link TestFormatException}.
* </ul>
* <li><p>{@code c} is not compiled nor inlined.
* <li><p>{@code c} must be part of the test class. Using {@code @Check} in nested or other classes is not allowed.</li>
* <li><p>{@code c} cannot specify any helper-method-specific compile command annotations
* ({@link ForceCompile @ForceCompile}, {@link DontCompile @DontCompile}, {@link ForceInline @ForceInline},
* {@link DontInline @DontInline}).</li>
* </ul>
*
* <p>
* If no verification is required, use a <i>base test</i> (see {@link Test}). If {@code t} must be invoked with more
* complex or varying arguments and/or the {@code t} must be invoked differently in subsequent invocations, use a
* <i>custom run test</i> (see {@link Run}).
*
* <p>
* Examples on how to write checked tests can be found in {@link jdk.test.lib.hotspot.ir_framework.examples.CheckedTestExample}
* and also as part of the internal testing in the package {@link jdk.test.lib.hotspot.ir_framework.tests}.
*
* @see Test
* @see TestInfo
* @see CheckAt
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface Check {
/**
* The unique associated {@link Test} method for this {@code @Check} annotated check method. The framework will directly
* invoke the {@code @Check} method after each invocation or only after the compilation of the associated {@code @Test}
* method (depending on the value set with {@link #when()}).
* <p>
* If a non-{@code @Test} annotated method or a {@code @Test} method that has already been used by another
* {@code @Check} or {@link Run} method is specified, then a {@link TestFormatException} is thrown.
*
* @see Test
*/
String test();
/**
* When should the {@code @Check} method be invoked? By default, the check is done after each invocation which is
* encouraged if performance is not critical.
*
* @see CheckAt
*/
CheckAt when() default CheckAt.EACH_INVOCATION;
}

View File

@ -0,0 +1,43 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
/**
* This enum is used in {@link Check#when()} of a <b>checked test</b> to specify when the framework will invoke the
* check method after invoking the associated {@link Test} method.
*
* @see Check
* @see Test
*/
public enum CheckAt {
/**
* Default: Invoke the {@link Check} method each time after invoking the associated {@link Test} method.
*/
EACH_INVOCATION,
/**
* Invoke the {@link Check} method only once after the warm-up of the associated {@link Test} method had been completed
* and the framework has compiled the associated {@link Test} method.
*/
COMPILED,
}

View File

@ -0,0 +1,171 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.shared.TestFrameworkException;
import compiler.lib.ir_framework.shared.TestRun;
import compiler.lib.ir_framework.shared.TestRunException;
import compiler.lib.ir_framework.test.TestVM;
import jdk.test.lib.Utils;
import java.lang.reflect.Executable;
import java.util.HashMap;
import java.util.Map;
/**
* Compilation levels used by the framework to initiate a compilation of a method. The compilation levels map to the used
* levels in HotSpot (apart from the framework specific values {@link #SKIP} and {@link #WAIT_FOR_COMPILATION} that cannot
* be found in HotSpot). The HotSpot specific levels must be in sync with hotspot/share/compiler/compilerDefinitions.hpp.
*
* <p>
* The compilation levels can be specified in the {@link Test}, {@link ForceCompile}, and
* {@link ForceCompileClassInitializer} annotation.
*
* @see Test
* @see ForceCompile
* @see ForceCompileClassInitializer
*/
public enum CompLevel {
/**
* Can only be used at {@link Test#compLevel()}. After the warm-up, the framework keeps invoking the test over a span
* of 10s (configurable by setting the property flag {@code -DWaitForCompilationTimeout}) until HotSpot compiles the
* {@link Test} method. If the method was not compiled after 10s, an exception is thrown. The framework does not wait
* for the compilation if the test VM is run with {@code -Xcomp}, {@code -XX:-UseCompiler}, or
* {@code -DExcludeRandom=true}.
*/
WAIT_FOR_COMPILATION(-4),
/**
* Can only be used at {@link Test#compLevel()}. Skip a compilation of the {@link Test @Test} method completely.
*/
SKIP(-3),
/**
* Use any compilation level depending on the usage:
* <ul>
* <li><p>{@link Test @Test}, {@link ForceCompile @ForceCompile}: Use the highest available compilation level
* which is usually C2.</li>
* <li><p>{@link DontCompile @DontCompile}: Prevents any compilation of the associated helper method.</li>
* </ul>
*/
ANY(-1),
/**
* Compilation level 1: C1 compilation without any profile information.
*/
C1_SIMPLE(1),
/**
* Compilation level 2: C1 compilation with limited profile information: Includes Invocation and backedge counters.
*/
C1_LIMITED_PROFILE(2),
/**
* Compilation level 3: C1 compilation with full profile information: Includes Invocation and backedge counters with MDO.
*/
C1_FULL_PROFILE(3),
/**
* Compilation level 4: C2 compilation with full optimizations.
*/
C2(4),
;
private static final Map<Integer, CompLevel> TYPES_BY_VALUE = new HashMap<>();
private final int value;
static {
for (CompLevel level : CompLevel.values()) {
TYPES_BY_VALUE.put(level.value, level);
}
}
CompLevel(int level) {
this.value = level;
}
/**
* Get the compilation level as integer value. These will match the levels specified in HotSpot (if available).
*
* @return the compilation level as integer.
*/
public int getValue() {
return value;
}
/**
* Get the compilation level enum from the specified integer.
*
* @param value the compilation level as integer.
* @throws TestRunException if {@code value} does not specify a valid compilation level.
* @return the compilation level enum for {@code value}.
*/
public static CompLevel forValue(int value) {
CompLevel level = TYPES_BY_VALUE.get(value);
TestRun.check(level != null, "Invalid compilation level " + value);
return level;
}
/**
* Called by {@link TestFramework} to check if this compilation level is not part of the compiler.
*/
public boolean isNotCompilationLevelOfCompiler(Compiler c) {
return switch (c) {
case C1 -> !isC1();
case C2 -> this != C2;
default -> throw new TestFrameworkException("Should not be called with compiler " + c);
};
}
/**
* Called by {@link TestFramework} to flip compilation levels.
*/
public CompLevel flipCompLevel() {
switch (this) {
case C1_SIMPLE, C1_LIMITED_PROFILE, C1_FULL_PROFILE -> {
return CompLevel.C2;
}
case C2 -> {
return CompLevel.C1_SIMPLE;
}
}
return this;
}
/**
* Called by {@link TestFramework}. Return the compilation level when only allowing a compilation with the specified
* compiler.
*/
public CompLevel excludeCompilationRandomly(Executable ex) {
if (Utils.getRandomInstance().nextBoolean()) {
// No exclusion
return this;
}
Compiler compiler = TestVM.excludeRandomly(ex);
return switch (compiler) {
case ANY -> SKIP;
case C1 -> isC1() ? SKIP : this;
case C2 -> this == C2 ? SKIP : this;
};
}
private boolean isC1() {
return this == C1_SIMPLE || this == C1_LIMITED_PROFILE || this == C1_FULL_PROFILE;
}
}

View File

@ -0,0 +1,63 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
/**
* Compilers to select for {@link DontCompile}. HotSpot does not handle the exclusion of a C1 method at a specific level.
* It can only exclude a method for the entire C1 compilation. Thus, this annotation is provided for {@link DontCompile}
* instead of {@link CompLevel}.
*
* @see DontCompile
*/
public enum Compiler {
/**
* Selecting both the C1 and C2 compiler. This must be in sync with hotspot/share/compiler/compilerDefinitions.hpp.
*/
ANY(-1),
/**
* The C1 compiler.
*/
C1(1),
/**
* The C2 compiler.
*/
C2(4),
;
private final int value;
Compiler(int level) {
this.value = level;
}
/**
* Get the compilation level as integer value. These will match the levels specified in HotSpot (if available).
*
* @return the compilation level as integer.
*/
public int getValue() {
return value;
}
}

View File

@ -0,0 +1,49 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.shared.TestFormatException;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* Prevent a compilation of the annotated <b>helper method</b> (not specifying {@link Test @Test},
* {@link Check @Check} or {@link Run @Run}) with the specified compiler.
*
* <ul>
* <li><p>{@link Compiler#ANY} (default): No C1 or C2 compilation.</li>
* <li><p>{@link Compiler#C1}: No C1 compilation, C2 compilation still possible.</li>
* <li><p>{@link Compiler#C2}: No C2 compilation, C1 compilation still possible.</li>
* </ul>
* <p>
* Using this annotation on <i>non-helper methods</i> results in a {@link TestFormatException TestFormatException}.
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface DontCompile {
/**
* The compiler with which a compilation of a helper method is excluded.
*/
Compiler value() default Compiler.ANY;
}

View File

@ -0,0 +1,38 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.shared.TestFormatException;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* Prevent an inlining of the annotated <b>helper method</b> (not specifying {@link Test @Test}, {@link Check @Check},
* or {@link Run @Run}). <i>Non-helper methods</i> are never inlined. Explicitly using this annotation on
* <i>non-helper methods</i> results in a {@link TestFormatException}.
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface DontInline {
}

View File

@ -0,0 +1,45 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.shared.TestFormatException;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* Force a compilation of the annotated <b>helper method</b> (not specifying {@link Test @Test}, {@link Check @Check},
* or {@link Test @Run}) immediately at the specified level. {@link CompLevel#SKIP} and
* {@link CompLevel#WAIT_FOR_COMPILATION} do not apply and result in a {@link TestFormatException}.
*
* <p>
* Using this annotation on <i>non-helper</i> methods also results in a {@link TestFormatException}.
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface ForceCompile {
/**
* The compilation level to compile the helper method at.
*/
CompLevel value() default CompLevel.ANY;
}

View File

@ -0,0 +1,45 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.shared.TestFormatException;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* Force a compilation of the static class initializer method ({@code <clinit>}) of the annotated test or helper class
* immediately at the specified level. {@link CompLevel#SKIP} and {@link CompLevel#WAIT_FOR_COMPILATION} do not apply
* and result in a {@link TestFormatException}.
* <p>
* Using this annotation on non-classes also results in a {@link TestFormatException}.
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface ForceCompileClassInitializer {
/**
* The compilation level to compile the static class initializer method ({@code <clinit>}) at.
*/
CompLevel value() default CompLevel.ANY;
}

View File

@ -0,0 +1,37 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.shared.TestFormatException;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* Force an inlining of the annotated <b>helper method</b> (not specifying {@link Test @Test}, {@link Check @Check},
* or {@link Test @Run}). Using this annotation on <i>non-helper methods</i> results in a {@link TestFormatException}.
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface ForceInline {
}

View File

@ -0,0 +1,144 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.driver.IRViolationException;
import java.lang.annotation.Repeatable;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* This annotation is used to define a constraint/rule/check on the resulting IR of a test method (method with
* {@link Test @Test} annotation). A test method can define multiple {@code @IR} rules.
* <p>
* There are two kinds of checks that can be specified:
* <ul>
* <li><p>{@link #failOn()}: Specify a list of (node) regexes that should not be matched on the {@code PrintIdeal} or
* {@code PrintOptoAssembly} output.</li>
* <li><p>{@link #counts()}: Specify a list of ({@code regex,count}) pairs: The (node) {@code regex} should be matched
* for the specified amount in {@code count} on the {@code PrintIdeal} or {@code PrintOptoAssembly} output.</li>
* </ul>
* An IR rule must specify either or both of these two checks. If one or both of the checks fails, an
* {@link IRViolationException} is thrown. A user can provide a custom regex string or specify any of the default node
* regexes defined in {@link IRNode}.
* <p>
* Sometimes, the shape of the resulting IR is changed by commonly used VM flags in such a way that an IR rule no longer
* applies. Generally, the framework does <b>not</b> apply any IR rules when any of the following flags are used:
* {@code -Xint, -XX:-UseCompiler, -XX:TieredStopAtLevel={1,2,3}, -DExcludeRandom=true, -DFlipC1C2=true}.
* Furthermore, a JTreg test could be run with additional VM and Javaoptions flags. The IR verification is <b>not</b>
* performed in this case if any of these JTreg flags is used that is not part of the whitelist specified by
* {@link TestFramework#JTREG_WHITELIST_FLAGS}.
* <p>
* For any other flag specified either by user code (e.g. {@link Scenario#Scenario(int, String...)},
* {@link TestFramework#runWithFlags(String...) etc.} or as part of the JTreg whitelist, IR verification is applied.
* To restrict the application of IR rules when certain flags are present that could change the IR, each {@code @IR}
* annotation can specify additional preconditions on the allowed test VM flags that must hold when an IR rule is applied.
* If the specified preconditions fail, then the framework does not apply the IR rule. These preconditions can be
* set with {@link #applyIf()}, {@link #applyIfNot()}, {@link #applyIfAnd()}, or {@link #applyIfOr()}.
* <p>
* Examples on how to write tests with IR rules can be found in {@link jdk.test.lib.hotspot.ir_framework.examples.IRExample}
* and also as part of the internal testing in {@link jdk.test.lib.hotspot.ir_framework.tests.TestIRMatching}.
*
* @see Test
* @see IRNode
*/
@Retention(RetentionPolicy.RUNTIME)
@Repeatable(IRs.class)
public @interface IR {
/**
* Define a list of (node) regexes. If any of these regexes are matched on the PrintIdeal or PrintOptoAssembly, the
* IR rule fails and an {@link IRViolationException} is thrown.
*/
String[] failOn() default {};
/**
* Define a list of ((node) regexes,count) string pairs: A regex to be matched on the PrintIdeal or PrintOptoAssembly
* is immediately followed by a number specifying how often the regex should be matched. The number can be proceeded
* by comparators ({@code =, !=, <, <=, =>, >}) where the equality operator is optional (default if no comparator is
* specified).
* <p>
* If any constraint on the number of regexes cannot be met, the IR rule fails and an
* {@link IRViolationException} is thrown.
*/
String[] counts() default {};
/**
* Define a single VM flag precondition which <i>must hold</i> when applying the IR rule. If the VM flag precondition
* fails, then the IR rule is not applied. This is useful if a commonly used flag alters the IR in such a way that an IR rule
* would fail.
* <p>
* The precondition is a (flag, value) string pair where the flag must be a valid VM flag and the value must conform
* with the type of the VM flag. A number based flag value can be proceeded with an additional comparator
* ({@code =, !=, <, <=, =>, >}) where the equality operator is optional (default if no comparator is specified).
* <p>
* This is the inverse of {@link #applyIfNot()}. For multiple preconditions, use {@link #applyIfAnd()} or
* {@link #applyIfOr()} depending on the use case.
*/
String[] applyIf() default {};
/**
* Define a single VM flag precondition which <i>must <b>not</b> hold</i> when applying the IR rule. If, however,
* the VM flag precondition holds, then the IR rule is not applied. This could also be defined as <i>negative</i>
* precondition. This is useful if a commonly used flag alters the IR in such a way that an IR rule would fail.
* <p>
* The precondition is a (flag, value) string pair where the flag must be a valid VM flag and the value must conform
* with the type of the VM flag. A number based flag value can be proceeded with an additional comparator
* ({@code =, !=, <, <=, =>, >}) where the equality operator is optional (default if no comparator is specified).
* <p>
* This is the inverse of {@link #applyIf()}. For multiple preconditions, use {@link #applyIfAnd()} or
* {@link #applyIfOr()} depending on the use case.
*/
String[] applyIfNot() default {};
/**
* Define a list of at least two VM flag precondition which <i><b>all</b> must hold</i> when applying the IR rule.
* If the one of the VM flag preconditions does not hold, then the IR rule is not applied. This is useful if
* commonly used flags alter the IR in such a way that an IR rule would fail. This can also be defined as conjunction
* of preconditions.
* <p>
* A precondition is a (flag, value) string pair where the flag must be a valid VM flag and the value must conform
* with the type of the VM flag. A number based flag value can be proceeded with an additional comparator
* ({@code =, !=, <, <=, =>, >}) where the equality operator is optional (default if no comparator is specified).
* <p>
* Use {@link #applyIfOr()} for disjunction and for single precondition constraints use {@link #applyIf()} or
* {@link #applyIfNot()} depending on the use case.
*/
String[] applyIfAnd() default {};
/**
* Define a list of at least two VM flag precondition from which <i><b>at least one</b> must hold</i> when applying
* the IR rule. If none of the VM flag preconditions holds, then the IR rule is not applied. This is useful if
* commonly used flags alter the IR in such a way that an IR rule would fail. This can also be defined as disjunction
* of preconditions.
* <p>
* A precondition is a (flag, value) string pair where the flag must be a valid VM flag and the value must conform
* with the type of the VM flag. A number based flag value can be proceeded with an additional comparator
* ({@code =, !=, <, <=, =>, >}) where the equality operator is optional (default if no comparator is specified).
* <p>
* Use {@link #applyIfAnd()} for conjunction and for single precondition constraints use {@link #applyIf()} or
* {@link #applyIfNot()} depending on the use case.
*/
String[] applyIfOr() default {};
}

View File

@ -0,0 +1,167 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.driver.IRMatcher;
import compiler.lib.ir_framework.shared.TestFormat;
import compiler.lib.ir_framework.shared.TestFormatException;
import java.util.ArrayList;
import java.util.List;
import java.util.regex.Pattern;
/**
* This class provides default regex strings that can be used in {@link IR @IR} annotations to specify IR constraints.
* <p>
* There are two types of default regexes:
* <ul>
* <li><p>Standalone regexes: Use them directly.</li>
* <li><p>Composite regexes: Their names contain "{@code _OF}" and expect another string in a list in
* {@link IR#failOn()} and {@link IR#counts()}. They cannot be use as standalone regex and will result in a
* {@link TestFormatException} when doing so.</li>
* </ul>
*
* @see IR
*/
public class IRNode {
private static final String START = "(\\d+(\\s){2}(";
private static final String MID = ".*)+(\\s){2}===.*";
private static final String END = ")";
public static final String ALLOC = "(.*precise klass .*\\R(.*(movl|xorl|nop|spill).*\\R)*.*call,static wrapper for: _new_instance_Java" + END;
public static final String ALLOC_OF = "(.*precise klass .*";
public static final String ALLOC_ARRAY = "(.*precise klass \\[L.*\\R(.*(movl|xorl|nop|spill).*\\R)*.*call,static wrapper for: _new_array_Java" + END;
public static final String ALLOC_ARRAY_OF = "(.*precise klass \\[L.*";
public static final String CHECKCAST_ARRAY = "(cmp.*precise klass \\[.*;:" + END;
public static final String CHECKCAST_ARRAY_OF = "(cmp.*precise klass \\[.*";
public static final String CHECKCAST_ARRAYCOPY = "(.*call_leaf_nofp,runtime checkcast_arraycopy.*" + END;
public static final String FIELD_ACCESS = "(.*Field: *" + END;
public static final String STORE = START + "Store(B|C|S|I|L|F|D|P|N)" + MID + END;
public static final String STORE_B = START + "StoreB" + MID + END; // Store to boolean is also mapped to byte
public static final String STORE_C = START + "StoreC" + MID + END;
public static final String STORE_I = START + "StoreI" + MID + END; // Store to short is also mapped to int
public static final String STORE_L = START + "StoreL" + MID + END;
public static final String STORE_F = START + "StoreF" + MID + END;
public static final String STORE_D = START + "StoreD" + MID + END;
public static final String STORE_P = START + "StoreP" + MID + END;
public static final String STORE_N = START + "StoreN" + MID + END;
public static final String STORE_OF_CLASS = START + "Store(B|C|S|I|L|F|D|P|N)" + MID + "@\\S*";
public static final String STORE_B_OF_CLASS = START + "StoreB" + MID + "@\\S*";
public static final String STORE_C_OF_CLASS = START + "StoreC" + MID + "@\\S*";
public static final String STORE_I_OF_CLASS = START + "StoreI" + MID + "@\\S*";
public static final String STORE_L_OF_CLASS = START + "StoreL" + MID + "@\\S*";
public static final String STORE_F_OF_CLASS = START + "StoreF" + MID + "@\\S*";
public static final String STORE_D_OF_CLASS = START + "StoreD" + MID + "@\\S*";
public static final String STORE_P_OF_CLASS = START + "StoreP" + MID + "@\\S*";
public static final String STORE_N_OF_CLASS = START + "StoreN" + MID + "@\\S*";
public static final String STORE_OF_FIELD = START + "Store(B|C|S|I|L|F|D|P|N)" + MID + "@.*name=";
public static final String LOAD = START + "Load(B|UB|S|US|I|L|F|D|P|N)" + MID + END;
public static final String LOAD_B = START + "LoadB" + MID + END;
public static final String LOAD_UB = START + "LoadUB" + MID + END; // Load from boolean
public static final String LOAD_S = START + "LoadS" + MID + END;
public static final String LOAD_US = START + "LoadUS" + MID + END; // Load from char
public static final String LOAD_I = START + "LoadI" + MID + END;
public static final String LOAD_L = START + "LoadL" + MID + END;
public static final String LOAD_F = START + "LoadF" + MID + END;
public static final String LOAD_D = START + "LoadD" + MID + END;
public static final String LOAD_P = START + "LoadP" + MID + END;
public static final String LOAD_N = START + "LoadN" + MID + END;
public static final String LOAD_OF_CLASS = START + "Load(B|UB|S|US|I|L|F|D|P|N)" + MID + "@\\S*";
public static final String LOAD_B_OF_CLASS = START + "LoadB" + MID + "@\\S*";
public static final String LOAD_UB_OF_CLASS = START + "LoadUB" + MID + "@\\S*";
public static final String LOAD_S_OF_CLASS = START + "LoadS" + MID + "@\\S*";
public static final String LOAD_US_OF_CLASS = START + "LoadUS" + MID + "@\\S*";
public static final String LOAD_I_OF_CLASS = START + "LoadI" + MID + "@\\S*";
public static final String LOAD_L_OF_CLASS = START + "LoadL" + MID + "@\\S*";
public static final String LOAD_F_OF_CLASS = START + "LoadF" + MID + "@\\S*";
public static final String LOAD_D_OF_CLASS = START + "LoadD" + MID + "@\\S*";
public static final String LOAD_P_OF_CLASS = START + "LoadP" + MID + "@\\S*";
public static final String LOAD_N_OF_CLASS = START + "LoadN" + MID + "@\\S*";
public static final String LOAD_OF_FIELD = START + "Load(B|C|S|I|L|F|D|P|N)" + MID + "@.*name=";
public static final String LOAD_KLASS = START + "LoadK" + MID + END;
public static final String LOOP = START + "Loop" + MID + "" + END;
public static final String COUNTEDLOOP = START + "CountedLoop\\b" + MID + "" + END;
public static final String COUNTEDLOOP_MAIN = START + "CountedLoop\\b" + MID + "main" + END;
public static final String CALL = START + "CallStaticJava" + MID + END;
public static final String TRAP = START + "CallStaticJava" + MID + "uncommon_trap.*reason" + END;
public static final String PREDICATE_TRAP = START + "CallStaticJava" + MID + "uncommon_trap.*predicate" + END;
public static final String UNSTABLE_IF_TRAP = START + "CallStaticJava" + MID + "uncommon_trap.*unstable_if" + END;
public static final String CLASS_CHECK_TRAP = START + "CallStaticJava" + MID + "uncommon_trap.*class_check" + END;
public static final String NULL_CHECK_TRAP = START + "CallStaticJava" + MID + "uncommon_trap.*null_check" + END;
public static final String NULL_ASSERT_TRAP = START + "CallStaticJava" + MID + "uncommon_trap.*null_assert" + END;
public static final String RANGE_CHECK_TRAP = START + "CallStaticJava" + MID + "uncommon_trap.*range_check" + END;
public static final String UNHANDLED_TRAP = START + "CallStaticJava" + MID + "uncommon_trap.*unhandled" + END;
public static final String INTRINSIC_OR_TYPE_CHECKED_INLINING_TRAP = START + "CallStaticJava" + MID + "uncommon_trap.*intrinsic_or_type_checked_inlining" + END;
public static final String SCOPE_OBJECT = "(.*# ScObj.*" + END;
public static final String MEMBAR = START + "MemBar" + MID + END;
private static final String ALLOC_OF_POSTFIX = ":.*\\R(.*(movl|xorl|nop|spill).*\\R)*.*call,static wrapper for: _new_instance_Java" + END;
private static final String ALLOC_ARRAY_OF_POSTFIX = ";:.*\\R(.*(movl|xorl|nop|spill).*\\R)*.*call,static wrapper for: _new_array_Java" + END;
private static final String CHECKCAST_ARRAY_OF_POSTFIX = ";:" + END;
private static final String STORE_OF_FIELD_POSTFIX = ",.*" + END;
private static final String STORE_OF_CLASS_POSTFIX = "(:|\\+)\\S* \\*" + END;
private static final String LOAD_OF_CLASS_POSTFIX = "(:|\\+)\\S* \\*" + END;
private static final String LOAD_OF_FIELD_POSTFIX = ",.*" + END;
/**
* Called by {@link IRMatcher} to merge special composite nodes together with additional user-defined input.
*/
public static List<String> mergeNodes(String[] nodes) {
List<String> mergedNodes = new ArrayList<>();
for (int i = 0; i < nodes.length; i += 2) {
String node = nodes[i];
switch (node) {
case ALLOC_OF -> mergeCompositeNodes(nodes, mergedNodes, i, node, ALLOC_OF_POSTFIX, "ALLOC_OF");
case ALLOC_ARRAY_OF -> mergeCompositeNodes(nodes, mergedNodes, i, node, ALLOC_ARRAY_OF_POSTFIX, "ALLOC_ARRAY_OF");
case CHECKCAST_ARRAY_OF -> mergeCompositeNodes(nodes, mergedNodes, i, node, CHECKCAST_ARRAY_OF_POSTFIX, "CHECKCAST_ARRAY_OF");
case STORE_OF_CLASS, STORE_B_OF_CLASS, STORE_C_OF_CLASS, STORE_D_OF_CLASS, STORE_F_OF_CLASS, STORE_I_OF_CLASS,
STORE_L_OF_CLASS, STORE_N_OF_CLASS, STORE_P_OF_CLASS
-> mergeCompositeNodes(nodes, mergedNodes, i, node, STORE_OF_CLASS_POSTFIX, "STORE_OF_CLASS");
case STORE_OF_FIELD -> mergeCompositeNodes(nodes, mergedNodes, i, node, STORE_OF_FIELD_POSTFIX, "STORE_OF_FIELD");
case LOAD_OF_CLASS, LOAD_B_OF_CLASS, LOAD_UB_OF_CLASS, LOAD_D_OF_CLASS, LOAD_F_OF_CLASS, LOAD_I_OF_CLASS, LOAD_L_OF_CLASS,
LOAD_N_OF_CLASS, LOAD_P_OF_CLASS, LOAD_S_OF_CLASS, LOAD_US_OF_CLASS
-> mergeCompositeNodes(nodes, mergedNodes, i, node, LOAD_OF_CLASS_POSTFIX, "LOAD_OF_CLASS");
case LOAD_OF_FIELD -> mergeCompositeNodes(nodes, mergedNodes, i, node, LOAD_OF_FIELD_POSTFIX, "LOAD_OF_FIELD");
default -> {
i--; // No composite node, do not increment by 2.
mergedNodes.add(node);
}
}
}
return mergedNodes;
}
private static void mergeCompositeNodes(String[] nodes, List<String> mergedNodes, int i, String node, String postFix, String varName) {
TestFormat.check(i + 1 < nodes.length, "Must provide class name at index " + (i + 1) + " right after " + varName);
mergedNodes.add(node + Pattern.quote(nodes[i + 1]) + postFix);
}
}

View File

@ -0,0 +1,35 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* Annotation to allow to specify multiple {@link IR @IR} annotations at a {@link Test @Test} method.
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface IRs {
IR[] value();
}

View File

@ -0,0 +1,135 @@
# IR Test Framework
This folder contains a test framework whose main purpose is to perform regex-based checks on the C2 IR shape of test methods emitted by the VM flags _-XX:+PrintIdeal_ and _-XX:+PrintOptoAssembly_. The framework can also be used for other non-IR matching (and non-compiler) tests by providing easy to use annotations for commonly used testing patterns and compiler control flags.
## 1. How to Use the Framework
The framework is intended to be used in JTreg tests. The JTreg header of the test must contain `@library /test/lib /` (2 paths) and should be run as a driver with `@run driver`. Annotate the test code with the supported framework annotations and call the framework from within the test's `main()` method. A simple example is shown below:
/*
* @test
* @summary A simple test using the test framework.
* @library /test/lib /
* @run driver my.package.MySimpleTest
*/
package my.package;
import compiler.lib.ir_framework.*;
public class MySimpleTest {
public static void main(String[] args) {
TestFramework.run(); // The framework runs all tests of this class.
}
@Test
@IR(failOn = IRNode.STORE) // Fail if the IR of myTest() contains any stores.
public void myTest() {
/* ... */
}
}
There are various ways how to set up and run a test within the `main()` method of a JTreg test. These are described and can be found in the [TestFramework](./TestFramework.java) class.
## 2. Features
The framework offers various annotations and flags to control how your test code should be invoked and being checked. This section gives an overview over all these features.
### 2.1 Different Tests
There are three kinds of tests depending on how much control is needed over the test invocation.
#### Base Tests
The simplest form of testing provides a single `@Test` annotated method which the framework will invoke as part of the testing. The test method has no or well-defined arguments that the framework can automatically provide.
More information on base tests with a precise definition can be found in the Javadocs of [Test](./Test.java). Concrete examples on how to specify a base test can be found in [BaseTestsExample](../../../testlibrary_tests/ir_framework/examples/BaseTestExample.java).
#### Checked Tests
The base tests do not provide any way of verification by user code. A checked test enables this by allowing the user to define an additional `@Check` annotated method which is invoked directly after the `@Test` annotated method. This allows the user to perform various checks about the test method including return value verification.
More information on checked tests with a precise definition can be found in the Javadocs of [Check](./Check.java). Concrete examples on how to specify a checked test can be found in [CheckedTestsExample](../../../testlibrary_tests/ir_framework/examples/CheckedTestExample.java).
#### Custom Run Tests
Neither the base nor the checked tests provide any control over how a `@Test` annotated method is invoked in terms of customized argument values and/or conditions for the invocation itself. A custom run test gives full control over the invocation of the `@Test` annotated method to the user. The framework calls a dedicated `@Run` annotated method from which the user can invoke the `@Test` method according to his/her needs.
More information on checked tests with a precise definition can be found in the Javadocs of [Run](./Run.java). Concrete examples on how to specify a custom run test can be found in [CustomRunTestsExample](../../../testlibrary_tests/ir_framework/examples/CustomRunTestExample.java).
### 2.2 IR Verification
The main feature of this framework is to perform a simple but yet powerful regex-based C2 IR matching on the output of _-XX:+PrintIdeal_ and _-XX:+PrintOptoAssembly_. For simplicity, we will refer to the "IR" or "IR matching" when actually meaning the combined output of _-XX:+PrintIdeal_ and _-XX:+PrintOptoAssembly_ for a C2 compilation.
The user has the possibility to add an additional `@IR` annotation to any `@Test` annotated method (regardless of the kind of test mentioned in section 2.1) to specify a constraint/rule on the compiled IR shape. The `@IR` annotation provides two kinds of regex checks:
- A `failOn` check that verifies that the provided regex is not matched in the C2 IR.
- A `counts` check that verifies that the provided regex is matched a user defined number of times in the C2 IR.
A regex can either be a custom string or any of the default regexes provided by the framework in [IRNode](./IRNode.java) for some commonly used IR nodes (also provides the possibility of composite regexes).
An IR verification cannot always be performed. For example, a JTreg test could be run with _-Xint_ or not a debug build (_-XX:+PrintIdeal_ and _-XX:+PrintOptoAssembly_ are debug build flags). But also CI tier testing could add additional JTreg VM and Javaoptions flags which could make an IR rule unstable.
In general, the framework will only perform IR verification if the used VM flags allow a C2 compilation and if non-critical additional JTreg VM and Javaoptions are provided (see whiteflag list in [TestFramework](./TestFramework.java)). The user test code, however, can specify any flags which still allow an IR verification to be performed if a C2 compilation is done (expected flags by user defined `@IR` annotations).
An `@IR` annotation allows additional preconditions/restrictions on the currently present VM flags to enable or disable rules when certain flags are present or have a specific value (see `applyIfXX` properties of an `@IR` annotation).
More information about IR matching can be found in the Javadocs of [IR](./IR.java). Concrete examples on how to specify IR constraint/rules can be found in [IRExample](../../../testlibrary_tests/ir_framework/examples/IRExample.java) and [TestIRMatching](../../../testlibrary_tests/ir_framework/tests/TestIRMatching.java) (an internal framework test).
### 2.3 Test VM Flags and Scenarios
The recommended way to use the framework is by defining a single `@run driver` statement in the JTreg header which, however, does not allow the specification of additional test VM flags. Instead, the user has the possibility to provide VM flags by calling `TestFramework.runWithFlags()` or by creating a `TestFramework` builder object on which `addFlags()` can be called.
If a user wants to provide multiple flag combinations for a single test, he or she has the option to provide different scenarios. A scenario based flag will always have precedence over other user defined flags. More information about scenarios can be found in the Javadocs of [Scenario](./Scenario.java).
### 2.4 Compiler Controls
The framework allows the use of additional compiler control annotations for helper method and classes in the same fashion as JMH does. The following annotations are supported and described in the referenced Javadocs for the annotation class:
- [@DontInline](./DontInline.java)
- [@ForceInline](./ForceInline.java)
- [@DontCompile](./DontCompile.java)
- [@ForceCompile](./DontCompile.java)
- [@ForceCompileClassInitializer](./ForceCompileClassInitializer.java)
### 2.5 Framework Debug and Stress Flags
The framework provides various stress and debug flags. They should mainly be used as JTreg VM and/or Javaoptions (apart from `VerifyIR`). The following (property) flags are supported:
- `-DVerifyIR=false`: Explicitly disable IR verification. This is useful, for example, if some scenarios use VM flags that let `@IR` annotation rules fail and the user does not want to provide separate IR rules or add flag preconditions to the already existing IR rules.
- `-DTest=test1,test2`: Provide a list of `@Test` method names which should be executed.
- `-DExclude=test3`: Provide a list of `@Test` method names which should be excluded from execution.
- `-DScenarios=1,2`: Provide a list of scenario indexes to specify which scenarios should be executed.
- `-DWarmup=200`: Provide a new default value of the number of warm-up iterations (framework default is 2000). This might have an influence on the resulting IR and could lead to matching failures (the user can also set a fixed default warm-up value in a test with `testFrameworkObject.setDefaultWarmup(200)`).
- `-DVerbose=true`: Enable more fain-grained logging (slows the execution down).
- `-DReproduce=true`: Flag to use when directly running a test VM to bypass dependencies to the driver VM state (for example, when reproducing an issue).
- `-DPrintTimes=true`: Print the execution time measurements of each executed test.
- `-DVerifyVM=true`: The framework runs the test VM with additional verification flags (slows the execution down).
- `-DExcluceRandom=true`: The framework randomly excludes some methods from compilation. IR verification is disabled completely with this flag.
- `-DFlipC1C2=true`: The framework compiles all `@Test` annotated method with C1 if a C2 compilation would have been applied and vice versa. IR verification is disabled completely with this flag.
- `-DShuffleTests=false`: Disables the random execution order of all tests (such a shuffling is always done by default).
- `-DDumpReplay=true`: Add the `DumpReplay` directive to the test VM.
- `-DGCAfter=true`: Perform `System.gc()` after each test (slows the execution down).
- `-TestCompilationTimeout=20`: Change the default waiting time (default: 10s) for a compilation of a normal `@Test` annotated method.
- `-DWaitForCompilationTimeout=20`: Change the default waiting time (default: 10s) for a compilation of a `@Test` annotated method with compilation level [WAIT\_FOR\_COMPILATION](./CompLevel.java).
- `-DIgnoreCompilerControls=false`: Ignore all compiler controls applied in the framework. This includes any compiler control annotations (`@DontCompile`, `@DontInline`, `@ForceCompile`, `@ForceInline`, `@ForceCompileStaticInitializer`), the exclusion of `@Run` and `@Check` methods from compilation, and the directive to not inline `@Test` annotated methods.
## 3. Test Framework Execution
This section gives an overview of how the framework is executing a JTreg test that calls the framework from within its `main()` method.
The framework will spawn a new "test VM" to execute the user defined tests. The test VM collects all tests of the test class specified by the user code in `main()` and ensures that there is no violation of the required format by the framework. In a next step, the framework does the following for each test in general:
1. Warm the test up for a predefined number of times (default 2000). This can also be adapted for all tests by using `testFrameworkobject.setDefaultWarmup(100)` or for individual tests with an additional [@Warmup](./Warmup.java) annotation.
2. After the warm-up is finished, the framework compiles the associated `@Test` annotated method at the specified compilation level (default: C2).
3. After the compilation, the test is invoked one more time.
Once the test VM terminates, IR verification (if possible) is performed on the output of the test VM. If any test throws an exception during its execution or if IR matching fails, the failures are collected and reported in a pretty format. Check the standard error and output for more information and how to reproduce these failures.
Some of the steps above can be different due to the kind of the test or due to using non-default annotation properties. These details and differences are described in the Javadocs for the three tests (see section 2.1 Different Tests).
More information about the internals and the workflow of the framework can be found in the Javadocs of [TestFramework](./TestFramework.java).
## 4. Internal Framework Tests
There are various tests to verify the correctness of the test framework. These tests can be found in [ir_framework](../../../testlibrary_tests/ir_framework) and can directly be run with JTreg. The tests are part of the normal JTreg tests of HotSpot and should be run upon changing the framework code as a minimal form of testing.
Additional testing was performed by converting all compiler Inline Types tests that used the currently present IR test framework in Valhalla (see [JDK-8263024](https://bugs.openjdk.java.net/browse/JDK-8263024)). It is strongly advised to make sure a change to the framework still lets these converted tests in Valhalla pass as part of an additional testing step.
## 5. Framework Package Structure
A user only needs to import classes from the package `compiler.lib.ir_framework` (e.g. `import compiler.lib.ir_framework.*;`) which represents the interface classes to the framework. The remaining framework internal classes are kept in separate subpackages and should not directly be imported:
- `compiler.lib.ir_framework.driver`: These classes are used while running the driver VM (same VM as the one running the user code's `main()` method of a JTreg test).
- `compiler.lib.ir_framework.flag`: These classes are used while running the flag VM to determine additional flags for the test VM which are required for IR verification.
- `compiler.lib.ir_framework.test`: These classes are used while running the test VM (i.e. the actual execution of the user tests as described in section 3).
- `compiler.lib.ir_framework.shared`: These classes can be called from either the driver, flag, or test VM.
## 6. Summary
The initial design and feature set was kept simple and straight forward and serves well for small to medium sized tests. There are a lot of possibilities to further enhance the framework and make it more powerful. This can be tackled in additional RFEs. A few ideas can be found as subtasks of the [initial RFE](https://bugs.openjdk.java.net/browse/JDK-8254129) for this framework.

View File

@ -0,0 +1,109 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.shared.TestFormatException;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* Annotation for a run method of a <b>custom run test</b>.
*
* <p>
* Let {@code t} be a test method specifying the {@link Test @Test} annotation and {@code r} be a run method specifying
* the {@code @Run(test = "t")} annotation. These two methods represent a so-called <i>custom run test</i>. The only
* difference to a <i>base test</i> (see {@link Test}) is that the framework will not invoke the test method {@code t}
* but instead the run method {@code r} which is then responsible to invoke {@code t} in any way and optionally do any
* additional verification (e.g. of the return value). If {@code r} does not specify {@link RunMode#STANDALONE} as
* {@link #mode()} property, the framework does the following, similar as for <i>base tests</i>:
* <ol>
* <li><p>The framework warms {@code r} up by invoking it for a predefined number of iterations (default: 2000)
* or any number specified by an additional {@link Warmup} annotation at the run method {@code r} or by using
* {@link TestFramework#setDefaultWarmup(int)} (could also be 0 which skips the warm-up completely which is
* similar to simulating {@code -Xcomp}). More information about the warm-up in general can be found in
* {@link Warmup @Warmup}.</li>
* <li><p>After the warm-up, the framework compiles the test method {@code t} at the specified compilation level set by
* {@link Test#compLevel()} (default {@link CompLevel#ANY} will pick the highest available level which is usually
* {@link CompLevel#C2}).</li>
* <li><p>The framework invokes the run method {@code r} one more time to check the compilation.</li>
* <li><p>The framework checks any specified {@link IR @IR} constraints at the test method {@code t}.
* More information about IR matching can be found at {@link IR}.</li>
* </ol>
*
* <p>
* If {@code r} specifies {@link RunMode#STANDALONE} as {@link #mode()} property, the framework gives complete
* control to the run method {@code r}:
* <ol>
* <li><p>The framework invokes the run method {@code r} only one time without any warm-up or compilation of
* {@code t} ({@link Warmup @Warmup} is not allowed at {@code r} in this case).</li>
* <li><p>After this single invocation, the framework directly checks any specified {@link IR} constraints at the test
* method {@code t}. The run method {@code r} needs to make sure to reliably trigger a C2 compilation. Otherwise,
* IR matching will fail. More information about IR matching can be found at {@link IR}.</li>
* </ol>
*
* <p>
* The test method {@code t} and run method {@code r} have the following properties:
* <ul>
* <li><p>{@code t} can specify any parameter or return type except {@link AbstractInfo} or any of its subclasses.</li>
* <li><p>{@code t} is not inlined.
* <li><p>{@code r} is not compiled nor inlined.
* <li><p>{@code r} is responsible to invoke {@code t} in any way (once, multiple times or even skipping on some
* invocations of {@code r}).
* <li><p>{@code r} can specify the following method parameter combinations:
* <ul>
* <li><p>void</li>
* <li><p>One parameter: {@link RunInfo} which provides some information about {@code t} and utility methods.</li>
* <li><p>Any other combination will result in a {@link TestFormatException}.
* </ul>
* <li><p>{@code t} and {@code r} must be part of the test class. Using {@code @Run} and {@code @Test} in nested or
* other helper classes is not allowed.</li>
* <li><p>{@code t} and {@code r} cannot specify any helper-method-specific compile command annotations
* ({@link ForceCompile @ForceCompile}, {@link DontCompile @DontCompile}, {@link ForceInline @ForceInline},
* {@link DontInline @DontInline}).</li>
* </ul>
*
* <p>
* Examples on how to write custom run tests can be found in {@link jdk.test.lib.hotspot.ir_framework.examples.CustomRunTestExample}
* and also as part of the internal testing in the package {@link jdk.test.lib.hotspot.ir_framework.tests}.
*
* @see Test
* @see RunInfo
* @see RunMode
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface Run {
/**
* The associated {@link Test @Test} methods (one or more) for this {@code @Run} annotated run method.
* The framework directly invokes the run method instead of the associated {@code @Test} methods.
*/
String[] test();
/**
* The mode of this custom run test.
*
* @see RunMode
*/
RunMode mode() default RunMode.NORMAL;
}

View File

@ -0,0 +1,237 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.test.DeclaredTest;
import compiler.lib.ir_framework.shared.TestRunException;
import compiler.lib.ir_framework.test.TestVM;
import java.lang.reflect.Method;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
/**
* Test info class which provides some useful utility methods and information about a <b>custom run test</b>.
*
* @see Run
*/
public class RunInfo extends AbstractInfo {
private final Method testMethod;
private final DeclaredTest test;
private final Map<String, DeclaredTest> tests;
private final boolean hasMultipleTests;
public RunInfo(List<DeclaredTest> tests) {
super(tests.get(0).getTestMethod().getDeclaringClass());
this.test = tests.get(0);
this.testMethod = test.getTestMethod();
this.hasMultipleTests = tests.size() != 1;
if (hasMultipleTests) {
this.tests = new HashMap<>();
for (DeclaredTest test : tests) {
this.tests.put(test.getTestMethod().getName(), test);
}
} else {
this.tests = null;
}
}
/**
* Get the associated test method object of this custom run test. This method can only be called if <i>one</i> test
* method is specified in the custom run test ({@link Run#test()}). Otherwise, use {@link #getTest(String)}.
*
* @return the associated test method object.
* @throws TestRunException if called for a custom run test that specifies multiple test methods in {@link Run#test()}.
*/
public Method getTest() {
checkSingleTest("getTest");
return testMethod;
}
/**
* Get the associated method object of the test method with the name {@code testName}. This method can only be called
* if the custom run test specifies more than one test method in ({@link Run#test()}). Otherwise, use {@link #getTest()}.
*
* @param testName the test method for which the method object should be returned.
* @return the associated test method object with the name {@code testName}.
* @throws TestRunException if there is no test method with the name {@code testName} or if called with only
* <i>one</i> associated test method.
*/
public Method getTest(String testName) {
checkMultipleTests("getTest");
return getMethod(testName);
}
/**
* Return a boolean indicating if the framework skipped a compilation of the associated test method after the warm-up
* due to VM flags not allowing a compilation on the requested level in {@link Test#compLevel()}. This method can only
* be called if <i>one</i> test method is specified in the custom run test ({@link Run#test()}). Otherwise, use
* {@link #isCompilationSkipped(String)}.
*
* @return {@code true} if the framework skipped compilation of the test;
* {@code false} otherwise.
* @throws TestRunException if called for a custom run test that specifies multiple test methods in {@link Run#test()}.
*/
public boolean isCompilationSkipped() {
checkSingleTest("isCompilationSkipped");
return test.getCompLevel() == CompLevel.SKIP;
}
/**
* Return a boolean indicating if the framework skipped a compilation of the associated test method with the name
* {@code testName} after the warm-up due to VM flags not allowing a compilation on the requested level in
* {@link Test#compLevel()}. This method can only be called if the custom run test specifies more than one test method
* in ({@link Run#test()}). Otherwise, use {@link #isCompilationSkipped()}.
*
* @param testName the test method for which the method object should be returned.
* @return {@code true} if the framework skipped compilation of the test;
* {@code false} otherwise.
* @throws TestRunException if there is no test method with the name {@code testName} or if called with only
* <i>one</i> associated test method.
*/
public boolean isCompilationSkipped(String testName) {
checkMultipleTests("isCompilationSkipped");
return getDeclaredTest(testName).getCompLevel() == CompLevel.SKIP;
}
/**
* Returns a boolean indicating if the associated test method is C1 compiled. This method can only be called if
* <i>one</i> test method is specified in the custom run test ({@link Run#test()}). Otherwise, use
* {@link #isTestC1Compiled(String)}.
*
* @return {@code true} if the associated test method is C1 compiled;
* {@code false} otherwise.
* @throws TestRunException if called for a custom run test that specifies multiple test methods in {@link Run#test()}.
*/
public boolean isTestC1Compiled() {
checkSingleTest("isTestC1Compiled");
return TestVM.isC1Compiled(testMethod);
}
/**
* Returns a boolean indicating if the associated test method with the name {@code testName} is C1 compiled.
* This method can only be called if the custom run test specifies more than one test method in ({@link Run#test()}).
* Otherwise, use {@link #isTestC1Compiled()}.
*
* @param testName the name of the test method.
* @return {@code true} if the test method with the name {@code testName} is C1 compiled;
* {@code false} otherwise.
* @throws TestRunException if there is no test method with the name {@code testName} or if called with only
* <i>one</i> associated test method.
*/
public boolean isTestC1Compiled(String testName) {
checkMultipleTests("isTestC1Compiled");
return TestVM.isC1Compiled(getMethod(testName));
}
/**
* Returns a boolean indicating if the associated test method is C2 compiled. This method can only be called if
* <i>one</i> test method is specified in the custom run test ({@link Run#test()}). Otherwise, use
* {@link #isTestC2Compiled(String)}.
*
* @return {@code true} if the associated test method is C2 compiled;
* {@code false} otherwise.
* @throws TestRunException if called for a custom run test that specifies multiple test methods in {@link Run#test()}.
*/
public boolean isTestC2Compiled() {
checkSingleTest("isTestC2Compiled");
return TestVM.isC2Compiled(testMethod);
}
/**
* Returns a boolean indicating if the associated test method with the name {@code testName} is C2 compiled.
* This method can only be called if the custom run test specifies more than one test method in ({@link Run#test()}).
* Otherwise, use {@link #isTestC2Compiled()}.
*
* @param testName the name of the test method.
* @return {@code true} if the test method with the name {@code testName} is C2 compiled;
* {@code false} otherwise.
* @throws TestRunException if there is no test method with the name {@code testName} or if called with only
* <i>one</i> associated test method.
*/
public boolean isTestC2Compiled(String testName) {
checkMultipleTests("isTestC2Compiled");
return TestVM.isC2Compiled(getMethod(testName));
}
/**
* Returns a boolean indicating if the associated test method is compiled at {@code compLevel}. This method can only
* be called if <i>one</i> test method is specified in the custom run test ({@link Run#test()}).
* Otherwise, use {@link #isTestCompiledAtLevel(String, CompLevel)}.
*
* @param compLevel the compilation level
* @return {@code true} if the associated test method is compiled at {@code compLevel};
* {@code false} otherwise.
* @throws TestRunException if called for a custom run test that specifies multiple test methods in {@link Run#test()}.
*/
public boolean isTestCompiledAtLevel(CompLevel compLevel) {
checkSingleTest("isTestCompiledAtLevel");
return TestVM.isCompiledAtLevel(testMethod, compLevel);
}
/**
* Returns a boolean indicating if the associated test method with the name {@code testName} is compiled at
* {@code compLevel}. This method can only be called if the custom run test specifies more than one test method
* in ({@link Run#test()}). Otherwise, use {@link #isTestCompiledAtLevel(CompLevel)}.
*
* @param testName the name of the test method.
* @param compLevel the compilation level.
* @return {@code true} if the test method with the name {@code testName} is compiled at {@code compLevel};
* {@code false} otherwise.
* @throws TestRunException if there is no test method with the name {@code testName} oor if called with only
* <i>one</i> associated test method.
*/
public boolean isTestCompiledAtLevel(String testName, CompLevel compLevel) {
checkMultipleTests("isTestCompiledAtLevel");
return TestVM.isCompiledAtLevel(getMethod(testName), compLevel);
}
private void checkSingleTest(String calledMethod) {
if (hasMultipleTests) {
throw new TestRunException("Use " + calledMethod + "(String) with testName String argument in @Run method " +
"for custom run test that specifies more than one @Test method.");
}
}
private void checkMultipleTests(String calledMethod) {
if (!hasMultipleTests) {
throw new TestRunException("Use " + calledMethod + "() without testName String argument in @Run method " +
"for custom run test that specifies exactly one @Test method.");
}
}
private DeclaredTest getDeclaredTest(String testName) {
DeclaredTest test = tests.get(testName);
if (test == null) {
throw new TestRunException("Could not find @Test \"" + testName + "\" in " + testClass + " being associated with" +
" corresponding @Run method.");
}
return test;
}
private Method getMethod(String testName) {
return getDeclaredTest(testName).getTestMethod();
}
}

View File

@ -0,0 +1,43 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
/**
* The run mode for a <b>custom run test</b> specified in {@link Run#mode}.
*
* @see Run
*/
public enum RunMode {
/**
* Default mode: First warm up the run method (if a warm-up is done), then compile the associated {@link Test}
* method and finally invoke the run method once more.
*/
NORMAL,
/**
* Standalone mode: There is no warm-up and no compilation done by the framework. The run method is responsible to
* trigger the compilation(s), especially in regard of possible {@link IR} annotations at the associated {@link Test}
* method.
*/
STANDALONE,
}

View File

@ -0,0 +1,150 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.shared.TestRunException;
import java.util.*;
import java.util.stream.Collectors;
/**
* This class represents a scenario that can be executed by the {@link TestFramework}.
* <p>
* A JTreg test should use the test framework with {@code @run driver} (without directly specify any additional flags).
* If a test should run with additional flags, use {@link TestFramework#runWithFlags(String...)} or
* {@link TestFramework#addFlags(String...)}. If, however, the test should be run with different settings (equivalent
* to having multiple {@code @run} entries in a normal JTreg test), use scenarios. A scenario will be run with the
* scenario specific VM flags, if any, and optionally specified VM flags with {@link TestFramework#addFlags(String...)}
* whereas scenario VM flags will have precedence.
* <p>
* There is also the possibility to specify additional VM flags for all scenarios by using {@code DScenarioFlags}.
*
* @see TestFramework
*/
public class Scenario {
private static final String ADDITIONAL_SCENARIO_FLAGS_PROPERTY = System.getProperty("ScenarioFlags", "");
private static final String SCENARIOS_PROPERTY = System.getProperty("Scenarios", "");
private static final List<String> ADDITIONAL_SCENARIO_FLAGS;
private static final Set<Integer> ENABLED_SCENARIOS;
private final List<String> flags;
private final int index;
private final boolean enabled;
private String testVMOutput;
static {
if (!SCENARIOS_PROPERTY.isEmpty()) {
var split = SCENARIOS_PROPERTY.split("\\s*,\\s*");
try {
ENABLED_SCENARIOS = Arrays.stream(split).map(Integer::parseInt).collect(Collectors.toSet());
} catch (NumberFormatException e) {
throw new TestRunException("Provided a scenario index in the -DScenario comma-separated list which is not "
+ "a number: " + SCENARIOS_PROPERTY);
}
} else {
ENABLED_SCENARIOS = Collections.emptySet();
}
ADDITIONAL_SCENARIO_FLAGS = ADDITIONAL_SCENARIO_FLAGS_PROPERTY.isEmpty() ? Collections.emptyList() :
Arrays.asList(ADDITIONAL_SCENARIO_FLAGS_PROPERTY.split("\\s*,\\s*"));
}
/**
* Create a scenario with {@code index} that will be run with the additional VM flags specified in {@code flags}
* (or without any if null or parameter not provided).
* <p>
* The scenario {@code index} must be unique to be distinguishable in the stdout and stderr output and when specifying
* {@code -DScenarios} (see {@link Scenario}).
*
* @param index the unique scenario index.
* @param flags the scenario flags or null (i.e. no parameter specified) if no flags should be used.
*/
public Scenario(int index, String... flags) {
this.index = index;
if (flags != null) {
this.flags = new ArrayList<>(Arrays.asList(flags));
this.flags.addAll(ADDITIONAL_SCENARIO_FLAGS);
} else {
this.flags = new ArrayList<>();
}
this.enabled = ENABLED_SCENARIOS.isEmpty() || ENABLED_SCENARIOS.contains(index);
}
/**
* Add additional VM flags to this scenario.
*
* @param flags the additional scenario VM flags.
*/
public void addFlags(String... flags) {
if (flags != null) {
this.flags.addAll(Arrays.asList(flags));
}
}
/**
* Get all scenario specific VM flags as defined in {@link #Scenario(int, String...)}.
*
* @return the scenario VM flags.
*/
public List<String> getFlags() {
return flags;
}
/**
* Get the unique scenario index as defined in {@link #Scenario(int, String...)}.
*
* @return the scenario index.
*/
public int getIndex() {
return index;
}
/**
* Get the test VM output (stdout + stderr) of this scenario from the last execution of the framework.
*
* @return the test VM output.
*/
public String getTestVMOutput() {
return testVMOutput;
}
/**
* Set the test VM output, called by the framework.
*/
void setTestVMOutput(String testVMOutput) {
this.testVMOutput = testVMOutput;
}
/**
* Returns a boolean indicating if this scenario will be executed by the test framework. This only depends on
* the property flag {@code -DScenarios} (see {@link Scenario}). This is only used by the framework internally.
*
* @return {@code true} if {@code -DScenarios} is either not set or if {@code -DScenarios} specifies the scenario
* index set by {@link #Scenario(int, String...)}.
* {@code false} otherwise.
*/
boolean isEnabled() {
return enabled;
}
}

View File

@ -0,0 +1,87 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* Annotate all methods in your test class which the framework should test with {@code @Test}.
* <p>
* Let {@code m} be a test method specifying the {@code @Test} annotation. If {@code m} is neither part of a
* <b>checked test</b> (an additional method specifying {@link Check @Check} with {@code @Check(test = "m")}) nor part
* of a <b>custom run test</b> (an additional method specifying {@link Run @Run} with {@code @Run(test = "m")}),
* then {@code m} is a so-called <b>base test</b> and the the framework invokes {@code m} in the following way:
* <ol>
* <li><p>The framework warms {@code m} up by invoking it for a predefined number of iterations (default: 2000)
* or any number specified by an additional {@link Warmup @Warmup} annotation at {@code m} or by using
* {@link TestFramework#setDefaultWarmup(int)} (could also be 0 which skips the warm-up completely which is similar
* to simulating {@code -Xcomp}). More information about the warm-up in general can be found at {@link Warmup}</li>
* <li><p>After the warm-up, the framework compiles {@code m} at the specified compilation level set by
* {@link #compLevel()} (default {@link CompLevel#ANY} will pick the highest available level which is usually
* {@link CompLevel#C2}).</li>
* <li><p>The framework invokes {@code m} one more time to run the compilation.</li>
* <li><p>The framework checks any specified {@link IR @IR} constraints at {@code m}. More information about IR matching
* can be found at {@link IR}.</li>
* </ol>
*
* <p>
* {@code m} has the following properties:
* <ul>
* <li><p>If {@code m} specifies no parameters, the framework can directly invoke {@code m}.</li>
* <li><p>If {@code m} specifies parameters, the framework needs to know how to invoke {@code m}. Use {@link Arguments}
* with {@link Argument} properties for each parameter to use well-defined parameters by the framework. If the method
* requires a more specific argument value, use a custom run test (see {@link Run}).</li>
* <li><p>{@code m} cannot specify {@link AbstractInfo} or any of its subclasses as parameter or return type.</li>
* <li><p>{@code m} is not inlined by the framework.</li>
* <li><p>Verification of the return value of {@code m} can only be done in a checked test (see {@link Check}) or
* custom run test (see {@link Run}).</li>
* </ul>
*
* <p>
* The following constraints must be met for the test method {@code m} specifying {@code @Test}:
* <ul>
* <li><p>{@code m} must be part of the test class. Using {@code @Test} in nested or helper classes is not allowed.</li>
* <li><p>{@code m} cannot have the same name as another {@code @Test} method in the same test class. Method
* overloading is only allowed (but not encouraged) with other non-{@code @Test} methods.</li>
* <li><p>{@code m} cannot specify any helper-method-specific compile command annotations
* ({@link ForceCompile @ForceCompile}, {@link DontCompile @DontCompile}, {@link ForceInline @ForceInline},
* {@link DontInline @DontInline}). </li>
* </ul>
*
* <p>
* Examples on how to write base tests can be found in {@link jdk.test.lib.hotspot.ir_framework.examples.BaseTestExample}
* and also as part of the internal testing in the package {@link jdk.test.lib.hotspot.ir_framework.tests}.
*
* @see Arguments
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface Test {
/**
* Specify at which compilation level the framework should eventually compile the test method after an optional
* warm-up period. The default {@link CompLevel#ANY} will let the framework compile the method at the highest
* available level which is usually {@link CompLevel#C2}.
*/
CompLevel compLevel() default CompLevel.ANY;
}

View File

@ -0,0 +1,718 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.driver.*;
import compiler.lib.ir_framework.shared.*;
import compiler.lib.ir_framework.test.*;
import jdk.test.lib.Platform;
import jdk.test.lib.Utils;
import jdk.test.lib.helpers.ClassFileInstaller;
import sun.hotspot.WhiteBox;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.lang.reflect.Method;
import java.util.*;
import java.util.stream.Collectors;
/**
* This class represents the main entry point to the test framework whose main purpose is to perform regex-based checks on
* the C2 IR shape emitted by the VM flags {@code -XX:+PrintIdeal} and {@code -XX:+PrintOptoAssembly}. The framework can
* also be used for other non-IR matching (and non-compiler) tests by providing easy to use annotations for commonly used
* testing patterns and compiler control flags.
* <p>
* The framework offers various annotations to control how your test code should be invoked and being checked. There are
* three kinds of tests depending on how much control is needed over the test invocation:
* <b>Base tests</b> (see {@link Test}), <b>checked tests</b> (see {@link Check}), and <b>custom run tests</b>
* (see {@link Run}). Each type of test needs to define a unique <i>test method</i> that specifies a {@link Test @Test}
* annotation which represents the test code that is eventually executed by the test framework. More information about
* the usage and how to write different tests can be found in {@link Test}, {@link Check}, and {@link Run}.
* <p>
* Each test method can specify an arbitrary number of IR rules. This is done by using {@link IR @IR} annotations which
* can define regex strings that are matched on the output of {@code -XX:+PrintIdeal} and {@code -XX:+PrintOptoAssembly}.
* The matching is done after the test method was (optionally) warmed up and compiled. More information about the usage
* and how to write different IR rules can be found at {@link IR}.
* <p>
* This framework should be used with the following JTreg setup in your Test.java file in package <i>some.package</i>:
* <pre>
* {@literal @}library /test/lib
* {@literal @}run driver some.package.Test
* </pre>
* Note that even though the framework uses the Whitebox API internally, it is not required to build and enabel it in the
* JTreg test if the test itself is not utilizing any Whitebox features directly.
* <p>
* To specify additional flags, use {@link #runWithFlags(String...)}, {@link #addFlags(String...)}, or
* {@link #addScenarios(Scenario...)} where the scenarios can also be used to run different flag combinations
* (instead of specifying multiple JTreg {@code @run} entries).
* <p>
* After annotating your test code with the framework specific annotations, the framework needs to be invoked from the
* {@code main()} method of your JTreg test. There are two ways to do so. The first way is by calling the various
* {@code runXX()} methods of {@link TestFramework}. The second way, which gives more control, is to create a new
* {@code TestFramework} builder object on which {@link #start()} needs to be eventually called to start the testing.
* <p>
* The framework is called from the <i>driver VM</i> in which the JTreg test is initially run by specifying {@code
* @run driver} in the JTreg header. This strips all additionally specified JTreg VM and Javaoptions.
* The framework creates a new <i>flag VM</i> with all these flags added again in order to figure out which flags are
* required to run the tests specified in the test class (e.g. {@code -XX:+PrintIdeal} and {@code -XX:+PrintOptoAssembly}
* for IR matching).
* <p>
* After the flag VM terminates, it starts a new <i>test VM</i> which performs the execution of the specified
* tests in the test class as described in {@link Test}, {@link Check}, and {@link Run}.
* <p>
* In a last step, once the test VM has terminated without exceptions, IR matching is performed if there are any IR
* rules and if no VM flags disable it (e.g. not running with {@code -Xint}, see {@link IR} for more details).
* The IR regex matching is done on the output of {@code -XX:+PrintIdeal} and {@code -XX:+PrintOptoAssembly} by parsing
* the hotspot_pid file of the test VM. Failing IR rules are reported by throwing a {@link IRViolationException}.
*
* @see Test
* @see Check
* @see Run
* @see IR
*/
public class TestFramework {
/**
* JTreg can define additional VM (-Dtest.vm.opts) and Javaoptions (-Dtest.java.opts) flags. IR verification is only
* performed when all these additional JTreg flags (does not include additionally added framework and scenario flags
* by user code) are whitelisted.
*
* A flag is whitelisted if it is a property flag (starting with -D), -ea, -esa, or if the flag name contains any of
* the entries of this list as a substring (partial match).
*/
public static final Set<String> JTREG_WHITELIST_FLAGS = new HashSet<>(
Arrays.asList(
// The following substrings are part of more than one VM flag
"RAM",
"Heap",
"Trace",
"Print",
"Verify",
"TLAB",
"UseNewCode",
"Xmn",
"Xms",
"Xmx",
"Xss",
// The following substrings are only part of one VM flag (=exact match)
"CreateCoredumpOnCrash",
"IgnoreUnrecognizedVMOptions",
"UnlockDiagnosticVMOptions",
"UnlockExperimentalVMOptions",
"BackgroundCompilation",
"Xbatch",
"TieredCompilation",
"Xmixed",
"server",
"Xlog",
"LogCompilation"
)
);
public static final boolean VERBOSE = Boolean.getBoolean("Verbose");
public static final boolean TESTLIST = !System.getProperty("Test", "").isEmpty();
public static final boolean EXCLUDELIST = !System.getProperty("Exclude", "").isEmpty();
private static final boolean REPORT_STDOUT = Boolean.getBoolean("ReportStdout");
// Only used for internal testing and should not be used for normal user testing.
private static final boolean SKIP_WHITEBOX_INSTALL = Boolean.getBoolean("SkipWhiteBoxInstall");
private static final String RERUN_HINT = """
#############################################################
- To only run the failed tests use -DTest, -DExclude,
and/or -DScenarios.
- To also get the standard output of the test VM run with
-DReportStdout=true or for even more fine-grained logging
use -DVerbose=true.
#############################################################
""" + System.lineSeparator();
private boolean irVerificationPossible = Boolean.parseBoolean(System.getProperty("VerifyIR", "true"));
private boolean shouldVerifyIR; // Should we perform IR matching?
private static boolean toggleBool;
private final Class<?> testClass;
private Set<Class<?>> helperClasses;
private List<Scenario> scenarios;
private Set<Integer> scenarioIndices;
private List<String> flags;
private int defaultWarmup = -1;
/*
* Public interface methods
*/
/**
* Creates an instance acting as a builder to test the class from which this constructor was invoked from.
* Use this constructor if you want to use multiple run options (flags, helper classes, scenarios).
* Use the associated add methods ({@link #addFlags(String...)}, {@link #addScenarios(Scenario...)},
* {@link #addHelperClasses(Class...)}) to set up everything and then start the testing by invoking {@link #start()}.
*/
public TestFramework() {
this(StackWalker.getInstance(StackWalker.Option.RETAIN_CLASS_REFERENCE).getCallerClass());
}
/**
* Creates an instance acting as a builder to test {@code testClass}.
* Use this constructor if you want to use multiple run options (flags, helper classes, scenarios).
* Use the associated add methods ({@link #addFlags(String...)}, @link #addScenarios(Scenario...)},
* {@link #addHelperClasses(Class...)}) to set up everything and then start the testing by invoking {@link #start()}.
*
* @param testClass the class to be tested by the framework.
* @see #TestFramework()
*/
public TestFramework(Class<?> testClass) {
TestRun.check(testClass != null, "Test class cannot be null");
this.testClass = testClass;
if (VERBOSE) {
System.out.println("Test class: " + testClass);
}
}
/**
* Tests the class from which this method was invoked from.
*/
public static void run() {
StackWalker walker = StackWalker.getInstance(StackWalker.Option.RETAIN_CLASS_REFERENCE);
run(walker.getCallerClass());
}
/**
* Tests {@code testClass}.
*
* @param testClass the class to be tested by the framework.
* @see #run()
*/
public static void run(Class<?> testClass) {
TestFramework framework = new TestFramework(testClass);
framework.start();
}
/**
* Tests the class from which this method was invoked from. The test VM is called with the specified {@code flags}.
* <ul>
* <li><p>The {@code flags} override any set VM or Javaoptions flags by JTreg by default.<p>
* Use {@code -DPreferCommandLineFlags=true} if you want to prefer the JTreg VM and Javaoptions flags over
* the specified {@code flags} of this method.</li>
* <li><p>If you want to run your entire JTreg test with additional flags, use this method.</li>
* <li><p>If you want to run your entire JTreg test with additional flags but for another test class then the one
* from which this method was called from, use {@link #addFlags(String...)}, use this method.</li>
* <li><p>If you want to run your JTreg test with multiple flag combinations, use
* {@link #addScenarios(Scenario...)}</li>
* </ul>
*
* @param flags VM flags to be used for the test VM.
*/
public static void runWithFlags(String... flags) {
StackWalker walker = StackWalker.getInstance(StackWalker.Option.RETAIN_CLASS_REFERENCE);
TestFramework framework = new TestFramework(walker.getCallerClass());
framework.addFlags(flags);
framework.start();
}
/**
* Add VM flags to be used for the test VM. These flags override any VM or Javaoptions set by JTreg by default.<p>
* Use {@code -DPreferCommandLineFlags=true} if you want to prefer the VM or Javaoptions over the scenario flags.
*
* <p>
* The testing can be started by invoking {@link #start()}
*
* @param flags VM options to be applied to the test VM.
* @return the same framework instance.
*/
public TestFramework addFlags(String... flags) {
TestRun.check(flags != null && Arrays.stream(flags).noneMatch(Objects::isNull), "A flag cannot be null");
if (this.flags == null) {
this.flags = new ArrayList<>();
}
this.flags.addAll(Arrays.asList(flags));
return this;
}
/**
* Add helper classes that can specify additional compile command annotations ({@link ForceCompile @ForceCompile},
* {@link DontCompile @DontCompile}, {@link ForceInline @ForceInline}, {@link DontInline @DontInline}) to be applied
* while testing {@code testClass} (also see description of {@link TestFramework}).
*
* <p>
* Duplicates in {@code helperClasses} are ignored. If a class is used by the test class that does not specify any
* compile command annotations, you do not need to include it with this method. If no helper class specifies any
* compile commands, you do not need to call this method at all.
*
* <p>
* The testing can be started by invoking {@link #start()}.
*
* @param helperClasses helper classes containing compile command annotations ({@link ForceCompile},
* {@link DontCompile}, {@link ForceInline}, {@link DontInline}) to be applied
* while testing {@code testClass} (also see description of {@link TestFramework}).
* @return the same framework instance.
*/
public TestFramework addHelperClasses(Class<?>... helperClasses) {
TestRun.check(helperClasses != null && Arrays.stream(helperClasses).noneMatch(Objects::isNull),
"A Helper class cannot be null");
if (this.helperClasses == null) {
this.helperClasses = new HashSet<>();
}
this.helperClasses.addAll(Arrays.asList(helperClasses));
return this;
}
/**
* Add scenarios to be used for the test VM. A test VM is called for each scenario in {@code scenarios} by using the
* specified VM flags in the scenario. The scenario flags override any flags set by {@link #addFlags(String...)}
* and thus also override any VM or Javaoptions set by JTreg by default.<p>
* Use {@code -DPreferCommandLineFlags=true} if you want to prefer the VM and Javaoptions over the scenario flags.
*
* <p>
* The testing can be started by invoking {@link #start()}
*
* @param scenarios scenarios which specify specific flags for the test VM.
* @return the same framework instance.
*/
public TestFramework addScenarios(Scenario... scenarios) {
TestFormat.check(scenarios != null && Arrays.stream(scenarios).noneMatch(Objects::isNull),
"A scenario cannot be null");
if (this.scenarios == null) {
this.scenarios = new ArrayList<>();
this.scenarioIndices = new HashSet<>();
}
for (Scenario scenario : scenarios) {
int scenarioIndex = scenario.getIndex();
TestFormat.check(scenarioIndices.add(scenarioIndex),
"Cannot define two scenarios with the same index " + scenarioIndex);
this.scenarios.add(scenario);
}
return this;
}
/**
* Start the testing of the implicitly (by {@link #TestFramework()}) or explicitly (by {@link #TestFramework(Class)})
* set test class.
*/
public void start() {
if (!SKIP_WHITEBOX_INSTALL) {
installWhiteBox();
}
disableIRVerificationIfNotFeasible();
if (scenarios == null) {
try {
start(null);
} catch (TestVMException e) {
System.err.println(System.lineSeparator() + e.getExceptionInfo() + RERUN_HINT);
throw e;
} catch (IRViolationException e) {
System.out.println("Compilation(s) of failed match(es):");
System.out.println(e.getCompilations());
System.err.println(System.lineSeparator() + e.getExceptionInfo() + System.lineSeparator() + RERUN_HINT);
throw e;
}
} else {
startWithScenarios();
}
}
/**
* Set a new default warm-up (overriding the framework default of 2000 at
* {@link TestVM#WARMUP_ITERATIONS}) to be applied for all tests that do not specify an explicit
* warm-up with {@link Warmup @Warmup}.
*
* @param defaultWarmup a new non-negative default warm-up.
* @return the same framework instance.
*/
public TestFramework setDefaultWarmup(int defaultWarmup) {
TestFormat.check(defaultWarmup >= 0, "Cannot specify a negative default warm-up");
this.defaultWarmup = defaultWarmup;
return this;
}
/**
* Get the VM output of the test VM. Use {@code -DVerbose=true} to enable more debug information. If scenarios
* were run, use {@link Scenario#getTestVMOutput()}.
*
* @return the last test VM output.
*/
public static String getLastTestVMOutput() {
return TestVMProcess.getLastTestVMOutput();
}
/*
* The following methods are only intended to be called from actual @Test methods and not from the main() method of
* a JTreg test. Calling these methods from main() results in a linking exception (Whitebox not yet loaded and enabled).
*/
/**
* Compile {@code m} at compilation level {@code compLevel}. {@code m} is first enqueued and might not be compiled,
* yet, upon returning from this method.
*
* @param m the method to be compiled.
* @param compLevel the (valid) compilation level at which the method should be compiled.
* @throws TestRunException if compilation level is {@link CompLevel#SKIP} or {@link CompLevel#WAIT_FOR_COMPILATION}.
*/
public static void compile(Method m, CompLevel compLevel) {
TestVM.compile(m, compLevel);
}
/**
* Deoptimize {@code m}.
*
* @param m the method to be deoptimized.
*/
public static void deoptimize(Method m) {
TestVM.deoptimize(m);
}
/**
* Returns a boolean indicating if {@code m} is compiled at any level.
*
* @param m the method to be checked.
* @return {@code true} if {@code m} is compiled at any level;
* {@code false} otherwise.
*/
public static boolean isCompiled(Method m) {
return TestVM.isCompiled(m);
}
/**
* Returns a boolean indicating if {@code m} is compiled with C1.
*
* @param m the method to be checked.
* @return {@code true} if {@code m} is compiled with C1;
* {@code false} otherwise.
*/
public static boolean isC1Compiled(Method m) {
return TestVM.isC1Compiled(m);
}
/**
* Returns a boolean indicating if {@code m} is compiled with C2.
*
* @param m the method to be checked.
* @return {@code true} if {@code m} is compiled with C2;
* {@code false} otherwise.
*/
public static boolean isC2Compiled(Method m) {
return TestVM.isC2Compiled(m);
}
/**
* Returns a boolean indicating if {@code m} is compiled at the specified {@code compLevel}.
*
* @param m the method to be checked.
* @param compLevel the compilation level.
* @return {@code true} if {@code m} is compiled at {@code compLevel};
* {@code false} otherwise.
*/
public static boolean isCompiledAtLevel(Method m, CompLevel compLevel) {
return TestVM.isCompiledAtLevel(m, compLevel);
}
/**
* Checks if {@code m} is compiled at any level.
*
* @param m the method to be checked.
* @throws TestRunException if {@code m} is not compiled at any level.
*/
public static void assertCompiled(Method m) {
TestVM.assertCompiled(m);
}
/**
* Checks if {@code m} is not compiled at any level.
*
* @param m the method to be checked.
* @throws TestRunException if {@code m} is compiled at any level.
*/
public static void assertNotCompiled(Method m) {
TestVM.assertNotCompiled(m);
}
/**
* Verifies that {@code m} is compiled with C1.
*
* @param m the method to be verified.
* @throws TestRunException if {@code m} is not compiled with C1.
*/
public static void assertCompiledByC1(Method m) {
TestVM.assertCompiledByC1(m);
}
/**
* Verifies that {@code m} is compiled with C2.
*
* @param m the method to be checked.
* @throws TestRunException if {@code m} is not compiled with C2.
*/
public static void assertCompiledByC2(Method m) {
TestVM.assertCompiledByC2(m);
}
/**
* Verifies that {@code m} is compiled at the specified {@code compLevel}.
*
* @param m the method to be checked.
* @param compLevel the compilation level.
* @throws TestRunException if {@code m} is not compiled at {@code compLevel}.
*/
public static void assertCompiledAtLevel(Method m, CompLevel compLevel) {
TestVM.assertCompiledAtLevel(m, compLevel);
}
/**
* Verifies that {@code m} was deoptimized after being C1 compiled.
*
* @param m the method to be checked.
* @throws TestRunException if {@code m} is was not deoptimized after being C1 compiled.
*/
public static void assertDeoptimizedByC1(Method m) {
TestVM.assertDeoptimizedByC1(m);
}
/**
* Verifies that {@code m} was deoptimized after being C2 compiled.
*
* @param m the method to be checked.
* @throws TestRunException if {@code m} is was not deoptimized after being C2 compiled.
*/
public static void assertDeoptimizedByC2(Method m) {
TestVM.assertDeoptimizedByC2(m);
}
/**
* Returns a different boolean each time this method is invoked (switching between {@code false} and {@code true}).
* The very first invocation returns {@code false}. Note that this method could be used by different tests and
* thus the first invocation for a test could be {@code true} or {@code false} depending on how many times
* other tests have already invoked this method.
*
* @return an inverted boolean of the result of the last invocation of this method.
*/
public static boolean toggleBoolean() {
toggleBool = !toggleBool;
return toggleBool;
}
/*
* End of public interface methods
*/
/**
* Used to move Whitebox class to the right folder in the JTreg test
*/
private void installWhiteBox() {
try {
ClassFileInstaller.main(WhiteBox.class.getName());
} catch (Exception e) {
throw new Error("failed to install whitebox classes", e);
}
}
/**
* Disable IR verification completely in certain cases.
*/
private void disableIRVerificationIfNotFeasible() {
if (irVerificationPossible) {
irVerificationPossible = Platform.isDebugBuild() && !Platform.isInt() && !Platform.isComp();
if (!irVerificationPossible) {
System.out.println("IR verification disabled due to not running a debug build (required for PrintIdeal" +
"and PrintOptoAssembly), running with -Xint, or -Xcomp (use warm-up of 0 instead)");
return;
}
irVerificationPossible = hasIRAnnotations();
if (!irVerificationPossible) {
System.out.println("IR verification disabled due to test " + testClass + " not specifying any @IR annotations");
return;
}
// No IR verification is done if additional non-whitelisted JTreg VM or Javaoptions flag is specified.
irVerificationPossible = onlyWhitelistedJTregVMAndJavaOptsFlags();
if (!irVerificationPossible) {
System.out.println("IR verification disabled due to using non-whitelisted JTreg VM or Javaoptions flag(s)."
+ System.lineSeparator());
}
}
}
/**
* For scenarios: Run the tests with the scenario settings and collect all exceptions to be able to run all
* scenarios without prematurely throwing an exception. Format violations, however, are wrong for all scenarios
* and thus is reported immediately on the first scenario execution.
*/
private void startWithScenarios() {
Map<Scenario, Exception> exceptionMap = new TreeMap<>(Comparator.comparingInt(Scenario::getIndex));
for (Scenario scenario : scenarios) {
try {
start(scenario);
} catch (TestFormatException e) {
// Test format violation is wrong for all the scenarios. Only report once.
throw e;
} catch (Exception e) {
exceptionMap.put(scenario, e);
}
}
if (!exceptionMap.isEmpty()) {
reportScenarioFailures(exceptionMap);
}
}
private void reportScenarioFailures(Map<Scenario, Exception> exceptionMap) {
String failedScenarios = "The following scenarios have failed: #"
+ exceptionMap.keySet().stream()
.map(s -> String.valueOf(s.getIndex()))
.collect(Collectors.joining(", #"));
StringBuilder builder = new StringBuilder(failedScenarios);
builder.append(System.lineSeparator()).append(System.lineSeparator());
for (Map.Entry<Scenario, Exception> entry : exceptionMap.entrySet()) {
Exception e = entry.getValue();
Scenario scenario = entry.getKey();
String errorMsg = "";
if (scenario != null) {
errorMsg = getScenarioTitleAndFlags(scenario);
}
if (e instanceof IRViolationException irException) {
// For IR violations, only show the actual violations and not the (uninteresting) stack trace.
System.out.println((scenario != null ? "Scenario #" + scenario.getIndex() + " - " : "")
+ "Compilation(s) of failed matche(s):");
System.out.println(irException.getCompilations());
builder.append(errorMsg).append(System.lineSeparator()).append(irException.getExceptionInfo());
} else if (e instanceof TestVMException testVMException) {
builder.append(errorMsg).append(System.lineSeparator()).append(testVMException.getExceptionInfo());
} else {
// Print stack trace otherwise
StringWriter errors = new StringWriter();
e.printStackTrace(new PrintWriter(errors));
builder.append(errors.toString());
}
builder.append(System.lineSeparator());
}
System.err.println(builder.toString());
if (!VERBOSE && !REPORT_STDOUT && !TESTLIST && !EXCLUDELIST) {
// Provide a hint to the user how to get additional output/debugging information.
System.err.println(RERUN_HINT);
}
throw new TestRunException(failedScenarios + ". Please check stderr for more information.");
}
private static String getScenarioTitleAndFlags(Scenario scenario) {
StringBuilder builder = new StringBuilder();
String title = "Scenario #" + scenario.getIndex();
builder.append(title).append(System.lineSeparator()).append("=".repeat(title.length()))
.append(System.lineSeparator());
builder.append("Scenario flags: [").append(String.join(", ", scenario.getFlags())).append("]")
.append(System.lineSeparator());
return builder.toString();
}
/**
* Execute a separate "flag" VM with White Box access to determine all test VM flags. The flag VM sends an encoding of
* all required flags for the test VM to the driver VM over a socket. Once the flag VM exits, this driver VM parses the
* test VM flags, which also determine if IR matching should be done, and then starts the test VM to execute all tests.
*/
private void start(Scenario scenario) {
if (scenario != null && !scenario.isEnabled()) {
System.out.println("Disabled scenario #" + scenario.getIndex() + "! This scenario is not present in set flag " +
"-DScenarios and is therefore not executed.");
return;
}
shouldVerifyIR = irVerificationPossible;
try {
// Use TestFramework flags and scenario flags for new VMs.
List<String> additionalFlags = new ArrayList<>();
if (flags != null) {
additionalFlags.addAll(flags);
}
if (scenario != null) {
List<String> scenarioFlags = scenario.getFlags();
String scenarioFlagsString = scenarioFlags.isEmpty() ? "" : " - [" + String.join(", ", scenarioFlags) + "]";
System.out.println("Scenario #" + scenario.getIndex() + scenarioFlagsString + ":");
additionalFlags.addAll(scenarioFlags);
}
String frameworkAndScenarioFlags = additionalFlags.isEmpty() ?
"" : " - [" + String.join(", ", additionalFlags) + "]";
if (shouldVerifyIR) {
// Only need to use flag VM if an IR verification is possibly done.
System.out.println("Run Flag VM:");
FlagVMProcess flagVMProcess = new FlagVMProcess(testClass, additionalFlags);
shouldVerifyIR = flagVMProcess.shouldVerifyIR();
if (shouldVerifyIR) {
// Add more flags for the test VM which are required to do IR verification.
additionalFlags.addAll(flagVMProcess.getTestVMFlags());
} // else: Flag VM found a reason to not do IR verification.
} else {
System.out.println("Skip Flag VM due to not performing IR verification.");
}
System.out.println("Run Test VM" + frameworkAndScenarioFlags + ":");
runTestVM(additionalFlags);
} finally {
if (scenario != null) {
scenario.setTestVMOutput(TestVMProcess.getLastTestVMOutput());
}
System.out.println();
}
}
private boolean hasIRAnnotations() {
return Arrays.stream(testClass.getDeclaredMethods()).anyMatch(m -> m.getAnnotationsByType(IR.class) != null);
}
private boolean onlyWhitelistedJTregVMAndJavaOptsFlags() {
List<String> flags = Arrays.stream(Utils.getTestJavaOpts())
.map(s -> s.replaceFirst("-XX:[+|-]?|-(?=[^D|^e])", ""))
.collect(Collectors.toList());
for (String flag : flags) {
// Property flags (prefix -D), -ea and -esa are whitelisted.
if (!flag.startsWith("-D") && !flag.startsWith("-e") && JTREG_WHITELIST_FLAGS.stream().noneMatch(flag::contains)) {
// Found VM flag that is not whitelisted
return false;
}
}
return true;
}
private void runTestVM(List<String> additionalFlags) {
TestVMProcess testVMProcess = new TestVMProcess(additionalFlags, testClass, helperClasses, defaultWarmup);
if (shouldVerifyIR) {
try {
new IRMatcher(testVMProcess.getHotspotPidFileName(), testVMProcess.getIrEncoding(), testClass);
} catch (IRViolationException e) {
e.addCommandLine(testVMProcess.getCommandLine());
throw e;
}
} else {
System.out.println("IR verification disabled either due to no @IR annotations, through explicitly setting " +
"-DVerify=false, due to not running a debug build, using a non-whitelisted JTreg VM or " +
"Javaopts flag like -Xint, or running the test VM with other VM flags added by user code " +
"that make the IR verification impossible (e.g. -XX:-UseCompile, " +
"-XX:TieredStopAtLevel=[1,2,3], etc.).");
}
}
public static void check(boolean test, String failureMessage) {
if (!test) {
throw new TestFrameworkException(failureMessage);
}
}
}

View File

@ -0,0 +1,97 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import compiler.lib.ir_framework.test.DeclaredTest;
import compiler.lib.ir_framework.test.TestVM;
import java.lang.reflect.Method;
/**
* Test info class which provides some useful utility methods and information about a <b>checked test</b>.
*
* @see Test
* @see Check
*/
public class TestInfo extends AbstractInfo {
private final Method testMethod;
private final boolean compilationSkipped;
public TestInfo(Method testMethod, CompLevel testCmpLevel) {
super(testMethod.getDeclaringClass());
this.testMethod = testMethod;
this.compilationSkipped = testCmpLevel == CompLevel.SKIP;
}
/**
* Get the associated test method object.
*
* @return the associated test method object.
*/
public Method getTest() {
return testMethod;
}
/**
* Return a boolean indicating if the framework skipped a compilation after the warm-up due to VM flags not
* allowing a compilation on the requested level in {@link Test#compLevel()}.
*
* @return {@code true} if the framework skipped compilation of the test;
* {@code false} otherwise.
*/
public boolean isCompilationSkipped() {
return compilationSkipped;
}
/**
* Returns a boolean indicating if the associated test method is C1 compiled.
*
* @return {@code true} if the test method is C1 compiled;
* {@code false} otherwise.
*/
public boolean isC1Compiled() {
return TestVM.isC1Compiled(testMethod);
}
/**
* Returns a boolean indicating if the associated test method is C2 compiled.
*
* @return {@code true} if the test method is C2 compiled;
* {@code false} otherwise.
*/
public boolean isC2Compiled() {
return TestVM.isC2Compiled(testMethod);
}
/**
* Returns a boolean indicating if the associated test method is compiled at {@code compLevel}.
*
* @param compLevel the compilation level.
* @return {@code true} if the test method is compiled at {@code compLevel};
* {@code false} otherwise.
*/
public boolean isCompiledAtLevel(CompLevel compLevel) {
return TestVM.isCompiledAtLevel(testMethod, compLevel);
}
}

View File

@ -0,0 +1,48 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
/**
* This annotation overrides the default number (2000) of times the framework should warm up a test.
* <ul>
* <li><p>Any positive value or zero is permitted. A warm-up of zero allows a simulation of {@code -Xcomp}.</li>
* <li><p>Custom run tests (see {@link Run}) must specify a {@code @Warmup} annotation at the run method.</li>
* <li><p>Base and checked tests (see {@link Test}, {@link Check}) must specify a {@code @Warmup} annotation at
* the test method.</li>
* </ul>
*
* @see Test
* @see Check
* @see Run
*/
@Retention(RetentionPolicy.RUNTIME)
public @interface Warmup {
/**
* The warm-up iterations for the test.
*/
int value();
}

View File

@ -0,0 +1,146 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.driver;
import compiler.lib.ir_framework.TestFramework;
import compiler.lib.ir_framework.shared.TestFrameworkException;
import compiler.lib.ir_framework.flag.FlagVM;
import compiler.lib.ir_framework.shared.TestRunException;
import jdk.test.lib.Utils;
import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.process.ProcessTools;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* This class prepares, creates, and runs the "flag" VM with verification of proper termination. The flag VM determines
* the flags required for the "test" VM. The flag VM writes these flags to a dedicated file which is then parsed by this
* class after the termination of the flag VM.
*
* @see FlagVM
*/
public class FlagVMProcess {
private static final boolean VERBOSE = Boolean.getBoolean("Verbose");
private final List<String> cmds;
private final List<String> testVMFlags;
private boolean shouldVerifyIR;
private String testVMFlagsFile;
private OutputAnalyzer oa;
public FlagVMProcess(Class<?> testClass, List<String> additionalFlags) {
cmds = new ArrayList<>();
testVMFlags = new ArrayList<>();
prepareVMFlags(testClass, additionalFlags);
start();
parseTestVMFlags();
}
private void parseTestVMFlags() {
String flags = readFlagsFromFile();
if (VERBOSE) {
System.out.println("Read data from " + testVMFlagsFile + ":");
System.out.println(flags);
}
String patternString = "(.*DShouldDoIRVerification=(true|false).*)";
Pattern pattern = Pattern.compile(patternString);
Matcher matcher = pattern.matcher(flags);
TestFramework.check(matcher.find(), "Invalid flag encoding emitted by flag VM");
// Maybe we run with flags that make IR verification impossible
shouldVerifyIR = Boolean.parseBoolean(matcher.group(2));
testVMFlags.addAll(Arrays.asList(matcher.group(1).split(FlagVM.TEST_VM_FLAGS_DELIMITER)));
}
private String readFlagsFromFile() {
try (var br = Files.newBufferedReader(Paths.get(testVMFlagsFile))) {
String flags = br.readLine();
TestFramework.check(br.readLine() == null, testVMFlagsFile + " should only contain one line.");
return flags;
} catch (IOException e) {
throw new TestFrameworkException("Error while reading from file " + testVMFlagsFile, e);
}
}
/**
* The flag VM needs White Box access to prepare all test VM flags. The flag VM will write the test VM flags to
* a dedicated file which is afterwards parsed by the driver VM and added as flags to the test VM.
*/
private void prepareVMFlags(Class<?> testClass, List<String> additionalFlags) {
cmds.add("-Dtest.jdk=" + Utils.TEST_JDK);
// Set java.library.path so JNI tests which rely on jtreg nativepath setting work
cmds.add("-Djava.library.path=" + Utils.TEST_NATIVE_PATH);
cmds.add("-cp");
cmds.add(Utils.TEST_CLASS_PATH);
cmds.add("-Xbootclasspath/a:.");
cmds.add("-XX:+UnlockDiagnosticVMOptions");
cmds.add("-XX:+WhiteBoxAPI");
// TestFramework and scenario flags might have an influence on the later used test VM flags. Add them as well.
cmds.addAll(additionalFlags);
cmds.add(FlagVM.class.getCanonicalName());
cmds.add(testClass.getCanonicalName());
}
private void start() {
try {
// Run "flag" VM with White Box access to determine the test VM flags and if IR verification should be done.
oa = ProcessTools.executeTestJvm(cmds);
} catch (Exception e) {
throw new TestRunException("Failed to execute TestFramework flag VM", e);
}
testVMFlagsFile = FlagVM.TEST_VM_FLAGS_FILE_PREFIX + oa.pid()
+ FlagVM.TEST_VM_FLAGS_FILE_POSTFIX;
checkFlagVMExitCode();
}
private void checkFlagVMExitCode() {
String flagVMOutput = oa.getOutput();
int exitCode = oa.getExitValue();
if (VERBOSE && exitCode == 0) {
System.out.println("--- OUTPUT TestFramework flag VM ---");
System.out.println(flagVMOutput);
}
if (exitCode != 0) {
System.err.println("--- OUTPUT TestFramework flag VM ---");
System.err.println(flagVMOutput);
throw new RuntimeException("TestFramework flag VM exited with " + exitCode);
}
}
public List<String> getTestVMFlags() {
return testVMFlags;
}
public boolean shouldVerifyIR() {
return shouldVerifyIR;
}
}

View File

@ -0,0 +1,493 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.driver;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.shared.*;
import compiler.lib.ir_framework.test.*;
import java.io.IOException;
import java.lang.reflect.Method;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* Parse the hotspot pid file of the test VM to match all @IR rules.
*/
public class IRMatcher {
private static final boolean PRINT_IR_ENCODING = Boolean.parseBoolean(System.getProperty("PrintIREncoding", "false"));
private static final Pattern IR_ENCODING_PATTERN =
Pattern.compile("(?<=" + IREncodingPrinter.START + "\r?\n)[\\s\\S]*(?=" + IREncodingPrinter.END + ")");
private static final Pattern COMPILE_ID_PATTERN = Pattern.compile("compile_id='(\\d+)'");
private final Map<String, IRMethod> compilations;
private final Class<?> testClass;
private final Map<Method, List<String>> fails;
private final Pattern compileIdPatternForTestClass;
private final String hotspotPidFileName;
private IRMethod irMethod; // Current IR method to which rules are applied
private Method method; // Current method to which rules are applied
private IR irAnno; // Current IR annotation that is processed.
private int irRuleIndex; // Current IR rule index;
public IRMatcher(String hotspotPidFileName, String irEncoding, Class<?> testClass) {
this.compilations = new HashMap<>();
this.fails = new HashMap<>();
this.testClass = testClass;
this.compileIdPatternForTestClass = Pattern.compile("compile_id='(\\d+)'.*" + Pattern.quote(testClass.getCanonicalName())
+ " (\\S+)");
this.hotspotPidFileName = hotspotPidFileName;
setupTestMethods(irEncoding);
if (TestFramework.VERBOSE || PRINT_IR_ENCODING) {
System.out.println("Read IR encoding from test VM:");
System.out.println(irEncoding);
}
if (!compilations.isEmpty()) {
parseHotspotPidFile();
applyRules();
}
}
/**
* Sets up a map testname -> IRMethod (containing the PrintIdeal and PrintOptoAssembly output for testname).
*/
private void setupTestMethods(String irEncoding) {
Map<String, int[]> irRulesMap = parseIREncoding(irEncoding);
for (Method m : testClass.getDeclaredMethods()) {
method = m;
IR[] irAnnos = m.getAnnotationsByType(IR.class);
if (irAnnos.length > 0) {
// Validation of legal @IR attributes and placement of the annotation was already done in Test VM.
int[] ids = irRulesMap.get(m.getName());
TestFramework.check(ids != null, "Should find method name in validIrRulesMap for " + m);
TestFramework.check(ids.length > 0, "Did not find any rule indices for " + m);
TestFramework.check(ids[ids.length - 1] < irAnnos.length, "Invalid IR rule index found in validIrRulesMap for " + m);
if (ids[0] != IREncodingPrinter.NO_RULE_APPLIED) {
// If -1, than there was no matching IR rule for the given conditions.
compilations.put(m.getName(), new IRMethod(m, ids, irAnnos));
}
}
}
}
/**
* Read the IR encoding emitted by the test VM to decide if an @IR rule must be checked for a method.
*/
private Map<String, int[]> parseIREncoding(String irEncoding) {
Map<String, int[]> irRulesMap = new HashMap<>();
Matcher matcher = IR_ENCODING_PATTERN.matcher(irEncoding);
TestFramework.check(matcher.find(), "Did not find IR encoding");
String[] lines = matcher.group(0).split("\\R");
// Skip first line containing information about the format only
for (int i = 1; i < lines.length; i++) {
String line = lines[i].trim();
String[] splitComma = line.split(",");
if (splitComma.length < 2) {
throw new TestFrameworkException("Invalid IR match rule encoding. No comma found: " + splitComma[0]);
}
String testName = splitComma[0];
int[] irRulesIdx = new int[splitComma.length - 1];
for (int j = 1; j < splitComma.length; j++) {
try {
irRulesIdx[j - 1] = Integer.parseInt(splitComma[j]);
} catch (NumberFormatException e) {
throw new TestFrameworkException("Invalid IR match rule encoding. No number found: " + splitComma[j]);
}
}
irRulesMap.put(testName, irRulesIdx);
}
return irRulesMap;
}
/**
* Parse the hotspot_pid*.log file from the test VM. Read the PrintIdeal and PrintOptoAssembly entries for all
* methods of the test class that need to be IR matched (according to IR encoding).
*/
private void parseHotspotPidFile() {
Map<Integer, String> compileIdMap = new HashMap<>();
try (var br = Files.newBufferedReader(Paths.get(hotspotPidFileName))) {
String line;
StringBuilder builder = new StringBuilder();
boolean append = false;
String currentMethod = "";
while ((line = br.readLine()) != null) {
if (append && line.startsWith("</")) {
flushOutput(line, builder, currentMethod);
append = false;
currentMethod = "";
continue;
} else if (append) {
appendLine(builder, line);
continue;
}
if (maybeTestEntry(line)) {
addTestMethodCompileId(compileIdMap, line);
} else if (isPrintIdealStart(line)) {
String methodName = getMethodName(compileIdMap, line);
if (methodName != null) {
currentMethod = methodName;
append = true; // Append all following lines until we hit the closing </ideal> tag.
}
} else if (isPrintOptoAssemblyStart(line)) {
String methodName = getMethodName(compileIdMap, line);
if (methodName != null) {
TestFramework.check(compilations.containsKey(methodName), "Must be second entry of " + methodName);
currentMethod = methodName;
append = true; // Append all following lines until we hit the closing </opto_assembly> tag.
}
}
}
} catch (IOException e) {
throw new TestFrameworkException("Error while reading " + hotspotPidFileName, e);
}
}
/**
* Write the input to the IR method and reset the builder.
*/
private void flushOutput(String line, StringBuilder builder, String currentMethod) {
TestFramework.check(!currentMethod.isEmpty(), "current method must be set");
IRMethod irMethod = compilations.get(currentMethod);
if (line.startsWith("</i")) {
// PrintIdeal
irMethod.setIdealOutput(builder.toString());
} else {
// PrintOptoAssembly
irMethod.setOptoAssemblyOutput(builder.toString());
}
builder.setLength(0);
}
/**
* Only consider non-osr (no "compile_kind") and compilations with C2 (no "level")
*/
private boolean maybeTestEntry(String line) {
return line.startsWith("<task_queued") && !line.contains("compile_kind='") && !line.contains("level='");
}
/**
* Need to escape XML special characters.
*/
private static void appendLine(StringBuilder builder, String line) {
if (line.contains("&")) {
line = line.replace("&lt;", "<");
line = line.replace("&gt;", ">");
line = line.replace("&quot;", "\"");
line = line.replace("&apos;", "'");
line = line.replace("&amp;", "&");
}
builder.append(line).append(System.lineSeparator());
}
private static int getCompileId(Matcher matcher) {
int compileId;
try {
compileId = Integer.parseInt(matcher.group(1));
} catch (NumberFormatException e) {
throw new TestRunException("Could not parse compile id", e);
}
return compileId;
}
/**
* Parse the compile id from this line if it belongs to a method that needs to be IR tested (part of test class
* and IR encoding from the test VM specifies that this method has @IR rules to be checked).
*/
private void addTestMethodCompileId(Map<Integer, String> compileIdMap, String line) {
Matcher matcher = compileIdPatternForTestClass.matcher(line);
if (matcher.find()) {
// Only care about test class entries. Might have non-class entries as well if user specified additional
// compile commands. Ignore these.
String methodName = matcher.group(2);
if (compilations.containsKey(methodName)) {
// We only care about methods that we are actually gonna IR match based on IR encoding.
int compileId = getCompileId(matcher);
TestRun.check(!methodName.isEmpty(), "method name cannot be empty");
compileIdMap.put(compileId, methodName);
}
}
}
/**
* Make sure that line does not contain compile_kind which is used for OSR compilations which we are not
* interested in.
*/
private static boolean isPrintIdealStart(String line) {
return line.startsWith("<ideal") && !line.contains("compile_kind='");
}
/**
* Make sure that line does not contain compile_kind which is used for OSR compilations which we are not
* interested in.
*/
private static boolean isPrintOptoAssemblyStart(String line) {
return line.startsWith("<opto_assembly") && !line.contains("compile_kind='");
}
/**
* Get method name for this line by looking up the compile id.
* Returns null if not an interesting method (i.e. from test class).
*/
private String getMethodName(Map<Integer, String> compileIdMap, String line) {
Matcher matcher = COMPILE_ID_PATTERN.matcher(line);
TestFramework.check(matcher.find(), "Is " + hotspotPidFileName + " corrupted?");
int compileId = getCompileId(matcher);
return compileIdMap.get(compileId);
}
/**
* Do an IR matching of all methods with appliable @IR rules fetched during parsing of the hotspot pid file.
*/
private void applyRules() {
compilations.values().forEach(this::applyRulesForMethod);
reportFailuresIfAny();
}
private void applyRulesForMethod(IRMethod irMethod) {
this.irMethod = irMethod;
method = irMethod.getMethod();
String testOutput = irMethod.getOutput();
if (testOutput.isEmpty()) {
String msg = "Method was not compiled. Did you specify any compiler directives preventing a compilation or used a " +
"@Run method in STANDALONE mode? In the latter case, make sure to always trigger a C2 compilation " +
"by invoking the test enough times.";
fails.computeIfAbsent(method, k -> new ArrayList<>()).add(msg);
return;
}
if (TestFramework.VERBOSE) {
System.out.println("Output of " + method + ":");
System.out.println(testOutput);
}
Arrays.stream(irMethod.getRuleIds()).forEach(this::applyIRRule);
}
/**
* Apply a single @IR rule as part of a method.
*/
private void applyIRRule(int id) {
irAnno = irMethod.getIrAnno(id);
irRuleIndex = id;
StringBuilder failMsg = new StringBuilder();
applyFailOn(failMsg);
try {
applyCounts(failMsg);
} catch (TestFormatException e) {
// Logged. Continue to check other rules.
}
if (!failMsg.isEmpty()) {
failMsg.insert(0, "@IR rule " + (id + 1) + ": \"" + irAnno + "\"" + System.lineSeparator());
fails.computeIfAbsent(method, k -> new ArrayList<>()).add(failMsg.toString());
}
}
/**
* Apply the failOn regexes of the @IR rule.
*/
private void applyFailOn(StringBuilder failMsg) {
if (irAnno.failOn().length != 0) {
String failOnRegex = String.join("|", IRNode.mergeNodes(irAnno.failOn()));
Pattern pattern = Pattern.compile(failOnRegex);
Matcher matcher = pattern.matcher(irMethod.getOutput());
long matchCount = matcher.results().count();
if (matchCount > 0) {
addFailOnFailsForOutput(failMsg, pattern, matchCount);
}
}
}
/**
* A failOn regex failed. Apply all regexes again to log the exact regex which failed. The failure is later reported
* to the user.
*/
private void addFailOnFailsForOutput(StringBuilder failMsg, Pattern pattern, long matchCount) {
long idealCount = pattern.matcher(irMethod.getIdealOutput()).results().count();
long optoAssemblyCount = pattern.matcher(irMethod.getOptoAssemblyOutput()).results().count();
if (matchCount != idealCount + optoAssemblyCount || (idealCount != 0 && optoAssemblyCount != 0)) {
// Report with Ideal and Opto Assembly
addFailOnFailsForOutput(failMsg, irMethod.getOutput());
irMethod.needsAllOutput();
} else if (optoAssemblyCount == 0) {
// Report with Ideal only
addFailOnFailsForOutput(failMsg, irMethod.getIdealOutput());
irMethod.needsIdeal();
} else {
// Report with Opto Assembly only
addFailOnFailsForOutput(failMsg, irMethod.getOptoAssemblyOutput());
irMethod.needsOptoAssembly();
}
}
/**
* Apply the regexes to the testOutput and log the failures.
*/
private void addFailOnFailsForOutput(StringBuilder failMsg, String testOutput) {
List<String> failOnNodes = IRNode.mergeNodes(irAnno.failOn());
Pattern pattern;
Matcher matcher;
failMsg.append("- failOn: Graph contains forbidden nodes:").append(System.lineSeparator());
int nodeId = 1;
for (String nodeRegex : failOnNodes) {
pattern = Pattern.compile(nodeRegex);
matcher = pattern.matcher(testOutput);
long matchCount = matcher.results().count();
if (matchCount > 0) {
matcher.reset();
failMsg.append(" Regex ").append(nodeId).append(": ").append(nodeRegex).append(System.lineSeparator());
failMsg.append(" Matched forbidden node").append(matchCount > 1 ? "s (" + matchCount + ")" : "")
.append(":").append(System.lineSeparator());
matcher.results().forEach(r -> failMsg.append(" ").append(r.group()).append(System.lineSeparator()));
}
nodeId++;
}
}
/**
* Apply the counts regexes of the @IR rule.
*/
private void applyCounts(StringBuilder failMsg) {
if (irAnno.counts().length != 0) {
boolean hasFails = false;
String testOutput = irMethod.getOutput();
int countsId = 1;
final List<String> nodesWithCount = IRNode.mergeNodes(irAnno.counts());
for (int i = 0; i < nodesWithCount.size(); i += 2) {
String node = nodesWithCount.get(i);
TestFormat.check(i + 1 < nodesWithCount.size(), "Missing count" + getPostfixErrorMsg(node));
String countString = nodesWithCount.get(i + 1);
long expectedCount;
ParsedComparator<Long> parsedComparator;
try {
parsedComparator = ParsedComparator.parseComparator(countString);
expectedCount = Long.parseLong(parsedComparator.getStrippedString());
} catch (NumberFormatException e) {
TestFormat.fail("Provided invalid count \"" + countString + "\"" + getPostfixErrorMsg(node));
return;
} catch (CheckedTestFrameworkException e) {
TestFormat.fail("Invalid comparator \"" + e.getMessage() + "\" in \"" + countString + "\" for count" + getPostfixErrorMsg(node));
return;
} catch (IndexOutOfBoundsException e) {
TestFormat.fail("Provided empty value" + getPostfixErrorMsg(node));
return;
}
TestFormat.check(expectedCount >= 0,"Provided invalid negative count \"" + countString + "\"" + getPostfixErrorMsg(node));
Pattern pattern = Pattern.compile(node);
Matcher matcher = pattern.matcher(testOutput);
long actualCount = matcher.results().count();
if (!parsedComparator.getPredicate().test(actualCount, expectedCount)) {
if (!hasFails) {
failMsg.append("- counts: Graph contains wrong number of nodes:").append(System.lineSeparator());
hasFails = true;
}
addCountsFail(failMsg, node, pattern, expectedCount, actualCount, countsId);
}
countsId++;
}
}
}
private String getPostfixErrorMsg(String node) {
return " for IR rule " + irRuleIndex + ", node \"" + node + "\" at " + method;
}
/**
* A counts regex failed. Apply all regexes again to log the exact regex which failed. The failure is later reported
* to the user.
*/
private void addCountsFail(StringBuilder failMsg, String node, Pattern pattern, long expectedCount, long actualCount, int countsId) {
failMsg.append(" Regex ").append(countsId).append(": ").append(node).append(System.lineSeparator());
failMsg.append(" Expected ").append(expectedCount).append(" but found ").append(actualCount);
if (actualCount > 0) {
Matcher matcher = pattern.matcher(irMethod.getOutput());
long idealCount = pattern.matcher(irMethod.getIdealOutput()).results().count();
long optoAssemblyCount = pattern.matcher(irMethod.getOptoAssemblyOutput()).results().count();
if (actualCount != idealCount + optoAssemblyCount || (idealCount != 0 && optoAssemblyCount != 0)) {
irMethod.needsAllOutput();
} else if (optoAssemblyCount == 0) {
irMethod.needsIdeal();
} else {
irMethod.needsOptoAssembly();
}
failMsg.append(" node").append(actualCount > 1 ? "s" : "").append(":").append(System.lineSeparator());
matcher.results().forEach(r -> failMsg.append(" ").append(r.group()).append(System.lineSeparator()));
} else {
irMethod.needsAllOutput();
failMsg.append(" nodes.").append(System.lineSeparator());
}
}
/**
* Report all IR violations in a pretty format to the user. Depending on the failed regex, we only report
* PrintIdeal or PrintOptoAssembly if the match failed there. If there were failures that matched things
* in both outputs than the entire output is reported. Throws IRViolationException from which the compilation
* can be read and reported to the stdout separately. The exception message only includes the summary of the
* failures.
*/
private void reportFailuresIfAny() {
TestFormat.reportIfAnyFailures();
if (!fails.isEmpty()) {
StringBuilder failuresBuilder = new StringBuilder();
StringBuilder compilationsBuilder = new StringBuilder();
int failures = 0;
for (Map.Entry<Method, List<String>> entry : fails.entrySet()) {
Method method = entry.getKey();
compilationsBuilder.append(">>> Compilation of ").append(method).append(":").append(System.lineSeparator());
IRMethod irMethod = compilations.get(method.getName());
String output;
if (irMethod.usesIdeal() && irMethod.usesOptoAssembly()) {
output = irMethod.getOutput();
} else if (irMethod.usesIdeal()) {
output = irMethod.getIdealOutput();
} else if (irMethod.usesOptoAssembly()) {
output = irMethod.getOptoAssemblyOutput();
} else {
output = "<empty>";
}
compilationsBuilder.append(output).append(System.lineSeparator()).append(System.lineSeparator());
List<String> list = entry.getValue();
failuresBuilder.append("- Method \"").append(method).append("\":").append(System.lineSeparator());
failures += list.size();
list.forEach(s -> failuresBuilder.append(" * ")
.append(s.replace(System.lineSeparator(),
System.lineSeparator() + " ").trim())
.append(System.lineSeparator()));
failuresBuilder.append(System.lineSeparator());
}
failuresBuilder.insert(0, ("One or more @IR rules failed:" + System.lineSeparator()
+ System.lineSeparator() + "Failed IR Rules (" + failures + ")"
+ System.lineSeparator()) + "-----------------"
+ "-".repeat(String.valueOf(failures).length()) + System.lineSeparator());
failuresBuilder.append(">>> Check stdout for compilation output of the failed methods")
.append(System.lineSeparator()).append(System.lineSeparator());
throw new IRViolationException(failuresBuilder.toString(), compilationsBuilder.toString());
}
}
}

View File

@ -0,0 +1,117 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.driver;
import compiler.lib.ir_framework.IR;
import java.lang.reflect.Method;
/**
* Helper class to store information about a method that needs to be IR matched.
*/
class IRMethod {
private final Method method;
private final int[] ruleIds;
private final IR[] irAnnos;
private final StringBuilder outputBuilder;
private String output;
private String idealOutput;
private String optoAssemblyOutput;
private boolean needsIdeal;
private boolean needsOptoAssembly;
public IRMethod(Method method, int[] ruleIds, IR[] irAnnos) {
this.method = method;
this.ruleIds = ruleIds;
this.irAnnos = irAnnos;
this.outputBuilder = new StringBuilder();
this.output = "";
this.idealOutput = "";
this.optoAssemblyOutput = "";
}
public Method getMethod() {
return method;
}
public int[] getRuleIds() {
return ruleIds;
}
public IR getIrAnno(int idx) {
return irAnnos[idx];
}
/**
* The Ideal output comes always before the Opto Assembly output. We might parse multiple C2 compilations of this method.
* Only keep the very last one by overriding 'output'.
*/
public void setIdealOutput(String idealOutput) {
outputBuilder.setLength(0);
this.idealOutput = "PrintIdeal:" + System.lineSeparator() + idealOutput;
outputBuilder.append(this.idealOutput);
}
/**
* The Opto Assembly output comes after the Ideal output. Simply append to 'output'.
*/
public void setOptoAssemblyOutput(String optoAssemblyOutput) {
this.optoAssemblyOutput = "PrintOptoAssembly:" + System.lineSeparator() + optoAssemblyOutput;
outputBuilder.append(System.lineSeparator()).append(System.lineSeparator()).append(this.optoAssemblyOutput);
output = outputBuilder.toString();
}
public String getOutput() {
return output;
}
public String getIdealOutput() {
return idealOutput;
}
public String getOptoAssemblyOutput() {
return optoAssemblyOutput;
}
public void needsAllOutput() {
needsIdeal();
needsOptoAssembly();
}
public void needsIdeal() {
needsIdeal = true;
}
public boolean usesIdeal() {
return needsIdeal;
}
public void needsOptoAssembly() {
needsOptoAssembly = true;
}
public boolean usesOptoAssembly() {
return needsOptoAssembly;
}
}

View File

@ -0,0 +1,63 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.driver;
import compiler.lib.ir_framework.IR;
import compiler.lib.ir_framework.Test;
/**
* Exception that is thrown if an {@link IR} rule/constraint failed. The exception message contains a detailed list of
* all failures, including failing method(s), {@code @IR} rule(s) (the first {@code @IR} constraint is rule 1) and the
* specific regex(es) that could not be matched.
*
* @see IR
* @see Test
*/
public class IRViolationException extends RuntimeException {
private final String compilations;
private String exceptionInfo;
IRViolationException(String message, String compilations) {
super("There were one or multiple IR rule failures. Please check stderr for more information.");
this.exceptionInfo = message;
this.compilations = compilations;
}
/**
* Get some more detailed information about the violated IR rule(s) and how to reproduce it.
*
* @return a formatted string containing information about the violated IR rule(s) and how to reproduce it.
*/
public String getExceptionInfo() {
return exceptionInfo;
}
public String getCompilations() {
return compilations;
}
public void addCommandLine(String commandLine) {
this.exceptionInfo = commandLine + System.lineSeparator() + exceptionInfo;
}
}

View File

@ -0,0 +1,47 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.driver;
import compiler.lib.ir_framework.shared.TestFormatException;
/**
* Exception that is thrown if the test VM has thrown any kind of exception (except for {@link TestFormatException}).
*/
public class TestVMException extends RuntimeException {
private final String exceptionInfo;
TestVMException(String exceptionInfo) {
super("There were one or multiple errors. Please check stderr for more information.");
this.exceptionInfo = exceptionInfo;
}
/**
* Get some more detailed information about the exception thrown in the test VM and how to reproduce it.
*
* @return a formatted string containing information about the exception of the test VM and how to reproduce it.
*/
public String getExceptionInfo() {
return exceptionInfo;
}
}

View File

@ -0,0 +1,236 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.driver;
import compiler.lib.ir_framework.TestFramework;
import compiler.lib.ir_framework.shared.TestFrameworkException;
import compiler.lib.ir_framework.shared.TestFrameworkSocket;
import compiler.lib.ir_framework.shared.NoTestsRunException;
import compiler.lib.ir_framework.shared.TestFormatException;
import compiler.lib.ir_framework.test.TestVM;
import jdk.test.lib.Platform;
import jdk.test.lib.Utils;
import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.process.ProcessTools;
import java.util.*;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* This class prepares, creates, and runs the "test" VM with verification of proper termination. The class also stores
* information about the test VM which is later queried for IR matching. The communication between this driver VM
* and the test VM is done over a dedicated socket.
*
* @see TestVM
* @see TestFrameworkSocket
*/
public class TestVMProcess {
private static final boolean VERBOSE = Boolean.getBoolean("Verbose");
private static final boolean PREFER_COMMAND_LINE_FLAGS = Boolean.getBoolean("PreferCommandLineFlags");
private static final int WARMUP_ITERATIONS = Integer.getInteger("Warmup", -1);
private static final boolean VERIFY_VM = Boolean.getBoolean("VerifyVM") && Platform.isDebugBuild();
private static final boolean REPORT_STDOUT = Boolean.getBoolean("ReportStdout");
private static final boolean EXCLUDE_RANDOM = Boolean.getBoolean("ExcludeRandom");
private static String lastTestVMOutput = "";
private final ArrayList<String> cmds;
private String hotspotPidFileName;
private String commandLine;
private OutputAnalyzer oa;
private String irEncoding;
public TestVMProcess(List<String> additionalFlags, Class<?> testClass, Set<Class<?>> helperClasses, int defaultWarmup) {
this.cmds = new ArrayList<>();
TestFrameworkSocket socket = new TestFrameworkSocket();
try (socket) {
prepareTestVMFlags(additionalFlags, socket, testClass, helperClasses, defaultWarmup);
start();
}
processSocketOutput(socket.getOutput());
checkTestVMExitCode();
}
public String getCommandLine() {
return commandLine;
}
public String getIrEncoding() {
return irEncoding;
}
public String getHotspotPidFileName() {
return hotspotPidFileName;
}
public static String getLastTestVMOutput() {
return lastTestVMOutput;
}
private void prepareTestVMFlags(List<String> additionalFlags, TestFrameworkSocket socket, Class<?> testClass,
Set<Class<?>> helperClasses, int defaultWarmup) {
// Set java.library.path so JNI tests which rely on jtreg nativepath setting work
cmds.add("-Djava.library.path=" + Utils.TEST_NATIVE_PATH);
// Need White Box access in test VM.
cmds.add("-Xbootclasspath/a:.");
cmds.add("-XX:+UnlockDiagnosticVMOptions");
cmds.add("-XX:+WhiteBoxAPI");
String[] jtregVMFlags = Utils.getTestJavaOpts();
if (!PREFER_COMMAND_LINE_FLAGS) {
cmds.addAll(Arrays.asList(jtregVMFlags));
}
// Add server property flag that enables test VM to print encoding for IR verification last and debug messages.
cmds.add(socket.getPortPropertyFlag());
cmds.addAll(additionalFlags);
cmds.addAll(Arrays.asList(getDefaultFlags()));
if (VERIFY_VM) {
cmds.addAll(Arrays.asList(getVerifyFlags()));
}
if (PREFER_COMMAND_LINE_FLAGS) {
// Prefer flags set via the command line over the ones set by scenarios.
cmds.addAll(Arrays.asList(jtregVMFlags));
}
if (WARMUP_ITERATIONS < 0 && defaultWarmup != -1) {
// Only use the set warmup for the framework if not overridden by a valid -DWarmup property set by a test.
cmds.add("-DWarmup=" + defaultWarmup);
}
cmds.add(TestVM.class.getName());
cmds.add(testClass.getName());
if (helperClasses != null) {
helperClasses.forEach(c -> cmds.add(c.getName()));
}
}
/**
* Default flags that are added used for the test VM.
*/
private static String[] getDefaultFlags() {
return new String[] {"-XX:-BackgroundCompilation", "-XX:CompileCommand=quiet"};
}
/**
* Additional verification flags that are used if -DVerifyVM=true is with a debug build.
*/
private static String[] getVerifyFlags() {
return new String[] {
"-XX:+UnlockDiagnosticVMOptions", "-XX:+VerifyOops", "-XX:+VerifyStack", "-XX:+VerifyLastFrame",
"-XX:+VerifyBeforeGC", "-XX:+VerifyAfterGC", "-XX:+VerifyDuringGC", "-XX:+VerifyAdapterSharing"
};
}
private void start() {
ProcessBuilder process = ProcessTools.createJavaProcessBuilder(cmds);
try {
// Calls 'main' of TestVM to run all specified tests with commands 'cmds'.
// Use executeProcess instead of executeTestJvm as we have already added the JTreg VM and
// Java options in prepareTestVMFlags().
oa = ProcessTools.executeProcess(process);
} catch (Exception e) {
throw new TestFrameworkException("Error while executing Test VM", e);
}
commandLine = "Command Line:" + System.lineSeparator() + String.join(" ", process.command())
+ System.lineSeparator();
hotspotPidFileName = String.format("hotspot_pid%d.log", oa.pid());
lastTestVMOutput = oa.getOutput();
}
/**
* Process the socket output: All prefixed lines are dumped to the standard output while the remaining lines
* represent the IR encoding used for IR matching later.
*/
private void processSocketOutput(String output) {
if (TestFramework.TESTLIST || TestFramework.EXCLUDELIST) {
StringBuilder builder = new StringBuilder();
Scanner scanner = new Scanner(output);
System.out.println(System.lineSeparator() + "Run flag defined test list");
System.out.println("--------------------------");
while (scanner.hasNextLine()) {
String line = scanner.nextLine();
if (line.startsWith(TestFrameworkSocket.STDOUT_PREFIX)) {
line = "> " + line.substring(TestFrameworkSocket.STDOUT_PREFIX.length());
System.out.println(line);
} else {
builder.append(line).append(System.lineSeparator());
}
}
System.out.println();
irEncoding = builder.toString();
} else {
irEncoding = output;
}
}
private void checkTestVMExitCode() {
final int exitCode = oa.getExitValue();
if (EXCLUDE_RANDOM || REPORT_STDOUT || (VERBOSE && exitCode == 0)) {
System.out.println("--- OUTPUT TestFramework test VM ---");
System.out.println(oa.getOutput());
}
if (exitCode != 0) {
throwTestVMException();
}
}
/**
* Exit code was non-zero of test VM. Check the stderr to determine what kind of exception that should be thrown to
* react accordingly later.
*/
private void throwTestVMException() {
String stdErr = oa.getStderr();
if (stdErr.contains("TestFormat.reportIfAnyFailures")) {
Pattern pattern = Pattern.compile("Violations \\(\\d+\\)[\\s\\S]*(?=/============/)");
Matcher matcher = pattern.matcher(stdErr);
TestFramework.check(matcher.find(), "Must find violation matches");
throw new TestFormatException(System.lineSeparator() + System.lineSeparator() + matcher.group());
} else if (stdErr.contains("NoTestsRunException")) {
throw new NoTestsRunException(">>> No tests run due to empty set specified with -DTest and/or -DExclude. " +
"Make sure to define a set of at least one @Test method");
} else {
throw new TestVMException(getExceptionInfo());
}
}
/**
* Get more detailed information about the exception in a pretty format.
*/
private String getExceptionInfo() {
int exitCode = oa.getExitValue();
String stdErr = oa.getStderr();
String stdOut = "";
if (exitCode == 134) {
// Also dump the stdout if we experience a JVM error (e.g. to show hit assertions etc.).
stdOut = System.lineSeparator() + System.lineSeparator() + "Standard Output" + System.lineSeparator()
+ "---------------" + System.lineSeparator() + oa.getOutput();
}
return "TestFramework test VM exited with code " + exitCode + System.lineSeparator() + stdOut
+ System.lineSeparator() + commandLine + System.lineSeparator() + System.lineSeparator()
+ "Error Output" + System.lineSeparator() + "------------" + System.lineSeparator() + stdErr
+ System.lineSeparator() + System.lineSeparator();
}
}

View File

@ -0,0 +1,135 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.flag;
import compiler.lib.ir_framework.CompLevel;
import compiler.lib.ir_framework.TestFramework;
import compiler.lib.ir_framework.shared.TestFrameworkException;
import compiler.lib.ir_framework.shared.TestRunException;
import jdk.test.lib.process.ProcessTools;
import sun.hotspot.WhiteBox;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
/**
* This class' main method is called from {@link TestFramework} and represents the so-called "flag VM". It uses the
* Whitebox API to determine the necessary additional flags to run the test VM (e.g. to do IR matching). It returns
* the flags over the dedicated TestFramework socket.
*/
public class FlagVM {
public static final String TEST_VM_FLAGS_FILE_PREFIX = "test-vm-flags-pid-";
public static final String TEST_VM_FLAGS_FILE_POSTFIX = ".log";
public static final String TEST_VM_FLAGS_DELIMITER = " ";
private static final String TEST_VM_FLAGS_FILE;
private static final WhiteBox WHITE_BOX;
static {
try {
WHITE_BOX = WhiteBox.getWhiteBox();
TEST_VM_FLAGS_FILE = TEST_VM_FLAGS_FILE_PREFIX + ProcessTools.getProcessId() + TEST_VM_FLAGS_FILE_POSTFIX;
} catch (UnsatisfiedLinkError e) {
throw new TestFrameworkException("Could not load WhiteBox", e);
} catch (Exception e) {
throw new TestFrameworkException("Could not get process id", e);
}
}
private static final boolean TIERED_COMPILATION = (Boolean)WHITE_BOX.getVMFlag("TieredCompilation");
private static final CompLevel TIERED_COMPILATION_STOP_AT_LEVEL =
CompLevel.forValue(((Long)WHITE_BOX.getVMFlag("TieredStopAtLevel")).intValue());
private static final boolean TEST_C1 = TIERED_COMPILATION
&& TIERED_COMPILATION_STOP_AT_LEVEL.getValue() < CompLevel.C2.getValue();
private static final boolean VERBOSE = Boolean.getBoolean("Verbose");
private static final boolean USE_COMPILER = WHITE_BOX.getBooleanVMFlag("UseCompiler");
private static final boolean EXCLUDE_RANDOM = Boolean.getBoolean("ExcludeRandom");
private static final boolean FLIP_C1_C2 = Boolean.getBoolean("FlipC1C2");
private static final boolean REQUESTED_VERIFY_IR = Boolean.parseBoolean(System.getProperty("VerifyIR", "true"));
private static final boolean VERIFY_IR = REQUESTED_VERIFY_IR && USE_COMPILER && !EXCLUDE_RANDOM && !FLIP_C1_C2 && !TEST_C1;
private static String[] getPrintFlags() {
return new String[] {"-XX:+PrintCompilation", "-XX:+UnlockDiagnosticVMOptions"};
}
/**
* Main entry point of the flag VM.
*/
public static void main(String[] args) {
String testClassName = args[0];
if (VERBOSE) {
System.out.println("FlagVM main() called. Prepare test VM flags to run class " + testClassName);
}
Class<?> testClass;
try {
testClass = Class.forName(testClassName);
} catch (Exception e) {
throw new TestRunException("Could not find test class " + testClassName, e);
}
emitTestVMFlags(prepareTestVmFlags(testClass));
}
/**
* Emit test VM flags to the dedicated test VM flags file to parse them from the TestFramework "driver" VM again
* which adds them to the test VM.
*/
private static void emitTestVMFlags(ArrayList<String> flags) {
try (var bw = Files.newBufferedWriter(Paths.get(TEST_VM_FLAGS_FILE))) {
bw.write(String.join(TEST_VM_FLAGS_DELIMITER, flags));
} catch (IOException e) {
throw new TestFrameworkException("Error while writing to file " + TEST_VM_FLAGS_FILE, e);
}
}
private static ArrayList<String> prepareTestVmFlags(Class<?> testClass) {
return setupIrVerificationFlags(testClass);
}
private static ArrayList<String> setupIrVerificationFlags(Class<?> testClass) {
ArrayList<String> cmds = new ArrayList<>();
if (VERIFY_IR) {
// Add print flags for IR verification
cmds.addAll(Arrays.asList(getPrintFlags()));
cmds.add("-XX:+LogCompilation");
cmds.add("-XX:CompileCommand=log," + testClass.getCanonicalName() + "::*");
addBoolOptionForClass(cmds, testClass, "PrintIdeal");
addBoolOptionForClass(cmds, testClass, "PrintOptoAssembly");
// Always trap for exception throwing to not confuse IR verification
cmds.add("-XX:-OmitStackTraceInFastThrow");
cmds.add("-DShouldDoIRVerification=true");
} else {
cmds.add("-DShouldDoIRVerification=false");
}
return cmds;
}
private static void addBoolOptionForClass(ArrayList<String> cmds, Class<?> testClass, String option) {
cmds.add("-XX:CompileCommand=option," + testClass.getCanonicalName() + "::*,bool," + option + ",true");
}
}

View File

@ -0,0 +1,33 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.shared;
/**
* Checked internal exceptions in the framework to propagate error handling.
*/
public class CheckedTestFrameworkException extends Exception {
public CheckedTestFrameworkException(String msg) {
super(msg);
}
}

View File

@ -0,0 +1,42 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.shared;
/**
* Exception that is thrown by the test VM if no tests are run as a result of specifying {@code -DTest} and/or
* {@code -DExclude} defining an empty set with the used test VM flags.
*/
public class NoTestsRunException extends RuntimeException {
/**
* Default constructor used by test VM
*/
public NoTestsRunException() {}
/**
* Constructor used to eventually throw the exception in the driver VM.
*/
public NoTestsRunException(String message) {
super(message);
}
}

View File

@ -0,0 +1,104 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.shared;
import java.util.function.BiPredicate;
/**
* Utility class to parse a comparator either in the applyIf* or in the counts properties of an @IR rules.
*/
public class ParsedComparator<T extends Comparable<T>> {
private final String strippedString;
private final BiPredicate<T, T> predicate;
private final String comparator;
public ParsedComparator(String strippedString, BiPredicate<T, T> predicate, String comparator) {
this.strippedString = strippedString;
this.predicate = predicate;
this.comparator = comparator;
}
public String getStrippedString() {
return strippedString;
}
public BiPredicate<T, T> getPredicate() {
return predicate;
}
public String getComparator() {
return comparator;
}
/**
* Return parsed comparator object which provides the predicate to perform the test.
* Allowed comparators: <, <=, >, =>, =, !=
*/
public static <T extends Comparable<T>> ParsedComparator<T> parseComparator(String value) throws CheckedTestFrameworkException {
BiPredicate<T, T> comparison;
value = value.trim();
String comparator = "";
switch (value.charAt(0)) {
case '<':
if (value.charAt(1) == '=') {
comparator = "<=";
comparison = (x, y) -> x.compareTo(y) <= 0;
value = value.substring(2).trim();
} else {
comparator = "<";
comparison = (x, y) -> x.compareTo(y) < 0;
value = value.substring(1).trim();
}
break;
case '>':
if (value.charAt(1) == '=') {
comparator = ">=";
comparison = (x, y) -> x.compareTo(y) >= 0;
value = value.substring(2).trim();
} else {
comparator = ">";
comparison = (x, y) -> x.compareTo(y) > 0;
value = value.substring(1).trim();
}
break;
case '!':
if (value.charAt(1) != '=') {
throw new CheckedTestFrameworkException(value.substring(0, 1));
}
comparator = "!=";
comparison = (x, y) -> x.compareTo(y) != 0;
value = value.substring(2).trim();
break;
case '=': // Allowed syntax, equivalent to not using any symbol.
comparator = "=";
value = value.substring(1).trim();
// Fall through
default:
comparison = (x, y) -> x.compareTo(y) == 0;
value = value.trim();
break;
}
return new ParsedComparator<>(value, comparison, comparator);
}
}

View File

@ -0,0 +1,74 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.shared;
import java.util.ArrayList;
import java.util.List;
/**
* Utility class to report a {@link TestFormatException}.
*/
public class TestFormat {
private static final List<String> FAILURES = new ArrayList<>();
public static void check(boolean test, String failureMessage) {
if (!test) {
fail(failureMessage);
}
}
public static void checkNoThrow(boolean test, String failureMessage) {
if (!test) {
failNoThrow(failureMessage);
}
}
public static void fail(String failureMessage) {
FAILURES.add(failureMessage);
throw new TestFormatException(failureMessage);
}
public static void failNoThrow(String failureMessage) {
FAILURES.add(failureMessage);
}
public static void reportIfAnyFailures() {
if (FAILURES.isEmpty()) {
// No format violation detected.
return;
}
StringBuilder builder = new StringBuilder();
builder.append(System.lineSeparator()).append("One or more format violations have been detected:")
.append(System.lineSeparator()).append(System.lineSeparator());
builder.append("Violations (").append(FAILURES.size()).append(")").append(System.lineSeparator());
builder.append("-------------").append("-".repeat(String.valueOf(FAILURES.size()).length()))
.append(System.lineSeparator());
for (String failure : FAILURES) {
builder.append(" - ").append(failure).append(System.lineSeparator());
}
builder.append("/============/");
FAILURES.clear();
throw new TestFormatException(builder.toString());
}
}

View File

@ -0,0 +1,33 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.shared;
/**
* Exception that is thrown if a JTreg test violates the supported format by the test framework.
*/
public class TestFormatException extends RuntimeException {
public TestFormatException(String message) {
super(message);
}
}

View File

@ -0,0 +1,38 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.shared;
/**
* Exception that is thrown if there is an internal error in the framework. This is most likely an indicator of a bug
* in the framework.
*/
public class TestFrameworkException extends RuntimeException {
public TestFrameworkException(String message) {
super("Internal Test Framework exception - please file a bug:" + System.lineSeparator() + message);
}
public TestFrameworkException(String message, Throwable e) {
super("Internal Test Framework exception - please file a bug:" + System.lineSeparator() + message, e);
}
}

View File

@ -0,0 +1,181 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.shared;
import compiler.lib.ir_framework.TestFramework;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;
/**
* Dedicated socket to send data from the flag and test VM back to the driver VM.
*/
public class TestFrameworkSocket implements AutoCloseable {
public static final String STDOUT_PREFIX = "[STDOUT]";
// Static fields used for test VM only.
private static final String SERVER_PORT_PROPERTY = "ir.framework.server.port";
private static final int SERVER_PORT = Integer.getInteger(SERVER_PORT_PROPERTY, -1);
private static final boolean REPRODUCE = Boolean.getBoolean("Reproduce");
private static final String HOSTNAME = null;
private static Socket clientSocket = null;
private static PrintWriter clientWriter = null;
private final String serverPortPropertyFlag;
private FutureTask<String> socketTask;
private final ServerSocket serverSocket;
public TestFrameworkSocket() {
try {
serverSocket = new ServerSocket(0);
} catch (IOException e) {
throw new TestFrameworkException("Failed to create TestFramework server socket", e);
}
int port = serverSocket.getLocalPort();
if (TestFramework.VERBOSE) {
System.out.println("TestFramework server socket uses port " + port);
}
serverPortPropertyFlag = "-D" + SERVER_PORT_PROPERTY + "=" + port;
start();
}
public String getPortPropertyFlag() {
return serverPortPropertyFlag;
}
private void start() {
socketTask = initSocketTask();
Thread socketThread = new Thread(socketTask);
socketThread.start();
}
/**
* Waits for a client (created by flag or test VM) to connect. Return the messages received from the client.
*/
private FutureTask<String> initSocketTask() {
return new FutureTask<>(() -> {
try (Socket clientSocket = serverSocket.accept();
BufferedReader in = new BufferedReader(new InputStreamReader(clientSocket.getInputStream()))
) {
StringBuilder builder = new StringBuilder();
String next;
while ((next = in.readLine()) != null) {
builder.append(next).append(System.lineSeparator());
}
return builder.toString();
} catch (IOException e) {
throw new TestFrameworkException("Server socket error", e);
}
});
}
@Override
public void close() {
try {
serverSocket.close();
} catch (IOException e) {
throw new TestFrameworkException("Could not close socket", e);
}
}
/**
* Only called by test VM to write to server socket.
*/
public static void write(String msg, String type) {
write(msg, type, false);
}
/**
* Only called by test VM to write to server socket.
*/
public static void write(String msg, String type, boolean stdout) {
if (REPRODUCE) {
System.out.println("Debugging Test VM: Skip writing due to -DReproduce");
return;
}
TestFramework.check(SERVER_PORT != -1, "Server port was not set correctly for flag and/or test VM "
+ "or method not called from flag or test VM");
try {
// Keep the client socket open until the test VM terminates (calls closeClientSocket before exiting main()).
if (clientSocket == null) {
clientSocket = new Socket(HOSTNAME, SERVER_PORT);
clientWriter = new PrintWriter(clientSocket.getOutputStream(), true);
}
if (stdout) {
msg = STDOUT_PREFIX + msg;
}
clientWriter.println(msg);
} catch (Exception e) {
// When the test VM is directly run, we should ignore all messages that would normally be sent to the
// driver VM.
String failMsg = System.lineSeparator() + System.lineSeparator() + """
###########################################################
Did you directly run the test VM (TestVM class)
to reproduce a bug?
=> Append the flag -DReproduce=true and try again!
###########################################################
""";
throw new TestRunException(failMsg, e);
}
if (TestFramework.VERBOSE) {
System.out.println("Written " + type + " to socket:");
System.out.println(msg);
}
}
/**
* Closes (and flushes) the printer to the socket and the socket itself. Is called as last thing before exiting
* the main() method of the flag and the test VM.
*/
public static void closeClientSocket() {
if (clientSocket != null) {
try {
clientWriter.close();
clientSocket.close();
} catch (IOException e) {
throw new RuntimeException("Could not close TestVM socket", e);
}
}
}
/**
* Get the socket output of the flag VM.
*/
public String getOutput() {
try {
return socketTask.get();
} catch (ExecutionException e) {
// Thrown when socket task was not finished, yet (i.e. no client sent data) but socket was already closed.
return "";
} catch (Exception e) {
throw new TestFrameworkException("Could not read from socket task", e);
}
}
}

View File

@ -0,0 +1,35 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.shared;
/**
* Utility class to report a {@link TestRunException}.
*/
public class TestRun {
public static void check(boolean test, String failureMessage) {
if (!test) {
throw new TestRunException(failureMessage);
}
}
}

View File

@ -0,0 +1,38 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.shared;
/**
* Exception that is thrown if the JTreg test throws an exception during the execution of individual tests of the
* test class.
*/
public class TestRunException extends RuntimeException {
public TestRunException(String message) {
super(message);
}
public TestRunException(String message, Exception e) {
super(message, e);
}
}

View File

@ -0,0 +1,221 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.test;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.shared.TestRun;
import compiler.lib.ir_framework.shared.TestRunException;
import sun.hotspot.WhiteBox;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
/**
* Abstract super class for base, checked and custom run tests.
*/
abstract class AbstractTest {
protected static final WhiteBox WHITE_BOX = WhiteBox.getWhiteBox();
protected static final int TEST_COMPILATION_TIMEOUT = Integer.parseInt(System.getProperty("TestCompilationTimeout", "10000"));
protected static final int WAIT_FOR_COMPILATION_TIMEOUT = Integer.parseInt(System.getProperty("WaitForCompilationTimeout", "10000"));
protected static final boolean VERIFY_OOPS = (Boolean)WHITE_BOX.getVMFlag("VerifyOops");
protected final int warmupIterations;
protected final boolean skip;
AbstractTest(int warmupIterations, boolean skip) {
this.warmupIterations = warmupIterations;
this.skip = skip;
}
protected boolean shouldCompile(DeclaredTest test) {
return test.getCompLevel() != CompLevel.SKIP;
}
abstract String getName();
/**
* Should test be executed?
*/
public boolean isSkipped() {
return skip;
}
/**
* See {@link CompLevel#WAIT_FOR_COMPILATION}.
*/
protected static boolean isWaitForCompilation(DeclaredTest test) {
return test.getCompLevel() == CompLevel.WAIT_FOR_COMPILATION;
}
protected static Object createInvocationTarget(Method method) {
Class<?> clazz = method.getDeclaringClass();
Object invocationTarget;
if (Modifier.isStatic(method.getModifiers())) {
invocationTarget = null;
} else {
try {
Constructor<?> constructor = clazz.getDeclaredConstructor();
constructor.setAccessible(true);
invocationTarget = constructor.newInstance();
} catch (Exception e) {
throw new TestRunException("Could not create instance of " + clazz
+ ". Make sure there is a constructor without arguments.", e);
}
}
return invocationTarget;
}
/**
* Run the associated test.
*/
public void run() {
if (skip) {
return;
}
onStart();
for (int i = 0; i < warmupIterations; i++) {
invokeTest();
}
onWarmupFinished();
compileTest();
// Always run the test as a last step of the test execution.
invokeTest();
}
protected void onStart() {
// Do nothing by default.
}
abstract protected void invokeTest();
abstract protected void onWarmupFinished();
abstract protected void compileTest();
protected void compileMethod(DeclaredTest test) {
final Method testMethod = test.getTestMethod();
TestRun.check(WHITE_BOX.isMethodCompilable(testMethod, test.getCompLevel().getValue(), false),
"Method " + testMethod + " not compilable at level " + test.getCompLevel()
+ ". Did you use compileonly without including all @Test methods?");
TestRun.check(WHITE_BOX.isMethodCompilable(testMethod),
"Method " + testMethod + " not compilable at level " + test.getCompLevel()
+ ". Did you use compileonly without including all @Test methods?");
if (TestFramework.VERBOSE) {
System.out.println("Compile method " + testMethod + " after warm-up...");
}
final boolean maybeCodeBufferOverflow = (TestVM.TEST_C1 && VERIFY_OOPS);
final long started = System.currentTimeMillis();
long elapsed = 0;
int lastCompilationLevel = -10;
enqueueMethodForCompilation(test);
do {
if (!WHITE_BOX.isMethodQueuedForCompilation(testMethod)) {
if (elapsed > 0) {
if (TestVM.VERBOSE) {
System.out.println(testMethod + " is not in queue anymore due to compiling it simultaneously on " +
"a different level. Enqueue again.");
}
enqueueMethodForCompilation(test);
}
}
if (maybeCodeBufferOverflow && elapsed > 1000 && !WHITE_BOX.isMethodCompiled(testMethod, false)) {
// Let's disable VerifyOops temporarily and retry.
WHITE_BOX.setBooleanVMFlag("VerifyOops", false);
WHITE_BOX.clearMethodState(testMethod);
enqueueMethodForCompilation(test);
WHITE_BOX.setBooleanVMFlag("VerifyOops", true);
}
lastCompilationLevel = WHITE_BOX.getMethodCompilationLevel(testMethod, false);
if (lastCompilationLevel == test.getCompLevel().getValue()) {
break;
}
elapsed = System.currentTimeMillis() - started;
} while (elapsed < TEST_COMPILATION_TIMEOUT);
TestRun.check(elapsed < TEST_COMPILATION_TIMEOUT,
"Could not compile " + testMethod + " at level " + test.getCompLevel() + " after "
+ TEST_COMPILATION_TIMEOUT/1000 + "s. Last compilation level: " + lastCompilationLevel);
checkCompilationLevel(test);
}
private void enqueueMethodForCompilation(DeclaredTest test) {
TestVM.enqueueForCompilation(test.getTestMethod(), test.getCompLevel());
}
protected void checkCompilationLevel(DeclaredTest test) {
CompLevel level = CompLevel.forValue(WHITE_BOX.getMethodCompilationLevel(test.getTestMethod()));
TestRun.check(level == test.getCompLevel(), "Compilation level should be " + test.getCompLevel().name()
+ " (requested) but was " + level.name() + " for " + test.getTestMethod());
}
final protected void waitForCompilation(DeclaredTest test) {
final Method testMethod = test.getTestMethod();
final boolean maybeCodeBufferOverflow = (TestVM.TEST_C1 && VERIFY_OOPS);
final long started = System.currentTimeMillis();
boolean stateCleared = false;
long elapsed;
do {
elapsed = System.currentTimeMillis() - started;
int level = WHITE_BOX.getMethodCompilationLevel(testMethod);
if (maybeCodeBufferOverflow && elapsed > 5000
&& (!WHITE_BOX.isMethodCompiled(testMethod, false) || level != test.getCompLevel().getValue())) {
retryDisabledVerifyOops(testMethod, stateCleared);
stateCleared = true;
} else {
invokeTest();
}
boolean isCompiled = WHITE_BOX.isMethodCompiled(testMethod, false);
if (TestVM.VERBOSE) {
System.out.println("Is " + testMethod + " compiled? " + isCompiled);
}
if (isCompiled || TestVM.XCOMP || TestVM.EXCLUDE_RANDOM) {
// Don't wait for compilation if -Xcomp is enabled or if we are randomly excluding methods from compilation.
return;
}
} while (elapsed < WAIT_FOR_COMPILATION_TIMEOUT);
throw new TestRunException(testMethod + " not compiled after waiting for "
+ WAIT_FOR_COMPILATION_TIMEOUT/1000 + " s");
}
/**
* If it takes too long, try to disable Verify Oops.
*/
private void retryDisabledVerifyOops(Method testMethod, boolean stateCleared) {
System.out.println("Temporarily disabling VerifyOops");
try {
WHITE_BOX.setBooleanVMFlag("VerifyOops", false);
if (!stateCleared) {
WHITE_BOX.clearMethodState(testMethod);
}
invokeTest();
} finally {
WHITE_BOX.setBooleanVMFlag("VerifyOops", true);
System.out.println("Re-enabled VerifyOops");
}
}
}

View File

@ -0,0 +1,333 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.test;
import compiler.lib.ir_framework.Argument;
import compiler.lib.ir_framework.Arguments;
import compiler.lib.ir_framework.shared.TestFrameworkException;
import compiler.lib.ir_framework.shared.TestFormat;
import compiler.lib.ir_framework.shared.TestFormatException;
import jdk.test.lib.Utils;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.lang.reflect.Parameter;
import java.util.Random;
/**
* This class represents an argument value specified by {@link Argument} in {@link Arguments}.
*/
class ArgumentValue {
private static final Random RANDOM = Utils.getRandomInstance();
private final Object argumentValue;
private final boolean isRandomEach;
private final boolean isFixedRandom;
private final Class<?> randomClass;
ArgumentValue() {
this.argumentValue = null;
this.isRandomEach = false;
this.randomClass = null;
this.isFixedRandom = false;
}
private ArgumentValue(Object argumentValue, boolean isFixedRandom) {
this.argumentValue = argumentValue;
this.isRandomEach = false;
this.randomClass = null;
this.isFixedRandom = isFixedRandom;
}
private ArgumentValue(Object argumentValue, Class<?> randomClass) {
this.argumentValue = argumentValue;
this.isRandomEach = true;
this.randomClass = randomClass;
this.isFixedRandom = false;
}
/**
* Return all arguments for the @Arguments annotation.
*
* @param m The @Test method.
* @return Returns an array with Argument objects for each specified argument in the @Arguments annotation of m.
* Returns null if method has no @Arguments annotation.
*/
public static ArgumentValue[] getArguments(Method m) {
Arguments argumentsAnno = m.getAnnotation(Arguments.class);
if (argumentsAnno == null) {
return null;
}
Argument[] values = argumentsAnno.value();
ArgumentValue[] arguments = new ArgumentValue[values.length];
Class<?>[] declaredParameters = m.getParameterTypes();
Parameter[] declaredParameterObjects = m.getParameters();
try {
TestFormat.check(values.length == declaredParameters.length,
"Number of argument values provided in @Arguments does not match the number " +
"of actual arguments in " + m);
for (int i = 0; i < values.length; i++) {
Argument specifiedArg = values[i];
Class<?> parameter = declaredParameters[i];
Parameter parameterObj = declaredParameterObjects[i];
try {
switch (specifiedArg) {
case DEFAULT -> {
try {
arguments[i] = createDefault(parameter);
} catch (NoSuchMethodException e) {
TestFormat.fail("Cannot create new default instance of " + parameter
+ " for " + m + " due to missing default constructor");
} catch (Exception e) {
TestFormat.fail("Cannot create new default instance of " + parameter
+ " for " + m + ": " + e.getCause());
}
}
case NUMBER_42 -> {
TestFormat.check(isNumber(parameter), "Provided invalid NUMBER_42 argument " +
"for non-number " + parameterObj + " for " + m);
arguments[i] = create((byte) 42);
}
case NUMBER_MINUS_42 -> {
TestFormat.check(isNumber(parameter), "Provided invalid NUMBER_MINUS_42 argument " +
"for non-number " + parameterObj + " for " + m);
arguments[i] = create((byte) -42);
}
case MIN -> {
TestFormat.check(isNumber(parameter) || isChar(parameter),
"Provided invalid MIN argument for non-number "
+ parameterObj + " for " + m);
arguments[i] = createMin(parameter);
}
case MAX -> {
TestFormat.check(isNumber(parameter) || isChar(parameter),
"Provided invalid MAX argument for non-number "
+ parameterObj + " for " + m);
arguments[i] = createMax(parameter);
}
case FALSE -> {
TestFormat.check(ArgumentValue.isBoolean(parameter),
"Provided invalid FALSE argument for non-boolean "
+ parameterObj + " for " + m);
arguments[i] = create(false);
}
case TRUE -> {
TestFormat.check(ArgumentValue.isBoolean(parameter),
"Provided invalid TRUE argument for non-boolean "
+ parameterObj + " for " + m);
arguments[i] = create(true);
}
case BOOLEAN_TOGGLE_FIRST_FALSE -> {
TestFormat.check(isBoolean(parameter),
"Provided invalid BOOLEAN_TOGGLE_FIRST_FALSE argument for non-boolean "
+ parameterObj + " for " + m);
arguments[i] = BooleanToggleValue.create(false);
}
case BOOLEAN_TOGGLE_FIRST_TRUE -> {
TestFormat.check(ArgumentValue.isBoolean(parameter),
"Provided invalid BOOLEAN_TOGGLE_FIRST_TRUE argument for non-boolean "
+ parameterObj + " for " + m);
arguments[i] = BooleanToggleValue.create(true);
}
case RANDOM_ONCE -> {
TestFormat.check(isPrimitiveType(parameter),
"Provided invalid RANDOM_ONCE argument for non-primitive type "
+ parameterObj + " for " + m);
arguments[i] = createRandom(parameter);
}
case RANDOM_EACH -> {
TestFormat.check(isPrimitiveType(parameter),
"Provided invalid RANDOM_EACH argument for non-primitive type "
+ parameterObj + " for " + m);
arguments[i] = createRandomEach(parameter);
}
}
} catch (TestFormatException e) {
// Catch and continue to check arguments.
}
}
} catch (TestFormatException e) {
// Catch and return empty array to check for additional failures.
return new ArgumentValue[0];
}
return arguments;
}
private static ArgumentValue create(Object argumentValue) {
return new ArgumentValue(argumentValue, false);
}
private static ArgumentValue createDefault(Class<?> c) throws Exception {
if (ArgumentValue.isNumber(c)) {
return ArgumentValue.create((byte)0);
} else if (ArgumentValue.isChar(c)) {
return ArgumentValue.create('\u0000');
} else if (ArgumentValue.isBoolean(c)) {
return ArgumentValue.create(false);
} else {
// Object
Constructor<?> constructor = c.getDeclaredConstructor();
constructor.setAccessible(true); // Make sure to have access to private default constructor
return ArgumentValue.create(constructor.newInstance());
}
}
private static ArgumentValue createMin(Class<?> c) {
Object argument;
if (c.equals(byte.class)) {
argument = Byte.MIN_VALUE;
} else if (isChar(c)) {
argument = Character.MIN_VALUE;
} else if (c.equals(short.class)) {
argument = Short.MIN_VALUE;
} else if (c.equals(int.class)) {
argument = Integer.MIN_VALUE;
} else if (c.equals(long.class)) {
argument = Long.MIN_VALUE;
} else if (c.equals(float.class)) {
argument = Float.MIN_VALUE;
} else if (c.equals(double.class)) {
argument = Double.MIN_VALUE;
} else {
throw new TestFrameworkException("Invalid class passed to createMin()");
}
return new ArgumentValue(argument, false);
}
private static ArgumentValue createMax(Class<?> c) {
Object argument;
if (c.equals(byte.class)) {
argument = Byte.MAX_VALUE;
} else if (isChar(c)) {
argument = Character.MAX_VALUE;
} else if (c.equals(short.class)) {
argument = Short.MAX_VALUE;
} else if (c.equals(int.class)) {
argument = Integer.MAX_VALUE;
} else if (c.equals(long.class)) {
argument = Long.MAX_VALUE;
} else if (c.equals(float.class)) {
argument = Float.MAX_VALUE;
} else if (c.equals(double.class)) {
argument = Double.MAX_VALUE;
} else {
throw new TestFrameworkException("Invalid class passed to createMax()");
}
return new ArgumentValue(argument, false);
}
private static ArgumentValue createRandom(Class<?> c) {
return new ArgumentValue(getRandom(c), true);
}
private static ArgumentValue createRandomEach(Class<?> c) {
return new ArgumentValue(null, c);
}
public boolean isFixedRandom() {
return isFixedRandom;
}
public Object getArgument() {
if (isRandomEach) {
return getRandom(randomClass);
} else {
return argumentValue;
}
}
private static boolean isPrimitiveType(Class<?> c) {
return isNumber(c) || isBoolean(c) || isChar(c);
}
private static boolean isBoolean(Class<?> c) {
return c.equals(boolean.class);
}
private static boolean isChar(Class<?> c) {
return c.equals(char.class);
}
private static boolean isNumber(Class<?> c) {
return isIntNumber(c) || isFloatNumber(c);
}
private static boolean isIntNumber(Class<?> c) {
return c.equals(byte.class)
|| c.equals(short.class)
|| c.equals(int.class)
|| c.equals(long.class);
}
private static boolean isFloatNumber(Class<?> c) {
return c.equals(float.class) || c.equals(double.class);
}
private static Object getRandom(Class<?> c) {
if (isBoolean(c)) {
return RANDOM.nextBoolean();
} else if (c.equals(byte.class)) {
return (byte) RANDOM.nextInt(256);
} else if (isChar(c)) {
return (char) RANDOM.nextInt(65536);
} else if (c.equals(short.class)) {
return (short) RANDOM.nextInt(65536);
} else if (c.equals(int.class)) {
return RANDOM.nextInt();
} else if (c.equals(long.class)) {
return RANDOM.nextLong();
} else if (c.equals(float.class)) {
// Get float between -10000 and 10000.
return RANDOM.nextFloat() * 20000 - 10000;
} else if (c.equals(double.class)) {
// Get double between -10000 and 10000.
return RANDOM.nextDouble() * 20000 - 10000;
} else {
TestFormat.fail("Cannot generate random value for non-primitive type");
return null;
}
}
}
/**
* Special class to handle boolean toggle argument values.
*/
class BooleanToggleValue extends ArgumentValue {
private boolean previousBoolean;
BooleanToggleValue(boolean firstBoolean) {
this.previousBoolean = !firstBoolean;
}
@Override
public Object getArgument() {
previousBoolean = !previousBoolean;
return previousBoolean;
}
static BooleanToggleValue create(boolean firstBoolean) {
return new BooleanToggleValue(firstBoolean);
}
}

View File

@ -0,0 +1,105 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.test;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.shared.TestRunException;
import java.lang.reflect.Method;
/**
* A base test only consists of a single @Test method. See {@link Test} for more details and its precise definition.
*/
class BaseTest extends AbstractTest {
private final DeclaredTest test;
protected final Method testMethod;
protected final TestInfo testInfo;
protected final Object invocationTarget;
private final boolean shouldCompile;
private final boolean waitForCompilation;
public BaseTest(DeclaredTest test, boolean skip) {
super(test.getWarmupIterations(), skip);
this.test = test;
this.testMethod = test.getTestMethod();
this.testInfo = new TestInfo(testMethod, test.getCompLevel());
this.invocationTarget = createInvocationTarget(testMethod);
this.shouldCompile = shouldCompile(test);
this.waitForCompilation = isWaitForCompilation(test);
}
@Override
public String toString() {
return "Base Test: @Test " + testMethod.getName();
}
@Override
public String getName() {
return testMethod.getName();
}
@Override
protected void onStart() {
test.printFixedRandomArguments();
}
@Override
public void onWarmupFinished() {
testInfo.setWarmUpFinished();
}
@Override
protected void invokeTest() {
verify(invokeTestMethod());
}
private Object invokeTestMethod() {
try {
if (test.hasArguments()) {
return testMethod.invoke(invocationTarget, test.getArguments());
} else {
return testMethod.invoke(invocationTarget);
}
} catch (Exception e) {
throw new TestRunException("There was an error while invoking @Test method " + testMethod
+ ". Used arguments: " + test.getArgumentsString(), e);
}
}
@Override
protected void compileTest() {
if (shouldCompile) {
if (waitForCompilation) {
waitForCompilation(test);
} else {
compileMethod(test);
}
}
}
/**
* Verify the result
*/
public void verify(Object result) { /* no verification in BaseTests */ }
}

View File

@ -0,0 +1,93 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.test;
import compiler.lib.ir_framework.Check;
import compiler.lib.ir_framework.CheckAt;
import compiler.lib.ir_framework.shared.TestRunException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
/**
* A checked test is an extension of a base test with additional verification done in a @Check method.
* See {@link Check} for more details and its precise definition.
*/
class CheckedTest extends BaseTest {
private final Method checkMethod;
private final CheckAt checkAt;
private final Parameter parameter;
private final Object checkInvocationTarget;
enum Parameter {
NONE, RETURN_ONLY, TEST_INFO_ONLY, BOTH
}
public CheckedTest(DeclaredTest test, Method checkMethod, Check checkSpecification, Parameter parameter, boolean excludedByUser) {
super(test, excludedByUser);
// Make sure we can also call non-public or public methods in package private classes
checkMethod.setAccessible(true);
this.checkMethod = checkMethod;
this.checkAt = checkSpecification.when();
this.parameter = parameter;
// Use the same invocation target
if (Modifier.isStatic(checkMethod.getModifiers())) {
this.checkInvocationTarget = null;
} else {
// Use the same invocation target as the test method if check method is non-static.
this.checkInvocationTarget = this.invocationTarget != null ? this.invocationTarget : createInvocationTarget(checkMethod);
}
}
@Override
public String toString() {
return "Checked Test: @Check " + checkMethod.getName() + " - @Test: " + testMethod.getName();
}
@Override
public String getName() {
return checkMethod.getName();
}
@Override
public void verify(Object result) {
boolean shouldVerify = false;
switch (checkAt) {
case EACH_INVOCATION -> shouldVerify = true;
case COMPILED -> shouldVerify = !testInfo.isWarmUp();
}
if (shouldVerify) {
try {
switch (parameter) {
case NONE -> checkMethod.invoke(checkInvocationTarget);
case RETURN_ONLY -> checkMethod.invoke(checkInvocationTarget, result);
case TEST_INFO_ONLY -> checkMethod.invoke(checkInvocationTarget, testInfo);
case BOTH -> checkMethod.invoke(checkInvocationTarget, result, testInfo);
}
} catch (Exception e) {
throw new TestRunException("There was an error while invoking @Check method " + checkMethod, e);
}
}
}
}

View File

@ -0,0 +1,181 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.test;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.shared.TestFormat;
import compiler.lib.ir_framework.shared.TestFrameworkException;
import compiler.lib.ir_framework.shared.TestRunException;
import sun.hotspot.WhiteBox;
import java.lang.reflect.Method;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;
/**
* A custom run test allows the user to have full control over how the @Test method is invoked by specifying
* a dedicated @Run method. See {@link Run} for more details and its precise definition.
*/
class CustomRunTest extends AbstractTest {
private final Method runMethod;
private final RunMode mode;
private final Object runInvocationTarget;
private final List<DeclaredTest> tests;
private final RunInfo runInfo;
public CustomRunTest(Method runMethod, Warmup warmUpAnno, Run runSpecification, List<DeclaredTest> tests, boolean skip) {
// Make sure we can also call non-public or public methods in package private classes
super(warmUpAnno != null ? warmUpAnno.value() : TestVM.WARMUP_ITERATIONS, skip);
TestFormat.checkNoThrow(warmupIterations >= 0, "Cannot have negative value for @Warmup at " + runMethod);
runMethod.setAccessible(true);
this.runMethod = runMethod;
this.runInvocationTarget = createInvocationTarget(runMethod);
this.mode = runSpecification.mode();
this.tests = tests;
this.runInfo = new RunInfo(tests);
}
@Override
public String toString() {
String s = "Custom Run Test: @Run: " + runMethod.getName() + " - @Test";
if (tests.size() == 1) {
s += ": " + tests.get(0).getTestMethod().getName();
} else {
s += "s: {" + tests.stream().map(t -> t.getTestMethod().getName())
.collect(Collectors.joining(",")) + "}";
}
return s;
}
@Override
String getName() {
return runMethod.getName();
}
@Override
public void run() {
if (skip) {
return;
}
switch (mode) {
case STANDALONE -> {
runInfo.setWarmUpFinished();
invokeTest();
}// Invoke once but do not apply anything else.
case NORMAL -> super.run();
}
}
@Override
public void onWarmupFinished() {
runInfo.setWarmUpFinished();
}
@Override
protected void compileTest() {
if (tests.size() == 1) {
compileSingleTest();
} else {
compileMultipleTests();
}
}
private void compileSingleTest() {
DeclaredTest test = tests.get(0);
if (shouldCompile(test)) {
if (isWaitForCompilation(test)) {
waitForCompilation(test);
} else {
compileMethod(test);
}
}
}
private void compileMultipleTests() {
boolean anyWaitForCompilation = false;
boolean anyCompileMethod = false;
ExecutorService executor = Executors.newFixedThreadPool(tests.size());
for (DeclaredTest test : tests) {
if (shouldCompile(test)) {
if (isWaitForCompilation(test)) {
anyWaitForCompilation = true;
executor.execute(() -> waitForCompilation(test));
} else {
anyCompileMethod = true;
executor.execute(() -> compileMethod(test));
}
}
}
executor.shutdown();
int timeout;
if (anyCompileMethod && anyWaitForCompilation) {
timeout = Math.max(WAIT_FOR_COMPILATION_TIMEOUT, TEST_COMPILATION_TIMEOUT) + 5000;
} else if (anyWaitForCompilation) {
timeout = WAIT_FOR_COMPILATION_TIMEOUT + 5000;
} else {
timeout = TEST_COMPILATION_TIMEOUT + 5000;
}
try {
executor.awaitTermination(timeout, TimeUnit.MILLISECONDS);
} catch (InterruptedException e) {
throw new TestRunException("Some compilations did not complete after " + timeout
+ "ms for @Run method " + runMethod);
}
}
/**
* Do not directly run the test but rather the run method that is responsible for invoking the actual test.
*/
@Override
protected void invokeTest() {
try {
if (runMethod.getParameterCount() == 1) {
runMethod.invoke(runInvocationTarget, runInfo);
} else {
runMethod.invoke(runInvocationTarget);
}
} catch (Exception e) {
throw new TestRunException("There was an error while invoking @Run method " + runMethod, e);
}
}
@Override
protected void checkCompilationLevel(DeclaredTest test) {
CompLevel level = CompLevel.forValue(WhiteBox.getWhiteBox().getMethodCompilationLevel(test.getTestMethod()));
if (level != test.getCompLevel()) {
String message = "Compilation level should be " + test.getCompLevel().name() + " (requested) but was "
+ level.name() + " for " + test.getTestMethod() + ".";
switch (mode) {
case STANDALONE -> throw new TestFrameworkException("Should not be called for STANDALONE method " + runMethod);
case NORMAL -> message = message + System.lineSeparator() + "Check your @Run method " + runMethod
+ " to ensure that " + test.getTestMethod()
+ " is called at least once in each iteration.";
}
throw new TestRunException(message);
}
}
}

View File

@ -0,0 +1,125 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.test;
import compiler.lib.ir_framework.CompLevel;
import compiler.lib.ir_framework.shared.TestRunException;
import java.lang.reflect.Method;
import java.util.Arrays;
/**
* This class represents a @Test method.
*/
public class DeclaredTest {
private final Method testMethod;
private final ArgumentValue[] arguments;
private final int warmupIterations;
private final CompLevel compLevel;
private Method attachedMethod;
public DeclaredTest(Method testMethod, ArgumentValue[] arguments, CompLevel compLevel, int warmupIterations) {
// Make sure we can also call non-public or public methods in package private classes
testMethod.setAccessible(true);
this.testMethod = testMethod;
this.compLevel = compLevel;
this.arguments = arguments;
this.warmupIterations = warmupIterations;
this.attachedMethod = null;
}
public Method getTestMethod() {
return testMethod;
}
public CompLevel getCompLevel() {
return compLevel;
}
public int getWarmupIterations() {
return warmupIterations;
}
public boolean hasArguments() {
return arguments != null;
}
public Object[] getArguments() {
return Arrays.stream(arguments).map(ArgumentValue::getArgument).toArray();
}
public void setAttachedMethod(Method m) {
attachedMethod = m;
}
public Method getAttachedMethod() {
return attachedMethod;
}
public void printFixedRandomArguments() {
if (hasArguments()) {
boolean hasRandomArgs = false;
StringBuilder builder = new StringBuilder("Fixed random arguments for method ").append(testMethod).append(": ");
for (int i = 0; i < arguments.length; i++) {
ArgumentValue argument = arguments[i];
if (argument.isFixedRandom()) {
hasRandomArgs = true;
Object argumentVal = argument.getArgument();
builder.append("arg ").append(i).append(": ").append(argumentVal.toString());
if (argumentVal instanceof Character) {
builder.append(" (").append((int)(Character)argumentVal).append(")");
}
builder.append(", ");
}
}
if (hasRandomArgs) {
// Drop the last comma and space.
builder.setLength(builder.length() - 2);
System.out.println(builder.toString());
}
}
}
public String getArgumentsString() {
if (hasArguments()) {
StringBuilder builder = new StringBuilder();
for (int i = 0; i < arguments.length; i++) {
builder.append("arg ").append(i).append(": ").append(arguments[i].getArgument()).append(", ");
}
builder.setLength(builder.length() - 2);
return builder.toString();
} else {
return "<void>";
}
}
public Object invoke(Object obj, Object... args) {
try {
return testMethod.invoke(obj, args);
} catch (Exception e) {
throw new TestRunException("There was an error while invoking @Test method " + testMethod, e);
}
}
}

View File

@ -0,0 +1,290 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.test;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.shared.*;
import sun.hotspot.WhiteBox;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.function.Function;
/**
* Prints an encoding to the dedicated test framework socket whether @IR rules of @Test methods should be applied or not.
* This is done during the execution of the test VM by checking the active VM flags. This encoding is eventually parsed
* and checked by the IRMatcher class in the driver VM after the termination of the test VM.
*/
public class IREncodingPrinter {
public static final String START = "##### IRMatchRulesEncoding - used by TestFramework #####";
public static final String END = "----- END -----";
public static final int NO_RULE_APPLIED = -1;
private static final WhiteBox WHITE_BOX = WhiteBox.getWhiteBox();
private static final List<Function<String, Object>> LONG_GETTERS = Arrays.asList(
WHITE_BOX::getIntVMFlag, WHITE_BOX::getUintVMFlag, WHITE_BOX::getIntxVMFlag,
WHITE_BOX::getUintxVMFlag, WHITE_BOX::getUint64VMFlag, WHITE_BOX::getSizeTVMFlag);
private final StringBuilder output = new StringBuilder();
private Method method;
private int ruleIndex;
public IREncodingPrinter() {
output.append(START).append(System.lineSeparator());
output.append("<method>,{comma separated applied @IR rule ids}").append(System.lineSeparator());
}
/**
* Emits "<method>,{ids}" where {ids} is either:
* - indices of all @IR rules that should be applied, separated by a comma
* - "-1" if no @IR rule should not be applied
*/
public void emitRuleEncoding(Method m, boolean skipped) {
method = m;
int i = 0;
ArrayList<Integer> validRules = new ArrayList<>();
IR[] irAnnos = m.getAnnotationsByType(IR.class);
if (!skipped) {
for (IR irAnno : irAnnos) {
ruleIndex = i + 1;
try {
if (shouldApplyIrRule(irAnno)) {
validRules.add(i);
}
} catch (TestFormatException e) {
// Catch logged failure and continue to check other IR annotations.
}
i++;
}
}
if (irAnnos.length != 0) {
output.append(m.getName());
if (validRules.isEmpty()) {
output.append("," + NO_RULE_APPLIED);
} else {
for (i = 0; i < validRules.size(); i++) {
output.append(",").append(validRules.get(i));
}
}
output.append(System.lineSeparator());
}
}
private boolean shouldApplyIrRule(IR irAnno) {
checkIRAnnotations(irAnno);
if (irAnno.applyIf().length != 0) {
return hasAllRequiredFlags(irAnno.applyIf(), "applyIf");
}
if (irAnno.applyIfNot().length != 0) {
return hasNoRequiredFlags(irAnno.applyIfNot(), "applyIfNot");
}
if (irAnno.applyIfAnd().length != 0) {
return hasAllRequiredFlags(irAnno.applyIfAnd(), "applyIfAnd");
}
if (irAnno.applyIfOr().length != 0) {
return !hasNoRequiredFlags(irAnno.applyIfOr(), "applyIfOr");
}
// No conditions, always apply.
return true;
}
private void checkIRAnnotations(IR irAnno) {
TestFormat.checkNoThrow(irAnno.counts().length != 0 || irAnno.failOn().length != 0,
"Must specify either counts or failOn constraint" + failAt());
int applyRules = 0;
if (irAnno.applyIfAnd().length != 0) {
applyRules++;
TestFormat.checkNoThrow(irAnno.applyIfAnd().length > 2,
"Use applyIf or applyIfNot or at least 2 conditions for applyIfAnd" + failAt());
}
if (irAnno.applyIfOr().length != 0) {
applyRules++;
TestFormat.checkNoThrow(irAnno.applyIfOr().length > 2,
"Use applyIf or applyIfNot or at least 2 conditions for applyIfOr" + failAt());
}
if (irAnno.applyIf().length != 0) {
applyRules++;
TestFormat.checkNoThrow(irAnno.applyIf().length <= 2,
"Use applyIfAnd or applyIfOr or only 1 condition for applyIf" + failAt());
}
if (irAnno.applyIfNot().length != 0) {
applyRules++;
TestFormat.checkNoThrow(irAnno.applyIfNot().length <= 2,
"Use applyIfAnd or applyIfOr or only 1 condition for applyIfNot" + failAt());
}
TestFormat.checkNoThrow(applyRules <= 1,
"Can only specify one apply constraint " + failAt());
}
private boolean hasAllRequiredFlags(String[] andRules, String ruleType) {
boolean returnValue = true;
for (int i = 0; i < andRules.length; i++) {
String flag = andRules[i].trim();
i++;
TestFormat.check(i < andRules.length, "Missing value for flag " + flag + " in " + ruleType + failAt());
String value = andRules[i].trim();
if (!check(flag, value) && returnValue) {
// Rule will not be applied but keep processing the other flags to verify that they are sane.
returnValue = false;
}
}
return returnValue;
}
private boolean hasNoRequiredFlags(String[] orRules, String ruleType) {
boolean returnValue = true;
for (int i = 0; i < orRules.length; i++) {
String flag = orRules[i];
i++;
TestFormat.check(i < orRules.length, "Missing value for flag " + flag + " in " + ruleType + failAt());
String value = orRules[i];
if (check(flag, value) && returnValue) {
// Rule will not be applied but keep processing the other flags to verify that they are sane.
returnValue = false;
}
}
return returnValue;
}
private boolean check(String flag, String value) {
if (flag.isEmpty()) {
TestFormat.failNoThrow("Provided empty flag" + failAt());
return false;
}
if (value.isEmpty()) {
TestFormat.failNoThrow("Provided empty value for flag " + flag + failAt());
return false;
}
Object actualFlagValue = WHITE_BOX.getBooleanVMFlag(flag);
if (actualFlagValue != null) {
return checkBooleanFlag(flag, value, (Boolean) actualFlagValue);
}
actualFlagValue = LONG_GETTERS.stream().map(f -> f.apply(flag)).filter(Objects::nonNull).findAny().orElse(null);
if (actualFlagValue != null) {
return checkLongFlag(flag, value, (Long) actualFlagValue);
}
actualFlagValue = WHITE_BOX.getDoubleVMFlag(flag);
if (actualFlagValue != null) {
return checkDoubleFlag(flag, value, (Double) actualFlagValue);
}
actualFlagValue = WHITE_BOX.getStringVMFlag(flag);
if (actualFlagValue != null) {
return value.equals(actualFlagValue);
}
// This could be improved if the Whitebox offers a "isVMFlag" function. For now, just check if we can actually set
// a value for a string flag. If we find this value, it's a string flag. If null is returned, the flag is unknown.
WHITE_BOX.setStringVMFlag(flag, "test");
String stringFlagValue = WHITE_BOX.getStringVMFlag(flag);
if (stringFlagValue == null) {
TestFormat.failNoThrow("Could not find VM flag \"" + flag + "\"" + failAt());
return false;
}
TestFramework.check(stringFlagValue.equals("test"),
"Must find newly set flag value \"test\" but found " + failAt());
WHITE_BOX.setStringVMFlag(flag, null); // reset flag to NULL
return false;
}
private boolean checkBooleanFlag(String flag, String value, boolean actualFlagValue) {
boolean booleanValue = false;
if ("true".equalsIgnoreCase(value)) {
booleanValue = true;
} else if (!"false".equalsIgnoreCase(value)) {
TestFormat.failNoThrow("Invalid value \"" + value + "\" for boolean flag " + flag + failAt());
return false;
}
return booleanValue == actualFlagValue;
}
private boolean checkLongFlag(String flag, String value, long actualFlagValue) {
long longValue;
ParsedComparator<Long> parsedComparator;
try {
parsedComparator = ParsedComparator.parseComparator(value);
} catch (CheckedTestFrameworkException e) {
TestFormat.failNoThrow("Invalid comparator in \"" + value + "\" for integer based flag " + flag + failAt());
return false;
} catch (IndexOutOfBoundsException e) {
TestFormat.failNoThrow("Provided empty value for integer based flag " + flag + failAt());
return false;
}
try {
longValue = Long.parseLong(parsedComparator.getStrippedString());
} catch (NumberFormatException e) {
String comparator = parsedComparator.getComparator();
if (!comparator.isEmpty()) {
comparator = "after comparator \"" + parsedComparator.getComparator() + "\"";
}
TestFormat.failNoThrow("Invalid value \"" + parsedComparator.getStrippedString() + "\" "
+ comparator + " for integer based flag " + flag + failAt());
return false;
}
return parsedComparator.getPredicate().test(actualFlagValue, longValue);
}
private boolean checkDoubleFlag(String flag, String value, double actualFlagValue) {
double doubleValue;
ParsedComparator<Double> parsedComparator;
try {
parsedComparator = ParsedComparator.parseComparator(value);
} catch (CheckedTestFrameworkException e) {
TestFormat.failNoThrow("Invalid comparator in \"" + value + "\" for floating point based flag " + flag + failAt());
return false;
} catch (IndexOutOfBoundsException e) {
TestFormat.failNoThrow("Provided empty value for floating point based flag " + flag + failAt());
return false;
}
try {
doubleValue = Double.parseDouble(parsedComparator.getStrippedString());
} catch (NumberFormatException e) {
String comparator = parsedComparator.getComparator();
if (!comparator.isEmpty()) {
comparator = "after comparator \"" + parsedComparator.getComparator() + "\"";
}
TestFormat.failNoThrow("Invalid value \"" + parsedComparator.getStrippedString() + "\" "
+ comparator + " for floating point based flag " + flag + failAt());
return false;
}
return parsedComparator.getPredicate().test(actualFlagValue, doubleValue);
}
private String failAt() {
return " for @IR rule " + ruleIndex + " at " + method;
}
public void emit() {
output.append(END);
TestFrameworkSocket.write(output.toString(), "IR rule application encoding");
}
}

View File

@ -0,0 +1,955 @@
/*
* Copyright (c) 2021, 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.
*
* 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 compiler.lib.ir_framework.test;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.Compiler;
import compiler.lib.ir_framework.shared.*;
import jdk.test.lib.Platform;
import jdk.test.lib.Utils;
import sun.hotspot.WhiteBox;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.lang.annotation.Annotation;
import java.lang.reflect.*;
import java.util.*;
import java.util.stream.Collectors;
import java.util.stream.Stream;
/**
* This class' main method is called from {@link TestFramework} and represents the so-called "test VM". The class is
* the heart of the framework and is responsible for executing all the specified tests in the test class. It uses the
* Whitebox API and reflection to achieve this task.
*/
public class TestVM {
private static final WhiteBox WHITE_BOX;
static {
try {
WHITE_BOX = WhiteBox.getWhiteBox();
} catch (UnsatisfiedLinkError e) {
System.err.println(System.lineSeparator() + """
##########################################################
- Did you call a test-related interface method from
TestFramework in main() of your test? Make sure to
only call setup/run methods and no checks or
assertions from main() of your test!
- Are you rerunning the test VM (TestVM class)
directly after a JTreg run? Make sure to start it
from within JTwork/scratch and with the flag
-DReproduce=true!
##########################################################
""");
throw e;
}
}
/**
* The default number of warm-up iterations used to warm up a {@link Test} annotated test method.
* Use {@code -DWarmup=XY} to specify a different default value. An individual warm-up can also be
* set by specifying a {@link Warmup} iteration for a test.
*/
public static final int WARMUP_ITERATIONS = Integer.parseInt(System.getProperty("Warmup", "2000"));
private static final boolean TIERED_COMPILATION = (Boolean)WHITE_BOX.getVMFlag("TieredCompilation");
private static final CompLevel TIERED_COMPILATION_STOP_AT_LEVEL = CompLevel.forValue(((Long)WHITE_BOX.getVMFlag("TieredStopAtLevel")).intValue());
public static final boolean TEST_C1 = TIERED_COMPILATION && TIERED_COMPILATION_STOP_AT_LEVEL.getValue() < CompLevel.C2.getValue();
static final boolean XCOMP = Platform.isComp();
static final boolean VERBOSE = Boolean.getBoolean("Verbose");
private static final boolean PRINT_TIMES = Boolean.getBoolean("PrintTimes");
public static final boolean USE_COMPILER = WHITE_BOX.getBooleanVMFlag("UseCompiler");
static final boolean EXCLUDE_RANDOM = Boolean.getBoolean("ExcludeRandom");
private static final String TESTLIST = System.getProperty("Test", "");
private static final String EXCLUDELIST = System.getProperty("Exclude", "");
private static final boolean DUMP_REPLAY = Boolean.getBoolean("DumpReplay");
private static final boolean GC_AFTER = Boolean.getBoolean("GCAfter");
private static final boolean SHUFFLE_TESTS = Boolean.parseBoolean(System.getProperty("ShuffleTests", "true"));
// Use separate flag as VERIFY_IR could have been set by user but due to other flags it was disabled by flag VM.
private static final boolean PRINT_VALID_IR_RULES = Boolean.getBoolean("ShouldDoIRVerification");
protected static final long PER_METHOD_TRAP_LIMIT = (Long)WHITE_BOX.getVMFlag("PerMethodTrapLimit");
protected static final boolean PROFILE_INTERPRETER = (Boolean)WHITE_BOX.getVMFlag("ProfileInterpreter");
private static final boolean FLIP_C1_C2 = Boolean.getBoolean("FlipC1C2");
private static final boolean IGNORE_COMPILER_CONTROLS = Boolean.getBoolean("IgnoreCompilerControls");
private final HashMap<Method, DeclaredTest> declaredTests = new HashMap<>();
private final List<AbstractTest> allTests = new ArrayList<>();
private final HashMap<String, Method> testMethodMap = new HashMap<>();
private final List<String> excludeList;
private final List<String> testList;
private Set<Class<?>> helperClasses = null; // Helper classes that contain framework annotations to be processed.
private final IREncodingPrinter irMatchRulePrinter;
private final Class<?> testClass;
private final Map<Executable, CompLevel> forceCompileMap = new HashMap<>();
private TestVM(Class<?> testClass) {
TestRun.check(testClass != null, "Test class cannot be null");
this.testClass = testClass;
this.testList = createTestFilterList(TESTLIST, testClass);
this.excludeList = createTestFilterList(EXCLUDELIST, testClass);
if (PRINT_VALID_IR_RULES) {
irMatchRulePrinter = new IREncodingPrinter();
} else {
irMatchRulePrinter = null;
}
}
/**
* Parse "test1,test2,test3" into a list.
*/
private static List<String> createTestFilterList(String list, Class<?> testClass) {
List<String> filterList = null;
if (!list.isEmpty()) {
String classPrefix = testClass.getSimpleName() + ".";
filterList = new ArrayList<>(Arrays.asList(list.split(",")));
for (int i = filterList.size() - 1; i >= 0; i--) {
String test = filterList.get(i);
if (test.indexOf(".") > 0) {
if (test.startsWith(classPrefix)) {
test = test.substring(classPrefix.length());
filterList.set(i, test);
} else {
filterList.remove(i);
}
}
}
}
return filterList;
}
/**
* Main entry point of the test VM.
*/
public static void main(String[] args) {
try {
String testClassName = args[0];
System.out.println("TestVM main() called - about to run tests in class " + testClassName);
Class<?> testClass = getClassObject(testClassName, "test");
TestVM framework = new TestVM(testClass);
framework.addHelperClasses(args);
framework.start();
} finally {
TestFrameworkSocket.closeClientSocket();
}
}
protected static Class<?> getClassObject(String className, String classType) {
try {
return Class.forName(className);
} catch (Exception e) {
throw new TestRunException("Could not find " + classType + " class", e);
}
}
/**
* Set up all helper classes and verify they are specified correctly.
*/
private void addHelperClasses(String[] args) {
Class<?>[] helperClassesList = getHelperClasses(args);
if (helperClassesList != null) {
TestRun.check(Arrays.stream(helperClassesList).noneMatch(Objects::isNull), "A Helper class cannot be null");
this.helperClasses = new HashSet<>();
for (Class<?> helperClass : helperClassesList) {
if (Arrays.stream(testClass.getDeclaredClasses()).anyMatch(c -> c == helperClass)) {
// Nested class of test class is automatically treated as helper class
TestFormat.failNoThrow("Nested " + helperClass + " inside test " + testClass + " is implicitly"
+ " treated as helper class and does not need to be specified as such.");
continue;
}
TestRun.check(!this.helperClasses.contains(helperClass), "Cannot add the same class twice: " + helperClass);
this.helperClasses.add(helperClass);
}
}
}
private static Class<?>[] getHelperClasses(String[] args) {
if (args.length == 1) {
return null;
}
Class<?>[] helperClasses = new Class<?>[args.length - 1]; // First argument is test class
for (int i = 1; i < args.length; i++) {
String helperClassName = args[i];
helperClasses[i - 1] = getClassObject(helperClassName, "helper");
}
return helperClasses;
}
private void checkHelperClass(Class<?> clazz) {
checkAnnotationsInClass(clazz, "helper");
for (Class<?> c : clazz.getDeclaredClasses()) {
checkAnnotationsInClass(c, "nested (and helper)");
}
}
private void checkAnnotationsInClass(Class<?> c, String clazzType) {
Method[] methods = c.getDeclaredMethods();
for (Method m : methods) {
TestFormat.checkNoThrow(getAnnotation(m, Test.class) == null,
"Cannot use @Test annotation in " + clazzType + " " + c + " at " + m);
TestFormat.checkNoThrow(getAnnotation(m, Run.class) == null,
"Cannot use @Run annotation in " + clazzType + " " + c + " at " + m);
TestFormat.checkNoThrow(getAnnotation(m, Check.class) == null,
"Cannot use @Check annotation in " + clazzType + " " + c + " at " + m);
}
}
/**
* Only called by internal tests testing the framework itself. Accessed by reflection. Not exposed to normal users.
*/
private static void runTestsOnSameVM(Class<?> testClass) {
if (testClass == null) {
StackWalker walker = StackWalker.getInstance(StackWalker.Option.RETAIN_CLASS_REFERENCE);
testClass = walker.getCallerClass();
}
TestVM framework = new TestVM(testClass);
framework.start();
}
/**
* Once everything is initialized and set up, start collecting tests and executing them afterwards.
*/
private void start() {
setupTests();
checkForcedCompilationsCompleted();
runTests();
}
private void setupTests() {
for (Class<?> clazz : testClass.getDeclaredClasses()) {
checkAnnotationsInClass(clazz, "inner");
}
if (DUMP_REPLAY) {
addReplay();
}
// Make sure to first setup test methods and make them non-inlineable and only then process compile commands.
setupDeclaredTests();
processControlAnnotations(testClass);
processHelperClasses();
setupCheckedAndCustomRunTests();
// All remaining tests are simple base tests without check or specific way to run them.
addBaseTests();
if (PRINT_VALID_IR_RULES) {
irMatchRulePrinter.emit();
}
TestFormat.reportIfAnyFailures();
declaredTests.clear();
testMethodMap.clear();
}
private void addBaseTests() {
declaredTests.forEach((m, test) -> {
if (test.getAttachedMethod() == null) {
try {
Arguments argumentsAnno = getAnnotation(m, Arguments.class);
TestFormat.check(argumentsAnno != null || m.getParameterCount() == 0, "Missing @Arguments annotation to define arguments of " + m);
BaseTest baseTest = new BaseTest(test, shouldExcludeTest(m.getName()));
allTests.add(baseTest);
if (PRINT_VALID_IR_RULES) {
irMatchRulePrinter.emitRuleEncoding(m, baseTest.isSkipped());
}
} catch (TestFormatException e) {
// Failure logged. Continue and report later.
}
}
});
}
/**
* Check if user wants to exclude this test by checking the -DTest and -DExclude lists.
*/
private boolean shouldExcludeTest(String testName) {
boolean hasTestList = testList != null;
boolean hasExcludeList = excludeList != null;
if (hasTestList) {
return !testList.contains(testName) || (hasExcludeList && excludeList.contains(testName));
} else if (hasExcludeList) {
return excludeList.contains(testName);
}
return false;
}
/**
* Generate replay compilation files.
*/
private void addReplay() {
String directive = "[{ match: \"*.*\", DumpReplay: true }]";
TestFramework.check(WHITE_BOX.addCompilerDirective(directive) == 1, "Failed to add DUMP_REPLAY directive");
}
private void processControlAnnotations(Class<?> clazz) {
if (IGNORE_COMPILER_CONTROLS) {
return;
}
// Also apply compile commands to all inner classes of 'clazz'.
ArrayList<Class<?>> classes = new ArrayList<>(Arrays.asList(clazz.getDeclaredClasses()));
classes.add(clazz);
for (Class<?> c : classes) {
applyClassAnnotations(c);
List<Executable> executables = new ArrayList<>(Arrays.asList(c.getDeclaredMethods()));
Collections.addAll(executables, c.getDeclaredConstructors());
for (Executable ex : executables) {
checkClassAnnotations(ex);
try {
applyIndependentCompilationCommands(ex);
} catch (TestFormatException e) {
// Failure logged. Continue and report later.
}
}
// Only force compilation now because above annotations affect inlining
for (Executable ex : executables) {
try {
applyForceCompileCommand(ex);
} catch (TestFormatException e) {
// Failure logged. Continue and report later.
}
}
}
}
private void applyClassAnnotations(Class<?> c) {
ForceCompileClassInitializer anno = getAnnotation(c, ForceCompileClassInitializer.class);
if (anno == null) {
return;
}
// Compile class initializer
CompLevel level = anno.value();
if (level == CompLevel.SKIP || level == CompLevel.WAIT_FOR_COMPILATION) {
TestFormat.failNoThrow("Cannot define compilation level SKIP or WAIT_FOR_COMPILATION in " +
"@ForceCompileClassInitializer at " + c);
return;
}
level = restrictCompLevel(anno.value());
if (level != CompLevel.SKIP) {
// Make sure class is initialized to avoid compilation bailout of <clinit>
getClassObject(c.getName(), "nested"); // calls Class.forName() to initialize 'c'
TestFormat.checkNoThrow(WHITE_BOX.enqueueInitializerForCompilation(c, level.getValue()),
"Failed to enqueue <clinit> of " + c + " for compilation. Did you specify "
+ "@ForceCompileClassInitializer without providing a static class initialization? "
+ "Make sure to provide any form of static initialization or remove the annotation.");
}
}
private void checkClassAnnotations(Executable ex) {
TestFormat.checkNoThrow(getAnnotation(ex, ForceCompileClassInitializer.class) == null,
"@ForceCompileClassInitializer only allowed at classes but not at method " + ex);
}
/**
* Exclude a method from compilation with a compiler randomly. Return the compiler for which the method was made
* not compilable.
*/
public static Compiler excludeRandomly(Executable ex) {
Compiler compiler = switch (Utils.getRandomInstance().nextInt() % 3) {
case 1 -> Compiler.C1;
case 2 -> Compiler.C2;
default -> Compiler.ANY;
};
WHITE_BOX.makeMethodNotCompilable(ex, compiler.getValue(), false);
WHITE_BOX.makeMethodNotCompilable(ex, compiler.getValue(), true);
System.out.println("Excluding from " + compiler.name() + " compilation: " + ex);
return compiler;
}
private void applyIndependentCompilationCommands(Executable ex) {
ForceInline forceInlineAnno = getAnnotation(ex, ForceInline.class);
DontInline dontInlineAnno = getAnnotation(ex, DontInline.class);
ForceCompile forceCompileAnno = getAnnotation(ex, ForceCompile.class);
DontCompile dontCompileAnno = getAnnotation(ex, DontCompile.class);
checkCompilationCommandAnnotations(ex, forceInlineAnno, dontInlineAnno, forceCompileAnno, dontCompileAnno);
// First handle inline annotations
if (dontInlineAnno != null) {
WHITE_BOX.testSetDontInlineMethod(ex, true);
} else if (forceInlineAnno != null) {
WHITE_BOX.testSetForceInlineMethod(ex, true);
}
if (dontCompileAnno != null) {
dontCompileWithCompiler(ex, dontCompileAnno.value());
}
if (EXCLUDE_RANDOM && getAnnotation(ex, Test.class) == null && forceCompileAnno == null && dontCompileAnno == null) {
// Randomly exclude helper methods from compilation
if (Utils.getRandomInstance().nextBoolean()) {
excludeRandomly(ex);
}
}
}
private void checkCompilationCommandAnnotations(Executable ex, ForceInline forceInlineAnno, DontInline dontInlineAnno, ForceCompile forceCompileAnno, DontCompile dontCompileAnno) {
Test testAnno = getAnnotation(ex, Test.class);
Run runAnno = getAnnotation(ex, Run.class);
Check checkAnno = getAnnotation(ex, Check.class);
TestFormat.check((testAnno == null && runAnno == null && checkAnno == null) || Stream.of(forceCompileAnno, dontCompileAnno, forceInlineAnno, dontInlineAnno).noneMatch(Objects::nonNull),
"Cannot use explicit compile command annotations (@ForceInline, @DontInline, " +
"@ForceCompile or @DontCompile) together with @Test, @Check or @Run: " + ex + ". Use compLevel in @Test for fine tuning.");
if (Stream.of(forceInlineAnno, dontCompileAnno, dontInlineAnno).filter(Objects::nonNull).count() > 1) {
// Failure
TestFormat.check(dontCompileAnno == null || dontInlineAnno == null,
"@DontInline is implicitely done with @DontCompile annotation at " + ex);
TestFormat.fail("Cannot mix @ForceInline, @DontInline and @DontCompile at the same time at " + ex);
}
TestFormat.check(forceInlineAnno == null || dontInlineAnno == null, "Cannot have @ForceInline and @DontInline at the same time at " + ex);
if (forceCompileAnno != null && dontCompileAnno != null) {
CompLevel forceCompileLevel = forceCompileAnno.value();
Compiler dontCompileCompiler = dontCompileAnno.value();
TestFormat.check(dontCompileCompiler != Compiler.ANY,
"Cannot have @DontCompile(Compiler.ANY) and @ForceCompile at the same time at " + ex);
TestFormat.check(forceCompileLevel != CompLevel.ANY,
"Cannot have @ForceCompile(CompLevel.ANY) and @DontCompile at the same time at " + ex);
TestFormat.check(forceCompileLevel.isNotCompilationLevelOfCompiler(dontCompileCompiler),
"Overlapping compilation level and compiler with @ForceCompile and @DontCompile at " + ex);
}
}
/**
* Exlude the method from compilation and make sure it is not inlined.
*/
private void dontCompileAndDontInlineMethod(Method m) {
if (!IGNORE_COMPILER_CONTROLS) {
WHITE_BOX.makeMethodNotCompilable(m, CompLevel.ANY.getValue(), true);
WHITE_BOX.makeMethodNotCompilable(m, CompLevel.ANY.getValue(), false);
WHITE_BOX.testSetDontInlineMethod(m, true);
}
}
private void dontCompileWithCompiler(Executable ex, Compiler compiler) {
if (VERBOSE) {
System.out.println("dontCompileWithCompiler " + ex + " , compiler = " + compiler.name());
}
WHITE_BOX.makeMethodNotCompilable(ex, compiler.getValue(), true);
WHITE_BOX.makeMethodNotCompilable(ex, compiler.getValue(), false);
if (compiler == Compiler.ANY) {
WHITE_BOX.testSetDontInlineMethod(ex, true);
}
}
private void applyForceCompileCommand(Executable ex) {
ForceCompile forceCompileAnno = getAnnotation(ex, ForceCompile.class);
if (forceCompileAnno != null) {
CompLevel complevel = forceCompileAnno.value();
TestFormat.check(complevel != CompLevel.SKIP && complevel != CompLevel.WAIT_FOR_COMPILATION,
"Cannot define compilation level SKIP or WAIT_FOR_COMPILATION in @ForceCompile at " + ex);
complevel = restrictCompLevel(forceCompileAnno.value());
if (FLIP_C1_C2) {
complevel = complevel.flipCompLevel();
}
if (EXCLUDE_RANDOM) {
complevel = complevel.excludeCompilationRandomly(ex);
}
if (complevel != CompLevel.SKIP) {
enqueueForCompilation(ex, complevel);
forceCompileMap.put(ex, complevel);
}
}
}
static void enqueueForCompilation(Executable ex, CompLevel requestedCompLevel) {
if (TestVM.VERBOSE) {
System.out.println("enqueueForCompilation " + ex + ", level = " + requestedCompLevel);
}
CompLevel compLevel = restrictCompLevel(requestedCompLevel);
if (compLevel != CompLevel.SKIP) {
WHITE_BOX.enqueueMethodForCompilation(ex, compLevel.getValue());
} else {
System.out.println("Skipped compilation on level " + requestedCompLevel + " due to VM flags not allowing it.");
}
}
/**
* Setup @Test annotated method an add them to the declaredTests map to have a convenient way of accessing them
* once setting up a framework test (base checked, or custom run test).
*/
private void setupDeclaredTests() {
for (Method m : testClass.getDeclaredMethods()) {
Test testAnno = getAnnotation(m, Test.class);
try {
if (testAnno != null) {
addDeclaredTest(m);
} else {
TestFormat.checkNoThrow(!m.isAnnotationPresent(IR.class), "Found @IR annotation on non-@Test method " + m);
TestFormat.checkNoThrow(!m.isAnnotationPresent(Warmup.class) || getAnnotation(m, Run.class) != null,
"Found @Warmup annotation on non-@Test or non-@Run method " + m);
}
} catch (TestFormatException e) {
// Failure logged. Continue and report later.
}
}
TestFormat.checkNoThrow(!declaredTests.isEmpty(), "Did not specify any @Test methods in " + testClass);
}
private void addDeclaredTest(Method m) {
Test testAnno = getAnnotation(m, Test.class);
checkTestAnnotations(m, testAnno);
Warmup warmup = getAnnotation(m, Warmup.class);
int warmupIterations = WARMUP_ITERATIONS;
if (warmup != null) {
warmupIterations = warmup.value();
TestFormat.checkNoThrow(warmupIterations >= 0, "Cannot have negative value for @Warmup at " + m);
}
if (!IGNORE_COMPILER_CONTROLS) {
// Don't inline test methods by default. Do not apply this when -DIgnoreCompilerControls=true is set.
WHITE_BOX.testSetDontInlineMethod(m, true);
}
CompLevel compLevel = restrictCompLevel(testAnno.compLevel());
if (FLIP_C1_C2) {
compLevel = compLevel.flipCompLevel();
}
if (EXCLUDE_RANDOM) {
compLevel = compLevel.excludeCompilationRandomly(m);
}
DeclaredTest test = new DeclaredTest(m, ArgumentValue.getArguments(m), compLevel, warmupIterations);
declaredTests.put(m, test);
testMethodMap.put(m.getName(), m);
}
private void checkTestAnnotations(Method m, Test testAnno) {
TestFormat.check(!testMethodMap.containsKey(m.getName()),
"Cannot overload two @Test methods: " + m + ", " + testMethodMap.get(m.getName()));
TestFormat.check(testAnno != null, m + " must be a method with a @Test annotation");
Check checkAnno = getAnnotation(m, Check.class);
Run runAnno = getAnnotation(m, Run.class);
TestFormat.check(checkAnno == null && runAnno == null,
m + " has invalid @Check or @Run annotation while @Test annotation is present.");
TestFormat.checkNoThrow(Arrays.stream(m.getParameterTypes()).noneMatch(AbstractInfo.class::isAssignableFrom),
"Cannot " + AbstractInfo.class + " or any of its subclasses as parameter type at " +
"@Test method " + m);
TestFormat.checkNoThrow(!AbstractInfo.class.isAssignableFrom(m.getReturnType()),
"Cannot " + AbstractInfo.class + " or any of its subclasses as return type at " +
"@Test method " + m);
}
/**
* Get the appropriate level as permitted by the test scenario and VM flags.
*/
private static CompLevel restrictCompLevel(CompLevel compLevel) {
if (!USE_COMPILER) {
return CompLevel.SKIP;
}
if (compLevel == CompLevel.ANY) {
// Use highest available compilation level by default (usually C2).
compLevel = TIERED_COMPILATION_STOP_AT_LEVEL;
}
if (!TIERED_COMPILATION && compLevel.getValue() < CompLevel.C2.getValue()) {
return CompLevel.SKIP;
}
if (TIERED_COMPILATION && compLevel.getValue() > TIERED_COMPILATION_STOP_AT_LEVEL.getValue()) {
return CompLevel.SKIP;
}
return compLevel;
}
/**
* Verify that the helper classes do not contain illegal framework annotations and then apply the actions as
* specified by the different helper class annotations.
*/
private void processHelperClasses() {
if (helperClasses != null) {
for (Class<?> helperClass : helperClasses) {
// Process the helper classes and apply the explicit compile commands
TestFormat.checkNoThrow(helperClass != testClass,
"Cannot specify test " + testClass + " as helper class, too.");
checkHelperClass(helperClass);
processControlAnnotations(helperClass);
}
}
}
/**
* First set up checked (with @Check) and custom run tests (with @Run). All remaining unmatched/unused @Test methods
* are treated as base tests and set up as such later.
*/
private void setupCheckedAndCustomRunTests() {
for (Method m : testClass.getDeclaredMethods()) {
Check checkAnno = getAnnotation(m, Check.class);
Run runAnno = getAnnotation(m, Run.class);
Arguments argumentsAnno = getAnnotation(m, Arguments.class);
try {
TestFormat.check(argumentsAnno == null || (checkAnno == null && runAnno == null),
"Cannot have @Argument annotation in combination with @Run or @Check at " + m);
if (checkAnno != null) {
addCheckedTest(m, checkAnno, runAnno);
} else if (runAnno != null) {
addCustomRunTest(m, runAnno);
}
} catch (TestFormatException e) {
// Failure logged. Continue and report later.
}
}
}
/**
* Set up a checked test by first verifying the correct format of the @Test and @Check method and then adding it
* to the allTests list which keeps track of all framework tests that are eventually executed.
*/
private void addCheckedTest(Method m, Check checkAnno, Run runAnno) {
Method testMethod = testMethodMap.get(checkAnno.test());
DeclaredTest test = declaredTests.get(testMethod);
checkCheckedTest(m, checkAnno, runAnno, testMethod, test);
test.setAttachedMethod(m);
CheckedTest.Parameter parameter = getCheckedTestParameter(m, testMethod);
dontCompileAndDontInlineMethod(m);
CheckedTest checkedTest = new CheckedTest(test, m, checkAnno, parameter, shouldExcludeTest(testMethod.getName()));
allTests.add(checkedTest);
if (PRINT_VALID_IR_RULES) {
// Only need to emit IR verification information if IR verification is actually performed.
irMatchRulePrinter.emitRuleEncoding(m, checkedTest.isSkipped());
}
}
private void checkCheckedTest(Method m, Check checkAnno, Run runAnno, Method testMethod, DeclaredTest test) {
TestFormat.check(runAnno == null, m + " has invalid @Run annotation while @Check annotation is present.");
TestFormat.check(testMethod != null, "Did not find associated test method \"" + m.getDeclaringClass().getName()
+ "." + checkAnno.test() + "\" for @Check at " + m);
TestFormat.check(test != null, "Missing @Test annotation for associated test method " + testMethod + " for @Check at " + m);
Method attachedMethod = test.getAttachedMethod();
TestFormat.check(attachedMethod == null,
"Cannot use @Test " + testMethod + " for more than one @Run or one @Check method. Found: " + m + ", " + attachedMethod);
}
/**
* Only allow parameters as specified in {@link Check}.
*/
private CheckedTest.Parameter getCheckedTestParameter(Method m, Method testMethod) {
boolean firstParameterTestInfo = m.getParameterCount() > 0 && m.getParameterTypes()[0].equals(TestInfo.class);
boolean secondParameterTestInfo = m.getParameterCount() > 1 && m.getParameterTypes()[1].equals(TestInfo.class);
CheckedTest.Parameter parameter = null;
Class<?> testReturnType = testMethod.getReturnType();
switch (m.getParameterCount()) {
case 0 -> parameter = CheckedTest.Parameter.NONE;
case 1 -> {
TestFormat.checkNoThrow(firstParameterTestInfo || m.getParameterTypes()[0] == testReturnType,
"Single-parameter version of @Check method " + m + " must match return type of @Test " + testMethod);
parameter = firstParameterTestInfo ? CheckedTest.Parameter.TEST_INFO_ONLY : CheckedTest.Parameter.RETURN_ONLY;
}
case 2 -> {
TestFormat.checkNoThrow(m.getParameterTypes()[0] == testReturnType && secondParameterTestInfo,
"Two-parameter version of @Check method " + m + " must provide as first parameter the same"
+ " return type as @Test method " + testMethod + " and as second parameter an object of " + TestInfo.class);
parameter = CheckedTest.Parameter.BOTH;
}
default -> TestFormat.failNoThrow("@Check method " + m + " must provide either a none, single or two-parameter variant.");
}
return parameter;
}
/**
* Set up a custom run test by first verifying the correct format of the @Test and @Run method and then adding it
* to the allTests list which keeps track of all framework tests that are eventually executed.
*/
private void addCustomRunTest(Method m, Run runAnno) {
checkRunMethod(m, runAnno);
List<DeclaredTest> tests = new ArrayList<>();
boolean shouldExcludeTest = true;
for (String testName : runAnno.test()) {
try {
Method testMethod = testMethodMap.get(testName);
DeclaredTest test = declaredTests.get(testMethod);
checkCustomRunTest(m, testName, testMethod, test, runAnno.mode());
test.setAttachedMethod(m);
tests.add(test);
// Only exclude custom run test if all test methods excluded
shouldExcludeTest &= shouldExcludeTest(testMethod.getName());
} catch (TestFormatException e) {
// Logged, continue.
}
}
if (tests.isEmpty()) {
return; // There was a format violation. Return.
}
dontCompileAndDontInlineMethod(m);
CustomRunTest customRunTest = new CustomRunTest(m, getAnnotation(m, Warmup.class), runAnno, tests, shouldExcludeTest);
allTests.add(customRunTest);
if (PRINT_VALID_IR_RULES) {
tests.forEach(test -> irMatchRulePrinter.emitRuleEncoding(test.getTestMethod(), customRunTest.isSkipped()));
}
}
/**
* Only allow parameters as specified in {@link Run}.
*/
private void checkCustomRunTest(Method m, String testName, Method testMethod, DeclaredTest test, RunMode runMode) {
TestFormat.check(testMethod != null, "Did not find associated @Test method \"" + m.getDeclaringClass().getName()
+ "." + testName + "\" specified in @Run at " + m);
TestFormat.check(test != null,
"Missing @Test annotation for associated test method " + testName + " for @Run at " + m);
Method attachedMethod = test.getAttachedMethod();
TestFormat.check(attachedMethod == null,
"Cannot use @Test " + testMethod + " for more than one @Run/@Check method. Found: "
+ m + ", " + attachedMethod);
TestFormat.check(!test.hasArguments(),
"Cannot use @Arguments at test method " + testMethod + " in combination with @Run method " + m);
Warmup warmupAnno = getAnnotation(testMethod, Warmup.class);
TestFormat.checkNoThrow(warmupAnno == null,
"Cannot set @Warmup at @Test method " + testMethod + " when used with its @Run method "
+ m + ". Use @Warmup at @Run method instead.");
Test testAnno = getAnnotation(testMethod, Test.class);
TestFormat.checkNoThrow(runMode != RunMode.STANDALONE || testAnno.compLevel() == CompLevel.ANY,
"Setting explicit compilation level for @Test method " + testMethod + " has no effect "
+ "when used with STANDALONE @Run method " + m);
}
private void checkRunMethod(Method m, Run runAnno) {
TestFormat.check(runAnno.test().length > 0, "@Run method " + m + " must specify at least one test method");
TestFormat.checkNoThrow(m.getParameterCount() == 0 || (m.getParameterCount() == 1 && m.getParameterTypes()[0].equals(RunInfo.class)),
"@Run method " + m + " must specify either no parameter or exactly one " + RunInfo.class + " parameter.");
Warmup warmupAnno = getAnnotation(m, Warmup.class);
TestFormat.checkNoThrow(warmupAnno == null || runAnno.mode() != RunMode.STANDALONE,
"Cannot set @Warmup at @Run method " + m + " when used with RunMode.STANDALONE. The @Run method is only invoked once.");
}
private static <T extends Annotation> T getAnnotation(AnnotatedElement element, Class<T> c) {
T[] annos = element.getAnnotationsByType(c);
TestFormat.check(annos.length < 2, element + " has duplicated annotations");
return Arrays.stream(annos).findFirst().orElse(null);
}
/**
* Ensure that all compilations that were enforced (added to compilation queue) by framework annotations are
* completed. Wait if necessary for a short amount of time for their completion.
*/
private void checkForcedCompilationsCompleted() {
if (forceCompileMap.isEmpty()) {
return;
}
final long started = System.currentTimeMillis();
long elapsed;
do {
forceCompileMap.entrySet().removeIf(entry -> WHITE_BOX.getMethodCompilationLevel(entry.getKey()) == entry.getValue().getValue());
if (forceCompileMap.isEmpty()) {
// All @ForceCompile methods are compiled at the requested level.
return;
}
// Retry again if not yet compiled.
forceCompileMap.forEach(TestVM::enqueueForCompilation);
elapsed = System.currentTimeMillis() - started;
} while (elapsed < 5000);
StringBuilder builder = new StringBuilder();
forceCompileMap.forEach((key, value) -> builder.append("- ").append(key).append(" at CompLevel.").append(value)
.append(System.lineSeparator()));
throw new TestRunException("Could not force compile the following @ForceCompile methods:"
+ System.lineSeparator() + builder.toString());
}
/**
* Once all framework tests are collected, they are run in this method.
*/
private void runTests() {
TreeMap<Long, String> durations = (PRINT_TIMES || VERBOSE) ? new TreeMap<>() : null;
long startTime = System.nanoTime();
List<AbstractTest> testList;
boolean testFilterPresent = testFilterPresent();
if (testFilterPresent) {
// Only run the specified tests by the user filters -DTest and/or -DExclude.
testList = allTests.stream().filter(test -> !test.isSkipped()).collect(Collectors.toList());
if (testList.isEmpty()) {
// Throw an exception to inform the user about an empty specified test set with -DTest and/or -DExclude
throw new NoTestsRunException();
}
} else {
testList = allTests;
}
if (SHUFFLE_TESTS) {
// Execute tests in random order (execution sequence affects profiling). This is done by default.
Collections.shuffle(testList, Utils.getRandomInstance());
}
StringBuilder builder = new StringBuilder();
int failures = 0;
// Execute all tests and keep track of each exception that is thrown. These are then reported once all tests
// are executing. This prevents a premature exit without running all tests.
for (AbstractTest test : testList) {
if (VERBOSE) {
System.out.println("Run " + test.toString());
}
if (testFilterPresent) {
TestFrameworkSocket.write("Run " + test.toString(), "testfilter", true);
}
try {
test.run();
} catch (TestRunException e) {
StringWriter sw = new StringWriter();
PrintWriter pw = new PrintWriter(sw);
e.printStackTrace(pw);
builder.append(test.toString()).append(":").append(System.lineSeparator()).append(sw.toString())
.append(System.lineSeparator()).append(System.lineSeparator());
failures++;
}
if (PRINT_TIMES || VERBOSE) {
long endTime = System.nanoTime();
long duration = (endTime - startTime);
durations.put(duration, test.getName());
if (VERBOSE) {
System.out.println("Done " + test.getName() + ": " + duration + " ns = " + (duration / 1000000) + " ms");
}
}
if (GC_AFTER) {
System.out.println("doing GC");
WHITE_BOX.fullGC();
}
}
// Print execution times
if (VERBOSE || PRINT_TIMES) {
System.out.println(System.lineSeparator() + System.lineSeparator() + "Test execution times:");
for (Map.Entry<Long, String> entry : durations.entrySet()) {
System.out.format("%-10s%15d ns%n", entry.getValue() + ":", entry.getKey());
}
}
if (failures > 0) {
// Finally, report all occurred exceptions in a nice format.
String msg = System.lineSeparator() + System.lineSeparator() + "Test Failures (" + failures + ")"
+ System.lineSeparator() + "----------------" + "-".repeat(String.valueOf(failures).length());
throw new TestRunException(msg + System.lineSeparator() + builder.toString());
}
}
private boolean testFilterPresent() {
return testList != null || excludeList != null;
}
enum TriState {
Maybe,
Yes,
No
}
public static void compile(Method m, CompLevel compLevel) {
TestRun.check(compLevel != CompLevel.SKIP && compLevel != CompLevel.WAIT_FOR_COMPILATION,
"Invalid compilation request with level " + compLevel);
enqueueForCompilation(m, compLevel);
}
public static void deoptimize(Method m) {
WHITE_BOX.deoptimizeMethod(m);
}
public static boolean isCompiled(Method m) {
return compiledAtLevel(m, CompLevel.ANY) == TriState.Yes;
}
public static boolean isC1Compiled(Method m) {
return compiledByC1(m) == TriState.Yes;
}
public static boolean isC2Compiled(Method m) {
return compiledByC2(m) == TriState.Yes;
}
public static boolean isCompiledAtLevel(Method m, CompLevel compLevel) {
return compiledAtLevel(m, compLevel) == TriState.Yes;
}
public static void assertDeoptimizedByC1(Method m) {
if (notUnstableDeoptAssertion(m, CompLevel.C1_SIMPLE)) {
TestRun.check(compiledByC1(m) != TriState.Yes || PER_METHOD_TRAP_LIMIT == 0 || !PROFILE_INTERPRETER,
m + " should have been deoptimized by C1");
}
}
public static void assertDeoptimizedByC2(Method m) {
if (notUnstableDeoptAssertion(m, CompLevel.C2)) {
TestRun.check(compiledByC2(m) != TriState.Yes || PER_METHOD_TRAP_LIMIT == 0 || !PROFILE_INTERPRETER,
m + " should have been deoptimized by C2");
}
}
/**
* Some VM flags could make the deopt assertions unstable.
*/
private static boolean notUnstableDeoptAssertion(Method m, CompLevel level) {
return (USE_COMPILER && !XCOMP && !IGNORE_COMPILER_CONTROLS && !TEST_C1 &&
(!EXCLUDE_RANDOM || WHITE_BOX.isMethodCompilable(m, level.getValue(), false)));
}
public static void assertCompiledByC1(Method m) {
TestRun.check(compiledByC1(m) != TriState.No, m + " should have been C1 compiled");
}
public static void assertCompiledByC2(Method m) {
TestRun.check(compiledByC2(m) != TriState.No, m + " should have been C2 compiled");
}
public static void assertCompiledAtLevel(Method m, CompLevel level) {
TestRun.check(compiledAtLevel(m, level) != TriState.No, m + " should have been compiled at level " + level.name());
}
public static void assertNotCompiled(Method m) {
TestRun.check(!isC1Compiled(m), m + " should not have been compiled by C1");
TestRun.check(!isC2Compiled(m), m + " should not have been compiled by C2");
}
public static void assertCompiled(Method m) {
TestRun.check(compiledByC1(m) != TriState.No || compiledByC2(m) != TriState.No, m + " should have been compiled");
}
private static TriState compiledByC1(Method m) {
TriState triState = compiledAtLevel(m, CompLevel.C1_SIMPLE);
if (triState != TriState.No) {
return triState;
}
triState = compiledAtLevel(m, CompLevel.C1_LIMITED_PROFILE);
if (triState != TriState.No) {
return triState;
}
triState = compiledAtLevel(m, CompLevel.C1_FULL_PROFILE);
return triState;
}
private static TriState compiledByC2(Method m) {
return compiledAtLevel(m, CompLevel.C2);
}
private static TriState compiledAtLevel(Method m, CompLevel level) {
if (WHITE_BOX.isMethodCompiled(m, false)) {
switch (level) {
case C1_SIMPLE, C1_LIMITED_PROFILE, C1_FULL_PROFILE, C2 -> {
if (WHITE_BOX.getMethodCompilationLevel(m, false) == level.getValue()) {
return TriState.Yes;
}
}
case ANY -> {
return TriState.Yes;
}
default -> throw new TestRunException("compiledAtLevel() should not be called with " + level);
}
}
if (!USE_COMPILER || XCOMP || TEST_C1 || IGNORE_COMPILER_CONTROLS ||
(EXCLUDE_RANDOM && !WHITE_BOX.isMethodCompilable(m, level.getValue(), false))) {
return TriState.Maybe;
}
return TriState.No;
}
}

View File

@ -0,0 +1,89 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.examples;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.test.TestVM; // Only used for Javadocs
/*
* @test
* @summary Example test to use the new test framework.
* @library /test/lib /
* @run driver ir_framework.examples.BaseTestExample
*/
/**
* If there is no warm up specified the Test Framework will do the following:
* <ol>
* <li><p>Invoke @Test method {@link TestVM#WARMUP_ITERATIONS} many times.</li>
* <li><p>Then do compilation of @Test method. <b>(**)</b></li>
* <li><p>Invoke @Test method once again</li>
* </ol>
* <p>
*
* Configurable things for simple tests (no @Run or @Check) at @Test method:
* <ul>
* <li><p>compLevel: Specify at which compilation level the test should be compiled by the framework at step <b>(**)</b>.
* If {@link CompLevel#WAIT_FOR_COMPILATION} is specified, the framework will continue invoke the
* method until HotSpot compiles it. If it is not compiled after 10s, an exception is thrown.</li>
* <li><p>@Warmup: Change warm-up iterations of test (defined by default by TestVM.WARMUP_ITERATIONS)</li>
* <li><p>@Arguments: If a @Test method specifies arguments, you need to provide arguments by using @Arguments such
* that the framework knows how to call the method. If you need more complex values, use @Run.</li>
* <li><p>@IR: Arbitrary number of @IR rules.</li>
* </ul>
*
* @see Test
* @see Arguments
* @see Warmup
* @see TestFramework
*/
public class BaseTestExample {
int iFld;
public static void main(String[] args) {
TestFramework.run(); // equivalent to TestFramework.run(BaseTestExample.class)
}
// Test without arguments.
@Test
public void mostBasicTest() {
iFld = 42;
}
// Test with arguments. Use Argument class to choose a value.
// Object arguments need to have an associated default constructor in its class.
@Test
@Arguments({Argument.DEFAULT, Argument.MAX})
public void basicTestWithArguments(int x, long y) {
iFld = x;
}
// @Warmup needs to be positive or zero. In case of zero, the method is directly compiled (simulated -Xcomp).
@Test
@Arguments({Argument.DEFAULT, Argument.MAX})
@Warmup(100)
public void basicTestWithDifferentWarmup(int x, long y) {
iFld = x;
}
}

View File

@ -0,0 +1,149 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.examples;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.test.TestVM; // Only used for Javadocs
/*
* @test
* @summary Example test to use the new test framework.
* @library /test/lib /
* @run driver ir_framework.examples.CheckedTestExample
*/
/**
* If there is no non-default warm-up specified, the Test Framework will do the following:
* <ol>
* <li><p>Invoke @Test method {@link TestVM#WARMUP_ITERATIONS} many times.</li>
* <li><p>By default, after each invocation, the @Check method of the @Test method is invoked. This can be disabled
* by using {@link CheckAt#COMPILED}</li>
* <li><p>After the warm-up, the @Test method is compiled.</li>
* <li><p>Invoke @Test method once again and then always invoke the @Check method once again.</li>
* </ol>
* <p>
*
* Configurable things for checked tests:
* <ul>
* <li><p>At @Test method:</li>
* <ul>
* <li><p>@Warmup: Change warm-up iterations of test (defined by default by TestVM.WARMUP_ITERATIONS)</li>
* <li><p>@Arguments: If a @Test method specifies arguments, you need to provide arguments by using @Arguments
* such that the framework knows how to call the method. If you need more complex values, use a
* custom run test with @Run.</li>
* <li><p>@IR: Arbitrary number of @IR rules.</li>
* </ul>
* <li><p>At @Check method:</li>
* <ul>
* <li><p>{@link Check#when}: When should the @Check method be invoked.</li>
* <li><p>No @IR annotations.</li>
* </ul>
* </ul>
*
* @see Check
* @see Test
* @see Arguments
* @see Warmup
* @see TestFramework
*/
public class CheckedTestExample {
public static void main(String[] args) {
TestFramework.run(); // equivalent to TestFramework.run(CheckedTestExample.class)
}
@Test
@Arguments(Argument.DEFAULT) // As with normal tests, you need to tell the framework what the argument is.
@Warmup(100) // As with normal tests, you can specify the warmup iterations.
public int test(int x) {
return 42;
}
// Check method for test(). Invoked directly after test() by the Test Framework.
@Check(test = "test") // Specify the @Test method for which this method is a check.
public void basicCheck() {
// Do some checks after an invocation.
}
@Test
public int test2() {
return 42;
}
// This version of @Check passes the return value from test2() as an argument.
// The return type and the parameter type must match.
@Check(test = "test2")
public void checkWithReturn(int returnValue) {
// Do some checks after an invocation.
if (returnValue != 42) {
throw new RuntimeException("Must match");
}
}
@Test
public int test3() {
return 42;
}
// This version of @Check passes a TestInfo object to the check which contains some additional information about the test.
@Check(test = "test3")
public void checkWithTestInfo(TestInfo info) {
// Do some checks after an invocation. Additional queries with TestInfo.
if (!info.isWarmUp()) {
// ...
}
}
@Test
public int test4() {
return 42;
}
// This version of @Check passes the return value and a TestInfo object to the check which contains some additional
// information about the test. The order of the arguments is important. The return value must come first and the
// the TestInfo parameter second. Any other combination or use of different arguments are forbidden for @Check methods.
@Check(test = "test4")
public void checkWithReturnAndTestInfo(int returnValue, TestInfo info) {
// Do some checks after an invocation. Additional queries with TestInfo.
if (returnValue != 42) {
throw new RuntimeException("Must match");
}
if (!info.isWarmUp()) {
// ...
}
}
@Test
public int test5() {
return 42;
}
// Check method for test5() is only invoked once warmup is finished and test() has been compiled by the Test Framework.
@Check(test = "test5", when = CheckAt.COMPILED) // Specify the @Test method for which this method is a check.
public void checkAfterCompiled(TestInfo info) {
// Do some checks after compilation.
TestFramework.assertCompiled(info.getTest()); // Test is compiled by framework after warm-up.
}
}

View File

@ -0,0 +1,168 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.examples;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.test.TestVM; // Only used for Javadocs
/*
* @test
* @summary Example test to use the new test framework.
* @library /test/lib /
* @run driver ir_framework.examples.CustomRunTestExample
*/
/**
* If there is no warm-up specified, the Test Framework will do the following:
* <ol>
* <li><p>Invoke @Run method {@link TestVM#WARMUP_ITERATIONS} many times. Note that the @Run method
* is responsible to invoke the @Test methods to warm it up properly. This is not done by the framework. Not
* invoking a @Test method will result in an -Xcomp like compilation of the method as there is no profile
* information for it. The @Run method can do any arbitrary argument setup and return value verification and
* can invoke the @Test methods multiple times in a single invocation of the @Run method or even skip some
* test invocations.</li>
* <li><p>After the warm-up, the @Test methods are compiled (there can be multiple @Test methods).</li>
* <li><p>Invoke the @Run method once again.</li>
* </ol>
* <p>
*
* Configurable things for custom run tests:
* <ul>
* <li><p>At @Test methods:</li>
* <ul>
* <li><p>@IR: Arbitrary number of @IR rules.</li>
* <li><p>No @Warmup, this must be set at @Run method.</li>
* <li><p>No @Arguments, the arguments are set by @Run method.</li>
* </ul>
* <li><p>At @Run method:</li>
* <ul>
* <li><p>@Warmup: Change warm-up iterations of @Run method (defined by default by
* TestVM.WARMUP_ITERATIONS)</li>
* <li><p>{@link Run#test}: Specify any number of @Test methods. They cannot be shared with other @Check or @Run
* methods.</li>
* <li><p>{@link Run#mode}: Choose between normal invocation as described above or {@link RunMode#STANDALONE}.
* STANDALONE only invokes the @Run method once without warm-up or a compilation by the
* Test Framework. The only thing done by the framework is the verification of any @IR
* rules afterwards. The STANDALONE @Run method needs to make sure that a C2 compilation
* is reliably triggered if there are any @IR rules.</li>
* <li><p>No @IR annotations</li>
* </ul>
* </ul>
*
* @see Run
* @see Test
* @see RunMode
* @see TestFramework
*/
public class CustomRunTestExample {
public static void main(String[] args) {
TestFramework.run(); // equivalent to TestFramework.run(CustomRunTestExample.class)
}
@Test
public int test(int x) {
return x;
}
// Run method for test(). Invoked directly by Test Framework instead of test().
// Can do anything you like. It's also possible to skip or do multiple invocations of test()
@Run(test = "test") // Specify the @Test method for which this method is a runner.
public void basicRun() {
int returnValue = test(34);
if (returnValue != 34) {
throw new RuntimeException("Must match");
}
}
@Test
public int test2(int x) {
return x;
}
// This version of @Run passes the RunInfo object as an argument. No other arguments and combiniations are allowed.
@Run(test = "test2")
public void runWithRunInfo(RunInfo info) {
// We could also skip some invocations. This might have an influence on possible @IR rules, need to be careful.
if (info.getRandom().nextBoolean()) {
int returnValue = test(34);
if (returnValue != 34) {
throw new RuntimeException("Must match");
}
}
}
@Test
public int test3(int x) {
return x;
}
// This version of @Run uses a user defined @Warmup.
@Run(test = "test3")
@Warmup(100)
public void runWithWarmUp() {
int returnValue = test3(34);
if (returnValue != 34) {
throw new RuntimeException("Must match");
}
}
@Test
public int test4(int x) {
return x;
}
// This version of @Run is only invoked once by the Test Framework. There is no warm-up and no compilation done
// by the Test Framework. The only thing done by the framework is @IR rule verification.
@Run(test = "test4", mode = RunMode.STANDALONE)
public void runOnlyOnce() {
int returnValue = test4(34);
if (returnValue != 34) {
throw new RuntimeException("Must match");
}
}
@Test
public int test5(int x) {
return x;
}
@Test
public int test6(int x) {
return x;
}
// This version of @Run can run multiple test methods and get them IR checked as part of this custom run test.
@Run(test = {"test5", "test6"})
public void runMultipleTests() {
int returnValue = test5(34);
if (returnValue != 34) {
throw new RuntimeException("Must match");
}
returnValue = test6(42);
if (returnValue != 42) {
throw new RuntimeException("Must match");
}
}
}

View File

@ -0,0 +1,171 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.examples;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.driver.IRViolationException;
/*
* @test
* @summary Example test to use the new test framework.
* @library /test/lib /
* @run driver ir_framework.examples.IRExample
*/
/**
* Multiple @IR rules can be specified at @Test methods. The framework performs a regex based match on the PrintIdeal
* and PrintOptoAssembly of the run test VM. Some default string regexes for IR nodes are defined in the framework
* IRNode class. There are two kinds of checks:
* <ul>
* <li><p>{@link IR#failOn}: One or more (IR node) regexes that are not allowed to occur in the IR (neither in
* PrintIdeal nor in PrintOptoAssembly)</li>
* <li><p>{@link IR#counts}: One or more regexes-count pairs specifies how often an (IR node) regex must be found in
* PrintIdeal and PrintOptoAssembly.</li>
* </ul>
* <p>
*
* One might also want to restrict the application of certain @IR rules depending on the used flags in the test VM.
* These could be flags defined by the user or by JTreg. In the latter case, the flags must be whitelisted (see
* {@link TestFramework}) most of them should not have an impact on the IR except for different GC flags which should
* be considered) to enable a verification by the framework (see below). The @IR rules thus have an option to restrict
* their application:
* <ul>
* <li><p>{@link IR#applyIf}: Only apply a rule if a flag has a certain value</li>
* <li><p>{@link IR#applyIfNot}: Only apply a rule if a flag has NOT a certain value (inverse of applyIf)</li>
* <li><p>{@link IR#applyIfAnd}: Only apply a rule if all flags have the specified value</li>
* <li><p>{@link IR#applyIfOr}: Only apply a rule if at least one flag has the specified value</li>
* </ul>
* <p>
*
* The framework, however, does not perform the verification if:
* <ul>
* <li><p>-DVerifyIR=false is used</li>
* <li><p>The test is run with a non-debug build</li>
* <li><p>-Xcomp, -Xint, -XX:-UseCompile, -XX:CompileThreshold, -DFlipC1C2=true, or -DExcludeRandom=true are used.</li>
* <li><p>JTreg specifies non-whitelisted flags as VM and/or Javaoptions (could change the IR in an unexpected way)</li>
* </ul>
*
* @see IR
* @see Test
* @see TestFramework
*/
// This test is expected to fail when run with JTreg.
public class IRExample {
int iFld, iFld2, iFld3;
public static void main(String[] args) {
TestFramework.run(); // First run tests from IRExample
try {
TestFramework.run(FailingExamples.class); // Secondly, run tests from FailingExamples
} catch (IRViolationException e) {
// Expected. Check stderr/stdout to see how IR failures are reported (always printed, regardless if
// exception is thrown or not). Uncomment the "throw" statement below to get a completely failing test.
//throw e;
}
}
// Rules with failOn constraint which all pass
@Test
@IR(failOn = IRNode.LOAD) // 1 default regex
@IR(failOn = {IRNode.LOAD, IRNode.LOOP}) // 2 default regexes
@IR(failOn = {IRNode.LOAD, "some regex that does not occur"}) // 1 default regex and a user-defined regex
// Rule with special configurable default regexes. All regexes with a "_OF" postfix in IR node expect a
// second string specifying an additional required information.
@IR(failOn = {IRNode.STORE_OF_FIELD, "iFld2", IRNode.LOAD, IRNode.STORE_OF_CLASS, "Foo"})
// Only apply this rule if the VM flag UseZGC is true
@IR(applyIf = {"UseZGC", "true"}, failOn = IRNode.LOAD)
// We can also use comparators (<, <=, >, >=, !=, =) to restrict the rules.
// This rule is only applied if the loop unroll limit is 10 or greater.
@IR(applyIf = {"LoopUnrollLimit", ">= 10"}, failOn = IRNode.LOAD)
public void goodFailOn() {
iFld = 42; // No load, no loop, no store to iFld2, no store to class Foo
}
// Rules with counts constraint which all pass
@Test
@IR(counts = {IRNode.STORE, "2"}) // 1 default regex
@IR(counts = {IRNode.LOAD, "0"}) // equivalent to failOn = IRNode.LOAD
@IR(counts = {IRNode.STORE, "2",
IRNode.LOAD, "0"}) // 2 default regexes
@IR(counts = {IRNode.STORE, "2",
"some regex that does not occur", "0"}) // 1 default regex and a user-defined regex
// Rule with special configurable default regexes. All regexes with a "_OF" postfix in IR node expect a
// second string specifying an additional required information.
@IR(counts = {IRNode.STORE_OF_FIELD, "iFld", "1",
IRNode.STORE, "2",
IRNode.STORE_OF_CLASS, "IRExample", "2"})
public void goodCounts() {
iFld = 42; // No load, store to iFld in class IRExample
iFld2 = 42; // No load, store to iFld2 in class IRExample
}
// @IR rules can also specify both type of checks in the same rule
@Test
@IR(failOn = {IRNode.ALLOC,
IRNode.LOOP},
counts = {IRNode.LOAD, "2",
IRNode.LOAD_OF_FIELD, "iFld2", "1",
IRNode.LOAD_OF_CLASS, "IRExample", "2"})
public void mixFailOnAndCounts() {
iFld = iFld2;
iFld2 = iFld3;
}
}
class FailingExamples {
int iFld2, iFld3;
IRExample irExample = new IRExample();
// Rules with failOn constraint which all fail.
@Test
@IR(failOn = IRNode.STORE)
@IR(failOn = {IRNode.STORE, IRNode.LOOP}) // LOOP regex not found but STORE regex, letting the rule fail
@IR(failOn = {IRNode.LOOP, IRNode.STORE}) // Order does not matter
@IR(failOn = {IRNode.STORE, IRNode.LOAD}) // STORE and LOAD regex found, letting the rule fail
@IR(failOn = {"LoadI"}) // LoadI can be found in PrintIdeal letting the rule fail
// Store to iFld, store, and store to class IRExample, all 3 regexes found letting the rule fail
@IR(failOn = {IRNode.STORE_OF_FIELD, "iFld", IRNode.STORE, IRNode.STORE_OF_CLASS, "IRExample"})
public void badFailOn() {
irExample.iFld = iFld2; // Store to iFld in class IRExample, load from iFld2
}
// Rules with counts constraint which all fail
@Test
@IR(counts = {IRNode.STORE, "1"}) // There are 2 stores
@IR(counts = {IRNode.LOAD, "0"}) // equivalent to failOn = IRNode.LOAD, there is 1 load
@IR(counts = {IRNode.STORE, "1",
IRNode.LOAD, "1"}) // first constraint holds (there is 1 load) but 2 stores, letting this rule fail
@IR(counts = {IRNode.LOAD, "1",
IRNode.STORE, "1"}) // order does not matter
@IR(counts = {"some regex that does not occur", "1"}) // user-defined regex does not occur once
// Rule with special configurable default regexes. All regexes with a "_OF" postfix in IR node expect a
// second string specifying an additional required information.
@IR(counts = {IRNode.STORE_OF_FIELD, "iFld", "2", // Only one store to iFld
IRNode.LOAD, "2", // Only 1 load
IRNode.STORE_OF_CLASS, "Foo", "1"}) // No store to class Foo
public void badCounts() {
irExample.iFld = iFld3; // No load, store to iFld in class IRExample
iFld2 = 42; // No load, store to iFld2 in class IRExample
}
}

View File

@ -0,0 +1,5 @@
# Framework internal tests
This folder contains tests which test the functionality of the framework. These are run with JTreg and are part of tier testing. All tests are run without additional VM and Javaopts flags. These tests must pass whenever the framework is updated.
Additional testing should be performed with the converted Valhalla tests to make sure a changeset is correct (these are part of the Valhalla CI).

View File

@ -0,0 +1,427 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.*;
/*
* @test
* @requires vm.flagless
* @summary Test different access modifiers an make sure, the framework can access all methods.
* @library /test/lib /
* @run driver ir_framework.tests.TestAccessModifiers
*/
public class TestAccessModifiers {
public static void main(String[] args) {
TestFramework.run(PackagePrivate.class);
}
}
class PackagePrivate {
@Test
public void test() {
}
@Test
@Arguments(Argument.DEFAULT)
public void test2(int x) {
}
@Test
public static int staticPublicPrivate() {
return 42;
}
@Check(test = "staticPublicPrivate")
private void staticPublicPrivateCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
protected static int staticProtectedPrivate() {
return 42;
}
@Check(test = "staticProtectedPrivate")
private void staticProtectedPrivateCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
static int staticDefaultPrivate() {
return 42;
}
@Check(test = "staticDefaultPrivate")
private void staticDefaultPrivateCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
private static int staticPrivatePrivate() {
return 42;
}
@Check(test = "staticPrivatePrivate")
private void staticPrivatePrivateCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
public static int staticPublicDefault() {
return 42;
}
@Check(test = "staticPublicDefault")
void staticPublicDefaultCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
protected static int staticProtectedDefault() {
return 42;
}
@Check(test = "staticProtectedDefault")
void staticProtectedDefaultCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
static int staticDefaultDefault() {
return 42;
}
@Check(test = "staticDefaultDefault")
void staticDefaultDefaultCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
private static int staticPrivateDefault() {
return 42;
}
@Check(test = "staticPrivateDefault")
void staticPrivateDefaultCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
public static int staticPublicProtected() {
return 42;
}
@Check(test = "staticPublicProtected")
protected void staticPublicProtectedCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
protected static int staticProtectedProtected() {
return 42;
}
@Check(test = "staticProtectedProtected")
protected void staticProtectedProtectedCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
static int staticDefaultProtected() {
return 42;
}
@Check(test = "staticDefaultProtected")
protected void staticDefaultProtectedCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
private static int staticPrivateProtected() {
return 42;
}
@Check(test = "staticPrivateProtected")
protected void staticPrivateProtectedCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
public static int staticPublicPublic() {
return 42;
}
@Check(test = "staticPublicPublic")
public void staticPublicPublicCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
protected static int staticProtectedPublic() {
return 42;
}
@Check(test = "staticProtectedPublic")
public void staticProtectedPublicCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
static int staticDefaultPublic() {
return 42;
}
@Check(test = "staticDefaultPublic")
public void staticDefaultPublicCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
private static int staticPrivatePublic() {
return 42;
}
@Check(test = "staticPrivatePublic")
public void staticPrivatePublicCheck(int retValue) {
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
static int staticDefaultPrivate2() {
return 42;
}
@Run(test = "staticDefaultPrivate2")
private void staticDefaultPrivateRun() {
int retValue = staticDefaultPrivate2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
private static int staticPrivatePrivate2() {
return 42;
}
@Run(test = "staticPrivatePrivate2")
private void staticPrivatePrivateRun() {
int retValue = staticPrivatePrivate2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
public static int staticPublicDefault2() {
return 42;
}
@Run(test = "staticPublicDefault2")
void staticPublicDefaultRun() {
int retValue = staticPublicDefault2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
protected static int staticProtectedDefault2() {
return 42;
}
@Run(test = "staticProtectedDefault2")
void staticProtectedDefaultRun() {
int retValue = staticProtectedDefault2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
static int staticDefaultDefault2() {
return 42;
}
@Run(test = "staticDefaultDefault2")
void staticDefaultDefaultRun() {
int retValue = staticDefaultDefault2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
private static int staticPrivateDefault2() {
return 42;
}
@Run(test = "staticPrivateDefault2")
void staticPrivateDefaultRun() {
int retValue = staticPrivateDefault2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
public static int staticPublicProtected2() {
return 42;
}
@Run(test = "staticPublicProtected2")
protected void staticPublicProtectedRun() {
int retValue = staticPublicProtected2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
protected static int staticProtectedProtected2() {
return 42;
}
@Run(test = "staticProtectedProtected2")
protected void staticProtectedProtectedRun() {
int retValue = staticProtectedProtected2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
static int staticDefaultProtected2() {
return 42;
}
@Run(test = "staticDefaultProtected2")
protected void staticDefaultProtectedRun() {
int retValue = staticDefaultProtected2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
private static int staticPrivateProtected2() {
return 42;
}
@Run(test = "staticPrivateProtected2")
protected void staticPrivateProtectedRun() {
int retValue = staticPrivateProtected2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
public static int staticPublicPublic2() {
return 42;
}
@Run(test = "staticPublicPublic2")
public void staticPublicPublicRun() {
int retValue = staticPublicPublic2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
protected static int staticProtectedPublic2() {
return 42;
}
@Run(test = "staticProtectedPublic2")
public void staticProtectedPublicRun() {
int retValue = staticProtectedPublic2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
static int staticDefaultPublic2() {
return 42;
}
@Run(test = "staticDefaultPublic2")
public void staticDefaultPublicRun() {
int retValue = staticDefaultPublic2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
@Test
private static int staticPrivatePublic2() {
return 42;
}
@Run(test = "staticPrivatePublic2")
public void staticPrivatePublicRun() {
int retValue = staticPrivatePublic2();
if (retValue != 42) {
throw new RuntimeException("Needs to be 42");
}
}
}

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,223 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.test.TestVM;
import java.lang.reflect.Method;
/*
* @test
* @requires vm.flagless
* @summary Test if compilation levels are used correctly in the framework.
* This test partly runs directly the test VM which normally does and should not happen in user tests.
* @library /test/lib /
* @build sun.hotspot.WhiteBox
* @run driver jdk.test.lib.helpers.ClassFileInstaller sun.hotspot.WhiteBox
* @run main/othervm -Xbootclasspath/a:. -DSkipWhiteBoxInstall=true -XX:+IgnoreUnrecognizedVMOptions -XX:+UnlockDiagnosticVMOptions
* -Xbatch -XX:+WhiteBoxAPI ir_framework.tests.TestCompLevels
*/
public class TestCompLevels {
static int[] testExecuted = new int[5];
public static void main(String[] args) throws Exception {
Method runTestsOnSameVM = TestVM.class.getDeclaredMethod("runTestsOnSameVM", Class.class);
runTestsOnSameVM.setAccessible(true);
runTestsOnSameVM.invoke(null, new Object[]{null});
for (int i = 0; i < testExecuted.length; i++) {
int value = testExecuted[i];
if (value != TestVM.WARMUP_ITERATIONS + 1) {
// Warmups + 1 compiled invocation
throw new RuntimeException("Test " + i + " was executed " + value + " times stead of "
+ TestVM.WARMUP_ITERATIONS + 1 + " times." );
}
}
TestFramework framework = new TestFramework(TestNoTiered.class);
framework.setDefaultWarmup(10).addFlags("-XX:-TieredCompilation").start();
framework = new TestFramework(TestNoTiered.class);
framework.setDefaultWarmup(10).addScenarios(new Scenario(0, "-XX:-TieredCompilation")).start();
framework = new TestFramework(TestStopAtLevel1.class);
framework.setDefaultWarmup(10).addFlags("-XX:TieredStopAtLevel=1").start();
framework = new TestFramework(TestStopAtLevel1.class);
framework.setDefaultWarmup(10).addScenarios(new Scenario(0, "-XX:TieredStopAtLevel=1")).start();
}
@Test(compLevel = CompLevel.C1_SIMPLE)
public void testC1() {
testExecuted[0]++;
}
@Check(test = "testC1", when = CheckAt.COMPILED)
public void checkTestC1(TestInfo info) {
TestFramework.assertCompiledAtLevel(info.getTest(), CompLevel.C1_SIMPLE);
}
@Test(compLevel = CompLevel.C1_LIMITED_PROFILE)
public void testC1Limited() {
testExecuted[1]++;
}
@Check(test = "testC1Limited", when = CheckAt.COMPILED)
public void checkTestLimited(TestInfo info) {
TestFramework.assertCompiledAtLevel(info.getTest(), CompLevel.C1_LIMITED_PROFILE);
}
@Test(compLevel = CompLevel.C1_FULL_PROFILE)
public void testC1Full() {
testExecuted[2]++;
}
@Check(test = "testC1Full", when = CheckAt.COMPILED)
public void checkTestC1Full(TestInfo info) {
TestFramework.assertCompiledAtLevel(info.getTest(), CompLevel.C1_FULL_PROFILE);
}
@Test(compLevel = CompLevel.C2)
public void testC2() {
testExecuted[3]++;
}
@Check(test = "testC2", when = CheckAt.COMPILED)
public void checkTestC2(TestInfo info) {
TestFramework.assertCompiledAtLevel(info.getTest(), CompLevel.C2);
}
@Test(compLevel = CompLevel.SKIP)
public void testSkip() {
testExecuted[4]++; // Executed by eventually not compiled by framework
}
}
class TestNoTiered {
@Test(compLevel = CompLevel.C1_SIMPLE)
public void level1() {
}
@Check(test = "level1")
public void check1(TestInfo info) {
TestFramework.assertNotCompiled(info.getTest()); // Never compiled without C1
}
@Test(compLevel = CompLevel.C1_LIMITED_PROFILE)
public void level2() {
}
@Check(test = "level2")
public void check2(TestInfo info) {
TestFramework.assertNotCompiled(info.getTest()); // Never compiled without C1
}
@Test(compLevel = CompLevel.C1_FULL_PROFILE)
public void level3() {
}
@Check(test = "level3")
public void check3(TestInfo info) {
TestFramework.assertNotCompiled(info.getTest()); // Never compiled without C1
}
@Test(compLevel = CompLevel.C2)
public void level4() {
}
@Check(test = "level4")
public void check4(TestInfo info) {
if (info.isWarmUp()) {
TestFramework.assertNotCompiled(info.getTest()); // Never compiled without C1
} else {
if (TestFramework.isC1Compiled(info.getTest())) {
throw new RuntimeException("Cannot be compiled with C1"); // Never compiled without C1
}
TestFramework.assertCompiledByC2(info.getTest());
}
}
@Test(compLevel = CompLevel.SKIP)
public void skip() {
}
@Check(test = "skip")
public void checkSkip(TestInfo info) {
TestFramework.assertNotCompiled(info.getTest()); // Never compiled
}
}
class TestStopAtLevel1 {
@Test(compLevel = CompLevel.C1_SIMPLE)
public int level1() {
return 34;
}
@Check(test = "level1")
public void check1(int result, TestInfo info) {
if (info.isWarmUp()) {
TestFramework.assertNotCompiled(info.getTest()); // Not compiled yet
} else {
TestFramework.assertCompiledByC1(info.getTest());
if (TestFramework.isC2Compiled(info.getTest())) {
throw new RuntimeException("Cannot be compiled by C2");
}
System.out.println("TestStopAtLevel1=" + result);
}
}
@Test(compLevel = CompLevel.C1_LIMITED_PROFILE)
public void level2() {
}
@Check(test = "level2")
public void check2(TestInfo info) {
TestFramework.assertNotCompiled(info.getTest()); // Never with level 2
}
@Test(compLevel = CompLevel.C1_FULL_PROFILE)
public void level3() {
}
@Check(test = "level3")
public void check3(TestInfo info) {
TestFramework.assertNotCompiled(info.getTest()); // Never with level 3
}
@Test(compLevel = CompLevel.C2)
public void level4() {
}
@Check(test = "level4")
public void check4(TestInfo info) {
TestFramework.assertNotCompiled(info.getTest()); // Never with level 4
}
@Test(compLevel = CompLevel.SKIP)
public void skip() {
}
@Check(test = "skip")
public void checkSkip(TestInfo info) {
TestFramework.assertNotCompiled(info.getTest()); // Never compiled
}
}

View File

@ -0,0 +1,389 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.Compiler;
import compiler.lib.ir_framework.test.TestVM;
import jdk.test.lib.Asserts;
import sun.hotspot.WhiteBox;
import java.lang.reflect.Method;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/*
* @test
* @requires vm.debug == true & vm.compMode != "Xint" & vm.compiler2.enabled & vm.flagless
* @summary Test if compilation control annotaions are handled correctly in the framework.
* This test partly runs directly the test VM which normally does and should not happen in user tests.
* @library /test/lib /
* @build sun.hotspot.WhiteBox
* @run driver jdk.test.lib.helpers.ClassFileInstaller sun.hotspot.WhiteBox
* @run main/othervm -Xbootclasspath/a:. -DSkipWhiteBoxInstall=true -XX:+IgnoreUnrecognizedVMOptions -XX:+UnlockDiagnosticVMOptions
* -Xbatch -XX:+WhiteBoxAPI ir_framework.tests.TestControls
*/
public class TestControls {
static int[] executed = new int[15];
static boolean wasExecuted = false;
static final WhiteBox WHITE_BOX = WhiteBox.getWhiteBox();
public int iFld;
public static void main(String[] args) throws Exception {
Method runTestsOnSameVM = TestVM.class.getDeclaredMethod("runTestsOnSameVM", Class.class);
runTestsOnSameVM.setAccessible(true);
runTestsOnSameVM.invoke(null, new Object[]{ null });
final int defaultIterations = TestVM.WARMUP_ITERATIONS + 1;
Asserts.assertEQ(executed[0], 1001);
Asserts.assertEQ(executed[1], 101);
Asserts.assertEQ(executed[2], 10000);
Asserts.assertEQ(executed[3], 10000);
Asserts.assertEQ(executed[4], defaultIterations);
Asserts.assertEQ(executed[5], defaultIterations);
Asserts.assertEQ(executed[6], 5001);
Asserts.assertEQ(executed[7], 5001);
Asserts.assertEQ(executed[8], 1);
Asserts.assertEQ(executed[9], 5000);
Asserts.assertEQ(executed[10], 1);
Asserts.assertEQ(executed[11], 2);
Asserts.assertEQ(executed[12], 1);
Asserts.assertEQ(executed[13], 1);
Asserts.assertFalse(wasExecuted);
final long started = System.currentTimeMillis();
long elapsed = 0;
Method overloadDouble = TestControls.class.getDeclaredMethod("overload", double.class);
Method overloadInt = TestControls.class.getDeclaredMethod("overload", int.class);
while (!(TestFramework.isC2Compiled(overloadInt) && TestFramework.isCompiledAtLevel(overloadDouble, CompLevel.C1_LIMITED_PROFILE)) && elapsed < 5000) {
elapsed = System.currentTimeMillis() - started;
}
TestFramework.assertCompiledAtLevel(TestControls.class.getDeclaredMethod("overload", double.class), CompLevel.C1_LIMITED_PROFILE);
TestFramework.assertCompiledByC2(TestControls.class.getDeclaredMethod("overload", int.class));
TestFramework framework = new TestFramework(ClassInitializerTest.class);
framework.addFlags("-XX:+PrintCompilation").addHelperClasses(ClassInitializerHelper.class).start();
String output = TestFramework.getLastTestVMOutput();
Pattern p = Pattern.compile("4.*ClassInitializerTest::<clinit>");
Matcher m = p.matcher(output);
Asserts.assertTrue(m.find());
p = Pattern.compile("2.*ClassInitializerHelper::<clinit>");
m = p.matcher(output);
Asserts.assertTrue(m.find());
new TestFramework(TestWarmup.class).setDefaultWarmup(500).start();
TestFramework.run(ExplicitSkip.class);
}
@Test
@Warmup(1000)
public void test1() {
executed[0]++;
}
@Check(test = "test1")
public void check1(TestInfo info) {
if (executed[0] <= 1000) {
Asserts.assertTrue(info.isWarmUp());
} else {
Asserts.assertTrue(!info.isWarmUp() && executed[0] == 1001);
TestFramework.assertCompiledByC2(info.getTest());
}
}
@Test
@Warmup(100)
public void test2() {
executed[1]++;
}
@Check(test = "test2", when = CheckAt.COMPILED)
public void check2(TestInfo info) {
Asserts.assertTrue(!info.isWarmUp() && executed[1] == 101);
TestFramework.assertCompiledByC2(info.getTest());
}
@Test
public void overload() {
executed[4]++;
}
@ForceCompile
@DontInline
public static void overload(int i) {
wasExecuted = true;
}
@ForceCompile(CompLevel.C1_LIMITED_PROFILE)
@ForceInline
public static void overload(double i) {
wasExecuted = true;
}
@Check(test = "overload")
public void checkOverload() {
executed[5]++;
}
@Test
public void testDontCompile() {
executed[2]++;
}
@DontCompile
public static void dontCompile() {
executed[3]++;
}
@Run(test = "testDontCompile", mode = RunMode.STANDALONE)
public void runTestDontCompile() throws NoSuchMethodException {
for (int i = 0; i < 10000; i++) {
dontCompile(); // Should not compile this method
testDontCompile();
}
TestFramework.assertNotCompiled(TestControls.class.getDeclaredMethod("dontCompile"));
}
@Test
public void testCompileAtLevel1() {
executed[6]++;
}
@DontCompile(Compiler.ANY)
public static void dontCompile2() {
executed[7]++;
}
@Run(test = "testCompileAtLevel1")
@Warmup(5000)
public void runTestDontCompile2(RunInfo info) throws NoSuchMethodException {
dontCompile2();
testCompileAtLevel1();
if (!info.isWarmUp()) {
executed[8]++;
int compLevel = WHITE_BOX.getMethodCompilationLevel(TestControls.class.getDeclaredMethod("dontCompile2"), false);
Asserts.assertLessThan(compLevel, CompLevel.C1_LIMITED_PROFILE.getValue());
} else {
executed[9]++;
}
}
@Test
@Warmup(0)
public void noWarmup() {
executed[10]++;
}
@Test
public void noWarmup2() {
executed[11]++;
}
@Run(test = "noWarmup2")
@Warmup(0)
public void runNoWarmup2(RunInfo info) {
noWarmup2();
noWarmup2();
Asserts.assertTrue(!info.isWarmUp());
executed[12]++;
}
@Test
public void testCompilation() {
wasExecuted = true;
}
@DontCompile(Compiler.ANY)
public void dontCompileAny() {
for (int i = 0; i < 10; i++) {
iFld = i;
}
}
@DontCompile(Compiler.C1)
public void dontCompileC1() {
for (int i = 0; i < 10; i++) {
iFld = 3;
}
}
@DontCompile(Compiler.C2)
public void dontCompileC2(int x, boolean b) {
for (int i = 0; i < 10; i++) {
iFld = x;
}
}
// Default is C2.
@ForceCompile
public void forceCompileDefault() {
wasExecuted = true;
}
// ANY maps to C2.
@ForceCompile
public void forceCompileAny() {
wasExecuted = true;
}
@ForceCompile(CompLevel.C1_SIMPLE)
public void forceCompileC1() {
wasExecuted = true;
}
@ForceCompile(CompLevel.C1_LIMITED_PROFILE)
public void forceCompileC1Limited() {
wasExecuted = true;
}
@ForceCompile(CompLevel.C1_FULL_PROFILE)
public void forceCompileC1Full() {
wasExecuted = true;
}
@ForceCompile(CompLevel.C2)
public void forceCompileC2() {
wasExecuted = true;
}
@ForceCompile(CompLevel.C1_SIMPLE)
@DontCompile(Compiler.C2)
public void forceC1DontC2() {
wasExecuted = true;
}
@ForceCompile(CompLevel.C2)
@DontCompile(Compiler.C1)
public void forceC2DontC1() {
wasExecuted = true;
}
@Run(test = "testCompilation")
@Warmup(0)
public void runTestCompilation(RunInfo info) {
for (int i = 0; i < 100000; i++) {
dontCompileAny();
dontCompileC1();
dontCompileC2(i, i % 2 == 0);
}
TestFramework.assertCompiledByC2(info.getTest());
TestFramework.assertNotCompiled(info.getTestClassMethod("dontCompileAny"));
TestFramework.assertCompiledByC2(info.getTestClassMethod("dontCompileC1"));
TestFramework.assertCompiledByC1(info.getTestClassMethod("dontCompileC2", int.class, boolean.class));
TestFramework.assertCompiledAtLevel(info.getTestClassMethod("forceCompileDefault"), CompLevel.C2);
TestFramework.assertCompiledAtLevel(info.getTestClassMethod("forceCompileAny"), CompLevel.C2);
TestFramework.assertCompiledAtLevel(info.getTestClassMethod("forceCompileC2"), CompLevel.C2);
TestFramework.assertCompiledAtLevel(info.getTestClassMethod("forceCompileC1"), CompLevel.C1_SIMPLE);
TestFramework.assertCompiledAtLevel(info.getTestClassMethod("forceCompileC1Limited"), CompLevel.C1_LIMITED_PROFILE);
TestFramework.assertCompiledAtLevel(info.getTestClassMethod("forceCompileC1Full"), CompLevel.C1_FULL_PROFILE);
TestFramework.assertCompiledAtLevel(info.getTestClassMethod("forceC1DontC2"), CompLevel.C1_SIMPLE);
TestFramework.assertCompiledAtLevel(info.getTestClassMethod("forceC2DontC1"), CompLevel.C2);
executed[13]++;
}
}
@ForceCompileClassInitializer
class ClassInitializerTest {
static int i;
static Object o;
static {
i = 3;
o = new Object();
}
@Test
public void test() {}
}
@ForceCompileClassInitializer(CompLevel.C1_LIMITED_PROFILE)
class ClassInitializerHelper {
static int i;
static {
i = 3;
}
}
class TestWarmup {
int iFld;
int iFld2;
int iFldCheck;
int iFldCheck2;
@Test
@Warmup(200)
public void test() {
iFld++;
}
@Test
public void test2() {
iFld2++;
}
@Check(test = "test")
public void checkTest(TestInfo info) {
iFldCheck++;
if (iFldCheck != iFld) {
throw new RuntimeException(iFld + " must be equal " + iFldCheck);
}
if (!info.isWarmUp()) {
if (iFld != 201) {
throw new RuntimeException("Must be 201 but was " + iFld);
}
}
}
@Check(test = "test2")
public void checkTest2(TestInfo info) {
iFldCheck2++;
if (iFldCheck2 != iFld2) {
throw new RuntimeException(iFld2 + " must be equal " + iFldCheck2);
}
if (!info.isWarmUp()) {
if (iFld2 != 501) {
throw new RuntimeException("Must be 501 but was " + iFld2);
}
}
}
}
class ExplicitSkip {
int iFld;
// Test skipped and thus also no IR verification should be done.
@Test(compLevel = CompLevel.SKIP)
@IR(counts = {IRNode.STORE_I, "1"})
public int test(int x) {
iFld = x;
return x;
}
@Run(test = "test")
public void run(RunInfo info) {
test(34);
}
}

View File

@ -0,0 +1,75 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.Test;
import compiler.lib.ir_framework.TestFramework;
/*
* @test
* @requires vm.debug == true & vm.compMode != "Xint" & vm.compiler2.enabled & vm.flagless
* @summary Sanity test remaining framework property flags.
* @library /test/lib /
* @run main/othervm -DFlipC1C2=true ir_framework.tests.TestDFlags
* @run main/othervm -DExcludeRandom=true ir_framework.tests.TestDFlags
* @run main/othervm -DVerifyVM=true ir_framework.tests.TestDFlags
* @run main/othervm -DDumpReplay=true ir_framework.tests.TestDFlags
* @run main/othervm -DVerbose=true ir_framework.tests.TestDFlags
* @run main/othervm -DShuffleTests=false ir_framework.tests.TestDFlags
* @run main/othervm -DReproduce=true ir_framework.tests.TestDFlags
* @run main/othervm -DReportStdout=true ir_framework.tests.TestDFlags
* @run main/othervm -DGCAfter=true ir_framework.tests.TestDFlags
* @run main/othervm -DPrintTimes=true ir_framework.tests.TestDFlags
* @run main/othervm -DVerifyIR=false ir_framework.tests.TestDFlags
*/
public class TestDFlags {
public static void main(String[] args) {
TestFramework.run();
}
@Test
public int c1() {
return 34;
}
@Test
public void c2() {
for (int i = 0; i < 100; i++) {
}
}
@Test
public void c2_2() {
for (int i = 0; i < 100; i++) {
}
}
@Test
public void c2_3() {
for (int i = 0; i < 100; i++) {
}
}
}

View File

@ -0,0 +1,89 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.*;
import jdk.test.lib.Asserts;
import jdk.test.lib.Utils;
import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.process.ProcessTools;
import sun.hotspot.WhiteBox;
/*
* @test
* @requires vm.debug == true & vm.flagless
* @summary Test -DIgnoreCompilerControls property flag.
* @library /test/lib /
* @run driver ir_framework.tests.TestDIgnoreCompilerControls
*/
public class TestDIgnoreCompilerControls {
public static void main(String[] args) throws Exception {
if (args.length != 0) {
TestFramework.run();
} else {
OutputAnalyzer oa = run("true");
oa.shouldHaveExitValue(0);
oa = run("false");
oa.shouldNotHaveExitValue(0);
Asserts.assertTrue(oa.getOutput().contains("fail run"), "did not find run: " + oa.getOutput());
Asserts.assertTrue(oa.getOutput().contains("fail check"), "did not find check" + oa.getOutput());
}
}
private static OutputAnalyzer run(String flagValue) throws Exception {
OutputAnalyzer oa;
ProcessBuilder process = ProcessTools.createJavaProcessBuilder(
"-Dtest.class.path=" + Utils.TEST_CLASS_PATH, "-Dtest.jdk=" + Utils.TEST_JDK,
"-Dtest.vm.opts=-DIgnoreCompilerControls=" + flagValue,
"ir_framework.tests.TestDIgnoreCompilerControls", flagValue);
oa = ProcessTools.executeProcess(process);
return oa;
}
@Test
public void test() { }
@Run(test = "test")
@Warmup(10000)
public void run(RunInfo info) throws NoSuchMethodException {
if (!info.isWarmUp()) {
// Should be compiled with -DIgnoreCompilerControls=true
Asserts.assertTrue(WhiteBox.getWhiteBox().isMethodCompiled(getClass().getDeclaredMethod("run", RunInfo.class)), "fail run");
}
}
@Test
@Warmup(10000)
public void test2() {}
@Check(test = "test2")
public void check(TestInfo info) throws NoSuchMethodException {
if (!info.isWarmUp()) {
// Should be compiled with -DIgnoreCompilerControls=true
Asserts.assertTrue(WhiteBox.getWhiteBox().isMethodCompiled(getClass().getDeclaredMethod("check", TestInfo.class)), "fail check");
}
}
}

View File

@ -0,0 +1,96 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.Scenario;
import compiler.lib.ir_framework.Test;
import compiler.lib.ir_framework.TestFramework;
import compiler.lib.ir_framework.driver.TestVMException;
import jdk.test.lib.Asserts;
import jdk.test.lib.Utils;
import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.process.ProcessTools;
/*
* @test
* @requires vm.debug == true & vm.flagless
* @summary Test -DScenarios property flag. Run with othervm which should not be done when writing tests using the framework.
* @library /test/lib /
* @run main/othervm -DScenarios=1,5,10 ir_framework.tests.TestDScenarios test
* @run main/othervm -DScenarios=1,4 ir_framework.tests.TestDScenarios test
* @run main/othervm -DScenarios=3,4,9 ir_framework.tests.TestDScenarios test
* @run driver ir_framework.tests.TestDScenarios test2
* @run driver ir_framework.tests.TestDScenarios
*/
public class TestDScenarios {
public static void main(String[] args) throws Exception {
if (args.length > 0) {
switch (args[0]) {
case "test" -> {
Scenario s1 = new Scenario(1);
Scenario s2 = new Scenario(5);
Scenario s3 = new Scenario(10);
Scenario bad = new Scenario(0, "-Flagdoesnotexist"); // not executed
new TestFramework().addScenarios(bad, s1, s2, s3).start();
}
case "test2" -> {
try {
TestFramework.run(DScenariosBad.class);
throw new RuntimeException("should not reach");
} catch (TestVMException e) {
System.out.println(e.getExceptionInfo());
Asserts.assertTrue(e.getExceptionInfo().contains("Expected DScenariosBad exception"));
}
}
default -> {
// Invalid -DScenarios set and thus exception thrown when Scenario class is statically initialized.
Scenario s = new Scenario(3);
throw new RuntimeException("should not reach");
}
}
} else {
// Test invalid -DScenario flag.
OutputAnalyzer oa;
ProcessBuilder process = ProcessTools.createJavaProcessBuilder(
"-Dtest.jdk=" + Utils.TEST_JDK, "-DScenarios=a,1,b,10",
"ir_framework.tests.TestDScenarios", " test3");
oa = ProcessTools.executeProcess(process);
oa.shouldNotHaveExitValue(0);
System.out.println(oa.getOutput());
Asserts.assertTrue(oa.getOutput().contains("TestRunException: Provided a scenario index"));
}
}
@Test
public void test() {
}
}
class DScenariosBad {
@Test
public void test() {
throw new RuntimeException("Expected DScenariosBad exception");
}
}

View File

@ -0,0 +1,164 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.Check;
import compiler.lib.ir_framework.Run;
import compiler.lib.ir_framework.Test;
import compiler.lib.ir_framework.TestFramework;
import compiler.lib.ir_framework.driver.TestVMException;
import compiler.lib.ir_framework.shared.NoTestsRunException;
import jdk.test.lib.Asserts;
import jdk.test.lib.Utils;
import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.process.ProcessTools;
/*
* @test
* @requires vm.flagless
* @summary Test -DTest and -DExclude property flag.
* @library /test/lib /
* @run driver ir_framework.tests.TestDTestAndExclude
*/
public class TestDTestAndExclude {
public static void main(String[] args) throws Exception {
if (args.length == 0) {
run("good1,good2", "", "good");
run("good1,good2", "bad1", "good");
run("good1,bad1", "bad1", "good");
run("good1,bad1", "bad1,good", "good");
run("good3,bad2", "bad1,bad2", "good");
run("goodMulti1,goodMulti2", "", "good");
run("bad1,good1", "", "bad1");
run("bad1,good1", "good1", "bad1");
run("bad1,good1", "asdf", "bad1");
run("bad2,good1", "", "runBadSingle");
run("bad2", "runBadSingle", "runBadSingle");
run("badMulti1,badMulti2", "", "runBadMulti");
run("badMulti1", "", "runBadMulti");
run("badMulti1", "badMulti2", "runBadMulti");
run("badMulti2", "badMulti1", "runBadMulti");
run("runBadSingle", "", "empty");
run("runBadMulti", "", "empty");
run("asdf", "", "empty");
run("", "good1,good2,good3,bad1,bad2,goodMulti1,goodMulti2,badMulti1,badMulti2", "empty");
run("asdf", "good1,good2,good3,bad1,bad2,goodMulti1,goodMulti2,badMulti1,badMulti2", "empty");
run("bad1", "bad1", "empty");
run("good1", "asdf,good,good1", "empty");
} else {
switch (args[0]) {
case "good" -> TestFramework.run();
case "bad1", "runBadMulti", "runBadSingle" -> {
try {
TestFramework.run();
throw new RuntimeException("should not reach");
} catch (TestVMException e) {
Asserts.assertTrue(e.getExceptionInfo().contains("expected " + args[0] + " exception"));
}
}
case "empty" -> {
try {
TestFramework.run();
throw new RuntimeException("should not reach");
} catch (NoTestsRunException e) {
// Expected
}
}
default -> throw new RuntimeException("should not reach");
}
}
}
/**
* Create a VM and simulate as if it was a driver VM spawned by JTreg that has -DTest/DExclude set as VM or Javaopts
*/
protected static void run(String dTest, String dExclude, String arg) throws Exception {
System.out.println("Run -DTest=" + dTest + " -DExclude=" + dExclude + " arg=" + arg);
OutputAnalyzer oa;
ProcessBuilder process = ProcessTools.createJavaProcessBuilder(
"-Dtest.class.path=" + Utils.TEST_CLASS_PATH, "-Dtest.jdk=" + Utils.TEST_JDK,
"-Dtest.vm.opts=-DTest=" + dTest + " -DExclude=" + dExclude,
"ir_framework.tests.TestDTestAndExclude", arg);
oa = ProcessTools.executeProcess(process);
oa.shouldHaveExitValue(0);
}
@Test
public void good1() { }
@Test
public void good2() { }
@Check(test = "good2")
public void check2() {}
@Test
public void bad1() {
throw new RuntimeException("expected bad1 exception");
}
@Test
public void good3() {}
@Test
public void goodMulti1() {}
@Test
public void goodMulti2() {}
@Run(test = "good3")
public void runGoodSingle() {
good3();
}
@Run(test = {"goodMulti1", "goodMulti2"})
public void runGoodMulti() {
goodMulti1();
goodMulti2();
}
@Test
public void bad2() {
}
@Test
public void badMulti1() {
}
@Test
public void badMulti2() {
}
@Run(test = "bad2")
public void runBadSingle() {
throw new RuntimeException("expected runBadSingle exception");
}
@Run(test = {"badMulti1", "badMulti2"})
public void runBadMulti() {
throw new RuntimeException("expected runBadMulti exception");
}
}

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,405 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.driver.IRViolationException;
import compiler.lib.ir_framework.shared.TestRunException;
import jdk.test.lib.Asserts;
import java.util.Arrays;
/*
* @test
* @requires vm.debug == true & vm.compMode != "Xint" & vm.compiler2.enabled & vm.flagless
* @summary Test different custom run tests.
* @library /test/lib /
* @run driver ir_framework.tests.TestRunTests
*/
public class TestRunTests {
public static void main(String[] args) {
TestFramework.run();
try {
TestFramework.run(BadStandalone.class);
throw new RuntimeException("Should not reach");
} catch (IRViolationException e) {
String[] matches = { "test(int)", "test2(int)", "Failed IR Rules (2)"};
Arrays.stream(matches).forEach(m -> Asserts.assertTrue(e.getExceptionInfo().contains(m)));
Asserts.assertEQ(e.getExceptionInfo().split("STANDALONE mode", -1).length - 1, 2);
}
new TestFramework(SkipCompilation.class).addFlags("-XX:-UseCompiler").start();
new TestFramework(SkipCompilation.class).addFlags("-Xint").start();
new TestFramework(SkipC2Compilation.class).addFlags("-XX:TieredStopAtLevel=1").start();
new TestFramework(SkipC2Compilation.class).addFlags("-XX:TieredStopAtLevel=2").start();
new TestFramework(SkipC2Compilation.class).addFlags("-XX:TieredStopAtLevel=3").start();
}
public int iFld;
@Test
@IR(counts = {IRNode.STORE_I, "1"})
public int test1(int x) {
iFld = x;
return x;
}
@Test
@IR(counts = {IRNode.STORE_I, "1"})
public int test2(int y) {
iFld = y;
return y;
}
@Run(test = {"test1", "test2"})
public void run(RunInfo info) {
test1(23);
test2(42);
Asserts.assertTrue(info.isC2CompilationEnabled());
if (!info.isWarmUp()) {
TestFramework.assertCompiledByC2(info.getTest("test1"));
TestFramework.assertCompiledByC2(info.getTest("test2"));
}
}
@Test
@IR(counts = {IRNode.STORE_I, "1"})
public int test3(int x) {
iFld = x;
return x;
}
@Run(test = "test3")
public void run2(RunInfo info) {
Asserts.assertTrue(info.isC2CompilationEnabled());
test3(42);
if (!info.isWarmUp()) {
TestFramework.assertCompiledByC2(info.getTest());
try {
info.getTest("test2");
throw new RuntimeException("should not reach");
} catch (TestRunException e) {
// Excepted, do not call this method for single associated test.
}
try {
info.isTestC1Compiled("test2");
throw new RuntimeException("should not reach");
} catch (TestRunException e) {
// Excepted, do not call this method for single associated test.
}
try {
info.isTestC2Compiled("test2");
throw new RuntimeException("should not reach");
} catch (TestRunException e) {
// Excepted, do not call this method for single associated test.
}
try {
info.isTestCompiledAtLevel("test2", CompLevel.C2);
throw new RuntimeException("should not reach");
} catch (TestRunException e) {
// Excepted, do not call this method for single associated test.
}
}
}
@Test
@IR(counts = {IRNode.STORE_I, "1"})
public int test4(int x) {
iFld = x;
return x;
}
@Run(test = "test4", mode = RunMode.STANDALONE)
public void run3(RunInfo info) {
for (int i = 0; i < 2000; i++) {
test4(i);
}
}
@Test
@IR(counts = {IRNode.STORE_I, "1"})
public int test5(int x) {
iFld = x;
return x;
}
@Test(compLevel = CompLevel.WAIT_FOR_COMPILATION)
@IR(counts = {IRNode.STORE_I, "1"})
public int test6(int y) {
iFld = y;
return y;
}
@Run(test = {"test5", "test6"})
public void run4(RunInfo info) {
test5(23);
test6(42);
if (!info.isWarmUp()) {
TestFramework.assertCompiledByC2(info.getTest("test5"));
TestFramework.assertCompiledByC2(info.getTest("test6"));
try {
info.getTest();
throw new RuntimeException("should not reach");
} catch (TestRunException e) {
// Excepted, do not call this method for single associated test.
}
try {
info.isTestC1Compiled();
throw new RuntimeException("should not reach");
} catch (TestRunException e) {
// Excepted, do not call this method for single associated test.
}
try {
info.isTestC2Compiled();
throw new RuntimeException("should not reach");
} catch (TestRunException e) {
// Excepted, do not call this method for single associated test.
}
try {
info.isTestCompiledAtLevel(CompLevel.C2);
throw new RuntimeException("should not reach");
} catch (TestRunException e) {
// Excepted, do not call this method for single associated test.
}
}
}
@Test
@IR(counts = {IRNode.STORE_I, "1"})
public int test7(int x) {
for (int i = 0; i < 100; i++) {}
iFld = x;
return x;
}
@Test
@IR(counts = {IRNode.STORE_I, "1"})
public int test8(int x) {
for (int i = 0; i < 100; i++) {}
iFld = x;
return x;
}
@Run(test = {"test7", "test8"}, mode = RunMode.STANDALONE)
public void run5() {
for (int i = 0; i < 10000; i++) {
test7(23);
test8(42);
}
}
@Test(compLevel = CompLevel.WAIT_FOR_COMPILATION)
@Warmup(0)
public void test9() {
TestClass tmp = new TestClass();
for (int i = 0; i < 100; ++i) {
tmp.test();
}
}
static class TestClass {
public int test() {
int res = 0;
for (int i = 1; i < 20_000; ++i) {
res -= i;
}
return res;
}
}
}
class BadStandalone {
int iFld;
@Test
@IR(counts = {IRNode.STORE_I, "1"})
public int test(int x) {
iFld = x;
return x;
}
@Run(test = "test", mode = RunMode.STANDALONE)
public void run(RunInfo info) {
test(42);
}
@Test
@IR(counts = {IRNode.STORE_I, "1"})
public int test2(int x) {
iFld = x;
return x;
}
@Run(test = "test2", mode = RunMode.STANDALONE)
public void run2(RunInfo info) {
}
}
// Run with TieredStopAt=[1,3]. IR verification is skipped.
class SkipC2Compilation {
int iFld;
@Test(compLevel = CompLevel.C2)
@IR(failOn = IRNode.STORE) // Would fail but not evaluated.
public void testC2() {
iFld = 34;
}
@Check(test = "testC2")
public void checkC2(TestInfo info) {
Asserts.assertFalse(info.isC2CompilationEnabled());
Asserts.assertTrue(info.isCompilationSkipped());
}
@Test(compLevel = CompLevel.C2)
@IR(failOn = IRNode.STORE) // Would fail but not evaluated.
public void test2C2() {
iFld = 34;
}
@Run(test = "test2C2")
public void run2C2(RunInfo info) {
Asserts.assertFalse(info.isC2CompilationEnabled());
Asserts.assertTrue(info.isCompilationSkipped());
test2C2();
Asserts.assertTrue(info.isCompilationSkipped());
try {
info.isCompilationSkipped("test2C2");
throw new RuntimeException("should not reach");
} catch (TestRunException e) {
// Excepted, do not call this method for single associated test.
}
}
@Test(compLevel = CompLevel.C2)
@IR(failOn = IRNode.STORE) // Would fail but not evaluated.
public void test3C2() {
iFld = 34;
}
@Test(compLevel = CompLevel.C2)
@IR(failOn = IRNode.STORE) // Would fail but not evaluated.
public void test4C2() {
iFld = 34;
}
@Test // Level any
@IR(failOn = IRNode.STORE) // Would fail but not evaluated.
public void testAny() {
iFld = 34;
}
@Run(test = {"test3C2", "test4C2", "testAny"})
public void runMulti(RunInfo info) {
Asserts.assertFalse(info.isC2CompilationEnabled());
if (!info.isWarmUp()) {
TestFramework.assertCompiledByC1(info.getTest("testAny"));
}
Asserts.assertTrue(info.isCompilationSkipped("test3C2"));
Asserts.assertTrue(info.isCompilationSkipped("test4C2"));
Asserts.assertFalse(info.isCompilationSkipped("testAny"));
test2C2();
Asserts.assertTrue(info.isCompilationSkipped("test3C2"));
Asserts.assertTrue(info.isCompilationSkipped("test4C2"));
Asserts.assertFalse(info.isCompilationSkipped("testAny"));
try {
info.isCompilationSkipped();
throw new RuntimeException("should not reach");
} catch (TestRunException e) {
// Excepted, do not call this method for multiple associated tests.
}
}
}
// Run with -Xint and -XX:-Compiler. IR verification is skipped.
class SkipCompilation {
int iFld;
@Test(compLevel = CompLevel.C2)
@IR(failOn = IRNode.STORE) // Would fail but not evaluated.
public void testC2() {
iFld = 34;
}
@Check(test = "testC2")
public void checkC2(TestInfo info) {
Asserts.assertTrue(info.isCompilationSkipped());
Asserts.assertFalse(info.isC2CompilationEnabled());
}
@Test(compLevel = CompLevel.C2)
@IR(failOn = IRNode.STORE) // Would fail but not evaluated.
public void test2C2() {
iFld = 34;
}
@Run(test = "test2C2")
public void run2C2(RunInfo info) {
Asserts.assertFalse(info.isC2CompilationEnabled());
Asserts.assertTrue(info.isCompilationSkipped());
test2C2();
Asserts.assertTrue(info.isCompilationSkipped());
}
@Test(compLevel = CompLevel.C2)
@IR(failOn = IRNode.STORE) // Would fail but not evaluated.
public void test3C2() {
iFld = 34;
}
@Test(compLevel = CompLevel.C2)
@IR(failOn = IRNode.STORE) // Would fail but not evaluated.
public void test4C2() {
iFld = 34;
}
@Test // Level any
@IR(failOn = IRNode.STORE) // Would fail but not evaluated.
public void testAny() {
iFld = 34;
}
@Run(test = {"test3C2", "test4C2", "testAny"})
public void runMulti(RunInfo info) {
Asserts.assertFalse(info.isC2CompilationEnabled());
if (!info.isWarmUp()) {
TestFramework.assertNotCompiled(info.getTest("testAny"));
TestFramework.assertNotCompiled(info.getTest("test3C2"));
TestFramework.assertNotCompiled(info.getTest("test4C2"));
}
Asserts.assertTrue(info.isCompilationSkipped("test3C2"));
Asserts.assertTrue(info.isCompilationSkipped("test4C2"));
Asserts.assertTrue(info.isCompilationSkipped("testAny"));
test2C2();
Asserts.assertTrue(info.isCompilationSkipped("test3C2"));
Asserts.assertTrue(info.isCompilationSkipped("test4C2"));
Asserts.assertTrue(info.isCompilationSkipped("testAny"));
}
}

View File

@ -0,0 +1,86 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.Scenario;
import compiler.lib.ir_framework.Test;
import compiler.lib.ir_framework.TestFramework;
/*
* @test
* @requires vm.flagless
* @summary Sanity test the different ways to start the test framework.
* @library /test/lib /
* @run driver ir_framework.tests.TestSanity
*/
public class TestSanity {
public static void main(String[] args) {
TestFramework.run();
TestFramework.run(TestSanity.class);
TestFramework.runWithFlags("-XX:+TieredCompilation");
new TestFramework().addFlags("-XX:TLABRefillWasteFraction=51", "-XX:+UseTLAB").start();
new TestFramework(TestSanity.class).addFlags("-XX:TLABRefillWasteFraction=51", "-XX:+UseTLAB").start();
new TestFramework().addHelperClasses(HelperA.class).start();
new TestFramework(TestSanity.class).addHelperClasses(HelperA.class, HelperB.class).start();
Scenario sDefault = new Scenario(0);
Scenario s1 = new Scenario(1, "-XX:TLABRefillWasteFraction=52", "-XX:+UseTLAB");
Scenario s2 = new Scenario(2, "-XX:TLABRefillWasteFraction=53", "-XX:+UseTLAB");
new TestFramework(TestSanity.class).addScenarios(s1).start();
new TestFramework().addScenarios(s1, s2).start();
new TestFramework(TestSanity.class).addScenarios(s1, s2).start();
new TestFramework().addScenarios(sDefault, s1).start();
new TestFramework().addScenarios(sDefault, s1, s2).start();
new TestFramework(TestSanity.class).addScenarios(sDefault, s1).start();
new TestFramework(TestSanity.class).addScenarios(sDefault, s1, s2).start();
TestFramework testFramework = new TestFramework();
testFramework.start();
testFramework.addFlags("-XX:TLABRefillWasteFraction=54").start();
testFramework = new TestFramework();
testFramework.addFlags("-XX:TLABRefillWasteFraction=55").addFlags("-XX:+UseTLAB").start();
testFramework = new TestFramework();
testFramework.addHelperClasses(HelperA.class, HelperB.class).start();
testFramework = new TestFramework();
testFramework.addHelperClasses(HelperA.class, HelperB.class).addHelperClasses(HelperC.class).start();
testFramework = new TestFramework();
testFramework.addScenarios(sDefault).addScenarios(s1, s2).start();
testFramework = new TestFramework();
testFramework.addHelperClasses(HelperA.class).addScenarios(sDefault).addFlags("-XX:+UseSuperWord").start();
testFramework = new TestFramework();
testFramework.addHelperClasses(HelperA.class).addFlags("-XX:+UseSuperWord", "-XX:+UseCompiler").addScenarios(sDefault)
.addHelperClasses(HelperB.class, HelperC.class).addScenarios(s1, s2).addFlags("-XX:+TieredCompilation").start();
testFramework = new TestFramework();
testFramework.addHelperClasses(HelperA.class).addFlags("-XX:+UseSuperWord", "-XX:+UseCompiler").addScenarios(sDefault)
.addHelperClasses(HelperB.class, HelperC.class).addScenarios(s1, s2).setDefaultWarmup(200)
.addFlags("-XX:+TieredCompilation").start();
}
@Test
public void test() {}
}
class HelperA { }
class HelperB { }
class HelperC { }

View File

@ -0,0 +1,106 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.shared.TestRunException;
import jdk.test.lib.Asserts;
/*
* @test
* @requires vm.debug == true & vm.compMode != "Xint" & vm.compiler2.enabled & vm.flagless
* @summary Test scenarios with the framework.
* @library /test/lib /
* @run driver ir_framework.tests.TestScenarios
*/
public class TestScenarios {
public static void main(String[] args) {
Scenario sDefault = new Scenario(0);
Scenario s1 = new Scenario(1, "-XX:TLABRefillWasteFraction=51");
Scenario s2 = new Scenario(2, "-XX:TLABRefillWasteFraction=52");
Scenario s3 = new Scenario(3, "-XX:TLABRefillWasteFraction=53");
Scenario s3dup = new Scenario(3, "-XX:TLABRefillWasteFraction=53");
try {
new TestFramework().addScenarios(sDefault, s1, s2, s3).start();
Asserts.fail("Should not reach");
} catch (TestRunException e) {
Asserts.assertTrue(e.getMessage().contains("The following scenarios have failed: #0, #1, #3"), e.getMessage());
}
try {
new TestFramework().addScenarios(s1, s2, s3).start();
Asserts.fail("Should not reach");
} catch (TestRunException e) {
Asserts.assertTrue(e.getMessage().contains("The following scenarios have failed: #1, #3"), e.getMessage());
}
new TestFramework(ScenarioTest.class).addScenarios(s1, s2, s3).start();
try {
new TestFramework().addScenarios(s1, s3dup, s2, s3).start();
Asserts.fail("Should not reach");
} catch (RuntimeException e) {
Asserts.assertTrue(e.getMessage().contains("Cannot define two scenarios with the same index 3"), e.getMessage());
}
try {
new TestFramework(MyExceptionTest.class).addScenarios(s1, s2, s3).start();
Asserts.fail("Should not reach");
} catch (TestRunException e) {
Asserts.assertTrue(s1.getTestVMOutput().contains("Caused by: ir_framework.tests.MyScenarioException"));
Asserts.assertTrue(s2.getTestVMOutput().contains("Caused by: ir_framework.tests.MyScenarioException"));
Asserts.assertTrue(s3.getTestVMOutput().contains("Caused by: ir_framework.tests.MyScenarioException"));
} catch (Exception e) {
Asserts.fail("Should not catch other exceptions");
}
}
@Test
@IR(applyIf = {"TLABRefillWasteFraction", "64"}, counts = {IRNode.CALL, "1"})
public void failDefault() {
}
@Test
@IR(applyIf = {"TLABRefillWasteFraction", "51"}, counts = {IRNode.CALL, "1"})
@IR(applyIf = {"TLABRefillWasteFraction", "53"}, counts = {IRNode.CALL, "1"})
public void failS3() {
}
}
class ScenarioTest {
@Test
@IR(applyIf = {"TLABRefillWasteFraction", "54"}, counts = {IRNode.CALL, "1"})
public void doesNotFail() {
}
}
class MyExceptionTest {
int iFld;
@Test
@IR(failOn = IRNode.STORE) // Not evaluated due to MyScenarioException
public void test() {
iFld = 42;
throw new MyScenarioException();
}
}
class MyScenarioException extends RuntimeException {}

View File

@ -0,0 +1,157 @@
/*
* Copyright (c) 2021, 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.
*
* 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 ir_framework.tests;
import compiler.lib.ir_framework.*;
import compiler.lib.ir_framework.driver.TestVMException;
import compiler.lib.ir_framework.shared.TestFormatException;
import jdk.test.lib.Asserts;
/*
* @test
* @requires vm.flagless
* @summary Test the framework with helper classes.
* @library /test/lib /
* @run driver ir_framework.tests.TestWithHelperClasses
*/
public class TestWithHelperClasses {
public static void main(String[] args) {
new TestFramework().addHelperClasses(Helper1.class, Helper2.class).start();
try {
new TestFramework().addHelperClasses(Helper1.class).start();
shouldNotReach();
} catch (TestVMException e) {
Asserts.assertFalse(e.getExceptionInfo().contains("public static void ir_framework.tests.Helper1.foo() should have been C2 compiled"));
Asserts.assertFalse(TestFramework.getLastTestVMOutput().contains("public static void ir_framework.tests.Helper1.foo() should have been C2 compiled"));
Asserts.assertTrue(TestFramework.getLastTestVMOutput().contains("public static void ir_framework.tests.Helper2.foo() should have been C2 compiled"));
Asserts.assertTrue(e.getExceptionInfo().contains("public static void ir_framework.tests.Helper2.foo() should have been C2 compiled"));
Asserts.assertFalse(TestFramework.getLastTestVMOutput().contains("Should not be executed"));
Asserts.assertFalse(e.getExceptionInfo().contains("Should not be executed"));
}
try {
new TestFramework(BadHelperClass.class).addHelperClasses(BadHelper.class).start();
shouldNotReach();
} catch (TestFormatException e) {
Asserts.assertTrue(e.getMessage().contains("Cannot use @Test annotation in helper class"));
Asserts.assertTrue(e.getMessage().contains("Cannot use @Check annotation in helper class"));
Asserts.assertTrue(e.getMessage().contains("Cannot use @Run annotation in helper class"));
Asserts.assertTrue(e.getMessage().contains("noTestInHelper"));
Asserts.assertTrue(e.getMessage().contains("test2"));
Asserts.assertTrue(e.getMessage().contains("check2"));
Asserts.assertTrue(e.getMessage().contains("test3"));
Asserts.assertTrue(e.getMessage().contains("run3"));
}
try {
new TestFramework(TestAsHelper.class).addHelperClasses(TestAsHelper.class).start();
shouldNotReach();
} catch (TestFormatException e) {
Asserts.assertTrue(e.getMessage().contains("Cannot specify test class ir_framework.tests." +
"TestAsHelper as helper class, too"));
}
try {
new TestFramework().addHelperClasses(NestedHelper.class).start();
shouldNotReach();
} catch (TestFormatException e) {
Asserts.assertTrue(e.getMessage().contains("Nested class"));
Asserts.assertTrue(e.getMessage().contains("TestWithHelperClasses$NestedHelper inside test class"));
}
}
public static void shouldNotReach() {
throw new RuntimeException("should not reach");
}
@Test
public void test() throws NoSuchMethodException {
TestFramework.assertCompiledByC2(Helper1.class.getMethod("foo"));
TestFramework.assertCompiledByC2(Helper2.class.getMethod("foo"));
TestFramework.assertCompiledByC2(NestedHelper.class.getMethod("foo"));
TestFramework.assertCompiledByC2(StaticNestedHelper.class.getMethod("foo"));
}
class NestedHelper {
@ForceCompile(CompLevel.C2)
public void foo() {
throw new RuntimeException("Should not be executed");
}
}
static class StaticNestedHelper {
@ForceCompile(CompLevel.C2)
public void foo() {
throw new RuntimeException("Should not be executed");
}
}
}
class TestAsHelper {
@Test
public void foo() {}
}
class Helper1 {
@ForceCompile(CompLevel.C2)
public static void foo() {
throw new RuntimeException("Should not be executed");
}
}
class Helper2 {
@ForceCompile(CompLevel.C2)
public static void foo() {
throw new RuntimeException("Should not be executed");
}
}
class BadHelperClass {
@Test
public void test1() {}
}
class BadHelper {
@Test
public void noTestInHelper() {}
@Test
public void test2() {}
@Check(test = "test2")
public void check2() {}
@Test
public void test3() {}
@Run(test = "test3")
public void run3() {}
}