Microsoft MVP성태의 닷넷 이야기
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일
(연관된 글이 3개 있습니다.)
(시리즈 글이 3개 있습니다.)
디버깅 기술: 57. C# - double 값에 대한 windbg 확인
; https://www.sysnet.pe.kr/2/0/1526

디버깅 기술: 97. windbg - 메모리 덤프로부터 DateTime 형식의 값을 알아내는 방법
; https://www.sysnet.pe.kr/2/0/11313

디버깅 기술: 202. windbg - ASP.NET MVC Web Application (.NET Framework) 응용 프로그램의 덤프 분석 시 요령
; https://www.sysnet.pe.kr/2/0/13741




windbg - 메모리 덤프로부터 DateTime 형식의 값을 알아내는 방법

예를 들어, 다음과 같이 DateTime을 사용하는 프로그램의

using System;

namespace ConsoleApp1
{
    class Program
    {
        DateTime _now = DateTime.Now;
        DateTime _utcNow = DateTime.UtcNow;
        DateTime _uninitialized;

        static  void Main(string[] args)
        {
            Program pg = new Program();
            Console.ReadLine();
        }
    }
}

메모리 덤프를 windbg에 로드하면 객체의 DateTime 필드를 다음과 같이 접근할 수 있습니다.

0:000> !DumpObj /d 030230dc
Name:        ConsoleApp1.Program
MethodTable: 01174d4c
EEClass:     011713c4
Size:        32(0x20) bytes
File:        C:\temp\ConsoleApp1\bin\Debug\ConsoleApp1.exe
Fields:
      MT    Field   Offset                 Type VT     Attr    Value Name
5c75f748  4000001        4      System.DateTime  1 instance 030230e0 _now
5c75f748  4000002        c      System.DateTime  1 instance 030230e8 _utcNow
5c75f748  4000003       14      System.DateTime  1 instance 030230f0 _uninitialized

그다음, 개별적으로 값을 덤프할 수 있습니다.

0:000> !DumpVC /d 5c75f748 030230e0
Name:        System.DateTime
MethodTable: 5c75f748
EEClass:     5c2fe6c4
Size:        16(0x10) bytes
File:        C:\WINDOWS\Microsoft.Net\assembly\GAC_32\mscorlib\v4.0_4.0.0.0__b77a5c561934e089\mscorlib.dll
Fields:
      MT    Field   Offset                 Type VT     Attr    Value Name
5c75d1d0  40002ce        0        System.UInt64  1 instance 9859791360354292701 dateData
5c763bc8  40002bf       68       System.Int32[]  0   shared   static DaysToMonth365
    >> Domain:Value  011eb430:0302315c <<
5c763bc8  40002c0       6c       System.Int32[]  0   shared   static DaysToMonth366
    >> Domain:Value  011eb430:0302319c <<
5c75f748  40002c1       60      System.DateTime  1   shared   static MinValue
    >> Domain:Value  011eb430:04021088 <<
5c75f748  40002c2       64      System.DateTime  1   shared   static MaxValue
    >> Domain:Value  011eb430:0402108c <<

0:000> !DumpVC /d 5c75f748 030230e8
Name:        System.DateTime
MethodTable: 5c75f748
EEClass:     5c2fe6c4
Size:        16(0x10) bytes
File:        C:\WINDOWS\Microsoft.Net\assembly\GAC_32\mscorlib\v4.0_4.0.0.0__b77a5c561934e089\mscorlib.dll
Fields:
      MT    Field   Offset                 Type VT     Attr    Value Name
5c75d1d0  40002ce        0        System.UInt64  1 instance 5248105017926914794 dateData
...[생략]...

0:000> !DumpVC /d 5c75f748 030230f0
Name:        System.DateTime
MethodTable: 5c75f748
EEClass:     5c2fe6c4
Size:        16(0x10) bytes
File:        C:\WINDOWS\Microsoft.Net\assembly\GAC_32\mscorlib\v4.0_4.0.0.0__b77a5c561934e089\mscorlib.dll
Fields:
      MT    Field   Offset                 Type VT     Attr    Value Name
5c75d1d0  40002ce        0        System.UInt64  1 instance 0 dateData
...[생략]...

리플렉션으로 보면, ulong dateData 필드가 DateTime 타입의 유일한 인스턴스 데이터임을 알 수 있습니다.

[Serializable, __DynamicallyInvokable]
public struct DateTime : IComparable, IFormattable, IConvertible, ISerializable, IComparable<DateTime>, IEquatable<DateTime>
{
    // Fields
    private ulong dateData;
    ...[생략]...

    // Methods
    ...[생략]...

    [__DynamicallyInvokable]
    public DateTime(long ticks)
    {
        if ((ticks < 0L) || (ticks > 0x2bca2875f4373fffL))
        {
            throw new ArgumentOutOfRangeException("ticks", Environment.GetResourceString("ArgumentOutOfRange_DateTimeBadTicks"));
        }
        this.dateData = (ulong) ticks;
    }

    ...[생략]...
}

따라서, (예제 코드의 _uninitialized 멤버처럼) dateData 값이 0이라는 점은 값 형식에 한 번도 값이 할당된 적이 없는 것입니다. 그리고 각각 UTC, Local에 대해 다음과 같이 dateData 값이 할당됩니다.

dateData == 5248105017926914794 (UTC)
dateData == 9859791360354292701 (Local)

이 값으로부터 어떻게 원래 시간 정보를 알 수 있을까요? DateTime 생성자에 보면 인자로 전달한 값(ticks)을 그대로 dateData 변수에 대입하는 것을 볼 수 있는데요. 실제로 그렇게 해보면 0x2bca2875f4373fffL 한계에 걸려 ArgumentOutOfRangeException 예외가 발생합니다.

Unhandled Exception: System.ArgumentOutOfRangeException: Ticks must be between DateTime.MinValue.Ticks and DateTime.MaxValue.Ticks.
Parameter name: ticks
   at System.DateTime..ctor(Int64 ticks)
   at ConsoleApp1.Program.Main(String[] args)

9859791360354292701, 5248105017926914794 값은 0x2bca2875f4373fff(10진수 3155378975999999999)값보다 크기 때문인데, 이유를 알기 위해 좀 더 추적해 보면 DateTime.Now에서 단서를 찾을 수 있습니다.

[__DynamicallyInvokable]
public static DateTime Now
{
    [__DynamicallyInvokable]
    get
    {
        DateTime utcNow = UtcNow;
        bool isAmbiguousLocalDst = false;
        long ticks = TimeZoneInfo.GetDateTimeNowUtcOffsetFromUtc(utcNow, out isAmbiguousLocalDst).Ticks;
        long num2 = utcNow.Ticks + ticks;
        if (num2 > 0x2bca2875f4373fffL)
        {
            return new DateTime(0x2bca2875f4373fffL, DateTimeKind.Local);
        }
        if (num2 < 0L)
        {
            return new DateTime(0L, DateTimeKind.Local);
        }
        return new DateTime(num2, DateTimeKind.Local, isAmbiguousLocalDst);
    }
}

[__DynamicallyInvokable]
public DateTime(long ticks, DateTimeKind kind)
{
    if ((ticks < 0L) || (ticks > 0x2bca2875f4373fffL))
    {
        throw new ArgumentOutOfRangeException("ticks", Environment.GetResourceString("ArgumentOutOfRange_DateTimeBadTicks"));
    }
    if ((kind < DateTimeKind.Unspecified) || (kind > DateTimeKind.Local))
    {
        throw new ArgumentException(Environment.GetResourceString("Argument_InvalidDateTimeKind"), "kind");
    }
    this.dateData = (ulong) (ticks | (((long) kind) << 0x3e));
}

보는 바와 같이 DateTime.dateData의 값은 원래의 시간 정보를 담은 ticks 값에 DateTimeKind 값을 OR 연산한 값이기 때문입니다.

따라서, 해당 값의 DateTimeKind를 알 수 있다면 다음과 같이 변환해서 원래의 시간 정보를 알아낼 수 있습니다.

long u = 5248105017926914794;

long mask = (long)DateTimeKind.Utc; // 또는 DateTimeKind.Local
u = (u ^ (mask << 0x3e));

DateTime dt = new DateTime(u, (DateTimeKind)mask);
Console.WriteLine(dt);

결국 DateTimeKind 값에 의해 dateData 값이 바뀌므로 이를 자동화하면 다음과 같은 식으로 코딩할 수 있습니다.

private static DateTime ToDateTime(long u)
{
    bool tryGetDateTime(long ticks, DateTimeKind kind, out DateTime result) // 로컬 함수 C# 7.0
    {
        result = DateTime.MinValue;

        long checkMask = (long)kind << 0x3e;
        long masked = ticks & checkMask;

        if (masked == checkMask)
        {
            result = new DateTime(ticks ^ checkMask, kind);
            return true;
        }

        return false;
    };

    switch (u)
    {
        // 패턴 매칭 C# 7.0
        case long value when tryGetDateTime(value, DateTimeKind.Utc, out DateTime time) == true: // out 사용 개선 C# 7.0
            return time;

        case long value when tryGetDateTime(value, DateTimeKind.Local, out DateTime time) == true: // out 사용 개선 C# 7.0
            return time;
    }

    return new DateTime(u, DateTimeKind.Unspecified);
}

이를 이용해 windbg에서 알아낸 값을 다음과 같이 친숙한 시간 값으로 바꿔줄 수 있습니다.

long u = 5248105017926914794;
DateTime dt = ToDateTime(u);
Console.WriteLine(dt); // 2017-09-25 오전 1:32:29

