Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
 
(연관된 글이 2개 있습니다.)
(시리즈 글이 7개 있습니다.)
개발 환경 구성: 319. windbg에서 python 스크립트 실행하는 방법 - pykd
; https://www.sysnet.pe.kr/2/0/11227

디버깅 기술: 96. windbg - 풀 덤프에 포함된 모든 닷넷 모듈을 파일로 저장하는 방법
; https://www.sysnet.pe.kr/2/0/11297

디버깅 기술: 103. windbg - .NET 4.0 이상의 환경에서 모든 DLL에 대한 심벌 파일을 로드하는 파이썬 스크립트
; https://www.sysnet.pe.kr/2/0/11331

디버깅 기술: 143. windbg/sos - Hashtable의 buckets 배열 내용을 모두 덤프하는 방법 (do_hashtable.py)
; https://www.sysnet.pe.kr/2/0/12084

디버깅 기술: 145. windbg/sos - Dictionary의 entries 배열 내용을 모두 덤프하는 방법 (do_hashtable.py)
; https://www.sysnet.pe.kr/2/0/12087

디버깅 기술: 178. windbg - 디버그 시작 시 스크립트 실행
; https://www.sysnet.pe.kr/2/0/12472

개발 환경 구성: 621. windbg에서 python 스크립트 실행하는 방법 - pykd (2)
; https://www.sysnet.pe.kr/2/0/12899




windbg/sos - Dictionary의 entries 배열 내용을 모두 덤프하는 방법 (do_hashtable.py)

지난 글에서,

windbg/sos - Hashtable의 buckets 배열 내용을 모두 덤프하는 방법 (do_hashtable.py)
; https://www.sysnet.pe.kr/2/0/12084

Hashtable 타입의 buckets 배열을 덤프하는 do_hashtable.py 스크립트를 알아봤는데요, 기왕 한 김에 이번에는 Dictionary 타입도 살펴보겠습니다. 실습을 위해 역시 Dictionary 인스턴스의 GC Heap 주소를 출력하는 코드를 작성하고,

{
    Dictionary<string, string> hash = new Dictionary<string, string>();
    hash["TEST"] = "1abc";
    hash["qwer"] = "2def";

    TypedReference tr = __makeref(hash);
    IntPtr ptr = **(IntPtr**)(&tr);

    Console.WriteLine(ptr.ToInt64().ToString("x")); // 1cb34d02f40

    Console.WriteLine(hash);

    Console.WriteLine("debug this...");
    Console.ReadLine();
}

windbg를 붙여 덤프합니다.

0:006> .loadby sos clr

0:006> !do 1cb34d02f40
Name:        System.Collections.Generic.Dictionary`2[[System.String, mscorlib],[System.String, mscorlib]]
MethodTable: 00007ffd977390a8
EEClass:     00007ffd978d4130
Size:        80(0x50) bytes
File:        C:\WINDOWS\Microsoft.Net\assembly\GAC_64\mscorlib\v4.0_4.0.0.0__b77a5c561934e089\mscorlib.dll
Fields:
              MT    Field   Offset                 Type VT     Attr            Value Name
00007ffd977c8090  4001887        8       System.Int32[]  0 instance 000001cb34d03018 buckets
00007ffd98992618  4001888       10 ...non, mscorlib]][]  0 instance 000001cb34d03040 entries
00007ffd977c80f8  4001889       38         System.Int32  1 instance                2 count
00007ffd977c80f8  400188a       3c         System.Int32  1 instance                2 version
00007ffd977c80f8  400188b       40         System.Int32  1 instance               -1 freeList
00007ffd977c80f8  400188c       44         System.Int32  1 instance                0 freeCount
00007ffd9774f228  400188d       18 ...Canon, mscorlib]]  0 instance 000001cb34d03000 comparer
00007ffd97751118  400188e       20 ...Canon, mscorlib]]  0 instance 0000000000000000 keys
00007ffd97762ae8  400188f       28 ...Canon, mscorlib]]  0 instance 0000000000000000 values
00007ffd977c5f88  4001890       30        System.Object  0 instance 0000000000000000 _syncRoot

이번에는 entries라는 필드로 내부 자료 구조를 유지하고 있는데 이것의 타입이 좀 깁니다.

0:006> !DumpObj /d 000001cb34d03040
Name:        System.Collections.Generic.Dictionary`2+Entry[[System.String, mscorlib],[System.String, mscorlib]][]
MethodTable: 00007ffd9775b7f0
EEClass:     00007ffd978bd288
Size:        96(0x60) bytes
Array:       Rank 1, Number of elements 3, Type VALUETYPE (Print Array)
Fields:
None

