mirror of
https://github.com/id-Software/DOOM-3-BFG.git
synced 2025-01-07 10:20:47 +00:00
708 lines
22 KiB
C++
708 lines
22 KiB
C++
/*
|
|
===========================================================================
|
|
|
|
Doom 3 BFG Edition GPL Source Code
|
|
Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
|
|
|
|
This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
|
|
|
|
Doom 3 BFG Edition Source Code is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
Doom 3 BFG Edition Source Code is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with Doom 3 BFG Edition Source Code. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
In addition, the Doom 3 BFG Edition Source Code is also subject to certain additional terms. You should have received a copy of these additional terms immediately following the terms and conditions of the GNU General Public License which accompanied the Doom 3 BFG Edition Source Code. If not, please request a copy in writing from id Software at the address below.
|
|
|
|
If you have questions concerning this license or the applicable additional terms, you may contact in writing id Software LLC, c/o ZeniMax Media Inc., Suite 120, Rockville, Maryland 20850 USA.
|
|
|
|
===========================================================================
|
|
*/
|
|
#include "precompiled.h"
|
|
#pragma hdrstop
|
|
|
|
#include "Common_local.h"
|
|
|
|
idCVar net_clientMaxPrediction( "net_clientMaxPrediction", "5000", CVAR_SYSTEM | CVAR_INTEGER | CVAR_NOCHEAT, "maximum number of milliseconds a client can predict ahead of server." );
|
|
idCVar net_snapRate( "net_snapRate", "100", CVAR_SYSTEM | CVAR_INTEGER, "How many milliseconds between sending snapshots" );
|
|
idCVar net_ucmdRate( "net_ucmdRate", "40", CVAR_SYSTEM | CVAR_INTEGER, "How many milliseconds between sending usercmds" );
|
|
|
|
idCVar net_debug_snapShotTime( "net_debug_snapShotTime", "0", CVAR_BOOL | CVAR_ARCHIVE, "" );
|
|
idCVar com_forceLatestSnap( "com_forceLatestSnap", "0", CVAR_BOOL, "" );
|
|
|
|
// Enables effective snap rate: dynamically adjust the client snap rate based on:
|
|
// -client FPS
|
|
// -server FPS (interpolated game time received / interval it was received over)
|
|
// -local buffered time (leave a cushion to absorb spikes, slow down when infront of it, speed up when behind it) ie: net_minBufferedSnapPCT_Static
|
|
idCVar net_effectiveSnapRateEnable( "net_effectiveSnapRateEnable", "1", CVAR_BOOL, "Dynamically adjust client snaprate" );
|
|
idCVar net_effectiveSnapRateDebug( "net_effectiveSnapRateDebug", "0", CVAR_BOOL, "Debug" );
|
|
|
|
// Min buffered snapshot time to keep as a percentage of the effective snaprate
|
|
// -ie we want to keep 50% of the amount of time difference between last two snaps.
|
|
// -we need to scale this because we may get throttled at the snaprate may change
|
|
// -Acts as a buffer to absorb spikes
|
|
idCVar net_minBufferedSnapPCT_Static( "net_minBufferedSnapPCT_Static", "1.0", CVAR_FLOAT, "Min amount of snapshot buffer time we want need to buffer" );
|
|
idCVar net_maxBufferedSnapMS( "net_maxBufferedSnapMS", "336", CVAR_INTEGER, "Max time to allow for interpolation cushion" );
|
|
idCVar net_minBufferedSnapWinPCT_Static( "net_minBufferedSnapWinPCT_Static", "1.0", CVAR_FLOAT, "Min amount of snapshot buffer time we want need to buffer" );
|
|
|
|
// Factor at which we catch speed up interpolation if we fall behind our optimal interpolation window
|
|
// -This is a static factor. We may experiment with a dynamic one that would be faster the farther you are from the ideal window
|
|
idCVar net_interpolationCatchupRate( "net_interpolationCatchupRate", "1.3", CVAR_FLOAT, "Scale interpolationg rate when we fall behind" );
|
|
idCVar net_interpolationFallbackRate( "net_interpolationFallbackRate", "0.95", CVAR_FLOAT, "Scale interpolationg rate when we fall behind" );
|
|
idCVar net_interpolationBaseRate( "net_interpolationBaseRate", "1.0", CVAR_FLOAT, "Scale interpolationg rate when we fall behind" );
|
|
|
|
// Enabled a dynamic ideal snap buffer window: we will scale the distance and size
|
|
idCVar net_optimalDynamic( "net_optimalDynamic", "1", CVAR_BOOL, "How fast to add to our optimal time buffer when we are playing snapshots faster than server is feeding them to us" );
|
|
|
|
// These values are used instead if net_optimalDynamic is 0 (don't scale by actual snap rate/interval)
|
|
idCVar net_optimalSnapWindow( "net_optimalSnapWindow", "112", CVAR_FLOAT, "" );
|
|
idCVar net_optimalSnapTime( "net_optimalSnapTime", "112", CVAR_FLOAT, "How fast to add to our optimal time buffer when we are playing snapshots faster than server is feeding them to us" );
|
|
|
|
// this is at what percentage of being ahead of the interpolation buffer that we start slowing down (we ramp down from 1.0 to 0.0 starting here)
|
|
// this is a percentage of the total cushion time.
|
|
idCVar net_interpolationSlowdownStart( "net_interpolationSlowdownStart", "0.5", CVAR_FLOAT, "Scale interpolation rate when we fall behind" );
|
|
|
|
|
|
// Extrapolation is now disabled
|
|
idCVar net_maxExtrapolationInMS( "net_maxExtrapolationInMS", "0", CVAR_INTEGER, "Max time in MS that extrapolation is allowed to occur." );
|
|
|
|
static const int SNAP_USERCMDS = 8192;
|
|
|
|
/*
|
|
===============
|
|
idCommonLocal::IsMultiplayer
|
|
===============
|
|
*/
|
|
bool idCommonLocal::IsMultiplayer()
|
|
{
|
|
idLobbyBase& lobby = session->GetPartyLobbyBase();
|
|
return ( ( ( lobby.GetMatchParms().matchFlags & MATCH_ONLINE ) != 0 ) && ( session->GetState() > idSession::IDLE ) );
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idCommonLocal::IsServer
|
|
===============
|
|
*/
|
|
bool idCommonLocal::IsServer()
|
|
{
|
|
return IsMultiplayer() && session->GetActingGameStateLobbyBase().IsHost();
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idCommonLocal::IsClient
|
|
===============
|
|
*/
|
|
bool idCommonLocal::IsClient()
|
|
{
|
|
return IsMultiplayer() && session->GetActingGameStateLobbyBase().IsPeer();
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idCommonLocal::SendSnapshots
|
|
===============
|
|
*/
|
|
int idCommonLocal::GetSnapRate()
|
|
{
|
|
return net_snapRate.GetInteger();
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idCommonLocal::SendSnapshots
|
|
===============
|
|
*/
|
|
void idCommonLocal::SendSnapshots()
|
|
{
|
|
if( !mapSpawned )
|
|
{
|
|
return;
|
|
}
|
|
int currentTime = Sys_Milliseconds();
|
|
if( currentTime < nextSnapshotSendTime )
|
|
{
|
|
return;
|
|
}
|
|
idLobbyBase& lobby = session->GetActingGameStateLobbyBase();
|
|
if( !lobby.IsHost() )
|
|
{
|
|
return;
|
|
}
|
|
if( !lobby.HasActivePeers() )
|
|
{
|
|
return;
|
|
}
|
|
idSnapShot ss;
|
|
game->ServerWriteSnapshot( ss );
|
|
|
|
session->SendSnapshot( ss );
|
|
nextSnapshotSendTime = MSEC_ALIGN_TO_FRAME( currentTime + net_snapRate.GetInteger() );
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idCommonLocal::NetReceiveSnapshot
|
|
===============
|
|
*/
|
|
void idCommonLocal::NetReceiveSnapshot( class idSnapShot& ss )
|
|
{
|
|
ss.SetRecvTime( Sys_Milliseconds() );
|
|
// If we are about to overwrite the oldest snap, then force a read, which will cause a pop on screen, but we have to do this.
|
|
if( writeSnapshotIndex - readSnapshotIndex >= RECEIVE_SNAPSHOT_BUFFER_SIZE )
|
|
{
|
|
idLib::Printf( "Overwritting oldest snapshot %d with new snapshot %d\n", readSnapshotIndex, writeSnapshotIndex );
|
|
assert( writeSnapshotIndex % RECEIVE_SNAPSHOT_BUFFER_SIZE == readSnapshotIndex % RECEIVE_SNAPSHOT_BUFFER_SIZE );
|
|
ProcessNextSnapshot();
|
|
}
|
|
|
|
receivedSnaps[ writeSnapshotIndex % RECEIVE_SNAPSHOT_BUFFER_SIZE ] = ss;
|
|
writeSnapshotIndex++;
|
|
|
|
// Force read the very first 2 snapshots
|
|
if( readSnapshotIndex < 2 )
|
|
{
|
|
ProcessNextSnapshot();
|
|
}
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idCommonLocal::SendUsercmd
|
|
===============
|
|
*/
|
|
void idCommonLocal::SendUsercmds( int localClientNum )
|
|
{
|
|
if( !mapSpawned )
|
|
{
|
|
return;
|
|
}
|
|
int currentTime = Sys_Milliseconds();
|
|
if( currentTime < nextUsercmdSendTime )
|
|
{
|
|
return;
|
|
}
|
|
idLobbyBase& lobby = session->GetActingGameStateLobbyBase();
|
|
if( lobby.IsHost() )
|
|
{
|
|
return;
|
|
}
|
|
// We always send the last NUM_USERCMD_SEND usercmds
|
|
// Which may result in duplicate usercmds being sent in the case of a low net_ucmdRate
|
|
// But the LZW compressor means the extra usercmds are not large and the redundancy can smooth packet loss
|
|
byte buffer[idPacketProcessor::MAX_FINAL_PACKET_SIZE];
|
|
idBitMsg msg( buffer, sizeof( buffer ) );
|
|
idSerializer ser( msg, true );
|
|
usercmd_t empty;
|
|
usercmd_t* last = ∅
|
|
|
|
usercmd_t* cmdBuffer[NUM_USERCMD_SEND];
|
|
const int numCmds = userCmdMgr.GetPlayerCmds( localClientNum, cmdBuffer, NUM_USERCMD_SEND );
|
|
msg.WriteByte( numCmds );
|
|
for( int i = 0; i < numCmds; i++ )
|
|
{
|
|
cmdBuffer[i]->Serialize( ser, *last );
|
|
|
|
last = cmdBuffer[i];
|
|
}
|
|
session->SendUsercmds( msg );
|
|
|
|
nextUsercmdSendTime = MSEC_ALIGN_TO_FRAME( currentTime + net_ucmdRate.GetInteger() );
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idCommonLocal::NetReceiveUsercmds
|
|
===============
|
|
*/
|
|
void idCommonLocal::NetReceiveUsercmds( int peer, idBitMsg& msg )
|
|
{
|
|
int clientNum = Game()->MapPeerToClient( peer );
|
|
if( clientNum == -1 )
|
|
{
|
|
idLib::Warning( "NetReceiveUsercmds: Could not find client for peer %d", peer );
|
|
return;
|
|
}
|
|
|
|
NetReadUsercmds( clientNum, msg );
|
|
}
|
|
|
|
/*
|
|
===============
|
|
idCommonLocal::NetReceiveReliable
|
|
===============
|
|
*/
|
|
void idCommonLocal::NetReceiveReliable( int peer, int type, idBitMsg& msg )
|
|
{
|
|
int clientNum = Game()->MapPeerToClient( peer );
|
|
// Only servers care about the client num. Band-aid for problems related to the host's peerIndex being -1 on clients.
|
|
if( common->IsServer() && clientNum == -1 )
|
|
{
|
|
idLib::Warning( "NetReceiveReliable: Could not find client for peer %d", peer );
|
|
return;
|
|
}
|
|
|
|
const byte* msgData = msg.GetReadData() + msg.GetReadCount();
|
|
int msgSize = msg.GetRemainingData();
|
|
reliableMsg_t& reliable = reliableQueue.Alloc();
|
|
reliable.client = clientNum;
|
|
reliable.type = type;
|
|
reliable.dataSize = msgSize;
|
|
reliable.data = ( byte* )Mem_Alloc( msgSize, TAG_NETWORKING );
|
|
memcpy( reliable.data, msgData, msgSize );
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idCommonLocal::ProcessSnapshot
|
|
========================
|
|
*/
|
|
void idCommonLocal::ProcessSnapshot( idSnapShot& ss )
|
|
{
|
|
int time = Sys_Milliseconds();
|
|
|
|
snapTime = time;
|
|
snapPrevious = snapCurrent;
|
|
snapCurrent.serverTime = ss.GetTime();
|
|
snapRate = snapCurrent.serverTime - snapPrevious.serverTime;
|
|
|
|
|
|
static int lastReceivedLocalTime = 0;
|
|
int timeSinceLastSnap = ( time - lastReceivedLocalTime );
|
|
if( net_debug_snapShotTime.GetBool() )
|
|
{
|
|
idLib::Printf( "^2ProcessSnapshot. delta serverTime: %d delta localTime: %d \n", ( snapCurrent.serverTime - snapPrevious.serverTime ), timeSinceLastSnap );
|
|
}
|
|
lastReceivedLocalTime = time;
|
|
|
|
/* JAF ?
|
|
for ( int i = 0; i < MAX_PLAYERS; i++ ) {
|
|
idBitMsg msg;
|
|
if ( ss.GetObjectMsgByID( idSession::SS_PLAYER + i, msg ) ) {
|
|
if ( msg.GetSize() == 0 ) {
|
|
snapCurrent.players[ i ].valid = false;
|
|
continue;
|
|
}
|
|
|
|
idSerializer ser( msg, false );
|
|
SerializePlayer( ser, snapCurrent.players[ i ] );
|
|
snapCurrent.players[ i ].valid = true;
|
|
|
|
extern idCVar com_drawSnapshots;
|
|
if ( com_drawSnapshots.GetInteger() == 3 ) {
|
|
console->AddSnapObject( "players", msg.GetSize(), ss.CompareObject( &oldss, idSession::SS_PLAYER + i ) );
|
|
}
|
|
}
|
|
}
|
|
*/
|
|
|
|
// Read usercmds from other players
|
|
for( int p = 0; p < MAX_PLAYERS; p++ )
|
|
{
|
|
if( p == game->GetLocalClientNum() )
|
|
{
|
|
continue;
|
|
}
|
|
idBitMsg msg;
|
|
if( ss.GetObjectMsgByID( SNAP_USERCMDS + p, msg ) )
|
|
{
|
|
NetReadUsercmds( p, msg );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set server game time here so that it accurately reflects the time when this frame was saved out, in case any serialize function needs it.
|
|
int oldTime = Game()->GetServerGameTimeMs();
|
|
Game()->SetServerGameTimeMs( snapCurrent.serverTime );
|
|
|
|
Game()->ClientReadSnapshot( ss ); //, &oldss );
|
|
|
|
// Restore server game time
|
|
Game()->SetServerGameTimeMs( oldTime );
|
|
|
|
snapTimeDelta = ss.GetRecvTime() - oldss.GetRecvTime();
|
|
oldss = ss;
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idCommonLocal::NetReadUsercmds
|
|
========================
|
|
*/
|
|
void idCommonLocal::NetReadUsercmds( int clientNum, idBitMsg& msg )
|
|
{
|
|
if( clientNum == -1 )
|
|
{
|
|
idLib::Warning( "NetReadUsercmds: Trying to read commands from invalid clientNum %d", clientNum );
|
|
return;
|
|
}
|
|
|
|
// TODO: This shouldn't actually happen. Figure out why it does.
|
|
// Seen on clients when another client leaves a match.
|
|
if( msg.GetReadData() == NULL )
|
|
{
|
|
return;
|
|
}
|
|
|
|
idSerializer ser( msg, false );
|
|
|
|
usercmd_t fakeCmd;
|
|
usercmd_t* base = &fakeCmd;
|
|
|
|
usercmd_t lastCmd;
|
|
|
|
bool gotNewCmd = false;
|
|
idStaticList< usercmd_t, NUM_USERCMD_RELAY > newCmdBuffer;
|
|
|
|
usercmd_t baseCmd = userCmdMgr.NewestUserCmdForPlayer( clientNum );
|
|
int curMilliseconds = baseCmd.clientGameMilliseconds;
|
|
|
|
const int numCmds = msg.ReadByte();
|
|
|
|
for( int i = 0; i < numCmds; i++ )
|
|
{
|
|
usercmd_t newCmd;
|
|
newCmd.Serialize( ser, *base );
|
|
|
|
lastCmd = newCmd;
|
|
base = &lastCmd;
|
|
|
|
int newMilliseconds = newCmd.clientGameMilliseconds;
|
|
|
|
if( newMilliseconds > curMilliseconds )
|
|
{
|
|
if( verify( i < NUM_USERCMD_RELAY ) )
|
|
{
|
|
newCmdBuffer.Append( newCmd );
|
|
gotNewCmd = true;
|
|
curMilliseconds = newMilliseconds;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Push the commands into the buffer.
|
|
for( int i = 0; i < newCmdBuffer.Num(); ++i )
|
|
{
|
|
userCmdMgr.PutUserCmdForPlayer( clientNum, newCmdBuffer[i] );
|
|
}
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idCommonLocal::ProcessNextSnapshot
|
|
========================
|
|
*/
|
|
void idCommonLocal::ProcessNextSnapshot()
|
|
{
|
|
if( readSnapshotIndex == writeSnapshotIndex )
|
|
{
|
|
idLib::Printf( "No snapshots to process.\n" );
|
|
return; // No snaps to process
|
|
}
|
|
ProcessSnapshot( receivedSnaps[ readSnapshotIndex % RECEIVE_SNAPSHOT_BUFFER_SIZE ] );
|
|
readSnapshotIndex++;
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idCommonLocal::CalcSnapTimeBuffered
|
|
Return the amount of game time left of buffered snapshots
|
|
totalBufferedTime - total amount of snapshot time (includng what we've already past in current interpolate)
|
|
totalRecvTime - total real time (sys_milliseconds) all of totalBufferedTime was received over
|
|
========================
|
|
*/
|
|
int idCommonLocal::CalcSnapTimeBuffered( int& totalBufferedTime, int& totalRecvTime )
|
|
{
|
|
|
|
totalBufferedTime = snapRate;
|
|
totalRecvTime = snapTimeDelta;
|
|
|
|
// oldSS = last ss we deserialized
|
|
int lastBuffTime = oldss.GetTime();
|
|
int lastRecvTime = oldss.GetRecvTime();
|
|
|
|
// receivedSnaps[readSnapshotIndex % RECEIVE_SNAPSHOT_BUFFER_SIZE] = next buffered snapshot we haven't processed yet (might not exist)
|
|
for( int i = readSnapshotIndex; i < writeSnapshotIndex; i++ )
|
|
{
|
|
int buffTime = receivedSnaps[i % RECEIVE_SNAPSHOT_BUFFER_SIZE].GetTime();
|
|
int recvTime = receivedSnaps[i % RECEIVE_SNAPSHOT_BUFFER_SIZE].GetRecvTime();
|
|
|
|
totalBufferedTime += buffTime - lastBuffTime;
|
|
totalRecvTime += recvTime - lastRecvTime;
|
|
|
|
lastRecvTime = recvTime;
|
|
lastBuffTime = buffTime;
|
|
}
|
|
|
|
totalRecvTime = Max( 1, totalRecvTime );
|
|
totalRecvTime = static_cast<float>( initialBaseTicksPerSec ) * static_cast<float>( totalRecvTime / 1000.0f ); // convert realMS to gameMS
|
|
|
|
// remove time we've already interpolated over
|
|
int timeLeft = totalBufferedTime - Min< int >( snapRate, snapCurrentTime );
|
|
|
|
//idLib::Printf( "CalcSnapTimeBuffered. timeLeft: %d totalRecvTime: %d, totalTimeBuffered: %d\n", timeLeft, totalRecvTime, totalBufferedTime );
|
|
return timeLeft;
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idCommonLocal::InterpolateSnapshot
|
|
========================
|
|
*/
|
|
void idCommonLocal::InterpolateSnapshot( netTimes_t& prev, netTimes_t& next, float fraction, bool predict )
|
|
{
|
|
|
|
int serverTime = Lerp( prev.serverTime, next.serverTime, fraction );
|
|
|
|
Game()->SetServerGameTimeMs( serverTime ); // Set the global server time to the interpolated time of the server
|
|
Game()->SetInterpolation( fraction, serverTime, prev.serverTime, next.serverTime );
|
|
|
|
//Game()->RunFrame( &userCmdMgr, &ret, true );
|
|
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idCommonLocal::RunNetworkSnapshotFrame
|
|
========================
|
|
*/
|
|
void idCommonLocal::RunNetworkSnapshotFrame()
|
|
{
|
|
|
|
// Process any reliable messages we've received
|
|
for( int i = 0; i < reliableQueue.Num(); i++ )
|
|
{
|
|
game->ProcessReliableMessage( reliableQueue[i].client, reliableQueue[i].type, idBitMsg( ( const byte* )reliableQueue[i].data, reliableQueue[i].dataSize ) );
|
|
Mem_Free( reliableQueue[i].data );
|
|
}
|
|
reliableQueue.Clear();
|
|
|
|
// abuse the game timing to time presentable thinking on clients
|
|
time_gameFrame = Sys_Microseconds();
|
|
time_maxGameFrame = 0;
|
|
count_numGameFrames = 0;
|
|
|
|
if( snapPrevious.serverTime >= 0 )
|
|
{
|
|
|
|
int msec_interval = 1 + idMath::Ftoi( ( float )initialBaseTicksPerSec );
|
|
|
|
static int clientTimeResidual = 0;
|
|
static int lastTime = Sys_Milliseconds();
|
|
int currentTime = Sys_Milliseconds();
|
|
int deltaFrameTime = idMath::ClampInt( 1, 33, currentTime - lastTime );
|
|
|
|
clientTimeResidual += idMath::ClampInt( 0, 50, currentTime - lastTime );
|
|
lastTime = currentTime;
|
|
|
|
extern idCVar com_fixedTic;
|
|
if( com_fixedTic.GetBool() )
|
|
{
|
|
clientTimeResidual = 0;
|
|
}
|
|
|
|
do
|
|
{
|
|
// If we are extrapolating and have fresher snapshots, then use the freshest one
|
|
while( ( snapCurrentTime >= snapRate || com_forceLatestSnap.GetBool() ) && readSnapshotIndex < writeSnapshotIndex )
|
|
{
|
|
snapCurrentTime -= snapRate;
|
|
ProcessNextSnapshot();
|
|
}
|
|
|
|
// this only matters when running < 60 fps
|
|
// JAF Game()->GetRenderWorld()->UpdateDeferredPositions();
|
|
|
|
// Clamp the current time so that it doesn't fall outside of our extrapolation bounds
|
|
snapCurrentTime = idMath::ClampInt( 0, snapRate + Min( ( int )snapRate, ( int )net_maxExtrapolationInMS.GetInteger() ), snapCurrentTime );
|
|
|
|
if( snapRate <= 0 )
|
|
{
|
|
idLib::Warning( "snapRate <= 0. Resetting to 100" );
|
|
snapRate = 100;
|
|
}
|
|
|
|
float fraction = ( float )snapCurrentTime / ( float )snapRate;
|
|
if( !IsValid( fraction ) )
|
|
{
|
|
idLib::Warning( "Interpolation Fraction invalid: snapCurrentTime %d / snapRate %d", ( int )snapCurrentTime, ( int )snapRate );
|
|
fraction = 0.0f;
|
|
}
|
|
|
|
InterpolateSnapshot( snapPrevious, snapCurrent, fraction, true );
|
|
|
|
// Default to a snap scale of 1
|
|
float snapRateScale = net_interpolationBaseRate.GetFloat();
|
|
|
|
snapTimeBuffered = CalcSnapTimeBuffered( totalBufferedTime, totalRecvTime );
|
|
effectiveSnapRate = static_cast< float >( totalBufferedTime ) / static_cast< float >( totalRecvTime );
|
|
|
|
if( net_minBufferedSnapPCT_Static.GetFloat() > 0.0f )
|
|
{
|
|
optimalPCTBuffer = session->GetTitleStorageFloat( "net_minBufferedSnapPCT_Static", net_minBufferedSnapPCT_Static.GetFloat() );
|
|
}
|
|
|
|
// Calculate optimal amount of buffered time we want
|
|
if( net_optimalDynamic.GetBool() )
|
|
{
|
|
optimalTimeBuffered = idMath::ClampInt( 0, net_maxBufferedSnapMS.GetInteger(), snapRate * optimalPCTBuffer );
|
|
optimalTimeBufferedWindow = snapRate * net_minBufferedSnapWinPCT_Static.GetFloat();
|
|
}
|
|
else
|
|
{
|
|
optimalTimeBuffered = net_optimalSnapTime.GetFloat();
|
|
optimalTimeBufferedWindow = net_optimalSnapWindow.GetFloat();
|
|
}
|
|
|
|
// Scale snapRate based on where we are in the buffer
|
|
if( snapTimeBuffered <= optimalTimeBuffered )
|
|
{
|
|
if( snapTimeBuffered <= idMath::FLT_SMALLEST_NON_DENORMAL )
|
|
{
|
|
snapRateScale = 0;
|
|
}
|
|
else
|
|
{
|
|
snapRateScale = net_interpolationFallbackRate.GetFloat();
|
|
// When we interpolate past our cushion of buffered snapshot, we want to slow smoothly slow the
|
|
// rate of interpolation. frac will go from 1.0 to 0.0 (if snapshots stop coming in).
|
|
float startSlowdown = ( net_interpolationSlowdownStart.GetFloat() * optimalTimeBuffered );
|
|
if( startSlowdown > 0 && snapTimeBuffered < startSlowdown )
|
|
{
|
|
float frac = idMath::ClampFloat( 0.0f, 1.0f, snapTimeBuffered / startSlowdown );
|
|
if( !IsValid( frac ) )
|
|
{
|
|
frac = 0.0f;
|
|
}
|
|
snapRateScale = Square( frac ) * snapRateScale;
|
|
if( !IsValid( snapRateScale ) )
|
|
{
|
|
snapRateScale = 0.0f;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
}
|
|
else if( snapTimeBuffered > optimalTimeBuffered + optimalTimeBufferedWindow )
|
|
{
|
|
// Go faster
|
|
snapRateScale = net_interpolationCatchupRate.GetFloat();
|
|
|
|
}
|
|
|
|
float delta_interpolate = ( float )initialBaseTicksPerSec * snapRateScale;
|
|
if( net_effectiveSnapRateEnable.GetBool() )
|
|
{
|
|
|
|
float deltaFrameGameMS = static_cast<float>( initialBaseTicksPerSec ) * static_cast<float>( deltaFrameTime / 1000.0f );
|
|
delta_interpolate = ( deltaFrameGameMS * snapRateScale * effectiveSnapRate ) + snapCurrentResidual;
|
|
if( !IsValid( delta_interpolate ) )
|
|
{
|
|
delta_interpolate = 0.0f;
|
|
}
|
|
|
|
snapCurrentResidual = idMath::Frac( delta_interpolate ); // fixme: snapCurrentTime should just be a float, but would require changes in d4 too
|
|
if( !IsValid( snapCurrentResidual ) )
|
|
{
|
|
snapCurrentResidual = 0.0f;
|
|
}
|
|
|
|
if( net_effectiveSnapRateDebug.GetBool() )
|
|
{
|
|
idLib::Printf( "%d/%.2f snapRateScale: %.2f effectiveSR: %.2f d.interp: %.2f snapTimeBuffered: %.2f res: %.2f\n", deltaFrameTime, deltaFrameGameMS, snapRateScale, effectiveSnapRate, delta_interpolate, snapTimeBuffered, snapCurrentResidual );
|
|
}
|
|
}
|
|
|
|
assert( IsValid( delta_interpolate ) );
|
|
int interpolate_interval = idMath::Ftoi( delta_interpolate );
|
|
|
|
snapCurrentTime += interpolate_interval; // advance interpolation time by the scaled interpolate_interval
|
|
clientTimeResidual -= msec_interval; // advance local client residual time (fixed step)
|
|
|
|
}
|
|
while( clientTimeResidual >= msec_interval );
|
|
|
|
if( clientTimeResidual < 0 )
|
|
{
|
|
clientTimeResidual = 0;
|
|
}
|
|
}
|
|
|
|
time_gameFrame = Sys_Microseconds() - time_gameFrame;
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idCommonLocal::ExecuteReliableMessages
|
|
========================
|
|
*/
|
|
void idCommonLocal::ExecuteReliableMessages()
|
|
{
|
|
|
|
// Process any reliable messages we've received
|
|
for( int i = 0; i < reliableQueue.Num(); i++ )
|
|
{
|
|
reliableMsg_t& reliable = reliableQueue[i];
|
|
game->ProcessReliableMessage( reliable.client, reliable.type, idBitMsg( ( const byte* )reliable.data, reliable.dataSize ) );
|
|
Mem_Free( reliable.data );
|
|
}
|
|
reliableQueue.Clear();
|
|
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idCommonLocal::ResetNetworkingState
|
|
========================
|
|
*/
|
|
void idCommonLocal::ResetNetworkingState()
|
|
{
|
|
snapTime = 0;
|
|
snapTimeWrite = 0;
|
|
snapCurrentTime = 0;
|
|
snapCurrentResidual = 0.0f;
|
|
|
|
snapTimeBuffered = 0.0f;
|
|
effectiveSnapRate = 0.0f;
|
|
totalBufferedTime = 0;
|
|
totalRecvTime = 0;
|
|
|
|
readSnapshotIndex = 0;
|
|
writeSnapshotIndex = 0;
|
|
snapRate = 100000;
|
|
optimalTimeBuffered = 0.0f;
|
|
optimalPCTBuffer = 0.5f;
|
|
optimalTimeBufferedWindow = 0.0;
|
|
|
|
// Clear snapshot queue
|
|
for( int i = 0; i < RECEIVE_SNAPSHOT_BUFFER_SIZE; i++ )
|
|
{
|
|
receivedSnaps[i].Clear();
|
|
}
|
|
|
|
userCmdMgr.SetDefaults();
|
|
|
|
snapCurrent.localTime = -1;
|
|
snapPrevious.localTime = -1;
|
|
snapCurrent.serverTime = -1;
|
|
snapPrevious.serverTime = -1;
|
|
|
|
// Make sure our current snap state is cleared so state from last game doesn't carry over into new game
|
|
oldss.Clear();
|
|
|
|
gameFrame = 0;
|
|
clientPrediction = 0;
|
|
nextUsercmdSendTime = 0;
|
|
nextSnapshotSendTime = 0;
|
|
}
|