You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			186 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			186 lines
		
	
	
		
			8.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //------------------------------------------------------------------------------
 | |
| // <copyright file="AspNetSynchronizationContext.cs" company="Microsoft">
 | |
| //     Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| // </copyright>                                                                
 | |
| //------------------------------------------------------------------------------
 | |
| 
 | |
| namespace System.Web {
 | |
|     using System;
 | |
|     using System.Collections.Generic;
 | |
|     using System.Diagnostics.CodeAnalysis;
 | |
|     using System.Runtime.ExceptionServices;
 | |
|     using System.Threading;
 | |
|     using System.Threading.Tasks;
 | |
|     using System.Web.Util;
 | |
| 
 | |
|     internal sealed class AspNetSynchronizationContext : AspNetSynchronizationContextBase {
 | |
| 
 | |
|         // we move all of the state to a separate field since our CreateCopy() method needs shallow copy semantics
 | |
|         private readonly State _state;
 | |
| 
 | |
|         internal AspNetSynchronizationContext(ISyncContext syncContext)
 | |
|             : this(new State(new SynchronizationHelper(syncContext))) {
 | |
|         }
 | |
| 
 | |
|         private AspNetSynchronizationContext(State state) {
 | |
|             _state = state;
 | |
|         }
 | |
| 
 | |
|         internal override bool AllowAsyncDuringSyncStages {
 | |
|             get {
 | |
|                 return _state.AllowAsyncDuringSyncStages;
 | |
|             }
 | |
|             set {
 | |
|                 _state.AllowAsyncDuringSyncStages = value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // We can't ever truly disable the AspNetSynchronizationContext, as the user and runtime can kick off asynchronous
 | |
|         // operations whether we wanted them to or not. But this property can be used as a flag by Page and other types
 | |
|         // to signal that asynchronous operations are not currently valid, so at least ASP.NET can avoid kicking them
 | |
|         // off and can bubble an appropriate exception back to the developer.
 | |
|         internal override bool Enabled {
 | |
|             get { return _state.Enabled; }
 | |
|         }
 | |
| 
 | |
|         internal override ExceptionDispatchInfo ExceptionDispatchInfo {
 | |
|             get { return _state.Helper.Error; }
 | |
|         }
 | |
| 
 | |
|         internal override int PendingOperationsCount {
 | |
|             get { return _state.Helper.PendingCount; }
 | |
|         }
 | |
| 
 | |
|         internal override void AllowVoidAsyncOperations() {
 | |
|             _state.AllowVoidAsyncOperations = true;
 | |
|         }
 | |
| 
 | |
|         [SuppressMessage("Microsoft.Globalization", "CA1305:SpecifyIFormatProvider", Justification = "Used only during debug.")]
 | |
|         internal override void AssociateWithCurrentThread() {
 | |
|             IDisposable disassociationAction = _state.Helper.EnterSynchronousControl();
 | |
| 
 | |
| #if DBG
 | |
|             IDisposable capturedDisassociationAction = disassociationAction;
 | |
|             Thread capturedThread = Thread.CurrentThread;
 | |
|             disassociationAction = new DisposableAction(() => {
 | |
|                 Debug.Assert(capturedThread == Thread.CurrentThread, String.Format("AssociateWithCurrentThread was called on thread ID '{0}', but DisassociateFromCurrentThread was called on thread ID '{1}'.", capturedThread.ManagedThreadId, Thread.CurrentThread.ManagedThreadId));
 | |
|                 capturedDisassociationAction.Dispose();
 | |
|             });
 | |
| #endif
 | |
| 
 | |
|             // Don't need to synchronize access to SyncControlDisassociationActions since only one thread can call
 | |
|             // EnterSynchronousControl() at a time.
 | |
|             _state.SyncControlDisassociationActions.Push(disassociationAction);
 | |
|         }
 | |
| 
 | |
|         internal override void ClearError() {
 | |
|             _state.Helper.Error = null;
 | |
|         }
 | |
| 
 | |
|         // Called by the BCL when it needs a SynchronizationContext that is identical to the existing context
 | |
|         // but does not have referential equality.
 | |
|         public override SynchronizationContext CreateCopy() {
 | |
|             return new AspNetSynchronizationContext(_state);
 | |
|         }
 | |
| 
 | |
|         internal override void Disable() {
 | |
|             _state.Enabled = false;
 | |
|         }
 | |
| 
 | |
|         internal override void DisassociateFromCurrentThread() {
 | |
|             // Don't need to synchronize access to SyncControlDisassociationActions since we assume that our callers are 
 | |
|             // well-behaved and won't call DisassociateFromCurrentThread() on a thread other than the one which called
 | |
|             // AssociateWithCurrentThread(), which itself serializes access.
 | |
|             Debug.Assert(_state.SyncControlDisassociationActions.Count > 0, "DisassociateFromCurrentThread() was called on a thread which hadn't previously called AssociateWithCurrentThread().");
 | |
|             IDisposable disassociationAction = _state.SyncControlDisassociationActions.Pop();
 | |
|             disassociationAction.Dispose();
 | |
|         }
 | |
| 
 | |
|         internal override void Enable() {
 | |
|             _state.Enabled = true;
 | |
|         }
 | |
| 
 | |
|         public override void OperationCompleted() {
 | |
|             Interlocked.Decrement(ref _state.VoidAsyncOutstandingOperationCount); // this line goes first since ChangeOperationCount might invoke a callback which depends on this value
 | |
|             _state.Helper.ChangeOperationCount(-1);
 | |
|         }
 | |
| 
 | |
|         public override void OperationStarted() {
 | |
|             // If the caller tries to kick off an asynchronous operation while we are not
 | |
|             // processing an async module, handler, or Page, we should prohibit the operation.
 | |
|             if (!AllowAsyncDuringSyncStages && !_state.AllowVoidAsyncOperations) {
 | |
|                 InvalidOperationException ex = new InvalidOperationException(SR.GetString(SR.Async_operation_cannot_be_started));
 | |
|                 throw ex;
 | |
|             }
 | |
| 
 | |
|             _state.Helper.ChangeOperationCount(+1);
 | |
|             Interlocked.Increment(ref _state.VoidAsyncOutstandingOperationCount);
 | |
|         }
 | |
| 
 | |
|         // Dev11 Bug 70908: Race condition involving SynchronizationContext allows ASP.NET requests to be abandoned in the pipeline
 | |
|         //  
 | |
|         // When the last completion occurs, the _pendingCount is decremented and then the _lastCompletionCallbackLock is acquired to get
 | |
|         // the _lastCompletionCallback.  If the _lastCompletionCallback is non-null, then the last completion will invoke the callback;
 | |
|         // otherwise, the caller of PendingCompletion will handle the completion.
 | |
|         internal override bool PendingCompletion(WaitCallback callback) {
 | |
|             return _state.Helper.TrySetCompletionContinuation(() => callback(null));
 | |
|         }
 | |
| 
 | |
|         public override void Post(SendOrPostCallback callback, Object state) {
 | |
|             _state.Helper.QueueAsynchronous(() => callback(state));
 | |
|         }
 | |
| 
 | |
|         // The method is used to post async func.
 | |
|         internal void PostAsync(Func<object, Task> callback, Object state) {
 | |
|             _state.Helper.QueueAsynchronousAsync(callback, state);
 | |
|         }
 | |
| 
 | |
|         internal override void ProhibitVoidAsyncOperations() {
 | |
|             _state.AllowVoidAsyncOperations = false;
 | |
| 
 | |
|             // If the caller tries to prohibit async operations while there are still some
 | |
|             // outstanding, we should treat this as an error condition. We can't throw from
 | |
|             // this method since (a) the caller generally isn't prepared for it and (b) we
 | |
|             // need to wait for the outstanding operations to finish anyway, so we instead
 | |
|             // need to mark the helper as faulted.
 | |
|             // 
 | |
|             // There is technically a race condition here: the caller isn't guaranteed to
 | |
|             // observe the error if the operation counter hits zero at just the right time.
 | |
|             // But it's actually not terrible if that happens, since the error is really
 | |
|             // just meant to be used for diagnostic purposes.
 | |
|             if (!AllowAsyncDuringSyncStages && Volatile.Read(ref _state.VoidAsyncOutstandingOperationCount) > 0) {
 | |
|                 InvalidOperationException ex = new InvalidOperationException(SR.GetString(SR.Async_operation_cannot_be_pending));
 | |
|                 _state.Helper.Error = ExceptionDispatchInfo.Capture(ex);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal override void ResetSyncCaller() {
 | |
|             // no-op
 | |
|             // this type doesn't special-case asynchronous work kicked off from a synchronous handler
 | |
|         }
 | |
| 
 | |
|         internal override void SetSyncCaller() {
 | |
|             // no-op
 | |
|             // this type doesn't special-case asynchronous work kicked off from a synchronous handler
 | |
|         }
 | |
| 
 | |
|         public override void Send(SendOrPostCallback callback, Object state) {
 | |
|             _state.Helper.QueueSynchronous(() => callback(state));
 | |
|         }
 | |
| 
 | |
|         private sealed class State {
 | |
|             internal bool AllowAsyncDuringSyncStages = AppSettings.AllowAsyncDuringSyncStages;
 | |
|             internal volatile bool AllowVoidAsyncOperations = false;
 | |
|             internal bool Enabled = true;
 | |
|             internal readonly SynchronizationHelper Helper; // handles scheduling of the asynchronous tasks
 | |
|             internal Stack<IDisposable> SyncControlDisassociationActions = new Stack<IDisposable>(capacity: 1);
 | |
|             internal int VoidAsyncOutstandingOperationCount = 0;
 | |
| 
 | |
|             internal State(SynchronizationHelper helper) {
 | |
|                 Helper = helper;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|     }
 | |
| }
 |