entries 필드의 값을 덤프해 보면,

0:006> !DumpArray /d 000001cb34d03040
Name:        System.Collections.Generic.Dictionary`2+Entry[[System.String, mscorlib],[System.String, mscorlib]][]
MethodTable: 00007ffd9775b7f0
EEClass:     00007ffd978bd288
Size:        96(0x60) bytes
Array:       Rank 1, Number of elements 3, Type VALUETYPE
Element Methodtable: 00007ffd9775b790
[0] 000001cb34d03050
[1] 000001cb34d03068
[2] 000001cb34d03080

역시 이번에도, DML로 제공하는 "000001cb34d03050" 링크를 누르면 Entry[]에 대한 MethodTable의 값을 이용하기 때문에 다음과 같이 정상적인 출력이 안 나옵니다.

0:006> !DumpVC /d 00007ffd9775b7f0 000001cb34d03050
Name:        System.Collections.Generic.Dictionary`2+Entry[[System.String, mscorlib],[System.String, mscorlib]][]
MethodTable: 00007ffd9775b7f0
EEClass:     00007ffd978bd288
Size:        24(0x18) bytes
File:        C:\WINDOWS\Microsoft.Net\assembly\GAC_64\mscorlib\v4.0_4.0.0.0__b77a5c561934e089\mscorlib.dll
Fields:

따라서 개별 Entry 값을 확인하려면 DML 링크를 사용하지 말고 직접 다음과 같이 dumpvc 명령을 내려야 합니다.

