Xamarin Public Jenkins (auto-signing) e79aa3c0ed Imported Upstream version 4.6.0.125
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
2016-08-03 10:59:49 +00:00

183 lines
5.7 KiB
C#

//-----------------------------------------------------------------------------
// Copyright (c) Microsoft Corporation. All rights reserved.
//-----------------------------------------------------------------------------
namespace System.Activities.Debugger
{
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Diagnostics.CodeAnalysis;
using System.Diagnostics.SymbolStore;
using System.Reflection;
using System.Reflection.Emit;
using System.Runtime;
using System.Threading;
// Define an auxiliary thread codegen technique for islands.
// This executes the islands on a dedicated worker thread. The worker thread's
// physical callstack then maps to the interpreter's virtual callstack.
// It has an excellent Step-in/Step-over/Step-Out experience.
[DebuggerNonUserCode]
public class ThreadWorkerController
{
StateManager stateManager;
// Set to true to notify to Break on first instruction. This helps the F11 on startup experience.
// Since the islands are on a new thread, there may be no user code on the main thread and so
// F11 doesn't work. Thus the new worker thread needs to fire some break event.
// This gets reset after the 'startup breakpoint'.
// The initial Properties can override this.
bool breakOnStartup;
// Own the worker thread.
Thread worker;
// Signalled when the main thread wants to send an event to the worker thread.
// The main thread fills out the data first.
AutoResetEvent eventSend;
// Signalled by the worker thread when it's finished handling the event and
// the main thread can resume.
AutoResetEvent eventDone;
EventCode eventCode;
// Parameter for enter message.
VirtualStackFrame enterStackParameter;
internal void Initialize(string threadName, StateManager manager)
{
this.stateManager = manager;
this.breakOnStartup = this.stateManager.ManagerProperties.BreakOnStartup;
CreateWorkerThread(threadName);
}
internal void Exit()
{
// Implement with an unbalanced Leave.
// This will get the Worker to return and the ThreadProc to exit.
this.LeaveState();
this.worker.Join();
}
[DebuggerHidden]
void WorkerThreadProc()
{
Worker(false);
this.eventDone.Set();
}
// Private Entry point called from islands. Must be public so that the islands can invoke it.
[Fx.Tag.InheritThrows(From = "Worker")]
[DebuggerHidden]
public static void IslandWorker(ThreadWorkerController controller)
{
if (controller == null)
{
throw FxTrace.Exception.ArgumentNull("controller");
}
controller.Worker(true);
}
[DebuggerHidden]
internal void Worker(bool isAtStartup)
{
if (isAtStartup)
{
// Fire the 1-time "startup" breakpoint.
if (this.breakOnStartup)
{
if (Debugger.IsAttached)
{
Debugger.Break();
}
this.breakOnStartup = false;
}
this.eventDone.Set();
}
// The final terminator is when leave returns, but from a recursive call.
bool leave = false;
while (!leave)
{
this.eventSend.WaitOne();
switch (eventCode)
{
case EventCode.Enter:
// Call Island for enterStackParameter
this.stateManager.InvokeWorker(this, enterStackParameter);
// resume from SendLeave()
this.eventDone.Set();
break;
case EventCode.Leave:
leave = true;
return;
case EventCode.Break:
Debugger.Break();
this.eventDone.Set();
break;
}
}
}
void CreateWorkerThread(string threadName)
{
this.eventSend = new AutoResetEvent(false);
this.eventDone = new AutoResetEvent(false);
this.worker = new Thread(new ThreadStart(WorkerThreadProc));
string name = string.IsNullOrEmpty(threadName) ? this.stateManager.ManagerProperties.AuxiliaryThreadName : threadName;
if (name != null)
{
this.worker.Name = name;
}
this.worker.Start();
}
internal void EnterState(VirtualStackFrame newFrame)
{
this.eventCode = EventCode.Enter;
this.enterStackParameter = newFrame;
this.eventSend.Set();
// Block until Island executes Nop,
// giving BPs a chance to be hit.
// Must block here if the island is stopped at a breakpoint.
this.eventDone.WaitOne();
}
internal void LeaveState()
{
this.eventCode = EventCode.Leave;
this.eventSend.Set();
// Block until call has exited.
this.eventDone.WaitOne();
}
internal void Break()
{
this.eventCode = EventCode.Break;
this.eventSend.Set();
this.eventDone.WaitOne();
}
// Type of event being fired.
enum EventCode
{
Enter,
Leave,
Break
};
}
}