출처 : http://www.gpgstudy.com/gpgiki/시간%20다루기#header1
Ansi C의 time(), _ftime()
Ansi 표준 C 라이브러리의 함수들로, time()은 최소 1초 단위의 값을 돌려준다. 메인 루프 제어용으로는 부적합하나 게임 저장 날짜/시간의 기록 등에는 유용할 것임. _ftime()은 밀리초(1000분의 1초) 단위의 값을 돌려준다.
Win32 API의 timeGetTime() 함수
흔히 멀티미디어 타이머라고 하는 것으로, 윈도우즈가 시작된 이후 흐른 시스템 시간을 돌려준다. 단위는 밀리초. 기본 정밀도는 Win9x의 경우 1 밀리초, NT 패밀리의 경우 5 밀리초 이상이라고 함. 이정도면 _ftime()과 함께 게임에서 써먹을 수 있을 만한 함수.
타이머의 해상도는 timeGetDevCaps()와 timeBeginPeriod()로 조정할 수 있다.
Win32 API의 WM_TIMER 메시지
일정한 시간 주기로 발생하는 메시지. 주기의 최소 단위는 역시 밀리초 단위. 메시지 펌프에서 잡거나 SetTimer()를 이용해서 콜백 함수를 호출하게 만들 수 있음. 그러나 메시지의 우선 순위가 낮기 때문에 정확하지 못하다.
Win32 API의 GetTickCount() 함수
시스템 틱 카운트를 돌려주며, 단위는 밀리초. Win9x의 경우 기본 정밀도는 약 55 밀리초, NT 급은 10에서 16 정도... timeGetTime()보다 못함...
고해상도 타이머
하드웨어가 고해상도 타이머를 지원하는 경우라면 가장 좋은 선택으로, 매우 정밀한 시간을 얻을 수 있다. 얻을 수 있는 시간의 해상도는 하드웨어마다 다르나, 어쨌든 적어도 위에 나온 것들보다는 정밀하다고 함.
관련 함수(자세한 사항은 MSDN 참고)
둘 다 windows.h만 포함시키면 됨...
BOOL QueryPerformanceCounter(
LARGE_INTEGER *lpPerformanceCount // pointer to counter value
);
BOOL QueryPerformanceFrequency(
LARGE_INTEGER *lpFrequency // address of current frequency
);
고해상도 타이머의 해상도 및 지원 여부 알아내기
QueryPerformanceFrequency(&ticksPerSecond) 의 반환값이 0이 아니면 고해상도 타이머를 지원하는 것이다. 이 때 tickPerSecond에는 초 당 틱 수가 설정된다.
예:
LARGE_INTEGER ticksPerSecond;
if (!QueryPerformanceFrequency(&ticksPerSecond))
{
// 지원하지 않음
return false;
}
현재 시간 얻기
QueryPerformanceCounter(&ticks)를 호출하면 ticks에 성능 카운터 수(음.. 컴퓨터가 켜진 후 지나간 틱 수)가 설정된다.
이를 위에서 얻은 ticksPerSecond로 나누면 초 단위의 시간을 얻을 수 있다.
예:
LARGE_INTEGER ticks;
QueryPerformanceCounter(&ticks);
float seconds = ((float)ticks.QuadPart / (float)ticksPerSecond.QuadPart;
지나간 시간 얻기
게임에서 실제로 필요한 것은 이전 프레임으로부터 흐른 시간인데, 그냥 현재 시간에서 이전 시간을 빼면 됨...
예:
float GetElapsedSeconds(unsigned long elapsedFrames = 1)
{
static LARGE_INTEGER s_lastTime = m_startTime;
LARGE_INTEGER currentTime;
QueryPerformanceCounter(¤tTime);
float seconds = ((float)currentTime.QuadPart -
(float)s_lastTime.QuadPart) / (float)m_ticksPerSecond.QuadPart;
// reset the timer
s_lastTime = currentTime;
return seconds;
} // end GetElapsedSeconds()
QueryPerformanceCounter의 문제점
일부 메인보드/칩셋에서 시간이 건너뛴다는 보고가 있음.
- http://www.etestinglabs.com/bi/cont2000/200012/perfcnt.asp - System clock problem can inflate benchmark scores
- http://support.microsoft.com/default.aspx?sd=msdn&scid=kb;en-us;274323 - PRB: Performance Counter Value May Unexpectedly Leap Forward
- GpgStudy 포럼의 관련 논의
RDTSC
인텔 펜티엄 계열 CPU에서 제공하는 어셈블리 명령이다. 펜티엄은 내부적으로 TSC(Time Stamp Counter)라는 64비트 카운터를 유지하는데 이 카운터의 값은 클럭 사이클마다 증가한다. RDTSC 명령은 내부 TSC 카운터의 값을 EDX와 EAX 레지스터에 복사하는 명령이다. 이 명령은 6~11 클럭을 소요한다. 고해상도 타이머가 이 명령을 이용해 구현되었다고 한다.
참고:
- Using the RDTSC Instruction for Performance Monitoring
- RDTSC Timer Class, by Paul Hope
- High-performance Timing Library, by Larry Hastings
참고 자료
- MSDN
- 고해상도 타이머 예제 코드 : OpenGlGameProgramming 19장
시간에 기반한 게임 갱신
고정 시간 간격
한 프레임에 걸리는 시간을 고정시키는 것. 간단히 말하면, 한 프레임에 걸리는 표준 목표 시간을 설정하고, 한 프레임에 걸린 시간이 그보다 작으면 남는 부분만큼 아무 일도 하지 않고 기다리는 것이다.
의사코드:
게임 메인 루프 {
현재시각을 얻는다.
한 프레임 처리;
흐른시간 = 현재시각 - 아까 얻은 현재시각
만일 ( 흐른시간 < 목표시간) {
(목표시간 - 흐른시간)만큼 쉰다. //예: Sleep(...);
}
- 장점: 시간 관리가 직관적이고 간편하다(시간 자체를 중심으로 바라볼 때). 또한 멀티미디어와의 시간 동기화가 편하다(애니메이션과 배경음악/대사를 일치시키는 등).
- 단점: 한 프레임 처리에 걸린 시간이 목표시간보다 더 긴 경우 게임의 속도가 일정하지 않게 된다. 또한 최소 사양 이하의 컴퓨터라면 게임 전체가 느리게 돌아가게 된다.
가변 시간 간격
게임의 기능 및 객체들이 이전 프레임 이후 흐른 시간(이하 프레임 시간)에 기반해서 갱신되게 하는 것. 예를 들어 어떤 객체가 1초당 10미터를 수평 이동한다고 하면, 객체의 x 좌표는:
x = x + 10 * (프레임 시간)
이렇게 하면 프레임 시간에 상관없이(즉 CPU 속도나 기타 다른 요인과 독립적으로) 객체는 항상 초 당 10 미터의 속력을 가지게 된다.
- 장점: 객체의 입장으로 바라볼 때 직관적. 게임을 실세계의 단위(미터, 초 등)에 기반해서 모델링할 때 크게 유용할 수 있다.
- 단점(?): 모든 객체들의 갱신 방식을 시간에 기반한 함수로 생각할 수 있어야 한다. 3D라면 당연하지만 전통적인 2D 게임이라면 사고의 전환이 필요함...
더 읽을거리
- 시간 간격 조절 패턴 : TimeStepRegulator
- 시간에 관련된 GpgStudy 포럼 논의들: GpgStudyForum:1191, GpgStudyForum:7101
- Gamza.net의 가변 프레임율이 지원되는 FrameSkip
- flipCode의 고정 간격 팁+토론
- gamedev.net의 Timing Pitfalls and Solutions
- 마이크로소프트웨어 1999년 2월호 테크니컬 컬럼.
'Tech & IT > 프로그래밍' 카테고리의 다른 글
"MFC에 RichEdit 1.0 컨트롤 사용"에 대한 잘못된 설명서 (0) | 2008.11.19 |
---|---|
간단 API Tip (0) | 2008.11.18 |
CPP Unit 관련 자료 (0) | 2008.10.31 |
[Scrap] C++ ReplaceAll 함수 (0) | 2008.09.17 |
HexToString code조각 (0) | 2008.09.08 |