You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			337 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			337 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| namespace System.Web.Services.Protocols {
 | |
|     using System;
 | |
|     using System.Web.Services;
 | |
|     using System.Diagnostics;
 | |
|     using System.Text;
 | |
|     using System.Runtime.InteropServices;
 | |
|     using System.Web.Services.Interop;
 | |
|     using System.Reflection;
 | |
|     using System.Threading;
 | |
|     using System.Security.Permissions;
 | |
|     using System.Net;
 | |
|     using System.ComponentModel; // for CompModSwitches
 | |
|     using System.Web.Services.Diagnostics;
 | |
| 
 | |
|     internal class RemoteDebugger : INotifySource2 {
 | |
|         private static INotifyConnection2 connection;
 | |
|         private static bool getConnection = true;
 | |
|         private INotifySink2 notifySink;
 | |
|         private NotifyFilter notifyFilter;
 | |
|         private UserThread userThread;
 | |
| 
 | |
|         private const int INPROC_SERVER = 1;
 | |
|         private static Guid IID_NotifyConnectionClassGuid = new Guid("12A5B9F0-7A1C-4fcb-8163-160A30F519B5");
 | |
|         private static Guid IID_NotifyConnection2Guid = new Guid("1AF04045-6659-4aaa-9F4B-2741AC56224B");
 | |
|         private static string debuggerHeader = "VsDebuggerCausalityData";
 | |
| 
 | |
|         private static Object s_InternalSyncObject;
 | |
|         private static Object InternalSyncObject {
 | |
|             get {
 | |
|                 if (s_InternalSyncObject == null) {
 | |
|                     Object o = new Object();
 | |
|                     Interlocked.CompareExchange(ref s_InternalSyncObject, o, null);
 | |
|                 }
 | |
|                 return s_InternalSyncObject;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         [DebuggerStepThrough]
 | |
|         [DebuggerHidden]
 | |
|         internal RemoteDebugger() {
 | |
|         }
 | |
| 
 | |
|         ~RemoteDebugger() {
 | |
|             Close();
 | |
|         }
 | |
| 
 | |
| 
 | |
|         internal static bool IsClientCallOutEnabled() {
 | |
|             bool enabled = false;
 | |
| 
 | |
|             try {
 | |
|                 enabled = !CompModSwitches.DisableRemoteDebugging.Enabled && Debugger.IsAttached && Connection != null;
 | |
|             }
 | |
|             catch (Exception e) {
 | |
|                 if (e is ThreadAbortException || e is StackOverflowException || e is OutOfMemoryException) {
 | |
|                     throw;
 | |
|                 }
 | |
|                 if (Tracing.On) Tracing.ExceptionCatch(TraceEventType.Warning, typeof(RemoteDebugger), "IsClientCallOutEnabled", e);
 | |
|             }
 | |
|             return enabled;
 | |
|         }
 | |
| 
 | |
|         internal static bool IsServerCallInEnabled(ServerProtocol protocol, out string stringBuffer) {
 | |
|             stringBuffer = null;
 | |
|             bool enabled = false;
 | |
|             try {
 | |
|                 if (CompModSwitches.DisableRemoteDebugging.Enabled)
 | |
|                     return false;
 | |
| 
 | |
|                 enabled = protocol.Context.IsDebuggingEnabled && Connection != null;
 | |
|                 if (enabled) {
 | |
|                     stringBuffer = protocol.Request.Headers[debuggerHeader];
 | |
|                     enabled = (stringBuffer != null && stringBuffer.Length > 0);
 | |
|                 }
 | |
|             }
 | |
|             catch (Exception e) {
 | |
|                 if (e is ThreadAbortException || e is StackOverflowException || e is OutOfMemoryException) {
 | |
|                     throw;
 | |
|                 }
 | |
|                 if (Tracing.On) Tracing.ExceptionCatch(TraceEventType.Warning, typeof(RemoteDebugger), "IsServerCallInEnabled", e);
 | |
|                 enabled = false;
 | |
|             }
 | |
|             return enabled;
 | |
|         }
 | |
| 
 | |
|         private static INotifyConnection2 Connection {
 | |
|             get {
 | |
|                 if (connection == null && getConnection) {
 | |
|                     lock (InternalSyncObject) {
 | |
|                         if (connection == null) {
 | |
|                             AppDomain.CurrentDomain.DomainUnload += new EventHandler(OnAppDomainUnload);
 | |
|                             AppDomain.CurrentDomain.ProcessExit += new EventHandler(OnProcessExit);
 | |
| 
 | |
|                             TraceMethod method = Tracing.On ? new TraceMethod(typeof(RemoteDebugger), "get_Connection") : null;
 | |
|                             if (Tracing.On) Tracing.Enter("RemoteDebugger", method);
 | |
| 
 | |
|                             object unk;
 | |
|                             int result = UnsafeNativeMethods.CoCreateInstance(ref IID_NotifyConnectionClassGuid,
 | |
|                                                                                null,
 | |
|                                                                                INPROC_SERVER,
 | |
|                                                                                ref IID_NotifyConnection2Guid,
 | |
|                                                                                out unk);
 | |
| 
 | |
|                             if (Tracing.On) Tracing.Exit("RemoteDebugger", method);
 | |
| 
 | |
|                             if (result >= 0) // success
 | |
|                                 connection = (INotifyConnection2)unk;
 | |
|                             else
 | |
|                                 connection = null;
 | |
|                         }
 | |
|                         getConnection = false;
 | |
|                     }
 | |
|                 }
 | |
|                 return connection;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         private INotifySink2 NotifySink {
 | |
|             get {
 | |
|                 if (this.notifySink == null && Connection != null) {
 | |
|                     TraceMethod method = Tracing.On ? new TraceMethod(this, "get_NotifySink") : null;
 | |
|                     if (Tracing.On) Tracing.Enter("RemoteDebugger", method);
 | |
| 
 | |
|                     this.notifySink = UnsafeNativeMethods.RegisterNotifySource(Connection, this);
 | |
| 
 | |
|                     if (Tracing.On) Tracing.Exit("RemoteDebugger", method);
 | |
|                 }
 | |
|                 return this.notifySink;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private static void CloseSharedResources() {
 | |
|             if (connection != null) {
 | |
|                 lock (InternalSyncObject) {
 | |
|                     if (connection != null) {
 | |
|                         TraceMethod method = Tracing.On ? new TraceMethod(typeof(RemoteDebugger), "CloseSharedResources") : null;
 | |
|                         if (Tracing.On) Tracing.Enter("RemoteDebugger", method);
 | |
| 
 | |
|                         try {
 | |
|                             Marshal.ReleaseComObject(connection);
 | |
|                         }
 | |
|                         catch (Exception e) {
 | |
|                             if (e is ThreadAbortException || e is StackOverflowException || e is OutOfMemoryException) {
 | |
|                                 throw;
 | |
|                             }
 | |
|                             if (Tracing.On) Tracing.ExceptionCatch(TraceEventType.Warning, typeof(RemoteDebugger), "CloseSharedResources", e);
 | |
|                         }
 | |
|                         if (Tracing.On) Tracing.Exit("RemoteDebugger", method);
 | |
|                         connection = null;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private void Close() {
 | |
|             if (this.notifySink != null && connection != null) {
 | |
|                 lock (InternalSyncObject) {
 | |
|                     if (this.notifySink != null && connection != null) {
 | |
|                         TraceMethod method = Tracing.On ? new TraceMethod(this, "Close") : null;
 | |
|                         if (Tracing.On) Tracing.Enter("RemoteDebugger", method);
 | |
| 
 | |
|                         try {
 | |
|                             UnsafeNativeMethods.UnregisterNotifySource(connection, this);
 | |
|                         }
 | |
|                         catch (Exception e) {
 | |
|                             if (e is ThreadAbortException || e is StackOverflowException || e is OutOfMemoryException) {
 | |
|                                 throw;
 | |
|                             }
 | |
|                             if (Tracing.On) Tracing.ExceptionCatch(TraceEventType.Warning, method, e);
 | |
|                         }
 | |
|                         if (Tracing.On) Tracing.Exit("RemoteDebugger", method);
 | |
|                         this.notifySink = null;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         [DebuggerStepThrough]
 | |
|         [DebuggerHidden]
 | |
|         internal void NotifyClientCallOut(WebRequest request) {
 | |
|             try {
 | |
|                 if (NotifySink == null) return;
 | |
| 
 | |
|                 IntPtr bufferPtr;
 | |
|                 int bufferSize = 0;
 | |
|                 CallId callId = new CallId(null, 0, (IntPtr)0, 0, null, request.RequestUri.Host);
 | |
| 
 | |
|                 TraceMethod method = Tracing.On ? new TraceMethod(this, "NotifyClientCallOut") : null;
 | |
|                 if (Tracing.On) Tracing.Enter("RemoteDebugger", method);
 | |
| 
 | |
|                 UnsafeNativeMethods.OnSyncCallOut(NotifySink, callId, out bufferPtr, ref bufferSize);
 | |
| 
 | |
|                 if (Tracing.On) Tracing.Exit("RemoteDebugger", method);
 | |
| 
 | |
|                 if (bufferPtr == IntPtr.Zero) return;
 | |
|                 byte[] buffer = null;
 | |
|                 try {
 | |
|                     buffer = new byte[bufferSize];
 | |
|                     Marshal.Copy(bufferPtr, buffer, 0, bufferSize);
 | |
|                 }
 | |
|                 finally {
 | |
|                     Marshal.FreeCoTaskMem(bufferPtr);
 | |
|                 }
 | |
|                 string bufferString = Convert.ToBase64String(buffer);
 | |
|                 request.Headers.Add(debuggerHeader, bufferString);
 | |
|             }
 | |
|             catch (Exception e) {
 | |
|                 if (e is ThreadAbortException || e is StackOverflowException || e is OutOfMemoryException) {
 | |
|                     throw;
 | |
|                 }
 | |
|                 if (Tracing.On) Tracing.ExceptionCatch(TraceEventType.Warning, typeof(RemoteDebugger), "NotifyClientCallOut", e);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         [DebuggerStepThrough]
 | |
|         [DebuggerHidden]
 | |
|         internal void NotifyClientCallReturn(WebResponse response) {
 | |
|             try {
 | |
|                 if (NotifySink == null) return;
 | |
| 
 | |
|                 byte[] buffer = new byte[0];
 | |
|                 if (response != null) {
 | |
|                     string bufferString = response.Headers[debuggerHeader];
 | |
|                     if (bufferString != null && bufferString.Length != 0)
 | |
|                         buffer = Convert.FromBase64String(bufferString);
 | |
|                 }
 | |
|                 CallId callId = new CallId(null, 0, (IntPtr)0, 0, null, null);
 | |
| 
 | |
|                 TraceMethod method = Tracing.On ? new TraceMethod(this, "NotifyClientCallReturn") : null;
 | |
|                 if (Tracing.On) Tracing.Enter("RemoteDebugger", method);
 | |
| 
 | |
|                 UnsafeNativeMethods.OnSyncCallReturn(NotifySink, callId, buffer, buffer.Length);
 | |
| 
 | |
|                 if (Tracing.On) Tracing.Exit("RemoteDebugger", method);
 | |
|             }
 | |
|             catch (Exception e) {
 | |
|                 if (e is ThreadAbortException || e is StackOverflowException || e is OutOfMemoryException) {
 | |
|                     throw;
 | |
|                 }
 | |
|                 if (Tracing.On) Tracing.ExceptionCatch(TraceEventType.Warning, typeof(RemoteDebugger), "NotifyClientCallReturn", e);
 | |
|             }
 | |
| 
 | |
|             this.Close();
 | |
|         }
 | |
| 
 | |
|         [DebuggerStepThrough]
 | |
|         [DebuggerHidden]
 | |
|         internal void NotifyServerCallEnter(ServerProtocol protocol, string stringBuffer) {
 | |
|             try {
 | |
|                 if (NotifySink == null) return;
 | |
|                 StringBuilder methodBuilder = new StringBuilder();
 | |
|                 methodBuilder.Append(protocol.Type.FullName);
 | |
|                 methodBuilder.Append('.');
 | |
|                 methodBuilder.Append(protocol.MethodInfo.Name);
 | |
|                 methodBuilder.Append('(');
 | |
|                 ParameterInfo[] parameterInfos = protocol.MethodInfo.Parameters;
 | |
|                 for (int i = 0; i < parameterInfos.Length; ++i) {
 | |
|                     if (i != 0)
 | |
|                         methodBuilder.Append(',');
 | |
| 
 | |
|                     methodBuilder.Append(parameterInfos[i].ParameterType.FullName);
 | |
|                 }
 | |
|                 methodBuilder.Append(')');
 | |
| 
 | |
|                 byte[] buffer = Convert.FromBase64String(stringBuffer);
 | |
|                 CallId callId = new CallId(null, 0, (IntPtr)0, 0, methodBuilder.ToString(), null);
 | |
| 
 | |
|                 TraceMethod method = Tracing.On ? new TraceMethod(this, "NotifyServerCallEnter") : null;
 | |
|                 if (Tracing.On) Tracing.Enter("RemoteDebugger", method);
 | |
| 
 | |
|                 UnsafeNativeMethods.OnSyncCallEnter(NotifySink, callId, buffer, buffer.Length);
 | |
| 
 | |
|                 if (Tracing.On) Tracing.Exit("RemoteDebugger", method);
 | |
|             }
 | |
|             catch (Exception e) {
 | |
|                 if (e is ThreadAbortException || e is StackOverflowException || e is OutOfMemoryException) {
 | |
|                     throw;
 | |
|                 }
 | |
|                 if (Tracing.On) Tracing.ExceptionCatch(TraceEventType.Warning, typeof(RemoteDebugger), "NotifyServerCallEnter", e);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         [DebuggerStepThrough]
 | |
|         [DebuggerHidden]
 | |
|         internal void NotifyServerCallExit(HttpResponse response) {
 | |
|             try {
 | |
|                 if (NotifySink == null) return;
 | |
| 
 | |
|                 IntPtr bufferPtr;
 | |
|                 int bufferSize = 0;
 | |
|                 CallId callId = new CallId(null, 0, (IntPtr)0, 0, null, null);
 | |
| 
 | |
|                 TraceMethod method = Tracing.On ? new TraceMethod(this, "NotifyServerCallExit") : null;
 | |
|                 if (Tracing.On) Tracing.Enter("RemoteDebugger", method);
 | |
| 
 | |
|                 UnsafeNativeMethods.OnSyncCallExit(NotifySink, callId, out bufferPtr, ref bufferSize);
 | |
| 
 | |
|                 if (Tracing.On) Tracing.Exit("RemoteDebugger", method);
 | |
| 
 | |
|                 if (bufferPtr == IntPtr.Zero) return;
 | |
|                 byte[] buffer = null;
 | |
|                 try {
 | |
|                     buffer = new byte[bufferSize];
 | |
|                     Marshal.Copy(bufferPtr, buffer, 0, bufferSize);
 | |
|                 }
 | |
|                 finally {
 | |
|                     Marshal.FreeCoTaskMem(bufferPtr);
 | |
|                 }
 | |
|                 string stringBuffer = Convert.ToBase64String(buffer);
 | |
|                 response.AddHeader(debuggerHeader, stringBuffer);
 | |
|             }
 | |
|             catch (Exception e) {
 | |
|                 if (e is ThreadAbortException || e is StackOverflowException || e is OutOfMemoryException) {
 | |
|                     throw;
 | |
|                 }
 | |
|                 if (Tracing.On) Tracing.ExceptionCatch(TraceEventType.Warning, typeof(RemoteDebugger), "NotifyServerCallExit", e);
 | |
|             }
 | |
|             this.Close();
 | |
|         }
 | |
| 
 | |
| 
 | |
|         private static void OnAppDomainUnload(object sender, EventArgs args) {
 | |
|             CloseSharedResources();
 | |
|         }
 | |
| 
 | |
|         private static void OnProcessExit(object sender, EventArgs args) {
 | |
|             CloseSharedResources();
 | |
|         }
 | |
| 
 | |
|         void INotifySource2.SetNotifyFilter(NotifyFilter in_NotifyFilter, UserThread in_pUserThreadFilter) {
 | |
|             notifyFilter = in_NotifyFilter;
 | |
|             userThread = in_pUserThreadFilter;
 | |
|         }
 | |
|     }
 | |
| }
 |