25

次のデリゲートオブジェクトのダンプがあります。

Name: MyEventHandler  
MethodTable: 132648fc  
EEClass: 1319e2b4  
Size: 32(0x20) bytes  
Fields:  
     MT    Field   Offset                 Type VT     Attr    Value Name  
790fd0f0  40000ff        4        System.Object  0 instance 014037a4 _target  
7910ebc8  4000100        8 ...ection.MethodBase  0 instance 00000000 _methodBase  
791016bc  4000101        c        System.IntPtr  1 instance 2ef38748 _methodPtr  
791016bc  4000102       10        System.IntPtr  1 instance        0 _methodPtrAux  
790fd0f0  400010c       14        System.Object  0 instance 00000000 _invocationList  
791016bc  400010d       18        System.IntPtr  1 instance        0 _invocationCount  

デリゲートが指すメソッドの名前を取得するにはどうすればよいですか?

4

5 に答える 5

40

私の経験では、hakanによって提供された提案は機能しません。これが私がすることです。

出力は、アタッチされたハンドラーが。で示されるオブジェクトのメンバーであることを示しています_target。ダンプすることで、メソッドテーブルを取得できます。

説明のために、同様の例を作成しました。

0:000> !do 02844de4 
Name: System.EventHandler
MethodTable: 0067afa4
EEClass: 0052ef88
Size: 32(0x20) bytes
 (C:\windows\assembly\GAC_32\mscorlib\2.0.0.0__b77a5c561934e089\mscorlib.dll)
Fields:
      MT    Field   Offset                 Type VT     Attr    Value Name
002e6d58  40000ff        4        System.Object  0 instance 02842d20 _target
0058df70  4000100        8 ...ection.MethodBase  0 instance 00000000 _methodBase
0058743c  4000101        c        System.IntPtr  1 instance   2cc060 _methodPtr
0058743c  4000102       10        System.IntPtr  1 instance        0 _methodPtrAux
002e6d58  400010c       14        System.Object  0 instance 00000000 _invocationList
0058743c  400010d       18        System.IntPtr  1 instance        0 _invocationCount

この場合、でオブジェクトを確認します02842d20

0:000> !do 02842d20 
Name: app.Foo
MethodTable: 002c30bc
EEClass: 002c13d4
Size: 12(0xc) bytes
 (C:\workspaces\TestBench\app\bin\x86\Debug\app.exe)
Fields:
None

したがって、ターゲットタイプはapp.Fooです。このタイプのメソッドをダンプしてみましょう。

0:000> !dumpmt -md 002c30bc
EEClass: 002c13d4
Module: 002c2c5c
Name: app.Foo
mdToken: 02000002  (C:\workspaces\TestBench\app\bin\x86\Debug\app.exe)
BaseSize: 0xc
ComponentSize: 0x0
Number of IFaces in IFaceMap: 0
Slots in VTable: 6
--------------------------------------
MethodDesc Table
   Entry MethodDesc      JIT Name
002ec015   002e6cbc     NONE System.Object.ToString()
002ec019   002e6cc4     NONE System.Object.Equals(System.Object)
002ec029   002e6cf4     NONE System.Object.GetHashCode()
005f4930   002e6d1c      JIT System.Object.Finalize()
005f8238   002c30b4      JIT app.Foo..ctor()
005f8270   002c30a8      JIT app.Foo.Bar(System.Object, System.EventArgs)

MethodDescテーブルの値を元の値と比較します_methodPtr。明らかな一致はありません。

_methodPtrはjmp、問題の関数のアドレスに対してaを実行するか、修正ルーチンを呼び出すコードの一部を指しているため、次のステップは!u、の値に対してコマンドを使用することです_methodPtrjmp指示が表示されれば、アドレスがわかり、それを使用!uしてメソッドを取得します。

一方、が表示された場合は、次のcallようにclr!PrecodeFixupThunkポイントされたメモリをダンプすることでMethodDescを取得できます。_methodPtr

