Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 1개 있습니다.)

GetFunctionPointer 호출 시 System.InvalidProgramException 예외 발생

현상은 간단합니다. 제네릭 메서드인 경우 그에 대해 GetFunctionPointer를 호출하면,

using System;
using System.Reflection;
using System.Threading;

public class Program
{
    public static void GenericMethod<T>(T obj)
    {
        Thread.Sleep(1);
    }

    static unsafe void Main()
    {
        Type type = typeof(Program);
        MethodInfo mi = type.GetMethod("GenericMethod", BindingFlags.Static | BindingFlags.Public);
        {
            string fullName = string.Format("{0}.{1}", type.FullName, mi.Name);
            IntPtr methodBody = mi.MethodHandle.GetFunctionPointer(); // 예외 발생
            Console.WriteLine(fullName);
        }
    }
}

MethodInfo.MethodHandle.GetFunctionPointer 메서드 호출에서 다음과 같은 예외가 발생합니다.

An unhandled exception of type 'System.InvalidProgramException' occurred in mscorlib.dll

Additional information: Common Language Runtime detected an invalid program.

이게... 이상한 듯 하면서도 사실 당연한 겁니다. 왜냐하면, 제네릭인 경우 컴파일러가 생성한 IL 단계에는 제네릭의 타입이 정해지지 않은 상태이고, 실제 메서드가 사용될 때 기계어 컴파일이 타입에 따라 확장되면서 컴파일되기 때문에 GenericMethod 자체의 FunctionPointer 값을 대표할 수 없는 것입니다. 가령 그 값이 0x00100으로 반환되었다고 해도 GenericMethod<int>(int obj)로 확장된 메서드의 FunctionPointer는 또 다른 값이 될 수 있는 것입니다.

그래서, 원래는 제네릭 메서드의 제대로 된 FunctionPointer를 구하고 싶다면 다음과 같이 해줘야 합니다.

MethodInfo mi = type.GetMethod("GenericMethod", BindingFlags.Static | BindingFlags.Public);
MethodInfo intMethod = mi.MakeGenericMethod(typeof(int));

string fullName = string.Format("{0}.{1}", type.FullName, intMethod.Name);
IntPtr methodBody = intMethod.MethodHandle.GetFunctionPointer();
Console.WriteLine(fullName); // Program.GenericMethod

그런데, 재미있는 것은 클래스 수준의 제네릭 인자가 있는 것은 또 잘됩니다.

public class GenericClass<T>
{
    public static void Test(T arg)
    {
        Console.WriteLine(arg);
    }
}

type = typeof(GenericClass<>);
mi = type.GetMethod("Test", BindingFlags.Static | BindingFlags.Public);
{
    string fullName = string.Format("{0}.{1}", type.FullName, mi.Name);
    IntPtr methodBody = mi.MethodHandle.GetFunctionPointer();
    Console.WriteLine(fullName); // GenericClass`1.Test
}




[이 글에 대해서 여러분들과 의견을 공유하고 싶습니다. 틀리거나 미흡한 부분 또는 의문 사항이 있으시면 언제든 댓글 남겨주십시오.]

[연관 글]






[최초 등록일: ]
[최종 수정일: 7/17/2021]

Creative Commons License
이 저작물은 크리에이티브 커먼즈 코리아 저작자표시-비영리-변경금지 2.0 대한민국 라이센스에 따라 이용하실 수 있습니다.
by SeongTae Jeong, mailto:techsharer at outlook.com

비밀번호

댓글 작성자
 




... 196  [197] 
NoWriterDateCnt.TitleFile(s)
17정성태7/17/200327528VC++: 6. Win32 API Hook - 소스는 "공개소스"에있습니다. [2]
16정성태7/17/200319854COM 개체 관련: 3. IE 툴밴드의 위치문제파일 다운로드1
15정성태7/17/200320879VC++: 5. 시행착오 - 클래스 포인터를 void * 로 대입후 delete 하는 경우.
14정성태7/17/200322407VC++: 4. MFC Message 처리 구조
13정성태7/17/200322170VC++: 3. template 활용의 최고 단계!
12정성태5/7/200619723VC++: 2. void func1( MYCLASS *&pBuildingElement ); 선언의 의미
10정성태7/17/200318039기타: 2. 데브피아 사이트의 클럽 서비스 내에 있는 Standard C++ Research
9정성태7/17/200323572기타: 1. Programming Applications for Microsoft Windows 4th Edition
8정성태7/17/200320966COM 개체 관련: 2. CWindowImpl 의 기본 윈도우 클래스 명을 바꾸려면?
7정성태7/17/200323547VS.NET IDE: 1. VC++ 프로파일링 사용법
4정성태7/17/200335924VC++: 1. C++ 클래스 멤버변수 초기화 [3]
3정성태7/17/200321880스크립트: 2. JScript에서의 Blocking 동작을 막아주는 COM 메서드
2정성태6/14/200640644COM 개체 관련: 1. IWebBrowser2와 IHTMLDocument2의 상호 변환 [2]
1정성태7/17/200324812스크립트: 1. IE 의 훌륭한 저장기능 - userData
... 196  [197]