C#[MSIL EMIT+] VirtualProxy 动态程序集与动态类型绑定

[assembly: RuntimeCompatibility(WrapNonExceptionThrows = true)]
[assembly: AllowPartiallyTrustedCallers]
[assembly: CLSCompliant(true)]
[assembly: NeutralResourcesLanguageAttribute("zh-CN")]
[assembly: CompilationRelaxations(CompilationRelaxations.NoStringInterning)]
[assembly: Debuggable(DebuggableAttribute.DebuggingModes.IgnoreSymbolStoreSequencePoints]
[assembly: StringFreezing]
[assembly: SecurityPermission(SecurityAction.RequestMinimum, SkipVerification = true)]
    using System;
    using System.Diagnostics;
    using System.IO;
    using System.Reflection;
    using System.Reflection.Emit;
    using System.Runtime.InteropServices;
    using System.Security;

    [TypeLibType(TypeLibTypeFlags.FHidden | TypeLibTypeFlags.FRestricted)]
    public static partial class VirtualProxy
    {
        [DebuggerBrowsable(DebuggerBrowsableState.Never)]
        internal static ModuleBuilder MainDynamicModule
        {
            get;
            set;
        }

        [SecuritySafeCritical]
        static VirtualProxy()
        {
            VirtualProxy.MainDynamicModule = VirtualProxy.CreateDynamicModule();
        }

        [SecuritySafeCritical]
        private static ModuleBuilder CreateDynamicModule()
        {
            AssemblyBuilder _asm = AppDomain.CurrentDomain.DefineDynamicAssembly(new AssemblyName(Path.GetRandomFileName()), AssemblyBuilderAccess.RunAndSave);
            return _asm.DefineDynamicModule(_asm.FullName);
        }

        [SecuritySafeCritical]
        private static TypeBuilder CreateDynamicType()
        {
            return VirtualProxy.MainDynamicModule.DefineType(Path.GetRandomFileName(), TypeAttributes.Public | TypeAttributes.Abstract | TypeAttributes.Sealed, null, null);
        }

        [SecuritySafeCritical]
        internal static void CreateDynamicMethods(Type objRefType)
        {
            MethodInfo[] _mets = objRefType.GetMethods();
            foreach (MethodInfo _met in _mets)
            {
                if (_met.DeclaringType == objRefType)
                {
                    VirtualProxy.CreateDynamicMethod(_met, VirtualProxy.CreateDynamicType());
                }
            }
        }

        [SecuritySafeCritical]
        private static Type[] GetDynamicParameterType(ParameterInfo[] args)
        {
            Type[] _params = new Type[args.Length + 1];
            for (int i = 1; i <= args.Length; i++)
                _params[i] = args[i - 1].ParameterType;
            _params[0] = typeof(object); return _params;
        }

        [SuppressUnmanagedCodeSecurity]
        private static void CreateDynamicMethod(MethodInfo met, TypeBuilder builder)
        {
            ParameterInfo[] _params = met.GetParameters();
            MethodBuilder _callback = builder.DefineMethod(builder.Name, MethodAttributes.Public | MethodAttributes.Static, CallingConventions.Standard, met.ReturnType, VirtualProxy.GetDynamicParameterType(_params));
            ILGenerator _il = _callback.GetILGenerator();
            _il.DeclareLocal(typeof(object[]));
            // 必须过滤VOID型,否则引发CLR/JIT内部错误
            if (met.ReturnType != typeof(void))
            {
                _il.DeclareLocal(met.ReturnType);
            }

            _il.Emit(OpCodes.Nop); // PUSH ECX
            _il.Emit(OpCodes.Ldc_I4, _params.Length + 2);
            _il.Emit(OpCodes.Newarr, typeof(object));
            _il.Emit(OpCodes.Stloc_0);

            _il.Emit(OpCodes.Ldloc_0);
            _il.Emit(OpCodes.Ldc_I4, 0);
            _il.Emit(OpCodes.Ldarg, 0);
            _il.Emit(OpCodes.Box, typeof(object));
            _il.Emit(OpCodes.Stelem_Ref);

            for (int i = 1; i <= _params.Length; i++)  // PUSH EBX+(8 + i * sizeof(void*))
            {
                _il.Emit(OpCodes.Ldloc_0); // 无论如何,您不可以BOX与调用PARAM不同类型
                _il.Emit(OpCodes.Ldc_I4, i);
                _il.Emit(OpCodes.Ldarg, i);
                _il.Emit(OpCodes.Box, _params[i - 1].ParameterType);
                _il.Emit(OpCodes.Stelem_Ref);
            }

            _il.Emit(OpCodes.Ldloc_0);
            _il.Emit(OpCodes.Ldc_I4, _params.Length + 1);
            _il.Emit(OpCodes.Ldstr, met.Name);
            _il.Emit(OpCodes.Box, typeof(object));
            _il.Emit(OpCodes.Stelem_Ref);

            _il.Emit(OpCodes.Ldloc_0);
            _il.Emit(OpCodes.Call, typeof(VirtualProxy).GetMethod("VirtualInvokeMember"));

            if (met.ReturnType == typeof(void)) // MOV EAX, DWORD PTR[EBP-(8 + sizeof(void*)]
            {
                _il.Emit(OpCodes.Pop);
            }
            else // 不可以移除现行计算堆栈顶部的(EAX)的值,一旦使用会引发CLR ClrSystemExecuteEngine发生难以解决的故障
            {
                _il.Emit(OpCodes.Unbox_Any, met.ReturnType);
                _il.Emit(OpCodes.Stloc_1);
                _il.Emit(OpCodes.Ldloc_1);
            }

// ExecutionEngineException

            _il.Emit(OpCodes.Ret);

            DirectedManagedMethods _metredir = new DirectedManagedMethods(); // 函数流程导向
            _metredir.Install(met.MethodHandle.GetFunctionPointer(), (builder.CreateType()).GetMethod(builder.Name));
        }

        [SuppressUnmanagedCodeSecurity]
        public static object VirtualInvokeMember(object[] args) // 您不可以修改函数返回值类型,不论被导向.NET函数需不需要返回
        {
            int len = args.Length - 1;
            object[] _retVal = new object[len - 1];
            for (int i = 1; i < len; i++)
            {
                _retVal[i - 1] = args[i];
            }
            object _sender = args[0];
            IRefObjProvider _refObj = (IRefObjProvider)_sender;
            return _refObj.InvokeMember(_sender, (string)args[len], _retVal);
        }
    }

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
MSIL(Microsoft Intermediate Language)权威指南是一本关于MSIL的详细指南,它是用于.NET平台的中间语言。MSIL权威指南的PDF版本是一份便携式文档,可以在电脑或其他设备上方便地阅读。这本指南提供了关于MSIL语法、结构和操作码的全面介绍。 MSIL是一种与平台无关的字节码,它是在将源代码编译成可执行代码之前生成的。通过学习和理解MSIL,开发人员可以更好地理解.Net应用程序的内部运作机制。这对于开发高效、可靠和安全的应用程序是至关重要的。 在MSIL权威指南中,读者将了解到如何编写和调试MSIL代码,以及如何使用不同的工具来检查和分析MSIL指令。此外,该指南还介绍了如何使用反汇编器来查看已编译的代码,并通过示例代码和说明展示了许多常见的MSIL编程场景。 通过PDF格式,读者可以随时随地访问MSIL权威指南,无论是在写代码时需要参考,还是在学习和研究MSIL时需要深入研究。此外,PDF格式还方便读者在电子设备上进行搜索、书签和注释,以便更好地管理和使用这个权威指南。 总之,MSIL权威指南的PDF版本提供了一种方便、可靠且易于阅读的方式,使开发人员了解和掌握MSIL的核心概念和技术。无论是初学者还是有经验的开发人员,都可以从这本指南中获得宝贵的知识和技巧,以提升他们的开发技能和应用程序的质量。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值