|
18 | 18 | import java.lang.Thread.UncaughtExceptionHandler;
|
19 | 19 | import java.util.List;
|
20 | 20 | import java.util.concurrent.FutureTask;
|
| 21 | +import java.util.concurrent.atomic.*; |
21 | 22 |
|
22 | 23 | import org.junit.Test;
|
23 | 24 |
|
@@ -283,4 +284,114 @@ public void run() {
|
283 | 284 | TestHelper.race(r1, r2);
|
284 | 285 | }
|
285 | 286 | }
|
| 287 | + |
| 288 | + @Test |
| 289 | + public void syncWorkerCancelRace() { |
| 290 | + for (int i = 0; i < 10000; i++) { |
| 291 | + final CompositeDisposable set = new CompositeDisposable(); |
| 292 | + final AtomicBoolean interrupted = new AtomicBoolean(); |
| 293 | + final AtomicInteger sync = new AtomicInteger(2); |
| 294 | + final AtomicInteger syncb = new AtomicInteger(2); |
| 295 | + |
| 296 | + Runnable r0 = new Runnable() { |
| 297 | + @Override |
| 298 | + public void run() { |
| 299 | + set.dispose(); |
| 300 | + if (sync.decrementAndGet() != 0) { |
| 301 | + while (sync.get() != 0) { } |
| 302 | + } |
| 303 | + if (syncb.decrementAndGet() != 0) { |
| 304 | + while (syncb.get() != 0) { } |
| 305 | + } |
| 306 | + for (int j = 0; j < 1000; j++) { |
| 307 | + if (Thread.currentThread().isInterrupted()) { |
| 308 | + interrupted.set(true); |
| 309 | + break; |
| 310 | + } |
| 311 | + } |
| 312 | + } |
| 313 | + }; |
| 314 | + |
| 315 | + final ScheduledRunnable run = new ScheduledRunnable(r0, set); |
| 316 | + set.add(run); |
| 317 | + |
| 318 | + final FutureTask<Void> ft = new FutureTask<Void>(run, null); |
| 319 | + |
| 320 | + Runnable r2 = new Runnable() { |
| 321 | + @Override |
| 322 | + public void run() { |
| 323 | + if (sync.decrementAndGet() != 0) { |
| 324 | + while (sync.get() != 0) { } |
| 325 | + } |
| 326 | + run.setFuture(ft); |
| 327 | + if (syncb.decrementAndGet() != 0) { |
| 328 | + while (syncb.get() != 0) { } |
| 329 | + } |
| 330 | + } |
| 331 | + }; |
| 332 | + |
| 333 | + TestHelper.race(ft, r2); |
| 334 | + |
| 335 | + assertFalse("The task was interrupted", interrupted.get()); |
| 336 | + } |
| 337 | + } |
| 338 | + |
| 339 | + @Test |
| 340 | + public void disposeAfterRun() { |
| 341 | + final ScheduledRunnable run = new ScheduledRunnable(Functions.EMPTY_RUNNABLE, null); |
| 342 | + |
| 343 | + run.run(); |
| 344 | + assertEquals(ScheduledRunnable.DONE, run.get(ScheduledRunnable.FUTURE_INDEX)); |
| 345 | + |
| 346 | + run.dispose(); |
| 347 | + assertEquals(ScheduledRunnable.DONE, run.get(ScheduledRunnable.FUTURE_INDEX)); |
| 348 | + } |
| 349 | + |
| 350 | + @Test |
| 351 | + public void syncDisposeIdempotent() { |
| 352 | + final ScheduledRunnable run = new ScheduledRunnable(Functions.EMPTY_RUNNABLE, null); |
| 353 | + run.set(ScheduledRunnable.THREAD_INDEX, Thread.currentThread()); |
| 354 | + |
| 355 | + run.dispose(); |
| 356 | + assertEquals(ScheduledRunnable.SYNC_DISPOSED, run.get(ScheduledRunnable.FUTURE_INDEX)); |
| 357 | + run.dispose(); |
| 358 | + assertEquals(ScheduledRunnable.SYNC_DISPOSED, run.get(ScheduledRunnable.FUTURE_INDEX)); |
| 359 | + run.run(); |
| 360 | + assertEquals(ScheduledRunnable.SYNC_DISPOSED, run.get(ScheduledRunnable.FUTURE_INDEX)); |
| 361 | + } |
| 362 | + |
| 363 | + @Test |
| 364 | + public void asyncDisposeIdempotent() { |
| 365 | + final ScheduledRunnable run = new ScheduledRunnable(Functions.EMPTY_RUNNABLE, null); |
| 366 | + |
| 367 | + run.dispose(); |
| 368 | + assertEquals(ScheduledRunnable.ASYNC_DISPOSED, run.get(ScheduledRunnable.FUTURE_INDEX)); |
| 369 | + run.dispose(); |
| 370 | + assertEquals(ScheduledRunnable.ASYNC_DISPOSED, run.get(ScheduledRunnable.FUTURE_INDEX)); |
| 371 | + run.run(); |
| 372 | + assertEquals(ScheduledRunnable.ASYNC_DISPOSED, run.get(ScheduledRunnable.FUTURE_INDEX)); |
| 373 | + } |
| 374 | + |
| 375 | + |
| 376 | + @Test |
| 377 | + public void noParentIsDisposed() { |
| 378 | + ScheduledRunnable run = new ScheduledRunnable(Functions.EMPTY_RUNNABLE, null); |
| 379 | + assertFalse(run.isDisposed()); |
| 380 | + run.run(); |
| 381 | + assertTrue(run.isDisposed()); |
| 382 | + } |
| 383 | + |
| 384 | + @Test |
| 385 | + public void withParentIsDisposed() { |
| 386 | + CompositeDisposable set = new CompositeDisposable(); |
| 387 | + ScheduledRunnable run = new ScheduledRunnable(Functions.EMPTY_RUNNABLE, set); |
| 388 | + set.add(run); |
| 389 | + |
| 390 | + assertFalse(run.isDisposed()); |
| 391 | + |
| 392 | + run.run(); |
| 393 | + assertTrue(run.isDisposed()); |
| 394 | + |
| 395 | + assertFalse(set.remove(run)); |
| 396 | + } |
286 | 397 | }
|
0 commit comments