public async Task TestReleaseThenReAcquireWithConcurrentAccess() { var sut = new FastAsyncLock(); using (var otherThread = new AsyncTestRunner(OtherThread)) { await otherThread.AdvanceTo(1); Assert.IsTrue(otherThread.HasLock()); await otherThread.AdvanceTo(2); Assert.IsFalse(otherThread.HasLock()); using (await sut.LockAsync(CancellationToken.None)) { await Task.Yield(); await otherThread.AdvanceAndFreezeBefore(3); Assert.IsFalse(otherThread.HasLock()); } await otherThread.AdvanceTo(4); Assert.IsTrue(otherThread.HasLock()); } async Task OtherThread(CancellationToken ct, AsyncTestRunner r) { using (await sut.LockAsync(CancellationToken.None)) { await Task.Yield(); r.HasLock(true); r.Sync(position: 1); } await Task.Yield(); r.HasLock(false); r.Sync(position: 2); using (await sut.LockAsync(CancellationToken.None)) { await Task.Yield(); r.HasLock(true); r.Sync(position: 3); r.Sync(position: 4); } await Task.Yield(); r.HasLock(false); } }
public async Task TestExitFromAnotherExecutionContext() { var sut = new FastAsyncLock(); var locking = default(IDisposable); var entryContext = default(AsyncTestContext); using (var externalThread = new AsyncTestRunner(ExternalThread)) { entryContext = AsyncTestContext.Current; locking = await sut.LockAsync(CancellationToken.None); await externalThread.AdvanceTo(1); Assert.IsTrue(externalThread.HasLock()); await externalThread.AdvanceToEnd(); } async Task ExternalThread(CancellationToken ct, AsyncTestRunner r) { Assert.AreNotEqual(entryContext, AsyncTestContext.Current); locking.Dispose(); // Try to relock from the external execution context to validate that the lock was effectively released using (await sut.LockAsync(ct)) { r.HasLock(true); r.Sync(position: 1); await Task.Yield(); } r.HasLock(false); } }
[Ignore] // https://github.com/nventive/Uno.Core/issues/48 public async Task TestUnlockReleaseNextSynchronously() { Console.WriteLine($"Running on thread {Thread.CurrentThread.ManagedThreadId}"); var sut = new FastAsyncLock(); var thread1ExitingThread = -1; var thread2LockingTask = default(Task <IDisposable>); var thread1 = new AsyncTestRunner(Thread1); var thread2 = new AsyncTestRunner(); thread2.Run(Thread2); using (thread1) using (thread2) { // Acquire the lock on thread 1 await thread1.AdvanceTo(1); // Wait for the thread 2 to be stuck to acquire the lock var thread2Locking = thread2.AdvanceTo(1); while (thread2LockingTask == null) { await Task.Yield(); } // Make sure that the thread 2 is really awaiting the thread2LockingTask before requesting thread1 to continue await Task.Delay(100); Assert.AreEqual(TaskStatus.WaitingForActivation, thread2LockingTask.Status); // Relase the thread 1 and make sure that the thread 2 is able to acquire the lock var thread1Release = thread1.AdvanceTo(2); await thread2Locking; } async Task Thread1(CancellationToken ct, AsyncTestRunner r) { Console.WriteLine($"Thread 1: {Thread.CurrentThread.ManagedThreadId}"); var ctx = AsyncTestContext.Current; using (await sut.LockAsync(ct)) { Console.WriteLine($"Acquired lock for Thread1 on thread: {Thread.CurrentThread.ManagedThreadId}"); ctx.Validate(); await Task.Yield(); r.Sync(position: 1); thread1ExitingThread = Thread.CurrentThread.ManagedThreadId; Console.WriteLine($"Releasing lock from Thread1 on thread: {Thread.CurrentThread.ManagedThreadId}"); } Console.WriteLine($"Released lock from Thread1 on thread: {Thread.CurrentThread.ManagedThreadId}"); // This must have run synchronously when lock got released (disposed). Assert.AreEqual(TaskStatus.RanToCompletion, thread2LockingTask.Status); r.Sync(position: 2); } async Task Thread2(CancellationToken ct, AsyncTestRunner r) { Console.WriteLine($"Thread 2: {Thread.CurrentThread.ManagedThreadId}"); var ctx = AsyncTestContext.Current; thread2LockingTask = sut.LockAsync(ct); // Validate that we are running on thread 2 Assert.AreEqual(thread2.ThreadId, Thread.CurrentThread.ManagedThreadId); Console.WriteLine("Thread 2 is waiting for lock"); using (await thread2LockingTask) { Console.WriteLine($"Acquired lock for Thread2 on thread: {Thread.CurrentThread.ManagedThreadId}"); // Here we should run on the thread 1 since the lock is released synchronously from thread 1 Assert.AreEqual(thread1ExitingThread, Thread.CurrentThread.ManagedThreadId); // But we should have kept the ExecutionContext from the thread 2 ctx.Validate(); await Task.Yield(); r.Sync(position: 1); Console.WriteLine($"Releasing lock from Thread2 on thread: {Thread.CurrentThread.ManagedThreadId}"); } Console.WriteLine($"Released lock from Thread2 on thread: {Thread.CurrentThread.ManagedThreadId}"); r.Sync(position: 2); } }
public async Task TestReEntrencyWithConcurrentAccess() { var sut = new FastAsyncLock(); using (var thread2 = new AsyncTestRunner(Thread2)) using (var thread1 = new AsyncTestRunner(Thread1)) { // Enter from main thread await thread1.AdvanceTo(1); // Try enter from second thread var thread2Locking = thread2.AdvanceTo(2); await Task.Delay(100); Assert.AreEqual(TaskStatus.WaitingForActivation, thread2Locking.Status); // ReEnter await thread1.AdvanceTo(2); // Exit once await thread1.AdvanceTo(3); Assert.AreEqual(TaskStatus.WaitingForActivation, thread2Locking.Status); // Final exit thread1.AdvanceTo(4); await thread2Locking; } async Task Thread1(CancellationToken ct, AsyncTestRunner r) { using (await sut.LockAsync(CancellationToken.None)) { await Task.Yield(); r.HasLock(true); r.Sync(position: 1); using (await sut.LockAsync(CancellationToken.None)) { await Task.Yield(); r.Sync(position: 2); } r.Sync(position: 3); } r.HasLock(false); r.Sync(position: 4); } async Task Thread2(CancellationToken ct, AsyncTestRunner r) { using (await sut.LockAsync(ct)) { r.HasLock(true); r.Sync(position: 1); } r.HasLock(false); r.Sync(position: 2); } }
public async Task TestConcurrentCancelSecondWithThird() { var sut = new FastAsyncLock(); var ct = new CancellationTokenSource(); var thread2LockingTask = default(Task <IDisposable>); using (var thread1 = new AsyncTestRunner(Thread1)) using (var thread2 = new AsyncTestRunner(Thread2)) using (var thread3 = new AsyncTestRunner(Thread3)) { // Acquire the lock on thread 1 THEN await thread1.AdvanceTo(1); Assert.IsTrue(thread1.HasLock()); // Try to acquire the lock from this async context await thread2.AdvanceAndFreezeBefore(1); //var locking = sut.LockAsync(ct.Token); Assert.AreEqual(TaskStatus.WaitingForActivation, thread2LockingTask.Status); Assert.IsFalse(thread2.HasLock()); // Try to acquire it on thread 3 var t3Locked = thread3.AdvanceTo(1); await thread3.IsFrozen(); Assert.IsFalse(thread3.HasLock()); // But cancel before the other async context completes ct.Cancel(); Assert.AreEqual(TaskStatus.Canceled, thread2LockingTask.Status); // Release the lock from thread1, and wait for thread 3 to acquire the lock await thread1.AdvanceAndFreezeBefore(2); // will freeze in continuation of thread 3 await t3Locked; //Assert.IsFalse(thread1.HasLock()); // flag not set yet: the thread 1 is dead locked by the continuation of thread 3 Assert.IsTrue(thread3.HasLock()); await thread3.AdvanceToEnd(); //await thread1.AdvanceToEnd(); await Task.Delay(500); Assert.IsFalse(thread1.HasLock()); Assert.IsFalse(thread3.HasLock()); } async Task Thread1(CancellationToken ct2, AsyncTestRunner r) { using (await sut.LockAsync(ct2)) { r.HasLock(true); r.Sync(position: 1); } r.HasLock(false); r.Sync(position: 2); }; async Task Thread2(CancellationToken ct2, AsyncTestRunner r) { thread2LockingTask = sut.LockAsync(ct.Token); using (await thread2LockingTask) { r.HasLock(true); r.Sync(position: 1); } r.HasLock(false); r.Sync(position: 2); } async Task Thread3(CancellationToken ct2, AsyncTestRunner r) { using (await sut.LockAsync(ct2)) { r.HasLock(true); r.Sync(position: 1); } r.HasLock(false); r.Sync(position: 2); }; }