2021-11-26 16:21:15 +00:00
|
|
|
/*
|
|
|
|
* Copyright Amazon.com Inc. 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
|
|
|
* @bug 8275908
|
|
|
|
* @summary Quick test for the new WhiteBox methods of JDK-8275908
|
|
|
|
*
|
|
|
|
* @requires vm.compiler2.enabled & vm.compMode != "Xcomp"
|
|
|
|
*
|
|
|
|
* @library /test/lib
|
|
|
|
* @build jdk.test.whitebox.WhiteBox
|
|
|
|
* @run driver jdk.test.lib.helpers.ClassFileInstaller jdk.test.whitebox.WhiteBox
|
|
|
|
* @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI
|
2021-11-29 12:29:06 +00:00
|
|
|
* -Xbatch -XX:-UseOnStackReplacement -XX:-TieredCompilation
|
2021-11-26 16:21:15 +00:00
|
|
|
* -XX:CompileCommand=compileonly,compiler.uncommontrap.Decompile::uncommonTrap
|
|
|
|
* -XX:CompileCommand=inline,compiler.uncommontrap.Decompile*::foo
|
|
|
|
* compiler.uncommontrap.Decompile
|
|
|
|
*/
|
|
|
|
|
|
|
|
package compiler.uncommontrap;
|
|
|
|
|
|
|
|
import java.lang.reflect.Method;
|
|
|
|
|
|
|
|
import jdk.test.lib.Asserts;
|
|
|
|
import jdk.test.whitebox.WhiteBox;
|
|
|
|
|
|
|
|
public class Decompile {
|
|
|
|
|
|
|
|
private static final WhiteBox WB = WhiteBox.getWhiteBox();
|
|
|
|
// The number of deoptimizations after which a method will be made not-entrant
|
|
|
|
private static final int PerBytecodeTrapLimit = WB.getIntxVMFlag("PerBytecodeTrapLimit").intValue();
|
|
|
|
// The number of interpreter invocations after which a decompiled method will be re-compiled.
|
|
|
|
private static final int Tier0InvokeNotifyFreq = (int)Math.pow(2, WB.getIntxVMFlag("Tier0InvokeNotifyFreqLog"));
|
|
|
|
// VM builds without JVMCI like x86_32 call the bimorphic inlining trap just 'bimorphic'
|
|
|
|
// while all the other builds with JVMCI call it 'bimorphic_or_optimized_type_check'.
|
2021-11-29 12:29:06 +00:00
|
|
|
// Only builds with JVMCI have the "EnableJVMCI" flag.
|
|
|
|
private static final boolean isJVMCISupported = (WB.getBooleanVMFlag("EnableJVMCI") != null);
|
2021-11-26 16:21:15 +00:00
|
|
|
private static final String bimorphicTrapName = isJVMCISupported ? "bimorphic_or_optimized_type_check" : "bimorphic";
|
|
|
|
|
|
|
|
static class Base {
|
|
|
|
void foo() {}
|
|
|
|
}
|
|
|
|
static class X extends Base {
|
|
|
|
void foo() {}
|
|
|
|
}
|
|
|
|
static class Y extends Base {
|
|
|
|
void foo() {}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void uncommonTrap(Base t) {
|
|
|
|
t.foo();
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void printCounters(Method uncommonTrap_m, int invocations) {
|
|
|
|
System.out.println("-----------------------------------------------------------------");
|
|
|
|
System.out.println("invocations=" + invocations + " " +
|
|
|
|
"method compiled=" + WB.isMethodCompiled(uncommonTrap_m) + " " +
|
|
|
|
"decompileCount=" + WB.getMethodDecompileCount(uncommonTrap_m) + "\n" +
|
|
|
|
"trapCount=" + WB.getMethodTrapCount(uncommonTrap_m) + " " +
|
|
|
|
"trapCount(class_check)=" + WB.getMethodTrapCount(uncommonTrap_m, "class_check") + " " +
|
|
|
|
"trapCount(" + bimorphicTrapName + ")=" +
|
|
|
|
WB.getMethodTrapCount(uncommonTrap_m, bimorphicTrapName) + "\n" +
|
|
|
|
"globalDeoptCount=" + WB.getDeoptCount() + " " +
|
|
|
|
"globalDeoptCount(class_check)=" + WB.getDeoptCount("class_check", null) + " " +
|
|
|
|
"globalDeoptCount(" + bimorphicTrapName + ")=" +
|
|
|
|
WB.getDeoptCount(bimorphicTrapName, null));
|
|
|
|
System.out.println("-----------------------------------------------------------------");
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void check(Method uncommonTrap_m, int invocations, boolean isCompiled, int decompileCount,
|
|
|
|
int trapCount, int trapCountClassCheck, int trapCountBimorphic,
|
|
|
|
int deoptCount, int deoptCountClassCheck, int deoptCountBimorphic) {
|
|
|
|
|
|
|
|
printCounters(uncommonTrap_m, invocations);
|
|
|
|
|
|
|
|
Asserts.assertEQ(isCompiled, WB.isMethodCompiled(uncommonTrap_m),
|
|
|
|
"Wrong compilation status.");
|
|
|
|
Asserts.assertEQ(decompileCount, WB.getMethodDecompileCount(uncommonTrap_m),
|
|
|
|
"Wrong number of decompilations.");
|
|
|
|
Asserts.assertEQ(trapCount, WB.getMethodTrapCount(uncommonTrap_m),
|
|
|
|
"Wrong number of traps.");
|
|
|
|
Asserts.assertEQ(trapCountClassCheck, WB.getMethodTrapCount(uncommonTrap_m, "class_check"),
|
|
|
|
"Wrong number of traps.");
|
|
|
|
Asserts.assertEQ(trapCountBimorphic, WB.getMethodTrapCount(uncommonTrap_m, bimorphicTrapName),
|
|
|
|
"Wrong number of traps.");
|
|
|
|
Asserts.assertEQ(deoptCount, WB.getDeoptCount(),
|
|
|
|
"Wrong number of deoptimizations.");
|
|
|
|
Asserts.assertEQ(deoptCountClassCheck, WB.getDeoptCount("class_check", null),
|
|
|
|
"Wrong number of class_check deoptimizations.");
|
|
|
|
Asserts.assertEQ(deoptCountBimorphic, WB.getDeoptCount(bimorphicTrapName, null),
|
|
|
|
"Wrong number of " + bimorphicTrapName + "deoptimizations.");
|
|
|
|
}
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
|
|
|
|
|
|
// Get a handle of the test method for usage with the WhiteBox API.
|
|
|
|
Method uncommonTrap_m = Decompile.class
|
|
|
|
.getDeclaredMethod("uncommonTrap", new Class[] { Base.class });
|
|
|
|
|
|
|
|
int invocations = 0;
|
|
|
|
Base b = new Base();
|
|
|
|
// This is a little tricky :) We have to define 'x' already here otherwise
|
|
|
|
// the class 'X' won't be loaded and 'uncommonTrap()' will be compiled without
|
|
|
|
// a class check but a CHA dependency that class 'B' has no subtypes.
|
|
|
|
X x = new X();
|
|
|
|
Y y = new Y();
|
|
|
|
|
|
|
|
// Warmup and compile with an object of type 'Base' as receiver, but don't invoke compiled code.
|
|
|
|
while(!WB.isMethodCompiled(uncommonTrap_m)) {
|
|
|
|
invocations++;
|
|
|
|
uncommonTrap(b);
|
|
|
|
}
|
|
|
|
check(uncommonTrap_m, invocations, true /* is_compiled */, 0 /* decompileCount */,
|
|
|
|
0 /* trapCount */, 0 /* trapCountClassCheck */, 0 /* trapCountBimorphic */,
|
|
|
|
0 /* deoptCount */, 0 /* deoptCountClassCheck */, 0 /* deoptCountBimorphic */);
|
|
|
|
|
|
|
|
// Invoke compiled code 'PerBytecodeTrapLimit' times with an receiver object of type 'X'.
|
|
|
|
// This should deoptimize 'PerBytecodeTrapLimit' times and finally decompile the method.
|
|
|
|
for (int i = 0; i < PerBytecodeTrapLimit; i++) {
|
|
|
|
invocations++;
|
|
|
|
uncommonTrap(x);
|
|
|
|
}
|
|
|
|
check(uncommonTrap_m, invocations, false /* is_compiled */, 1 /* decompileCount */,
|
|
|
|
PerBytecodeTrapLimit /* trapCount */, PerBytecodeTrapLimit /* trapCountClassCheck */, 0 /* trapCountBimorphic */,
|
|
|
|
PerBytecodeTrapLimit /* deoptCount */, PerBytecodeTrapLimit /* deoptCountClassCheck */, 0 /* deoptCountBimorphic */);
|
|
|
|
|
|
|
|
// Invoke the method 'Tier0InvokeNotifyFreq' more times with an receiver object of type 'X'.
|
|
|
|
// This should re-compile the method again with bimorphic inlining for receiver types 'Base' and 'X'.
|
|
|
|
for (int i = 0; i < Tier0InvokeNotifyFreq; i++) {
|
|
|
|
invocations++;
|
|
|
|
uncommonTrap(x);
|
|
|
|
}
|
|
|
|
check(uncommonTrap_m, invocations, true /* is_compiled */, 1 /* decompileCount */,
|
|
|
|
PerBytecodeTrapLimit /* trapCount */, PerBytecodeTrapLimit /* trapCountClassCheck */, 0 /* trapCountBimorphic */,
|
|
|
|
PerBytecodeTrapLimit /* deoptCount */, PerBytecodeTrapLimit /* deoptCountClassCheck */, 0 /* deoptCountBimorphic */);
|
|
|
|
|
|
|
|
// Invoke compiled code 'PerBytecodeTrapLimit' times with an receiver object of type 'Y'.
|
|
|
|
// This should deoptimize 'PerBytecodeTrapLimit' times and finally decompile the method.
|
|
|
|
for (int i = 0; i < PerBytecodeTrapLimit; i++) {
|
|
|
|
invocations++;
|
|
|
|
uncommonTrap(y);
|
|
|
|
}
|
|
|
|
check(uncommonTrap_m, invocations, false /* is_compiled */, 2 /* decompileCount */,
|
|
|
|
2*PerBytecodeTrapLimit /* trapCount */, PerBytecodeTrapLimit /* trapCountClassCheck */, PerBytecodeTrapLimit /* trapCountBimorphic */,
|
|
|
|
2*PerBytecodeTrapLimit /* deoptCount */, PerBytecodeTrapLimit /* deoptCountClassCheck */, PerBytecodeTrapLimit /* deoptCountBimorphic */);
|
|
|
|
}
|
|
|
|
}
|