You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			393 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			393 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //----------------------------------------------------------------------------- | ||
|  | // Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | //----------------------------------------------------------------------------- | ||
|  | 
 | ||
|  | namespace System.ServiceModel.Dispatcher | ||
|  | { | ||
|  |     using System.Diagnostics; | ||
|  |     using System.Runtime; | ||
|  |     using System.ServiceModel; | ||
|  |     using System.ServiceModel.Channels; | ||
|  |     using System.Threading; | ||
|  |     using System.ServiceModel.Diagnostics.Application; | ||
|  | 
 | ||
|  |     interface IInstanceContextManager | ||
|  |     { | ||
|  |         void Abort(); | ||
|  |         void Add(InstanceContext instanceContext); | ||
|  |         IAsyncResult BeginClose(TimeSpan timeout, AsyncCallback callback, object state); | ||
|  |         IAsyncResult BeginCloseInput(TimeSpan timeout, AsyncCallback callback, object state); | ||
|  |         void Close(TimeSpan timeout); | ||
|  |         void CloseInput(TimeSpan timeout); | ||
|  |         void EndClose(IAsyncResult result); | ||
|  |         void EndCloseInput(IAsyncResult result); | ||
|  |         bool Remove(InstanceContext instanceContext); | ||
|  |         InstanceContext[] ToArray(); | ||
|  |     } | ||
|  | 
 | ||
|  |     class InstanceContextManager : LifetimeManager, IInstanceContextManager | ||
|  |     { | ||
|  |         int firstFreeIndex; | ||
|  |         Item[] items; | ||
|  | 
 | ||
|  |         public InstanceContextManager(object mutex) | ||
|  |             : base(mutex) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         public void Add(InstanceContext instanceContext) | ||
|  |         { | ||
|  |             bool added = false; | ||
|  | 
 | ||
|  |             lock (this.ThisLock) | ||
|  |             { | ||
|  |                 if (this.State == LifetimeState.Opened) | ||
|  |                 { | ||
|  |                     if (instanceContext.InstanceContextManagerIndex != 0) | ||
|  |                         return; | ||
|  |                     if (this.firstFreeIndex == 0) | ||
|  |                         this.GrowItems(); | ||
|  |                     this.AddItem(instanceContext); | ||
|  |                     base.IncrementBusyCountWithoutLock(); | ||
|  |                     added = true; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             if (!added) | ||
|  |             { | ||
|  |                 instanceContext.Abort(); | ||
|  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ObjectDisposedException(this.GetType().ToString())); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void AddItem(InstanceContext instanceContext) | ||
|  |         { | ||
|  |             int index = this.firstFreeIndex; | ||
|  |             this.firstFreeIndex = this.items[index].nextFreeIndex; | ||
|  |             this.items[index].instanceContext = instanceContext; | ||
|  |             instanceContext.InstanceContextManagerIndex = index; | ||
|  |         } | ||
|  | 
 | ||
|  |         public IAsyncResult BeginCloseInput(TimeSpan timeout, AsyncCallback callback, object state) | ||
|  |         { | ||
|  |             return new CloseInputAsyncResult(timeout, callback, state, this.ToArray()); | ||
|  |         } | ||
|  | 
 | ||
|  |         void CloseInitiate(TimeSpan timeout) | ||
|  |         { | ||
|  |             TimeoutHelper timeoutHelper = new TimeoutHelper(timeout); | ||
|  |             InstanceContext[] instances = this.ToArray(); | ||
|  |             for (int index = 0; index < instances.Length; index++) | ||
|  |             { | ||
|  |                 InstanceContext instance = instances[index]; | ||
|  |                 try | ||
|  |                 { | ||
|  |                     if (instance.State == CommunicationState.Opened) | ||
|  |                     { | ||
|  |                         IAsyncResult result = instance.BeginClose(timeoutHelper.RemainingTime(), Fx.ThunkCallback(new AsyncCallback(CloseInstanceContextCallback)), instance); | ||
|  |                         if (!result.CompletedSynchronously) | ||
|  |                             continue; | ||
|  |                         instance.EndClose(result); | ||
|  |                     } | ||
|  |                     else | ||
|  |                     { | ||
|  |                         instance.Abort(); | ||
|  |                     } | ||
|  |                 } | ||
|  |                 catch (ObjectDisposedException e) | ||
|  |                 { | ||
|  |                     DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |                 } | ||
|  |                 catch (InvalidOperationException e) | ||
|  |                 { | ||
|  |                     DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |                 } | ||
|  |                 catch (CommunicationException e) | ||
|  |                 { | ||
|  |                     DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |                 } | ||
|  |                 catch (TimeoutException e) | ||
|  |                 { | ||
|  |                     if (TD.CloseTimeoutIsEnabled()) | ||
|  |                     { | ||
|  |                         TD.CloseTimeout(e.Message); | ||
|  |                     } | ||
|  |                     DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         public void CloseInput(TimeSpan timeout) | ||
|  |         { | ||
|  |             TimeoutHelper timeoutHelper = new TimeoutHelper(timeout); | ||
|  |             InstanceContext[] instances = this.ToArray(); | ||
|  |             for (int index = 0; index < instances.Length; index++) | ||
|  |                 instances[index].CloseInput(timeoutHelper.RemainingTime()); | ||
|  |         } | ||
|  | 
 | ||
|  |         static void CloseInstanceContextCallback(IAsyncResult result) | ||
|  |         { | ||
|  |             if (result.CompletedSynchronously) | ||
|  |                 return; | ||
|  |             InstanceContext instanceContext = (InstanceContext)result.AsyncState; | ||
|  |             try | ||
|  |             { | ||
|  |                 instanceContext.EndClose(result); | ||
|  |             } | ||
|  |             catch (ObjectDisposedException e) | ||
|  |             { | ||
|  |                 DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |              } | ||
|  |              catch (InvalidOperationException e) | ||
|  |              { | ||
|  |                  DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |              } | ||
|  |              catch (CommunicationException e) | ||
|  |              { | ||
|  |                  DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |              } | ||
|  |              catch (TimeoutException e) | ||
|  |              { | ||
|  |                  if (TD.CloseTimeoutIsEnabled()) | ||
|  |                  { | ||
|  |                      TD.CloseTimeout(e.Message); | ||
|  |                  } | ||
|  |                  DiagnosticUtility.TraceHandledException(e, TraceEventType.Information); | ||
|  |              } | ||
|  |         } | ||
|  | 
 | ||
|  |         public void EndCloseInput(IAsyncResult result) | ||
|  |         { | ||
|  |             CloseInputAsyncResult.End(result); | ||
|  |         } | ||
|  | 
 | ||
|  |         void GrowItems() | ||
|  |         { | ||
|  |             Item[] existingItems = this.items; | ||
|  |             if (existingItems != null) | ||
|  |             { | ||
|  |                 this.InitItems(existingItems.Length * 2); | ||
|  |                 for (int i = 1; i < existingItems.Length; i++) | ||
|  |                     this.AddItem(existingItems[i].instanceContext); | ||
|  |             } | ||
|  |             else | ||
|  |             { | ||
|  |                 this.InitItems(4); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void InitItems(int count) | ||
|  |         { | ||
|  |             this.items = new Item[count]; | ||
|  |             for (int i = count - 2; i > 0; i--) | ||
|  |             { | ||
|  |                 this.items[i].nextFreeIndex = i + 1; | ||
|  |             } | ||
|  |             this.firstFreeIndex = 1; | ||
|  |         } | ||
|  | 
 | ||
|  |         protected override void OnAbort() | ||
|  |         { | ||
|  |             InstanceContext[] instances = this.ToArray(); | ||
|  |             for (int index = 0; index < instances.Length; index++) | ||
|  |             { | ||
|  |                 instances[index].Abort(); | ||
|  |             } | ||
|  | 
 | ||
|  |             base.OnAbort(); | ||
|  |         } | ||
|  | 
 | ||
|  |         protected override IAsyncResult OnBeginClose(TimeSpan timeout, AsyncCallback callback, object state) | ||
|  |         { | ||
|  |             TimeoutHelper timeoutHelper = new TimeoutHelper(timeout); | ||
|  |             CloseInitiate(timeoutHelper.RemainingTime()); | ||
|  |             return base.OnBeginClose(timeoutHelper.RemainingTime(), callback, state); | ||
|  |         } | ||
|  | 
 | ||
|  |         protected override void OnClose(TimeSpan timeout) | ||
|  |         { | ||
|  |             TimeoutHelper timeoutHelper = new TimeoutHelper(timeout); | ||
|  |             CloseInitiate(timeoutHelper.RemainingTime()); | ||
|  |             base.OnClose(timeoutHelper.RemainingTime()); | ||
|  |         } | ||
|  | 
 | ||
|  |         protected override void OnEndClose(IAsyncResult result) | ||
|  |         { | ||
|  |             base.OnEndClose(result); | ||
|  |         } | ||
|  | 
 | ||
|  |         public bool Remove(InstanceContext instanceContext) | ||
|  |         { | ||
|  |             if (instanceContext == null) | ||
|  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("instanceContext")); | ||
|  | 
 | ||
|  |             lock (this.ThisLock) | ||
|  |             { | ||
|  |                 int index = instanceContext.InstanceContextManagerIndex; | ||
|  |                 if (index == 0) | ||
|  |                     return false; | ||
|  |                 instanceContext.InstanceContextManagerIndex = 0; | ||
|  |                 this.items[index].nextFreeIndex = this.firstFreeIndex; | ||
|  |                 this.items[index].instanceContext = null; | ||
|  |                 this.firstFreeIndex = index; | ||
|  |             } | ||
|  | 
 | ||
|  |             base.DecrementBusyCount(); | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         public InstanceContext[] ToArray() | ||
|  |         { | ||
|  |             if (this.items == null) | ||
|  |             { | ||
|  |                 return EmptyArray<InstanceContext>.Instance; | ||
|  |             } | ||
|  | 
 | ||
|  |             lock (this.ThisLock) | ||
|  |             { | ||
|  |                 int count = 0; | ||
|  |                 for (int i = 1; i < this.items.Length; i++) | ||
|  |                     if (this.items[i].instanceContext != null) | ||
|  |                         count++; | ||
|  | 
 | ||
|  |                 if (count == 0) | ||
|  |                     return EmptyArray<InstanceContext>.Instance; | ||
|  | 
 | ||
|  |                 InstanceContext[] array = new InstanceContext[count]; | ||
|  |                 count = 0; | ||
|  |                 for (int i = 1; i < this.items.Length; i++) | ||
|  |                 { | ||
|  |                     InstanceContext instanceContext = this.items[i].instanceContext; | ||
|  |                     if (instanceContext != null) | ||
|  |                     { | ||
|  |                         array[count++] = instanceContext; | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 return array; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         struct Item | ||
|  |         { | ||
|  |             public int nextFreeIndex; | ||
|  |             public InstanceContext instanceContext; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     class CloseInputAsyncResult : AsyncResult | ||
|  |     { | ||
|  |         bool completedSynchronously; | ||
|  |         Exception exception; | ||
|  |         static AsyncCallback nestedCallback = Fx.ThunkCallback(new AsyncCallback(Callback)); | ||
|  |         int count; | ||
|  |         TimeoutHelper timeoutHelper; | ||
|  | 
 | ||
|  |         public CloseInputAsyncResult(TimeSpan timeout, AsyncCallback otherCallback, object state, InstanceContext[] instances) | ||
|  |             : base(otherCallback, state) | ||
|  |         { | ||
|  |             this.timeoutHelper = new TimeoutHelper(timeout); | ||
|  |             completedSynchronously = true; | ||
|  | 
 | ||
|  |             count = instances.Length; | ||
|  |             if (count == 0) | ||
|  |             { | ||
|  |                 Complete(true); | ||
|  |                 return; | ||
|  |             } | ||
|  | 
 | ||
|  |             for (int index = 0; index < instances.Length; index++) | ||
|  |             { | ||
|  |                 CallbackState callbackState = new CallbackState(this, instances[index]); | ||
|  |                 IAsyncResult result; | ||
|  |                 try | ||
|  |                 { | ||
|  |                     result = instances[index].BeginCloseInput(this.timeoutHelper.RemainingTime(), nestedCallback, callbackState); | ||
|  |                 } | ||
|  | #pragma warning suppress 56500 // covered by FxCOP | ||
|  |                 catch (Exception e) | ||
|  |                 { | ||
|  |                     if (Fx.IsFatal(e)) | ||
|  |                     { | ||
|  |                         throw; | ||
|  |                     } | ||
|  |                     Decrement(true, e); | ||
|  |                     continue; | ||
|  |                 } | ||
|  |                 if (result.CompletedSynchronously) | ||
|  |                 { | ||
|  |                     instances[index].EndCloseInput(result); | ||
|  |                     Decrement(true); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         static void Callback(IAsyncResult result) | ||
|  |         { | ||
|  |             if (result.CompletedSynchronously) | ||
|  |                 return; | ||
|  |             CallbackState callbackState = (CallbackState)result.AsyncState; | ||
|  |             try | ||
|  |             { | ||
|  |                 callbackState.Instance.EndCloseInput(result); | ||
|  |                 callbackState.Result.Decrement(false); | ||
|  |             } | ||
|  | #pragma warning suppress 56500 // covered by FxCOP | ||
|  |             catch (Exception e) | ||
|  |             { | ||
|  |                 if (Fx.IsFatal(e)) | ||
|  |                 { | ||
|  |                     throw; | ||
|  |                 } | ||
|  |                 callbackState.Result.Decrement(false, e); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void Decrement(bool completedSynchronously) | ||
|  |         { | ||
|  |             if (completedSynchronously == false) | ||
|  |                 this.completedSynchronously = false; | ||
|  |             if (Interlocked.Decrement(ref count) == 0) | ||
|  |             { | ||
|  |                 if (this.exception != null) | ||
|  |                     Complete(this.completedSynchronously, this.exception); | ||
|  |                 else | ||
|  |                     Complete(this.completedSynchronously); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         void Decrement(bool completedSynchronously, Exception exception) | ||
|  |         { | ||
|  |             this.exception = exception; | ||
|  |             this.Decrement(completedSynchronously); | ||
|  |         } | ||
|  | 
 | ||
|  |         public static void End(IAsyncResult result) | ||
|  |         { | ||
|  |             AsyncResult.End<CloseInputAsyncResult>(result); | ||
|  |         } | ||
|  | 
 | ||
|  |         class CallbackState | ||
|  |         { | ||
|  |             InstanceContext instance; | ||
|  |             CloseInputAsyncResult result; | ||
|  | 
 | ||
|  |             public CallbackState(CloseInputAsyncResult result, InstanceContext instance) | ||
|  |             { | ||
|  |                 this.result = result; | ||
|  |                 this.instance = instance; | ||
|  |             } | ||
|  | 
 | ||
|  |             public InstanceContext Instance | ||
|  |             { | ||
|  |                 get { return instance; } | ||
|  |             } | ||
|  | 
 | ||
|  |             public CloseInputAsyncResult Result | ||
|  |             { | ||
|  |                 get { return result; } | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | } |