241 lines
6 KiB
C++
241 lines
6 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose: module for gathering performance stats for upload so that we can
|
|
// monitor performance regressions and improvements
|
|
//
|
|
//=====================================================================================//
|
|
|
|
|
|
#include "cbase.h"
|
|
#include "statgather.h"
|
|
|
|
// memdbgon must be the last include file in a .cpp file!!!
|
|
#include "tier0/memdbgon.h"
|
|
|
|
|
|
#define STATS_WINDOW_SIZE ( 60 * 10 ) // # of records to hold
|
|
#define STATS_RECORD_INTERVAL 1 // # of seconds between data grabs. 2 * 300 = every 10 minutes
|
|
|
|
|
|
struct StatsBufferRecord_t
|
|
{
|
|
float m_flFrameRate; // fps
|
|
|
|
};
|
|
|
|
const int PERFDATA_LEVEL = 1;
|
|
const int PERFDATA_SHUTDOWN = 2;
|
|
|
|
class CStatsRecorder : public CAutoGameSystem
|
|
{
|
|
|
|
StatsBufferRecord_t m_StatsBuffer[STATS_WINDOW_SIZE];
|
|
bool m_bBufferFull;
|
|
float m_flLastRealTime;
|
|
float m_flLastSampleTime;
|
|
|
|
template<class T> T AverageStat( T StatsBufferRecord_t::*field ) const
|
|
{
|
|
T sum = 0;
|
|
for( int i = 0; i < STATS_WINDOW_SIZE; i++ )
|
|
sum += m_StatsBuffer[i].*field;
|
|
return sum / STATS_WINDOW_SIZE;
|
|
}
|
|
|
|
template<class T> T MaxStat( T StatsBufferRecord_t::*field ) const
|
|
{
|
|
T maxsofar = -16000000;
|
|
for( int i = 0; i < STATS_WINDOW_SIZE; i++ )
|
|
maxsofar = MAX( maxsofar, m_StatsBuffer[i].*field );
|
|
return maxsofar;
|
|
}
|
|
|
|
template<class T> T MinStat( T StatsBufferRecord_t::*field ) const
|
|
{
|
|
T minsofar = 16000000;
|
|
for( int i = 0; i < STATS_WINDOW_SIZE; i++ )
|
|
minsofar = MIN( minsofar, m_StatsBuffer[i].*field );
|
|
return minsofar;
|
|
}
|
|
|
|
inline void AdvanceIndex( void )
|
|
{
|
|
m_nWriteIndex++;
|
|
if ( m_nWriteIndex == STATS_WINDOW_SIZE )
|
|
{
|
|
m_nWriteIndex = 0;
|
|
m_bBufferFull = true;
|
|
}
|
|
}
|
|
|
|
void LevelInitPreEntity()
|
|
{
|
|
m_flTimeLevelStart = gpGlobals->curtime;
|
|
}
|
|
|
|
void LevelShutdownPreEntity()
|
|
{
|
|
float flLevelTime = gpGlobals->curtime - m_flTimeLevelStart;
|
|
m_flTotalTimeInLevels += flLevelTime;
|
|
m_iNumLevels ++;
|
|
UploadPerfData( PERFDATA_LEVEL );
|
|
}
|
|
|
|
void Shutdown()
|
|
{
|
|
UploadPerfData( PERFDATA_SHUTDOWN );
|
|
}
|
|
|
|
public:
|
|
int m_nWriteIndex;
|
|
float m_flTimeLevelStart;
|
|
float m_flTotalTimeInLevels;
|
|
int m_iNumLevels;
|
|
CStatsRecorder( void )
|
|
{
|
|
m_bBufferFull = false;
|
|
m_nWriteIndex = 0;
|
|
m_flLastRealTime = -1;
|
|
m_flLastSampleTime = -1;
|
|
m_flTimeLevelStart = 0;
|
|
m_flTotalTimeInLevels = 0;
|
|
m_iNumLevels = 0;
|
|
}
|
|
|
|
char const *GetPerfStatsString( int iType );
|
|
void UploadPerfData( int iType );
|
|
void UpdatePerfStats( void );
|
|
};
|
|
|
|
char s_cPerfString[2048];
|
|
|
|
static inline char const *SafeString( char const *pStr )
|
|
{
|
|
return ( pStr ) ? pStr : "?";
|
|
}
|
|
|
|
// get the string record for sending to the server. Contains perf data and hardware/software
|
|
// info. Returns NULL if there isn't a good record to send (i.e. not enough data yet).
|
|
// A successful Get() resets the stats
|
|
char const *CStatsRecorder::GetPerfStatsString( int iType )
|
|
{
|
|
switch ( iType )
|
|
{
|
|
case PERFDATA_LEVEL:
|
|
{
|
|
if ( ! m_bBufferFull )
|
|
return NULL;
|
|
|
|
float flAverageFrameRate = AverageStat( &StatsBufferRecord_t::m_flFrameRate );
|
|
float flMinFrameRate = MinStat( &StatsBufferRecord_t::m_flFrameRate );
|
|
float flMaxFrameRate = MaxStat( &StatsBufferRecord_t::m_flFrameRate );
|
|
|
|
const CPUInformation &cpu = GetCPUInformation();
|
|
MaterialAdapterInfo_t gpu;
|
|
materials->GetDisplayAdapterInfo( materials->GetCurrentAdapter(), gpu );
|
|
|
|
CMatRenderContextPtr pRenderContext( materials );
|
|
int dest_width,dest_height;
|
|
pRenderContext->GetRenderTargetDimensions( dest_width, dest_height );
|
|
|
|
char szMap[MAX_PATH+1]="";
|
|
Q_FileBase( engine->GetLevelName(), szMap, ARRAYSIZE( szMap ) );
|
|
|
|
V_snprintf( s_cPerfString, sizeof( s_cPerfString ),
|
|
"PERFDATA:AvgFps=%4.2f MinFps=%4.2f MaxFps=%4.2f CPUID=\"%s\" CPUGhz=%2.2f "
|
|
"NumCores=%d GPUDrv=\"%s\" "
|
|
"GPUVendor=%d GPUDeviceID=%d "
|
|
"GPUDriverVersion=\"%d.%d\" DxLvl=%d "
|
|
"Width=%d Height=%d MapName=%s",
|
|
flAverageFrameRate,
|
|
flMinFrameRate,
|
|
flMaxFrameRate,
|
|
cpu.m_szProcessorID,
|
|
cpu.m_Speed * ( 1.0 / 1.0e9 ),
|
|
cpu.m_nPhysicalProcessors,
|
|
SafeString( gpu.m_pDriverName ),
|
|
gpu.m_VendorID,
|
|
gpu.m_DeviceID,
|
|
gpu.m_nDriverVersionHigh,
|
|
gpu.m_nDriverVersionLow,
|
|
g_pMaterialSystemHardwareConfig->GetDXSupportLevel(),
|
|
dest_width, dest_height, szMap
|
|
);
|
|
// get rid of chars that we hate in vendor strings
|
|
for( char *i = s_cPerfString; *i; i++ )
|
|
{
|
|
if ( ( i[0]=='\n' ) || ( i[0]=='\r' ) || ( i[0]==';' ) )
|
|
i[0]=' ';
|
|
}
|
|
|
|
// clear buffer
|
|
m_nWriteIndex = 0;
|
|
m_bBufferFull = false;
|
|
|
|
return s_cPerfString;
|
|
}
|
|
case PERFDATA_SHUTDOWN:
|
|
V_snprintf( s_cPerfString, sizeof( s_cPerfString ), "PERFDATA:TotalLevelTime=%d NumLevels=%d",
|
|
(int) m_flTotalTimeInLevels, m_iNumLevels );
|
|
return s_cPerfString;
|
|
|
|
default:
|
|
Assert( false );
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
void CStatsRecorder::UpdatePerfStats( void )
|
|
{
|
|
float flCurTime = Plat_FloatTime();
|
|
if (
|
|
( m_flLastSampleTime == -1 ) ||
|
|
( flCurTime - m_flLastSampleTime >= STATS_RECORD_INTERVAL ) )
|
|
{
|
|
if ( ( m_flLastRealTime > 0 ) && ( flCurTime > m_flLastRealTime ) )
|
|
{
|
|
float flFrameRate = 1.0 / ( flCurTime - m_flLastRealTime );
|
|
StatsBufferRecord_t &stat = m_StatsBuffer[m_nWriteIndex];
|
|
stat.m_flFrameRate = flFrameRate;
|
|
AdvanceIndex();
|
|
m_flLastSampleTime = flCurTime;
|
|
}
|
|
}
|
|
m_flLastRealTime = flCurTime;
|
|
}
|
|
|
|
static CStatsRecorder s_StatsRecorder;
|
|
|
|
void UpdatePerfStats( void )
|
|
{
|
|
s_StatsRecorder.UpdatePerfStats();
|
|
}
|
|
|
|
static void ShowPerfStats( void )
|
|
{
|
|
char const *pStr = s_StatsRecorder.GetPerfStatsString( PERFDATA_LEVEL );
|
|
if ( pStr )
|
|
Warning( "%s\n", pStr );
|
|
else
|
|
Warning( "%d records stored. buffer not full.\n", s_StatsRecorder.m_nWriteIndex );
|
|
}
|
|
|
|
static ConCommand perfstats( "cl_perfstats", ShowPerfStats, "Dump the perf monitoring string" );
|
|
|
|
|
|
// upload performance to steam, if we have any. This is a blocking call.
|
|
void CStatsRecorder::UploadPerfData( int iType )
|
|
{
|
|
if( g_pClientGameStatsUploader )
|
|
{
|
|
char const *pPerfData = GetPerfStatsString( iType );
|
|
|
|
if ( pPerfData )
|
|
{
|
|
g_pClientGameStatsUploader->UploadGameStats( "",
|
|
1,
|
|
1 + strlen( pPerfData ),
|
|
pPerfData );
|
|
}
|
|
}
|
|
}
|