Microsoft MVP성태의 닷넷 이야기
.NET Framework: 460. GetTickCount / GetTickCount64와 0x7FFE0000 주솟값 [링크 복사], [링크+제목 복사],
조회: 31019
글쓴 사람
정성태 (techsharer at outlook.com)
홈페이지
첨부 파일

GetTickCount / GetTickCount64와 0x7FFE0000 주솟값

GetTickCount API는 시스템이 시작된 이후 경과한 시간을 밀리초 단위로 반환합니다. (10ms ~ 16ms 단위입니다.)

GetTickCount function
; https://learn.microsoft.com/en-us/windows/win32/api/sysinfoapi/nf-sysinfoapi-gettickcount

위의 문서에도 나와있지만 이것의 반환값이 "DWORD"라서 32비트이므로 경과 시간을 표현하는데 한계가 있습니다. 대략 49.7일이 지나면 다시 0으로 돌아온다는 것이지요. 그래서 이 문제를 해결하려면 GetTickCount64를 쓰라고 권장하고 있습니다.

GetTickCount64 function
; https://learn.microsoft.com/en-us/windows/win32/api/sysinfoapi/nf-sysinfoapi-gettickcount64

위의 문서에 추가된 덧글에 보면 (64비트까지 확장되었으므로) 기간이 "585 million years"까지 늘어나기 때문에 기간 걱정은 할 필요가 없어진 것입니다.

새삼스럽게 GetTickCount 이야기를 꺼낸 것은 재미있는 글을 보았기 때문입니다. ^^

GetTickCount를 대체할만한 방법 있을까요? 
; http://lab.gamecodi.com/board/zboard.php?id=GAMECODILAB_QnA_etc&no=3187&z=

위의 덧글 중에 보면 0x7FFE0000라는 주소에 32비트/64비트 윈도우 운영체제 공통으로 KSYSTEM_TIME이라는 구조체 값이 들어 있다는 것입니다.

typedef struct _KSYSTEM_TIME  
{  
     ULONG LowPart;  
     LONG High1Time;  
     LONG High2Time;  
} KSYSTEM_TIME, *PKSYSTEM_TIME;

C#으로 이 값을 구해 보면 다음과 같이 코딩할 수 있습니다.

using System;
using System.Runtime.InteropServices;

class Program
{
    static void Main(string[] args)
    {
        IntPtr ptr = new IntPtr(0x7FFE0008);
        KSYSTEM_TIME systemTime = new KSYSTEM_TIME();
        Marshal.PtrToStructure(ptr, systemTime);

        Console.WriteLine("LowPart: " + systemTime.LowPart);
        Console.WriteLine("HighTime: " + systemTime.HighTime);
        Console.WriteLine("High2Time: " + systemTime.High2Time);

        ulong fullTime = (ulong)systemTime.LowPart | ((ulong)systemTime.HighTime << 32);
        Console.WriteLine("KSYSTEM_TIME: \t\t" + fullTime);
/*

출력결과

LowPart:   2242934063
HighTime:  102
High2Time: 102
KSYSTEM_TIME:  441971822558
*/
    }
}

[StructLayout(LayoutKind.Sequential)]
public class KSYSTEM_TIME
{
    public uint LowPart;
    public int HighTime;
    public int High2Time;
}


재미있는 것은 GetTickCount/64 결과와의 비교값입니다.

Console.WriteLine("Environment.TickCount: \t" + Environment.TickCount);
Console.WriteLine("GetTickCount: \t\t" + GetTickCount());
Console.WriteLine("GetTickCount64: \t" + GetTickCount64());

위의 코드를 한꺼번에 실행시키면 다음과 같은 결과를 얻을 수 있습니다.

KSYSTEM_TIME:           442837484163
Environment.TickCount:  44283734
GetTickCount:           44283750
GetTickCount64:         44283750

