jdk-24/test/jdk/java/lang/Thread/virtual/MiscMonitorTests.java

510 lines
16 KiB
Java
Raw Normal View History

/*
* Copyright (c) 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 id=default
* @summary Tests for object monitors that have been useful to find bugs
* @library /test/lib
* @requires vm.continuations & vm.opt.LockingMode != 1
* @modules java.base/java.lang:+open
* @run junit/othervm MiscMonitorTests
*/
/*
* @test id=Xint
* @library /test/lib
* @requires vm.continuations & vm.opt.LockingMode != 1
* @modules java.base/java.lang:+open
* @run junit/othervm -Xint MiscMonitorTests
*/
/*
* @test id=Xcomp
* @library /test/lib
* @requires vm.continuations & vm.opt.LockingMode != 1
* @modules java.base/java.lang:+open
* @run junit/othervm -Xcomp MiscMonitorTests
*/
/*
* @test id=Xcomp-TieredStopAtLevel3
* @library /test/lib
* @requires vm.continuations & vm.opt.LockingMode != 1
* @modules java.base/java.lang:+open
* @run junit/othervm -Xcomp -XX:TieredStopAtLevel=3 MiscMonitorTests
*/
/*
* @test id=Xcomp-noTieredCompilation
* @summary Test virtual threads using synchronized
* @library /test/lib
* @requires vm.continuations & vm.opt.LockingMode != 1
* @modules java.base/java.lang:+open
* @run junit/othervm -Xcomp -XX:-TieredCompilation MiscMonitorTests
*/
/*
* @test id=gc
* @requires vm.debug == true & vm.continuations & vm.opt.LockingMode != 1
* @library /test/lib
* @modules java.base/java.lang:+open
* @run junit/othervm -XX:+UnlockDiagnosticVMOptions -XX:+FullGCALot -XX:FullGCALotInterval=1000 MiscMonitorTests
*/
import java.util.concurrent.atomic.AtomicInteger;
import java.util.concurrent.*;
import java.util.ArrayList;
import java.util.List;
import jdk.test.lib.thread.VThreadScheduler;
import org.junit.jupiter.api.Test;
import static org.junit.jupiter.api.Assertions.*;
class MiscMonitorTests {
static final int CARRIER_COUNT = Runtime.getRuntime().availableProcessors();
/**
* Test that yielding while holding monitors releases carrier.
*/
@Test
void testReleaseOnYield() throws Exception {
try (var test = new TestReleaseOnYield()) {
test.runTest();
}
}
private static class TestReleaseOnYield extends TestBase {
final Object lock = new Object();
volatile boolean finish;
volatile int counter;
@Override
void runTest() throws Exception {
int vthreadCount = CARRIER_COUNT;
startVThreads(() -> foo(), vthreadCount, "Batch1");
sleep(500); // Give time for threads to reach Thread.yield
startVThreads(() -> bar(), vthreadCount, "Batch2");
while (counter != vthreadCount) {
Thread.onSpinWait();
}
finish = true;
joinVThreads();
}
void foo() {
Object lock = new Object();
synchronized (lock) {
while (!finish) {
Thread.yield();
}
}
System.err.println("Exiting foo from thread " + Thread.currentThread().getName());
}
void bar() {
synchronized (lock) {
counter++;
}
System.err.println("Exiting bar from thread " + Thread.currentThread().getName());
}
}
/**
* Test yielding while holding monitors with recursive locking releases carrier.
*/
@Test
void testReleaseOnYieldRecursive() throws Exception {
try (var test = new TestReleaseOnYieldRecursive()) {
test.runTest();
}
}
private static class TestReleaseOnYieldRecursive extends TestBase {
final Object lock = new Object();
volatile boolean finish;
volatile int counter;
@Override
void runTest() throws Exception {
int vthreadCount = CARRIER_COUNT;
startVThreads(() -> foo(), vthreadCount, "Batch1");
sleep(500); // Give time for threads to reach Thread.yield
startVThreads(() -> bar(), vthreadCount, "Batch2");
while (counter != 2 * vthreadCount) {
Thread.onSpinWait();
}
finish = true;
joinVThreads();
}
void foo() {
Object lock = new Object();
synchronized (lock) {
while (!finish) {
Thread.yield();
}
}
System.err.println("Exiting foo from thread " + Thread.currentThread().getName());
}
void bar() {
synchronized (lock) {
counter++;
}
recursive(10);
System.err.println("Exiting bar from thread " + Thread.currentThread().getName());
};
void recursive(int count) {
synchronized (Thread.currentThread()) {
if (count > 0) {
recursive(count - 1);
} else {
synchronized (lock) {
counter++;
Thread.yield();
}
}
}
}
}
/**
* Test that contention on monitorenter releases carrier.
*/
@Test
void testReleaseOnContention() throws Exception {
try (var test = new TestReleaseOnContention()) {
test.runTest();
}
}
private static class TestReleaseOnContention extends TestBase {
final Object lock = new Object();
volatile boolean finish;
volatile int counter;
@Override
void runTest() throws Exception {
int vthreadCount = CARRIER_COUNT * 8;
startVThreads(() -> foo(), vthreadCount, "VThread");
sleep(500); // Give time for threads to reach synchronized (lock)
finish = true;
joinVThreads();
}
void foo() {
synchronized (lock) {
while (!finish) {
Thread.yield();
}
}
System.err.println("Exiting foo from thread " + Thread.currentThread().getName());
}
}
/**
* Test contention on monitorenter with extra monitors on stack shared by all threads.
*/
@Test
void testContentionMultipleMonitors() throws Exception {
try (var test = new TestContentionMultipleMonitors()) {
test.runTest();
}
}
private static class TestContentionMultipleMonitors extends TestBase {
static int MONITOR_COUNT = 12;
final Object[] lockArray = new Object[MONITOR_COUNT];
final AtomicInteger workerCount = new AtomicInteger(0);
volatile boolean finish;
@Override
void runTest() throws Exception {
int vthreadCount = CARRIER_COUNT * 8;
for (int i = 0; i < MONITOR_COUNT; i++) {
lockArray[i] = new Object();
}
startVThreads(() -> foo(), vthreadCount, "VThread");
sleep(5000);
finish = true;
joinVThreads();
assertEquals(vthreadCount, workerCount.get());
}
void foo() {
while (!finish) {
int lockNumber = ThreadLocalRandom.current().nextInt(0, MONITOR_COUNT - 1);
synchronized (lockArray[lockNumber]) {
recursive1(lockNumber, lockNumber);
}
}
workerCount.getAndIncrement();
System.err.println("Exiting foo from thread " + Thread.currentThread().getName());
}
public void recursive1(int depth, int lockNumber) {
if (depth > 0) {
recursive1(depth - 1, lockNumber);
} else {
if (Math.random() < 0.5) {
Thread.yield();
}
recursive2(lockNumber);
}
}
public void recursive2(int lockNumber) {
if (lockNumber + 2 <= MONITOR_COUNT - 1) {
lockNumber += 2;
synchronized (lockArray[lockNumber]) {
Thread.yield();
recursive2(lockNumber);
}
}
}
}
/**
* Test contention on monitorenter with extra monitors on stack both local only and shared by all threads.
*/
@Test
void testContentionMultipleMonitors2() throws Exception {
try (var test = new TestContentionMultipleMonitors2()) {
test.runTest();
}
}
private static class TestContentionMultipleMonitors2 extends TestBase {
int MONITOR_COUNT = 12;
final Object[] lockArray = new Object[MONITOR_COUNT];
final AtomicInteger workerCount = new AtomicInteger(0);
volatile boolean finish;
@Override
void runTest() throws Exception {
int vthreadCount = CARRIER_COUNT * 8;
for (int i = 0; i < MONITOR_COUNT; i++) {
lockArray[i] = new Object();
}
startVThreads(() -> foo(), vthreadCount, "VThread");
sleep(5000);
finish = true;
joinVThreads();
assertEquals(vthreadCount, workerCount.get());
}
void foo() {
Object[] myLockArray = new Object[MONITOR_COUNT];
for (int i = 0; i < MONITOR_COUNT; i++) {
myLockArray[i] = new Object();
}
while (!finish) {
int lockNumber = ThreadLocalRandom.current().nextInt(0, MONITOR_COUNT - 1);
synchronized (myLockArray[lockNumber]) {
synchronized (lockArray[lockNumber]) {
recursive1(lockNumber, lockNumber, myLockArray);
}
}
}
workerCount.getAndIncrement();
System.err.println("Exiting foo from thread " + Thread.currentThread().getName());
}
public void recursive1(int depth, int lockNumber, Object[] myLockArray) {
if (depth > 0) {
recursive1(depth - 1, lockNumber, myLockArray);
} else {
if (Math.random() < 0.5) {
Thread.yield();
}
recursive2(lockNumber, myLockArray);
}
}
public void recursive2(int lockNumber, Object[] myLockArray) {
if (lockNumber + 2 <= MONITOR_COUNT - 1) {
lockNumber += 2;
synchronized (myLockArray[lockNumber]) {
if (Math.random() < 0.5) {
Thread.yield();
}
synchronized (lockArray[lockNumber]) {
Thread.yield();
recursive2(lockNumber, myLockArray);
}
}
}
}
}
/**
* Test contention on monitorenter with synchronized methods.
*/
@Test
void testContentionWithSyncMethods() throws Exception {
try (var test = new TestContentionWithSyncMethods()) {
test.runTest();
}
}
private static class TestContentionWithSyncMethods extends TestBase {
static final int MONITOR_COUNT = 12;
final Object[] lockArray = new Object[MONITOR_COUNT];
final AtomicInteger workerCount = new AtomicInteger(0);
volatile boolean finish;
@Override
void runTest() throws Exception {
int vthreadCount = CARRIER_COUNT * 8;
for (int i = 0; i < MONITOR_COUNT; i++) {
lockArray[i] = new Object();
}
startVThreads(() -> foo(), vthreadCount, "VThread");
sleep(5000);
finish = true;
joinVThreads();
assertEquals(vthreadCount, workerCount.get());
}
void foo() {
Object myLock = new Object();
while (!finish) {
int lockNumber = ThreadLocalRandom.current().nextInt(0, MONITOR_COUNT - 1);
synchronized (myLock) {
synchronized (lockArray[lockNumber]) {
recursive(lockNumber, myLock);
}
}
}
workerCount.getAndIncrement();
System.err.println("Exiting foo from thread " + Thread.currentThread().getName());
};
synchronized void recursive(int depth, Object myLock) {
if (depth > 0) {
recursive(depth - 1, myLock);
} else {
if (Math.random() < 0.5) {
Thread.yield();
} else {
synchronized (myLock) {
Thread.yield();
}
}
}
}
}
/**
* Test wait/notify mechanism.
*/
@Test
void waitNotifyTest() throws Exception {
int threadCount = 1000;
int waitTime = 50;
Thread[] vthread = new Thread[threadCount];
long start = System.currentTimeMillis();
while (System.currentTimeMillis() - start < 5000) {
CountDownLatch latchStart = new CountDownLatch(threadCount);
CountDownLatch latchFinish = new CountDownLatch(threadCount);
Object object = new Object();
for (int i = 0; i < threadCount; i++) {
vthread[i] = Thread.ofVirtual().start(() -> {
synchronized (object) {
try {
latchStart.countDown();
object.wait(waitTime);
} catch (InterruptedException e) {
//do nothing;
}
}
latchFinish.countDown();
});
}
try {
latchStart.await();
synchronized (object) {
object.notifyAll();
}
latchFinish.await();
for (int i = 0; i < threadCount; i++) {
vthread[i].join();
}
} catch (InterruptedException e) {
//do nothing;
}
}
}
private static abstract class TestBase implements AutoCloseable {
final ExecutorService scheduler = Executors.newFixedThreadPool(CARRIER_COUNT);
final List<Thread[]> vthreadList = new ArrayList<>();
abstract void runTest() throws Exception;
void startVThreads(Runnable r, int count, String name) {
Thread vthreads[] = new Thread[count];
for (int i = 0; i < count; i++) {
vthreads[i] = VThreadScheduler.virtualThreadBuilder(scheduler).name(name + "-" + i).start(r);
}
vthreadList.add(vthreads);
}
void joinVThreads() throws Exception {
for (Thread[] vthreads : vthreadList) {
for (Thread vthread : vthreads) {
vthread.join();
}
}
}
void sleep(long ms) throws Exception {
Thread.sleep(ms);
}
@Override
public void close() {
scheduler.close();
}
}
}