//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // //------------------------------------------------------------------------------ namespace System.Web.Hosting { using System; using System.Collections.Concurrent; using System.Collections.Generic; using System.Threading; using System.Web.Util; // Handles calling suspend and resume methods, including issues around // synchronization and timeout handling. internal sealed class SuspendManager { private static readonly TimeSpan _suspendMethodTimeout = TimeSpan.FromSeconds(5); private readonly ConcurrentDictionary _registeredObjects = new ConcurrentDictionary(); public void RegisterObject(ISuspendibleRegisteredObject o) { Debug.Assert(o != null); _registeredObjects[o] = null; } public void UnregisterObject(ISuspendibleRegisteredObject o) { Debug.Assert(o != null); ((IDictionary)_registeredObjects).Remove(o); } // Returns a state object that will be passed to the Resume method. public object Suspend() { // ConcurrentDictionary.Count / IsEmpty are basically as expensive as getting // the entire collection of keys, so we may as well just read the keys anyway. var allRegisteredObjects = _registeredObjects.Keys; return SuspendImpl(allRegisteredObjects); } private static SuspendState SuspendImpl(ICollection allRegisteredObjects) { // Our behavior is: // - We'll call each registered object's suspend method serially. // - All methods have a combined 5 seconds to respond, at which // point we'll forcibly return to our caller. // - If a Resume call comes in, we'll not call any Suspend methods // we haven't yet gotten around to, and we'll execute each // resume callback we got. // - Resume callbacks may fire in parallel, even if Suspend methods // fire sequentially. // - Resume methods fire asynchronously, so other events (such as // Stop or a new Suspend call) could happen while a Resume callback // is in progress. CountdownEvent countdownEvent = new CountdownEvent(2); SuspendState suspendState = new SuspendState(allRegisteredObjects); // Unsafe QUWI since occurs outside the context of a request. // We are not concerned about impersonation, identity, etc. // Invoke any registered subscribers to let them know that we're about // to suspend. This is done in parallel with ASP.NET's own cleanup below. if (allRegisteredObjects.Count > 0) { ThreadPool.UnsafeQueueUserWorkItem(_ => { suspendState.Suspend(); countdownEvent.Signal(); }, null); } else { countdownEvent.Signal(); // nobody is subscribed } // Release any unnecessary memory that we're holding on to. The GC will // be able to reclaim these, which means that we'll have to page in less // memory when the next request comes in. ThreadPool.UnsafeQueueUserWorkItem(_ => { // Release any char[] buffers we're keeping around HttpWriter.ReleaseAllPooledBuffers(); // Trim expired entries from the runtime cache var iCache = HttpRuntime.Cache.GetInternalCache(createIfDoesNotExist: false); var oCache = HttpRuntime.Cache.GetObjectCache(createIfDoesNotExist: false); if (iCache != null) { iCache.Trim(0); } if (oCache != null && !oCache.Equals(iCache)) { oCache.Trim(0); } // Trim all pooled HttpApplication instances HttpApplicationFactory.TrimApplicationInstances(removeAll: true); countdownEvent.Signal(); }, null); if (Debug.IsDebuggerPresent()) { countdownEvent.Wait(); // to assist with debugging, don't time out if a debugger is attached } else { countdownEvent.Wait(_suspendMethodTimeout); // blocking call, ok for our needs since has finite wait time } return suspendState; } public void Resume(object state) { ((SuspendState)state).Resume(); } internal sealed class SuspendState { private static readonly WaitCallback _quwiThunk = (state) => ((Action)state)(); private readonly ICollection _suspendibleObjects; // these two fields should only ever be accessed under lock private readonly List _resumeCallbacks; private bool _resumeWasCalled; public SuspendState(ICollection suspendibleObjects) { _suspendibleObjects = suspendibleObjects; _resumeCallbacks = new List(suspendibleObjects.Count); } public void Suspend() { foreach (ISuspendibleRegisteredObject suspendibleObject in _suspendibleObjects) { Action callback = suspendibleObject.Suspend(); lock (this) { // If Resume was called while the Suspend method was still executing, // this callback won't be invoked, so we need to invoke it manually. if (_resumeWasCalled) { if (callback != null) { InvokeResumeCallbackAsync(callback); } return; // don't run any other Suspend methods } if (callback != null) { _resumeCallbacks.Add(callback); } } } } public void Resume() { lock (this) { Debug.Assert(!_resumeWasCalled, "Resume was called too many times!"); _resumeWasCalled = true; foreach (Action callback in _resumeCallbacks) { InvokeResumeCallbackAsync(callback); } } } private static void InvokeResumeCallbackAsync(Action callback) { // Unsafe QUWI since occurs outside the context of a request. // We are not concerned about impersonation, identity, etc. ThreadPool.UnsafeQueueUserWorkItem(_quwiThunk, callback); } } } }