Как извлечь код класса IL из загруженной сборки и сохранить на диск?

Как мне извлечь код IL для классов, которые генерируются во время выполнения с помощью отражения, чтобы я мог сохранить его на диск? Если вообще возможно. У меня нет контроля над кодом, который генерирует эти классы.

В конце концов, я хотел бы загрузить этот код IL с диска в другую сборку.

Я знаю, что могу сериализовать / десериализовать классы, но я хочу использовать чисто IL-код. Я не обеспокоен последствиями для безопасности.

Запуск Mono 2.10.1

2 ответа

Решение

Или еще лучше, используйте Mono.Cecil.

Это позволит вам получить отдельные инструкции, даже манипулировать ими и разбирать их (с добавлением монодекомпилятора).

Обратите внимание, что декомпилятор находится в стадии разработки (в прошлый раз, когда я проверял, он не полностью поддерживает лямбда-выражения и блоки исключений Visual Basic), но вы можете получить довольно декомпилированный вывод в C# довольно легко, поскольку вы не попадаете в эти граничные условия, Кроме того, работа продвинулась с тех пор.

Mono Cecil, в общем, давайте также напишем IL для новой сборки, которую вы можете затем впоследствии загрузить в свой домен приложения, если вы хотите поиграть с острыми краями.

Обновление я пришел, чтобы попробовать это. К сожалению, я думаю, что нашел проблему, с которой вы столкнулись. Оказывается, что нет никакого способа получить байты IL для сгенерированного типа, если только сборка не была записана где-то, откуда вы можете загрузить ее.

Я предположил, что вы можете просто получить биты с помощью отражения (поскольку классы поддерживают требуемые методы), однако связанные методы просто вызывают исключение The invoked member is not supported in a dynamic module. на вызов. Вы можете попробовать это с помощью приведенного ниже кода, но вкратце я полагаю, это означает, что этого не произойдет, если вы не хотите f*ck сMarshal::GetFunctionPointerForDelegate(), Вам нужно будет выполнить двоичный вывод инструкций и вручную разобрать их как коды операций IL. Там будут драконы.

Фрагмент кода:

using System;
using System.Linq;
using Mono.Cecil;
using Mono.Cecil.Cil;
using System.Reflection.Emit;
using System.Reflection;

namespace REFLECT
{
    class Program
    {
        private static Type EmitType()
        {
            var dyn = AppDomain.CurrentDomain.DefineDynamicAssembly(new AssemblyName("Emitted"), AssemblyBuilderAccess.RunAndSave);
            var mod = dyn.DefineDynamicModule("Emitted", "Emitted.dll");
            var typ = mod.DefineType("EmittedNS.EmittedType", System.Reflection.TypeAttributes.Public);
            var mth = typ.DefineMethod("SuperSecretEncryption", System.Reflection.MethodAttributes.Public | System.Reflection.MethodAttributes.Static, typeof(String), new [] {typeof(String)});

            var il = mth.GetILGenerator();
            il.EmitWriteLine("Emit was here");
            il.Emit(System.Reflection.Emit.OpCodes.Ldarg_0);    
            il.Emit(System.Reflection.Emit.OpCodes.Ret);
            var result = typ.CreateType();
            dyn.Save("Emitted.dll");
            return result;
        }

        private static Type TestEmit()
        {
            var result = EmitType();
            var instance = Activator.CreateInstance(result);
            var encrypted = instance.GetType().GetMethod("SuperSecretEncryption").Invoke(null, new [] { "Hello world" });
            Console.WriteLine(encrypted); // This works happily, print "Emit was here" first

            return result;
        }

        public static void Main (string[] args)
        {
            Type emitted = TestEmit();

              // CRASH HERE: even if the assembly was actually for SaveAndRun _and_ it 
              // has actually been saved, there seems to be no way to get at the image
              // directly:
            var ass = AssemblyFactory.GetAssembly(emitted.Assembly.GetFiles(false)[0]);

              // the rest was intended as mockup on how to isolate the interesting bits
              // but I didn't get much chance to test that :)
            var types = ass.Modules.Cast<ModuleDefinition>().SelectMany(m => m.Types.Cast<TypeDefinition>()).ToList();
            var typ = types.FirstOrDefault(t => t.Name == emitted.Name);

            var operands = typ.Methods.Cast<MethodDefinition>()
                .SelectMany(m => m.Body.Instructions.Cast<Instruction>())
                .Select(i => i.Operand);

            var requiredTypes = operands.OfType<TypeReference>()
                .Concat(operands.OfType<MethodReference>().Select(mr => mr.DeclaringType))
                .Select(tr => tr.Resolve()).OfType<TypeDefinition>()
                .Distinct();
            var requiredAssemblies = requiredTypes
                .Select(tr => tr.Module).OfType<ModuleDefinition>()
                .Select(md => md.Assembly.Name as AssemblyNameReference);

            foreach (var t in types.Except(requiredTypes))
                ass.MainModule.Types.Remove(t);

            foreach (var unused in ass.MainModule
                     .AssemblyReferences.Cast<AssemblyNameReference>().ToList()
                     .Except(requiredAssemblies))
                ass.MainModule.AssemblyReferences.Remove(unused);

            AssemblyFactory.SaveAssembly(ass, "/tmp/TestCecil.dll");
        }
    }
}

Если все, что вы хотите, это IL для вашего User класс, у вас уже есть. Это в DLL, что вы скомпилировали его.

Из вашей другой сборки, вы можете загрузить DLL с User Класс динамически и использовать его через отражение.

ОБНОВЛЕНИЕ:

Если у вас есть динамический класс, созданный с Reflection.Emitу вас есть AssemblyBuilder что вы можете использовать, чтобы сохранить его на диск.

Если ваш динамический тип был создан вместо Mono.Cecilу вас есть AssemblyDefinition что вы можете сохранить на диск с myAssemblyDefinition.Write("MyAssembly.dll") (в Mono.Cecil 0,9).

Другие вопросы по тегам