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

비밀번호

댓글 작성자
 




1  2  3  4  5  6  7  8  [9]  10  11  12  13  14  15  ...
NoWriterDateCnt.TitleFile(s)
13411정성태9/12/20233555Windows: 252. 권한 상승 전/후 따로 관리되는 공유 네트워크 드라이브 정보
13410정성태9/11/20235091닷넷: 2141. C# 12 - Interceptor (컴파일 시에 메서드 호출 재작성) [1]
13409정성태9/8/20233924닷넷: 2140. C# - Win32 API를 이용한 모니터 전원 끄기
13408정성태9/5/20233895Windows: 251. 임의로 만든 EXE 파일을 포함한 ZIP 파일의 압축을 해제할 때 Windows Defender에 의해 삭제되는 경우
13407정성태9/4/20233619닷넷: 2139. C# - ParallelEnumerable을 이용한 IEnumerable에 대한 병렬 처리
13406정성태9/4/20233606VS.NET IDE: 186. Visual Studio Community 버전의 라이선스
13405정성태9/3/20234020닷넷: 2138. C# - async 메서드 호출 원칙
13404정성태8/29/20233592오류 유형: 876. Windows - 키보드의 등호(=, Equals sign) 키가 눌리지 않는 경우
13403정성태8/21/20233389오류 유형: 875. The following signatures couldn't be verified because the public key is not available: NO_PUBKEY EB3E94ADBE1229CF
13402정성태8/20/20233491닷넷: 2137. ILSpy의 nuget 라이브러리 버전 - ICSharpCode.Decompiler
13401정성태8/19/20233742닷넷: 2136. .NET 5+ 환경에서 P/Invoke의 성능을 높이기 위한 SuppressGCTransition 특성 [1]
13400정성태8/10/20233580오류 유형: 874. 파이썬 - pymssql을 윈도우 환경에서 설치 불가
13399정성태8/9/20233513닷넷: 2135. C# - 지역 변수로 이해하는 메서드 매개변수의 값/참조 전달
13398정성태8/3/20234352스크립트: 55. 파이썬 - pyodbc를 이용한 SQL Server 연결 사용법
13397정성태7/23/20233830닷넷: 2134. C# - 문자열 연결 시 string.Create를 이용한 GC 할당 최소화
13396정성태7/22/20233611스크립트: 54. 파이썬 pystack 소개 - 메모리 덤프로부터 콜 스택 열거
13395정성태7/20/20233493개발 환경 구성: 685. 로컬에서 개발 중인 ASP.NET Core/5+ 웹 사이트에 대해 localhost 이외의 호스트 이름으로 접근하는 방법
13394정성태7/16/20233473오류 유형: 873. Oracle.ManagedDataAccess.Client - 쿼리 수행 시 System.InvalidOperationException
13393정성태7/16/20233643닷넷: 2133. C# - Oracle 데이터베이스의 Sleep 쿼리 실행하는 방법
13392정성태7/16/20233538오류 유형: 872. Oracle - ORA-01031: insufficient privileges
13391정성태7/14/20233560닷넷: 2132. C# - sealed 클래스의 메서드를 callback 호출했을 때 인라인 처리가 될까요?
13390정성태7/12/20233499스크립트: 53. 파이썬 - localhost 호출 시의 hang 현상
13389정성태7/5/20233542개발 환경 구성: 684. IIS Express로 호스팅하는 웹을 WSL 환경에서 접근하는 방법
13388정성태7/3/20233688오류 유형: 871. 윈도우 탐색기에서 열리지 않는 zip 파일 - The Compressed (zipped) Folder '[...].zip' is invalid. [1]파일 다운로드1
13387정성태6/28/20233740오류 유형: 870. _mysql - Commands out of sync; you can't run this command now
13386정성태6/27/20233799Linux: 61. docker - 원격 제어를 위한 TCP 바인딩 추가
1  2  3  4  5  6  7  8  [9]  10  11  12  13  14  15  ...