jdk-24/test/hotspot/jtreg/compiler/codecache/stress/OverloadCompileQueueTest.java

156 lines
6.7 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2014, 2024, 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 OverloadCompileQueueTest
* @key stress randomness
* @summary stressing code cache by overloading compile queues
* @library /test/lib /
* @modules java.base/jdk.internal.misc
* java.management
*
* @build jdk.test.whitebox.WhiteBox
* compiler.codecache.stress.TestCaseImpl
* @run driver jdk.test.lib.helpers.ClassFileInstaller jdk.test.whitebox.WhiteBox
* @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions
* -XX:+WhiteBoxAPI
* -XX:CompileCommand=dontinline,compiler.codecache.stress.Helper$TestCase::method
* -XX:CompileCommand=exclude,java.lang.Thread::sleep
* -XX:CompileCommand=exclude,java.lang.Thread::sleepNanos
* -XX:CompileCommand=exclude,java.lang.Thread::beforeSleep
* -XX:CompileCommand=exclude,java.lang.Thread::afterSleep
* -XX:CompileCommand=exclude,java.util.concurrent.TimeUnit::toNanos
* -XX:CompileCommand=exclude,java.lang.Math::min
* -XX:CompileCommand=exclude,jdk.internal.event.ThreadSleepEvent::*
* -XX:CompileCommand=exclude,jdk.internal.event.SleepEvent::*
* -XX:CompileCommand=exclude,jdk.internal.event.Event::*
* -XX:-SegmentedCodeCache
* compiler.codecache.stress.OverloadCompileQueueTest
* @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions
* -XX:+WhiteBoxAPI
* -XX:CompileCommand=dontinline,compiler.codecache.stress.Helper$TestCase::method
* -XX:CompileCommand=exclude,java.lang.Thread::sleep
* -XX:CompileCommand=exclude,java.lang.Thread::sleepNanos
* -XX:CompileCommand=exclude,java.lang.Thread::beforeSleep
* -XX:CompileCommand=exclude,java.lang.Thread::afterSleep
* -XX:CompileCommand=exclude,java.util.concurrent.TimeUnit::toNanos
* -XX:CompileCommand=exclude,java.lang.Math::min
* -XX:CompileCommand=exclude,jdk.internal.event.ThreadSleepEvent::*
* -XX:CompileCommand=exclude,jdk.internal.event.SleepEvent::*
* -XX:CompileCommand=exclude,jdk.internal.event.Event::*
* -XX:+SegmentedCodeCache
* compiler.codecache.stress.OverloadCompileQueueTest
*/
package compiler.codecache.stress;
import jdk.test.lib.Platform;
import jdk.test.lib.Utils;
import java.lang.reflect.Method;
import java.util.stream.IntStream;
import java.util.Random;
class LockUnlockThread extends Thread {
private static final int MAX_SLEEP = 10000;
private static final int DELAY_BETWEEN_LOCKS = 100;
private final Random rng = Utils.getRandomInstance();
public volatile boolean isActive = true;
@Override
public void run() {
try {
while (isActive) {
int timeInLockedState = rng.nextInt(MAX_SLEEP);
Helper.WHITE_BOX.lockCompilation();
Thread.sleep(timeInLockedState);
Helper.WHITE_BOX.unlockCompilation();
Thread.sleep(DELAY_BETWEEN_LOCKS);
}
} catch (InterruptedException e) {
if (isActive) {
throw new Error("TESTBUG: LockUnlockThread was unexpectedly interrupted", e);
}
} finally {
Helper.WHITE_BOX.unlockCompilation();
}
}
}
public class OverloadCompileQueueTest implements Runnable {
private static final String METHOD_TO_ENQUEUE = "method";
private static final int LEVEL_SIMPLE = 1;
private static final int LEVEL_FULL_OPTIMIZATION = 4;
private static final boolean TIERED_COMPILATION
= Helper.WHITE_BOX.getBooleanVMFlag("TieredCompilation");
private static final int TIERED_STOP_AT_LEVEL
= Helper.WHITE_BOX.getIntxVMFlag("TieredStopAtLevel").intValue();
private static final int[] AVAILABLE_LEVELS;
static {
if (TIERED_COMPILATION) {
AVAILABLE_LEVELS = IntStream
.rangeClosed(LEVEL_SIMPLE, TIERED_STOP_AT_LEVEL)
.toArray();
} else if (Platform.isServer() && !Platform.isEmulatedClient()) {
AVAILABLE_LEVELS = new int[] { LEVEL_FULL_OPTIMIZATION };
} else if (Platform.isClient() || Platform.isMinimal() || Platform.isEmulatedClient()) {
AVAILABLE_LEVELS = new int[] { LEVEL_SIMPLE };
} else {
throw new Error("TESTBUG: unknown VM: " + Platform.vmName);
}
}
public static void main(String[] args) throws InterruptedException {
LockUnlockThread lockUnlockThread = new LockUnlockThread();
lockUnlockThread.start();
if (Platform.isInt()) {
throw new Error("TESTBUG: test can not be run in interpreter");
}
new CodeCacheStressRunner(new OverloadCompileQueueTest()).runTest();
lockUnlockThread.isActive = false;
lockUnlockThread.interrupt();
lockUnlockThread.join();
}
@Override
public void run() {
Helper.TestCase obj = Helper.TestCase.get();
Class clazz = obj.getClass();
Method mEnqueue;
try {
mEnqueue = clazz.getMethod(METHOD_TO_ENQUEUE);
} catch (NoSuchMethodException | SecurityException e) {
throw new Error(String.format(
"TESTBUG: cannot get method '%s' of class %s",
METHOD_TO_ENQUEUE, clazz.getName()), e);
}
for (int compLevel : AVAILABLE_LEVELS) {
Helper.WHITE_BOX.enqueueMethodForCompilation(mEnqueue, compLevel);
}
}
}