| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586 |
- // Licensed to the .NET Foundation under one or more agreements.
- // The .NET Foundation licenses this file to you under the Apache 2.0 License.
- // See the LICENSE file in the project root for more information.
- using System.Collections.Generic;
- namespace System.Reactive.Concurrency
- {
- /// <summary>
- /// Asynchronous lock.
- /// </summary>
- public sealed class AsyncLock : IDisposable
- {
- private readonly Queue<Action> queue = new Queue<Action>();
- private bool isAcquired = false;
- private bool hasFaulted = false;
- /// <summary>
- /// Queues the action for execution. If the caller acquires the lock and becomes the owner,
- /// the queue is processed. If the lock is already owned, the action is queued and will get
- /// processed by the owner.
- /// </summary>
- /// <param name="action">Action to queue for execution.</param>
- /// <exception cref="ArgumentNullException"><paramref name="action"/> is null.</exception>
- public void Wait(Action action)
- {
- if (action == null)
- throw new ArgumentNullException(nameof(action));
- var isOwner = false;
- lock (queue)
- {
- if (!hasFaulted)
- {
- queue.Enqueue(action);
- isOwner = !isAcquired;
- isAcquired = true;
- }
- }
- if (isOwner)
- {
- while (true)
- {
- var work = default(Action);
- lock (queue)
- {
- if (queue.Count > 0)
- work = queue.Dequeue();
- else
- {
- isAcquired = false;
- break;
- }
- }
- try
- {
- work();
- }
- catch
- {
- lock (queue)
- {
- queue.Clear();
- hasFaulted = true;
- }
- throw;
- }
- }
- }
- }
- /// <summary>
- /// Clears the work items in the queue and drops further work being queued.
- /// </summary>
- public void Dispose()
- {
- lock (queue)
- {
- queue.Clear();
- hasFaulted = true;
- }
- }
- }
- }
|