u = (long)9859791360354292701;
dt = ToDateTime(u);
Console.WriteLine(dt); // 2017-09-25 오전 10:32:29




참고로, DateTime 생성자를 통해 dateData 값을 넣어줄 필요 없이 Reflection을 이용하는 방법도 있습니다.

// C# - 구조체(값 형식)의 필드를 리플렉션을 이용해 값을 바꾸는 방법
// ; https://www.sysnet.pe.kr/2/0/11312

unchecked
{
    long u = 5248105017926914794;
    DateTime dt = new DateTime();

    FieldInfo fi = typeof(DateTime).GetField("dateData", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic);
    TypedReference tr = __makeref(dt);
    fi.SetValueDirect(tr, (ulong)u);
    Console.WriteLine(dt); // 2017-09-25 오전 1:32:29

    u = (long)9859791360354292701;
    fi.SetValueDirect(tr, (ulong)u);
    Console.WriteLine(dt); // 2017-09-25 오전 10:32:29
}

(첨부 파일은 이 글의 예제 코드를 포함합니다.)




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

[연관 글]






[최초 등록일: ]
[최종 수정일: 9/26/2017]

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

비밀번호

댓글 작성자
 




1  2  3  4  [5]  6  7  8  9  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13818정성태11/15/20245274Windows: 272. Windows 11 24H2 - sudo 추가
13817정성태11/14/20244929Linux: 106. eBPF / bpf2go - (BPF_MAP_TYPE_HASH) Map을 이용한 전역 변수 구현
13816정성태11/14/20245386닷넷: 2312. C#, C++ - Windows / Linux 환경의 Thread Name 설정파일 다운로드1
13815정성태11/13/20244809Linux: 105. eBPF - bpf2go에서 전역 변수 설정 방법
13814정성태11/13/20245283닷넷: 2311. C# - Windows / Linux 환경에서 Native Thread ID 가져오기파일 다운로드1
13813정성태11/12/20245032닷넷: 2310. .NET의 Rune 타입과 emoji 표현파일 다운로드1
13812정성태11/11/20245264오류 유형: 933. Active Directory - The forest functional level is not supported.
13811정성태11/11/20244848Linux: 104. Linux - COLUMNS 환경변수가 언제나 80으로 설정되는 환경
13810정성태11/10/20245374Linux: 103. eBPF (bpf2go) - Tracepoint를 이용한 트레이스 (BPF_PROG_TYPE_TRACEPOINT)
13809정성태11/10/20245261Windows: 271. 윈도우 서버 2025 마이그레이션
13808정성태11/9/20245257오류 유형: 932. Linux - 커널 업그레이드 후 "error: bad shim signature" 오류 발생
13807정성태11/9/20244994Linux: 102. Linux - 커널 이미지 파일 서명 (Ubuntu 환경)
13806정성태11/8/20244902Windows: 270. 어댑터 상세 정보(Network Connection Details) 창의 내용이 비어 있는 경우
13805정성태11/8/20244736오류 유형: 931. Active Directory의 adprep 또는 복제가 안 되는 경우
13804정성태11/7/20245366Linux: 101. eBPF 함수의 인자를 다루는 방법
13803정성태11/7/20245319닷넷: 2309. C# - .NET Core에서 바뀐 DateTime.Ticks의 정밀도
13802정성태11/6/20245703Windows: 269. GetSystemTimeAsFileTime과 GetSystemTimePreciseAsFileTime의 차이점파일 다운로드1
13801정성태11/5/20245481Linux: 100. eBPF의 2가지 방식 - libbcc와 libbpf(CO-RE)
13800정성태11/3/20246321닷넷: 2308. C# - ICU 라이브러리를 활용한 문자열의 대소문자 변환 [2]파일 다운로드1
13799정성태11/2/20244907개발 환경 구성: 732. 모바일 웹 브라우저에서 유니코드 문자가 표시되지 않는 경우
13798정성태11/2/20245505개발 환경 구성: 731. 유니코드 - 출력 예시 및 폰트 찾기
13797정성태11/1/20245493C/C++: 185. C++ - 문자열의 대소문자를 변환하는 transform + std::tolower/toupper 방식의 문제점파일 다운로드1
13796정성태10/31/20245382C/C++: 184. C++ - ICU dll을 이용하는 예제 코드 (Windows)파일 다운로드1
13795정성태10/31/20245164Windows: 268. Windows - 리눅스 환경처럼 공백으로 끝나는 프롬프트 만들기
13794정성태10/30/20245261닷넷: 2307. C# - 윈도우에서 한글(및 유니코드)을 포함한 콘솔 프로그램을 컴파일 및 실행하는 방법
13793정성태10/28/20245134C/C++: 183. C++ - 윈도우에서 한글(및 유니코드)을 포함한 콘솔 프로그램을 컴파일 및 실행하는 방법
1  2  3  4  [5]  6  7  8  9  10  11  12  13  14  15  ...