성태의 닷넷 이야기
홈 주인
모아 놓은 자료
프로그래밍
질문/답변
사용자 관리
사용자
메뉴
아티클
외부 아티클
유용한 코드
온라인 기능
MathJax 입력기
최근 덧글
[정성태] Java - How to use the Foreign Funct...
[정성태] 제가 큰 실수를 했군요. ^^; Delegate를 통한 Bein...
[정성태] Working with Rust Libraries from C#...
[정성태] Detecting blocking calls using asyn...
[정성태] 아쉽게도, 커뮤니티는 아니고 개인 블로그입니다. ^^
[정성태] 질문이 잘 이해가 안 됩니다. 우선, 해당 소스코드에서 ILis...
[양승조
] var대신 dinamic으로 선언해서 해결은 했습니다. 맞는 해...
[양승조
] 또 막혔습니다. ㅠㅠ var list = props[i].Ge...
[양승조
] 아. 감사합니다. 어제는 안됐던것 같은데....정신을 차려야겠네...
[정성태] "props[i].GetValue(props[i])" 코드에서 ...
글쓰기
제목
이름
암호
전자우편
HTML
홈페이지
유형
제니퍼 .NET
닷넷
COM 개체 관련
스크립트
VC++
VS.NET IDE
Windows
Team Foundation Server
디버깅 기술
오류 유형
개발 환경 구성
웹
기타
Linux
Java
DDK
Math
Phone
Graphics
사물인터넷
부모글 보이기/감추기
내용
<div style='display: inline'> <div style='font-family: 맑은 고딕, Consolas; font-size: 20pt; color: #006699; text-align: center; font-weight: bold'>windbg로 확인하는 .NET CLR LCG 메서드(DynamicMethod)</div> <br /> 지난번 글에 이어서.<BR> <BR> <PRE style="PADDING-BOTTOM: 10px; OVERFLOW-X: scroll; BACKGROUND-COLOR: #fbedbb; MARGIN: 10px 0px 10px 10px; PADDING-LEFT: 10px; WIDTH: 800px; PADDING-RIGHT: 0px; FONT-FAMILY: Consolas, Verdana; PADDING-TOP: 10px"> windbg로 확인하는 .NET CLR 메서드 ; <A class=con_link href="http://www.sysnet.pe.kr/2/0/940" target=_blank>http://www.sysnet.pe.kr/2/0/940</A> </PRE> <BR> 그럼, 동적으로 생성되는 Dynamic 메서드는 어떻게 확인이 될까요?<BR><BR>예제 코드 먼저 만들어야겠지요. (아래의 실습은 .NET 3.5 x64 환경에서 행해졌습니다.)<BR> <BR> <PRE style="PADDING-BOTTOM: 10px; OVERFLOW-X: scroll; BACKGROUND-COLOR: #fbedbb; MARGIN: 10px 0px 10px 10px; PADDING-LEFT: 10px; WIDTH: 800px; PADDING-RIGHT: 0px; FONT-FAMILY: Consolas, Verdana; PADDING-TOP: 10px"> static void Main(string[] args) { DynamicMethod dynamicMethod = CreateTestMethod(<B style="COLOR: blue">typeof(Program)</B>, <B style="COLOR: blue">"Test"</B>); dynamicMethod.Invoke(null, null); Console.ReadLine(); } private static DynamicMethod CreateTestMethod(Type type, string name) { DynamicMethod dynamicMethod = new DynamicMethod( name, MethodAttributes.Static | MethodAttributes.Public, CallingConventions.Standard, typeof(void), Type.EmptyTypes, type, false ); ILGenerator gen = dynamicMethod.GetILGenerator(); gen.EmitWriteLine(type.Name + "." + name); gen.Emit(OpCodes.Ret); dynamicMethod.CreateDelegate(typeof(Action)); return dynamicMethod; } </PRE> <BR> windbg로 연결해 보면, 일단 어려운 것이 기존 MethodDesc 테이블에는 이 항목이 추가되지 않는다는 점입니다. 사실 동적으로 생성된 "Test" 메서드를 찾아갈 방법도 없기 때문에 아래와 같이 "Main" 메서드를 기준으로 MethodDesc 테이블을 확인할 수밖에 없습니다.<BR> <BR> <PRE style="PADDING-BOTTOM: 10px; OVERFLOW-X: scroll; BACKGROUND-COLOR: #fbedbb; MARGIN: 10px 0px 10px 10px; PADDING-LEFT: 10px; WIDTH: 800px; PADDING-RIGHT: 0px; FONT-FAMILY: Consolas, Verdana; PADDING-TOP: 10px"> .loadby sos mscorwks 0:003> <B style="COLOR: blue">!name2ee ConsoleApplication1.exe!ConsoleApplication1.Program.Main</B> Module: 000007ff000333d0 (ConsoleApplication1.exe) Token: 0x0000000006000001 MethodDesc: 000007ff00033988 Name: ConsoleApplication1.Program.Main(System.String[]) JITTED Code Address: 000007ff00180120 0:003> <B style="COLOR: blue">!dumpmd 000007ff00033988</B> Method Name: ConsoleApplication1.Program.Main(System.String[]) Class: 000007ff00172218 MethodTable: 000007ff000339c0 mdToken: 06000001 Module: 000007ff000333d0 IsJitted: yes CodeAddr: 000007ff00180120 0:003> <B style="COLOR: blue">!dumpmt -md 000007ff000339c0</B> EEClass: 000007ff00172218 Module: 000007ff000333d0 Name: ConsoleApplication1.Program mdToken: 02000002 (D:\...[경로생략]...\ConsoleApplication1.exe) BaseSize: 0x18 ComponentSize: 0x0 Number of IFaces in IFaceMap: 0 Slots in VTable: 8 -------------------------------------- MethodDesc Table Entry MethodDesc JIT Name 000007fef0a4abe0 000007fef07ce828 PreJIT System.Object.ToString() 000007fef0a52560 000007fef07ce830 PreJIT System.Object.Equals(System.Object) 000007fef0a4bc70 000007fef07ce870 PreJIT System.Object.GetHashCode() 000007fef0afe5f0 000007fef07ce8a0 PreJIT System.Object.Finalize() 000007ff0003c038 000007ff000339b8 NONE ConsoleApplication1.Program..ctor() 000007ff00180120 000007ff00033988 JIT ConsoleApplication1.Program.Main(System.String[]) 000007ff001801e0 000007ff00033998 JIT ConsoleApplication1.Program.CreateTestMethod(System.Type, System.String) 000007ff001804a0 000007ff000339a8 JIT ConsoleApplication1.Program.ShowMethodDesc(System.Reflection.Emit.DynamicMethod) </PRE> <BR> 보시는 것처럼 Program 타입에는 LCG 메서드가 없습니다. 아쉽지만, DynamicMethod를 찾을 수 있는 명확한 방법이 없습니다. 그나마 찾을 수 있었던 방법이 있는데,<BR> <BR> <PRE style="PADDING-BOTTOM: 10px; OVERFLOW-X: scroll; BACKGROUND-COLOR: #fbedbb; MARGIN: 10px 0px 10px 10px; PADDING-LEFT: 10px; WIDTH: 800px; PADDING-RIGHT: 0px; FONT-FAMILY: Consolas, Verdana; PADDING-TOP: 10px"> Executing dynamic IL with DynamicMethod ; <A class=con_link href="http://www.josebonnin.com/post/2008/08/23/Executing-dynamic-IL-with-DynamicMethod.aspx" target=_blank>http://www.josebonnin.com/post/2008/08/23/Executing-dynamic-IL-with-DynamicMethod.aspx</A> </PRE> <BR> 위의 방법을 이번 예제에 적용해 보면 다음과 같이 찾아들어갈 수 있습니다.<BR> <BR> <PRE style="PADDING-BOTTOM: 10px; OVERFLOW-X: scroll; BACKGROUND-COLOR: #fbedbb; MARGIN: 10px 0px 10px 10px; PADDING-LEFT: 10px; WIDTH: 800px; PADDING-RIGHT: 0px; FONT-FAMILY: Consolas, Verdana; PADDING-TOP: 10px"> 0:003> <B style="COLOR: blue">!DumpHeap -stat</B> total 6210 objects Statistics: MT Count TotalSize Class Name 000007fef0bde5d0 1 24 System.IO.TextReader+NullTextReader 000007fef0bde0a0 1 24 System.Collections.Generic.GenericEqualityComparer`1[[System.String, mscorlib]] ...[생략]... 000007fef0b98520 1 104 System.Threading.Thread <B style="COLOR: blue">000007fef0bb3588 1 112 System.Reflection.Emit.DynamicMethod</B> ...[생략]... 000007fef0b9fac0 62 126024 System.Byte[] 000007fef0b85870 1361 149296 System.Object[] 000007fef0b97a80 903 154344 System.String <B style="COLOR: blue">Total 6210 objects</B> // 위의 출력결과가 너무 많기 때문에, // 다음과 같이 필터링하는 방법이 제공됩니다. 0:003> .shell -ci "!dumpheap -stat" findstr DynamicMethod 000007fef0bb38d8 1 56 System.Reflection.Emit.DynamicMethod+RTDynamicMethod <B style="COLOR: blue">000007fef0bb3588 1 112 System.Reflection.Emit.DynamicMethod</B> .shell: Process exited 0:003> <B style="COLOR: blue">!DumpHeap -mt 000007fef0bb3588</B> Address MT Size <B style="COLOR: blue">00000000027c63f0</B> 000007fef0bb3588 112 ... [프로그램에서 생성한 DynamicMethod 목록이 여기에 출력됨]... total 1 objects Statistics: MT Count TotalSize Class Name 000007fef0bb3588 1 112 System.Reflection.Emit.DynamicMethod Total 1 objects 0:003> <B style="COLOR: blue">!DumpIL 00000000027c63f0</B> This is dynamic IL. Exception info is not reported at this time. If a token is unresolved, run "!do <addr>" on the addr given in parenthesis. You can also look at the token table yourself, by running "!DumpArray 00000000027c99a0". IL_0000: ldstr 70000002 "Program.Test" IL_0005: call 6000003 System.Console.WriteLine(System.String) IL_000a: ret </PRE> <BR> 그런데, 현실 세계의 프로그램에서 저렇게 간단하게 하나만 정의하는 프로젝트가 몇 개나 되겠습니까? "!DumpHeap -mt ..."로 출력된 Address 값들을 일일이 하나씩 DumpIL로 출력해 봐야만 알 수 있다는 것이니,,, 아쉬운 데로 이번에는 여기까지!<BR> <BR> <HR style="WIDTH: 50%"> <BR> <BR> LCG 메서드가 MethodDesc 테이블에 등록되어 있지 않다고 해서 MethodDesc 자료구조가 없다는 것은 아닙니다. 위에서 알아본 것처럼 DynamicMethod 타입의 힙에 제공이 되고 있으며, 이 값은 Reflection을 이용해서 알아보는 것이 가능합니다.<BR> <BR> <PRE style="PADDING-BOTTOM: 10px; OVERFLOW-X: scroll; BACKGROUND-COLOR: #fbedbb; MARGIN: 10px 0px 10px 10px; PADDING-LEFT: 10px; WIDTH: 800px; PADDING-RIGHT: 0px; FONT-FAMILY: Consolas, Verdana; PADDING-TOP: 10px"> static void ShowMethodDesc(DynamicMethod dynamicMethod) { // RuntimeMethodHandle methodHandle = dynamicMethod.MethodHandle; // 예외 발생 FieldInfo fieldInfo = typeof(DynamicMethod).GetField("<B style="COLOR: blue">m_method</B>", BindingFlags.NonPublic | BindingFlags.Instance); <B style="COLOR: blue">RuntimeMethodHandle methodHandle</B> = ((RuntimeMethodHandle)fieldInfo.GetValue(dynamicMethod)); Console.WriteLine("[" + dynamicMethod.Name + "]"); Console.WriteLine("MethodHandle == 0x" + methodHandle.Value.ToString("x")); IntPtr pAddress = methodHandle.GetFunctionPointer(); Console.WriteLine("Stub or Jitted Address == 0x" + pAddress.ToString("x")); Console.WriteLine(); } </PRE> <BR> 여기서도 아쉬운 점이 있는데요. DynamicMethod의 경우에는 일반 메서드에서처럼 MethodHandle 값을 직접 구하려고 시도하면 예외가 발생합니다. 대신 내부의 private에 저장된 m_method 필드를 액세스해서 그 값을 구해오는 방식을 사용해야 합니다.<BR> <BR> 일단, MethodHandle 값을 얻어오면 나머지는 <A href="http://www.sysnet.pe.kr/2/0/940" target=_blank>이전 글</A>에서 했던 것과 동일하게 JIT 컴파일 이전/이후에 대해서 GetFunctionPointer의 값이 달라지는 것을 확인할 수 있습니다. 게다가 methodHandle.Value 값으로 windbg에서 "!dumpmd [methodHandle.Value로 얻은 값]" 명령어를 내리게 되면 동일하게 다음과 같은 식으로 출력이 됩니다.<BR> <BR> <PRE style="PADDING-BOTTOM: 10px; OVERFLOW-X: scroll; BACKGROUND-COLOR: #fbedbb; MARGIN: 10px 0px 10px 10px; PADDING-LEFT: 10px; WIDTH: 800px; PADDING-RIGHT: 0px; FONT-FAMILY: Consolas, Verdana; PADDING-TOP: 10px"> 0:003> <B style="COLOR: blue">!dumpmd 0x7ff00054bf0</B> Method Name: DynamicClass.Test() Class: 000007ff00054ad8 <B style="COLOR: blue">MethodTable: 000007ff00054b80</B> mdToken: 06000000 Module: 000007ff000533d0 <B style="COLOR: blue">IsJitted: yes CodeAddr: 000007ff001e0130</B> // 이 값이 methodHandle.GetFunctionPointer(); 0:003> <B style="COLOR: blue">!dumpmt -md 000007ff00054b80</B> EEClass: 000007ff00054ad8 Module: 000007ff000533d0 Name: mdToken: 02000000 (D:\...[경로 생략]...\ConsoleApplication1.exe) BaseSize: 0x10 ComponentSize: 0x0 Number of IFaces in IFaceMap: 0 Slots in VTable: 0 -------------------------------------- MethodDesc Table Entry MethodDesc JIT Name // 보는 것처럼, MethodDesc 테이블은 비어 있음 </PRE> <BR> 계속 아쉬운 이야기만 하게 되는데요. ^^;<BR> <BR> .NET 4.0에서는 m_method마저 없어져버려서 다른 방법을 찾아야만 했습니다. .NET Reflector로 보면, 다행히 "internal RuntimeMethodHandle GetMethodDescriptor()" 메서드가 제공되어 다음과 같이 구하는 것이 가능했습니다.<BR> <BR> <PRE style="PADDING-BOTTOM: 10px; OVERFLOW-X: scroll; BACKGROUND-COLOR: #fbedbb; MARGIN: 10px 0px 10px 10px; PADDING-LEFT: 10px; WIDTH: 800px; PADDING-RIGHT: 0px; FONT-FAMILY: Consolas, Verdana; PADDING-TOP: 10px"> RuntimeMethodHandle methodHandle; if (Environment.Version.Major == 4) { MethodInfo getMethodDescriptorInfo = typeof(DynamicMethod).GetMethod("GetMethodDescriptor", BindingFlags.NonPublic | BindingFlags.Instance); methodHandle = (RuntimeMethodHandle)getMethodDescriptorInfo.Invoke(dynamicMethod, null); } </PRE> <BR> 끝!
첨부파일
스팸 방지용 인증 번호
3466
(왼쪽의 숫자를 입력해야 합니다.)