thug/Code/Sys/ngps/p_timer.cpp
2016-02-14 08:39:12 +11:00

638 lines
17 KiB
C++

/*****************************************************************************
** **
** Neversoft Entertainment **
** **
** Copyright (C) 1999 - All Rights Reserved **
** **
******************************************************************************
** **
** Project: SYS Library **
** **
** Module: Timer (TMR) **
** **
** File name: ngps/p_timer.cpp **
** **
** Created by: 01/13/00 - mjb **
** **
** Description: NGPS System Timer **
** **
*****************************************************************************/
/*****************************************************************************
** Includes **
*****************************************************************************/
#include <eeregs.h>
#include <sys/timer.h>
#include <sys/profiler.h>
#include <gel/scripting/script.h>
#include <sys/config/config.h>
#ifdef TESTING_REPLAY
static volatile uint64 s_num_game_renders = 0;
uint64 GameRenders( void )
{
return ( s_num_game_renders );
}
#endif
//uint32 s_ints;
/*****************************************************************************
** DBG Information **
*****************************************************************************/
namespace Tmr
{
/*****************************************************************************
** Externals **
*****************************************************************************/
/*****************************************************************************
** Defines **
*****************************************************************************/
/*****************************************************************************
** Private Types **
*****************************************************************************/
/*****************************************************************************
** Private Data **
*****************************************************************************/
static sint s_timer_handler_id = -1;
#ifdef __USE_PROFILER__
static volatile uint64 s_count;
// K: This is a uint32 version of s_count, used by the now inline function GetTimeInCPUCycles
// defined in timer.h
volatile uint32 gCPUTimeCount;
#endif
static uint64 GetGameVblanks( void );
#define vSMOOTH_N 4
volatile uint32 s_vblank = 0; // updated every vblanks under intrrupts
Time s_milliseconds = 0; // updated every call to VSync() or VSync1()
static volatile uint64 s_total_vblanks = 0;
static float s_slomo = 1.0f;
static uint64 s_render_frame = 0;
static uint64 s_stored_vblank = 0;
struct STimerInfo
{
float render_length;
float frame_length;
double uncapped_render_length;
int render_buffer[vSMOOTH_N];
uint32 render_vblank;
uint32 render_last_vblank;
int render_index;
};
static STimerInfo gTimerInfo;
static STimerInfo gStoredTimerInfo;
static void InitTimerInfo( void )
{
int i;
gTimerInfo.render_length = 1; // defualt to 1 frame, so 60/50 fps
gTimerInfo.frame_length = 1.0f/60.0f; // default to 1/60th, will be correct after first frame is rendered
gTimerInfo.uncapped_render_length = 1; // default to 1 (uncappe is used by network code)
for ( i = 0; i < vSMOOTH_N; i++ )
{
gTimerInfo.render_buffer[ i ] = 1;
}
gTimerInfo.render_vblank = 0;
gTimerInfo.render_last_vblank = 0;
gTimerInfo.render_index = 0;
gStoredTimerInfo = gTimerInfo;
s_stored_vblank = 0;
s_vblank = 0;
#ifdef TESTING_REPLAY
s_num_game_renders = 0;
#endif
}
/*****************************************************************************
** Public Data **
*****************************************************************************/
/*****************************************************************************
** Private Prototypes **
*****************************************************************************/
/*****************************************************************************
** Private Functions **
*****************************************************************************/
#ifdef __USE_PROFILER__
static sint timer_handler ( sint ca )
{
// s_ints++;
#if 1
if (( ca == INTC_TIM0 ) && ( *T0_MODE & 0x400 )) // compare interrupt
{
*T0_COUNT = 0; // reset s_count
*T0_MODE |= 0x400; // clear compare flag
s_count++;
++gCPUTimeCount;
}
#else
if (( ca == INTC_TIM0 ) && ( *T0_MODE & 0x800 )) // overflow interrupt
{
*T0_MODE |= 0x800; // clear overflow
s_count += 0x10000;
}
#endif
ExitHandler(); // Mick: Sony requires interrupts to end with an ExitHandler() call
// this enables interrupts
return 0;
}
#endif
static sint s_vsync_handler_id = -1;
static sint s_vsync_handler ( sint ca )
{
// Warning! Don't do any floating point calculations here, or game will crash,
// cos interrupt handlers are strange that way.
// Also, if writing to a variable, make sure it is volatile (or game will crash)
// inc vblanks
s_vblank++;
s_total_vblanks++;
ExitHandler(); // Mick: Sony requires interrupts to end with an ExitHandler() call
// this enables interrupts
return 0;
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void Init( void )
{
InitTimerInfo( );
#ifdef __USE_PROFILER__
s_timer_handler_id = AddIntcHandler ( INTC_TIM0, timer_handler, 0 );
Dbg_MsgAssert (( s_timer_handler_id >= 0 ),( "Failed to add timer handler" ));
#endif
s_vsync_handler_id = AddIntcHandler ( INTC_VBLANK_S, s_vsync_handler, 0 );
Dbg_MsgAssert (( s_vsync_handler_id >= 0 ),( "Failed to add vsync handler" ));
#ifdef __USE_PROFILER__
s_count = 0;
*T0_COUNT = 0;
*T0_COMP = vHIREZ_CMP; // used for hi-res counter (milli-second interrupt)
*T0_HOLD = 0;
#if 1
s_count = 0;
gCPUTimeCount = 0;
*T0_MODE = 0xc80; // clear overflow and equal flag
*T0_MODE = 0x180; // compare int enabled, start clock, speed 150MHz
#else
*T0_MODE = 0xc81; // clear overflow and equal flag
*T0_MODE = 0x281; // overflow int enabled, start clock, speed 150Mhz/16
#endif
EnableIntc ( INTC_TIM0 );
#endif
EnableIntc ( INTC_VBLANK_S );
}
/******************************************************************/
/* */
/* */
/******************************************************************/
void DeInit(void)
{
#ifdef __USE_PROFILER__
Dbg_MsgAssert (( s_timer_handler_id >= 0 ),( "Invalid Timer Handler (%d)", s_timer_handler_id ));
DisableIntc ( INTC_TIM0 );
RemoveIntcHandler ( INTC_TIM0, s_timer_handler_id );
#endif
Dbg_MsgAssert (( s_vsync_handler_id >= 0 ),( "Invalid VSync Handler (%d)", s_vsync_handler_id ));
RemoveIntcHandler ( INTC_VBLANK_S, s_vsync_handler_id );
s_timer_handler_id = -1;
}
/*****************************************************************************
** Public Functions **
*****************************************************************************/
// when pausing the game, call this to store the current state
// of OncePerRender( ) (only essential in replay mode)
void StoreTimerInfo( void )
{
//if ( Replay::GetReplayMode( ) == Replay::REPLAY_MODE_OFF )
//{
// return;
//}
//gStoredTimerInfo = gTimerInfo;
//s_stored_vblank = s_vblank;
}
void RecallTimerInfo( void )
{
/*if ( Replay::GetReplayMode( ) == Replay::REPLAY_MODE_OFF )
{
return;
}
gTimerInfo = gStoredTimerInfo;
s_vblank = s_stored_vblank;
if ( Replay::GetReplayMode( ) & ( Replay::REPLAY_MODE_PLAY_SAVED | Replay::REPLAY_MODE_PLAY_AFTER_RUN ) )
{
s_vblank += Replay::GetLastFrameLength( );
}*/
}
// Call this function once per rendering loop, to increment the
// m_render_frame variable
// This function should be synchronized in some way to the vblank, so that it is called
// directly after the vblank that rendering starts on
void OncePerRender()
{
s_render_frame++;
// printf ("%d ints/second\n", s_ints * 60 / s_vblank);
#ifdef TESTING_REPLAY
if ( s_num_game_renders < 5 )
{
Dbg_Message( "** OncePerRender %d **", s_num_game_renders );
}
s_num_game_renders++;
#endif
#ifdef STOPWATCH_STUFF
Script::IncrementStopwatchFrameIndices();
Script::DumpFunctionTimes();
Script::ResetFunctionCounts();
#endif
int diff;
gTimerInfo.render_last_vblank = gTimerInfo.render_vblank;
/*
Replay::ReplayMode replayMode;
replayMode = Replay::GetReplayMode( );
if ( replayMode & ( Replay::REPLAY_MODE_PLAY_SAVED | Replay::REPLAY_MODE_PLAY_AFTER_RUN ) )
{
diff = Replay::GetFrameLength( );
gTimerInfo.render_vblank += diff;
}
else
{
gTimerInfo.render_vblank = GetGameVblanks();
diff = (int)(gTimerInfo.render_vblank - gTimerInfo.render_last_vblank);
if ( Replay::REPLAY_MODE_RECORD == replayMode )
{
if ( Replay::FixSmoothing( ) ) // the 'smoothing' feature will fuck up our logic if,
// during replay, there is a framerate difference
// between the initial frames of play/record before
// getting our data synced up.
{
diff = 1; // keep it at 60 hz before synchronization.
}
Replay::StoreFrameLength( diff );
}
}
*/
gTimerInfo.render_vblank = GetGameVblanks();
diff = (int)(gTimerInfo.render_vblank - gTimerInfo.render_last_vblank);
gTimerInfo.render_buffer[ gTimerInfo.render_index++] = diff;
if ( gTimerInfo.render_index == vSMOOTH_N)
{
gTimerInfo.render_index = 0;
}
int total = 0;
int uncapped_total = 0;
for (int i=0;i<vSMOOTH_N;i++)
{
diff = gTimerInfo.render_buffer[i];
uncapped_total += diff;
if (diff > 10 || diff <= 1) // handle very bad values
{
diff = 1;
}
if (diff >4) // limit to 4
{
diff = 4;
}
total += diff;
}
gTimerInfo.render_length = (float)total/(float)vSMOOTH_N;
gTimerInfo.uncapped_render_length = (double)uncapped_total/(double)vSMOOTH_N;
gTimerInfo.frame_length = gTimerInfo.render_length/(float)Config::FPS()*GetSlomo();
// printf ("%d: %d, %f\n",(int)GetRenderFrame(),diff, gTimerInfo.render_length);
}
uint64 GetRenderFrame( void )
{
return s_render_frame;
}
void Vblank( void )
{
s_total_vblanks++;
s_vblank++;
}
static uint64 GetGameVblanks( void )
{
return s_vblank;
}
uint64 GetVblanks( void )
{
return ( s_total_vblanks );
}
// float FrameLength()
// returns the PROJECTED length of the next frame, for use in logic calculations
// This time is in SECONDS, so typically will be 0.016666 if running at 1 60th
// done by averaging the frame lengths of the last N frames (default N = 4)
// at 60fps (NTSC), this would return 1/60
// at 50fps (PAL), this would return 1/50
// complications arise when the framerate changes
// we want the physics to move smoothly and consistently at all framerates
// however, all we know is the number of ticks the last frames took
// we also need to ensure that the total of all the "FrameLength" calls will
// reflect the actual time that has passed
// | | | | | | | | | | | | | | | | | |
// say we have 1 1 1 . 2 1 1 1 . 2 1 . 2 1 1 1 . 2
// (10x1 + 4x2 = 18 frames
//
//
// now, let's work in 1/60ths of a second for now....
// let's take the average of the last 4 frames, as the logic time we will return for this frame
//
// 1 1 1 1.25 1.25 1.25 1.25 1.25 1.25 1.5 1.5 1.25 1.25 1.25
//
// the reason we do the averaging is to smooth out discontinuities
// if we simply use the length of the previous frame, then if we were to be alternating between 1 and 2 frames
// then the logic would move 2 on frames of length 1, and move 1 on frames of length 2
// this would essentially make the skater visually move alternately .5x and 2x his actual speed
// this looks incredibly jerky, as there is a 4x difference per frame
// by smoothing it out, you would move 1.5 on every frame, giving relative motion of 0.6666 and 1.3333,
// this is still jerky, but half as much as the unsmoothed version.
void SetSlomo(float slomo)
{
s_slomo = slomo;
}
float GetSlomo()
{
return s_slomo;
}
float FrameLength()
{
return gTimerInfo.frame_length;
// return gTimerInfo.render_length/(float)Config::FPS()*GetSlomo();
}
double UncappedFrameLength()
{
return gTimerInfo.uncapped_render_length/(float)Config::FPS();
}
uint64 now;
// wait until Vsyncs changes
void VSync()
{
now = GetVblanks();
while (now == GetVblanks());
now = GetVblanks();
s_milliseconds = (Time) (((float)(uint32)s_vblank)*(1000.0f / (float)Config::FPS()));
}
// wait until VSync changes from what it was
// (Might return immediately, if it's already changed)
void VSync1()
{
while (now == GetVblanks());
now = GetVblanks();
s_milliseconds = (Time) (((float)(uint32)s_vblank)*(1000.0f / (float)Config::FPS()));
}
#if 1
// K: These are now inline in timer.h
/*
MicroSeconds GetTimeInUSeconds ( void )
{
uint64 high0, low0, high1, low1;
high0 = s_count;
low0 = *T0_COUNT;
high1 = s_count;
low1 = *T0_COUNT;
if ( high0 == high1 )
{
return ((high0 * vHIREZ_CMP) + low0)/150;
}
else
{
return ((high1 * vHIREZ_CMP) + low1)/150;
}
}
// Added by Ken, to get higher resolution when timing Script::Update, which needs to
// subtract out the time spent in calling C-functions to get an accurate measurement
// of how long the actual script interpretation is taking.
CPUCycles GetTimeInCPUCycles ( void )
{
uint64 high0, low0, high1, low1;
high0 = s_count;
low0 = *T0_COUNT;
high1 = s_count;
low1 = *T0_COUNT;
if ( high0 == high1 )
{
return ((high0 * vHIREZ_CMP) + low0);
}
else
{
return ((high1 * vHIREZ_CMP) + low1);
}
}
*/
/******************************************************************/
/* */
/* */
/******************************************************************/
void RestartClock( void )
{
InitTimerInfo( );
}
Time GetTime ( void )
{
//return ( GetTimeInUSeconds() / 1000 );
//return s_count;
// A less accurate time, but much faster than doing a 64 bit divide.
//return ((int)s_vblank)*1000/60;
// Ken Note: The casting of s_vblank (which is a uint64) to a uint32 before casting to a float is important. If the
// uint64 is cast straight to a float it will do tons of floating point stuff which take ages.
//
// (Mick: updated to use vVBLANK_HZ)
// if ( Replay::UseNormalClock( ) )
// {
return (int) (((float)(uint32)s_vblank)*(1000.0f / (float)Config::FPS()));
// return (int) s_milliseconds;
// }
// else
// {
// return (int) (((float)(uint32)gTimerInfo.render_vblank )*(1000.0f/ (float)Config::FPS()));
// }
}
/******************************************************************/
/* */
/* */
/******************************************************************/
#else // __NOPT_PROFILE__
#ifdef __USER_MATT__
#error Wrong Motherfuckin Timer
#endif
Time Manager::GetTime ( void )
{
uint64 high0, low0, high1, low1;
high0 = s_count;
low0 = *T0_COUNT;
high1 = s_count;
low1 = *T0_COUNT;
if ( high0 == high1 )
{
return static_cast<uint64>(( high0 | ( low0 & 0xffff )) / 9375 );
}
else
{
return static_cast<uint64>(( high1 | ( low1 & 0xffff )) / 9375 );
}
}
/******************************************************************/
/* */
/* */
/******************************************************************/
// K: This is now inline in timer.h
/*
MicroSeconds GetTimeInUSeconds ( void ) // remove ?
{
uint64 high0, low0, high1, low1;
high0 = s_count;
low0 = *T0_COUNT;
high1 = s_count;
low1 = *T0_COUNT;
if ( high0 == high1 )
{
return static_cast<uint64>(( high0 + ( low0 & 0xffff )) / 9 );
}
else
{
return static_cast<uint64>(( high1 + ( low1 & 0xffff )) / 9 );
}
}
*/
#endif //
/******************************************************************/
/* */
/* */
/******************************************************************/
} // namespace Tmr