mirror of
https://github.com/id-Software/DOOM-3-BFG.git
synced 2024-11-24 21:12:03 +00:00
546 lines
No EOL
18 KiB
C++
546 lines
No EOL
18 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.
|
|
|
|
===========================================================================
|
|
*/
|
|
#pragma hdrstop
|
|
#include "../idlib/precompiled.h"
|
|
#include "sys_lobby.h"
|
|
|
|
idCVar net_migration_debug( "net_migration_debug", "0", CVAR_BOOL, "debug" );
|
|
idCVar net_migration_disable( "net_migration_disable", "0", CVAR_BOOL, "debug" );
|
|
idCVar net_migration_forcePeerAsHost( "net_migration_forcePeerAsHost", "-1", CVAR_INTEGER, "When set to >-1, it forces that peer number to be the new host during migration" );
|
|
|
|
|
|
/*
|
|
========================
|
|
idLobby::IsBetterHost
|
|
========================
|
|
*/
|
|
bool idLobby::IsBetterHost( int ping1, lobbyUserID_t userId1, int ping2, lobbyUserID_t userId2 ) {
|
|
if ( lobbyType == TYPE_PARTY ) {
|
|
return userId1 < userId2; // Only use user id for party, since ping doesn't matter
|
|
}
|
|
|
|
if ( ping1 < ping2 ) {
|
|
// Better ping wins
|
|
return true;
|
|
} else if ( ping1 == ping2 && userId1 < userId2 ) {
|
|
// User id is tie breaker
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::FindMigrationInviteIndex
|
|
========================
|
|
*/
|
|
int idLobby::FindMigrationInviteIndex( lobbyAddress_t & address ) {
|
|
if ( migrationInfo.state == MIGRATE_NONE ) {
|
|
return -1;
|
|
}
|
|
|
|
for ( int i = 0; i < migrationInfo.invites.Num(); i++ ) {
|
|
if ( migrationInfo.invites[i].address.Compare( address, true ) ) {
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::UpdateHostMigration
|
|
========================
|
|
*/
|
|
void idLobby::UpdateHostMigration() {
|
|
|
|
int time = Sys_Milliseconds();
|
|
|
|
// If we are picking a new host, then update that
|
|
if ( migrationInfo.state == MIGRATE_PICKING_HOST ) {
|
|
const int MIGRATION_PICKING_HOST_TIMEOUT_IN_SECONDS = 20; // FIXME: set back to 5 // Give other hosts 5 seconds
|
|
|
|
if ( time - migrationInfo.migrationStartTime > session->GetTitleStorageInt( "MIGRATION_PICKING_HOST_TIMEOUT_IN_SECONDS", MIGRATION_PICKING_HOST_TIMEOUT_IN_SECONDS ) * 1000 ) {
|
|
// Just become the host if we haven't heard from a host in awhile
|
|
BecomeHost();
|
|
} else {
|
|
return;
|
|
}
|
|
}
|
|
|
|
// See if we are a new migrated host that needs to invite the original members back
|
|
if ( migrationInfo.state != MIGRATE_BECOMING_HOST ) {
|
|
return;
|
|
}
|
|
|
|
if ( lobbyBackend == NULL || lobbyBackend->GetState() != idLobbyBackend::STATE_READY ) {
|
|
return;
|
|
}
|
|
|
|
if ( state != STATE_IDLE ) {
|
|
return;
|
|
}
|
|
|
|
if ( !IsHost() ) {
|
|
return;
|
|
}
|
|
|
|
const int MIGRATION_TIMEOUT_IN_SECONDS = 30; // FIXME: setting to 30 for dev purposes. 10 seems more reasonable. Need to make unloading game / loading lobby async
|
|
const int MIGRATION_INVITE_TIME_IN_SECONDS = 2;
|
|
|
|
if ( migrationInfo.invites.Num() == 0 || time - migrationInfo.migrationStartTime > session->GetTitleStorageInt( "MIGRATION_TIMEOUT_IN_SECONDS", MIGRATION_TIMEOUT_IN_SECONDS ) * 1000 ) {
|
|
// Either everyone acked, or we timed out, just keep who we have, and stop sending invites
|
|
EndMigration();
|
|
return;
|
|
}
|
|
|
|
// Send invites to anyone who hasn't responded
|
|
for ( int i = 0; i < migrationInfo.invites.Num(); i++ ) {
|
|
if ( time - migrationInfo.invites[i].lastInviteTime < session->GetTitleStorageInt( "MIGRATION_INVITE_TIME_IN_SECONDS", MIGRATION_INVITE_TIME_IN_SECONDS ) * 1000 ) {
|
|
continue; // Not enough time passed
|
|
}
|
|
|
|
// Mark the time
|
|
migrationInfo.invites[i].lastInviteTime = time;
|
|
|
|
byte buffer[ idPacketProcessor::MAX_PACKET_SIZE - 2 ];
|
|
idBitMsg outmsg( buffer, sizeof( buffer ) );
|
|
|
|
// Have lobbyBackend fill out msg with connection info
|
|
lobbyConnectInfo_t connectInfo = lobbyBackend->GetConnectInfo();
|
|
connectInfo.WriteToMsg( outmsg );
|
|
|
|
// Let them know whether or not this was from in game
|
|
outmsg.WriteBool( migrationInfo.persistUntilGameEndsData.wasMigratedGame );
|
|
|
|
NET_VERBOSE_PRINT( "NET: Sending migration invite to %s\n", migrationInfo.invites[i].address.ToString() );
|
|
|
|
// Send the migration invite
|
|
SendConnectionLess( migrationInfo.invites[i].address, OOB_MIGRATE_INVITE, outmsg.GetReadData(), outmsg.GetSize() );
|
|
}
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::BuildMigrationInviteList
|
|
========================
|
|
*/
|
|
void idLobby::BuildMigrationInviteList( bool inviteOldHost ) {
|
|
migrationInfo.invites.Clear();
|
|
|
|
// Build a list of addresses we will send invites to (gather all unique remote addresses from the session user list)
|
|
for ( int i = 0; i < GetNumLobbyUsers(); i++ ) {
|
|
lobbyUser_t * user = GetLobbyUser( i );
|
|
|
|
if ( !verify( user != NULL ) ) {
|
|
continue;
|
|
}
|
|
|
|
if ( user->IsDisconnected() ) {
|
|
continue;
|
|
}
|
|
|
|
if ( IsSessionUserIndexLocal( i ) ) {
|
|
migrationInfo.ourPingMs = user->pingMs;
|
|
migrationInfo.ourUserId = user->lobbyUserID;
|
|
migrationInfo.persistUntilGameEndsData.ourGameData = user->migrationGameData;
|
|
NET_VERBOSE_PRINT( "^2NET: Migration game data for local user is index %d \n", user->migrationGameData );
|
|
|
|
continue; // Only interested in remote users
|
|
}
|
|
|
|
if ( !inviteOldHost && user->peerIndex == -1 ) {
|
|
continue; // Don't invite old host if told not to do so
|
|
}
|
|
|
|
if ( FindMigrationInviteIndex( user->address ) == -1 ) {
|
|
migrationInvite_t invite;
|
|
invite.address = user->address;
|
|
invite.pingMs = user->pingMs;
|
|
invite.userId = user->lobbyUserID;
|
|
invite.migrationGameData = user->migrationGameData;
|
|
invite.lastInviteTime = 0;
|
|
|
|
NET_VERBOSE_PRINT( "^2NET: Migration game data for user %s is index %d \n", user->gamertag, user->migrationGameData );
|
|
|
|
migrationInfo.invites.Append( invite );
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::PickNewHost
|
|
========================
|
|
*/
|
|
void idLobby::PickNewHost( bool forceMe, bool inviteOldHost ) {
|
|
if ( IsHost() ) {
|
|
idLib::Printf( "PickNewHost: Already host of session %s\n", GetLobbyName() );
|
|
return;
|
|
}
|
|
|
|
sessionCB->PrePickNewHost( *this, forceMe, inviteOldHost );
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::PickNewHostInternal
|
|
========================
|
|
*/
|
|
void idLobby::PickNewHostInternal( bool forceMe, bool inviteOldHost ) {
|
|
|
|
if ( migrationInfo.state == MIGRATE_PICKING_HOST ) {
|
|
return; // Already picking new host
|
|
}
|
|
|
|
idLib::Printf( "PickNewHost: Started picking new host %s.\n", GetLobbyName() );
|
|
|
|
if ( IsHost() ) {
|
|
idLib::Printf( "PickNewHost: Already host of session %s\n", GetLobbyName() );
|
|
return;
|
|
}
|
|
|
|
// Find the user with the lowest ping
|
|
int bestUserIndex = -1;
|
|
int bestPingMs = 0;
|
|
lobbyUserID_t bestUserId;
|
|
|
|
for ( int i = 0; i < GetNumLobbyUsers(); i++ ) {
|
|
lobbyUser_t * user = GetLobbyUser( i );
|
|
|
|
if ( !verify( user != NULL ) ) {
|
|
continue;
|
|
}
|
|
|
|
if ( user->IsDisconnected() ) {
|
|
continue;
|
|
}
|
|
|
|
if ( user->peerIndex == -1 ) {
|
|
continue; // Don't try and pick old host
|
|
}
|
|
|
|
if ( bestUserIndex == -1 || IsBetterHost( user->pingMs, user->lobbyUserID, bestPingMs, bestUserId ) ) {
|
|
bestUserIndex = i;
|
|
bestPingMs = user->pingMs;
|
|
bestUserId = user->lobbyUserID;
|
|
}
|
|
|
|
if ( user->peerIndex == net_migration_forcePeerAsHost.GetInteger() ) {
|
|
bestUserIndex = i;
|
|
bestPingMs = user->pingMs;
|
|
bestUserId = user->lobbyUserID;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Remember when we first started picking a new host
|
|
migrationInfo.state = MIGRATE_PICKING_HOST;
|
|
migrationInfo.migrationStartTime = Sys_Milliseconds();
|
|
|
|
migrationInfo.persistUntilGameEndsData.wasMigratedGame = sessionCB->GetState() == idSession::INGAME;
|
|
|
|
if ( bestUserIndex == -1 ) { // This can happen if we call PickNewHost on an lobby that was Shutdown
|
|
NET_VERBOSE_PRINT( "MIGRATION: PickNewHost was called on an lobby that was Shutdown\n" );
|
|
BecomeHost();
|
|
return;
|
|
}
|
|
|
|
NET_VERBOSE_PRINT( "MIGRATION: Chose user index %d (%s) for new host\n", bestUserIndex, GetLobbyUser( bestUserIndex )->gamertag );
|
|
|
|
bool bestWasLocal = IsSessionUserIndexLocal( bestUserIndex ); // Check before shutting down the lobby
|
|
migrateMsgFlags = parms.matchFlags; // Save off match parms
|
|
|
|
// Build invite list
|
|
BuildMigrationInviteList( inviteOldHost );
|
|
|
|
// If the best user is on this machine, then we become the host now, otherwise, wait for a new host to contact us
|
|
if ( forceMe || bestWasLocal ) {
|
|
BecomeHost();
|
|
}
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::BecomeHost
|
|
========================
|
|
*/
|
|
void idLobby::BecomeHost() {
|
|
|
|
if ( !verify( migrationInfo.state == MIGRATE_PICKING_HOST ) ) {
|
|
idLib::Printf( "BecomeHost: Must be called from PickNewHost.\n" );
|
|
EndMigration();
|
|
return;
|
|
}
|
|
|
|
if ( IsHost() ) {
|
|
idLib::Printf( "BecomeHost: Already host of session.\n" );
|
|
EndMigration();
|
|
return;
|
|
}
|
|
|
|
if ( !sessionCB->BecomingHost( *this ) ) {
|
|
EndMigration();
|
|
return;
|
|
}
|
|
|
|
idLib::Printf( "BecomeHost: Sending %i invites on %s.\n", migrationInfo.invites.Num(), GetLobbyName() );
|
|
|
|
migrationInfo.state = MIGRATE_BECOMING_HOST;
|
|
migrationInfo.migrationStartTime = Sys_Milliseconds();
|
|
|
|
if ( lobbyBackend == NULL ) {
|
|
// If we don't have a lobbyBackend, then just create one
|
|
Shutdown();
|
|
StartCreating();
|
|
return;
|
|
}
|
|
|
|
// Shutdown the current lobby, but keep the lobbyBackend (we'll migrate it)
|
|
Shutdown( true );
|
|
|
|
// Migrate the lobbyBackend to host
|
|
lobbyBackend->BecomeHost( migrationInfo.invites.Num() );
|
|
|
|
// Wait for it to complete
|
|
SetState( STATE_CREATE_LOBBY_BACKEND );
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::EndMigration
|
|
This gets called when we are done migrating, and invites will no longer be sent out.
|
|
========================
|
|
*/
|
|
void idLobby::EndMigration() {
|
|
if ( migrationInfo.state == MIGRATE_NONE ) {
|
|
idLib::Printf( "idSessionLocal::EndMigration: Not migrating.\n" );
|
|
return;
|
|
}
|
|
|
|
sessionCB->MigrationEnded( *this );
|
|
|
|
if ( lobbyBackend != NULL ) {
|
|
lobbyBackend->FinishBecomeHost();
|
|
}
|
|
|
|
migrationInfo.state = MIGRATE_NONE;
|
|
migrationInfo.invites.Clear();
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::ResetAllMigrationState
|
|
This will reset all state related to host migration. Should be called
|
|
at match end so our next game is not treated as a migrated game
|
|
========================
|
|
*/
|
|
void idLobby::ResetAllMigrationState() {
|
|
migrationInfo.state = MIGRATE_NONE;
|
|
migrationInfo.invites.Clear();
|
|
migrationInfo.persistUntilGameEndsData.Clear();
|
|
|
|
migrateMsgFlags = 0;
|
|
|
|
common->Dialog().ClearDialog( GDM_MIGRATING );
|
|
common->Dialog().ClearDialog( GDM_MIGRATING_WAITING );
|
|
common->Dialog().ClearDialog( GDM_MIGRATING_RELAUNCHING );
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::GetMigrationGameData
|
|
This will setup the passed in idBitMsg to either read or write from the global migration game data buffer
|
|
========================
|
|
*/
|
|
bool idLobby::GetMigrationGameData( idBitMsg &msg, bool reading ) {
|
|
if ( reading ) {
|
|
if ( !IsMigratedStatsGame() || !migrationInfo.persistUntilGameEndsData.wasMigratedHost ) {
|
|
// This was not a migrated session, we have no migration data
|
|
return false;
|
|
}
|
|
msg.InitRead( migrationInfo.persistUntilGameEndsData.gameData, sizeof( migrationInfo.persistUntilGameEndsData.gameData ) );
|
|
} else {
|
|
migrationInfo.persistUntilGameEndsData.hasGameData = true;
|
|
memset( migrationInfo.persistUntilGameEndsData.gameData, 0, sizeof( migrationInfo.persistUntilGameEndsData.gameData ) );
|
|
msg.InitWrite( migrationInfo.persistUntilGameEndsData.gameData, sizeof( migrationInfo.persistUntilGameEndsData.gameData ) );
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::GetMigrationGameDataUser
|
|
This will setup the passed in idBitMsg to either read or write from the user's migration game data buffer
|
|
========================
|
|
*/
|
|
bool idLobby::GetMigrationGameDataUser( lobbyUserID_t lobbyUserID, idBitMsg & msg, bool reading ) {
|
|
const int userNum = GetLobbyUserIndexByID( lobbyUserID );
|
|
|
|
if ( !verify( userNum >=0 && userNum < MAX_PLAYERS ) ) {
|
|
return false;
|
|
}
|
|
|
|
lobbyUser_t * u = GetLobbyUser( userNum );
|
|
if ( u != NULL ) {
|
|
if ( reading ) {
|
|
|
|
if ( !IsMigratedStatsGame() || !migrationInfo.persistUntilGameEndsData.wasMigratedHost ) {
|
|
// This was not a migrated session, we have no migration data
|
|
return false;
|
|
}
|
|
|
|
if ( u->migrationGameData >= 0 && u->migrationGameData < MAX_PLAYERS ) {
|
|
msg.InitRead( migrationInfo.persistUntilGameEndsData.gameDataUser[ u->migrationGameData ], sizeof( migrationInfo.persistUntilGameEndsData.gameDataUser[ 0 ] ) );
|
|
} else {
|
|
// We don't have migration data for this user
|
|
idLib::Warning( "No migration data for user %d in a migrated game (%d)", userNum, u->migrationGameData );
|
|
return false;
|
|
}
|
|
} else {
|
|
// Writing
|
|
migrationInfo.persistUntilGameEndsData.hasGameData = true;
|
|
u->migrationGameData = userNum;
|
|
memset( migrationInfo.persistUntilGameEndsData.gameDataUser[ userNum ], 0, sizeof( migrationInfo.persistUntilGameEndsData.gameDataUser[0] ) );
|
|
msg.InitWrite( migrationInfo.persistUntilGameEndsData.gameDataUser[ userNum ], sizeof( migrationInfo.persistUntilGameEndsData.gameDataUser[0] ) );
|
|
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::HandleMigrationGameData
|
|
========================
|
|
*/
|
|
void idLobby::HandleMigrationGameData( idBitMsg & msg ) {
|
|
// Receives game migration data from the server. Just save off the raw data. If we ever become host we'll let the game code read
|
|
// that chunk in (we can't do anything with it now anyways: we don't have entities or any server code to read it in to)
|
|
migrationInfo.persistUntilGameEndsData.hasGameData = true;
|
|
|
|
// Reset each user's migration game data. If we don't receive new data for them in this msg, we don't want to use the old data
|
|
for ( int i=0; i < GetNumLobbyUsers(); i++ ) {
|
|
lobbyUser_t * u = GetLobbyUser( i );
|
|
if ( u != NULL ) {
|
|
u->migrationGameData = -1;
|
|
}
|
|
}
|
|
|
|
msg.ReadData( migrationInfo.persistUntilGameEndsData.gameData, sizeof( migrationInfo.persistUntilGameEndsData.gameData ) );
|
|
int numUsers = msg.ReadByte();
|
|
int dataIndex=0;
|
|
for ( int i=0; i < numUsers; i++ ) {
|
|
lobbyUserID_t lobbyUserID;
|
|
lobbyUserID.ReadFromMsg( msg );
|
|
lobbyUser_t * user = GetLobbyUser( GetLobbyUserIndexByID( lobbyUserID ) );
|
|
if ( user != NULL ) {
|
|
|
|
NET_VERBOSE_PRINT( "NET: Got migration data[%d] for user %s\n", dataIndex, user->gamertag );
|
|
|
|
user->migrationGameData = dataIndex;
|
|
msg.ReadData( migrationInfo.persistUntilGameEndsData.gameDataUser[ dataIndex ], sizeof( migrationInfo.persistUntilGameEndsData.gameDataUser[ dataIndex ] ) );
|
|
dataIndex++;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
========================
|
|
idLobby::SendMigrationGameData
|
|
========================
|
|
*/
|
|
void idLobby::SendMigrationGameData() {
|
|
if ( net_migration_disable.GetBool() ) {
|
|
return;
|
|
}
|
|
|
|
if ( sessionCB->GetState() != idSession::INGAME ) {
|
|
return;
|
|
}
|
|
|
|
if ( !migrationInfo.persistUntilGameEndsData.hasGameData ) {
|
|
// Haven't been given any migration game data yet
|
|
return;
|
|
}
|
|
|
|
const int now = Sys_Milliseconds();
|
|
if ( nextSendMigrationGameTime > now ) {
|
|
return;
|
|
}
|
|
|
|
byte packetData[ idPacketProcessor::MAX_MSG_SIZE ];
|
|
idBitMsg msg( packetData, sizeof(packetData) );
|
|
|
|
// Write global data
|
|
msg.WriteData( &migrationInfo.persistUntilGameEndsData.gameData, sizeof( migrationInfo.persistUntilGameEndsData.gameData ) );
|
|
msg.WriteByte( GetNumLobbyUsers() );
|
|
|
|
// Write user data
|
|
for ( int userIndex = 0; userIndex < GetNumLobbyUsers(); ++userIndex ) {
|
|
lobbyUser_t * u = GetLobbyUser( userIndex );
|
|
if ( u->IsDisconnected() || u->migrationGameData < 0 ) {
|
|
continue;
|
|
}
|
|
|
|
u->lobbyUserID.WriteToMsg( msg );
|
|
msg.WriteData( migrationInfo.persistUntilGameEndsData.gameDataUser[ u->migrationGameData ], sizeof( migrationInfo.persistUntilGameEndsData.gameDataUser[ u->migrationGameData ] ) );
|
|
}
|
|
|
|
// Send to 1 peer
|
|
for ( int i=0; i < peers.Num(); i++ ) {
|
|
int peerToSend = ( nextSendMigrationGamePeer + i ) % peers.Num();
|
|
|
|
if ( peers[ peerToSend ].IsConnected() && peers[ peerToSend ].loaded ) {
|
|
if ( peers[ peerToSend ].packetProc->NumQueuedReliables() > idPacketProcessor::MAX_RELIABLE_QUEUE / 2 ) {
|
|
// This is kind of a hack for development so we don't DC clients by sending them too many reliable migration messages
|
|
// when they aren't responding. Doesn't seem like a horrible thing to have in a shipping product but is not necessary.
|
|
NET_VERBOSE_PRINT("NET: Skipping reliable game migration data msg because client reliable queue is > half full\n");
|
|
|
|
} else {
|
|
if ( net_migration_debug.GetBool() ) {
|
|
idLib::Printf( "NET: Sending migration game data to peer %d. size: %d\n", peerToSend, msg.GetSize() );
|
|
}
|
|
QueueReliableMessage( peerToSend, RELIABLE_MIGRATION_GAME_DATA, msg.GetReadData(), msg.GetSize() );
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Increment next send time / next send peer
|
|
nextSendMigrationGamePeer++;
|
|
if ( nextSendMigrationGamePeer >= peers.Num() ) {
|
|
nextSendMigrationGamePeer = 0;
|
|
}
|
|
|
|
nextSendMigrationGameTime = now + MIGRATION_GAME_DATA_INTERVAL_MS;
|
|
} |