/*
|
* 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.Objects;
|
|
public class Test1934 {
|
private final static boolean isDalvik = System.getProperty("java.vm.name").equals("Dalvik");
|
|
public static final boolean PRINT_STACK_TRACE = false;
|
|
public static void run() throws Exception {
|
ensureClassesLoaded();
|
|
System.out.println("Interrupt before start");
|
testInterruptBeforeStart();
|
|
System.out.println("Stop before start");
|
testStopBeforeStart();
|
|
System.out.println("Interrupt recur");
|
testInterruptRecur();
|
|
System.out.println("Stop Recur");
|
testStopRecur();
|
|
System.out.println("Interrupt spinning");
|
testInterruptSpinning();
|
|
System.out.println("Stop spinning");
|
testStopSpinning();
|
|
System.out.println("Interrupt wait");
|
testInterruptWait();
|
|
System.out.println("Stop wait");
|
testStopWait();
|
|
System.out.println("Stop in native");
|
testStopInNative();
|
}
|
|
private static void ensureInitialized(Class c) {
|
try {
|
Class.forName(c.getName());
|
} catch (Exception e) {
|
throw new Error("Failed to initialize " + c, e);
|
}
|
}
|
|
private static void ensureClassesLoaded() {
|
// Depending on timing this class might (or might not) be loaded during testing of Stop. If it
|
// is and the StopThread occurs inside of it we will get a ExceptionInInitializerError which is
|
// not what we are looking for. In order to avoid this ever happening simply initialize the
|
// class that can cause it early.
|
ensureInitialized(java.util.concurrent.locks.LockSupport.class);
|
}
|
|
public static Thread createThread(Runnable r, String name) {
|
return new Thread(Thread.currentThread().getThreadGroup(), r, name, /* 10 mb */ 10 * 1000000);
|
}
|
|
public static void testStopBeforeStart() throws Exception {
|
final Throwable[] out_err = new Throwable[] { null, };
|
final Object tst = new Object();
|
Thread target = createThread(() -> { while (true) { } }, "waiting thread!");
|
target.setUncaughtExceptionHandler((t, e) -> { out_err[0] = e; });
|
System.out.println("stopping other thread before starting");
|
try {
|
Threads.stopThread(target, new Error("AWESOME"));
|
target.start();
|
target.join();
|
System.out.println("Other thread Stopped by: " + out_err[0]);
|
if (PRINT_STACK_TRACE && out_err[0] != null) {
|
out_err[0].printStackTrace();
|
}
|
} catch (Exception e) {
|
System.out.println("Caught exception " + e);
|
}
|
}
|
|
public static void testInterruptBeforeStart() throws Exception {
|
final Throwable[] out_err = new Throwable[] { null, };
|
final Object tst = new Object();
|
Thread target = createThread(() -> { while (true) { } }, "waiting thread!");
|
target.setUncaughtExceptionHandler((t, e) -> { out_err[0] = e; });
|
System.out.println("interrupting other thread before starting");
|
try {
|
Threads.interruptThread(target);
|
target.start();
|
target.join();
|
System.out.println("Other thread interrupted. err: " + out_err[0]);
|
if (PRINT_STACK_TRACE && out_err[0] != null) {
|
out_err[0].printStackTrace();
|
}
|
} catch (Exception e) {
|
System.out.println("Caught exception " + e);
|
}
|
}
|
|
public static void testStopWait() throws Exception {
|
final Throwable[] out_err = new Throwable[] { null, };
|
final Object tst = new Object();
|
final Semaphore sem = new Semaphore(0);
|
Thread target = createThread(() -> {
|
sem.release();
|
while (true) {
|
try {
|
synchronized (tst) {
|
tst.wait();
|
}
|
} catch (InterruptedException e) { throw new Error("Interrupted!", e); }
|
}
|
}, "waiting thread!");
|
target.setUncaughtExceptionHandler((t, e) -> { out_err[0] = e; });
|
target.start();
|
sem.acquire();
|
while (!Objects.equals(Monitors.getCurrentContendedMonitor(target), tst)) {}
|
System.out.println("stopping other thread waiting");
|
Threads.stopThread(target, new Error("AWESOME"));
|
target.join();
|
System.out.println("Other thread Stopped by: " + out_err[0]);
|
if (PRINT_STACK_TRACE && out_err[0] != null) {
|
out_err[0].printStackTrace();
|
}
|
}
|
|
public static void testInterruptWait() throws Exception {
|
final Throwable[] out_err = new Throwable[] { null, };
|
final Object tst = new Object();
|
final Semaphore sem = new Semaphore(0);
|
Thread target = createThread(() -> {
|
sem.release();
|
while (true) {
|
try {
|
synchronized (tst) {
|
tst.wait();
|
}
|
} catch (InterruptedException e) { throw new Error("Interrupted!", e); }
|
}
|
}, "waiting thread!");
|
target.setUncaughtExceptionHandler((t, e) -> { out_err[0] = e; });
|
target.start();
|
sem.acquire();
|
while (!Objects.equals(Monitors.getCurrentContendedMonitor(target), tst)) {}
|
System.out.println("interrupting other thread waiting");
|
Threads.interruptThread(target);
|
target.join();
|
System.out.println("Other thread interrupted. err: " + out_err[0]);
|
if (PRINT_STACK_TRACE && out_err[0] != null) {
|
out_err[0].printStackTrace();
|
}
|
}
|
|
public static void doNothing() {}
|
public static native long allocNativeMonitor();
|
public static native void nativeWaitForOtherThread(long id);
|
public static native void nativeDoInterleaved(long id, Runnable op);
|
public static native void destroyNativeMonitor(long id);
|
public static void testStopInNative() throws Exception {
|
final Throwable[] out_err = new Throwable[] { null, };
|
final long native_monitor_id = allocNativeMonitor();
|
final Semaphore sem = new Semaphore(0);
|
Thread target = createThread(() -> {
|
sem.release();
|
nativeWaitForOtherThread(native_monitor_id);
|
// We need to make sure we do something that can get the exception to be actually noticed.
|
doNothing();
|
}, "native waiting thread!");
|
target.setUncaughtExceptionHandler((t, e) -> { out_err[0] = e; });
|
target.start();
|
sem.acquire();
|
System.out.println("stopping other thread");
|
nativeDoInterleaved(
|
native_monitor_id,
|
() -> { Threads.stopThread(target, new Error("AWESOME")); });
|
target.join();
|
|
String out_err_msg;
|
if (isDalvik || out_err[0] != null) {
|
out_err_msg = out_err[0].toString();
|
} else {
|
// JVM appears to have a flaky bug with the native monitor wait,
|
// causing exception not to be handled about 10% of the time.
|
out_err_msg = "java.lang.Error: AWESOME";
|
}
|
System.out.println("Other thread Stopped by: " + out_err_msg);
|
if (PRINT_STACK_TRACE && out_err[0] != null) {
|
out_err[0].printStackTrace();
|
}
|
destroyNativeMonitor(native_monitor_id);
|
}
|
|
public static void doRecurCnt(Runnable r, int cnt) {
|
if (r != null) {
|
r.run();
|
}
|
if (cnt != 0) {
|
doRecurCnt(r, cnt - 1);
|
}
|
}
|
|
public static void testStopRecur() throws Exception {
|
final Throwable[] out_err = new Throwable[] { null, };
|
final Semaphore sem = new Semaphore(0);
|
Thread target = createThread(() -> {
|
sem.release();
|
while (true) {
|
doRecurCnt(null, 50);
|
}
|
}, "recuring thread!");
|
target.setUncaughtExceptionHandler((t, e) -> { out_err[0] = e; });
|
target.start();
|
sem.acquire();
|
System.out.println("stopping other thread recurring");
|
Threads.stopThread(target, new Error("AWESOME!"));
|
target.join();
|
System.out.println("Other thread Stopped by: " + out_err[0]);
|
if (PRINT_STACK_TRACE && out_err[0] != null) {
|
out_err[0].printStackTrace();
|
}
|
}
|
|
public static void testInterruptRecur() throws Exception {
|
final Throwable[] out_err = new Throwable[] { null, };
|
final Semaphore sem = new Semaphore(0);
|
Thread target = createThread(() -> {
|
sem.release();
|
while (true) {
|
doRecurCnt(() -> {
|
if (Thread.currentThread().isInterrupted()) { throw new Error("Interrupted!"); }
|
}, 50);
|
}
|
}, "recuring thread!");
|
target.setUncaughtExceptionHandler((t, e) -> { out_err[0] = e; });
|
target.start();
|
sem.acquire();
|
System.out.println("Interrupting other thread recurring");
|
Threads.interruptThread(target);
|
target.join();
|
System.out.println("Other thread Interrupted. err: " + out_err[0]);
|
if (PRINT_STACK_TRACE && out_err[0] != null) {
|
out_err[0].printStackTrace();
|
}
|
}
|
|
public static void testStopSpinning() throws Exception {
|
final Throwable[] out_err = new Throwable[] { null, };
|
final Semaphore sem = new Semaphore(0);
|
Thread target = createThread(() -> { sem.release(); while (true) {} }, "Spinning thread!");
|
target.setUncaughtExceptionHandler((t, e) -> { out_err[0] = e; });
|
target.start();
|
sem.acquire();
|
System.out.println("stopping other thread spinning");
|
Threads.stopThread(target, new Error("AWESOME!"));
|
target.join();
|
System.out.println("Other thread Stopped by: " + out_err[0]);
|
if (PRINT_STACK_TRACE && out_err[0] != null) {
|
out_err[0].printStackTrace();
|
}
|
}
|
|
public static void testInterruptSpinning() throws Exception {
|
final Semaphore sem = new Semaphore(0);
|
Thread target = createThread(() -> {
|
sem.release();
|
while (!Thread.currentThread().isInterrupted()) { }
|
}, "Spinning thread!");
|
target.start();
|
sem.acquire();
|
System.out.println("Interrupting other thread spinning");
|
Threads.interruptThread(target);
|
target.join();
|
System.out.println("Other thread Interrupted.");
|
}
|
}
|