當前位置: 首頁>>代碼示例>>C#>>正文


C# ILProcessor.Clone方法代碼示例

本文整理匯總了C#中Mono.Cecil.Cil.ILProcessor.Clone方法的典型用法代碼示例。如果您正苦於以下問題:C# ILProcessor.Clone方法的具體用法?C# ILProcessor.Clone怎麽用?C# ILProcessor.Clone使用的例子?那麽, 這裏精選的方法代碼示例或許可以為您提供幫助。您也可以進一步了解該方法所在Mono.Cecil.Cil.ILProcessor的用法示例。


在下文中一共展示了ILProcessor.Clone方法的1個代碼示例,這些例子默認根據受歡迎程度排序。您可以為喜歡或者感覺有用的代碼點讚,您的評價將有助於係統推薦出更棒的C#代碼示例。

示例1: ImplementBody

            protected override void ImplementBody(ILProcessor il, FieldReference methodInfoField, FieldReference propertyInfoField, MethodReference proceed, MethodReference proceedTargetMethod)
            {
                // If it's abstract, then the method is entirely implemented by the InvocationHandler
                if (Method.IsAbstract)
                {
                    base.ImplementBody(il, methodInfoField, propertyInfoField, proceed, proceedTargetMethod);
                }
                // Otherwise, it is implemented by the class itself, and calling this.Invocation().Proceed() calls the InvocationHandler
                else
                {
                    Method.Body.InitLocals = true;

                    // First declare the invocation in a private local variable
                    var invocation = new VariableDefinition(ClassWeaver.Context.InvocationType);
                    var instructions = Method.Body.Instructions.ToList();
                    Method.Body.Instructions.Clear();                   // Wipe out the existing instructions for the method
                    Method.Body.Variables.Add(invocation);              // Add a variable to store the invocation
                    EmitInvocation(il, methodInfoField, propertyInfoField, proceed);       // Put the invocation on the stack
                    il.Emit(OpCodes.Dup);                               // Duplicate invocation for below
                    il.Emit(OpCodes.Stloc, invocation);                 // Store the invocation in the variable declared (just) earlier

                    // Add the invocation to the end of the array
                    il.Emit(OpCodes.Call, ClassWeaver.Context.InvocationGetArguments);  // Array now on the stack with the invocation above it
                    il.Emit(OpCodes.Ldc_I4, Method.Parameters.Count);       // Array index
                    il.Emit(OpCodes.Ldloc, invocation);                     // Element value
                    il.Emit(OpCodes.Stelem_Any, ClassWeaver.Context.ModuleDefinition.TypeSystem.Object);  // Set array at index to element value

                    // Special instrumentation for async methods
                    var returnType = Method.ReturnType;
                    if (ClassWeaver.Context.TaskType.IsAssignableFrom(returnType))
                    {
                        // If the return type is Task<T>
                        if (returnType.IsTaskT())
                        {
                            var actualReturnType = returnType.GetTaskType();
                            var expectedAsyncBuilder = ClassWeaver.Context.AsyncTaskMethodBuilder.MakeGenericInstanceType(actualReturnType);

                            // Now find the call to .Start() (only will be found if we're async)
                            var startInstructionMethod = (GenericInstanceMethod)instructions
                                .Where(x => x.OpCode == OpCodes.Call)
                                .Select(x => (MethodReference)x.Operand)
                                .Where(x => x.IsGenericInstance && x.Name == "Start" && x.DeclaringType.CompareTo(expectedAsyncBuilder))
                                .SingleOrDefault();
                            if (startInstructionMethod != null)
                            {
                                var asyncType = startInstructionMethod.GenericArguments[0];
                                var invocationField = InstrumentAsyncType(asyncType);

                                // Find the first instruction that is setting a field on the async type.  We can just assume it's a three instruction 
                                // set (it always is in this context)  And we add our instructions to set the invocation field.
                                var nextSetFieldIndex = instructions.IndexOf(x => x.OpCode == OpCodes.Stfld && x.Operand is FieldDefinition && ((FieldDefinition)x.Operand).DeclaringType.CompareTo(asyncType));
                                if (nextSetFieldIndex == -1)
                                    throw new Exception($"Could not find expected stfld of async type: {asyncType}");
                                var setFieldLoadInstance = instructions[nextSetFieldIndex - 2];
                                instructions.Insert(nextSetFieldIndex - 2, il.Clone(setFieldLoadInstance));
                                instructions.Insert(nextSetFieldIndex - 1, il.Create(OpCodes.Ldloc, invocation));
                                instructions.Insert(nextSetFieldIndex, il.Create(OpCodes.Stfld, invocationField));
                            }
                        }
                    }

                    InstrumentInstructions(il, instructions, 2, x => x.Emit(OpCodes.Ldloc, invocation));
                }
            }
開發者ID:kswoll,項目名稱:sexy-proxy,代碼行數:64,代碼來源:ReverseProxyClassWeaver.cs


注:本文中的Mono.Cecil.Cil.ILProcessor.Clone方法示例由純淨天空整理自Github/MSDocs等開源代碼及文檔管理平台,相關代碼片段篩選自各路編程大神貢獻的開源項目,源碼版權歸原作者所有,傳播和使用請參考對應項目的License;未經允許,請勿轉載。