a575963da9
Former-commit-id: da6be194a6b1221998fc28233f2503bd61dd9d14
64 lines
2.2 KiB
C#
64 lines
2.2 KiB
C#
namespace System.Web.Mvc.Async {
|
|
using System;
|
|
using System.Threading;
|
|
|
|
internal static class AsyncUtil {
|
|
|
|
public static void WaitForAsyncResultCompletion(IAsyncResult asyncResult, HttpApplication app) {
|
|
// based on HttpServerUtility.ExecuteInternal()
|
|
|
|
if (!asyncResult.IsCompleted) {
|
|
// suspend app lock while waiting, else might deadlock
|
|
bool needToRelock = false;
|
|
|
|
try {
|
|
// .NET 2.0+ will not allow a ThreadAbortException to be thrown while a
|
|
// thread is inside a finally block, so this pattern ensures that the
|
|
// value of 'needToRelock' is correct.
|
|
try { }
|
|
finally {
|
|
Monitor.Exit(app);
|
|
needToRelock = true;
|
|
}
|
|
|
|
WaitHandle waitHandle = asyncResult.AsyncWaitHandle;
|
|
|
|
if (waitHandle != null) {
|
|
waitHandle.WaitOne();
|
|
}
|
|
else {
|
|
while (!asyncResult.IsCompleted) {
|
|
Thread.Sleep(1);
|
|
}
|
|
}
|
|
}
|
|
finally {
|
|
if (needToRelock) {
|
|
Monitor.Enter(app);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public static AsyncCallback WrapCallbackForSynchronizedExecution(AsyncCallback callback, SynchronizationContext syncContext) {
|
|
if (callback == null || syncContext == null) {
|
|
return callback;
|
|
}
|
|
|
|
AsyncCallback newCallback = delegate(IAsyncResult asyncResult) {
|
|
if (asyncResult.CompletedSynchronously) {
|
|
callback(asyncResult);
|
|
}
|
|
else {
|
|
// Only take the application lock if this request completed asynchronously,
|
|
// else we might end up in a deadlock situation.
|
|
syncContext.Sync(() => callback(asyncResult));
|
|
}
|
|
};
|
|
|
|
return newCallback;
|
|
}
|
|
|
|
}
|
|
}
|