You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			243 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			243 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | #region Imports | ||
|  | 
 | ||
|  | using System; | ||
|  | using System.Xml; | ||
|  | using System.Collections; | ||
|  | using System.Collections.Generic; | ||
|  | using System.Collections.Specialized; | ||
|  | using System.Threading; | ||
|  | using System.Transactions; | ||
|  | using System.Reflection; | ||
|  | using System.Workflow.ComponentModel; | ||
|  | using System.Workflow.Runtime.Hosting; | ||
|  | using System.Diagnostics; | ||
|  | 
 | ||
|  | #endregion | ||
|  | 
 | ||
|  | namespace System.Workflow.Runtime | ||
|  | { | ||
|  |     #region Class AmbientEnvironment | ||
|  | 
 | ||
|  |     internal abstract class AmbientEnvironment : IDisposable | ||
|  |     { | ||
|  |         /// <summary> | ||
|  |         /// Indicates that the value of a static field is unique for each thread | ||
|  |         /// CLR Perf suggests using this attribute over the slot approach. | ||
|  |         /// </summary> | ||
|  |         [ThreadStatic()] | ||
|  |         static EnvWrapper threadData; | ||
|  | 
 | ||
|  |         readonly object _prevEnv; | ||
|  |         readonly int _prevRC; | ||
|  | 
 | ||
|  |         protected AmbientEnvironment(object env) | ||
|  |         { | ||
|  |             if (threadData == null) | ||
|  |             { | ||
|  |                 //Setting TLS for the first time | ||
|  |                 threadData = new EnvWrapper(); | ||
|  |             } | ||
|  | 
 | ||
|  |             threadData.Push(env, out _prevEnv, out _prevRC); | ||
|  |         } | ||
|  | 
 | ||
|  |         void IDisposable.Dispose() | ||
|  |         { | ||
|  |             Debug.Assert(null != threadData); | ||
|  |             threadData.Pop(_prevEnv, _prevRC); | ||
|  |             if (_prevRC == 0) | ||
|  |             { | ||
|  |                 threadData = null; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static object Retrieve() | ||
|  |         { | ||
|  |             if (threadData != null) | ||
|  |                 return threadData.Retrieve(); | ||
|  |             else | ||
|  |                 return null; | ||
|  |         } | ||
|  | 
 | ||
|  |         private class EnvWrapper | ||
|  |         { | ||
|  |             int _rc; | ||
|  | 
 | ||
|  |             object _currEnv; | ||
|  | 
 | ||
|  |             internal void Push(object env, out object prevEnv, out int prevRc) | ||
|  |             { | ||
|  |                 Debug.Assert(_rc >= 0); | ||
|  |                 prevEnv = _currEnv; | ||
|  |                 prevRc = _rc; | ||
|  |                 _rc++; | ||
|  |                 _currEnv = env; | ||
|  |             } | ||
|  | 
 | ||
|  |             internal void Pop(object prevEnv, int prevRC) | ||
|  |             { | ||
|  |                 Debug.Assert(_rc > 0); | ||
|  |                 _rc--; | ||
|  |                 _currEnv = prevEnv; | ||
|  |                 if (_rc != prevRC) | ||
|  |                 { | ||
|  |                     Debug.Assert(false); | ||
|  |                     // | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             internal object Retrieve() | ||
|  |             { | ||
|  |                 Debug.Assert(_rc > 0); | ||
|  |                 return _currEnv; | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     #endregion | ||
|  | 
 | ||
|  |     #region Class ServiceEnvironment | ||
|  | 
 | ||
|  |     // This class presents the transactional view of a WF instance: | ||
|  |     // mainly the current batch in transaction, and NOT the runtime view | ||
|  |     // of currently executing activity. | ||
|  |     internal sealed class ServiceEnvironment : AmbientEnvironment | ||
|  |     { | ||
|  |         internal static readonly Guid debuggerThreadGuid = new Guid("54D747AE-5CC6-4171-95C8-0A8C40443915"); | ||
|  | 
 | ||
|  |         internal ServiceEnvironment(Activity currentActivity) | ||
|  |             : base(currentActivity) | ||
|  |         { | ||
|  |             GC.SuppressFinalize(this); | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static IWorkBatch WorkBatch | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 Activity currentActivity = ServiceEnvironment.CurrentActivity; | ||
|  |                 if (currentActivity == null) | ||
|  |                     return null; | ||
|  |                 else | ||
|  |                     return (IWorkBatch)currentActivity.GetValue(WorkflowExecutor.TransientBatchProperty); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static Guid WorkflowInstanceId | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 Activity currentActivity = ServiceEnvironment.CurrentActivity; | ||
|  |                 if (currentActivity == null) | ||
|  |                     return Guid.Empty; | ||
|  | 
 | ||
|  |                 return ((Guid)ContextActivityUtils.RootContextActivity(currentActivity).GetValue(WorkflowExecutor.WorkflowInstanceIdProperty)); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static WorkflowQueuingService QueuingService | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 Activity currentActivity = ServiceEnvironment.CurrentActivity; | ||
|  | 
 | ||
|  |                 // fetch workflow executor | ||
|  |                 IWorkflowCoreRuntime workflowExecutor = null; | ||
|  |                 if (currentActivity != null) | ||
|  |                     workflowExecutor = ContextActivityUtils.RetrieveWorkflowExecutor(currentActivity); | ||
|  | 
 | ||
|  |                 while (currentActivity != null) | ||
|  |                 { | ||
|  |                     if (currentActivity == workflowExecutor.CurrentAtomicActivity) | ||
|  |                     { | ||
|  |                         TransactionalProperties transactionalProperties = (TransactionalProperties)currentActivity.GetValue(WorkflowExecutor.TransactionalPropertiesProperty); | ||
|  |                         if (transactionalProperties != null) | ||
|  |                         { | ||
|  |                             if (transactionalProperties.LocalQueuingService != null) | ||
|  |                             { | ||
|  |                                 WorkflowQueuingService queuingService = transactionalProperties.LocalQueuingService; | ||
|  |                                 return queuingService; // return local queuing service | ||
|  |                             } | ||
|  |                         } | ||
|  |                     } | ||
|  |                     currentActivity = currentActivity.Parent; | ||
|  |                 } | ||
|  |                 return null; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         // DO NOT change this to internal/public | ||
|  |         // Technically we only want to store the Batch in the TLS,  | ||
|  |         // but because we also want the queueing service and instId,  | ||
|  |         // we are storing the object encapsulating all the info. | ||
|  |         // The service environment only represents the transactional view: | ||
|  |         // the current batch; and not the current executing view: | ||
|  |         // e.g. some caller/caller, send/receive scenarios where  | ||
|  |         // we want the current batch to be the caller's so this activity  | ||
|  |         // does not reflect the executing activity (the callee). | ||
|  |         private static Activity CurrentActivity | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 object o = AmbientEnvironment.Retrieve(); | ||
|  |                 return o as Activity; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static bool IsInServiceThread(Guid instanceId) | ||
|  |         { | ||
|  |             System.Diagnostics.Debug.Assert(instanceId != Guid.Empty, "IsInServiceThread expects valid guid."); | ||
|  |             if (WorkflowInstanceId == instanceId) | ||
|  |                 return true; | ||
|  | 
 | ||
|  |             return DebuggerThreadMarker.IsInDebuggerThread(); | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     #endregion | ||
|  | 
 | ||
|  |     #region Class DebuggerThreadMarker | ||
|  | 
 | ||
|  |     internal class DebuggerThreadMarker : AmbientEnvironment | ||
|  |     { | ||
|  |         public DebuggerThreadMarker() | ||
|  |             : base(new object()) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static bool IsInDebuggerThread() | ||
|  |         { | ||
|  |             return AmbientEnvironment.Retrieve() != null; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     #endregion | ||
|  | 
 | ||
|  |     #region Class RuntimeEnvironment | ||
|  | 
 | ||
|  |     internal class RuntimeEnvironment : IDisposable | ||
|  |     { | ||
|  |         [ThreadStatic()] | ||
|  |         static WorkflowRuntime workflowRuntime; | ||
|  | 
 | ||
|  |         public RuntimeEnvironment(WorkflowRuntime runtime) | ||
|  |         { | ||
|  |             workflowRuntime = runtime; | ||
|  |         } | ||
|  | 
 | ||
|  |         internal static WorkflowRuntime CurrentRuntime | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return RuntimeEnvironment.workflowRuntime; | ||
|  |             } | ||
|  |         } | ||
|  |         void IDisposable.Dispose() | ||
|  |         { | ||
|  |             workflowRuntime = null; | ||
|  |         } | ||
|  |     } | ||
|  | 
 | ||
|  |     #endregion | ||
|  | } |