/* * 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. */ /* * @test * @bug 8254108 * @library / /test/lib * @summary Testing of ciReplay with incremental inlining. * @requires vm.flightRecorder != true & vm.compMode != "Xint" & vm.compMode != "Xcomp" & vm.debug == true & vm.compiler2.enabled * @modules java.base/jdk.internal.misc * @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 * compiler.ciReplay.TestIncrementalInlining */ package compiler.ciReplay; import jdk.test.lib.Asserts; import jdk.test.lib.Utils; import jdk.test.whitebox.WhiteBox; import java.io.IOException; import java.nio.file.Files; import java.nio.file.Path; import java.nio.file.Paths; import java.nio.file.StandardOpenOption; import java.util.List; import java.util.regex.Matcher; import java.util.regex.Pattern; public class TestIncrementalInlining extends InliningBase { private List inlineesNormal; private List inlineesReplay; public static void main(String[] args) { new TestIncrementalInlining(); } TestIncrementalInlining() { super(IncrementalInliningTest.class); // Enable Whitebox access for test VM. commandLineNormal.add("-Dtest.jdk=" + Utils.TEST_JDK); commandLineNormal.add("-cp"); commandLineNormal.add(Utils.TEST_CLASS_PATH); commandLineNormal.add("-Xbootclasspath/a:."); commandLineNormal.add("-XX:+UnlockDiagnosticVMOptions"); commandLineNormal.add("-XX:+WhiteBoxAPI"); commandLineNormal.add("-XX:MaxInlineLevel=2"); commandLineNormal.add("-XX:-AlwaysIncrementalInline"); runTest(); } @Override public void testAction() { positiveTest(commandLineReplay); inlineesNormal = parseLogFile(LOG_FILE_NORMAL, getTestClass() + " " + "test", "compile_id='" + getCompileIdFromFile(getReplayFileName()), 5); verify(true); // Incremental inlining is supported in version 2+ // Test replay file version 1. removeIncrementalInlineInfo(); setNewVersionInReplayFile(1); positiveTest(commandLineReplay); verify(false); // Test replay file without version. removeVersionFromReplayFile(); positiveTest(commandLineReplay); verify(false); } private void verify(boolean isNewFormat) { inlineesReplay = parseLogFile(LOG_FILE_REPLAY, getTestClass() + " " + "test", "test ()V", 5); verifyLists(inlineesNormal, inlineesReplay, 5); checkInlining(isNewFormat); } // Check if inlining is done correctly in ciReplay. private void checkInlining(boolean isNewFormat) { String klass = getTestClass(); Asserts.assertTrue(inlineesNormal.get(0).compare(klass, "level0", inlineesNormal.get(0).isForcedInline())); Asserts.assertTrue(inlineesReplay.get(0).compare(klass, "level0", inlineesReplay.get(0).isForcedByReplay())); Asserts.assertTrue(inlineesNormal.get(1).compare(klass, "level1", inlineesNormal.get(1).isNormalInline())); Asserts.assertTrue(inlineesReplay.get(1).compare(klass, "level1", inlineesReplay.get(1).isForcedByReplay())); Asserts.assertTrue(inlineesNormal.get(2).compare(klass, "level2", inlineesNormal.get(2).isForcedInline())); Asserts.assertTrue(inlineesReplay.get(2).compare(klass, "level2", inlineesReplay.get(2).isForcedByReplay())); Asserts.assertTrue(inlineesNormal.get(3).compare(klass, "late", inlineesNormal.get(3).isForcedInline())); Asserts.assertTrue(inlineesReplay.get(3).compare(klass, "late", isNewFormat ? inlineesReplay.get(3).isForcedIncrementalInlineByReplay() : inlineesReplay.get(3).isForcedByReplay())); Asserts.assertTrue(inlineesNormal.get(4).compare(klass, "level4", inlineesNormal.get(4).isTooDeep())); Asserts.assertTrue(inlineesReplay.get(4).compare(klass, "level4", inlineesReplay.get(4).isDisallowedByReplay())); } private void removeIncrementalInlineInfo() { try { Path replayFilePath = Paths.get(getReplayFileName()); List replayContent = Files.readAllLines(replayFilePath); for (int i = 0; i < replayContent.size(); i++) { String line = replayContent.get(i); if (line.startsWith("compile ")) { int lastIndex = 0; StringBuilder newLine = new StringBuilder(); Pattern p = Pattern.compile("(\\d (-?\\d)) \\d compiler"); Matcher m = p.matcher(line); boolean firstMatch = true; while (m.find()) { newLine.append(line, lastIndex, m.start()) .append(m.group(1)) .append(" compiler"); lastIndex = m.end(); String bci = m.group(2); Asserts.assertTrue(firstMatch ? bci.equals("-1") : bci.equals("0"), "only root has -1"); firstMatch = false; } Asserts.assertLessThan(lastIndex, line.length(), "not reached end of line, yet"); newLine.append(line, lastIndex, line.length()); replayContent.set(i, newLine.toString()); } } Files.write(replayFilePath, replayContent, StandardOpenOption.TRUNCATE_EXISTING); } catch (IOException ioe) { throw new Error("Failed to read/write replay data: " + ioe, ioe); } } } class IncrementalInliningTest { private static final WhiteBox WB = WhiteBox.getWhiteBox(); private static String s; public static void main(String[] args) throws NoSuchMethodException { WB.testSetForceInlineMethod(IncrementalInliningTest.class.getDeclaredMethod("level0"), true); WB.testSetForceInlineMethod(IncrementalInliningTest.class.getDeclaredMethod("level2"), true); WB.testSetForceInlineMethod(IncrementalInliningTest.class.getDeclaredMethod("late"), true); for (int i = 0; i < 10000; i++) { test(); } } private static void test() { level0(); } public static void level0() { level1(); } public static void level1() { level2(); } public static void level2() { late(); } // Reached max inline level but forced to be inlined -> inline late. public static void late() { level4(); } // Reached max inline level and not forced to be inlined -> no inline. public static void level4() { s = "HelloWorld"; } }