Я играю с DynamicMethod и стремлюсь сделать следующее:
У меня есть действие, из которого я получаю код IL в виде байтов, используя GetILAsByteArray()
. Из этих байтов я хотел бы создать динамический метод и выполнить его. Вот пример того, что я пытаюсь сделать:
class Program
{
static void Main(string[] args)
{
//Create action and execute
Action<string> myAction = s =>
{
Console.WriteLine("Hello " + s);
};
myAction("World");
//Get IL bytes
byte[] ilBytes = myAction.GetMethodInfo().GetMethodBody().GetILAsByteArray();
DynamicMethod dynamicCallback = new DynamicMethod("myAction", typeof(void), new Type[] { typeof(string) });
DynamicILInfo dynamicIlInfo = dynamicCallback.GetDynamicILInfo();
dynamicIlInfo.SetCode(ilBytes, 100);
dynamicCallback.Invoke(null, new object[] { "World" });
}
}
При вызове dynamicCallback.Invoke(null, new object[] { "World" })
мы получаем «Исключение:« System.BadImageFormatException »в mscorlib.dll».
Одна вещь, о которой я понятия не имею, это то, что я должен использовать в качестве второго аргумента для SetCode()
, что следует использовать как «maxStackSize»? Как я могу установить то же значение, что и для начального действия? Но я полагаю, что это не причина для исключения.
Как я могу правильно создать динамический метод из байтов IL?
Решение
Здесь я хотел бы обобщить полное решение, предоставленное Дуди Келети:
static void Main(string[] args)
{
Action<string> myAction = s =>
{
Console.WriteLine("Hello " + s);
};
MethodInfo method = myAction.GetMethodInfo();
object target = myAction.Target;
DynamicMethod dm = new DynamicMethod(
method.Name,
method.ReturnType,
new[] {method.DeclaringType}.
Concat(method.GetParameters().
Select(pi => pi.ParameterType)).ToArray(),
method.DeclaringType,
skipVisibility: true);
DynamicILInfo ilInfo = dm.GetDynamicILInfo();
var body = method.GetMethodBody();
SignatureHelper sig = SignatureHelper.GetLocalVarSigHelper();
foreach (LocalVariableInfo lvi in body.LocalVariables)
{
sig.AddArgument(lvi.LocalType, lvi.IsPinned);
}
ilInfo.SetLocalSignature(sig.GetSignature());
byte[] code = body.GetILAsByteArray();
ILReader reader = new ILReader(method);
DynamicMethodHelper.ILInfoGetTokenVisitor visitor = new DynamicMethodHelper.ILInfoGetTokenVisitor(ilInfo, code);
reader.Accept(visitor);
ilInfo.SetCode(code, body.MaxStackSize);
dm.Invoke(target, new object[] { target, "World" });
Console.ReadLine(); //Just to see the result
}
Примечание. DynamicMethodHelper — это класс, разработанный Хайбо Луо и описанный в сообщение в блоге, но его также можно загрузить напрямую здесь.
Console.WriteLine
закодирован как токен метаданных (вероятно, MethodRef), а токены метаданных действительны только в области модуля, объявляющего их. Взгляните на функцииDynamicILInfo.GetTokenFor
, они будут импортировать другие элементы метаданных и создавать токены, действительные дляDynamicMethod
. - person thehennyy   schedule 20.10.2016GetTokenFor
. - person thehennyy   schedule 20.10.2016Module.Resolvexxx
. - person thehennyy   schedule 20.10.2016