You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			273 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			273 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //-----------------------------------------------------------------------------
 | |
| // Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| //-----------------------------------------------------------------------------
 | |
| 
 | |
| namespace System.ServiceModel.Dispatcher
 | |
| {
 | |
|     using System;
 | |
|     using System.Reflection;
 | |
|     using System.Runtime.Diagnostics;
 | |
|     using System.ServiceModel.Diagnostics;
 | |
|     using System.ServiceModel.Diagnostics.Application;
 | |
|     using System.Globalization;
 | |
|     using System.Threading;
 | |
|     using System.Collections;
 | |
|     using System.Diagnostics;
 | |
|     using System.Security;
 | |
|     using System.Runtime;
 | |
| 
 | |
|     class SyncMethodInvoker : IOperationInvoker
 | |
|     {
 | |
|         Type type;
 | |
|         string methodName;
 | |
|         MethodInfo method;
 | |
|         InvokeDelegate invokeDelegate;
 | |
|         int inputParameterCount;
 | |
|         int outputParameterCount;
 | |
| 
 | |
|         public SyncMethodInvoker(MethodInfo method)
 | |
|         {
 | |
|             if (method == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("method"));
 | |
| 
 | |
|             this.method = method;
 | |
|         }
 | |
| 
 | |
|         public SyncMethodInvoker(Type type, string methodName)
 | |
|         {
 | |
|             if (type == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("type"));
 | |
| 
 | |
|             if (methodName == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentNullException("methodName"));
 | |
| 
 | |
|             this.type = type;
 | |
|             this.methodName = methodName;
 | |
|         }
 | |
| 
 | |
|         public bool IsSynchronous
 | |
|         {
 | |
|             get { return true; }
 | |
|         }
 | |
| 
 | |
|         public MethodInfo Method
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (method == null)
 | |
|                     method = type.GetMethod(methodName);
 | |
|                 return method;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public string MethodName
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (methodName == null)
 | |
|                     methodName = method.Name;
 | |
|                 return methodName;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public object[] AllocateInputs()
 | |
|         {
 | |
|             EnsureIsInitialized();
 | |
| 
 | |
|             return EmptyArray.Allocate(this.inputParameterCount);
 | |
|         }
 | |
| 
 | |
|         public object Invoke(object instance, object[] inputs, out object[] outputs)
 | |
|         {
 | |
|             EnsureIsInitialized();
 | |
| 
 | |
|             if (instance == null)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxNoServiceObject)));
 | |
|             if (inputs == null)
 | |
|             {
 | |
|                 if (this.inputParameterCount > 0)
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxInputParametersToServiceNull, this.inputParameterCount)));
 | |
|             }
 | |
|             else if (inputs.Length != this.inputParameterCount)
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(SR.GetString(SR.SFxInputParametersToServiceInvalid, this.inputParameterCount, inputs.Length)));
 | |
| 
 | |
|             outputs = EmptyArray.Allocate(this.outputParameterCount);
 | |
| 
 | |
|             long startCounter = 0;
 | |
|             long stopCounter = 0;
 | |
|             long beginOperation = 0;
 | |
|             bool callSucceeded = false;
 | |
|             bool callFaulted = false;
 | |
| 
 | |
|             if (PerformanceCounters.PerformanceCountersEnabled)
 | |