0:006> !DumpVC /d 00007ffd9775b790 000001cb34d03050
Name:        System.Collections.Generic.Dictionary`2+Entry[[System.String, mscorlib],[System.String, mscorlib]]
MethodTable: 00007ffd9775b790
EEClass:     00007ffd978bce48
Size:        40(0x28) bytes
File:        C:\WINDOWS\Microsoft.Net\assembly\GAC_64\mscorlib\v4.0_4.0.0.0__b77a5c561934e089\mscorlib.dll
Fields:
              MT    Field   Offset                 Type VT     Attr            Value Name
00007ffd977c80f8  4003502       10         System.Int32  1 instance        775949382 hashCode
00007ffd977c80f8  4003503       14         System.Int32  1 instance               -1 next
00007ffd977c9a60  4003504        0       System.__Canon  0 instance 000001cb34d02e48 key
00007ffd977c9a60  4003505        8       System.__Canon  0 instance 000001cb34d02e70 value

자... 그런데 Entry 타입이 가진 필드가 (Hashtable의 bucket이 3개를 가진 것과는 달리) 4개입니다. 하지만, 다행히 hashCode와 next 필드의 값이 Int32이고 4바이트 정렬이 되어 있어 최종적으로는 24바이트씩 차지하는 데에는 변함이 없습니다. 따라서 배열 요소를 한 줄 씩 덤프하는 방법도 같고,

0:006> dq /c3 000001cb34d03050 L9
000001cb`34d03050  000001cb`34d02e48 000001cb`34d02e70 ffffffff`2e400c46
000001cb`34d03068  000001cb`34d02e98 000001cb`34d02ec0 00000000`34a6e611
000001cb`34d03080  00000000`00000000 00000000`00000000 00000000`00000000

결국 지난 글에 작성한 do_hashtable.py를 그대로 사용해 첫 번째 원소의 주솟값을 기준으로 값을 덤프할 수 있습니다.

0:006> !py d:\wext\do_hashtable.py 000001cb34d03050 3
[0] key: "TEST", value:  "1abc"
[1] key: "qwer", value:  "2def"
[2] (empty)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 12/21/2019]

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

비밀번호

댓글 작성자
 



2020-12-22 04시32분
rodneyviana/netext
; https://github.com/rodneyviana/netext#wdict---dump-items-in-a-dictionary-type

!wdict [...addr_of_dictionary_instance...]
정성태

... 151  152  153  154  155  156  157  158  159  160  161  162  [163]  164  165  ...
NoWriterDateCnt.TitleFile(s)
972정성태1/7/201124187개발 환경 구성: 95. SQL Server 2008 R2 이하 버전 정보 확인
971정성태1/5/201133763.NET Framework: 199. .NET 코드 - Named Pipe 닷넷 서버와 VC++ 클라이언트 제작 [2]파일 다운로드1
970정성태1/4/201134291.NET Framework: 198. 윈도우 응용 프로그램에 Facebook 로그인 연동 [1]파일 다운로드1
969정성태12/31/201040365VC++: 45. Winsock 2 Layered Service Provider - Visual Studio 2010용 프로젝트 [1]파일 다운로드1
968정성태12/30/201026650개발 환경 구성: 94. 개발자가 선택할 수 있는 윈도우에서의 네트워크 프로그래밍 기술 [2]
967정성태12/27/201028425.NET Framework: 197. .NET 코드 - 단일 Process 실행파일 다운로드1
966정성태12/26/201026375.NET Framework: 196. .NET 코드 - 창 흔드는 효과파일 다운로드1
965정성태12/25/201027889개발 환경 구성: 93. MSBuild를 이용한 닷넷 응용프로그램의 다중 어셈블리 출력 빌드파일 다운로드1
964정성태12/21/2010143046개발 환경 구성: 92. 윈도우 서버 환경에서, 최대 생성 가능한 소켓(socket) 연결 수는 얼마일까? [14]
963정성태12/13/201027907개발 환경 구성: 91. MSBuild를 이용한 닷넷 응용프로그램의 플랫폼(x86/x64)별 빌드 [2]파일 다운로드1
962정성태12/10/201022766오류 유형: 110. GAC 등록 - Failure adding assembly to the cache: Invalid file or assembly name.
961정성태12/10/201099809개발 환경 구성: 90. 닷넷에서 접근해보는 PostgreSQL DB [5]
960정성태12/8/201045131.NET Framework: 195. .NET에서 코어(Core) 관련 CPU 정보 알아내는 방법파일 다운로드1
959정성태12/8/201031955.NET Framework: 194. Facebook 연동 - API Error Description: Invalid OAuth 2.0 Access Token
958정성태12/7/201028952개발 환경 구성: 89. 배치(batch) 파일에서 또 다른 배치 파일을 동기 방식으로 실행 및 반환값 얻기 [2]
957정성태12/6/201031712디버깅 기술: 31. Windbg - Visual Studio 디버그 상태에서 종료해 버리는 응용 프로그램 [3]
953정성태11/28/201036928.NET Framework: 193. 페이스북(Facebook) 계정으로 로그인하는 C# 웹 사이트 제작 [5]
952정성태11/25/201025356.NET Framework: 192. GC의 부하는 상대적인 것! [4]
950정성태11/18/201076726.NET Framework: 191. ClickOnce - 관리자 권한 상승하는 방법 [17]파일 다운로드2
954정성태11/29/201048716    답변글 .NET Framework: 191.1. [답변] 클릭원스 - 요청한 작업을 수행하려면 권한 상승이 필요합니다. (Exception from HRESULT: 0x800702E4) [2]
949정성태11/16/201027254오류 유형: 109. System.ServiceModel.Security.SecurityNegotiationException
948정성태11/16/201036080.NET Framework: 190. 트위터 계정으로 로그인하는 C# 웹 사이트 제작 [7]파일 다운로드1
947정성태11/14/201041695.NET Framework: 189. Mono Cecil로 만들어 보는 .NET Decompiler [1]파일 다운로드1
946정성태11/11/201041554.NET Framework: 188. .NET 64비트 응용 프로그램에서 왜 (2GB) OutOfMemoryException 예외가 발생할까? [1]파일 다운로드1
945정성태11/11/201025049VC++: 44. C++/CLI 컴파일 오류 - error C4368: mixed types are not supported
944정성태11/11/201031587VC++: 43. C++/CLI 컴파일 오류 - error C2872: 'IServiceProvider' : ambiguous symbol could be ...
... 151  152  153  154  155  156  157  158  159  160  161  162  [163]  164  165  ...