Code:
/ WCF / WCF / 3.5.30729.1 / untmp / Orcas / SP / ndp / cdf / src / WCF / ServiceModel / System / ServiceModel / Channels / ConnectionAcceptor.cs / 1 / ConnectionAcceptor.cs
//------------------------------------------------------------
// Copyright (c) Microsoft Corporation. All rights reserved.
//-----------------------------------------------------------
namespace System.ServiceModel.Channels
{
using System.Diagnostics;
using System.ServiceModel;
using System.ServiceModel.Diagnostics;
using System.ServiceModel.Dispatcher;
using System.Threading;
delegate void ConnectionAvailableCallback(IConnection connection, ItemDequeuedCallback connectionDequeuedCallback);
delegate void ErrorCallback(Exception exception);
class ConnectionAcceptor : IDisposable
{
int maxAccepts;
int maxPendingConnections;
int connections;
int pendingAccepts;
IConnectionListener listener;
AsyncCallback acceptCompletedCallback;
WaitCallback scheduleAcceptCallback;
ItemDequeuedCallback onConnectionDequeued;
bool isDisposed;
ConnectionAvailableCallback callback;
ErrorCallback errorCallback;
public ConnectionAcceptor(IConnectionListener listener, int maxAccepts, int maxPendingConnections,
ConnectionAvailableCallback callback)
: this(listener, maxAccepts, maxPendingConnections, callback, null)
{
// empty
}
public ConnectionAcceptor(IConnectionListener listener, int maxAccepts, int maxPendingConnections,
ConnectionAvailableCallback callback, ErrorCallback errorCallback)
{
if (maxAccepts <= 0)
{
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentOutOfRangeException("maxAccepts", maxAccepts,
SR.GetString(SR.ValueMustBePositive)));
}
DiagnosticUtility.DebugAssert(maxPendingConnections > 0, "maxPendingConnections must be positive");
this.listener = listener;
this.maxAccepts = maxAccepts;
this.maxPendingConnections = maxPendingConnections;
this.callback = callback;
this.errorCallback = errorCallback;
this.onConnectionDequeued = new ItemDequeuedCallback(OnConnectionDequeued);
this.acceptCompletedCallback = DiagnosticUtility.ThunkAsyncCallback(new AsyncCallback(AcceptCompletedCallback));
this.scheduleAcceptCallback = new WaitCallback(ScheduleAcceptCallback);
}
bool IsAcceptNecessary
{
get
{
return (pendingAccepts < maxAccepts)
&& ((connections + pendingAccepts) < maxPendingConnections)
&& !isDisposed;
}
}
public int ConnectionCount
{
get { return connections; }
}
object ThisLock
{
get { return this; }
}
void AcceptIfNecessary(bool startAccepting)
{
if (IsAcceptNecessary)
{
lock (ThisLock)
{
while (IsAcceptNecessary)
{
IAsyncResult result = null;
Exception unexpectedException = null;
try
{
result = listener.BeginAccept(acceptCompletedCallback, null);
}
catch (CommunicationException exception)
{
if (DiagnosticUtility.ShouldTraceInformation)
{
DiagnosticUtility.ExceptionUtility.TraceHandledException(exception, TraceEventType.Information);
}
}
catch (Exception exception)
{
if (DiagnosticUtility.IsFatal(exception))
{
throw;
}
if (startAccepting)
{
// Since we're under a call to StartAccepting(), just throw the exception up the stack.
throw;
}
if ((errorCallback == null) && !ExceptionHandler.HandleTransportExceptionHelper(exception))
{
throw;
}
unexpectedException = exception;
}
if ((unexpectedException != null) && (errorCallback != null))
{
errorCallback(unexpectedException);
}
if (result != null)
{
// don't block our accept processing loop
if (result.CompletedSynchronously)
{
IOThreadScheduler.ScheduleCallback(scheduleAcceptCallback, result);
}
pendingAccepts++;
}
}
}
}
}
void AcceptCompletedCallback(IAsyncResult result)
{
if (result.CompletedSynchronously)
{
return;
}
HandleCompletedAccept(result);
}
public void Dispose()
{
lock (ThisLock)
{
if (!isDisposed)
{
isDisposed = true;
listener.Dispose();
}
}
}
void HandleCompletedAccept(IAsyncResult result)
{
IConnection connection = null;
lock (ThisLock)
{
bool success = false;
Exception unexpectedException = null;
try
{
if (!isDisposed)
{
connection = listener.EndAccept(result);
if (connection != null)
{
if (DiagnosticUtility.ShouldTraceWarning)
{
if (connections + 1 >= maxPendingConnections)
{
TraceUtility.TraceEvent(TraceEventType.Warning,
TraceCode.MaxPendingConnectionsReached,
new StringTraceRecord("MaxPendingConnections", maxPendingConnections.ToString(System.Globalization.CultureInfo.InvariantCulture)),
this,
null);
}
}
// This is incremented after the Trace just in case the Trace throws.
connections++;
}
}
success = true;
}
catch (CommunicationException exception)
{
if (DiagnosticUtility.ShouldTraceInformation)
{
DiagnosticUtility.ExceptionUtility.TraceHandledException(exception, TraceEventType.Information);
}
}
catch (Exception exception)
{
if (DiagnosticUtility.IsFatal(exception))
{
throw;
}
if ((errorCallback == null) && !ExceptionHandler.HandleTransportExceptionHelper(exception))
{
throw;
}
unexpectedException = exception;
}
finally
{
if (!success)
{
connection = null;
}
pendingAccepts--;
}
if ((unexpectedException != null) && (errorCallback != null))
{
errorCallback(unexpectedException);
}
}
AcceptIfNecessary(false);
if (connection != null)
{
callback(connection, onConnectionDequeued);
}
}
void OnConnectionDequeued()
{
lock (ThisLock)
{
connections--;
}
AcceptIfNecessary(false);
}
void ScheduleAcceptCallback(object state)
{
HandleCompletedAccept((IAsyncResult)state);
}
public void StartAccepting()
{
listener.Listen();
AcceptIfNecessary(true);
}
}
}
// File provided for Reference Use Only by Microsoft Corporation (c) 2007.
// Copyright (c) Microsoft Corporation. All rights reserved.
Link Menu

