4

我创建了一个动态方法,但是当我尝试访问外部资源时,它给了我一个异常

TargetInvocationException:调用的目标已引发异常。

基本上我想写一个字节数组的方法,然后将它作为动态方法加载。我知道一个简单的字节数组不足以重建元数据链接,但是如何制作一个使用来自同一程序集的变量的动态方法?

我试图转换该代码:

public static int z = 10;
public static int sum(int x, int y) {
    return x + y + z;
}

这给了我IL:

0        L_0000:      ldarg.0      
1        L_0001:      ldarg.1      
2        L_0002:      add          
3        L_0003:      ldsfld       System.Int32 CodeGen.Program::z
4        L_0008:      add          
5        L_0009:      ret  

这是字节:

02 03 58 7E 06 00 00 04 58 2A

我是这样测试的:

public static int z = 10;

static void Main(string[] args) {

    DynamicMethod sum = new DynamicMethod("sum", typeof(int), new Type[] { typeof(int), typeof(int) });
    var info = sum.GetDynamicILInfo();
    var bytes = new byte[] { 0x02, 0x03, 0x58, 0x7E, 0x06, 0x00, 0x00, 0x04, 0x58, 0x2A }; // { 0x02, 0x17, 0x58, 0x2A }; // this is a code for int sum(int x, int y) { return x + y; }

    info.SetCode(bytes, 8);

    var sig = SignatureHelper.GetMethodSigHelper(CallingConventions.Standard, typeof(int));
    byte[] bsig = sig.GetSignature();
    bsig[0] = 0x7;

    info.SetLocalSignature(bsig);

    int x = (int) sum.Invoke(null, new object[] { 10, 20 });
    Console.WriteLine(x.ToString());
    Console.ReadLine();
}

TL;DR 我想修复代表动态方法 IL 的字节数组的引用。怎么做?另外,我不想使用 ILGenerator(),我想要一个字节数组。

4

1 回答 1

4

元数据令牌不能按原样重复使用;相反,您需要使用DynamicILInfo.GetTokenFor来获取与您可以使用的字段相对应的新令牌。在您的情况下,这看起来像这样:

var tok = info.GetTokenFor(typeof(...).GetField("z").FieldHandle);
bytes[4] = tok;    
bytes[5] = tok >> 8;
bytes[6] = tok >> 16;
bytes[7] = tok >> 24;

但是,您可能还需要创建您的DynamicMethod,以便禁用 JIT 可见性检查,否则您的方法将无法访问私有字段。

在一个不相关的说明中,我不清楚你在用签名信息做什么 - 看起来你出于某种原因将局部变量签名设置为修改后的方法签名......我认为你应该做更多的事情像这样:

info.SetLocalSignature(SignatureHelper.GetLocalVarSigHelper().GetSignature())

当然,您需要根据您的方法实际使用的本地人来改变这一点。

于 2012-08-22T16:47:44.253 回答