1 /*
2  * Written by Doug Lea with assistance from members of JCP JSR-166
3  * Expert Group and released to the public domain, as explained at
4  * http://creativecommons.org/publicdomain/zero/1.0/
5  * Other contributors include Andrew Wright, Jeffrey Hayes,
6  * Pat Fisher, Mike Judd.
7  */
8 
9 package jsr166;
10 
11 import static java.util.concurrent.TimeUnit.MILLISECONDS;
12 import static java.util.concurrent.TimeUnit.NANOSECONDS;
13 import static java.util.concurrent.TimeUnit.SECONDS;
14 
15 import java.util.ArrayList;
16 import java.util.List;
17 import java.util.concurrent.ArrayBlockingQueue;
18 import java.util.concurrent.BlockingQueue;
19 import java.util.concurrent.Callable;
20 import java.util.concurrent.CancellationException;
21 import java.util.concurrent.CountDownLatch;
22 import java.util.concurrent.ExecutionException;
23 import java.util.concurrent.Executors;
24 import java.util.concurrent.ExecutorService;
25 import java.util.concurrent.Future;
26 import java.util.concurrent.FutureTask;
27 import java.util.concurrent.LinkedBlockingQueue;
28 import java.util.concurrent.RejectedExecutionException;
29 import java.util.concurrent.RejectedExecutionHandler;
30 import java.util.concurrent.SynchronousQueue;
31 import java.util.concurrent.ThreadFactory;
32 import java.util.concurrent.ThreadPoolExecutor;
33 import java.util.concurrent.TimeUnit;
34 import java.util.concurrent.atomic.AtomicInteger;
35 
36 import junit.framework.Test;
37 import junit.framework.TestSuite;
38 
39 public class ThreadPoolExecutorTest extends JSR166TestCase {
40     // android-note: Removed because the CTS runner does a bad job of
41     // retrying tests that have suite() declarations.
42     //
43     // public static void main(String[] args) {
44     //     main(suite(), args);
45     // }
46     // public static Test suite() {
47     //     return new TestSuite(ThreadPoolExecutorTest.class);
48     // }
49 
50     static class ExtendedTPE extends ThreadPoolExecutor {
51         final CountDownLatch beforeCalled = new CountDownLatch(1);
52         final CountDownLatch afterCalled = new CountDownLatch(1);
53         final CountDownLatch terminatedCalled = new CountDownLatch(1);
54 
55         public ExtendedTPE() {
56             super(1, 1, LONG_DELAY_MS, MILLISECONDS, new SynchronousQueue<Runnable>());
57         }
58         protected void beforeExecute(Thread t, Runnable r) {
59             beforeCalled.countDown();
60         }
61         protected void afterExecute(Runnable r, Throwable t) {
62             afterCalled.countDown();
63         }
64         protected void terminated() {
65             terminatedCalled.countDown();
66         }
67 
68         public boolean beforeCalled() {
69             return beforeCalled.getCount() == 0;
70         }
71         public boolean afterCalled() {
72             return afterCalled.getCount() == 0;
73         }
74         public boolean terminatedCalled() {
75             return terminatedCalled.getCount() == 0;
76         }
77     }
78 
79     static class FailingThreadFactory implements ThreadFactory {
80         int calls = 0;
81         public Thread newThread(Runnable r) {
82             if (++calls > 1) return null;
83             return new Thread(r);
84         }
85     }
86 
87     /**
88      * execute successfully executes a runnable
89      */
90     public void testExecute() throws InterruptedException {
91         final ThreadPoolExecutor p =
92             new ThreadPoolExecutor(1, 1,
93                                    LONG_DELAY_MS, MILLISECONDS,
94                                    new ArrayBlockingQueue<Runnable>(10));
95         try (PoolCleaner cleaner = cleaner(p)) {
96             final CountDownLatch done = new CountDownLatch(1);
97             final Runnable task = new CheckedRunnable() {
98                 public void realRun() { done.countDown(); }};
99             p.execute(task);
100             assertTrue(done.await(LONG_DELAY_MS, MILLISECONDS));
101         }
102     }
103 
104     /**
105      * getActiveCount increases but doesn't overestimate, when a
106      * thread becomes active
107      */
108     public void testGetActiveCount() throws InterruptedException {
109         final CountDownLatch done = new CountDownLatch(1);
110         final ThreadPoolExecutor p =
111             new ThreadPoolExecutor(2, 2,
112                                    LONG_DELAY_MS, MILLISECONDS,
113                                    new ArrayBlockingQueue<Runnable>(10));
114         try (PoolCleaner cleaner = cleaner(p, done)) {
115             final CountDownLatch threadStarted = new CountDownLatch(1);
116             assertEquals(0, p.getActiveCount());
117             p.execute(new CheckedRunnable() {
118                 public void realRun() throws InterruptedException {
119                     threadStarted.countDown();
120                     assertEquals(1, p.getActiveCount());
121                     await(done);
122                 }});
123             await(threadStarted);
124             assertEquals(1, p.getActiveCount());
125         }
126     }
127 
128     /**
129      * prestartCoreThread starts a thread if under corePoolSize, else doesn't
130      */
131     public void testPrestartCoreThread() {
132         final ThreadPoolExecutor p =
133             new ThreadPoolExecutor(2, 6,
134                                    LONG_DELAY_MS, MILLISECONDS,
135                                    new ArrayBlockingQueue<Runnable>(10));
136         try (PoolCleaner cleaner = cleaner(p)) {
137             assertEquals(0, p.getPoolSize());
138             assertTrue(p.prestartCoreThread());
139             assertEquals(1, p.getPoolSize());
140             assertTrue(p.prestartCoreThread());
141             assertEquals(2, p.getPoolSize());
142             assertFalse(p.prestartCoreThread());
143             assertEquals(2, p.getPoolSize());
144             p.setCorePoolSize(4);
145             assertTrue(p.prestartCoreThread());
146             assertEquals(3, p.getPoolSize());
147             assertTrue(p.prestartCoreThread());
148             assertEquals(4, p.getPoolSize());
149             assertFalse(p.prestartCoreThread());
150             assertEquals(4, p.getPoolSize());
151         }
152     }
153 
154     /**
155      * prestartAllCoreThreads starts all corePoolSize threads
156      */
157     public void testPrestartAllCoreThreads() {
158         final ThreadPoolExecutor p =
159             new ThreadPoolExecutor(2, 6,
160                                    LONG_DELAY_MS, MILLISECONDS,
161                                    new ArrayBlockingQueue<Runnable>(10));
162         try (PoolCleaner cleaner = cleaner(p)) {
163             assertEquals(0, p.getPoolSize());
164             p.prestartAllCoreThreads();
165             assertEquals(2, p.getPoolSize());
166             p.prestartAllCoreThreads();
167             assertEquals(2, p.getPoolSize());
168             p.setCorePoolSize(4);
169             p.prestartAllCoreThreads();
170             assertEquals(4, p.getPoolSize());
171             p.prestartAllCoreThreads();
172             assertEquals(4, p.getPoolSize());
173         }
174     }
175 
176     /**
177      * getCompletedTaskCount increases, but doesn't overestimate,
178      * when tasks complete
179      */
180     public void testGetCompletedTaskCount() throws InterruptedException {
181         final ThreadPoolExecutor p =
182             new ThreadPoolExecutor(2, 2,
183                                    LONG_DELAY_MS, MILLISECONDS,
184                                    new ArrayBlockingQueue<Runnable>(10));
185         try (PoolCleaner cleaner = cleaner(p)) {
186             final CountDownLatch threadStarted = new CountDownLatch(1);
187             final CountDownLatch threadProceed = new CountDownLatch(1);
188             final CountDownLatch threadDone = new CountDownLatch(1);
189             assertEquals(0, p.getCompletedTaskCount());
190             p.execute(new CheckedRunnable() {
191                 public void realRun() throws InterruptedException {
192                     threadStarted.countDown();
193                     assertEquals(0, p.getCompletedTaskCount());
194                     threadProceed.await();
195                     threadDone.countDown();
196                 }});
197             await(threadStarted);
198             assertEquals(0, p.getCompletedTaskCount());
199             threadProceed.countDown();
200             threadDone.await();
201             long startTime = System.nanoTime();
202             while (p.getCompletedTaskCount() != 1) {
203                 if (millisElapsedSince(startTime) > LONG_DELAY_MS)
204                     fail("timed out");
205                 Thread.yield();
206             }
207         }
208     }
209 
210     /**
211      * getCorePoolSize returns size given in constructor if not otherwise set
212      */
213     public void testGetCorePoolSize() {
214         final ThreadPoolExecutor p =
215             new ThreadPoolExecutor(1, 1,
216                                    LONG_DELAY_MS, MILLISECONDS,
217                                    new ArrayBlockingQueue<Runnable>(10));
218         try (PoolCleaner cleaner = cleaner(p)) {
219             assertEquals(1, p.getCorePoolSize());
220         }
221     }
222 
223     /**
224      * getKeepAliveTime returns value given in constructor if not otherwise set
225      */
226     public void testGetKeepAliveTime() {
227         final ThreadPoolExecutor p =
228             new ThreadPoolExecutor(2, 2,
229                                    1000, MILLISECONDS,
230                                    new ArrayBlockingQueue<Runnable>(10));
231         try (PoolCleaner cleaner = cleaner(p)) {
232             assertEquals(1, p.getKeepAliveTime(SECONDS));
233         }
234     }
235 
236     /**
237      * getThreadFactory returns factory in constructor if not set
238      */
239     public void testGetThreadFactory() {
240         ThreadFactory threadFactory = new SimpleThreadFactory();
241         final ThreadPoolExecutor p =
242             new ThreadPoolExecutor(1, 2,
243                                    LONG_DELAY_MS, MILLISECONDS,
244                                    new ArrayBlockingQueue<Runnable>(10),
245                                    threadFactory,
246                                    new NoOpREHandler());
247         try (PoolCleaner cleaner = cleaner(p)) {
248             assertSame(threadFactory, p.getThreadFactory());
249         }
250     }
251 
252     /**
253      * setThreadFactory sets the thread factory returned by getThreadFactory
254      */
255     public void testSetThreadFactory() {
256         final ThreadPoolExecutor p =
257             new ThreadPoolExecutor(1, 2,
258                                    LONG_DELAY_MS, MILLISECONDS,
259                                    new ArrayBlockingQueue<Runnable>(10));
260         try (PoolCleaner cleaner = cleaner(p)) {
261             ThreadFactory threadFactory = new SimpleThreadFactory();
262             p.setThreadFactory(threadFactory);
263             assertSame(threadFactory, p.getThreadFactory());
264         }
265     }
266 
267     /**
268      * setThreadFactory(null) throws NPE
269      */
270     public void testSetThreadFactoryNull() {
271         final ThreadPoolExecutor p =
272             new ThreadPoolExecutor(1, 2,
273                                    LONG_DELAY_MS, MILLISECONDS,
274                                    new ArrayBlockingQueue<Runnable>(10));
275         try (PoolCleaner cleaner = cleaner(p)) {
276             try {
277                 p.setThreadFactory(null);
278                 shouldThrow();
279             } catch (NullPointerException success) {}
280         }
281     }
282 
283     /**
284      * getRejectedExecutionHandler returns handler in constructor if not set
285      */
286     public void testGetRejectedExecutionHandler() {
287         final RejectedExecutionHandler handler = new NoOpREHandler();
288         final ThreadPoolExecutor p =
289             new ThreadPoolExecutor(1, 2,
290                                    LONG_DELAY_MS, MILLISECONDS,
291                                    new ArrayBlockingQueue<Runnable>(10),
292                                    handler);
293         try (PoolCleaner cleaner = cleaner(p)) {
294             assertSame(handler, p.getRejectedExecutionHandler());
295         }
296     }
297 
298     /**
299      * setRejectedExecutionHandler sets the handler returned by
300      * getRejectedExecutionHandler
301      */
302     public void testSetRejectedExecutionHandler() {
303         final ThreadPoolExecutor p =
304             new ThreadPoolExecutor(1, 2,
305                                    LONG_DELAY_MS, MILLISECONDS,
306                                    new ArrayBlockingQueue<Runnable>(10));
307         try (PoolCleaner cleaner = cleaner(p)) {
308             RejectedExecutionHandler handler = new NoOpREHandler();
309             p.setRejectedExecutionHandler(handler);
310             assertSame(handler, p.getRejectedExecutionHandler());
311         }
312     }
313 
314     /**
315      * setRejectedExecutionHandler(null) throws NPE
316      */
317     public void testSetRejectedExecutionHandlerNull() {
318         final ThreadPoolExecutor p =
319             new ThreadPoolExecutor(1, 2,
320                                    LONG_DELAY_MS, MILLISECONDS,
321                                    new ArrayBlockingQueue<Runnable>(10));
322         try (PoolCleaner cleaner = cleaner(p)) {
323             try {
324                 p.setRejectedExecutionHandler(null);
325                 shouldThrow();
326             } catch (NullPointerException success) {}
327         }
328     }
329 
330     /**
331      * getLargestPoolSize increases, but doesn't overestimate, when
332      * multiple threads active
333      */
334     public void testGetLargestPoolSize() throws InterruptedException {
335         final int THREADS = 3;
336         final CountDownLatch done = new CountDownLatch(1);
337         final ThreadPoolExecutor p =
338             new ThreadPoolExecutor(THREADS, THREADS,
339                                    LONG_DELAY_MS, MILLISECONDS,
340                                    new ArrayBlockingQueue<Runnable>(10));
341         try (PoolCleaner cleaner = cleaner(p, done)) {
342             assertEquals(0, p.getLargestPoolSize());
343             final CountDownLatch threadsStarted = new CountDownLatch(THREADS);
344             for (int i = 0; i < THREADS; i++)
345                 p.execute(new CheckedRunnable() {
346                     public void realRun() throws InterruptedException {
347                         threadsStarted.countDown();
348                         await(done);
349                         assertEquals(THREADS, p.getLargestPoolSize());
350                     }});
351             await(threadsStarted);
352             assertEquals(THREADS, p.getLargestPoolSize());
353         }
354         assertEquals(THREADS, p.getLargestPoolSize());
355     }
356 
357     /**
358      * getMaximumPoolSize returns value given in constructor if not
359      * otherwise set
360      */
361     public void testGetMaximumPoolSize() {
362         final ThreadPoolExecutor p =
363             new ThreadPoolExecutor(2, 3,
364                                    LONG_DELAY_MS, MILLISECONDS,
365                                    new ArrayBlockingQueue<Runnable>(10));
366         try (PoolCleaner cleaner = cleaner(p)) {
367             assertEquals(3, p.getMaximumPoolSize());
368             p.setMaximumPoolSize(5);
369             assertEquals(5, p.getMaximumPoolSize());
370             p.setMaximumPoolSize(4);
371             assertEquals(4, p.getMaximumPoolSize());
372         }
373     }
374 
375     /**
376      * getPoolSize increases, but doesn't overestimate, when threads
377      * become active
378      */
379     public void testGetPoolSize() throws InterruptedException {
380         final CountDownLatch done = new CountDownLatch(1);
381         final ThreadPoolExecutor p =
382             new ThreadPoolExecutor(1, 1,
383                                    LONG_DELAY_MS, MILLISECONDS,
384                                    new ArrayBlockingQueue<Runnable>(10));
385         try (PoolCleaner cleaner = cleaner(p, done)) {
386             assertEquals(0, p.getPoolSize());
387             final CountDownLatch threadStarted = new CountDownLatch(1);
388             p.execute(new CheckedRunnable() {
389                 public void realRun() throws InterruptedException {
390                     threadStarted.countDown();
391                     assertEquals(1, p.getPoolSize());
392                     await(done);
393                 }});
394             await(threadStarted);
395             assertEquals(1, p.getPoolSize());
396         }
397     }
398 
399     /**
400      * getTaskCount increases, but doesn't overestimate, when tasks submitted
401      */
402     public void testGetTaskCount() throws InterruptedException {
403         final int TASKS = 3;
404         final CountDownLatch done = new CountDownLatch(1);
405         final ThreadPoolExecutor p =
406             new ThreadPoolExecutor(1, 1,
407                                    LONG_DELAY_MS, MILLISECONDS,
408                                    new ArrayBlockingQueue<Runnable>(10));
409         try (PoolCleaner cleaner = cleaner(p, done)) {
410             final CountDownLatch threadStarted = new CountDownLatch(1);
411             assertEquals(0, p.getTaskCount());
412             assertEquals(0, p.getCompletedTaskCount());
413             p.execute(new CheckedRunnable() {
414                 public void realRun() throws InterruptedException {
415                     threadStarted.countDown();
416                     await(done);
417                 }});
418             await(threadStarted);
419             assertEquals(1, p.getTaskCount());
420             assertEquals(0, p.getCompletedTaskCount());
421             for (int i = 0; i < TASKS; i++) {
422                 assertEquals(1 + i, p.getTaskCount());
423                 p.execute(new CheckedRunnable() {
424                     public void realRun() throws InterruptedException {
425                         threadStarted.countDown();
426                         assertEquals(1 + TASKS, p.getTaskCount());
427                         await(done);
428                     }});
429             }
430             assertEquals(1 + TASKS, p.getTaskCount());
431             assertEquals(0, p.getCompletedTaskCount());
432         }
433         assertEquals(1 + TASKS, p.getTaskCount());
434         assertEquals(1 + TASKS, p.getCompletedTaskCount());
435     }
436 
437     /**
438      * isShutdown is false before shutdown, true after
439      */
440     public void testIsShutdown() {
441         final ThreadPoolExecutor p =
442             new ThreadPoolExecutor(1, 1,
443                                    LONG_DELAY_MS, MILLISECONDS,
444                                    new ArrayBlockingQueue<Runnable>(10));
445         try (PoolCleaner cleaner = cleaner(p)) {
446             assertFalse(p.isShutdown());
447             try { p.shutdown(); } catch (SecurityException ok) { return; }
448             assertTrue(p.isShutdown());
449         }
450     }
451 
452     /**
453      * awaitTermination on a non-shutdown pool times out
454      */
455     public void testAwaitTermination_timesOut() throws InterruptedException {
456         final ThreadPoolExecutor p =
457             new ThreadPoolExecutor(1, 1,
458                                    LONG_DELAY_MS, MILLISECONDS,
459                                    new ArrayBlockingQueue<Runnable>(10));
460         try (PoolCleaner cleaner = cleaner(p)) {
461             assertFalse(p.isTerminated());
462             assertFalse(p.awaitTermination(Long.MIN_VALUE, NANOSECONDS));
463             assertFalse(p.awaitTermination(Long.MIN_VALUE, MILLISECONDS));
464             assertFalse(p.awaitTermination(-1L, NANOSECONDS));
465             assertFalse(p.awaitTermination(-1L, MILLISECONDS));
466             assertFalse(p.awaitTermination(0L, NANOSECONDS));
467             assertFalse(p.awaitTermination(0L, MILLISECONDS));
468             long timeoutNanos = 999999L;
469             long startTime = System.nanoTime();
470             assertFalse(p.awaitTermination(timeoutNanos, NANOSECONDS));
471             assertTrue(System.nanoTime() - startTime >= timeoutNanos);
472             assertFalse(p.isTerminated());
473             startTime = System.nanoTime();
474             long timeoutMillis = timeoutMillis();
475             assertFalse(p.awaitTermination(timeoutMillis, MILLISECONDS));
476             assertTrue(millisElapsedSince(startTime) >= timeoutMillis);
477             assertFalse(p.isTerminated());
478             try { p.shutdown(); } catch (SecurityException ok) { return; }
479             assertTrue(p.awaitTermination(LONG_DELAY_MS, MILLISECONDS));
480             assertTrue(p.isTerminated());
481         }
482     }
483 
484     /**
485      * isTerminated is false before termination, true after
486      */
487     public void testIsTerminated() throws InterruptedException {
488         final ThreadPoolExecutor p =
489             new ThreadPoolExecutor(1, 1,
490                                    LONG_DELAY_MS, MILLISECONDS,
491                                    new ArrayBlockingQueue<Runnable>(10));
492         try (PoolCleaner cleaner = cleaner(p)) {
493             final CountDownLatch threadStarted = new CountDownLatch(1);
494             final CountDownLatch done = new CountDownLatch(1);
495             assertFalse(p.isTerminating());
496             p.execute(new CheckedRunnable() {
497                 public void realRun() throws InterruptedException {
498                     assertFalse(p.isTerminating());
499                     threadStarted.countDown();
500                     await(done);
501                 }});
502             await(threadStarted);
503             assertFalse(p.isTerminating());
504             done.countDown();
505             try { p.shutdown(); } catch (SecurityException ok) { return; }
506             assertTrue(p.awaitTermination(LONG_DELAY_MS, MILLISECONDS));
507             assertTrue(p.isTerminated());
508             assertFalse(p.isTerminating());
509         }
510     }
511 
512     /**
513      * isTerminating is not true when running or when terminated
514      */
515     public void testIsTerminating() throws InterruptedException {
516         final ThreadPoolExecutor p =
517             new ThreadPoolExecutor(1, 1,
518                                    LONG_DELAY_MS, MILLISECONDS,
519                                    new ArrayBlockingQueue<Runnable>(10));
520         try (PoolCleaner cleaner = cleaner(p)) {
521             final CountDownLatch threadStarted = new CountDownLatch(1);
522             final CountDownLatch done = new CountDownLatch(1);
523             assertFalse(p.isTerminating());
524             p.execute(new CheckedRunnable() {
525                 public void realRun() throws InterruptedException {
526                     assertFalse(p.isTerminating());
527                     threadStarted.countDown();
528                     await(done);
529                 }});
530             await(threadStarted);
531             assertFalse(p.isTerminating());
532             done.countDown();
533             try { p.shutdown(); } catch (SecurityException ok) { return; }
534             assertTrue(p.awaitTermination(LONG_DELAY_MS, MILLISECONDS));
535             assertTrue(p.isTerminated());
536             assertFalse(p.isTerminating());
537         }
538     }
539 
540     /**
541      * getQueue returns the work queue, which contains queued tasks
542      */
543     public void testGetQueue() throws InterruptedException {
544         final CountDownLatch done = new CountDownLatch(1);
545         final BlockingQueue<Runnable> q = new ArrayBlockingQueue<Runnable>(10);
546         final ThreadPoolExecutor p =
547             new ThreadPoolExecutor(1, 1,
548                                    LONG_DELAY_MS, MILLISECONDS,
549                                    q);
550         try (PoolCleaner cleaner = cleaner(p, done)) {
551             final CountDownLatch threadStarted = new CountDownLatch(1);
552             FutureTask[] tasks = new FutureTask[5];
553             for (int i = 0; i < tasks.length; i++) {
554                 Callable task = new CheckedCallable<Boolean>() {
555                     public Boolean realCall() throws InterruptedException {
556                         threadStarted.countDown();
557                         assertSame(q, p.getQueue());
558                         await(done);
559                         return Boolean.TRUE;
560                     }};
561                 tasks[i] = new FutureTask(task);
562                 p.execute(tasks[i]);
563             }
564             await(threadStarted);
565             assertSame(q, p.getQueue());
566             assertFalse(q.contains(tasks[0]));
567             assertTrue(q.contains(tasks[tasks.length - 1]));
568             assertEquals(tasks.length - 1, q.size());
569         }
570     }
571 
572     /**
573      * remove(task) removes queued task, and fails to remove active task
574      */
575     public void testRemove() throws InterruptedException {
576         final CountDownLatch done = new CountDownLatch(1);
577         BlockingQueue<Runnable> q = new ArrayBlockingQueue<Runnable>(10);
578         final ThreadPoolExecutor p =
579             new ThreadPoolExecutor(1, 1,
580                                    LONG_DELAY_MS, MILLISECONDS,
581                                    q);
582         try (PoolCleaner cleaner = cleaner(p, done)) {
583             Runnable[] tasks = new Runnable[6];
584             final CountDownLatch threadStarted = new CountDownLatch(1);
585             for (int i = 0; i < tasks.length; i++) {
586                 tasks[i] = new CheckedRunnable() {
587                     public void realRun() throws InterruptedException {
588                         threadStarted.countDown();
589                         await(done);
590                     }};
591                 p.execute(tasks[i]);
592             }
593             await(threadStarted);
594             assertFalse(p.remove(tasks[0]));
595             assertTrue(q.contains(tasks[4]));
596             assertTrue(q.contains(tasks[3]));
597             assertTrue(p.remove(tasks[4]));
598             assertFalse(p.remove(tasks[4]));
599             assertFalse(q.contains(tasks[4]));
600             assertTrue(q.contains(tasks[3]));
601             assertTrue(p.remove(tasks[3]));
602             assertFalse(q.contains(tasks[3]));
603         }
604     }
605 
606     /**
607      * purge removes cancelled tasks from the queue
608      */
609     public void testPurge() throws InterruptedException {
610         final CountDownLatch threadStarted = new CountDownLatch(1);
611         final CountDownLatch done = new CountDownLatch(1);
612         final BlockingQueue<Runnable> q = new ArrayBlockingQueue<Runnable>(10);
613         final ThreadPoolExecutor p =
614             new ThreadPoolExecutor(1, 1,
615                                    LONG_DELAY_MS, MILLISECONDS,
616                                    q);
617         try (PoolCleaner cleaner = cleaner(p, done)) {
618             FutureTask[] tasks = new FutureTask[5];
619             for (int i = 0; i < tasks.length; i++) {
620                 Callable task = new CheckedCallable<Boolean>() {
621                     public Boolean realCall() throws InterruptedException {
622                         threadStarted.countDown();
623                         await(done);
624                         return Boolean.TRUE;
625                     }};
626                 tasks[i] = new FutureTask(task);
627                 p.execute(tasks[i]);
628             }
629             await(threadStarted);
630             assertEquals(tasks.length, p.getTaskCount());
631             assertEquals(tasks.length - 1, q.size());
632             assertEquals(1L, p.getActiveCount());
633             assertEquals(0L, p.getCompletedTaskCount());
634             tasks[4].cancel(true);
635             tasks[3].cancel(false);
636             p.purge();
637             assertEquals(tasks.length - 3, q.size());
638             assertEquals(tasks.length - 2, p.getTaskCount());
639             p.purge();         // Nothing to do
640             assertEquals(tasks.length - 3, q.size());
641             assertEquals(tasks.length - 2, p.getTaskCount());
642         }
643     }
644 
645     /**
646      * shutdownNow returns a list containing tasks that were not run,
647      * and those tasks are drained from the queue
648      */
649     public void testShutdownNow() throws InterruptedException {
650         final int poolSize = 2;
651         final int count = 5;
652         final AtomicInteger ran = new AtomicInteger(0);
653         final ThreadPoolExecutor p =
654             new ThreadPoolExecutor(poolSize, poolSize,
655                                    LONG_DELAY_MS, MILLISECONDS,
656                                    new ArrayBlockingQueue<Runnable>(10));
657         final CountDownLatch threadsStarted = new CountDownLatch(poolSize);
658         Runnable waiter = new CheckedRunnable() { public void realRun() {
659             threadsStarted.countDown();
660             try {
661                 MILLISECONDS.sleep(2 * LONG_DELAY_MS);
662             } catch (InterruptedException success) {}
663             ran.getAndIncrement();
664         }};
665         for (int i = 0; i < count; i++)
666             p.execute(waiter);
667         await(threadsStarted);
668         assertEquals(poolSize, p.getActiveCount());
669         assertEquals(0, p.getCompletedTaskCount());
670         final List<Runnable> queuedTasks;
671         try {
672             queuedTasks = p.shutdownNow();
673         } catch (SecurityException ok) {
674             return; // Allowed in case test doesn't have privs
675         }
676         assertTrue(p.isShutdown());
677         assertTrue(p.getQueue().isEmpty());
678         assertEquals(count - poolSize, queuedTasks.size());
679         assertTrue(p.awaitTermination(LONG_DELAY_MS, MILLISECONDS));
680         assertTrue(p.isTerminated());
681         assertEquals(poolSize, ran.get());
682         assertEquals(poolSize, p.getCompletedTaskCount());
683     }
684 
685     // Exception Tests
686 
687     /**
688      * Constructor throws if corePoolSize argument is less than zero
689      */
690     public void testConstructor1() {
691         try {
692             new ThreadPoolExecutor(-1, 1, 1L, SECONDS,
693                                    new ArrayBlockingQueue<Runnable>(10));
694             shouldThrow();
695         } catch (IllegalArgumentException success) {}
696     }
697 
698     /**
699      * Constructor throws if maximumPoolSize is less than zero
700      */
701     public void testConstructor2() {
702         try {
703             new ThreadPoolExecutor(1, -1, 1L, SECONDS,
704                                    new ArrayBlockingQueue<Runnable>(10));
705             shouldThrow();
706         } catch (IllegalArgumentException success) {}
707     }
708 
709     /**
710      * Constructor throws if maximumPoolSize is equal to zero
711      */
712     public void testConstructor3() {
713         try {
714             new ThreadPoolExecutor(1, 0, 1L, SECONDS,
715                                    new ArrayBlockingQueue<Runnable>(10));
716             shouldThrow();
717         } catch (IllegalArgumentException success) {}
718     }
719 
720     /**
721      * Constructor throws if keepAliveTime is less than zero
722      */
723     public void testConstructor4() {
724         try {
725             new ThreadPoolExecutor(1, 2, -1L, SECONDS,
726                                    new ArrayBlockingQueue<Runnable>(10));
727             shouldThrow();
728         } catch (IllegalArgumentException success) {}
729     }
730 
731     /**
732      * Constructor throws if corePoolSize is greater than the maximumPoolSize
733      */
734     public void testConstructor5() {
735         try {
736             new ThreadPoolExecutor(2, 1, 1L, SECONDS,
737                                    new ArrayBlockingQueue<Runnable>(10));
738             shouldThrow();
739         } catch (IllegalArgumentException success) {}
740     }
741 
742     /**
743      * Constructor throws if workQueue is set to null
744      */
745     public void testConstructorNullPointerException() {
746         try {
747             new ThreadPoolExecutor(1, 2, 1L, SECONDS,
748                                    (BlockingQueue) null);
749             shouldThrow();
750         } catch (NullPointerException success) {}
751     }
752 
753     /**
754      * Constructor throws if corePoolSize argument is less than zero
755      */
756     public void testConstructor6() {
757         try {
758             new ThreadPoolExecutor(-1, 1, 1L, SECONDS,
759                                    new ArrayBlockingQueue<Runnable>(10),
760                                    new SimpleThreadFactory());
761             shouldThrow();
762         } catch (IllegalArgumentException success) {}
763     }
764 
765     /**
766      * Constructor throws if maximumPoolSize is less than zero
767      */
768     public void testConstructor7() {
769         try {
770             new ThreadPoolExecutor(1, -1, 1L, SECONDS,
771                                    new ArrayBlockingQueue<Runnable>(10),
772                                    new SimpleThreadFactory());
773             shouldThrow();
774         } catch (IllegalArgumentException success) {}
775     }
776 
777     /**
778      * Constructor throws if maximumPoolSize is equal to zero
779      */
780     public void testConstructor8() {
781         try {
782             new ThreadPoolExecutor(1, 0, 1L, SECONDS,
783                                    new ArrayBlockingQueue<Runnable>(10),
784                                    new SimpleThreadFactory());
785             shouldThrow();
786         } catch (IllegalArgumentException success) {}
787     }
788 
789     /**
790      * Constructor throws if keepAliveTime is less than zero
791      */
792     public void testConstructor9() {
793         try {
794             new ThreadPoolExecutor(1, 2, -1L, SECONDS,
795                                    new ArrayBlockingQueue<Runnable>(10),
796                                    new SimpleThreadFactory());
797             shouldThrow();
798         } catch (IllegalArgumentException success) {}
799     }
800 
801     /**
802      * Constructor throws if corePoolSize is greater than the maximumPoolSize
803      */
804     public void testConstructor10() {
805         try {
806             new ThreadPoolExecutor(2, 1, 1L, SECONDS,
807                                    new ArrayBlockingQueue<Runnable>(10),
808                                    new SimpleThreadFactory());
809             shouldThrow();
810         } catch (IllegalArgumentException success) {}
811     }
812 
813     /**
814      * Constructor throws if workQueue is set to null
815      */
816     public void testConstructorNullPointerException2() {
817         try {
818             new ThreadPoolExecutor(1, 2, 1L, SECONDS,
819                                    (BlockingQueue) null,
820                                    new SimpleThreadFactory());
821             shouldThrow();
822         } catch (NullPointerException success) {}
823     }
824 
825     /**
826      * Constructor throws if threadFactory is set to null
827      */
828     public void testConstructorNullPointerException3() {
829         try {
830             new ThreadPoolExecutor(1, 2, 1L, SECONDS,
831                                    new ArrayBlockingQueue<Runnable>(10),
832                                    (ThreadFactory) null);
833             shouldThrow();
834         } catch (NullPointerException success) {}
835     }
836 
837     /**
838      * Constructor throws if corePoolSize argument is less than zero
839      */
840     public void testConstructor11() {
841         try {
842             new ThreadPoolExecutor(-1, 1, 1L, SECONDS,
843                                    new ArrayBlockingQueue<Runnable>(10),
844                                    new NoOpREHandler());
845             shouldThrow();
846         } catch (IllegalArgumentException success) {}
847     }
848 
849     /**
850      * Constructor throws if maximumPoolSize is less than zero
851      */
852     public void testConstructor12() {
853         try {
854             new ThreadPoolExecutor(1, -1, 1L, SECONDS,
855                                    new ArrayBlockingQueue<Runnable>(10),
856                                    new NoOpREHandler());
857             shouldThrow();
858         } catch (IllegalArgumentException success) {}
859     }
860 
861     /**
862      * Constructor throws if maximumPoolSize is equal to zero
863      */
864     public void testConstructor13() {
865         try {
866             new ThreadPoolExecutor(1, 0, 1L, SECONDS,
867                                    new ArrayBlockingQueue<Runnable>(10),
868                                    new NoOpREHandler());
869             shouldThrow();
870         } catch (IllegalArgumentException success) {}
871     }
872 
873     /**
874      * Constructor throws if keepAliveTime is less than zero
875      */
876     public void testConstructor14() {
877         try {
878             new ThreadPoolExecutor(1, 2, -1L, SECONDS,
879                                    new ArrayBlockingQueue<Runnable>(10),
880                                    new NoOpREHandler());
881             shouldThrow();
882         } catch (IllegalArgumentException success) {}
883     }
884 
885     /**
886      * Constructor throws if corePoolSize is greater than the maximumPoolSize
887      */
888     public void testConstructor15() {
889         try {
890             new ThreadPoolExecutor(2, 1, 1L, SECONDS,
891                                    new ArrayBlockingQueue<Runnable>(10),
892                                    new NoOpREHandler());
893             shouldThrow();
894         } catch (IllegalArgumentException success) {}
895     }
896 
897     /**
898      * Constructor throws if workQueue is set to null
899      */
900     public void testConstructorNullPointerException4() {
901         try {
902             new ThreadPoolExecutor(1, 2, 1L, SECONDS,
903                                    (BlockingQueue) null,
904                                    new NoOpREHandler());
905             shouldThrow();
906         } catch (NullPointerException success) {}
907     }
908 
909     /**
910      * Constructor throws if handler is set to null
911      */
912     public void testConstructorNullPointerException5() {
913         try {
914             new ThreadPoolExecutor(1, 2, 1L, SECONDS,
915                                    new ArrayBlockingQueue<Runnable>(10),
916                                    (RejectedExecutionHandler) null);
917             shouldThrow();
918         } catch (NullPointerException success) {}
919     }
920 
921     /**
922      * Constructor throws if corePoolSize argument is less than zero
923      */
924     public void testConstructor16() {
925         try {
926             new ThreadPoolExecutor(-1, 1, 1L, SECONDS,
927                                    new ArrayBlockingQueue<Runnable>(10),
928                                    new SimpleThreadFactory(),
929                                    new NoOpREHandler());
930             shouldThrow();
931         } catch (IllegalArgumentException success) {}
932     }
933 
934     /**
935      * Constructor throws if maximumPoolSize is less than zero
936      */
937     public void testConstructor17() {
938         try {
939             new ThreadPoolExecutor(1, -1, 1L, SECONDS,
940                                    new ArrayBlockingQueue<Runnable>(10),
941                                    new SimpleThreadFactory(),
942                                    new NoOpREHandler());
943             shouldThrow();
944         } catch (IllegalArgumentException success) {}
945     }
946 
947     /**
948      * Constructor throws if maximumPoolSize is equal to zero
949      */
950     public void testConstructor18() {
951         try {
952             new ThreadPoolExecutor(1, 0, 1L, SECONDS,
953                                    new ArrayBlockingQueue<Runnable>(10),
954                                    new SimpleThreadFactory(),
955                                    new NoOpREHandler());
956             shouldThrow();
957         } catch (IllegalArgumentException success) {}
958     }
959 
960     /**
961      * Constructor throws if keepAliveTime is less than zero
962      */
963     public void testConstructor19() {
964         try {
965             new ThreadPoolExecutor(1, 2, -1L, SECONDS,
966                                    new ArrayBlockingQueue<Runnable>(10),
967                                    new SimpleThreadFactory(),
968                                    new NoOpREHandler());
969             shouldThrow();
970         } catch (IllegalArgumentException success) {}
971     }
972 
973     /**
974      * Constructor throws if corePoolSize is greater than the maximumPoolSize
975      */
976     public void testConstructor20() {
977         try {
978             new ThreadPoolExecutor(2, 1, 1L, SECONDS,
979                                    new ArrayBlockingQueue<Runnable>(10),
980                                    new SimpleThreadFactory(),
981                                    new NoOpREHandler());
982             shouldThrow();
983         } catch (IllegalArgumentException success) {}
984     }
985 
986     /**
987      * Constructor throws if workQueue is null
988      */
989     public void testConstructorNullPointerException6() {
990         try {
991             new ThreadPoolExecutor(1, 2, 1L, SECONDS,
992                                    (BlockingQueue) null,
993                                    new SimpleThreadFactory(),
994                                    new NoOpREHandler());
995             shouldThrow();
996         } catch (NullPointerException success) {}
997     }
998 
999     /**
1000      * Constructor throws if handler is null
1001      */
1002     public void testConstructorNullPointerException7() {
1003         try {
1004             new ThreadPoolExecutor(1, 2, 1L, SECONDS,
1005                                    new ArrayBlockingQueue<Runnable>(10),
1006                                    new SimpleThreadFactory(),
1007                                    (RejectedExecutionHandler) null);
1008             shouldThrow();
1009         } catch (NullPointerException success) {}
1010     }
1011 
1012     /**
1013      * Constructor throws if ThreadFactory is null
1014      */
1015     public void testConstructorNullPointerException8() {
1016         try {
1017             new ThreadPoolExecutor(1, 2, 1L, SECONDS,
1018                                    new ArrayBlockingQueue<Runnable>(10),
1019                                    (ThreadFactory) null,
1020                                    new NoOpREHandler());
1021             shouldThrow();
1022         } catch (NullPointerException success) {}
1023     }
1024 
1025     /**
1026      * get of submitted callable throws InterruptedException if interrupted
1027      */
1028     public void testInterruptedSubmit() throws InterruptedException {
1029         final CountDownLatch done = new CountDownLatch(1);
1030         final ThreadPoolExecutor p =
1031             new ThreadPoolExecutor(1, 1,
1032                                    60, SECONDS,
1033                                    new ArrayBlockingQueue<Runnable>(10));
1034 
1035         try (PoolCleaner cleaner = cleaner(p, done)) {
1036             final CountDownLatch threadStarted = new CountDownLatch(1);
1037             Thread t = newStartedThread(new CheckedInterruptedRunnable() {
1038                 public void realRun() throws Exception {
1039                     Callable task = new CheckedCallable<Boolean>() {
1040                         public Boolean realCall() throws InterruptedException {
1041                             threadStarted.countDown();
1042                             await(done);
1043                             return Boolean.TRUE;
1044                         }};
1045                     p.submit(task).get();
1046                 }});
1047 
1048             await(threadStarted);
1049             t.interrupt();
1050             awaitTermination(t);
1051         }
1052     }
1053 
1054     /**
1055      * execute throws RejectedExecutionException if saturated.
1056      */
1057     public void testSaturatedExecute() {
1058         final CountDownLatch done = new CountDownLatch(1);
1059         final ThreadPoolExecutor p =
1060             new ThreadPoolExecutor(1, 1,
1061                                    LONG_DELAY_MS, MILLISECONDS,
1062                                    new ArrayBlockingQueue<Runnable>(1));
1063         try (PoolCleaner cleaner = cleaner(p, done)) {
1064             Runnable task = new CheckedRunnable() {
1065                 public void realRun() throws InterruptedException {
1066                     await(done);
1067                 }};
1068             for (int i = 0; i < 2; ++i)
1069                 p.execute(task);
1070             for (int i = 0; i < 2; ++i) {
1071                 try {
1072                     p.execute(task);
1073                     shouldThrow();
1074                 } catch (RejectedExecutionException success) {}
1075                 assertTrue(p.getTaskCount() <= 2);
1076             }
1077         }
1078     }
1079 
1080     /**
1081      * submit(runnable) throws RejectedExecutionException if saturated.
1082      */
1083     public void testSaturatedSubmitRunnable() {
1084         final CountDownLatch done = new CountDownLatch(1);
1085         final ThreadPoolExecutor p =
1086             new ThreadPoolExecutor(1, 1,
1087                                    LONG_DELAY_MS, MILLISECONDS,
1088                                    new ArrayBlockingQueue<Runnable>(1));
1089         try (PoolCleaner cleaner = cleaner(p, done)) {
1090             Runnable task = new CheckedRunnable() {
1091                 public void realRun() throws InterruptedException {
1092                     await(done);
1093                 }};
1094             for (int i = 0; i < 2; ++i)
1095                 p.submit(task);
1096             for (int i = 0; i < 2; ++i) {
1097                 try {
1098                     p.execute(task);
1099                     shouldThrow();
1100                 } catch (RejectedExecutionException success) {}
1101                 assertTrue(p.getTaskCount() <= 2);
1102             }
1103         }
1104     }
1105 
1106     /**
1107      * submit(callable) throws RejectedExecutionException if saturated.
1108      */
1109     public void testSaturatedSubmitCallable() {
1110         final CountDownLatch done = new CountDownLatch(1);
1111         final ThreadPoolExecutor p =
1112             new ThreadPoolExecutor(1, 1,
1113                                    LONG_DELAY_MS, MILLISECONDS,
1114                                    new ArrayBlockingQueue<Runnable>(1));
1115         try (PoolCleaner cleaner = cleaner(p, done)) {
1116             Runnable task = new CheckedRunnable() {
1117                 public void realRun() throws InterruptedException {
1118                     await(done);
1119                 }};
1120             for (int i = 0; i < 2; ++i)
1121                 p.submit(Executors.callable(task));
1122             for (int i = 0; i < 2; ++i) {
1123                 try {
1124                     p.execute(task);
1125                     shouldThrow();
1126                 } catch (RejectedExecutionException success) {}
1127                 assertTrue(p.getTaskCount() <= 2);
1128             }
1129         }
1130     }
1131 
1132     /**
1133      * executor using CallerRunsPolicy runs task if saturated.
1134      */
1135     public void testSaturatedExecute2() {
1136         final ThreadPoolExecutor p =
1137             new ThreadPoolExecutor(1, 1,
1138                                    LONG_DELAY_MS,
1139                                    MILLISECONDS,
1140                                    new ArrayBlockingQueue<Runnable>(1),
1141                                    new ThreadPoolExecutor.CallerRunsPolicy());
1142         try (PoolCleaner cleaner = cleaner(p)) {
1143             final CountDownLatch done = new CountDownLatch(1);
1144             Runnable blocker = new CheckedRunnable() {
1145                 public void realRun() throws InterruptedException {
1146                     await(done);
1147                 }};
1148             p.execute(blocker);
1149             TrackedNoOpRunnable[] tasks = new TrackedNoOpRunnable[5];
1150             for (int i = 0; i < tasks.length; i++)
1151                 tasks[i] = new TrackedNoOpRunnable();
1152             for (int i = 0; i < tasks.length; i++)
1153                 p.execute(tasks[i]);
1154             for (int i = 1; i < tasks.length; i++)
1155                 assertTrue(tasks[i].done);
1156             assertFalse(tasks[0].done); // waiting in queue
1157             done.countDown();
1158         }
1159     }
1160 
1161     /**
1162      * executor using DiscardPolicy drops task if saturated.
1163      */
1164     public void testSaturatedExecute3() {
1165         final CountDownLatch done = new CountDownLatch(1);
1166         final TrackedNoOpRunnable[] tasks = new TrackedNoOpRunnable[5];
1167         for (int i = 0; i < tasks.length; ++i)
1168             tasks[i] = new TrackedNoOpRunnable();
1169         final ThreadPoolExecutor p =
1170             new ThreadPoolExecutor(1, 1,
1171                           LONG_DELAY_MS, MILLISECONDS,
1172                           new ArrayBlockingQueue<Runnable>(1),
1173                           new ThreadPoolExecutor.DiscardPolicy());
1174         try (PoolCleaner cleaner = cleaner(p, done)) {
1175             p.execute(awaiter(done));
1176 
1177             for (TrackedNoOpRunnable task : tasks)
1178                 p.execute(task);
1179             for (int i = 1; i < tasks.length; i++)
1180                 assertFalse(tasks[i].done);
1181         }
1182         for (int i = 1; i < tasks.length; i++)
1183             assertFalse(tasks[i].done);
1184         assertTrue(tasks[0].done); // was waiting in queue
1185     }
1186 
1187     /**
1188      * executor using DiscardOldestPolicy drops oldest task if saturated.
1189      */
1190     public void testSaturatedExecute4() {
1191         final CountDownLatch done = new CountDownLatch(1);
1192         LatchAwaiter r1 = awaiter(done);
1193         LatchAwaiter r2 = awaiter(done);
1194         LatchAwaiter r3 = awaiter(done);
1195         final ThreadPoolExecutor p =
1196             new ThreadPoolExecutor(1, 1,
1197                                    LONG_DELAY_MS, MILLISECONDS,
1198                                    new ArrayBlockingQueue<Runnable>(1),
1199                                    new ThreadPoolExecutor.DiscardOldestPolicy());
1200         try (PoolCleaner cleaner = cleaner(p, done)) {
1201             assertEquals(LatchAwaiter.NEW, r1.state);
1202             assertEquals(LatchAwaiter.NEW, r2.state);
1203             assertEquals(LatchAwaiter.NEW, r3.state);
1204             p.execute(r1);
1205             p.execute(r2);
1206             assertTrue(p.getQueue().contains(r2));
1207             p.execute(r3);
1208             assertFalse(p.getQueue().contains(r2));
1209             assertTrue(p.getQueue().contains(r3));
1210         }
1211         assertEquals(LatchAwaiter.DONE, r1.state);
1212         assertEquals(LatchAwaiter.NEW, r2.state);
1213         assertEquals(LatchAwaiter.DONE, r3.state);
1214     }
1215 
1216     /**
1217      * execute throws RejectedExecutionException if shutdown
1218      */
1219     public void testRejectedExecutionExceptionOnShutdown() {
1220         final ThreadPoolExecutor p =
1221             new ThreadPoolExecutor(1, 1,
1222                                    LONG_DELAY_MS, MILLISECONDS,
1223                                    new ArrayBlockingQueue<Runnable>(1));
1224         try { p.shutdown(); } catch (SecurityException ok) { return; }
1225         try (PoolCleaner cleaner = cleaner(p)) {
1226             try {
1227                 p.execute(new NoOpRunnable());
1228                 shouldThrow();
1229             } catch (RejectedExecutionException success) {}
1230         }
1231     }
1232 
1233     /**
1234      * execute using CallerRunsPolicy drops task on shutdown
1235      */
1236     public void testCallerRunsOnShutdown() {
1237         RejectedExecutionHandler h = new ThreadPoolExecutor.CallerRunsPolicy();
1238         final ThreadPoolExecutor p =
1239             new ThreadPoolExecutor(1, 1,
1240                                    LONG_DELAY_MS, MILLISECONDS,
1241                                    new ArrayBlockingQueue<Runnable>(1), h);
1242 
1243         try { p.shutdown(); } catch (SecurityException ok) { return; }
1244         try (PoolCleaner cleaner = cleaner(p)) {
1245             TrackedNoOpRunnable r = new TrackedNoOpRunnable();
1246             p.execute(r);
1247             assertFalse(r.done);
1248         }
1249     }
1250 
1251     /**
1252      * execute using DiscardPolicy drops task on shutdown
1253      */
1254     public void testDiscardOnShutdown() {
1255         final ThreadPoolExecutor p =
1256             new ThreadPoolExecutor(1, 1,
1257                                    LONG_DELAY_MS, MILLISECONDS,
1258                                    new ArrayBlockingQueue<Runnable>(1),
1259                                    new ThreadPoolExecutor.DiscardPolicy());
1260 
1261         try { p.shutdown(); } catch (SecurityException ok) { return; }
1262         try (PoolCleaner cleaner = cleaner(p)) {
1263             TrackedNoOpRunnable r = new TrackedNoOpRunnable();
1264             p.execute(r);
1265             assertFalse(r.done);
1266         }
1267     }
1268 
1269     /**
1270      * execute using DiscardOldestPolicy drops task on shutdown
1271      */
1272     public void testDiscardOldestOnShutdown() {
1273         final ThreadPoolExecutor p =
1274             new ThreadPoolExecutor(1, 1,
1275                                    LONG_DELAY_MS, MILLISECONDS,
1276                                    new ArrayBlockingQueue<Runnable>(1),
1277                                    new ThreadPoolExecutor.DiscardOldestPolicy());
1278 
1279         try { p.shutdown(); } catch (SecurityException ok) { return; }
1280         try (PoolCleaner cleaner = cleaner(p)) {
1281             TrackedNoOpRunnable r = new TrackedNoOpRunnable();
1282             p.execute(r);
1283             assertFalse(r.done);
1284         }
1285     }
1286 
1287     /**
1288      * execute(null) throws NPE
1289      */
1290     public void testExecuteNull() {
1291         final ThreadPoolExecutor p =
1292             new ThreadPoolExecutor(1, 2,
1293                                    1L, SECONDS,
1294                                    new ArrayBlockingQueue<Runnable>(10));
1295         try (PoolCleaner cleaner = cleaner(p)) {
1296             try {
1297                 p.execute(null);
1298                 shouldThrow();
1299             } catch (NullPointerException success) {}
1300         }
1301     }
1302 
1303     /**
1304      * setCorePoolSize of negative value throws IllegalArgumentException
1305      */
1306     public void testCorePoolSizeIllegalArgumentException() {
1307         final ThreadPoolExecutor p =
1308             new ThreadPoolExecutor(1, 2,
1309                                    LONG_DELAY_MS, MILLISECONDS,
1310                                    new ArrayBlockingQueue<Runnable>(10));
1311         try (PoolCleaner cleaner = cleaner(p)) {
1312             try {
1313                 p.setCorePoolSize(-1);
1314                 shouldThrow();
1315             } catch (IllegalArgumentException success) {}
1316         }
1317     }
1318 
1319     /**
1320      * setMaximumPoolSize(int) throws IllegalArgumentException if
1321      * given a value less the core pool size
1322      */
1323     public void testMaximumPoolSizeIllegalArgumentException() {
1324         final ThreadPoolExecutor p =
1325             new ThreadPoolExecutor(2, 3,
1326                                    LONG_DELAY_MS, MILLISECONDS,
1327                                    new ArrayBlockingQueue<Runnable>(10));
1328         try (PoolCleaner cleaner = cleaner(p)) {
1329             try {
1330                 p.setMaximumPoolSize(1);
1331                 shouldThrow();
1332             } catch (IllegalArgumentException success) {}
1333         }
1334     }
1335 
1336     /**
1337      * setMaximumPoolSize throws IllegalArgumentException
1338      * if given a negative value
1339      */
1340     public void testMaximumPoolSizeIllegalArgumentException2() {
1341         final ThreadPoolExecutor p =
1342             new ThreadPoolExecutor(2, 3,
1343                                    LONG_DELAY_MS, MILLISECONDS,
1344                                    new ArrayBlockingQueue<Runnable>(10));
1345         try (PoolCleaner cleaner = cleaner(p)) {
1346             try {
1347                 p.setMaximumPoolSize(-1);
1348                 shouldThrow();
1349             } catch (IllegalArgumentException success) {}
1350         }
1351     }
1352 
1353     /**
1354      * Configuration changes that allow core pool size greater than
1355      * max pool size result in IllegalArgumentException.
1356      */
1357     public void testPoolSizeInvariants() {
1358         final ThreadPoolExecutor p =
1359             new ThreadPoolExecutor(1, 1,
1360                                    LONG_DELAY_MS, MILLISECONDS,
1361                                    new ArrayBlockingQueue<Runnable>(10));
1362         try (PoolCleaner cleaner = cleaner(p)) {
1363             for (int s = 1; s < 5; s++) {
1364                 p.setMaximumPoolSize(s);
1365                 p.setCorePoolSize(s);
1366                 try {
1367                     p.setMaximumPoolSize(s - 1);
1368                     shouldThrow();
1369                 } catch (IllegalArgumentException success) {}
1370                 assertEquals(s, p.getCorePoolSize());
1371                 assertEquals(s, p.getMaximumPoolSize());
1372                 try {
1373                     p.setCorePoolSize(s + 1);
1374                     // Android-changed: changeset dfec9b5386ca028cc1468f3e2717120ab6274702
1375                     // disables this check for compatibility reason.
1376                     //    shouldThrow();
1377                 } catch (IllegalArgumentException success) {}
1378                 // Android-changed: changeset dfec9b5386ca028cc1468f3e2717120ab6274702
1379                 // disables maximumpoolsize check for compatibility reason.
1380                 // assertEquals(s, p.getCorePoolSize());
1381                 assertEquals(s + 1, p.getCorePoolSize());
1382                 assertEquals(s, p.getMaximumPoolSize());
1383             }
1384         }
1385     }
1386 
1387     /**
1388      * setKeepAliveTime throws IllegalArgumentException
1389      * when given a negative value
1390      */
1391     public void testKeepAliveTimeIllegalArgumentException() {
1392         final ThreadPoolExecutor p =
1393             new ThreadPoolExecutor(2, 3,
1394                                    LONG_DELAY_MS, MILLISECONDS,
1395                                    new ArrayBlockingQueue<Runnable>(10));
1396         try (PoolCleaner cleaner = cleaner(p)) {
1397             try {
1398                 p.setKeepAliveTime(-1, MILLISECONDS);
1399                 shouldThrow();
1400             } catch (IllegalArgumentException success) {}
1401         }
1402     }
1403 
1404     /**
1405      * terminated() is called on termination
1406      */
1407     public void testTerminated() {
1408         ExtendedTPE p = new ExtendedTPE();
1409         try (PoolCleaner cleaner = cleaner(p)) {
1410             try { p.shutdown(); } catch (SecurityException ok) { return; }
1411             assertTrue(p.terminatedCalled());
1412             assertTrue(p.isShutdown());
1413         }
1414     }
1415 
1416     /**
1417      * beforeExecute and afterExecute are called when executing task
1418      */
1419     public void testBeforeAfter() throws InterruptedException {
1420         ExtendedTPE p = new ExtendedTPE();
1421         try (PoolCleaner cleaner = cleaner(p)) {
1422             final CountDownLatch done = new CountDownLatch(1);
1423             p.execute(new CheckedRunnable() {
1424                 public void realRun() {
1425                     done.countDown();
1426                 }});
1427             await(p.afterCalled);
1428             assertEquals(0, done.getCount());
1429             assertTrue(p.afterCalled());
1430             assertTrue(p.beforeCalled());
1431         }
1432     }
1433 
1434     /**
1435      * completed submit of callable returns result
1436      */
1437     public void testSubmitCallable() throws Exception {
1438         final ExecutorService e =
1439             new ThreadPoolExecutor(2, 2,
1440                                    LONG_DELAY_MS, MILLISECONDS,
1441                                    new ArrayBlockingQueue<Runnable>(10));
1442         try (PoolCleaner cleaner = cleaner(e)) {
1443             Future<String> future = e.submit(new StringTask());
1444             String result = future.get();
1445             assertSame(TEST_STRING, result);
1446         }
1447     }
1448 
1449     /**
1450      * completed submit of runnable returns successfully
1451      */
1452     public void testSubmitRunnable() throws Exception {
1453         final ExecutorService e =
1454             new ThreadPoolExecutor(2, 2,
1455                                    LONG_DELAY_MS, MILLISECONDS,
1456                                    new ArrayBlockingQueue<Runnable>(10));
1457         try (PoolCleaner cleaner = cleaner(e)) {
1458             Future<?> future = e.submit(new NoOpRunnable());
1459             future.get();
1460             assertTrue(future.isDone());
1461         }
1462     }
1463 
1464     /**
1465      * completed submit of (runnable, result) returns result
1466      */
1467     public void testSubmitRunnable2() throws Exception {
1468         final ExecutorService e =
1469             new ThreadPoolExecutor(2, 2,
1470                                    LONG_DELAY_MS, MILLISECONDS,
1471                                    new ArrayBlockingQueue<Runnable>(10));
1472         try (PoolCleaner cleaner = cleaner(e)) {
1473             Future<String> future = e.submit(new NoOpRunnable(), TEST_STRING);
1474             String result = future.get();
1475             assertSame(TEST_STRING, result);
1476         }
1477     }
1478 
1479     /**
1480      * invokeAny(null) throws NPE
1481      */
1482     public void testInvokeAny1() throws Exception {
1483         final ExecutorService e =
1484             new ThreadPoolExecutor(2, 2,
1485                                    LONG_DELAY_MS, MILLISECONDS,
1486                                    new ArrayBlockingQueue<Runnable>(10));
1487         try (PoolCleaner cleaner = cleaner(e)) {
1488             try {
1489                 e.invokeAny(null);
1490                 shouldThrow();
1491             } catch (NullPointerException success) {}
1492         }
1493     }
1494 
1495     /**
1496      * invokeAny(empty collection) throws IAE
1497      */
1498     public void testInvokeAny2() throws Exception {
1499         final ExecutorService e =
1500             new ThreadPoolExecutor(2, 2,
1501                                    LONG_DELAY_MS, MILLISECONDS,
1502                                    new ArrayBlockingQueue<Runnable>(10));
1503         try (PoolCleaner cleaner = cleaner(e)) {
1504             try {
1505                 e.invokeAny(new ArrayList<Callable<String>>());
1506                 shouldThrow();
1507             } catch (IllegalArgumentException success) {}
1508         }
1509     }
1510 
1511     /**
1512      * invokeAny(c) throws NPE if c has null elements
1513      */
1514     public void testInvokeAny3() throws Exception {
1515         final CountDownLatch latch = new CountDownLatch(1);
1516         final ExecutorService e =
1517             new ThreadPoolExecutor(2, 2,
1518                                    LONG_DELAY_MS, MILLISECONDS,
1519                                    new ArrayBlockingQueue<Runnable>(10));
1520         try (PoolCleaner cleaner = cleaner(e)) {
1521             List<Callable<String>> l = new ArrayList<Callable<String>>();
1522             l.add(latchAwaitingStringTask(latch));
1523             l.add(null);
1524             try {
1525                 e.invokeAny(l);
1526                 shouldThrow();
1527             } catch (NullPointerException success) {}
1528             latch.countDown();
1529         }
1530     }
1531 
1532     /**
1533      * invokeAny(c) throws ExecutionException if no task completes
1534      */
1535     public void testInvokeAny4() throws Exception {
1536         final ExecutorService e =
1537             new ThreadPoolExecutor(2, 2,
1538                                    LONG_DELAY_MS, MILLISECONDS,
1539                                    new ArrayBlockingQueue<Runnable>(10));
1540         try (PoolCleaner cleaner = cleaner(e)) {
1541             List<Callable<String>> l = new ArrayList<Callable<String>>();
1542             l.add(new NPETask());
1543             try {
1544                 e.invokeAny(l);
1545                 shouldThrow();
1546             } catch (ExecutionException success) {
1547                 assertTrue(success.getCause() instanceof NullPointerException);
1548             }
1549         }
1550     }
1551 
1552     /**
1553      * invokeAny(c) returns result of some task
1554      */
1555     public void testInvokeAny5() throws Exception {
1556         final ExecutorService e =
1557             new ThreadPoolExecutor(2, 2,
1558                                    LONG_DELAY_MS, MILLISECONDS,
1559                                    new ArrayBlockingQueue<Runnable>(10));
1560         try (PoolCleaner cleaner = cleaner(e)) {
1561             List<Callable<String>> l = new ArrayList<Callable<String>>();
1562             l.add(new StringTask());
1563             l.add(new StringTask());
1564             String result = e.invokeAny(l);
1565             assertSame(TEST_STRING, result);
1566         }
1567     }
1568 
1569     /**
1570      * invokeAll(null) throws NPE
1571      */
1572     public void testInvokeAll1() throws Exception {
1573         final ExecutorService e =
1574             new ThreadPoolExecutor(2, 2,
1575                                    LONG_DELAY_MS, MILLISECONDS,
1576                                    new ArrayBlockingQueue<Runnable>(10));
1577         try (PoolCleaner cleaner = cleaner(e)) {
1578             try {
1579                 e.invokeAll(null);
1580                 shouldThrow();
1581             } catch (NullPointerException success) {}
1582         }
1583     }
1584 
1585     /**
1586      * invokeAll(empty collection) returns empty collection
1587      */
1588     public void testInvokeAll2() throws InterruptedException {
1589         final ExecutorService e =
1590             new ThreadPoolExecutor(2, 2,
1591                                    LONG_DELAY_MS, MILLISECONDS,
1592                                    new ArrayBlockingQueue<Runnable>(10));
1593         try (PoolCleaner cleaner = cleaner(e)) {
1594             List<Future<String>> r = e.invokeAll(new ArrayList<Callable<String>>());
1595             assertTrue(r.isEmpty());
1596         }
1597     }
1598 
1599     /**
1600      * invokeAll(c) throws NPE if c has null elements
1601      */
1602     public void testInvokeAll3() throws Exception {
1603         final ExecutorService e =
1604             new ThreadPoolExecutor(2, 2,
1605                                    LONG_DELAY_MS, MILLISECONDS,
1606                                    new ArrayBlockingQueue<Runnable>(10));
1607         try (PoolCleaner cleaner = cleaner(e)) {
1608             List<Callable<String>> l = new ArrayList<Callable<String>>();
1609             l.add(new StringTask());
1610             l.add(null);
1611             try {
1612                 e.invokeAll(l);
1613                 shouldThrow();
1614             } catch (NullPointerException success) {}
1615         }
1616     }
1617 
1618     /**
1619      * get of element of invokeAll(c) throws exception on failed task
1620      */
1621     public void testInvokeAll4() throws Exception {
1622         final ExecutorService e =
1623             new ThreadPoolExecutor(2, 2,
1624                                    LONG_DELAY_MS, MILLISECONDS,
1625                                    new ArrayBlockingQueue<Runnable>(10));
1626         try (PoolCleaner cleaner = cleaner(e)) {
1627             List<Callable<String>> l = new ArrayList<Callable<String>>();
1628             l.add(new NPETask());
1629             List<Future<String>> futures = e.invokeAll(l);
1630             assertEquals(1, futures.size());
1631             try {
1632                 futures.get(0).get();
1633                 shouldThrow();
1634             } catch (ExecutionException success) {
1635                 assertTrue(success.getCause() instanceof NullPointerException);
1636             }
1637         }
1638     }
1639 
1640     /**
1641      * invokeAll(c) returns results of all completed tasks
1642      */
1643     public void testInvokeAll5() throws Exception {
1644         final ExecutorService e =
1645             new ThreadPoolExecutor(2, 2,
1646                                    LONG_DELAY_MS, MILLISECONDS,
1647                                    new ArrayBlockingQueue<Runnable>(10));
1648         try (PoolCleaner cleaner = cleaner(e)) {
1649             List<Callable<String>> l = new ArrayList<Callable<String>>();
1650             l.add(new StringTask());
1651             l.add(new StringTask());
1652             List<Future<String>> futures = e.invokeAll(l);
1653             assertEquals(2, futures.size());
1654             for (Future<String> future : futures)
1655                 assertSame(TEST_STRING, future.get());
1656         }
1657     }
1658 
1659     /**
1660      * timed invokeAny(null) throws NPE
1661      */
1662     public void testTimedInvokeAny1() throws Exception {
1663         final ExecutorService e =
1664             new ThreadPoolExecutor(2, 2,
1665                                    LONG_DELAY_MS, MILLISECONDS,
1666                                    new ArrayBlockingQueue<Runnable>(10));
1667         try (PoolCleaner cleaner = cleaner(e)) {
1668             try {
1669                 e.invokeAny(null, MEDIUM_DELAY_MS, MILLISECONDS);
1670                 shouldThrow();
1671             } catch (NullPointerException success) {}
1672         }
1673     }
1674 
1675     /**
1676      * timed invokeAny(,,null) throws NPE
1677      */
1678     public void testTimedInvokeAnyNullTimeUnit() throws Exception {
1679         final ExecutorService e =
1680             new ThreadPoolExecutor(2, 2,
1681                                    LONG_DELAY_MS, MILLISECONDS,
1682                                    new ArrayBlockingQueue<Runnable>(10));
1683         try (PoolCleaner cleaner = cleaner(e)) {
1684             List<Callable<String>> l = new ArrayList<Callable<String>>();
1685             l.add(new StringTask());
1686             try {
1687                 e.invokeAny(l, MEDIUM_DELAY_MS, null);
1688                 shouldThrow();
1689             } catch (NullPointerException success) {}
1690         }
1691     }
1692 
1693     /**
1694      * timed invokeAny(empty collection) throws IAE
1695      */
1696     public void testTimedInvokeAny2() throws Exception {
1697         final ExecutorService e =
1698             new ThreadPoolExecutor(2, 2,
1699                                    LONG_DELAY_MS, MILLISECONDS,
1700                                    new ArrayBlockingQueue<Runnable>(10));
1701         try (PoolCleaner cleaner = cleaner(e)) {
1702             try {
1703                 e.invokeAny(new ArrayList<Callable<String>>(),
1704                             MEDIUM_DELAY_MS, MILLISECONDS);
1705                 shouldThrow();
1706             } catch (IllegalArgumentException success) {}
1707         }
1708     }
1709 
1710     /**
1711      * timed invokeAny(c) throws NPE if c has null elements
1712      */
1713     public void testTimedInvokeAny3() throws Exception {
1714         final CountDownLatch latch = new CountDownLatch(1);
1715         final ExecutorService e =
1716             new ThreadPoolExecutor(2, 2,
1717                                    LONG_DELAY_MS, MILLISECONDS,
1718                                    new ArrayBlockingQueue<Runnable>(10));
1719         try (PoolCleaner cleaner = cleaner(e)) {
1720             List<Callable<String>> l = new ArrayList<Callable<String>>();
1721             l.add(latchAwaitingStringTask(latch));
1722             l.add(null);
1723             try {
1724                 e.invokeAny(l, MEDIUM_DELAY_MS, MILLISECONDS);
1725                 shouldThrow();
1726             } catch (NullPointerException success) {}
1727             latch.countDown();
1728         }
1729     }
1730 
1731     /**
1732      * timed invokeAny(c) throws ExecutionException if no task completes
1733      */
1734     public void testTimedInvokeAny4() throws Exception {
1735         final ExecutorService e =
1736             new ThreadPoolExecutor(2, 2,
1737                                    LONG_DELAY_MS, MILLISECONDS,
1738                                    new ArrayBlockingQueue<Runnable>(10));
1739         try (PoolCleaner cleaner = cleaner(e)) {
1740             long startTime = System.nanoTime();
1741             List<Callable<String>> l = new ArrayList<Callable<String>>();
1742             l.add(new NPETask());
1743             try {
1744                 e.invokeAny(l, LONG_DELAY_MS, MILLISECONDS);
1745                 shouldThrow();
1746             } catch (ExecutionException success) {
1747                 assertTrue(success.getCause() instanceof NullPointerException);
1748             }
1749             assertTrue(millisElapsedSince(startTime) < LONG_DELAY_MS);
1750         }
1751     }
1752 
1753     /**
1754      * timed invokeAny(c) returns result of some task
1755      */
1756     public void testTimedInvokeAny5() throws Exception {
1757         final ExecutorService e =
1758             new ThreadPoolExecutor(2, 2,
1759                                    LONG_DELAY_MS, MILLISECONDS,
1760                                    new ArrayBlockingQueue<Runnable>(10));
1761         try (PoolCleaner cleaner = cleaner(e)) {
1762             long startTime = System.nanoTime();
1763             List<Callable<String>> l = new ArrayList<Callable<String>>();
1764             l.add(new StringTask());
1765             l.add(new StringTask());
1766             String result = e.invokeAny(l, LONG_DELAY_MS, MILLISECONDS);
1767             assertSame(TEST_STRING, result);
1768             assertTrue(millisElapsedSince(startTime) < LONG_DELAY_MS);
1769         }
1770     }
1771 
1772     /**
1773      * timed invokeAll(null) throws NPE
1774      */
1775     public void testTimedInvokeAll1() throws Exception {
1776         final ExecutorService e =
1777             new ThreadPoolExecutor(2, 2,
1778                                    LONG_DELAY_MS, MILLISECONDS,
1779                                    new ArrayBlockingQueue<Runnable>(10));
1780         try (PoolCleaner cleaner = cleaner(e)) {
1781             try {
1782                 e.invokeAll(null, MEDIUM_DELAY_MS, MILLISECONDS);
1783                 shouldThrow();
1784             } catch (NullPointerException success) {}
1785         }
1786     }
1787 
1788     /**
1789      * timed invokeAll(,,null) throws NPE
1790      */
1791     public void testTimedInvokeAllNullTimeUnit() throws Exception {
1792         final ExecutorService e =
1793             new ThreadPoolExecutor(2, 2,
1794                                    LONG_DELAY_MS, MILLISECONDS,
1795                                    new ArrayBlockingQueue<Runnable>(10));
1796         try (PoolCleaner cleaner = cleaner(e)) {
1797             List<Callable<String>> l = new ArrayList<Callable<String>>();
1798             l.add(new StringTask());
1799             try {
1800                 e.invokeAll(l, MEDIUM_DELAY_MS, null);
1801                 shouldThrow();
1802             } catch (NullPointerException success) {}
1803         }
1804     }
1805 
1806     /**
1807      * timed invokeAll(empty collection) returns empty collection
1808      */
1809     public void testTimedInvokeAll2() throws InterruptedException {
1810         final ExecutorService e =
1811             new ThreadPoolExecutor(2, 2,
1812                                    LONG_DELAY_MS, MILLISECONDS,
1813                                    new ArrayBlockingQueue<Runnable>(10));
1814         try (PoolCleaner cleaner = cleaner(e)) {
1815             List<Future<String>> r = e.invokeAll(new ArrayList<Callable<String>>(),
1816                                                  MEDIUM_DELAY_MS, MILLISECONDS);
1817             assertTrue(r.isEmpty());
1818         }
1819     }
1820 
1821     /**
1822      * timed invokeAll(c) throws NPE if c has null elements
1823      */
1824     public void testTimedInvokeAll3() throws Exception {
1825         final ExecutorService e =
1826             new ThreadPoolExecutor(2, 2,
1827                                    LONG_DELAY_MS, MILLISECONDS,
1828                                    new ArrayBlockingQueue<Runnable>(10));
1829         try (PoolCleaner cleaner = cleaner(e)) {
1830             List<Callable<String>> l = new ArrayList<Callable<String>>();
1831             l.add(new StringTask());
1832             l.add(null);
1833             try {
1834                 e.invokeAll(l, MEDIUM_DELAY_MS, MILLISECONDS);
1835                 shouldThrow();
1836             } catch (NullPointerException success) {}
1837         }
1838     }
1839 
1840     /**
1841      * get of element of invokeAll(c) throws exception on failed task
1842      */
1843     public void testTimedInvokeAll4() throws Exception {
1844         final ExecutorService e =
1845             new ThreadPoolExecutor(2, 2,
1846                                    LONG_DELAY_MS, MILLISECONDS,
1847                                    new ArrayBlockingQueue<Runnable>(10));
1848         try (PoolCleaner cleaner = cleaner(e)) {
1849             List<Callable<String>> l = new ArrayList<Callable<String>>();
1850             l.add(new NPETask());
1851             List<Future<String>> futures =
1852                 e.invokeAll(l, LONG_DELAY_MS, MILLISECONDS);
1853             assertEquals(1, futures.size());
1854             try {
1855                 futures.get(0).get();
1856                 shouldThrow();
1857             } catch (ExecutionException success) {
1858                 assertTrue(success.getCause() instanceof NullPointerException);
1859             }
1860         }
1861     }
1862 
1863     /**
1864      * timed invokeAll(c) returns results of all completed tasks
1865      */
1866     public void testTimedInvokeAll5() throws Exception {
1867         final ExecutorService e =
1868             new ThreadPoolExecutor(2, 2,
1869                                    LONG_DELAY_MS, MILLISECONDS,
1870                                    new ArrayBlockingQueue<Runnable>(10));
1871         try (PoolCleaner cleaner = cleaner(e)) {
1872             List<Callable<String>> l = new ArrayList<Callable<String>>();
1873             l.add(new StringTask());
1874             l.add(new StringTask());
1875             List<Future<String>> futures =
1876                 e.invokeAll(l, LONG_DELAY_MS, MILLISECONDS);
1877             assertEquals(2, futures.size());
1878             for (Future<String> future : futures)
1879                 assertSame(TEST_STRING, future.get());
1880         }
1881     }
1882 
1883     /**
1884      * timed invokeAll(c) cancels tasks not completed by timeout
1885      */
1886     public void testTimedInvokeAll6() throws Exception {
1887         for (long timeout = timeoutMillis();;) {
1888             final CountDownLatch done = new CountDownLatch(1);
1889             final Callable<String> waiter = new CheckedCallable<String>() {
1890                 public String realCall() {
1891                     try { done.await(LONG_DELAY_MS, MILLISECONDS); }
1892                     catch (InterruptedException ok) {}
1893                     return "1"; }};
1894             final ExecutorService p =
1895                 new ThreadPoolExecutor(2, 2,
1896                                        LONG_DELAY_MS, MILLISECONDS,
1897                                        new ArrayBlockingQueue<Runnable>(10));
1898             try (PoolCleaner cleaner = cleaner(p, done)) {
1899                 List<Callable<String>> tasks = new ArrayList<>();
1900                 tasks.add(new StringTask("0"));
1901                 tasks.add(waiter);
1902                 tasks.add(new StringTask("2"));
1903                 long startTime = System.nanoTime();
1904                 List<Future<String>> futures =
1905                     p.invokeAll(tasks, timeout, MILLISECONDS);
1906                 assertEquals(tasks.size(), futures.size());
1907                 assertTrue(millisElapsedSince(startTime) >= timeout);
1908                 for (Future future : futures)
1909                     assertTrue(future.isDone());
1910                 assertTrue(futures.get(1).isCancelled());
1911                 try {
1912                     assertEquals("0", futures.get(0).get());
1913                     assertEquals("2", futures.get(2).get());
1914                     break;
1915                 } catch (CancellationException retryWithLongerTimeout) {
1916                     timeout *= 2;
1917                     if (timeout >= LONG_DELAY_MS / 2)
1918                         fail("expected exactly one task to be cancelled");
1919                 }
1920             }
1921         }
1922     }
1923 
1924     /**
1925      * Execution continues if there is at least one thread even if
1926      * thread factory fails to create more
1927      */
1928     public void testFailingThreadFactory() throws InterruptedException {
1929         final ExecutorService e =
1930             new ThreadPoolExecutor(100, 100,
1931                                    LONG_DELAY_MS, MILLISECONDS,
1932                                    new LinkedBlockingQueue<Runnable>(),
1933                                    new FailingThreadFactory());
1934         try (PoolCleaner cleaner = cleaner(e)) {
1935             final int TASKS = 100;
1936             final CountDownLatch done = new CountDownLatch(TASKS);
1937             for (int k = 0; k < TASKS; ++k)
1938                 e.execute(new CheckedRunnable() {
1939                     public void realRun() {
1940                         done.countDown();
1941                     }});
1942             assertTrue(done.await(LONG_DELAY_MS, MILLISECONDS));
1943         }
1944     }
1945 
1946     /**
1947      * allowsCoreThreadTimeOut is by default false.
1948      */
1949     public void testAllowsCoreThreadTimeOut() {
1950         final ThreadPoolExecutor p =
1951             new ThreadPoolExecutor(2, 2,
1952                                    1000, MILLISECONDS,
1953                                    new ArrayBlockingQueue<Runnable>(10));
1954         try (PoolCleaner cleaner = cleaner(p)) {
1955             assertFalse(p.allowsCoreThreadTimeOut());
1956         }
1957     }
1958 
1959     /**
1960      * allowCoreThreadTimeOut(true) causes idle threads to time out
1961      */
1962     public void testAllowCoreThreadTimeOut_true() throws Exception {
1963         long keepAliveTime = timeoutMillis();
1964         final ThreadPoolExecutor p =
1965             new ThreadPoolExecutor(2, 10,
1966                                    keepAliveTime, MILLISECONDS,
1967                                    new ArrayBlockingQueue<Runnable>(10));
1968         try (PoolCleaner cleaner = cleaner(p)) {
1969             final CountDownLatch threadStarted = new CountDownLatch(1);
1970             p.allowCoreThreadTimeOut(true);
1971             p.execute(new CheckedRunnable() {
1972                 public void realRun() {
1973                     threadStarted.countDown();
1974                     assertEquals(1, p.getPoolSize());
1975                 }});
1976             await(threadStarted);
1977             delay(keepAliveTime);
1978             long startTime = System.nanoTime();
1979             while (p.getPoolSize() > 0
1980                    && millisElapsedSince(startTime) < LONG_DELAY_MS)
1981                 Thread.yield();
1982             assertTrue(millisElapsedSince(startTime) < LONG_DELAY_MS);
1983             assertEquals(0, p.getPoolSize());
1984         }
1985     }
1986 
1987     /**
1988      * allowCoreThreadTimeOut(false) causes idle threads not to time out
1989      */
1990     public void testAllowCoreThreadTimeOut_false() throws Exception {
1991         long keepAliveTime = timeoutMillis();
1992         final ThreadPoolExecutor p =
1993             new ThreadPoolExecutor(2, 10,
1994                                    keepAliveTime, MILLISECONDS,
1995                                    new ArrayBlockingQueue<Runnable>(10));
1996         try (PoolCleaner cleaner = cleaner(p)) {
1997             final CountDownLatch threadStarted = new CountDownLatch(1);
1998             p.allowCoreThreadTimeOut(false);
1999             p.execute(new CheckedRunnable() {
2000                 public void realRun() throws InterruptedException {
2001                     threadStarted.countDown();
2002                     assertTrue(p.getPoolSize() >= 1);
2003                 }});
2004             delay(2 * keepAliveTime);
2005             assertTrue(p.getPoolSize() >= 1);
2006         }
2007     }
2008 
2009     /**
2010      * execute allows the same task to be submitted multiple times, even
2011      * if rejected
2012      */
2013     public void testRejectedRecycledTask() throws InterruptedException {
2014         final int nTasks = 1000;
2015         final CountDownLatch done = new CountDownLatch(nTasks);
2016         final Runnable recycledTask = new Runnable() {
2017             public void run() {
2018                 done.countDown();
2019             }};
2020         final ThreadPoolExecutor p =
2021             new ThreadPoolExecutor(1, 30,
2022                                    60, SECONDS,
2023                                    new ArrayBlockingQueue(30));
2024         try (PoolCleaner cleaner = cleaner(p)) {
2025             for (int i = 0; i < nTasks; ++i) {
2026                 for (;;) {
2027                     try {
2028                         p.execute(recycledTask);
2029                         break;
2030                     }
2031                     catch (RejectedExecutionException ignore) {}
2032                 }
2033             }
2034             // enough time to run all tasks
2035             assertTrue(done.await(nTasks * SHORT_DELAY_MS, MILLISECONDS));
2036         }
2037     }
2038 
2039     /**
2040      * get(cancelled task) throws CancellationException
2041      */
2042     public void testGet_cancelled() throws Exception {
2043         final CountDownLatch done = new CountDownLatch(1);
2044         final ExecutorService e =
2045             new ThreadPoolExecutor(1, 1,
2046                                    LONG_DELAY_MS, MILLISECONDS,
2047                                    new LinkedBlockingQueue<Runnable>());
2048         try (PoolCleaner cleaner = cleaner(e, done)) {
2049             final CountDownLatch blockerStarted = new CountDownLatch(1);
2050             final List<Future<?>> futures = new ArrayList<>();
2051             for (int i = 0; i < 2; i++) {
2052                 Runnable r = new CheckedRunnable() { public void realRun()
2053                                                          throws Throwable {
2054                     blockerStarted.countDown();
2055                     assertTrue(done.await(2 * LONG_DELAY_MS, MILLISECONDS));
2056                 }};
2057                 futures.add(e.submit(r));
2058             }
2059             await(blockerStarted);
2060             for (Future<?> future : futures) future.cancel(false);
2061             for (Future<?> future : futures) {
2062                 try {
2063                     future.get();
2064                     shouldThrow();
2065                 } catch (CancellationException success) {}
2066                 try {
2067                     future.get(LONG_DELAY_MS, MILLISECONDS);
2068                     shouldThrow();
2069                 } catch (CancellationException success) {}
2070                 assertTrue(future.isCancelled());
2071                 assertTrue(future.isDone());
2072             }
2073         }
2074     }
2075 
2076 }
2077