123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941 |
- #pragma warning disable CS1591 // Missing XML comment for publicly visible type or member
- using System;
- using System.Collections.Generic;
- using System.Diagnostics;
- using System.Runtime.CompilerServices;
- using System.Runtime.ExceptionServices;
- using System.Runtime.InteropServices;
- using System.Threading;
- using Cysharp.Threading.Tasks.Internal;
- namespace Cysharp.Threading.Tasks
- {
- public interface IResolvePromise
- {
- bool TrySetResult();
- }
- public interface IResolvePromise<T>
- {
- bool TrySetResult(T value);
- }
- public interface IRejectPromise
- {
- bool TrySetException(Exception exception);
- }
- public interface ICancelPromise
- {
- bool TrySetCanceled(CancellationToken cancellationToken = default);
- }
- public interface IPromise<T> : IResolvePromise<T>, IRejectPromise, ICancelPromise
- {
- }
- public interface IPromise : IResolvePromise, IRejectPromise, ICancelPromise
- {
- }
- internal class ExceptionHolder
- {
- ExceptionDispatchInfo exception;
- bool calledGet = false;
- public ExceptionHolder(ExceptionDispatchInfo exception)
- {
- this.exception = exception;
- }
- public ExceptionDispatchInfo GetException()
- {
- if (!calledGet)
- {
- calledGet = true;
- GC.SuppressFinalize(this);
- }
- return exception;
- }
- ~ExceptionHolder()
- {
- if (!calledGet)
- {
- UniTaskScheduler.PublishUnobservedTaskException(exception.SourceException);
- }
- }
- }
- [StructLayout(LayoutKind.Auto)]
- public struct UniTaskCompletionSourceCore<TResult>
- {
- // Struct Size: TResult + (8 + 2 + 1 + 1 + 8 + 8)
- TResult result;
- object error; // ExceptionHolder or OperationCanceledException
- short version;
- bool hasUnhandledError;
- int completedCount; // 0: completed == false
- Action<object> continuation;
- object continuationState;
- [DebuggerHidden]
- public void Reset()
- {
- ReportUnhandledError();
- unchecked
- {
- version += 1; // incr version.
- }
- completedCount = 0;
- result = default;
- error = null;
- hasUnhandledError = false;
- continuation = null;
- continuationState = null;
- }
- void ReportUnhandledError()
- {
- if (hasUnhandledError)
- {
- try
- {
- if (error is OperationCanceledException oc)
- {
- UniTaskScheduler.PublishUnobservedTaskException(oc);
- }
- else if (error is ExceptionHolder e)
- {
- UniTaskScheduler.PublishUnobservedTaskException(e.GetException().SourceException);
- }
- }
- catch
- {
- }
- }
- }
- internal void MarkHandled()
- {
- hasUnhandledError = false;
- }
- /// <summary>Completes with a successful result.</summary>
- /// <param name="result">The result.</param>
- [DebuggerHidden]
- public bool TrySetResult(TResult result)
- {
- if (Interlocked.Increment(ref completedCount) == 1)
- {
- // setup result
- this.result = result;
- if (continuation != null || Interlocked.CompareExchange(ref this.continuation, UniTaskCompletionSourceCoreShared.s_sentinel, null) != null)
- {
- continuation(continuationState);
- return true;
- }
- }
- return false;
- }
- /// <summary>Completes with an error.</summary>
- /// <param name="error">The exception.</param>
- [DebuggerHidden]
- public bool TrySetException(Exception error)
- {
- if (Interlocked.Increment(ref completedCount) == 1)
- {
- // setup result
- this.hasUnhandledError = true;
- if (error is OperationCanceledException)
- {
- this.error = error;
- }
- else
- {
- this.error = new ExceptionHolder(ExceptionDispatchInfo.Capture(error));
- }
- if (continuation != null || Interlocked.CompareExchange(ref this.continuation, UniTaskCompletionSourceCoreShared.s_sentinel, null) != null)
- {
- continuation(continuationState);
- return true;
- }
- }
- return false;
- }
- [DebuggerHidden]
- public bool TrySetCanceled(CancellationToken cancellationToken = default)
- {
- if (Interlocked.Increment(ref completedCount) == 1)
- {
- // setup result
- this.hasUnhandledError = true;
- this.error = new OperationCanceledException(cancellationToken);
- if (continuation != null || Interlocked.CompareExchange(ref this.continuation, UniTaskCompletionSourceCoreShared.s_sentinel, null) != null)
- {
- continuation(continuationState);
- return true;
- }
- }
- return false;
- }
- /// <summary>Gets the operation version.</summary>
- [DebuggerHidden]
- public short Version => version;
- /// <summary>Gets the status of the operation.</summary>
- /// <param name="token">Opaque value that was provided to the <see cref="UniTask"/>'s constructor.</param>
- [DebuggerHidden]
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- public UniTaskStatus GetStatus(short token)
- {
- ValidateToken(token);
- return (continuation == null || (completedCount == 0)) ? UniTaskStatus.Pending
- : (error == null) ? UniTaskStatus.Succeeded
- : (error is OperationCanceledException) ? UniTaskStatus.Canceled
- : UniTaskStatus.Faulted;
- }
- /// <summary>Gets the status of the operation without token validation.</summary>
- [DebuggerHidden]
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- public UniTaskStatus UnsafeGetStatus()
- {
- return (continuation == null || (completedCount == 0)) ? UniTaskStatus.Pending
- : (error == null) ? UniTaskStatus.Succeeded
- : (error is OperationCanceledException) ? UniTaskStatus.Canceled
- : UniTaskStatus.Faulted;
- }
- /// <summary>Gets the result of the operation.</summary>
- /// <param name="token">Opaque value that was provided to the <see cref="UniTask"/>'s constructor.</param>
- // [StackTraceHidden]
- [DebuggerHidden]
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- public TResult GetResult(short token)
- {
- ValidateToken(token);
- if (completedCount == 0)
- {
- throw new InvalidOperationException("Not yet completed, UniTask only allow to use await.");
- }
- if (error != null)
- {
- hasUnhandledError = false;
- if (error is OperationCanceledException oce)
- {
- throw oce;
- }
- else if (error is ExceptionHolder eh)
- {
- eh.GetException().Throw();
- }
- throw new InvalidOperationException("Critical: invalid exception type was held.");
- }
- return result;
- }
- /// <summary>Schedules the continuation action for this operation.</summary>
- /// <param name="continuation">The continuation to invoke when the operation has completed.</param>
- /// <param name="state">The state object to pass to <paramref name="continuation"/> when it's invoked.</param>
- /// <param name="token">Opaque value that was provided to the <see cref="UniTask"/>'s constructor.</param>
- [DebuggerHidden]
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- public void OnCompleted(Action<object> continuation, object state, short token /*, ValueTaskSourceOnCompletedFlags flags */)
- {
- if (continuation == null)
- {
- throw new ArgumentNullException(nameof(continuation));
- }
- ValidateToken(token);
- /* no use ValueTaskSourceOnCOmpletedFlags, always no capture ExecutionContext and SynchronizationContext. */
- /*
- PatternA: GetStatus=Pending => OnCompleted => TrySet*** => GetResult
- PatternB: TrySet*** => GetStatus=!Pending => GetResult
- PatternC: GetStatus=Pending => TrySet/OnCompleted(race condition) => GetResult
- C.1: win OnCompleted -> TrySet invoke saved continuation
- C.2: win TrySet -> should invoke continuation here.
- */
- // not set continuation yet.
- object oldContinuation = this.continuation;
- if (oldContinuation == null)
- {
- continuationState = state;
- oldContinuation = Interlocked.CompareExchange(ref this.continuation, continuation, null);
- }
- if (oldContinuation != null)
- {
- // already running continuation in TrySet.
- // It will cause call OnCompleted multiple time, invalid.
- if (!ReferenceEquals(oldContinuation, UniTaskCompletionSourceCoreShared.s_sentinel))
- {
- throw new InvalidOperationException("Already continuation registered, can not await twice or get Status after await.");
- }
- continuation(state);
- }
- }
- [DebuggerHidden]
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- private void ValidateToken(short token)
- {
- if (token != version)
- {
- throw new InvalidOperationException("Token version is not matched, can not await twice or get Status after await.");
- }
- }
- }
- internal static class UniTaskCompletionSourceCoreShared // separated out of generic to avoid unnecessary duplication
- {
- internal static readonly Action<object> s_sentinel = CompletionSentinel;
- private static void CompletionSentinel(object _) // named method to aid debugging
- {
- throw new InvalidOperationException("The sentinel delegate should never be invoked.");
- }
- }
- public class AutoResetUniTaskCompletionSource : IUniTaskSource, ITaskPoolNode<AutoResetUniTaskCompletionSource>, IPromise
- {
- static TaskPool<AutoResetUniTaskCompletionSource> pool;
- AutoResetUniTaskCompletionSource nextNode;
- public ref AutoResetUniTaskCompletionSource NextNode => ref nextNode;
- static AutoResetUniTaskCompletionSource()
- {
- TaskPool.RegisterSizeGetter(typeof(AutoResetUniTaskCompletionSource), () => pool.Size);
- }
- UniTaskCompletionSourceCore<AsyncUnit> core;
- AutoResetUniTaskCompletionSource()
- {
- }
- [DebuggerHidden]
- public static AutoResetUniTaskCompletionSource Create()
- {
- if (!pool.TryPop(out var result))
- {
- result = new AutoResetUniTaskCompletionSource();
- }
- TaskTracker.TrackActiveTask(result, 2);
- return result;
- }
- [DebuggerHidden]
- public static AutoResetUniTaskCompletionSource CreateFromCanceled(CancellationToken cancellationToken, out short token)
- {
- var source = Create();
- source.TrySetCanceled(cancellationToken);
- token = source.core.Version;
- return source;
- }
- [DebuggerHidden]
- public static AutoResetUniTaskCompletionSource CreateFromException(Exception exception, out short token)
- {
- var source = Create();
- source.TrySetException(exception);
- token = source.core.Version;
- return source;
- }
- [DebuggerHidden]
- public static AutoResetUniTaskCompletionSource CreateCompleted(out short token)
- {
- var source = Create();
- source.TrySetResult();
- token = source.core.Version;
- return source;
- }
- public UniTask Task
- {
- [DebuggerHidden]
- get
- {
- return new UniTask(this, core.Version);
- }
- }
- [DebuggerHidden]
- public bool TrySetResult()
- {
- return core.TrySetResult(AsyncUnit.Default);
- }
- [DebuggerHidden]
- public bool TrySetCanceled(CancellationToken cancellationToken = default)
- {
- return core.TrySetCanceled(cancellationToken);
- }
- [DebuggerHidden]
- public bool TrySetException(Exception exception)
- {
- return core.TrySetException(exception);
- }
- [DebuggerHidden]
- public void GetResult(short token)
- {
- try
- {
- core.GetResult(token);
- }
- finally
- {
- TryReturn();
- }
- }
- [DebuggerHidden]
- public UniTaskStatus GetStatus(short token)
- {
- return core.GetStatus(token);
- }
- [DebuggerHidden]
- public UniTaskStatus UnsafeGetStatus()
- {
- return core.UnsafeGetStatus();
- }
- [DebuggerHidden]
- public void OnCompleted(Action<object> continuation, object state, short token)
- {
- core.OnCompleted(continuation, state, token);
- }
- [DebuggerHidden]
- bool TryReturn()
- {
- TaskTracker.RemoveTracking(this);
- core.Reset();
- return pool.TryPush(this);
- }
- }
- public class AutoResetUniTaskCompletionSource<T> : IUniTaskSource<T>, ITaskPoolNode<AutoResetUniTaskCompletionSource<T>>, IPromise<T>
- {
- static TaskPool<AutoResetUniTaskCompletionSource<T>> pool;
- AutoResetUniTaskCompletionSource<T> nextNode;
- public ref AutoResetUniTaskCompletionSource<T> NextNode => ref nextNode;
- static AutoResetUniTaskCompletionSource()
- {
- TaskPool.RegisterSizeGetter(typeof(AutoResetUniTaskCompletionSource<T>), () => pool.Size);
- }
- UniTaskCompletionSourceCore<T> core;
- AutoResetUniTaskCompletionSource()
- {
- }
- [DebuggerHidden]
- public static AutoResetUniTaskCompletionSource<T> Create()
- {
- if (!pool.TryPop(out var result))
- {
- result = new AutoResetUniTaskCompletionSource<T>();
- }
- TaskTracker.TrackActiveTask(result, 2);
- return result;
- }
- [DebuggerHidden]
- public static AutoResetUniTaskCompletionSource<T> CreateFromCanceled(CancellationToken cancellationToken, out short token)
- {
- var source = Create();
- source.TrySetCanceled(cancellationToken);
- token = source.core.Version;
- return source;
- }
- [DebuggerHidden]
- public static AutoResetUniTaskCompletionSource<T> CreateFromException(Exception exception, out short token)
- {
- var source = Create();
- source.TrySetException(exception);
- token = source.core.Version;
- return source;
- }
- [DebuggerHidden]
- public static AutoResetUniTaskCompletionSource<T> CreateFromResult(T result, out short token)
- {
- var source = Create();
- source.TrySetResult(result);
- token = source.core.Version;
- return source;
- }
- public UniTask<T> Task
- {
- [DebuggerHidden]
- get
- {
- return new UniTask<T>(this, core.Version);
- }
- }
- [DebuggerHidden]
- public bool TrySetResult(T result)
- {
- return core.TrySetResult(result);
- }
- [DebuggerHidden]
- public bool TrySetCanceled(CancellationToken cancellationToken = default)
- {
- return core.TrySetCanceled(cancellationToken);
- }
- [DebuggerHidden]
- public bool TrySetException(Exception exception)
- {
- return core.TrySetException(exception);
- }
- [DebuggerHidden]
- public T GetResult(short token)
- {
- try
- {
- return core.GetResult(token);
- }
- finally
- {
- TryReturn();
- }
- }
- [DebuggerHidden]
- void IUniTaskSource.GetResult(short token)
- {
- GetResult(token);
- }
- [DebuggerHidden]
- public UniTaskStatus GetStatus(short token)
- {
- return core.GetStatus(token);
- }
- [DebuggerHidden]
- public UniTaskStatus UnsafeGetStatus()
- {
- return core.UnsafeGetStatus();
- }
- [DebuggerHidden]
- public void OnCompleted(Action<object> continuation, object state, short token)
- {
- core.OnCompleted(continuation, state, token);
- }
- [DebuggerHidden]
- bool TryReturn()
- {
- TaskTracker.RemoveTracking(this);
- core.Reset();
- return pool.TryPush(this);
- }
- }
- public class UniTaskCompletionSource : IUniTaskSource, IPromise
- {
- CancellationToken cancellationToken;
- ExceptionHolder exception;
- object gate;
- Action<object> singleContinuation;
- object singleState;
- List<(Action<object>, object)> secondaryContinuationList;
- int intStatus; // UniTaskStatus
- bool handled = false;
- public UniTaskCompletionSource()
- {
- TaskTracker.TrackActiveTask(this, 2);
- }
- [DebuggerHidden]
- internal void MarkHandled()
- {
- if (!handled)
- {
- handled = true;
- TaskTracker.RemoveTracking(this);
- }
- }
- public UniTask Task
- {
- [DebuggerHidden]
- get
- {
- return new UniTask(this, 0);
- }
- }
- [DebuggerHidden]
- public bool TrySetResult()
- {
- return TrySignalCompletion(UniTaskStatus.Succeeded);
- }
- [DebuggerHidden]
- public bool TrySetCanceled(CancellationToken cancellationToken = default)
- {
- if (UnsafeGetStatus() != UniTaskStatus.Pending) return false;
- this.cancellationToken = cancellationToken;
- return TrySignalCompletion(UniTaskStatus.Canceled);
- }
- [DebuggerHidden]
- public bool TrySetException(Exception exception)
- {
- if (exception is OperationCanceledException oce)
- {
- return TrySetCanceled(oce.CancellationToken);
- }
- if (UnsafeGetStatus() != UniTaskStatus.Pending) return false;
- this.exception = new ExceptionHolder(ExceptionDispatchInfo.Capture(exception));
- return TrySignalCompletion(UniTaskStatus.Faulted);
- }
- [DebuggerHidden]
- public void GetResult(short token)
- {
- MarkHandled();
- var status = (UniTaskStatus)intStatus;
- switch (status)
- {
- case UniTaskStatus.Succeeded:
- return;
- case UniTaskStatus.Faulted:
- exception.GetException().Throw();
- return;
- case UniTaskStatus.Canceled:
- throw new OperationCanceledException(cancellationToken);
- default:
- case UniTaskStatus.Pending:
- throw new InvalidOperationException("not yet completed.");
- }
- }
- [DebuggerHidden]
- public UniTaskStatus GetStatus(short token)
- {
- return (UniTaskStatus)intStatus;
- }
- [DebuggerHidden]
- public UniTaskStatus UnsafeGetStatus()
- {
- return (UniTaskStatus)intStatus;
- }
- [DebuggerHidden]
- public void OnCompleted(Action<object> continuation, object state, short token)
- {
- if (gate == null)
- {
- Interlocked.CompareExchange(ref gate, new object(), null);
- }
- var lockGate = Thread.VolatileRead(ref gate);
- lock (lockGate) // wait TrySignalCompletion, after status is not pending.
- {
- if ((UniTaskStatus)intStatus != UniTaskStatus.Pending)
- {
- continuation(state);
- return;
- }
- if (singleContinuation == null)
- {
- singleContinuation = continuation;
- singleState = state;
- }
- else
- {
- if (secondaryContinuationList == null)
- {
- secondaryContinuationList = new List<(Action<object>, object)>();
- }
- secondaryContinuationList.Add((continuation, state));
- }
- }
- }
- [DebuggerHidden]
- bool TrySignalCompletion(UniTaskStatus status)
- {
- if (Interlocked.CompareExchange(ref intStatus, (int)status, (int)UniTaskStatus.Pending) == (int)UniTaskStatus.Pending)
- {
- if (gate == null)
- {
- Interlocked.CompareExchange(ref gate, new object(), null);
- }
- var lockGate = Thread.VolatileRead(ref gate);
- lock (lockGate) // wait OnCompleted.
- {
- if (singleContinuation != null)
- {
- try
- {
- singleContinuation(singleState);
- }
- catch (Exception ex)
- {
- UniTaskScheduler.PublishUnobservedTaskException(ex);
- }
- }
- if (secondaryContinuationList != null)
- {
- foreach (var (c, state) in secondaryContinuationList)
- {
- try
- {
- c(state);
- }
- catch (Exception ex)
- {
- UniTaskScheduler.PublishUnobservedTaskException(ex);
- }
- }
- }
- singleContinuation = null;
- singleState = null;
- secondaryContinuationList = null;
- }
- return true;
- }
- return false;
- }
- }
- public class UniTaskCompletionSource<T> : IUniTaskSource<T>, IPromise<T>
- {
- CancellationToken cancellationToken;
- T result;
- ExceptionHolder exception;
- object gate;
- Action<object> singleContinuation;
- object singleState;
- List<(Action<object>, object)> secondaryContinuationList;
- int intStatus; // UniTaskStatus
- bool handled = false;
- public UniTaskCompletionSource()
- {
- TaskTracker.TrackActiveTask(this, 2);
- }
- [DebuggerHidden]
- internal void MarkHandled()
- {
- if (!handled)
- {
- handled = true;
- TaskTracker.RemoveTracking(this);
- }
- }
- public UniTask<T> Task
- {
- [DebuggerHidden]
- get
- {
- return new UniTask<T>(this, 0);
- }
- }
- [DebuggerHidden]
- public bool TrySetResult(T result)
- {
- if (UnsafeGetStatus() != UniTaskStatus.Pending) return false;
- this.result = result;
- return TrySignalCompletion(UniTaskStatus.Succeeded);
- }
- [DebuggerHidden]
- public bool TrySetCanceled(CancellationToken cancellationToken = default)
- {
- if (UnsafeGetStatus() != UniTaskStatus.Pending) return false;
- this.cancellationToken = cancellationToken;
- return TrySignalCompletion(UniTaskStatus.Canceled);
- }
- [DebuggerHidden]
- public bool TrySetException(Exception exception)
- {
- if (exception is OperationCanceledException oce)
- {
- return TrySetCanceled(oce.CancellationToken);
- }
- if (UnsafeGetStatus() != UniTaskStatus.Pending) return false;
- this.exception = new ExceptionHolder(ExceptionDispatchInfo.Capture(exception));
- return TrySignalCompletion(UniTaskStatus.Faulted);
- }
- [DebuggerHidden]
- public T GetResult(short token)
- {
- MarkHandled();
- var status = (UniTaskStatus)intStatus;
- switch (status)
- {
- case UniTaskStatus.Succeeded:
- return result;
- case UniTaskStatus.Faulted:
- exception.GetException().Throw();
- return default;
- case UniTaskStatus.Canceled:
- throw new OperationCanceledException(cancellationToken);
- default:
- case UniTaskStatus.Pending:
- throw new InvalidOperationException("not yet completed.");
- }
- }
- [DebuggerHidden]
- void IUniTaskSource.GetResult(short token)
- {
- GetResult(token);
- }
- [DebuggerHidden]
- public UniTaskStatus GetStatus(short token)
- {
- return (UniTaskStatus)intStatus;
- }
- [DebuggerHidden]
- public UniTaskStatus UnsafeGetStatus()
- {
- return (UniTaskStatus)intStatus;
- }
- [DebuggerHidden]
- public void OnCompleted(Action<object> continuation, object state, short token)
- {
- if (gate == null)
- {
- Interlocked.CompareExchange(ref gate, new object(), null);
- }
- var lockGate = Thread.VolatileRead(ref gate);
- lock (lockGate) // wait TrySignalCompletion, after status is not pending.
- {
- if ((UniTaskStatus)intStatus != UniTaskStatus.Pending)
- {
- continuation(state);
- return;
- }
- if (singleContinuation == null)
- {
- singleContinuation = continuation;
- singleState = state;
- }
- else
- {
- if (secondaryContinuationList == null)
- {
- secondaryContinuationList = new List<(Action<object>, object)>();
- }
- secondaryContinuationList.Add((continuation, state));
- }
- }
- }
- [DebuggerHidden]
- bool TrySignalCompletion(UniTaskStatus status)
- {
- if (Interlocked.CompareExchange(ref intStatus, (int)status, (int)UniTaskStatus.Pending) == (int)UniTaskStatus.Pending)
- {
- if (gate == null)
- {
- Interlocked.CompareExchange(ref gate, new object(), null);
- }
- var lockGate = Thread.VolatileRead(ref gate);
- lock (lockGate) // wait OnCompleted.
- {
- if (singleContinuation != null)
- {
- try
- {
- singleContinuation(singleState);
- }
- catch (Exception ex)
- {
- UniTaskScheduler.PublishUnobservedTaskException(ex);
- }
- }
- if (secondaryContinuationList != null)
- {
- foreach (var (c, state) in secondaryContinuationList)
- {
- try
- {
- c(state);
- }
- catch (Exception ex)
- {
- UniTaskScheduler.PublishUnobservedTaskException(ex);
- }
- }
- }
- singleContinuation = null;
- singleState = null;
- secondaryContinuationList = null;
- }
- return true;
- }
- return false;
- }
- }
- }
|