96 lines
4.2 KiB
C#
96 lines
4.2 KiB
C#
|
//------------------------------------------------------------------------------
|
|||
|
// <copyright file="CustomRuntimeSuspendManager.cs" company="Microsoft">
|
|||
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|||
|
// </copyright>
|
|||
|
//------------------------------------------------------------------------------
|
|||
|
|
|||
|
namespace System.Web.Hosting {
|
|||
|
using System.Collections.Concurrent;
|
|||
|
using System.Collections.Generic;
|
|||
|
using System.Web.Util;
|
|||
|
|
|||
|
// Handles calling suspend and resume methods, including issues around
|
|||
|
// synchronization and timeout handling.
|
|||
|
|
|||
|
internal sealed class CustomRuntimeManager : ICustomRuntimeManager {
|
|||
|
|
|||
|
private readonly ConcurrentDictionary<CustomRuntimeRegistration, object> _activeRegistrations = new ConcurrentDictionary<CustomRuntimeRegistration, object>();
|
|||
|
|
|||
|
private List<IProcessSuspendListener> GetAllSuspendListeners() {
|
|||
|
List<IProcessSuspendListener> suspendListeners = new List<IProcessSuspendListener>();
|
|||
|
|
|||
|
foreach (var registration in _activeRegistrations.Keys) {
|
|||
|
var suspendListener = registration.CustomRuntime as IProcessSuspendListener;
|
|||
|
if (suspendListener != null) {
|
|||
|
suspendListeners.Add(suspendListener);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return suspendListeners;
|
|||
|
}
|
|||
|
|
|||
|
public ICustomRuntimeRegistrationToken Register(ICustomRuntime customRuntime) {
|
|||
|
CustomRuntimeRegistration registration = new CustomRuntimeRegistration(this, customRuntime);
|
|||
|
_activeRegistrations[registration] = null;
|
|||
|
return registration;
|
|||
|
}
|
|||
|
|
|||
|
// Custom runtimes are expected to be well-behaved so will be suspended sequentially and aren't
|
|||
|
// subject to the 5-second timeout that normal applications are subject to.
|
|||
|
public Action SuspendAllCustomRuntimes() {
|
|||
|
var suspendListeners = GetAllSuspendListeners();
|
|||
|
if (suspendListeners == null || suspendListeners.Count == 0) {
|
|||
|
return null;
|
|||
|
}
|
|||
|
|
|||
|
List<IProcessResumeCallback> callbacks = new List<IProcessResumeCallback>(suspendListeners.Count);
|
|||
|
foreach (var suspendListener in suspendListeners) {
|
|||
|
IProcessResumeCallback callback = null;
|
|||
|
try {
|
|||
|
callback = suspendListener.Suspend();
|
|||
|
}
|
|||
|
catch (AppDomainUnloadedException) {
|
|||
|
// There exists a race condition where a custom runtime could have been stopped (unloaded)
|
|||
|
// while a call to Suspend is in progress, so AD unloads may leak out. Don't treat this
|
|||
|
// as a failure; just move on.
|
|||
|
}
|
|||
|
|
|||
|
if (callback != null) {
|
|||
|
callbacks.Add(callback);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return () => {
|
|||
|
foreach (var callback in callbacks) {
|
|||
|
try {
|
|||
|
callback.Resume();
|
|||
|
}
|
|||
|
catch (AppDomainUnloadedException) {
|
|||
|
// There exists a race condition where a custom runtime could have been stopped (unloaded)
|
|||
|
// while a call to Suspend is in progress, so AD unloads may leak out. Don't treat this
|
|||
|
// as a failure; just move on.
|
|||
|
}
|
|||
|
}
|
|||
|
};
|
|||
|
}
|
|||
|
|
|||
|
private sealed class CustomRuntimeRegistration : ICustomRuntimeRegistrationToken {
|
|||
|
private readonly CustomRuntimeManager _customRuntimeManager;
|
|||
|
|
|||
|
public CustomRuntimeRegistration(CustomRuntimeManager customRuntimeManager, ICustomRuntime customRuntime) {
|
|||
|
_customRuntimeManager = customRuntimeManager;
|
|||
|
CustomRuntime = customRuntime;
|
|||
|
}
|
|||
|
|
|||
|
public ICustomRuntime CustomRuntime { get; private set; }
|
|||
|
|
|||
|
public void Unregister() {
|
|||
|
object dummy;
|
|||
|
bool removed = _customRuntimeManager._activeRegistrations.TryRemove(this, out dummy);
|
|||
|
|
|||
|
Debug.Assert(removed, "Entry did not exist in the dictionary; was it removed twice?");
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|