Code:
/ FXUpdate3074 / FXUpdate3074 / 1.1 / DEVDIV / depot / DevDiv / releases / whidbey / QFE / ndp / fx / src / xsp / System / Web / RequestTimeoutManager.cs / 3 / RequestTimeoutManager.cs
//------------------------------------------------------------------------------ //// Copyright (c) Microsoft Corporation. All rights reserved. // //----------------------------------------------------------------------------- /* * Request timeout manager -- implements the request timeout mechanism */ namespace System.Web { using System.Threading; using System.Collections; using System.Web.Util; internal class RequestTimeoutManager { private int _requestCount; private DoubleLinkList[] _lists; // partitioned to avoid contention private int _currentList; private int _inProgressLock; // only 1 thread can be cancelling private readonly TimeSpan _timerPeriod = new TimeSpan(0, 0, 15); // 15 second init precision private Timer _timer; internal RequestTimeoutManager() { // initialize request lists _requestCount = 0; _lists = new DoubleLinkList[13]; for (int i = 0; i < _lists.Length; i++) _lists[i] = new DoubleLinkList(); _currentList = 0; // init lock _inProgressLock = 0; // create the timer #if DBG if (!Debug.IsTagPresent("Timer") || Debug.IsTagEnabled("Timer")) #endif { _timer = new Timer(new TimerCallback(this.TimerCompletionCallback), null, _timerPeriod, _timerPeriod); } } internal void Stop() { // stop the timer if (_timer != null) { ((IDisposable)_timer).Dispose(); _timer = null; } while (_inProgressLock != 0) Thread.Sleep(100); // cancel all cancelable requests if (_requestCount > 0) CancelTimedOutRequests(DateTime.UtcNow.AddYears(1)); // } private void TimerCompletionCallback(Object state) { if (_requestCount > 0) CancelTimedOutRequests(DateTime.UtcNow); } private void CancelTimedOutRequests(DateTime now) { // only one thread can be doing it if (Interlocked.CompareExchange(ref _inProgressLock, 1, 0) != 0) return; // collect them all into a separate list with minimal locking ArrayList entries = new ArrayList(_requestCount); // size can change DoubleLinkListEnumerator en; for (int i = 0; i < _lists.Length; i++) { lock (_lists[i]) { en = _lists[i].GetEnumerator(); while (en.MoveNext()) entries.Add(en.GetDoubleLink()); en = null; } } // walk through the collected list to timeout what's needed int n = entries.Count; for (int i = 0; i < n; i++) ((RequestTimeoutEntry)entries[i]).TimeoutIfNeeded(now); // this thread is done -- unlock Interlocked.Exchange(ref _inProgressLock, 0); } internal void Add(HttpContext context) { if (context.TimeoutLink != null) { ((RequestTimeoutEntry)context.TimeoutLink).IncrementCount(); return; } // create new entry RequestTimeoutEntry entry = new RequestTimeoutEntry(context); // add it to the list int i = _currentList++; if (i >= _lists.Length) { i = 0; _currentList = 0; } entry.AddToList(_lists[i]); Interlocked.Increment(ref _requestCount); // update HttpContext context.TimeoutLink = entry; } internal void Remove(HttpContext context) { RequestTimeoutEntry entry = (RequestTimeoutEntry)context.TimeoutLink; // remove from the list if (entry != null) { if( entry.DecrementCount() == 0 ) { entry.RemoveFromList(); Interlocked.Decrement(ref _requestCount); } else { return; } } // update HttpContext context.TimeoutLink = null; } private class RequestTimeoutEntry : DoubleLink { private HttpContext _context; // the request private DoubleLinkList _list; private int _count; internal RequestTimeoutEntry(HttpContext context) { _context = context; _count = 1; } internal void AddToList(DoubleLinkList list) { lock(list) { list.InsertTail(this); _list = list; } } internal void RemoveFromList() { if (_list != null) { lock(_list) { Remove(); _list = null; } } } internal void TimeoutIfNeeded(DateTime now) { Thread thread = _context.MustTimeout(now); if (thread != null) { RemoveFromList(); thread.Abort(new HttpApplication.CancelModuleException(true)); } } internal void IncrementCount() { Interlocked.Increment( ref _count ); } internal int DecrementCount() { return Interlocked.Decrement( ref _count ); } } } } // File provided for Reference Use Only by Microsoft Corporation (c) 2007. // Copyright (c) Microsoft Corporation. All rights reserved. //------------------------------------------------------------------------------ //// Copyright (c) Microsoft Corporation. All rights reserved. // //----------------------------------------------------------------------------- /* * Request timeout manager -- implements the request timeout mechanism */ namespace System.Web { using System.Threading; using System.Collections; using System.Web.Util; internal class RequestTimeoutManager { private int _requestCount; private DoubleLinkList[] _lists; // partitioned to avoid contention private int _currentList; private int _inProgressLock; // only 1 thread can be cancelling private readonly TimeSpan _timerPeriod = new TimeSpan(0, 0, 15); // 15 second init precision private Timer _timer; internal RequestTimeoutManager() { // initialize request lists _requestCount = 0; _lists = new DoubleLinkList[13]; for (int i = 0; i < _lists.Length; i++) _lists[i] = new DoubleLinkList(); _currentList = 0; // init lock _inProgressLock = 0; // create the timer #if DBG if (!Debug.IsTagPresent("Timer") || Debug.IsTagEnabled("Timer")) #endif { _timer = new Timer(new TimerCallback(this.TimerCompletionCallback), null, _timerPeriod, _timerPeriod); } } internal void Stop() { // stop the timer if (_timer != null) { ((IDisposable)_timer).Dispose(); _timer = null; } while (_inProgressLock != 0) Thread.Sleep(100); // cancel all cancelable requests if (_requestCount > 0) CancelTimedOutRequests(DateTime.UtcNow.AddYears(1)); // } private void TimerCompletionCallback(Object state) { if (_requestCount > 0) CancelTimedOutRequests(DateTime.UtcNow); } private void CancelTimedOutRequests(DateTime now) { // only one thread can be doing it if (Interlocked.CompareExchange(ref _inProgressLock, 1, 0) != 0) return; // collect them all into a separate list with minimal locking ArrayList entries = new ArrayList(_requestCount); // size can change DoubleLinkListEnumerator en; for (int i = 0; i < _lists.Length; i++) { lock (_lists[i]) { en = _lists[i].GetEnumerator(); while (en.MoveNext()) entries.Add(en.GetDoubleLink()); en = null; } } // walk through the collected list to timeout what's needed int n = entries.Count; for (int i = 0; i < n; i++) ((RequestTimeoutEntry)entries[i]).TimeoutIfNeeded(now); // this thread is done -- unlock Interlocked.Exchange(ref _inProgressLock, 0); } internal void Add(HttpContext context) { if (context.TimeoutLink != null) { ((RequestTimeoutEntry)context.TimeoutLink).IncrementCount(); return; } // create new entry RequestTimeoutEntry entry = new RequestTimeoutEntry(context); // add it to the list int i = _currentList++; if (i >= _lists.Length) { i = 0; _currentList = 0; } entry.AddToList(_lists[i]); Interlocked.Increment(ref _requestCount); // update HttpContext context.TimeoutLink = entry; } internal void Remove(HttpContext context) { RequestTimeoutEntry entry = (RequestTimeoutEntry)context.TimeoutLink; // remove from the list if (entry != null) { if( entry.DecrementCount() == 0 ) { entry.RemoveFromList(); Interlocked.Decrement(ref _requestCount); } else { return; } } // update HttpContext context.TimeoutLink = null; } private class RequestTimeoutEntry : DoubleLink { private HttpContext _context; // the request private DoubleLinkList _list; private int _count; internal RequestTimeoutEntry(HttpContext context) { _context = context; _count = 1; } internal void AddToList(DoubleLinkList list) { lock(list) { list.InsertTail(this); _list = list; } } internal void RemoveFromList() { if (_list != null) { lock(_list) { Remove(); _list = null; } } } internal void TimeoutIfNeeded(DateTime now) { Thread thread = _context.MustTimeout(now); if (thread != null) { RemoveFromList(); thread.Abort(new HttpApplication.CancelModuleException(true)); } } internal void IncrementCount() { Interlocked.Increment( ref _count ); } internal int DecrementCount() { return Interlocked.Decrement( ref _count ); } } } } // 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
- DbTransaction.cs
- Wizard.cs
- BasePropertyDescriptor.cs
- XmlNodeReader.cs
- MergeExecutor.cs
- SqlBinder.cs
- SqlRowUpdatedEvent.cs
- RuleSet.cs
- IBuiltInEvidence.cs
- UIElementParagraph.cs
- JsonWriter.cs
- DataSourceView.cs
- Math.cs
- ChannelTerminatedException.cs
- Label.cs
- InvokeMethodActivityDesigner.cs
- PropertyChangeTracker.cs
- XmlAttributeOverrides.cs
- ConfigUtil.cs
- CapabilitiesAssignment.cs
- Encoder.cs
- ChangePassword.cs
- SmuggledIUnknown.cs
- CodeExporter.cs
- MissingMemberException.cs
- BamlReader.cs
- MatrixStack.cs
- HashSetDebugView.cs
- WindowsToolbarItemAsMenuItem.cs
- ControlCachePolicy.cs
- DataGridViewLayoutData.cs
- WebPartActionVerb.cs
- MemberDescriptor.cs
- CultureMapper.cs
- DataServiceRequestOfT.cs
- ZoneLinkButton.cs
- ThreadExceptionEvent.cs
- TextParaClient.cs
- LinearGradientBrush.cs
- CodeTypeDeclarationCollection.cs
- GenericXmlSecurityToken.cs
- XPathMessageFilter.cs
- FragmentNavigationEventArgs.cs
- Config.cs
- SubclassTypeValidatorAttribute.cs
- DataControlImageButton.cs
- ProjectionPath.cs
- FileIOPermission.cs
- GuidConverter.cs
- DataGridViewCellValidatingEventArgs.cs
- coordinator.cs
- LogicalExpressionEditor.cs
- ArgumentDesigner.xaml.cs
- SplineKeyFrames.cs
- TextServicesHost.cs
- ServerIdentity.cs
- DecimalStorage.cs
- EmptyQuery.cs
- ObjectDataSourceChooseMethodsPanel.cs
- OracleMonthSpan.cs
- ArrayEditor.cs
- ModulesEntry.cs
- TextProperties.cs
- TraceListener.cs
- CodeAccessSecurityEngine.cs
- Missing.cs
- SignatureHelper.cs
- TraceEventCache.cs
- RefExpr.cs
- EntitySet.cs
- SqlOuterApplyReducer.cs
- bidPrivateBase.cs
- PointAnimationUsingKeyFrames.cs
- NameValueCollection.cs
- XmlEnumAttribute.cs
- PrintDialogException.cs
- OperationResponse.cs
- TimelineGroup.cs
- DefaultShape.cs
- IdentityNotMappedException.cs
- EnterpriseServicesHelper.cs
- TypeViewSchema.cs
- Baml2006SchemaContext.cs
- Invariant.cs
- BoundColumn.cs
- ConfigurationManagerInternal.cs
- Nullable.cs
- MouseButtonEventArgs.cs
- ProxyManager.cs
- UserPreferenceChangedEventArgs.cs
- RestrictedTransactionalPackage.cs
- SocketAddress.cs
- CodeExpressionStatement.cs
- SwitchAttribute.cs
- WebSysDescriptionAttribute.cs
- MediaElement.cs
- Point3DCollection.cs
- DecoderExceptionFallback.cs
- BrowserTree.cs
- Int32RectConverter.cs