| /* |
| * Copyright (C) 2006 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. |
| */ |
| |
| import java.util.ArrayList; |
| import java.util.Arrays; |
| import java.util.Iterator; |
| import java.util.List; |
| |
| /** |
| * Test some basic thread stuff. |
| */ |
| public class Main { |
| public static void main(String[] args) throws Exception { |
| System.loadLibrary(args[0]); |
| System.out.println("thread test starting"); |
| testThreadCapacity(); |
| testThreadDaemons(); |
| testSleepZero(); |
| testSetName(); |
| testThreadPriorities(); |
| testMainThreadGroup(); |
| testMainThreadAllStackTraces(); |
| System.out.println("thread test done"); |
| } |
| |
| /* |
| * Simple thread capacity test. |
| */ |
| private static void testThreadCapacity() throws Exception { |
| TestCapacityThread[] threads = new TestCapacityThread[128]; |
| for (int i = 0; i < threads.length; i++) { |
| threads[i] = new TestCapacityThread(); |
| } |
| |
| for (TestCapacityThread thread : threads) { |
| thread.start(); |
| } |
| for (TestCapacityThread thread : threads) { |
| thread.join(); |
| } |
| |
| System.out.println("testThreadCapacity thread count: " + TestCapacityThread.mCount); |
| } |
| |
| private static class TestCapacityThread extends Thread { |
| static int mCount = 0; |
| public void run() { |
| synchronized (TestCapacityThread.class) { |
| ++mCount; |
| } |
| try { |
| sleep(1000); |
| } catch (Exception ex) { |
| } |
| } |
| } |
| |
| private static void testThreadDaemons() { |
| Thread t = new Thread(null, new TestDaemonThread(), "TestDaemonThread", 7168); |
| |
| t.setDaemon(false); |
| |
| System.out.print("testThreadDaemons starting thread '" + t.getName() + "'\n"); |
| t.start(); |
| |
| try { |
| t.join(); |
| } catch (InterruptedException ex) { |
| ex.printStackTrace(System.out); |
| } |
| |
| System.out.print("testThreadDaemons finished\n"); |
| } |
| |
| private static class TestDaemonThread implements Runnable { |
| public void run() { |
| System.out.print("testThreadDaemons @ Thread running\n"); |
| |
| try { |
| Thread.currentThread().setDaemon(true); |
| System.out.print("testThreadDaemons @ FAILED: setDaemon() succeeded\n"); |
| } catch (IllegalThreadStateException itse) { |
| System.out.print("testThreadDaemons @ Got expected setDaemon exception\n"); |
| } |
| |
| try { |
| Thread.sleep(2000); |
| } |
| catch (InterruptedException ie) { |
| System.out.print("testThreadDaemons @ Interrupted!\n"); |
| } |
| finally { |
| System.out.print("testThreadDaemons @ Thread bailing\n"); |
| } |
| } |
| } |
| |
| private static void testSleepZero() throws Exception { |
| Thread.currentThread().interrupt(); |
| try { |
| Thread.sleep(0); |
| throw new AssertionError("unreachable"); |
| } catch (InterruptedException e) { |
| if (Thread.currentThread().isInterrupted()) { |
| throw new AssertionError("thread is interrupted"); |
| } |
| } |
| System.out.print("testSleepZero finished\n"); |
| } |
| |
| private static void testSetName() throws Exception { |
| System.out.print("testSetName starting\n"); |
| Thread thread = new Thread() { |
| @Override |
| public void run() { |
| System.out.print("testSetName running\n"); |
| } |
| }; |
| thread.start(); |
| thread.setName("HelloWorld"); // b/17302037 hang if setName called after start |
| if (!thread.getName().equals("HelloWorld")) { |
| throw new AssertionError("Unexpected thread name: " + thread.getName()); |
| } |
| thread.join(); |
| if (!thread.getName().equals("HelloWorld")) { |
| throw new AssertionError("Unexpected thread name after join: " + thread.getName()); |
| } |
| System.out.print("testSetName finished\n"); |
| } |
| |
| private static void testThreadPriorities() throws Exception { |
| System.out.print("testThreadPriorities starting\n"); |
| |
| PriorityStoringThread t1 = new PriorityStoringThread(false); |
| t1.setPriority(Thread.MAX_PRIORITY); |
| t1.start(); |
| t1.join(); |
| if (supportsThreadPriorities() && (t1.getNativePriority() != Thread.MAX_PRIORITY)) { |
| System.out.print("thread priority for t1 was " + t1.getNativePriority() + |
| " [expected Thread.MAX_PRIORITY]\n"); |
| } |
| |
| PriorityStoringThread t2 = new PriorityStoringThread(true); |
| t2.start(); |
| t2.join(); |
| if (supportsThreadPriorities() && (t2.getNativePriority() != Thread.MAX_PRIORITY)) { |
| System.out.print("thread priority for t2 was " + t2.getNativePriority() + |
| " [expected Thread.MAX_PRIORITY]\n"); |
| } |
| |
| System.out.print("testThreadPriorities finished\n"); |
| } |
| |
| private static void testMainThreadGroup() { |
| Thread threads[] = new Thread[10]; |
| Thread current = Thread.currentThread(); |
| current.getThreadGroup().enumerate(threads); |
| |
| for (Thread t : threads) { |
| if (t == current) { |
| System.out.println("Found current Thread in ThreadGroup"); |
| return; |
| } |
| } |
| throw new RuntimeException("Did not find main thread: " + Arrays.toString(threads)); |
| } |
| |
| private static void testMainThreadAllStackTraces() { |
| StackTraceElement[] trace = Thread.getAllStackTraces().get(Thread.currentThread()); |
| if (trace == null) { |
| throw new RuntimeException("Did not find main thread: " + Thread.getAllStackTraces()); |
| } |
| List<StackTraceElement> list = Arrays.asList(trace); |
| Iterator<StackTraceElement> it = list.iterator(); |
| while (it.hasNext()) { |
| StackTraceElement ste = it.next(); |
| if (ste.getClassName().equals("Main")) { |
| if (!ste.getMethodName().equals("testMainThreadAllStackTraces")) { |
| throw new RuntimeException(list.toString()); |
| } |
| |
| StackTraceElement ste2 = it.next(); |
| if (!ste2.getClassName().equals("Main")) { |
| throw new RuntimeException(list.toString()); |
| } |
| if (!ste2.getMethodName().equals("main")) { |
| throw new RuntimeException(list.toString()); |
| } |
| |
| System.out.println("Found expected stack in getAllStackTraces()"); |
| return; |
| } |
| } |
| throw new RuntimeException(list.toString()); |
| } |
| |
| private static native int getNativePriority(); |
| private static native boolean supportsThreadPriorities(); |
| |
| static class PriorityStoringThread extends Thread { |
| private final boolean setPriority; |
| private volatile int nativePriority; |
| |
| public PriorityStoringThread(boolean setPriority) { |
| this.setPriority = setPriority; |
| this.nativePriority = -1; |
| } |
| |
| @Override |
| public void run() { |
| if (setPriority) { |
| setPriority(Thread.MAX_PRIORITY); |
| } |
| |
| nativePriority = Main.getNativePriority(); |
| } |
| |
| public int getNativePriority() { |
| return nativePriority; |
| } |
| } |
| } |