1 /* 2 * Written by Doug Lea and Martin Buchholz with assistance from 3 * members of JCP JSR-166 Expert Group and released to the public 4 * domain, as explained at 5 * http://creativecommons.org/publicdomain/zero/1.0/ 6 * Other contributors include Andrew Wright, Jeffrey Hayes, 7 * Pat Fisher, Mike Judd. 8 */ 9 10 package jsr166; 11 12 import static java.util.concurrent.TimeUnit.MILLISECONDS; 13 14 import java.util.concurrent.CountDownLatch; 15 import java.util.concurrent.atomic.AtomicBoolean; 16 import java.util.concurrent.locks.LockSupport; 17 18 import junit.framework.Test; 19 import junit.framework.TestSuite; 20 21 public class LockSupportTest 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(LockSupportTest.class); 30 // } 31 32 static { 33 // Reduce the risk of rare disastrous classloading in first call to 34 // LockSupport.park: https://bugs.openjdk.java.net/browse/JDK-8074773 35 Class<?> ensureLoaded = LockSupport.class; 36 } 37 38 /** 39 * Returns the blocker object used by tests in this file. 40 * Any old object will do; we'll return a convenient one. 41 */ 42 static Object theBlocker() { 43 return LockSupportTest.class; 44 } 45 46 enum ParkMethod { 47 park() { 48 void park() { 49 LockSupport.park(); 50 } 51 void park(long millis) { 52 throw new UnsupportedOperationException(); 53 } 54 }, 55 parkUntil() { 56 void park(long millis) { 57 LockSupport.parkUntil(deadline(millis)); 58 } 59 }, 60 parkNanos() { 61 void park(long millis) { 62 LockSupport.parkNanos(MILLISECONDS.toNanos(millis)); 63 } 64 }, 65 parkBlocker() { 66 void park() { 67 LockSupport.park(theBlocker()); 68 } 69 void park(long millis) { 70 throw new UnsupportedOperationException(); 71 } 72 }, 73 parkUntilBlocker() { 74 void park(long millis) { 75 LockSupport.parkUntil(theBlocker(), deadline(millis)); 76 } 77 }, 78 parkNanosBlocker() { 79 void park(long millis) { 80 LockSupport.parkNanos(theBlocker(), 81 MILLISECONDS.toNanos(millis)); 82 } 83 }; 84 85 void park() { park(2 * LONG_DELAY_MS); } 86 abstract void park(long millis); 87 88 /** Returns a deadline to use with parkUntil. */ 89 long deadline(long millis) { 90 // beware of rounding 91 return System.currentTimeMillis() + millis + 1; 92 } 93 } 94 95 /** 96 * park is released by subsequent unpark 97 */ 98 public void testParkBeforeUnpark_park() { 99 testParkBeforeUnpark(ParkMethod.park); 100 } 101 public void testParkBeforeUnpark_parkNanos() { 102 testParkBeforeUnpark(ParkMethod.parkNanos); 103 } 104 public void testParkBeforeUnpark_parkUntil() { 105 testParkBeforeUnpark(ParkMethod.parkUntil); 106 } 107 public void testParkBeforeUnpark_parkBlocker() { 108 testParkBeforeUnpark(ParkMethod.parkBlocker); 109 } 110 public void testParkBeforeUnpark_parkNanosBlocker() { 111 testParkBeforeUnpark(ParkMethod.parkNanosBlocker); 112 } 113 public void testParkBeforeUnpark_parkUntilBlocker() { 114 testParkBeforeUnpark(ParkMethod.parkUntilBlocker); 115 } 116 public void testParkBeforeUnpark(final ParkMethod parkMethod) { 117 final CountDownLatch pleaseUnpark = new CountDownLatch(1); 118 Thread t = newStartedThread(new CheckedRunnable() { 119 public void realRun() { 120 pleaseUnpark.countDown(); 121 parkMethod.park(); 122 }}); 123 124 await(pleaseUnpark); 125 LockSupport.unpark(t); 126 awaitTermination(t); 127 } 128 129 /** 130 * park is released by preceding unpark 131 */ 132 public void testParkAfterUnpark_park() { 133 testParkAfterUnpark(ParkMethod.park); 134 } 135 public void testParkAfterUnpark_parkNanos() { 136 testParkAfterUnpark(ParkMethod.parkNanos); 137 } 138 public void testParkAfterUnpark_parkUntil() { 139 testParkAfterUnpark(ParkMethod.parkUntil); 140 } 141 public void testParkAfterUnpark_parkBlocker() { 142 testParkAfterUnpark(ParkMethod.parkBlocker); 143 } 144 public void testParkAfterUnpark_parkNanosBlocker() { 145 testParkAfterUnpark(ParkMethod.parkNanosBlocker); 146 } 147 public void testParkAfterUnpark_parkUntilBlocker() { 148 testParkAfterUnpark(ParkMethod.parkUntilBlocker); 149 } 150 public void testParkAfterUnpark(final ParkMethod parkMethod) { 151 final CountDownLatch pleaseUnpark = new CountDownLatch(1); 152 final AtomicBoolean pleasePark = new AtomicBoolean(false); 153 Thread t = newStartedThread(new CheckedRunnable() { 154 public void realRun() { 155 pleaseUnpark.countDown(); 156 while (!pleasePark.get()) 157 Thread.yield(); 158 parkMethod.park(); 159 }}); 160 161 await(pleaseUnpark); 162 LockSupport.unpark(t); 163 pleasePark.set(true); 164 awaitTermination(t); 165 } 166 167 /** 168 * park is released by subsequent interrupt 169 */ 170 public void testParkBeforeInterrupt_park() { 171 testParkBeforeInterrupt(ParkMethod.park); 172 } 173 public void testParkBeforeInterrupt_parkNanos() { 174 testParkBeforeInterrupt(ParkMethod.parkNanos); 175 } 176 public void testParkBeforeInterrupt_parkUntil() { 177 testParkBeforeInterrupt(ParkMethod.parkUntil); 178 } 179 public void testParkBeforeInterrupt_parkBlocker() { 180 testParkBeforeInterrupt(ParkMethod.parkBlocker); 181 } 182 public void testParkBeforeInterrupt_parkNanosBlocker() { 183 testParkBeforeInterrupt(ParkMethod.parkNanosBlocker); 184 } 185 public void testParkBeforeInterrupt_parkUntilBlocker() { 186 testParkBeforeInterrupt(ParkMethod.parkUntilBlocker); 187 } 188 public void testParkBeforeInterrupt(final ParkMethod parkMethod) { 189 final CountDownLatch pleaseInterrupt = new CountDownLatch(1); 190 Thread t = newStartedThread(new CheckedRunnable() { 191 public void realRun() { 192 pleaseInterrupt.countDown(); 193 do { 194 parkMethod.park(); 195 // park may return spuriously 196 } while (! Thread.currentThread().isInterrupted()); 197 }}); 198 199 await(pleaseInterrupt); 200 assertThreadStaysAlive(t); 201 t.interrupt(); 202 awaitTermination(t); 203 } 204 205 /** 206 * park is released by preceding interrupt 207 */ 208 public void testParkAfterInterrupt_park() { 209 testParkAfterInterrupt(ParkMethod.park); 210 } 211 public void testParkAfterInterrupt_parkNanos() { 212 testParkAfterInterrupt(ParkMethod.parkNanos); 213 } 214 public void testParkAfterInterrupt_parkUntil() { 215 testParkAfterInterrupt(ParkMethod.parkUntil); 216 } 217 public void testParkAfterInterrupt_parkBlocker() { 218 testParkAfterInterrupt(ParkMethod.parkBlocker); 219 } 220 public void testParkAfterInterrupt_parkNanosBlocker() { 221 testParkAfterInterrupt(ParkMethod.parkNanosBlocker); 222 } 223 public void testParkAfterInterrupt_parkUntilBlocker() { 224 testParkAfterInterrupt(ParkMethod.parkUntilBlocker); 225 } 226 public void testParkAfterInterrupt(final ParkMethod parkMethod) { 227 final CountDownLatch pleaseInterrupt = new CountDownLatch(1); 228 final AtomicBoolean pleasePark = new AtomicBoolean(false); 229 Thread t = newStartedThread(new CheckedRunnable() { 230 public void realRun() throws Exception { 231 pleaseInterrupt.countDown(); 232 while (!pleasePark.get()) 233 Thread.yield(); 234 assertTrue(Thread.currentThread().isInterrupted()); 235 parkMethod.park(); 236 assertTrue(Thread.currentThread().isInterrupted()); 237 }}); 238 239 await(pleaseInterrupt); 240 t.interrupt(); 241 pleasePark.set(true); 242 awaitTermination(t); 243 } 244 245 /** 246 * timed park times out if not unparked 247 */ 248 public void testParkTimesOut_parkNanos() { 249 testParkTimesOut(ParkMethod.parkNanos); 250 } 251 public void testParkTimesOut_parkUntil() { 252 testParkTimesOut(ParkMethod.parkUntil); 253 } 254 public void testParkTimesOut_parkNanosBlocker() { 255 testParkTimesOut(ParkMethod.parkNanosBlocker); 256 } 257 public void testParkTimesOut_parkUntilBlocker() { 258 testParkTimesOut(ParkMethod.parkUntilBlocker); 259 } 260 public void testParkTimesOut(final ParkMethod parkMethod) { 261 Thread t = newStartedThread(new CheckedRunnable() { 262 public void realRun() { 263 for (;;) { 264 long startTime = System.nanoTime(); 265 parkMethod.park(timeoutMillis()); 266 // park may return spuriously 267 if (millisElapsedSince(startTime) >= timeoutMillis()) 268 return; 269 } 270 }}); 271 272 awaitTermination(t); 273 } 274 275 /** 276 * getBlocker(null) throws NullPointerException 277 */ 278 public void testGetBlockerNull() { 279 try { 280 LockSupport.getBlocker(null); 281 shouldThrow(); 282 } catch (NullPointerException success) {} 283 } 284 285 /** 286 * getBlocker returns the blocker object passed to park 287 */ 288 public void testGetBlocker_parkBlocker() { 289 testGetBlocker(ParkMethod.parkBlocker); 290 } 291 public void testGetBlocker_parkNanosBlocker() { 292 testGetBlocker(ParkMethod.parkNanosBlocker); 293 } 294 public void testGetBlocker_parkUntilBlocker() { 295 testGetBlocker(ParkMethod.parkUntilBlocker); 296 } 297 public void testGetBlocker(final ParkMethod parkMethod) { 298 final CountDownLatch started = new CountDownLatch(1); 299 Thread t = newStartedThread(new CheckedRunnable() { 300 public void realRun() { 301 Thread t = Thread.currentThread(); 302 started.countDown(); 303 do { 304 assertNull(LockSupport.getBlocker(t)); 305 parkMethod.park(); 306 assertNull(LockSupport.getBlocker(t)); 307 // park may return spuriously 308 } while (! Thread.currentThread().isInterrupted()); 309 }}); 310 311 long startTime = System.nanoTime(); 312 await(started); 313 for (;;) { 314 Object x = LockSupport.getBlocker(t); 315 if (x == theBlocker()) { // success 316 t.interrupt(); 317 awaitTermination(t); 318 assertNull(LockSupport.getBlocker(t)); 319 return; 320 } else { 321 assertNull(x); // ok 322 if (millisElapsedSince(startTime) > LONG_DELAY_MS) 323 fail("timed out"); 324 Thread.yield(); 325 } 326 } 327 } 328 329 /** 330 * timed park(0) returns immediately. 331 * 332 * Requires hotspot fix for: 333 * 6763959 java.util.concurrent.locks.LockSupport.parkUntil(0) blocks forever 334 * which is in jdk7-b118 and 6u25. 335 */ 336 public void testPark0_parkNanos() { 337 testPark0(ParkMethod.parkNanos); 338 } 339 public void testPark0_parkUntil() { 340 testPark0(ParkMethod.parkUntil); 341 } 342 public void testPark0_parkNanosBlocker() { 343 testPark0(ParkMethod.parkNanosBlocker); 344 } 345 public void testPark0_parkUntilBlocker() { 346 testPark0(ParkMethod.parkUntilBlocker); 347 } 348 public void testPark0(final ParkMethod parkMethod) { 349 Thread t = newStartedThread(new CheckedRunnable() { 350 public void realRun() { 351 parkMethod.park(0L); 352 }}); 353 354 awaitTermination(t); 355 } 356 357 /** 358 * timed park(Long.MIN_VALUE) returns immediately. 359 */ 360 public void testParkNeg_parkNanos() { 361 testParkNeg(ParkMethod.parkNanos); 362 } 363 public void testParkNeg_parkUntil() { 364 testParkNeg(ParkMethod.parkUntil); 365 } 366 public void testParkNeg_parkNanosBlocker() { 367 testParkNeg(ParkMethod.parkNanosBlocker); 368 } 369 public void testParkNeg_parkUntilBlocker() { 370 testParkNeg(ParkMethod.parkUntilBlocker); 371 } 372 public void testParkNeg(final ParkMethod parkMethod) { 373 Thread t = newStartedThread(new CheckedRunnable() { 374 public void realRun() { 375 parkMethod.park(Long.MIN_VALUE); 376 }}); 377 378 awaitTermination(t); 379 } 380 } 381