| /* |
| * 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."); |
| } |
| } |