blob: ee03d73760734999def9f571744d2b3998b6e6c1 [file] [log] [blame]
/*
* Copyright (C) 2017 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package art;
import java.util.concurrent.Semaphore;
import java.util.Arrays;
public class Test1930 {
public static final int NUM_RETRY = 100;
private static void testSingleThread() {
Monitors.NamedLock lk = new Monitors.NamedLock("Test1930 - testSingleThread");
executeLocked(() -> { printMonitorUsage(lk); }, lk);
}
private static void testSingleThreadNative() {
Monitors.NamedLock lk = new Monitors.NamedLock("Test1930 - testSingleThread");
executeLockedNative(() -> { printMonitorUsage(lk); }, lk);
}
private static void testLockedTwice() {
final Monitors.NamedLock lk = new Monitors.NamedLock("Test1930 - testLockedTwice");
executeLocked(() -> { executeLocked(() -> { printMonitorUsage(lk); }, lk); }, lk);
}
private static void testLockedTwiceNJ() {
final Monitors.NamedLock lk = new Monitors.NamedLock("Test1930 - testLockedTwiceNJ");
executeLockedNative(() -> { executeLockedNative(() -> { printMonitorUsage(lk); }, lk); }, lk);
}
private static void testLockedTwiceJN() {
final Monitors.NamedLock lk = new Monitors.NamedLock("Test1930 - testLockedTwiceJN");
executeLockedNative(() -> { executeLockedNative(() -> { printMonitorUsage(lk); }, lk); }, lk);
}
private static void testLockedTwiceNative() {
final Monitors.NamedLock lk = new Monitors.NamedLock("Test1930 - testLockedTwiceNative");
executeLockedNative(() -> { executeLockedNative(() -> { printMonitorUsage(lk); }, lk); }, lk);
}
public final static class ThreadSignaler {
public volatile boolean signal = false;
}
private static void testLockWait() throws Exception {
final Monitors.NamedLock lk = new Monitors.NamedLock("Test1930 - testLockWait");
final Semaphore sem = new Semaphore(0);
final Thread t = new Thread(() -> {
sem.release();
synchronized (lk) {
printMonitorUsage(lk);
}
}, "Test1930 Thread - testLockWait");
synchronized (lk) {
t.start();
// Wait for the other thread to actually start.
sem.acquire();
// Wait for the other thread to go to sleep trying to get the mutex. This might take a (short)
// time since we try spinning first for better performance.
boolean found_wait = false;
for (long i = 0; i < NUM_RETRY; i++) {
if (Arrays.asList(Monitors.getObjectMonitorUsage(lk).waiters).contains(t)) {
found_wait = true;
break;
} else {
Thread.sleep(500);
Thread.yield();
}
}
if (!found_wait) {
System.out.println("other thread doesn't seem to be waiting.");
}
printMonitorUsage(lk);
}
t.join();
printMonitorUsage(lk);
}
private static void testNotifyWait() throws Exception {
final Monitors.NamedLock lk = new Monitors.NamedLock("Test1930 - testNotifyWait");
final Semaphore sem = new Semaphore(0);
Thread t = new Thread(() -> {
synchronized (lk) {
printMonitorUsage(lk);
sem.release();
try {
lk.DoWait();
} catch (Exception e) {
throw new Error("Error waiting!", e);
}
printMonitorUsage(lk);
}
}, "Test1930 Thread - testLockWait");
t.start();
sem.acquire();
synchronized (lk) {
printMonitorUsage(lk);
lk.DoNotifyAll();
}
t.join();
printMonitorUsage(lk);
}
public static void run() throws Exception {
// Single threaded tests.
System.out.println("Running with single thread.");
testSingleThread();
System.out.println("Running with single thread in native.");
testSingleThreadNative();
System.out.println("Lock twice");
testLockedTwice();
System.out.println("Lock twice native");
testLockedTwiceNative();
System.out.println("Lock twice Java then native");
testLockedTwiceJN();
System.out.println("Lock twice native then Java");
testLockedTwiceNJ();
// Mutli threaded tests.
System.out.println("lock with wait");
testLockWait();
System.out.println("Wait for notify.");
testNotifyWait();
}
public static void printPreLock(Object lock) {
System.out.println(String.format("Pre-lock[%s]: %s",
Thread.currentThread().getName(), Monitors.getObjectMonitorUsage(lock)));
}
public static void executeLocked(Runnable r, Object lock) {
printPreLock(lock);
synchronized (lock) {
r.run();
}
}
public native static void executeLockedNative(Runnable r, Object m);
public static void printMonitorUsage(Object m) {
System.out.println(String.format("Thread[%s]: %s",
Thread.currentThread().getName(), Monitors.getObjectMonitorUsage(m)));
}
}