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

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

비밀번호

댓글 작성자
 




... [31]  32  33  34  35  36  37  38  39  40  41  42  43  44  45  ...
NoWriterDateCnt.TitleFile(s)
12862정성태12/3/20215979오류 유형: 768. Golang - 빌드 시 "cmd/go: unsupported GOOS/GOARCH pair linux /amd64" 오류
12861정성태12/3/20218186개발 환경 구성: 609. 파이썬 - "Windows embeddable package"로 개발 환경 구성하는 방법
12860정성태12/1/20216292오류 유형: 767. SQL Server - 127.0.0.1로 접속하는 경우 "Access is denied"가 발생한다면?
12859정성태12/1/202112470개발 환경 구성: 608. Hyper-V 가상 머신에 Console 모드로 로그인하는 방법
12858정성태11/30/20219720개발 환경 구성: 607. 로컬의 USB 장치를 원격 머신에 제공하는 방법 - usbip-win
12857정성태11/24/20217146개발 환경 구성: 606. WSL Ubuntu 20.04에서 파이썬을 위한 uwsgi 설치 방법
12856정성태11/23/20218971.NET Framework: 1121. C# - 동일한 IP:Port로 바인딩 가능한 서버 소켓 [2]
12855정성태11/13/20216274개발 환경 구성: 605. Azure App Service - Kudu SSH 환경에서 FTP를 이용한 파일 전송
12854정성태11/13/20217900개발 환경 구성: 604. Azure - 윈도우 VM에서 FTP 여는 방법
12853정성태11/10/20216235오류 유형: 766. Azure App Service - JBoss 호스팅 생성 시 "This region has quota of 0 PremiumV3 instances for your subscription. Try selecting different region or SKU."
12851정성태11/1/20217628스크립트: 34. 파이썬 - MySQLdb 기본 예제 코드
12850정성태10/27/20218802오류 유형: 765. 우분투에서 pip install mysqlclient 실행 시 "OSError: mysql_config not found" 오류
12849정성태10/17/20217909스크립트: 33. JavaScript와 C#의 시간 변환 [1]
12848정성태10/17/20218876스크립트: 32. 파이썬 - sqlite3 기본 예제 코드 [1]
12847정성태10/14/20218722스크립트: 31. 파이썬 gunicorn - WORKER TIMEOUT 오류 발생
12846정성태10/7/20218489스크립트: 30. 파이썬 __debug__ 플래그 변수에 따른 코드 실행 제어
12845정성태10/6/20218325.NET Framework: 1120. C# - BufferBlock<T> 사용 예제 [5]파일 다운로드1
12844정성태10/3/20216321오류 유형: 764. MSI 설치 시 "... is accessible and not read-only." 오류 메시지
12843정성태10/3/20216773스크립트: 29. 파이썬 - fork 시 기존 클라이언트 소켓 및 스레드의 동작파일 다운로드1
12842정성태10/1/202125196오류 유형: 763. 파이썬 오류 - AttributeError: type object '...' has no attribute '...'
12841정성태10/1/20218604스크립트: 28. 모든 파이썬 프로세스에 올라오는 특별한 파일 - sitecustomize.py
12840정성태9/30/20218665.NET Framework: 1119. Entity Framework의 Join 사용 시 다중 칼럼에 대한 OR 조건 쿼리파일 다운로드1
12839정성태9/15/20219712.NET Framework: 1118. C# 11 - 제네릭 타입의 특성 적용파일 다운로드1
12838정성태9/13/20219350.NET Framework: 1117. C# - Task에 전달한 Action, Func 유형에 따라 달라지는 async/await 비동기 처리 [2]파일 다운로드1
12837정성태9/11/20218274VC++: 151. Golang - fmt.Errorf, errors.Is, errors.As 설명
12836정성태9/10/20217879Linux: 45. 리눅스 - 실행 중인 다른 프로그램의 출력을 확인하는 방법
... [31]  32  33  34  35  36  37  38  39  40  41  42  43  44  45  ...