You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			257 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			257 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //----------------------------------------------------------------------------- | ||
|  | // Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | //----------------------------------------------------------------------------- | ||
|  | 
 | ||
|  | namespace System.Activities.Runtime | ||
|  | { | ||
|  |     using System; | ||
|  |     using System.Activities.DynamicUpdate; | ||
|  |     using System.Collections.ObjectModel; | ||
|  |     using System.Runtime; | ||
|  |     using System.Runtime.Serialization; | ||
|  |     using System.Security; | ||
|  | 
 | ||
|  |     // can't add FuncCompletionCallbackWrapper<T> since we don't know what to close the generic with | ||
|  |     [KnownType(typeof(ActivityCompletionCallbackWrapper))] | ||
|  |     [KnownType(typeof(DelegateCompletionCallbackWrapper))] | ||
|  |     [DataContract] | ||
|  |     abstract class CompletionCallbackWrapper : CallbackWrapper | ||
|  |     { | ||
|  |         static Type completionCallbackType = typeof(CompletionCallback); | ||
|  |         static Type[] completionCallbackParameters = new Type[] { typeof(NativeActivityContext), typeof(ActivityInstance) }; | ||
|  | 
 | ||
|  |         bool checkForCancelation; | ||
|  | 
 | ||
|  |         bool needsToGatherOutputs; | ||
|  | 
 | ||
|  |         protected CompletionCallbackWrapper(Delegate callback, ActivityInstance owningInstance) | ||
|  |             : base(callback, owningInstance) | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |         protected bool NeedsToGatherOutputs | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return this.needsToGatherOutputs; | ||
|  |             } | ||
|  | 
 | ||
|  |             set | ||
|  |             { | ||
|  |                 this.needsToGatherOutputs = value; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         [DataMember(EmitDefaultValue = false, Name = "checkForCancelation")] | ||
|  |         internal bool SerializedCheckForCancelation | ||
|  |         { | ||
|  |             get { return this.checkForCancelation; } | ||
|  |             set { this.checkForCancelation = value; } | ||
|  |         } | ||
|  | 
 | ||
|  |         [DataMember(EmitDefaultValue = false, Name = "needsToGatherOutputs")] | ||
|  |         internal bool SerializedNeedsToGatherOutputs | ||
|  |         { | ||
|  |             get { return this.needsToGatherOutputs; } | ||
|  |             set { this.needsToGatherOutputs = value; } | ||
|  |         } | ||
|  | 
 | ||
|  |         public void CheckForCancelation() | ||
|  |         { | ||
|  |             this.checkForCancelation = true; | ||
|  |         } | ||
|  | 
 | ||
|  |         protected virtual void GatherOutputs(ActivityInstance completedInstance) | ||
|  |         { | ||
|  |             // No-op in the base class | ||
|  |         } | ||
|  | 
 | ||
|  |         internal WorkItem CreateWorkItem(ActivityInstance completedInstance, ActivityExecutor executor) | ||
|  |         { | ||
|  |             // We use the property to guard against the virtual method call | ||
|  |             // since we don't need it in the common case | ||
|  |             if (this.NeedsToGatherOutputs) | ||
|  |             { | ||
|  |                 this.GatherOutputs(completedInstance); | ||
|  |             } | ||
|  | 
 | ||
|  |             CompletionWorkItem workItem; | ||
|  | 
 | ||
|  |             if (this.checkForCancelation) | ||
|  |             { | ||
|  |                 workItem = new CompletionWithCancelationCheckWorkItem(this, completedInstance); | ||
|  |             } | ||
|  |             else | ||
|  |             { | ||
|  |                 workItem = executor.CompletionWorkItemPool.Acquire(); | ||
|  |                 workItem.Initialize(this, completedInstance); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (completedInstance.InstanceMap != null) | ||
|  |             { | ||
|  |                 completedInstance.InstanceMap.AddEntry(workItem); | ||
|  |             } | ||
|  | 
 | ||
|  |             return workItem; | ||
|  |         } | ||
|  | 
 | ||
|  |         [Fx.Tag.SecurityNote(Critical = "Because any implementation will be calling EnsureCallback", | ||
|  |             Safe = "Safe because the method needs to be part of an Activity and we are casting to the callback type and it has a very specific signature. The author of the callback is buying into being invoked from PT.")] | ||
|  |         [SecuritySafeCritical] | ||
|  |         protected internal abstract void Invoke(NativeActivityContext context, ActivityInstance completedInstance); | ||
|  | 
 | ||
|  |         [DataContract] | ||
|  |         public class CompletionWorkItem : ActivityExecutionWorkItem, ActivityInstanceMap.IActivityReference | ||
|  |         { | ||
|  |             CompletionCallbackWrapper callbackWrapper; | ||
|  |             ActivityInstance completedInstance; | ||
|  | 
 | ||
|  |             // Called by the Pool. | ||
|  |             public CompletionWorkItem() | ||
|  |             { | ||
|  |                 this.IsPooled = true; | ||
|  |             } | ||
|  | 
 | ||
|  |             // Only used by non-pooled base classes. | ||
|  |             protected CompletionWorkItem(CompletionCallbackWrapper callbackWrapper, ActivityInstance completedInstance) | ||
|  |                 : base(callbackWrapper.ActivityInstance) | ||
|  |             { | ||
|  |                 this.callbackWrapper = callbackWrapper; | ||
|  |                 this.completedInstance = completedInstance; | ||
|  |             } | ||
|  | 
 | ||
|  |             protected ActivityInstance CompletedInstance | ||
|  |             { | ||
|  |                 get | ||
|  |                 { | ||
|  |                     return this.completedInstance; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             [DataMember(Name = "callbackWrapper")] | ||
|  |             internal CompletionCallbackWrapper SerializedCallbackWrapper | ||
|  |             { | ||
|  |                 get { return this.callbackWrapper; } | ||
|  |                 set { this.callbackWrapper = value; } | ||
|  |             } | ||
|  | 
 | ||
|  |             [DataMember(Name = "completedInstance")] | ||
|  |             internal ActivityInstance SerializedCompletedInstance | ||
|  |             { | ||
|  |                 get { return this.completedInstance; } | ||
|  |                 set { this.completedInstance = value; } | ||
|  |             } | ||
|  | 
 | ||
|  |             public void Initialize(CompletionCallbackWrapper callbackWrapper, ActivityInstance completedInstance) | ||
|  |             { | ||
|  |                 base.Reinitialize(callbackWrapper.ActivityInstance); | ||
|  |                 this.callbackWrapper = callbackWrapper; | ||
|  |                 this.completedInstance = completedInstance; | ||
|  |             } | ||
|  | 
 | ||
|  |             protected override void ReleaseToPool(ActivityExecutor executor) | ||
|  |             { | ||
|  |                 base.ClearForReuse(); | ||
|  |                 this.callbackWrapper = null; | ||
|  |                 this.completedInstance = null; | ||
|  |              | ||
|  |                 executor.CompletionWorkItemPool.Release(this); | ||
|  |             } | ||
|  | 
 | ||
|  |             public override void TraceCompleted() | ||
|  |             { | ||
|  |                 if (TD.CompleteCompletionWorkItemIsEnabled()) | ||
|  |                 { | ||
|  |                     TD.CompleteCompletionWorkItem(this.ActivityInstance.Activity.GetType().ToString(), this.ActivityInstance.Activity.DisplayName, this.ActivityInstance.Id, this.completedInstance.Activity.GetType().ToString(), this.completedInstance.Activity.DisplayName, this.completedInstance.Id); | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public override void TraceScheduled() | ||
|  |             { | ||
|  |                 if (TD.ScheduleCompletionWorkItemIsEnabled()) | ||
|  |                 { | ||
|  |                     TD.ScheduleCompletionWorkItem(this.ActivityInstance.Activity.GetType().ToString(), this.ActivityInstance.Activity.DisplayName, this.ActivityInstance.Id, this.completedInstance.Activity.GetType().ToString(), this.completedInstance.Activity.DisplayName, this.completedInstance.Id); | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public override void TraceStarting() | ||
|  |             { | ||
|  |                 if (TD.StartCompletionWorkItemIsEnabled()) | ||
|  |                 { | ||
|  |                     TD.StartCompletionWorkItem(this.ActivityInstance.Activity.GetType().ToString(), this.ActivityInstance.Activity.DisplayName, this.ActivityInstance.Id, this.completedInstance.Activity.GetType().ToString(), this.completedInstance.Activity.DisplayName, this.completedInstance.Id); | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             public override bool Execute(ActivityExecutor executor, BookmarkManager bookmarkManager) | ||
|  |             { | ||
|  |                 NativeActivityContext context = executor.NativeActivityContextPool.Acquire(); | ||
|  | 
 | ||
|  |                 Fx.Assert(this.completedInstance.Activity != null, "Activity definition should always be associated with an activity instance."); | ||
|  | 
 | ||
|  |                 try | ||
|  |                 { | ||
|  |                     context.Initialize(this.ActivityInstance, executor, bookmarkManager); | ||
|  |                     this.callbackWrapper.Invoke(context, this.completedInstance); | ||
|  |                 } | ||
|  |                 catch (Exception e) | ||
|  |                 { | ||
|  |                     if (Fx.IsFatal(e)) | ||
|  |                     { | ||
|  |                         throw; | ||
|  |                     } | ||
|  | 
 | ||
|  |                     this.ExceptionToPropagate = e; | ||
|  |                 } | ||
|  |                 finally | ||
|  |                 { | ||
|  |                     context.Dispose(); | ||
|  |                     executor.NativeActivityContextPool.Release(context); | ||
|  | 
 | ||
|  |                     if (this.ActivityInstance.InstanceMap != null) | ||
|  |                     { | ||
|  |                         this.ActivityInstance.InstanceMap.RemoveEntry(this); | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 return true; | ||
|  |             } | ||
|  | 
 | ||
|  |             Activity ActivityInstanceMap.IActivityReference.Activity | ||
|  |             { | ||
|  |                 get  | ||
|  |                 { | ||
|  |                     return this.completedInstance.Activity; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             void ActivityInstanceMap.IActivityReference.Load(Activity activity, ActivityInstanceMap instanceMap) | ||
|  |             { | ||
|  |                 if (this.completedInstance.Activity == null) | ||
|  |                 { | ||
|  |                     ((ActivityInstanceMap.IActivityReference)this.completedInstance).Load(activity, instanceMap); | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |         } | ||
|  | 
 | ||
|  |         [DataContract] | ||
|  |         class CompletionWithCancelationCheckWorkItem : CompletionWorkItem | ||
|  |         { | ||
|  |             public CompletionWithCancelationCheckWorkItem(CompletionCallbackWrapper callbackWrapper, ActivityInstance completedInstance) | ||
|  |                 : base(callbackWrapper, completedInstance) | ||
|  |             { | ||
|  |             } | ||
|  | 
 | ||
|  |             public override bool Execute(ActivityExecutor executor, BookmarkManager bookmarkManager) | ||
|  |             { | ||
|  |                 if (this.CompletedInstance.State != ActivityInstanceState.Closed && this.ActivityInstance.IsPerformingDefaultCancelation) | ||
|  |                 { | ||
|  |                     this.ActivityInstance.MarkCanceled(); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 return base.Execute(executor, bookmarkManager); | ||
|  |             } | ||
|  |         } | ||
|  |     } | ||
|  | } |