程序化MSIL注入

时间:2009-02-04 18:38:24

标签: .net cil

假设我有一个像这样的错误应用程序:

using System;

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            Console.WriteLine("2 + 1 = {0}", Add(2, 1));
        }

        static int Add(int x, int y)
        {
            return x + x; // <-- oops!
        }
    }
}

该应用程序已经编译并部署到野外。有人找到了这个bug,现在他们要求修复它。虽然我可以使用修复程序重新部署此应用程序,但由于我无法控制的原因,这是一个极其麻烦的事情 - 我只想为该bug编写补丁。

具体来说,我想将自己的MSIL插入有问题的assmembly源文件中。我之前从未做过这样的事情,谷歌搜索没有发现任何有用的信息。如果我能在上面的代码中看到如何执行此操作的示例,它将极大地帮助我:)

如何以编程方式将自己的MSIL注入已编译的.NET程序集中?

[编辑添加:]给那些问:我不需要运行时hotswapping。对我来说完全没问题,关闭应用程序,操纵程序集,然后重新启动程序。

[编辑一次:]看起来普遍的共识是“操纵程序集是一种糟糕修补程序的方式”。如果是bad idea,我就不会走那条路。

我将问题保持开放,因为MSIL注入可能仍然可用于其他目的:)

7 个答案:

答案 0 :(得分:5)

我想我会问的问题是“你将如何部署补丁”?在某个地方,你必须部署一些东西来修复已经存在的bug。为什么重新编译dll并发布固定版本确实是一个问题?我的猜测是,弄清楚如何以编程方式注入MSIL比单纯重新部署固定程序集更麻烦。

答案 1 :(得分:4)

您是否考虑过将源直接插入到程序集中,而不是在运行时注入MSIL?

你可以用ildasm反汇编,插入你的MSIL,然后用ilasm重新组装,然后部署它的产品。

答案 2 :(得分:3)

如果您不想重新部署整个内容的原因是因为它实际上是半场演出,您应该使用某种二进制补丁工具 - 这是Google上的第一个结果:

Binary Diff http://www.daemonology.net/bsdiff/

答案 3 :(得分:2)

如果你的exe签名那么那是不可能的。为什么不能只发布单个组件,而不是再次发送整个应用程序?看起来你的目标是解决一个简单问题的复杂解决方案。

答案 4 :(得分:1)

您可以使用静态MSIL注入。 Mono Cecil或PostSharp可能会有所帮助。

答案 5 :(得分:1)

良好的信息来源......

IL Programming ebook

另一个是关于IL编程的MS Press书。

反编译为IL并重新编译的步骤

1。 ildasm /output=ConsoleApplication1.il ConsoleApplication1.exe

2。 //修改ConsoleApplication1.il代码

3。 ilasm ConsoleApplication1.il

答案 6 :(得分:1)

您需要非托管代码才能成功执行此操作,但可以完成...

请参阅http://www.codeproject.com/Articles/463508/NET-CLR-Injection-Modify-IL-Code-during-Run-time