210 lines
7.7 KiB
C#
210 lines
7.7 KiB
C#
|
#region Using directives
|
|||
|
|
|||
|
using System;
|
|||
|
using System.Threading;
|
|||
|
using System.Reflection;
|
|||
|
using System.Diagnostics;
|
|||
|
using System.Runtime.InteropServices;
|
|||
|
using System.Globalization;
|
|||
|
using Microsoft.Win32;
|
|||
|
#endregion
|
|||
|
|
|||
|
namespace System.Workflow.Runtime.DebugEngine
|
|||
|
{
|
|||
|
internal sealed class DebugControllerThread
|
|||
|
{
|
|||
|
#region Data members
|
|||
|
|
|||
|
private Thread controllerThread = null;
|
|||
|
private int threadId = 0;
|
|||
|
private ManualResetEvent threadInitializedEvent = null;
|
|||
|
private volatile bool runThread = false;
|
|||
|
private static readonly string ExpressionEvaluationFrameTypeName = "ExpressionEvaluationFrameTypeName";
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Methods
|
|||
|
|
|||
|
public DebugControllerThread()
|
|||
|
{
|
|||
|
this.threadInitializedEvent = new ManualResetEvent(false);
|
|||
|
this.threadInitializedEvent.Reset();
|
|||
|
|
|||
|
this.controllerThread = new Thread(ControllerThreadFunction);
|
|||
|
this.controllerThread.IsBackground = true;
|
|||
|
this.controllerThread.Priority = ThreadPriority.Lowest;
|
|||
|
this.controllerThread.Name = "__dct__";
|
|||
|
}
|
|||
|
|
|||
|
public void RunThread(IInstanceTable instanceTable)
|
|||
|
{
|
|||
|
Debug.WriteLine(string.Format(CultureInfo.CurrentCulture, "WDE: DebugControllerThread.DebugControllerThread():"));
|
|||
|
if (this.controllerThread == null)
|
|||
|
return;
|
|||
|
|
|||
|
this.runThread = true;
|
|||
|
this.controllerThread.Start(instanceTable);
|
|||
|
this.threadInitializedEvent.WaitOne();
|
|||
|
}
|
|||
|
|
|||
|
public void StopThread()
|
|||
|
{
|
|||
|
Debug.WriteLine(string.Format(CultureInfo.CurrentCulture, "WDE: DebugControllerThread.StopThread():"));
|
|||
|
|
|||
|
try
|
|||
|
{
|
|||
|
if (this.controllerThread != null)
|
|||
|
{
|
|||
|
this.runThread = false;
|
|||
|
Thread.Sleep(10);
|
|||
|
|
|||
|
// On x64 we put the debug controller thread to Sleep(Timeout.Infinite) in
|
|||
|
// ExpressionEvaluationFunction. This thread needs to be started before it
|
|||
|
// a Join can execute.
|
|||
|
if (this.controllerThread.IsAlive && IntPtr.Size == 8)
|
|||
|
{
|
|||
|
while (this.controllerThread.ThreadState == System.Threading.ThreadState.WaitSleepJoin)
|
|||
|
{
|
|||
|
this.controllerThread.Start();
|
|||
|
this.controllerThread.Join();
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
this.controllerThread.Join();
|
|||
|
}
|
|||
|
}
|
|||
|
catch (ThreadStateException)
|
|||
|
{
|
|||
|
// Ignore the ThreadStateException which will be thrown when StopThread() is called during
|
|||
|
// AppDomain unload.
|
|||
|
}
|
|||
|
finally
|
|||
|
{
|
|||
|
this.controllerThread = null;
|
|||
|
}
|
|||
|
|
|||
|
this.controllerThread = null;
|
|||
|
this.threadId = 0;
|
|||
|
this.threadInitializedEvent = null;
|
|||
|
}
|
|||
|
|
|||
|
private void ControllerThreadFunction(object instanceTable)
|
|||
|
{
|
|||
|
try
|
|||
|
{
|
|||
|
Debug.WriteLine(string.Format(CultureInfo.CurrentCulture, "WDE: DebugControllerThread.ControllerThreadFunction():"));
|
|||
|
|
|||
|
IExpressionEvaluationFrame expressionEvaluationFrame = null;
|
|||
|
|
|||
|
try
|
|||
|
{
|
|||
|
RegistryKey debugEngineSubKey = Registry.LocalMachine.OpenSubKey(RegistryKeys.DebuggerSubKey);
|
|||
|
if (debugEngineSubKey != null)
|
|||
|
{
|
|||
|
string evaluationFrameTypeName = debugEngineSubKey.GetValue(ExpressionEvaluationFrameTypeName, String.Empty) as string;
|
|||
|
if (!String.IsNullOrEmpty(evaluationFrameTypeName) && Type.GetType(evaluationFrameTypeName) != null)
|
|||
|
expressionEvaluationFrame = Activator.CreateInstance(Type.GetType(evaluationFrameTypeName)) as IExpressionEvaluationFrame;
|
|||
|
}
|
|||
|
}
|
|||
|
catch { }
|
|||
|
|
|||
|
if (expressionEvaluationFrame == null)
|
|||
|
{
|
|||
|
Type eeFrameType = null;
|
|||
|
|
|||
|
const string eeFrameTypeNameFormat = "Microsoft.Workflow.DebugEngine.ExpressionEvaluationFrame, Microsoft.Workflow.ExpressionEvaluation, Version={0}.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad364e35";
|
|||
|
|
|||
|
// Try versions 12.0.0.0, 11.0.0.0, 10.0.0.0
|
|||
|
for (int version = 12; eeFrameType == null && version >= 10; --version)
|
|||
|
{
|
|||
|
try
|
|||
|
{
|
|||
|
eeFrameType = Type.GetType(string.Format(CultureInfo.InvariantCulture, eeFrameTypeNameFormat, version));
|
|||
|
}
|
|||
|
catch (TypeLoadException)
|
|||
|
{
|
|||
|
// Fall back to next-lower version
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if (eeFrameType != null)
|
|||
|
{
|
|||
|
expressionEvaluationFrame = Activator.CreateInstance(eeFrameType) as IExpressionEvaluationFrame;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
Debug.Assert(expressionEvaluationFrame != null, "Failed to create Expression Evaluation Frame.");
|
|||
|
|
|||
|
if (expressionEvaluationFrame != null)
|
|||
|
expressionEvaluationFrame.CreateEvaluationFrame((IInstanceTable)instanceTable, (DebugEngineCallback)Delegate.CreateDelegate(typeof(DebugEngineCallback), this, "ExpressionEvaluationFunction"));
|
|||
|
}
|
|||
|
catch
|
|||
|
{
|
|||
|
// Don't throw exceptions to the Runtime, it would terminate the debugee.
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// This thread spins forever. It is used by the Debug Engine to perform expression evaluation as a "carrier
|
|||
|
// wave". It is created only when the debugger is attached and terminates itself when the debugger isn't
|
|||
|
// attached anymore.
|
|||
|
public void ExpressionEvaluationFunction()
|
|||
|
{
|
|||
|
this.threadId = NativeMethods.GetCurrentThreadId();
|
|||
|
this.threadInitializedEvent.Set();
|
|||
|
|
|||
|
|
|||
|
using (new DebuggerThreadMarker())
|
|||
|
{
|
|||
|
// If an exception occurs somehow, continue to spin.
|
|||
|
while (this.runThread)
|
|||
|
{
|
|||
|
try
|
|||
|
{
|
|||
|
// Expression eval on x64 does not work (bug 18143) so
|
|||
|
// don't let the thread spin.
|
|||
|
if (IntPtr.Size == 8)
|
|||
|
{
|
|||
|
Thread.Sleep(Timeout.Infinite);
|
|||
|
}
|
|||
|
else
|
|||
|
// Spin within the try catch.
|
|||
|
while (this.runThread);
|
|||
|
}
|
|||
|
catch (ThreadAbortException)
|
|||
|
{
|
|||
|
Debug.WriteLine(string.Format(CultureInfo.CurrentCulture, "WDE: DebugControllerThread.ExpressionEvaluationFunction(): ThreadAbortException"));
|
|||
|
|
|||
|
// Explicitly do not call ResetAbort().
|
|||
|
throw;
|
|||
|
}
|
|||
|
catch
|
|||
|
{
|
|||
|
Debug.WriteLine(string.Format(CultureInfo.CurrentCulture, "WDE: DebugControllerThread.ExpressionEvaluationFunction(): other exception"));
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
#endregion
|
|||
|
|
|||
|
#region Properties
|
|||
|
|
|||
|
public int ThreadId
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
return this.threadId;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public int ManagedThreadId
|
|||
|
{
|
|||
|
get
|
|||
|
{
|
|||
|
return this.controllerThread.ManagedThreadId;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
#endregion
|
|||
|
}
|
|||
|
}
|