|             {
 | |
|                 PerformanceCounters.MethodCalled(this.MethodName);
 | |
|                 try
 | |
|                 {
 | |
|                     if (System.ServiceModel.Channels.UnsafeNativeMethods.QueryPerformanceCounter(out startCounter) == 0)
 | |
|                     {
 | |
|                         startCounter = -1;
 | |
|                     }
 | |
|                 }
 | |
|                 catch (SecurityException securityException)
 | |
|                 {
 | |
|                     DiagnosticUtility.TraceHandledException(securityException, TraceEventType.Warning);
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(
 | |
|                         new SecurityException(SR.GetString(
 | |
|                                 SR.PartialTrustPerformanceCountersNotEnabled), securityException));
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             EventTraceActivity eventTraceActivity = null;
 | |
|             if (TD.OperationCompletedIsEnabled() ||
 | |
|                     TD.OperationFaultedIsEnabled() ||
 | |
|                     TD.OperationFailedIsEnabled())
 | |
|             {
 | |
|                 beginOperation = DateTime.UtcNow.Ticks;
 | |
|                 OperationContext context = OperationContext.Current;
 | |
|                 if (context != null && context.IncomingMessage != null)
 | |
|                 {
 | |
|                     eventTraceActivity = EventTraceActivityHelper.TryExtractActivity(context.IncomingMessage);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             object returnValue;
 | |
|             try
 | |
|             {
 | |
|                 ServiceModelActivity activity = null;
 | |
|                 IDisposable boundActivity = null;
 | |
|                 if (DiagnosticUtility.ShouldUseActivity)
 | |
|                 {
 | |
|                     activity = ServiceModelActivity.CreateBoundedActivity(true);
 | |
|                     boundActivity = activity;
 | |
|                 }
 | |
|                 else if (TraceUtility.MessageFlowTracingOnly)
 | |
|                 {
 | |
|                     Guid activityId = TraceUtility.GetReceivedActivityId(OperationContext.Current);
 | |
|                     if (activityId != Guid.Empty)
 | |
|                     {
 | |
|                         DiagnosticTraceBase.ActivityId = activityId;
 | |
|                     }
 | |
|                 }
 | |
|                 else if (TraceUtility.ShouldPropagateActivity)
 | |
|                 {
 | |
|                     //Message flow tracing only scenarios use a light-weight ActivityID management logic
 | |
|                     Guid activityId = ActivityIdHeader.ExtractActivityId(OperationContext.Current.IncomingMessage);
 | |
|                     if (activityId != Guid.Empty)
 | |
|                     {
 | |
|                         boundActivity = Activity.CreateActivity(activityId);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 using (boundActivity)
 | |
|                 {
 | |
|                     if (DiagnosticUtility.ShouldUseActivity)
 | |
|                     {
 | |
|                         ServiceModelActivity.Start(activity, SR.GetString(SR.ActivityExecuteMethod, this.method.DeclaringType.FullName, this.method.Name), ActivityType.ExecuteUserCode);
 | |
|                     }
 | |
|                     if (TD.OperationInvokedIsEnabled())
 | |
|                     {
 | |
|                         TD.OperationInvoked(eventTraceActivity, this.MethodName, TraceUtility.GetCallerInfo(OperationContext.Current));
 | |
|                     }
 | |
|                     returnValue = this.invokeDelegate(instance, inputs, outputs);
 | |
|                     callSucceeded = true;
 | |
|                 }
 | |
|             }
 | |
|             catch (System.ServiceModel.FaultException)
 | |
|             {
 | |
|                 callFaulted = true;
 | |
|                 throw;
 | |
|             }
 | |
|             catch (System.Security.SecurityException e)
 | |
|             {
 | |
|                 DiagnosticUtility.TraceHandledException(e, TraceEventType.Warning);
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(AuthorizationBehavior.CreateAccessDeniedFaultException());
 | |
|             }
 | |
|             finally
 | |
|             {
 | |
|                 if (PerformanceCounters.PerformanceCountersEnabled)
 | |
|                 {
 | |
|                     long elapsedTime = 0;
 | |
|                     if (startCounter >= 0 && System.ServiceModel.Channels.UnsafeNativeMethods.QueryPerformanceCounter(out stopCounter) != 0)
 | |
|                     {
 | |
|                         elapsedTime = stopCounter - startCounter;
 | |
|                     }
 | |
| 
 | |
|                     if (callSucceeded) // call succeeded
 | |
|                     {
 | |
|                         PerformanceCounters.MethodReturnedSuccess(this.MethodName, elapsedTime);
 | |
|                     }
 | |
|                     else if (callFaulted) // call faulted
 | |
|                     {
 | |
|                         PerformanceCounters.MethodReturnedFault(this.MethodName, elapsedTime);
 | |
|                     }
 | |
|                     else // call failed
 | |
|                     {
 | |
|                         PerformanceCounters.MethodReturnedError(this.MethodName, elapsedTime);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (beginOperation != 0)
 | |
|                 {
 | |
|                     if (callSucceeded)
 | |
|                     {
 | |
|                         if (TD.OperationCompletedIsEnabled())
 | |
|                         {
 | |
|                             TD.OperationCompleted(eventTraceActivity, this.methodName,
 | |
|                                 TraceUtility.GetUtcBasedDurationForTrace(beginOperation));
 | |
|                         }
 | |
|                     }
 | |
|                     else if (callFaulted)
 | |
|                     {
 | |
|                         if (TD.OperationFaultedIsEnabled())
 | |
|                         {
 | |
|                             TD.OperationFaulted(eventTraceActivity, this.methodName,
 | |
|                                 TraceUtility.GetUtcBasedDurationForTrace(beginOperation));
 | |
|                         }
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         if (TD.OperationFailedIsEnabled())
 | |
|                         {
 | |
|                             TD.OperationFailed(eventTraceActivity, this.methodName,
 | |
|                                 TraceUtility.GetUtcBasedDurationForTrace(beginOperation));
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return returnValue;
 | |
|         }
 | |
| 
 | |
|         public IAsyncResult InvokeBegin(object instance, object[] inputs, AsyncCallback callback, object state)
 | |
|         {
 | |
|             throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new NotImplementedException());
 | |
|         }
 | |
| 
 | |
|         public object InvokeEnd(object instance, out object[] outputs, IAsyncResult result)
 | |
|         {
 | |
|             throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new NotImplementedException());
 | |
|         }
 | |
| 
 | |
|         void EnsureIsInitialized()
 | |
|         {
 | |
|             if (this.invokeDelegate == null)
 | |
|             {
 | |
|                 EnsureIsInitializedCore();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         void EnsureIsInitializedCore()
 | |
|         {
 | |
|             // Only pass locals byref because InvokerUtil may store temporary results in the byref.
 | |
|             // If two threads both reference this.count, temporary results may interact.
 | |
|             int inputParameterCount;
 | |
|             int outputParameterCount;
 | |
|             InvokeDelegate invokeDelegate = new InvokerUtil().GenerateInvokeDelegate(this.Method, out inputParameterCount, out outputParameterCount);
 | |
|             this.outputParameterCount = outputParameterCount;
 | |
|             this.inputParameterCount = inputParameterCount;
 | |
|             this.invokeDelegate = invokeDelegate;  // must set this last due to ----
 | |
|         }
 | |
|     }
 | |
| }
 |