0:000> dd 2cc060 
002cc060  7e5d65e8 00005e6e 002c30a8 00000000
002cc070  00000000 00000000 00000000 00000000
002cc080  00000000 00000000 00000000 00000000

3番目のDWORDとして、メソッドテーブルエントリのように見えるものが表示されます。値を上記のメソッドテーブルと比較する002c30a8と、メソッドの名前がであることがわかりますapp.Foo.Bar

これは構築された例であるため、私はメソッドを見つけたことを知っています。この場合、私は探していました。

実際には、イベントの実際の使用法に応じてフィールドの使用方法が異なるため、上記の例が示すよりも少し複雑になる可能性があります。ただし、私の経験では、上記のアプローチは一般的なパブリッシャー/サブスクライバーのシナリオで機能します。

実装の詳細の詳細についてcomdelegate.cppは、共有ソースCLIのファイルを確認してください。

于 2010-09-10T06:16:46.177 に答える
5

methodPtrの値に!ip2mdを使用できると思います。これでメソッドの説明が得られます。

于 2010-09-08T14:29:09.983 に答える
5

格納されたメソッドをmethodPtr値から直接解決するために、littelWindbgスクリプトを作成しました。あなたはここでもっと読むことができます。

スクリプトは次のとおりです。

r $t0 = ${$arg1}+5
r $t1 = $t0 + 8*by($t0+2) + 3
r $t2 = 8*by($t0+1)
r $t3 = poi($t1) + $t2
!DumpMD $t3

それをファイルに保存し、次のようなデリゲートの_methodPtr値を使用して実行します。

$$>a< "c:\source\DelegateTest\Resolve.txt" 2ef38748 

これで、すべてのプラットフォームと.NET2.0から.NET4.5まででうまくいくはずです。

于 2012-05-20T22:49:57.883 に答える
4

もう1つの方法は、_methodPtrでデータを逆アセンブルすることです。

EventHandlerが次のようになっているとしましょう。

      MT    Field   Offset                 Type VT     Attr    Value Name
6da484dc  40000ff        4        System.Object  0 instance 02d8ff64 _target
6da4d0ac  4000100        8 ...ection.MethodBase  0 instance 00000000 _methodBase
6da4b188  4000101        c        System.IntPtr  1 instance  d955840 _methodPtr

d955840の分解を見てみましょう

!U d955840
Unmanaged code
08577a50 b884f8a007      mov     eax,7A0F884h
08577a55 90              nop
08577a56 e855b4d665      call    mscorwks+0x2eb0 (6e2e2eb0)
08577a5b e9ac8de4f7      jmp     003c080c
08577a60 b8d4f9a007      mov     eax,7A0F9D4h
08577a65 90              nop
08577a66 e845b4d665      call    mscorwks+0x2eb0 (6e2e2eb0)
08577a6b e99c8de4f7      jmp     003c080c
08577a70 00b000eb0cb0    add     byte ptr [eax-4FF31500h],dh
08577a76 03eb            add     ebp,ebx

ここに7A0F884への移動が表示されるので、これが私たちが探している方法である可能性があります。

!dumpmd 7A0F884 
Method Name: DemoClass.OnDemoEvent(System.Object, System.EventArgs)
Class: 07c079e8
MethodTable: 07c10034
mdToken: 060010ee
Module: 07a0b7ac
IsJitted: no
CodeAddr: ffffffff

ビンゴ!

メソッドの名前を取得するにはさまざまな方法がありますが、すべての状況ですべてが機能するわけではありません

于 2010-09-16T13:51:04.977 に答える
1

ClrMdの助けを借りて、ユーザーフレンドリーなGUIを使用して.Netダンプファイルを探索するツールに取り組んでいます。デリゲートを検索し、その呼び出しリストとメソッド名を表示する機能があります。

于 2017-01-02T13:47:48.147 に答える