Question

I am trying this simple example on creating a generic method using Reflection.Emit but it is causing an exception when calling Invoke and I can't find the problem.

public class Program
{
    public static void Main(string[] args)
    {
        AppDomain appDomain = AppDomain.CurrentDomain;
        AssemblyName name = new AssemblyName("MyAssembly") { Version = new Version("1.0.0.0") };
        AssemblyBuilder ab = appDomain.DefineDynamicAssembly(name, AssemblyBuilderAccess.RunAndSave);
        ModuleBuilder mb = ab.DefineDynamicModule("MyModule", "MyAssembly.dll");
        TypeBuilder tb = mb.DefineType("Widget", TypeAttributes.Public);

        // Define a method builder
        MethodBuilder methodBuilder = tb.DefineMethod("MyGenericMethod", MethodAttributes.Public | MethodAttributes.Static);

        // Get generic type parameter builders 
        GenericTypeParameterBuilder[] genericParams =
            methodBuilder.DefineGenericParameters("TKey", "TValue");

        methodBuilder.SetSignature(typeof(int), null, null,
            genericParams,
            null, null);
        methodBuilder.DefineParameter(1, ParameterAttributes.None, "key");
        methodBuilder.DefineParameter(2, ParameterAttributes.None, "val");

        ILGenerator gen = methodBuilder.GetILGenerator();
        MethodInfo writeLineStr = typeof(Console).GetMethod("WriteLine", new[] { typeof(object) });
        gen.Emit(OpCodes.Ldarg_0);
        gen.Emit(OpCodes.Box, genericParams[0]);
        gen.Emit(OpCodes.Call, writeLineStr);
        gen.Emit(OpCodes.Ldarg_1);
        gen.Emit(OpCodes.Box, genericParams[1]);
        gen.Emit(OpCodes.Call, writeLineStr);
        gen.Emit(OpCodes.Ret);

        Type t = tb.CreateType();
        MethodInfo genMeth = t.GetMethod("MyGenericMethod").MakeGenericMethod(typeof(string), typeof(int));
        genMeth.Invoke(null, new object[] { "NumberKey", 100 });
    }
}

When calling genMeth.Invoke, the following exception is thrown

Unhandled Exception: System.Reflection.TargetInvocationException: Exception has been thrown by the t
arget of an invocation. ---> System.InvalidProgramException: Common Language Runtime detected an inv
alid program.
   at Widget.MyGenericMethod[TKey,TValue](TKey key, TValue val)
   --- End of inner exception stack trace ---
   at System.RuntimeMethodHandle.InvokeMethod(Object target, Object[] arguments, Signature sig, Bool
ean constructor)
   at System.Reflection.RuntimeMethodInfo.UnsafeInvokeInternal(Object obj, Object[] parameters, Obje
ct[] arguments)
   at System.Reflection.RuntimeMethodInfo.Invoke(Object obj, BindingFlags invokeAttr, Binder binder,
 Object[] parameters, CultureInfo culture)
   at System.Reflection.MethodBase.Invoke(Object obj, Object[] parameters)
   at ReflectionEmitDemo14.Program.Main(String[] args) in d:\Projects\ReflectionEmitDemo\ReflectionE
mitDemo14\Program.cs:line 50
Was it helpful?

Solution

Your method is static and has two arguments: argument 0 and argument 1. But in the generated IL code you're accessing an argument 2 which does not exist.

Another problem is that you're trying to pass an Int32 value to a method that expects a String reference. Box both arguments and pass them to the WriteLine overload that accepts an Object reference.

You also have declared your method to return an Int32, but you're not putting anything on the stack before returning.

Try to write the method in C# and then take a look at what IL the C# compiler emits (for example, using ILspy). It should look like this:

ldarg.0
box !TKey
call void [mscorlib]System.Console::WriteLine(object)

ldarg.1
box !TValue
call void [mscorlib]System.Console::WriteLine(object)

ldc.i4.s 42
ret
Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow
scroll top