This book is available now!
Buy at Amazon US or
Buy at Amazon UK
- XmlSchemaException.cs
- FamilyMap.cs
- QueryReaderSettings.cs
- InteropAutomationProvider.cs
- SafeLibraryHandle.cs
- Boolean.cs
- FixedSOMTable.cs
- WebBrowserDocumentCompletedEventHandler.cs
- FrameAutomationPeer.cs
- DataRowComparer.cs
- AppDomain.cs
- XmlResolver.cs
- MultipleViewPatternIdentifiers.cs
- HttpListenerPrefixCollection.cs
- FtpCachePolicyElement.cs
- ToolStripComboBox.cs
- SetIterators.cs
- OleDbConnectionPoolGroupProviderInfo.cs
- LinkTarget.cs
- VersionValidator.cs
- XamlFilter.cs
- BuildResultCache.cs
- PersonalizationState.cs
- ADRoleFactoryConfiguration.cs
- MDIWindowDialog.cs
- HashCryptoHandle.cs
- Control.cs
- Rules.cs
- XmlILConstructAnalyzer.cs
- MailHeaderInfo.cs
- SizeValueSerializer.cs
- DesignTimeResourceProviderFactoryAttribute.cs
- SspiWrapper.cs
- DetailsViewCommandEventArgs.cs
- AssociationType.cs
- CompiledQuery.cs
- BooleanToVisibilityConverter.cs
- SamlAuthenticationClaimResource.cs
- ResourceContainer.cs
- VisualTarget.cs
- GetWinFXPath.cs
- TypedRowGenerator.cs
- DataPagerField.cs
- DrawingContextWalker.cs
- BitSet.cs
- SwitchLevelAttribute.cs
- RoleService.cs
- Drawing.cs
- HasRunnableWorkflowEvent.cs
- CommandID.cs
- HMACRIPEMD160.cs
- XmlSiteMapProvider.cs
- ManagedCodeMarkers.cs
- DragEventArgs.cs
- QueryExpression.cs
- CompareValidator.cs
- UIElement.cs
- ThaiBuddhistCalendar.cs
- ActiveXSite.cs
- TraceSection.cs
- CustomBindingElementCollection.cs
- XmlUtf8RawTextWriter.cs
- ConfigurationManagerHelper.cs
- SHA256.cs
- ProfileModule.cs
- ToolStripSplitStackLayout.cs
- XmlDocument.cs
- OledbConnectionStringbuilder.cs
- CellParagraph.cs
- MenuItemBindingCollection.cs
- ArrangedElement.cs
- XsdDuration.cs
- FtpRequestCacheValidator.cs
- EntityDataSourceWrapperPropertyDescriptor.cs
- ApplyImportsAction.cs
- MemberDescriptor.cs
- ExceptionHelpers.cs
- Section.cs
- ClientTargetSection.cs
- TextBoxAutoCompleteSourceConverter.cs
- GeneralTransform3D.cs
- InternalDuplexChannelListener.cs
- DataGridHelper.cs
- wmiprovider.cs
- MetadataItemSerializer.cs
- SBCSCodePageEncoding.cs
- CheckBoxList.cs
- GeometryDrawing.cs
- HttpHeaderCollection.cs
- PlainXmlSerializer.cs
- DocumentOrderComparer.cs
- AssemblyBuilder.cs
- EntityDataSourceValidationException.cs
- SQLDateTime.cs
- DPCustomTypeDescriptor.cs
- MessageDispatch.cs
- TCPClient.cs
- XpsPartBase.cs
- JournalEntryStack.cs
- KnowledgeBase.cs