DynamicMethod возвращает неправильное значение, когда тип свойства Int64 - PullRequest
5 голосов
/ 30 августа 2009

Я работаю над подпрограммой, чтобы использовать DynamicMethod для получения значений из объекта. Он работал нормально с большинством типов данных, за исключением DateTime.Ticks, который является int64

В следующем тестовом приложении. Я использую как MethodInfo, так и DynamicMethod, methodInfo возвращает правильное значение, а DynamicMethod - нет. Есть идеи?

using System;
using System.Reflection;
using System.Reflection.Emit;

namespace ConsoleApplication2
{
    public delegate object MemberGetDelegate(object obj);

    class Program
    {
        static void Main(string[] args)
        {
            DateTime dat = DateTime.Today;
            PropertyInfo pi = typeof(DateTime).GetProperty("Ticks");
            MethodInfo mi = pi.GetGetMethod();
            Type type = pi.PropertyType;
            object ticks = mi.Invoke(dat, new object[] { });
            Console.WriteLine("Get by MethodInfo " + ticks.ToString());

            MemberGetDelegate mget=TypeUtils.GetMemberFunc(pi);
            object ret = mget(dat);
            Console.WriteLine("Get by DynamicMethod " + ret.ToString());

            Console.Read();
        }
    }

    static class TypeUtils
    {
        public static readonly Type objectType = typeof(object);
        public static readonly Type[] typeArray = new[] { typeof(object) };

        public static MemberGetDelegate GetMemberFunc(PropertyInfo pi)
        {

            MethodInfo mi = pi.GetGetMethod();

            if (mi != null)
            {
                DynamicMethod dm = new DynamicMethod("_" + mi.Name,
                                                     objectType,
                                                     typeArray,
                                                     pi.Module, true);
                ILGenerator il = dm.GetILGenerator();

                // Load the instance of the object (argument 0) onto the stack
                il.Emit(OpCodes.Ldarg_0);

                // Call underlying get method
                il.EmitCall(OpCodes.Callvirt, mi, null);

                //boxing
                if (pi.PropertyType.IsValueType)
                {
                    il.Emit(OpCodes.Box, pi.PropertyType);                   
                }

                // return the value on the top of the stack
                il.Emit(OpCodes.Ret);

                return  (MemberGetDelegate) dm.CreateDelegate(typeof (MemberGetDelegate));

            }
            return null;
        }
    }
}

1 Ответ

4 голосов
/ 31 августа 2009

Вы генерируете неверный код. Если вы компилируете полученный IL с Ilasm

ldarg.0
callvirt instance int64 [mscorlib]System.DateTime::get_Ticks()
box int64
ret

А затем запустите PEVerify для исполняемого файла, он скажет вам, что код недействителен. (Вы не можете использовать callvirt для такого метода типа значения). Рабочий код должен выглядеть следующим образом

ldarg.0
unbox [mscorlib]System.DateTime
call instance int64 [mscorlib]System.DateTime::get_Ticks()
box int64
ret

Адаптируйте генерацию кода соответствующим образом, и она вернет правильное значение.

...