e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
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; }
|
|
}
|
|
}
|
|
}
|
|
}
|