Mono.Cecil:从字符串创建指令

时间:2016-10-30 06:08:04

标签: c# mono.cecil

我怎样才能像这样转换字符串:

"call        System.Console.WriteLine" 
"ldstr       \"hello\""

进入带有操作数的Intructions?

1 个答案:

答案 0 :(得分:1)

如果你现在如何使用Mono.Cecil(或Reflection.Emit),那么更常见的问题是将文本解析为代码操作。

你有一些方法可以做到这一点,我可以给你一个提示,你可以选择自己的方式。

首先,您需要一些先决条件(如果IL文本是有效的IL代码,则这些先决条件已经存在)。 例如,您无法猜出Console.WriteLine是什么。 Console是一个程序集,类型,方法?关于WriteLine的相同问题。即使我们知道WriteLine是一种方法,我们需要选择哪种超载?什么是泛型?因此,您需要设置一个合约,例如定义点是分隔符,第一部分是程序集,第二部分是命名空间,依此类推。

例如:

"mscorlib.System.Console.WriteLine(string)"将被翻译为System.Console.WriteLine(string)

签订严格的合同后,您需要几个步骤(对于WriteLine示例):

  1. 解析Assembly并获取ModuleDefinition
  2. 解析并导入Type
  3. 解析并导入MethodReference描述所请求的方法
  4. 发出通话
  5. 实现这一目标的一种方法是保持操作码的结构及其所需的操作。

    例如,我们知道call指令需要发出对静态方法的调用(在大多数情况下),所以你需要将操作数发送到ParseStaticCall,你需要在那里解析string并发出调用指令

    伪代码:

    new Dictionary<string, Tuple<OpCode, Action<string>>>
    {
        {
            "call",
            Tuple.Create<OpCode, Action<string>>
              (OpCodes.Call,ParseStaticCall)
        }
    };
    
    
    static void ParseStaticCall(Opcpde opcode,
                                string call,
                                ILProcessor processor)
    {
        string assembly, namespaceName, type, method;
        int numOfParameters;
    
        var moduleDefenition = AssemblyResolver.Resolve(assembly).MainModule;
    
        var methodReference = 
            new ReferenceFinder(moduleDefenition).
                GetMethodReference(typeof (Console),
                    md => md.Name == methodName &&
                    md.Parameters.Count == numOfParameters);
            processor.Emit(opcode, methodReference);
    }
    

    AssemblyResolver是一个帮助程序类,用于按名称和路径查找程序集(可以是常量路径)。 ReferenceFinder是一个帮助程序类,它在特定模块中查找类型\。

    所以你需要为方法体创建方法和ILProccesor,然后对于你拥有的每个字符串,你需要将操作数的指令操作码分开,然后在字典中查找所需的操作并传递操作码,操作数为字符串和ILProccesor