using System.Threading.Tasks;
public static void Main()
await Task.Delay(TimeSpan.FromSeconds(2));
Console.WriteLine("Deadlocked?");
var asyncLock = new CellWars.Threading.AsyncLock();
for (var i = 0; i < 1000; ++i)
using (await asyncLock.LockAsync(CancellationToken.None))
using (await asyncLock.LockAsync(CancellationToken.None))
using (await asyncLock.LockAsync(CancellationToken.None))
Console.WriteLine("Not okay");
internal class AsyncLock : IAsyncDisposable
private static readonly ObjectPool<SemaphoreSlim> s_semaphorePool = (new DefaultObjectPoolProvider { MaximumRetained = 100 })
.Create(new SemaphoreSlimPooledObjectPolicy());
private AsyncLocal<SemaphoreSlim> _currentSemaphore;
private SemaphoreSlim _topLevelSemaphore;
private bool _isDisposed;
_topLevelSemaphore = s_semaphorePool.Get();
_currentSemaphore = new AsyncLocal<SemaphoreSlim>();
public Task<IAsyncDisposable> TakeLockAsync()
throw new ObjectDisposedException(nameof(AsyncLock));
_currentSemaphore.Value ??= _topLevelSemaphore;
SemaphoreSlim currentSem = _currentSemaphore.Value;
var nextSem = s_semaphorePool.Get();
_currentSemaphore.Value = nextSem;
var safeRelease = new SafeSemaphoreRelease(currentSem, nextSem, this);
return TakeLockCoreAsync(currentSem, safeRelease);
private async Task<IAsyncDisposable> TakeLockCoreAsync(SemaphoreSlim currentSemaphore, SafeSemaphoreRelease safeSemaphoreRelease)
await currentSemaphore.WaitAsync();
return safeSemaphoreRelease;
public IDisposable TakeLock()
throw new ObjectDisposedException(nameof(AsyncLock));
_currentSemaphore.Value ??= _topLevelSemaphore;
SemaphoreSlim currentSem = _currentSemaphore.Value;
var nextSem = s_semaphorePool.Get();
_currentSemaphore.Value = nextSem;
return new SafeSemaphoreRelease(currentSem, nextSem, this);
public async ValueTask DisposeAsync()
await _topLevelSemaphore.WaitAsync();
_topLevelSemaphore.Release();
s_semaphorePool.Return(_topLevelSemaphore);
_topLevelSemaphore = null;
private struct SafeSemaphoreRelease : IAsyncDisposable, IDisposable
private SemaphoreSlim _currentSemaphore;
private SemaphoreSlim _nextSemaphore;
private AsyncLock _asyncLock;
public SafeSemaphoreRelease(SemaphoreSlim currentSemaphore, SemaphoreSlim nextSemaphore, AsyncLock asyncLock)
_currentSemaphore = currentSemaphore;
_nextSemaphore = nextSemaphore;
public ValueTask DisposeAsync()
Fx.Assert(_nextSemaphore == _asyncLock._currentSemaphore.Value, "_nextSemaphore was expected to by the current semaphore");
if (_currentSemaphore == _asyncLock._topLevelSemaphore)
_asyncLock._currentSemaphore.Value = null;
_asyncLock._currentSemaphore.Value = _currentSemaphore;
return DisposeCoreAsync();
private async ValueTask DisposeCoreAsync()
await _nextSemaphore.WaitAsync();
_currentSemaphore.Release();
_nextSemaphore.Release();
s_semaphorePool.Return(_nextSemaphore);
Fx.Assert(_nextSemaphore == _asyncLock._currentSemaphore.Value, "_nextSemaphore was expected to by the current semaphore");
if (_currentSemaphore == _asyncLock._topLevelSemaphore)
_asyncLock._currentSemaphore.Value = null;
_asyncLock._currentSemaphore.Value = _currentSemaphore;
_currentSemaphore.Release();
_nextSemaphore.Release();
s_semaphorePool.Return(_nextSemaphore);
private class SemaphoreSlimPooledObjectPolicy : PooledObjectPolicy<SemaphoreSlim>
public override SemaphoreSlim Create()
return new SemaphoreSlim(1);
public override bool Return(SemaphoreSlim obj)
if (obj.CurrentCount != 1)
Fx.Assert("Shouldn't be returning semaphore with a count != 1");