Environment.TickCount는 내부적으로 GetTickCount를 부를 것이므로 값이 같아야 하지만 JIT 컴파일 시간으로 인해 약간 차이가 벌어지는 것 같습니다. JIT 컴파일로 인한 부작용을 없애기 위해 한번 컴파일된 상태로 호출하면 다음과 같이 마지막 3개의 값이 대개 동일하게 나옵니다.

KSYSTEM_TIME:           442837503715
Environment.TickCount:  44283750
GetTickCount:           44283750
GetTickCount64:         44283750

문제는 KSYSTEM_TIME값과 TickCount의 관계입니다. 위의 값을 보면 KSYSTEM_TIME 값에서 정밀도를 절삭한 유형이 TickCount로 해석되는 듯합니다. 즉, GetTickCount가 49.7일이 지나서 0부터 다시 시작한다면 KSYSTEM_TIME 역시 정밀도만 좋은 것일 뿐 49.7일에 대한 문제는 해결되지 않는 것으로 보입니다. (따라서, KSYSTEM_TIME 값이 GetTickCount에 대한 해결책으로 보이지는 않습니다.)

제가 해당 API를 설계한 사람이 아니어서 더 이상의 해석은 불가능하지만, 암튼 0x7FFE0000 이라는 특수한 주소를 알았다는 사실만으로 오늘은 만족해야겠습니다. ^^

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




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







[최초 등록일: ]
[최종 수정일: 1/3/2024]

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

비밀번호

댓글 작성자
 



2015-09-08 07시29분
[deepdark] 맨 아래 49.7일에 대해서 테스트는 해 보지 못했지만..
GetTickCount()와 GetTickCount64() 가 모두 같이 참조하는 메모리 주소로 보이고, 말씀하신대로 정밀도와 관련된 하부 얼마는 버리는 것이 맞는 것 같습니다.
다만, GetTickCount()의 경우 DWORD에 값을 받아오므로 Overflow 되는 부분에 대해 0부터 다시 시작하는 것이고, GetTickCount64()를 생각해볼때,
49.7일에 무관하게 더 큰 값이 기록될 것으로 보입니다.
[guest]
2015-09-08 07시47분
@deepdark 님, 이 글에서 GetTickCount64가 49.7일에 제약받는다고 쓰지 않았습니다. 처음 부분에도 나오지만 GetTickCount64는 "585 million years" 동안의 값이 표현될 것입니다. 이 글의 요지는 KSYSTEM_TIME이 49.7일에 제약을 받는지에 대해 쓴 것입니다.

그런데, 이번 기회를 통해 다시 코드를 보니 왠지 KSYSTEM_TIME이 49.7일 이후 GetTickCount64와 동일한 값은 아니어도 어떤 식으로든 상위 8바이트 내에서의 값이 변화하지 않을까 싶습니다. 한번 테스트를 해봐야겠군요. ^^
정성태
2016-06-22 03시15분
[kernel] 단순히 HighTime이 밀리세컨드를 표현하는 게 아니라 HighTime을 32비트 왼쪽으로 밀고 10000으로 나누는 거라서 KSYSTEM_TIME도 49.7일 이후에 0으로 바뀌지는 않을 것 같습니다.
HighTime = FFFFFF <-- 이게 밀리세컨드 표현이면 4294967295 / 1000 / 60 / 60 / 24 = 49.7 이지만
((ulong)(FFFFFF << 32)) / 10000 <--- 실제로는 이것이 밀리세컨드 표현이므로 1844674406941458 / 1000 / 60 / 60 / 24 = 21350398 일정도이지 않을까요.

ps: 여담이지만 홈페이지 너무 잘보고 있습니다. 닷넷과 C#의 역사를 볼 수 있는 곳이네요
[guest]
2016-06-22 11시56분
@kernel 님 설명 감사합니다. ^^
정성태

