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 
13 import java.util.Collection;
14 import java.util.concurrent.CountDownLatch;
15 import java.util.concurrent.Semaphore;
16 
17 import junit.framework.AssertionFailedError;
18 import junit.framework.Test;
19 import junit.framework.TestSuite;
20 
21 public class SemaphoreTest extends JSR166TestCase {
22     // android-note: Removed because the CTS runner does a bad job of
23     // retrying tests that have suite() declarations.
24     //
25     // public static void main(String[] args) {
26     //     main(suite(), args);
27     // }
28     // public static Test suite() {
29     //     return new TestSuite(SemaphoreTest.class);
30     // }
31 
32     /**
33      * Subclass to expose protected methods
34      */
35     static class PublicSemaphore extends Semaphore {
36         PublicSemaphore(int permits) { super(permits); }
37         PublicSemaphore(int permits, boolean fair) { super(permits, fair); }
38         public Collection<Thread> getQueuedThreads() {
39             return super.getQueuedThreads();
40         }
41         public boolean hasQueuedThread(Thread t) {
42             return super.getQueuedThreads().contains(t);
43         }
44         public void reducePermits(int reduction) {
45             super.reducePermits(reduction);
46         }
47     }
48 
49     /**
50      * A runnable calling acquire
51      */
52     class InterruptibleLockRunnable extends CheckedRunnable {
53         final Semaphore lock;
54         InterruptibleLockRunnable(Semaphore s) { lock = s; }
55         public void realRun() {
56             try {
57                 lock.acquire();
58             }
59             catch (InterruptedException ignored) {}
60         }
61     }
62 
63     /**
64      * A runnable calling acquire that expects to be interrupted
65      */
66     class InterruptedLockRunnable extends CheckedInterruptedRunnable {
67         final Semaphore lock;
68         InterruptedLockRunnable(Semaphore s) { lock = s; }
69         public void realRun() throws InterruptedException {
70             lock.acquire();
71         }
72     }
73 
74     /**
75      * Spin-waits until s.hasQueuedThread(t) becomes true.
76      */
77     void waitForQueuedThread(PublicSemaphore s, Thread t) {
78         long startTime = System.nanoTime();
79         while (!s.hasQueuedThread(t)) {
80             if (millisElapsedSince(startTime) > LONG_DELAY_MS)
81                 throw new AssertionFailedError("timed out");
82             Thread.yield();
83         }
84         assertTrue(s.hasQueuedThreads());
85         assertTrue(t.isAlive());
86     }
87 
88     /**
89      * Spin-waits until s.hasQueuedThreads() becomes true.
90      */
91     void waitForQueuedThreads(Semaphore s) {
92         long startTime = System.nanoTime();
93         while (!s.hasQueuedThreads()) {
94             if (millisElapsedSince(startTime) > LONG_DELAY_MS)
95                 throw new AssertionFailedError("timed out");
96             Thread.yield();
97         }
98     }
99 
100     enum AcquireMethod {
101         acquire() {
102             void acquire(Semaphore s) throws InterruptedException {
103                 s.acquire();
104             }
105         },
106         acquireN() {
107             void acquire(Semaphore s, int permits) throws InterruptedException {
108                 s.acquire(permits);
109             }
110         },
111         acquireUninterruptibly() {
112             void acquire(Semaphore s) {
113                 s.acquireUninterruptibly();
114             }
115         },
116         acquireUninterruptiblyN() {
117             void acquire(Semaphore s, int permits) {
118                 s.acquireUninterruptibly(permits);
119             }
120         },
121         tryAcquire() {
122             void acquire(Semaphore s) {
123                 assertTrue(s.tryAcquire());
124             }
125         },
126         tryAcquireN() {
127             void acquire(Semaphore s, int permits) {
128                 assertTrue(s.tryAcquire(permits));
129             }
130         },
131         tryAcquireTimed() {
132             void acquire(Semaphore s) throws InterruptedException {
133                 assertTrue(s.tryAcquire(2 * LONG_DELAY_MS, MILLISECONDS));
134             }
135         },
136         tryAcquireTimedN {
137             void acquire(Semaphore s, int permits) throws InterruptedException {
138                 assertTrue(s.tryAcquire(permits, 2 * LONG_DELAY_MS, MILLISECONDS));
139             }
140         };
141 
142         // Intentionally meta-circular
143 
144         /** Acquires 1 permit. */
145         void acquire(Semaphore s) throws InterruptedException {
146             acquire(s, 1);
147         }
148         /** Acquires the given number of permits. */
149         void acquire(Semaphore s, int permits) throws InterruptedException {
150             for (int i = 0; i < permits; i++)
151                 acquire(s);
152         }
153     }
154 
155     /**
156      * Zero, negative, and positive initial values are allowed in constructor
157      */
158     public void testConstructor()      { testConstructor(false); }
159     public void testConstructor_fair() { testConstructor(true); }
160     public void testConstructor(boolean fair) {
161         for (int permits : new int[] { -42, -1, 0, 1, 42 }) {
162             Semaphore s = new Semaphore(permits, fair);
163             assertEquals(permits, s.availablePermits());
164             assertEquals(fair, s.isFair());
165         }
166     }
167 
168     /**
169      * Constructor without fairness argument behaves as nonfair
170      */
171     public void testConstructorDefaultsToNonFair() {
172         for (int permits : new int[] { -42, -1, 0, 1, 42 }) {
173             Semaphore s = new Semaphore(permits);
174             assertEquals(permits, s.availablePermits());
175             assertFalse(s.isFair());
176         }
177     }
178 
179     /**
180      * tryAcquire succeeds when sufficient permits, else fails
181      */
182     public void testTryAcquireInSameThread()      { testTryAcquireInSameThread(false); }
183     public void testTryAcquireInSameThread_fair() { testTryAcquireInSameThread(true); }
184     public void testTryAcquireInSameThread(boolean fair) {
185         Semaphore s = new Semaphore(2, fair);
186         assertEquals(2, s.availablePermits());
187         assertTrue(s.tryAcquire());
188         assertTrue(s.tryAcquire());
189         assertEquals(0, s.availablePermits());
190         assertFalse(s.tryAcquire());
191         assertFalse(s.tryAcquire());
192         assertEquals(0, s.availablePermits());
193     }
194 
195     /**
196      * timed tryAcquire times out
197      */
198     public void testTryAcquire_timeout()      { testTryAcquire_timeout(false); }
199     public void testTryAcquire_timeout_fair() { testTryAcquire_timeout(true); }
200     public void testTryAcquire_timeout(boolean fair) {
201         Semaphore s = new Semaphore(0, fair);
202         long startTime = System.nanoTime();
203         try { assertFalse(s.tryAcquire(timeoutMillis(), MILLISECONDS)); }
204         catch (InterruptedException e) { threadUnexpectedException(e); }
205         assertTrue(millisElapsedSince(startTime) >= timeoutMillis());
206     }
207 
208     /**
209      * timed tryAcquire(N) times out
210      */
211     public void testTryAcquireN_timeout()      { testTryAcquireN_timeout(false); }
212     public void testTryAcquireN_timeout_fair() { testTryAcquireN_timeout(true); }
213     public void testTryAcquireN_timeout(boolean fair) {
214         Semaphore s = new Semaphore(2, fair);
215         long startTime = System.nanoTime();
216         try { assertFalse(s.tryAcquire(3, timeoutMillis(), MILLISECONDS)); }
217         catch (InterruptedException e) { threadUnexpectedException(e); }
218         assertTrue(millisElapsedSince(startTime) >= timeoutMillis());
219     }
220 
221     /**
222      * acquire(), acquire(N), timed tryAcquired, timed tryAcquire(N)
223      * are interruptible
224      */
225     public void testInterruptible_acquire()               { testInterruptible(false, AcquireMethod.acquire); }
226     public void testInterruptible_acquire_fair()          { testInterruptible(true,  AcquireMethod.acquire); }
227     public void testInterruptible_acquireN()              { testInterruptible(false, AcquireMethod.acquireN); }
228     public void testInterruptible_acquireN_fair()         { testInterruptible(true,  AcquireMethod.acquireN); }
229     public void testInterruptible_tryAcquireTimed()       { testInterruptible(false, AcquireMethod.tryAcquireTimed); }
230     public void testInterruptible_tryAcquireTimed_fair()  { testInterruptible(true,  AcquireMethod.tryAcquireTimed); }
231     public void testInterruptible_tryAcquireTimedN()      { testInterruptible(false, AcquireMethod.tryAcquireTimedN); }
232     public void testInterruptible_tryAcquireTimedN_fair() { testInterruptible(true,  AcquireMethod.tryAcquireTimedN); }
233     public void testInterruptible(boolean fair, final AcquireMethod acquirer) {
234         final PublicSemaphore s = new PublicSemaphore(0, fair);
235         final Semaphore pleaseInterrupt = new Semaphore(0, fair);
236         Thread t = newStartedThread(new CheckedRunnable() {
237             public void realRun() {
238                 // Interrupt before acquire
239                 Thread.currentThread().interrupt();
240                 try {
241                     acquirer.acquire(s);
242                     shouldThrow();
243                 } catch (InterruptedException success) {}
244 
245                 // Interrupt during acquire
246                 try {
247                     acquirer.acquire(s);
248                     shouldThrow();
249                 } catch (InterruptedException success) {}
250 
251                 // Interrupt before acquire(N)
252                 Thread.currentThread().interrupt();
253                 try {
254                     acquirer.acquire(s, 3);
255                     shouldThrow();
256                 } catch (InterruptedException success) {}
257 
258                 pleaseInterrupt.release();
259 
260                 // Interrupt during acquire(N)
261                 try {
262                     acquirer.acquire(s, 3);
263                     shouldThrow();
264                 } catch (InterruptedException success) {}
265             }});
266 
267         waitForQueuedThread(s, t);
268         t.interrupt();
269         await(pleaseInterrupt);
270         waitForQueuedThread(s, t);
271         t.interrupt();
272         awaitTermination(t);
273     }
274 
275     /**
276      * acquireUninterruptibly(), acquireUninterruptibly(N) are
277      * uninterruptible
278      */
279     public void testUninterruptible_acquireUninterruptibly()       { testUninterruptible(false, AcquireMethod.acquireUninterruptibly); }
280     public void testUninterruptible_acquireUninterruptibly_fair()  { testUninterruptible(true,  AcquireMethod.acquireUninterruptibly); }
281     public void testUninterruptible_acquireUninterruptiblyN()      { testUninterruptible(false, AcquireMethod.acquireUninterruptiblyN); }
282     public void testUninterruptible_acquireUninterruptiblyN_fair() { testUninterruptible(true,  AcquireMethod.acquireUninterruptiblyN); }
283     public void testUninterruptible(boolean fair, final AcquireMethod acquirer) {
284         final PublicSemaphore s = new PublicSemaphore(0, fair);
285         final Semaphore pleaseInterrupt = new Semaphore(-1, fair);
286 
287         Thread t1 = newStartedThread(new CheckedRunnable() {
288             public void realRun() throws InterruptedException {
289                 // Interrupt before acquire
290                 pleaseInterrupt.release();
291                 Thread.currentThread().interrupt();
292                 acquirer.acquire(s);
293                 assertTrue(Thread.interrupted());
294             }});
295 
296         Thread t2 = newStartedThread(new CheckedRunnable() {
297             public void realRun() throws InterruptedException {
298                 // Interrupt during acquire
299                 pleaseInterrupt.release();
300                 acquirer.acquire(s);
301                 assertTrue(Thread.interrupted());
302             }});
303 
304         await(pleaseInterrupt);
305         waitForQueuedThread(s, t1);
306         waitForQueuedThread(s, t2);
307         t2.interrupt();
308 
309         assertThreadStaysAlive(t1);
310         assertTrue(t2.isAlive());
311 
312         s.release(2);
313 
314         awaitTermination(t1);
315         awaitTermination(t2);
316     }
317 
318     /**
319      * hasQueuedThreads reports whether there are waiting threads
320      */
321     public void testHasQueuedThreads()      { testHasQueuedThreads(false); }
322     public void testHasQueuedThreads_fair() { testHasQueuedThreads(true); }
323     public void testHasQueuedThreads(boolean fair) {
324         final PublicSemaphore lock = new PublicSemaphore(1, fair);
325         assertFalse(lock.hasQueuedThreads());
326         lock.acquireUninterruptibly();
327         Thread t1 = newStartedThread(new InterruptedLockRunnable(lock));
328         waitForQueuedThread(lock, t1);
329         assertTrue(lock.hasQueuedThreads());
330         Thread t2 = newStartedThread(new InterruptibleLockRunnable(lock));
331         waitForQueuedThread(lock, t2);
332         assertTrue(lock.hasQueuedThreads());
333         t1.interrupt();
334         awaitTermination(t1);
335         assertTrue(lock.hasQueuedThreads());
336         lock.release();
337         awaitTermination(t2);
338         assertFalse(lock.hasQueuedThreads());
339     }
340 
341     /**
342      * getQueueLength reports number of waiting threads
343      */
344     public void testGetQueueLength()      { testGetQueueLength(false); }
345     public void testGetQueueLength_fair() { testGetQueueLength(true); }
346     public void testGetQueueLength(boolean fair) {
347         final PublicSemaphore lock = new PublicSemaphore(1, fair);
348         assertEquals(0, lock.getQueueLength());
349         lock.acquireUninterruptibly();
350         Thread t1 = newStartedThread(new InterruptedLockRunnable(lock));
351         waitForQueuedThread(lock, t1);
352         assertEquals(1, lock.getQueueLength());
353         Thread t2 = newStartedThread(new InterruptibleLockRunnable(lock));
354         waitForQueuedThread(lock, t2);
355         assertEquals(2, lock.getQueueLength());
356         t1.interrupt();
357         awaitTermination(t1);
358         assertEquals(1, lock.getQueueLength());
359         lock.release();
360         awaitTermination(t2);
361         assertEquals(0, lock.getQueueLength());
362     }
363 
364     /**
365      * getQueuedThreads includes waiting threads
366      */
367     public void testGetQueuedThreads()      { testGetQueuedThreads(false); }
368     public void testGetQueuedThreads_fair() { testGetQueuedThreads(true); }
369     public void testGetQueuedThreads(boolean fair) {
370         final PublicSemaphore lock = new PublicSemaphore(1, fair);
371         assertTrue(lock.getQueuedThreads().isEmpty());
372         lock.acquireUninterruptibly();
373         assertTrue(lock.getQueuedThreads().isEmpty());
374         Thread t1 = newStartedThread(new InterruptedLockRunnable(lock));
375         waitForQueuedThread(lock, t1);
376         assertTrue(lock.getQueuedThreads().contains(t1));
377         Thread t2 = newStartedThread(new InterruptibleLockRunnable(lock));
378         waitForQueuedThread(lock, t2);
379         assertTrue(lock.getQueuedThreads().contains(t1));
380         assertTrue(lock.getQueuedThreads().contains(t2));
381         t1.interrupt();
382         awaitTermination(t1);
383         assertFalse(lock.getQueuedThreads().contains(t1));
384         assertTrue(lock.getQueuedThreads().contains(t2));
385         lock.release();
386         awaitTermination(t2);
387         assertTrue(lock.getQueuedThreads().isEmpty());
388     }
389 
390     /**
391      * drainPermits reports and removes given number of permits
392      */
393     public void testDrainPermits()      { testDrainPermits(false); }
394     public void testDrainPermits_fair() { testDrainPermits(true); }
395     public void testDrainPermits(boolean fair) {
396         Semaphore s = new Semaphore(0, fair);
397         assertEquals(0, s.availablePermits());
398         assertEquals(0, s.drainPermits());
399         s.release(10);
400         assertEquals(10, s.availablePermits());
401         assertEquals(10, s.drainPermits());
402         assertEquals(0, s.availablePermits());
403         assertEquals(0, s.drainPermits());
404     }
405 
406     /**
407      * release(-N) throws IllegalArgumentException
408      */
409     public void testReleaseIAE()      { testReleaseIAE(false); }
410     public void testReleaseIAE_fair() { testReleaseIAE(true); }
411     public void testReleaseIAE(boolean fair) {
412         Semaphore s = new Semaphore(10, fair);
413         try {
414             s.release(-1);
415             shouldThrow();
416         } catch (IllegalArgumentException success) {}
417     }
418 
419     /**
420      * reducePermits(-N) throws IllegalArgumentException
421      */
422     public void testReducePermitsIAE()      { testReducePermitsIAE(false); }
423     public void testReducePermitsIAE_fair() { testReducePermitsIAE(true); }
424     public void testReducePermitsIAE(boolean fair) {
425         PublicSemaphore s = new PublicSemaphore(10, fair);
426         try {
427             s.reducePermits(-1);
428             shouldThrow();
429         } catch (IllegalArgumentException success) {}
430     }
431 
432     /**
433      * reducePermits reduces number of permits
434      */
435     public void testReducePermits()      { testReducePermits(false); }
436     public void testReducePermits_fair() { testReducePermits(true); }
437     public void testReducePermits(boolean fair) {
438         PublicSemaphore s = new PublicSemaphore(10, fair);
439         assertEquals(10, s.availablePermits());
440         s.reducePermits(0);
441         assertEquals(10, s.availablePermits());
442         s.reducePermits(1);
443         assertEquals(9, s.availablePermits());
444         s.reducePermits(10);
445         assertEquals(-1, s.availablePermits());
446         s.reducePermits(10);
447         assertEquals(-11, s.availablePermits());
448         s.reducePermits(0);
449         assertEquals(-11, s.availablePermits());
450     }
451 
452     /**
453      * a reserialized semaphore has same number of permits and
454      * fairness, but no queued threads
455      */
456     public void testSerialization()      { testSerialization(false); }
457     public void testSerialization_fair() { testSerialization(true); }
458     public void testSerialization(boolean fair) {
459         try {
460             Semaphore s = new Semaphore(3, fair);
461             s.acquire();
462             s.acquire();
463             s.release();
464 
465             Semaphore clone = serialClone(s);
466             assertEquals(fair, s.isFair());
467             assertEquals(fair, clone.isFair());
468             assertEquals(2, s.availablePermits());
469             assertEquals(2, clone.availablePermits());
470             clone.acquire();
471             clone.acquire();
472             clone.release();
473             assertEquals(2, s.availablePermits());
474             assertEquals(1, clone.availablePermits());
475             assertFalse(s.hasQueuedThreads());
476             assertFalse(clone.hasQueuedThreads());
477         } catch (InterruptedException e) { threadUnexpectedException(e); }
478 
479         {
480             PublicSemaphore s = new PublicSemaphore(0, fair);
481             Thread t = newStartedThread(new InterruptibleLockRunnable(s));
482             // waitForQueuedThreads(s); // suffers from "flicker", so ...
483             waitForQueuedThread(s, t);  // ... we use this instead
484             PublicSemaphore clone = serialClone(s);
485             assertEquals(fair, s.isFair());
486             assertEquals(fair, clone.isFair());
487             assertEquals(0, s.availablePermits());
488             assertEquals(0, clone.availablePermits());
489             assertTrue(s.hasQueuedThreads());
490             assertFalse(clone.hasQueuedThreads());
491             s.release();
492             awaitTermination(t);
493             assertFalse(s.hasQueuedThreads());
494             assertFalse(clone.hasQueuedThreads());
495         }
496     }
497 
498     /**
499      * tryAcquire(n) succeeds when sufficient permits, else fails
500      */
501     public void testTryAcquireNInSameThread()      { testTryAcquireNInSameThread(false); }
502     public void testTryAcquireNInSameThread_fair() { testTryAcquireNInSameThread(true); }
503     public void testTryAcquireNInSameThread(boolean fair) {
504         Semaphore s = new Semaphore(2, fair);
505         assertEquals(2, s.availablePermits());
506         assertFalse(s.tryAcquire(3));
507         assertEquals(2, s.availablePermits());
508         assertTrue(s.tryAcquire(2));
509         assertEquals(0, s.availablePermits());
510         assertFalse(s.tryAcquire(1));
511         assertFalse(s.tryAcquire(2));
512         assertEquals(0, s.availablePermits());
513     }
514 
515     /**
516      * acquire succeeds if permits available
517      */
518     public void testReleaseAcquireSameThread_acquire()       { testReleaseAcquireSameThread(false, AcquireMethod.acquire); }
519     public void testReleaseAcquireSameThread_acquire_fair()  { testReleaseAcquireSameThread(true, AcquireMethod.acquire); }
520     public void testReleaseAcquireSameThread_acquireN()      { testReleaseAcquireSameThread(false, AcquireMethod.acquireN); }
521     public void testReleaseAcquireSameThread_acquireN_fair() { testReleaseAcquireSameThread(true, AcquireMethod.acquireN); }
522     public void testReleaseAcquireSameThread_acquireUninterruptibly()       { testReleaseAcquireSameThread(false, AcquireMethod.acquireUninterruptibly); }
523     public void testReleaseAcquireSameThread_acquireUninterruptibly_fair()  { testReleaseAcquireSameThread(true, AcquireMethod.acquireUninterruptibly); }
524     public void testReleaseAcquireSameThread_acquireUninterruptiblyN()      { testReleaseAcquireSameThread(false, AcquireMethod.acquireUninterruptibly); }
525     public void testReleaseAcquireSameThread_acquireUninterruptiblyN_fair() { testReleaseAcquireSameThread(true, AcquireMethod.acquireUninterruptibly); }
526     public void testReleaseAcquireSameThread_tryAcquire()       { testReleaseAcquireSameThread(false, AcquireMethod.tryAcquire); }
527     public void testReleaseAcquireSameThread_tryAcquire_fair()  { testReleaseAcquireSameThread(true, AcquireMethod.tryAcquire); }
528     public void testReleaseAcquireSameThread_tryAcquireN()      { testReleaseAcquireSameThread(false, AcquireMethod.tryAcquireN); }
529     public void testReleaseAcquireSameThread_tryAcquireN_fair() { testReleaseAcquireSameThread(true, AcquireMethod.tryAcquireN); }
530     public void testReleaseAcquireSameThread_tryAcquireTimed()       { testReleaseAcquireSameThread(false, AcquireMethod.tryAcquireTimed); }
531     public void testReleaseAcquireSameThread_tryAcquireTimed_fair()  { testReleaseAcquireSameThread(true, AcquireMethod.tryAcquireTimed); }
532     public void testReleaseAcquireSameThread_tryAcquireTimedN()      { testReleaseAcquireSameThread(false, AcquireMethod.tryAcquireTimedN); }
533     public void testReleaseAcquireSameThread_tryAcquireTimedN_fair() { testReleaseAcquireSameThread(true, AcquireMethod.tryAcquireTimedN); }
534     public void testReleaseAcquireSameThread(boolean fair,
535                                              final AcquireMethod acquirer) {
536         Semaphore s = new Semaphore(1, fair);
537         for (int i = 1; i < 6; i++) {
538             s.release(i);
539             assertEquals(1 + i, s.availablePermits());
540             try {
541                 acquirer.acquire(s, i);
542             } catch (InterruptedException e) { threadUnexpectedException(e); }
543             assertEquals(1, s.availablePermits());
544         }
545     }
546 
547     /**
548      * release in one thread enables acquire in another thread
549      */
550     public void testReleaseAcquireDifferentThreads_acquire()       { testReleaseAcquireDifferentThreads(false, AcquireMethod.acquire); }
551     public void testReleaseAcquireDifferentThreads_acquire_fair()  { testReleaseAcquireDifferentThreads(true, AcquireMethod.acquire); }
552     public void testReleaseAcquireDifferentThreads_acquireN()      { testReleaseAcquireDifferentThreads(false, AcquireMethod.acquireN); }
553     public void testReleaseAcquireDifferentThreads_acquireN_fair() { testReleaseAcquireDifferentThreads(true, AcquireMethod.acquireN); }
554     public void testReleaseAcquireDifferentThreads_acquireUninterruptibly()       { testReleaseAcquireDifferentThreads(false, AcquireMethod.acquireUninterruptibly); }
555     public void testReleaseAcquireDifferentThreads_acquireUninterruptibly_fair()  { testReleaseAcquireDifferentThreads(true, AcquireMethod.acquireUninterruptibly); }
556     public void testReleaseAcquireDifferentThreads_acquireUninterruptiblyN()      { testReleaseAcquireDifferentThreads(false, AcquireMethod.acquireUninterruptibly); }
557     public void testReleaseAcquireDifferentThreads_acquireUninterruptiblyN_fair() { testReleaseAcquireDifferentThreads(true, AcquireMethod.acquireUninterruptibly); }
558     public void testReleaseAcquireDifferentThreads_tryAcquireTimed()       { testReleaseAcquireDifferentThreads(false, AcquireMethod.tryAcquireTimed); }
559     public void testReleaseAcquireDifferentThreads_tryAcquireTimed_fair()  { testReleaseAcquireDifferentThreads(true, AcquireMethod.tryAcquireTimed); }
560     public void testReleaseAcquireDifferentThreads_tryAcquireTimedN()      { testReleaseAcquireDifferentThreads(false, AcquireMethod.tryAcquireTimedN); }
561     public void testReleaseAcquireDifferentThreads_tryAcquireTimedN_fair() { testReleaseAcquireDifferentThreads(true, AcquireMethod.tryAcquireTimedN); }
562     public void testReleaseAcquireDifferentThreads(boolean fair,
563                                                    final AcquireMethod acquirer) {
564         final Semaphore s = new Semaphore(0, fair);
565         final int rounds = 4;
566         long startTime = System.nanoTime();
567         Thread t = newStartedThread(new CheckedRunnable() {
568             public void realRun() throws InterruptedException {
569                 for (int i = 0; i < rounds; i++) {
570                     assertFalse(s.hasQueuedThreads());
571                     if (i % 2 == 0)
572                         acquirer.acquire(s);
573                     else
574                         acquirer.acquire(s, 3);
575                 }}});
576 
577         for (int i = 0; i < rounds; i++) {
578             while (! (s.availablePermits() == 0 && s.hasQueuedThreads()))
579                 Thread.yield();
580             assertTrue(t.isAlive());
581             if (i % 2 == 0)
582                 s.release();
583             else
584                 s.release(3);
585         }
586         awaitTermination(t);
587         assertEquals(0, s.availablePermits());
588         assertTrue(millisElapsedSince(startTime) < LONG_DELAY_MS);
589     }
590 
591     /**
592      * fair locks are strictly FIFO
593      */
594     public void testFairLocksFifo() {
595         final PublicSemaphore s = new PublicSemaphore(1, true);
596         final CountDownLatch pleaseRelease = new CountDownLatch(1);
597         Thread t1 = newStartedThread(new CheckedRunnable() {
598             public void realRun() throws InterruptedException {
599                 // Will block; permits are available, but not three
600                 s.acquire(3);
601             }});
602 
603         waitForQueuedThread(s, t1);
604 
605         Thread t2 = newStartedThread(new CheckedRunnable() {
606             public void realRun() throws InterruptedException {
607                 // Will fail, even though 1 permit is available
608                 assertFalse(s.tryAcquire(0L, MILLISECONDS));
609                 assertFalse(s.tryAcquire(1, 0L, MILLISECONDS));
610 
611                 // untimed tryAcquire will barge and succeed
612                 assertTrue(s.tryAcquire());
613                 s.release(2);
614                 assertTrue(s.tryAcquire(2));
615                 s.release();
616 
617                 pleaseRelease.countDown();
618                 // Will queue up behind t1, even though 1 permit is available
619                 s.acquire();
620             }});
621 
622         await(pleaseRelease);
623         waitForQueuedThread(s, t2);
624         s.release(2);
625         awaitTermination(t1);
626         assertTrue(t2.isAlive());
627         s.release();
628         awaitTermination(t2);
629     }
630 
631     /**
632      * toString indicates current number of permits
633      */
634     public void testToString()      { testToString(false); }
635     public void testToString_fair() { testToString(true); }
636     public void testToString(boolean fair) {
637         PublicSemaphore s = new PublicSemaphore(0, fair);
638         assertTrue(s.toString().contains("Permits = 0"));
639         s.release();
640         assertTrue(s.toString().contains("Permits = 1"));
641         s.release(2);
642         assertTrue(s.toString().contains("Permits = 3"));
643         s.reducePermits(5);
644         assertTrue(s.toString().contains("Permits = -2"));
645     }
646 
647 }
648