You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			297 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			297 lines
		
	
	
		
			9.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //-----------------------------------------------------------------------------
 | |
| // Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| //-----------------------------------------------------------------------------
 | |
| 
 | |
| namespace System.ServiceModel.Dispatcher
 | |
| {
 | |
|     using System;
 | |
|     using System.Collections.Generic;
 | |
|     using System.Runtime;
 | |
|     using System.ServiceModel;
 | |
|     using System.Threading;
 | |
| 
 | |
|     class ConcurrencyBehavior
 | |
|     {
 | |
|         ConcurrencyMode concurrencyMode;
 | |
|         bool enforceOrderedReceive;
 | |
|         bool supportsTransactedBatch;
 | |
| 
 | |
|         internal ConcurrencyBehavior(DispatchRuntime runtime)
 | |
|         {
 | |
|             this.concurrencyMode = runtime.ConcurrencyMode;
 | |
|             this.enforceOrderedReceive = runtime.EnsureOrderedDispatch;
 | |
|             this.supportsTransactedBatch = ConcurrencyBehavior.SupportsTransactedBatch(runtime.ChannelDispatcher);
 | |
|         }
 | |
| 
 | |
|         static bool SupportsTransactedBatch(ChannelDispatcher channelDispatcher)
 | |
|         {
 | |
|             return channelDispatcher.IsTransactedReceive && (channelDispatcher.MaxTransactedBatchSize > 0);
 | |
|         }
 | |
| 
 | |
|         internal bool IsConcurrent(ref MessageRpc rpc)
 | |
|         {
 | |
|             return IsConcurrent(this.concurrencyMode, this.enforceOrderedReceive, rpc.Channel.HasSession, this.supportsTransactedBatch);
 | |
|         }
 | |
| 
 | |
|         internal static bool IsConcurrent(ConcurrencyMode concurrencyMode, bool ensureOrderedDispatch, bool hasSession, bool supportsTransactedBatch)
 | |
|         {
 | |
|             if (supportsTransactedBatch)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             if (concurrencyMode != ConcurrencyMode.Single)
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             if (hasSession)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             if (ensureOrderedDispatch)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         internal static bool IsConcurrent(ChannelDispatcher runtime, bool hasSession)
 | |
|         {
 | |
|             bool isConcurrencyModeSingle = true;
 | |
| 
 | |
|             if (ConcurrencyBehavior.SupportsTransactedBatch(runtime))
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             foreach (EndpointDispatcher endpointDispatcher in runtime.Endpoints)
 | |
|             {
 | |
|                 if (endpointDispatcher.DispatchRuntime.EnsureOrderedDispatch)
 | |
|                 {
 | |
|                     return false;
 | |
|                 }
 | |
| 
 | |
|                 if (endpointDispatcher.DispatchRuntime.ConcurrencyMode != ConcurrencyMode.Single)
 | |
|                 {
 | |
|                     isConcurrencyModeSingle = false;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (!isConcurrencyModeSingle)
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             if (!hasSession)
 | |
|             {
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         internal void LockInstance(ref MessageRpc rpc)
 | |
|         {
 | |
|             if (this.concurrencyMode != ConcurrencyMode.Multiple)
 | |
|             {
 | |
|                 ConcurrencyInstanceContextFacet resource = rpc.InstanceContext.Concurrency;
 | |
|                 lock (rpc.InstanceContext.ThisLock)
 | |
|                 {
 | |
|                     if (!resource.Locked)
 | |
|                     {
 | |
|                         resource.Locked = true;
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         MessageRpcWaiter waiter = new MessageRpcWaiter(rpc.Pause());
 | |
|                         resource.EnqueueNewMessage(waiter);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (this.concurrencyMode == ConcurrencyMode.Reentrant)
 | |
|                 {
 | |
|                     rpc.OperationContext.IsServiceReentrant = true;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal void UnlockInstance(ref MessageRpc rpc)
 | |
|         {
 | |
|             if (this.concurrencyMode != ConcurrencyMode.Multiple)
 | |
|             {
 | |
|                 ConcurrencyBehavior.UnlockInstance(rpc.InstanceContext);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal static void UnlockInstanceBeforeCallout(OperationContext operationContext)
 | |
|         {
 | |
|             if (operationContext != null && operationContext.IsServiceReentrant)
 | |
|             {
 | |
|                 ConcurrencyBehavior.UnlockInstance(operationContext.InstanceContext);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         static void UnlockInstance(InstanceContext instanceContext)
 | |
|         {
 | |
|             ConcurrencyInstanceContextFacet resource = instanceContext.Concurrency;
 | |
| 
 | |
|             lock (instanceContext.ThisLock)
 | |
|             {
 | |
|                 if (resource.HasWaiters)
 | |
|                 {
 | |
|                     IWaiter nextWaiter = resource.DequeueWaiter();
 | |
|                     nextWaiter.Signal();
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     //We have no pending Callouts and no new Messages to process
 | |
|                     resource.Locked = false;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal static void LockInstanceAfterCallout(OperationContext operationContext)
 | |
|         {
 | |
|             if (operationContext != null)
 | |
|             {
 | |
|                 InstanceContext instanceContext = operationContext.InstanceContext;
 | |
|                 
 | |
|                 if (operationContext.IsServiceReentrant)
 | |
|                 {
 | |
|                     ConcurrencyInstanceContextFacet resource = instanceContext.Concurrency;
 | |
|                     ThreadWaiter waiter = null;
 | |
| 
 | |
|                     lock (instanceContext.ThisLock)
 | |
|                     {
 | |
|                         if (!resource.Locked)
 | |
|                         {
 | |
|                             resource.Locked = true;
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             waiter = new ThreadWaiter();
 | |
|                             resource.EnqueueCalloutMessage(waiter);
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     if (waiter != null)
 | |
|                     {
 | |
|                         waiter.Wait();
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal interface IWaiter
 | |
|         {
 | |
|             void Signal();
 | |
|         }
 | |
| 
 | |
|         class MessageRpcWaiter : IWaiter
 | |
|         {
 | |
|             IResumeMessageRpc resume;
 | |
| 
 | |
|             internal MessageRpcWaiter(IResumeMessageRpc resume)
 | |
|             {
 | |
|                 this.resume = resume;
 | |
|             }
 | |
| 
 | |
|             void IWaiter.Signal()
 | |
|             {
 | |
|                 try
 | |
|                 {
 | |
|                     bool alreadyResumedNoLock;
 | |
|                     this.resume.Resume(out alreadyResumedNoLock);
 | |
| 
 | |
|                     if (alreadyResumedNoLock)
 | |
|                     {
 | |
|                         Fx.Assert("ConcurrencyBehavior resumed more than once for same call.");
 | |
|                     }
 | |
|                 }
 | |
|                 catch (Exception e)
 | |
|                 {
 | |
|                     if (Fx.IsFatal(e))
 | |
|                     {
 | |
|                         throw;
 | |
|                     }
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperCallback(e);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         class ThreadWaiter : IWaiter
 | |
|         {
 | |
|             ManualResetEvent wait = new ManualResetEvent(false);
 | |
| 
 | |
|             void IWaiter.Signal()
 | |
|             {
 | |
|                 this.wait.Set();
 | |
|             }
 | |
| 
 | |
|             internal void Wait()
 | |
|             {
 | |
|                 this.wait.WaitOne();
 | |
|                 this.wait.Close();
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     internal class ConcurrencyInstanceContextFacet
 | |
|     {
 | |
|         internal bool Locked;
 | |
|         Queue<ConcurrencyBehavior.IWaiter> calloutMessageQueue;
 | |
|         Queue<ConcurrencyBehavior.IWaiter> newMessageQueue;
 | |
| 
 | |
|         internal bool HasWaiters
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 return (((this.calloutMessageQueue != null) && (this.calloutMessageQueue.Count > 0)) ||
 | |
|                         ((this.newMessageQueue != null) && (this.newMessageQueue.Count > 0)));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         ConcurrencyBehavior.IWaiter DequeueFrom(Queue<ConcurrencyBehavior.IWaiter> queue)
 | |
|         {
 | |
|             ConcurrencyBehavior.IWaiter waiter = queue.Dequeue();
 | |
| 
 | |
|             if (queue.Count == 0)
 | |
|             {
 | |
|                 queue.TrimExcess();
 | |
|             }
 | |
| 
 | |
|             return waiter;
 | |
|         }
 | |
| 
 | |
|         internal ConcurrencyBehavior.IWaiter DequeueWaiter()
 | |
|         {
 | |
|             // Finishing old work takes precedence over new work.
 | |
|             if ((this.calloutMessageQueue != null) && (this.calloutMessageQueue.Count > 0))
 | |
|             {
 | |
|                 return this.DequeueFrom(this.calloutMessageQueue);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return this.DequeueFrom(this.newMessageQueue);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal void EnqueueNewMessage(ConcurrencyBehavior.IWaiter waiter)
 | |
|         {
 | |
|             if (this.newMessageQueue == null)
 | |
|                 this.newMessageQueue = new Queue<ConcurrencyBehavior.IWaiter>();
 | |
|             this.newMessageQueue.Enqueue(waiter);
 | |
|         }
 | |
| 
 | |
|         internal void EnqueueCalloutMessage(ConcurrencyBehavior.IWaiter waiter)
 | |
|         {
 | |
|             if (this.calloutMessageQueue == null)
 | |
|                 this.calloutMessageQueue = new Queue<ConcurrencyBehavior.IWaiter>();
 | |
|             this.calloutMessageQueue.Enqueue(waiter);
 | |
|         }
 | |
|     }
 | |
| }
 |