... 136  137  138  139  140  141  [142]  143  144  145  146  147  148  149  150  ...
NoWriterDateCnt.TitleFile(s)
1504정성태9/24/201330235.NET Framework: 387. UDP 브로드캐스팅을 이용해 서비스 측의 IP 주소를 구하는 방법 [1]파일 다운로드1
1503정성태9/21/201335369개발 환경 구성: 199. Visual Studio - github 연동 [7]
1502정성태9/21/201338962개발 환경 구성: 198. Visual Studio - git을 이용한 로컬 소스 컨트롤
1501정성태9/21/201346056개발 환경 구성: 197. Visual Studio를 위한 Git 환경 설정 [5]
1500정성태9/20/201345033.NET Framework: 386. C# 버전의 한글 형태소 분석기 [1]파일 다운로드1
1499정성태9/20/201321638개발 환경 구성: 196. Windows Azure - Cloud Service의 인스턴스 타입 변경하는 방법
1498정성태9/20/201327788Windows: 76. 윈도우 8.1 / 서버 2012 R2 마이그레이션 [5]
1497정성태9/20/201360051웹: 28. IE 11로 바꾼 후 발생하는 문제 정리
1496정성태9/20/201332351Windows: 75. 윈도우 8.1, 2012 R2 설치 후 원격 접속이 안 되는 문제
1495정성태9/20/201323493웹: 27. IE 11 - YBM Sisa.com에서 검색된 영단어의 발음 기호가 안 나오는 문제
1494정성태9/13/201333097.NET Framework: 385. Html Agility Pack 소개 - 웹 문서에서 텍스트만 분리하는 방법 [2]파일 다운로드1
1493정성태9/13/201334863.NET Framework: 384. WebClient.DownloadString 문자열 인코딩 문제
1492정성태9/13/201322304오류 유형: 186. The .NET assembly 'Microsoft.Vsa' could not be found.
1491정성태9/9/201325462.NET Framework: 383. RSAParameters의 ToXmlString과 ExportParameters의 결과 비교
1490정성태9/7/201360469기타: 34. 도서: 시작하세요! C# 프로그래밍: 기본 문법부터 실전 예제까지 [7]
1489정성태9/4/201344903오류 유형: 185. 오피스 워드 파일이 저장되지 않는 문제 [2]
1488정성태8/27/201329035.NET Framework: 382. WCF에서 DataSet을 binary encoding으로 직렬화하는 방법파일 다운로드1
1487정성태8/27/201331349개발 환경 구성: 195. 로컬 PC에서의 WCF 통신을 Fiddler로 보는 방법 [1]
1486정성태8/27/201328848.NET Framework: 381. SqlCommand를 이용해 Microsoft SQL 서버의 쿼리 실행 계획을 구하는 방법파일 다운로드1
1485정성태8/26/201332544.NET Framework: 380. 프로세스 스스로 풀 덤프 남기는 방법 [3]파일 다운로드1
1484정성태8/23/201326783제니퍼 .NET: 24. 제니퍼 닷넷 적용 사례 (4) - GZIP 인코딩으로 인한 성능 하락
1483정성태8/23/201326902.NET Framework: 379. System.IO.MemoryStream, ArraySegment&lt;T&gt; 의 효율적인 사용법 [1]
1482정성태8/23/201320347.NET Framework: 378. Java / C# - 정수의 부호 유무에 따른 16진수 문자열 변환
1481정성태8/22/201321178오류 유형: 184. PaaS 유형(Cloud Services)의 Azure VM에 연결할 때 계정 만료 에러가 발생하는 경우
1480정성태8/22/201337847개발 환경 구성: 194. 윈도우 서버의 80 포트에 대한 port forwarding 설정 방법파일 다운로드1
1479정성태8/14/201325173오류 유형: 183. IIS - 바인딩 추가 시 Object reference not set to an instance of an object 오류 [5]
... 136  137  138  139  140  141  [142]  143  144  145  146  147  148  149  150  ...