SnapshotProcessor.cpp 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525
  1. /*
  2. ===========================================================================
  3. Doom 3 BFG Edition GPL Source Code
  4. Copyright (C) 1993-2012 id Software LLC, a ZeniMax Media company.
  5. This file is part of the Doom 3 BFG Edition GPL Source Code ("Doom 3 BFG Edition Source Code").
  6. Doom 3 BFG Edition Source Code is free software: you can redistribute it and/or modify
  7. it under the terms of the GNU General Public License as published by
  8. the Free Software Foundation, either version 3 of the License, or
  9. (at your option) any later version.
  10. Doom 3 BFG Edition Source Code is distributed in the hope that it will be useful,
  11. but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. GNU General Public License for more details.
  14. You should have received a copy of the GNU General Public License
  15. along with Doom 3 BFG Edition Source Code. If not, see <http://www.gnu.org/licenses/>.
  16. 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.
  17. 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.
  18. ===========================================================================
  19. */
  20. #pragma hdrstop
  21. #include "../idlib/precompiled.h"
  22. idCVar net_optimalSnapDeltaSize( "net_optimalSnapDeltaSize", "1000", CVAR_INTEGER, "Optimal size of snapshot delta msgs." );
  23. idCVar net_debugBaseStates( "net_debugBaseStates", "0", CVAR_BOOL, "Log out base state information" );
  24. idCVar net_skipClientDeltaAppend( "net_skipClientDeltaAppend", "0", CVAR_BOOL, "Simulate delta receive buffer overflowing" );
  25. /*
  26. ========================
  27. idSnapshotProcessor::idSnapshotProcessor
  28. ========================
  29. */
  30. idSnapshotProcessor::idSnapshotProcessor() {
  31. //assert( mem.IsGlobalHeap() );
  32. jobMemory = (jobMemory_t*)Mem_Alloc( sizeof( jobMemory_t) , TAG_NETWORKING );
  33. assert_16_byte_aligned( jobMemory );
  34. assert_16_byte_aligned( jobMemory->objParms.Ptr() );
  35. assert_16_byte_aligned( jobMemory->headers.Ptr() );
  36. assert_16_byte_aligned( jobMemory->lzwParms.Ptr() );
  37. Reset( true );
  38. }
  39. /*
  40. ========================
  41. idSnapshotProcessor::idSnapshotProcessor
  42. ========================
  43. */
  44. idSnapshotProcessor::~idSnapshotProcessor() {
  45. //mem.PushHeap();
  46. Mem_Free( jobMemory );
  47. //mem.PopHeap();
  48. }
  49. /*
  50. ========================
  51. idSnapshotProcessor::Reset
  52. ========================
  53. */
  54. void idSnapshotProcessor::Reset( bool cstor ) {
  55. hasPendingSnap = false;
  56. snapSequence = INITIAL_SNAP_SEQUENCE;
  57. baseSequence = -1;
  58. lastFullSnapBaseSequence = -1;
  59. if ( !cstor && net_debugBaseStates.GetBool() ) {
  60. idLib::Printf( "NET: Reset snapshot base");
  61. }
  62. baseState.Clear();
  63. submittedState.Clear();
  64. pendingSnap.Clear();
  65. deltas.Clear();
  66. partialBaseSequence = -1;
  67. memset( &jobMemory->lzwInOutData, 0, sizeof( jobMemory->lzwInOutData ) );
  68. }
  69. /*
  70. ========================
  71. idSnapshotProcessor::TrySetPendingSnapshot
  72. ========================
  73. */
  74. bool idSnapshotProcessor::TrySetPendingSnapshot( idSnapShot & ss ) {
  75. // Don't advance to the next snap until the last one was fully sent
  76. if ( hasPendingSnap ) {
  77. return false;
  78. }
  79. pendingSnap = ss;
  80. hasPendingSnap = true;
  81. return true;
  82. }
  83. /*
  84. ========================
  85. idSnapshotProcessor::PeekDeltaSequence
  86. ========================
  87. */
  88. void idSnapshotProcessor::PeekDeltaSequence( const char * deltaMem, int deltaSize, int & deltaSequence, int & deltaBaseSequence ) {
  89. idSnapShot::PeekDeltaSequence( deltaMem, deltaSize, deltaSequence, deltaBaseSequence );
  90. }
  91. /*
  92. ========================
  93. idSnapshotProcessor::ApplyDeltaToSnapshot
  94. ========================
  95. */
  96. bool idSnapshotProcessor::ApplyDeltaToSnapshot( idSnapShot & snap, const char * deltaMem, int deltaSize, int visIndex ) {
  97. return snap.ReadDeltaForJob( deltaMem, deltaSize, visIndex, &templateStates );
  98. }
  99. #ifdef STRESS_LZW_MEM
  100. // When this defined, we'll stress the lzw compressor with the smallest possible buffer, and detect when we need to grow it to make
  101. // sure we are gacefully detecting the situation.
  102. static int g_maxlwMem = 100;
  103. #endif
  104. /*
  105. ========================
  106. idSnapshotProcessor::SubmitPendingSnap
  107. ========================
  108. */
  109. void idSnapshotProcessor::SubmitPendingSnap( int visIndex, uint8 * objMemory, int objMemorySize, lzwCompressionData_t * lzwData ) {
  110. assert_16_byte_aligned( objMemory );
  111. assert_16_byte_aligned( lzwData );
  112. assert( hasPendingSnap );
  113. assert( jobMemory->lzwInOutData.numlzwDeltas == 0 );
  114. assert( net_optimalSnapDeltaSize.GetInteger() < jobMemory_t::MAX_LZW_MEM - 128 ); // Leave padding
  115. jobMemory->lzwInOutData.lzwDeltas = jobMemory->lzwDeltas.Ptr();
  116. jobMemory->lzwInOutData.maxlzwDeltas = jobMemory->lzwDeltas.Num();
  117. jobMemory->lzwInOutData.lzwMem = jobMemory->lzwMem.Ptr();
  118. #ifdef STRESS_LZW_MEM
  119. jobMemory->lzwInOutData.maxlzwMem = g_maxlwMem;
  120. #else
  121. jobMemory->lzwInOutData.maxlzwMem = jobMemory_t::MAX_LZW_MEM;
  122. #endif
  123. jobMemory->lzwInOutData.lzwDmaOut = jobMemory_t::MAX_LZW_MEM;
  124. jobMemory->lzwInOutData.numlzwDeltas = 0;
  125. jobMemory->lzwInOutData.lzwBytes = 0;
  126. jobMemory->lzwInOutData.optimalLength = net_optimalSnapDeltaSize.GetInteger();
  127. jobMemory->lzwInOutData.snapSequence = snapSequence;
  128. jobMemory->lzwInOutData.lastObjId = 0;
  129. jobMemory->lzwInOutData.lzwData = lzwData;
  130. idSnapShot::submitDeltaJobsInfo_t submitInfo;
  131. submitInfo.objParms = jobMemory->objParms.Ptr();
  132. submitInfo.maxObjParms = jobMemory->objParms.Num();
  133. submitInfo.headers = jobMemory->headers.Ptr();
  134. submitInfo.maxHeaders = jobMemory->headers.Num();
  135. submitInfo.objMemory = objMemory;
  136. submitInfo.maxObjMemory = objMemorySize;
  137. submitInfo.lzwParms = jobMemory->lzwParms.Ptr();
  138. submitInfo.maxDeltaParms = jobMemory->lzwParms.Num();
  139. // Use a copy of base state to avoid race conditions.
  140. // The main thread could change it behind the jobs backs.
  141. submittedState = baseState;
  142. submittedTemplateStates = templateStates;
  143. submitInfo.templateStates = &submittedTemplateStates;
  144. submitInfo.oldSnap = &submittedState;
  145. submitInfo.visIndex = visIndex;
  146. submitInfo.baseSequence = baseSequence;
  147. submitInfo.lzwInOutData = &jobMemory->lzwInOutData;
  148. pendingSnap.SubmitWriteDeltaToJobs( submitInfo );
  149. }
  150. /*
  151. ========================
  152. idSnapshotProcessor::GetPendingSnapDelta
  153. ========================
  154. */
  155. int idSnapshotProcessor::GetPendingSnapDelta( byte * outBuffer, int maxLength ) {
  156. assert( PendingSnapReadyToSend() );
  157. if ( !verify( jobMemory->lzwInOutData.numlzwDeltas == 1 ) ) {
  158. jobMemory->lzwInOutData.numlzwDeltas = 0;
  159. return 0; // No more deltas left to send
  160. }
  161. assert( hasPendingSnap );
  162. jobMemory->lzwInOutData.numlzwDeltas = 0;
  163. int size = jobMemory->lzwDeltas[0].size;
  164. if ( !verify( size != -1 ) ) {
  165. #ifdef STRESS_LZW_MEM
  166. if ( g_maxlwMem < MAX_LZW_MEM ) {
  167. g_maxlwMem += 50;
  168. g_maxlwMem = Min( g_maxlwMem, MAX_LZW_MEM );
  169. return 0;
  170. }
  171. #endif
  172. // This can happen if there wasn't enough maxlzwMem to process one full obj in a single delta
  173. idLib::Error( "GetPendingSnapDelta: Delta failed." );
  174. }
  175. uint8 * deltaData = &jobMemory->lzwMem[jobMemory->lzwDeltas[0].offset];
  176. int deltaSequence = 0;
  177. int deltaBaseSequence = 0;
  178. PeekDeltaSequence( (const char *)deltaData, size, deltaSequence, deltaBaseSequence );
  179. // sanity check: does the compressed data we are about to send have the sequence number we expect
  180. assert( deltaSequence == jobMemory->lzwDeltas[0].snapSequence );
  181. if ( !verify( size <= maxLength ) ) {
  182. idLib::Error( "GetPendingSnapDelta: Size overflow." );
  183. }
  184. // Copy to out buffer
  185. memcpy( outBuffer, deltaData, size );
  186. // Set the sequence to what this delta actually belongs to
  187. assert( jobMemory->lzwDeltas[0].snapSequence == snapSequence + 1 );
  188. snapSequence = jobMemory->lzwDeltas[0].snapSequence;
  189. //idLib::Printf( "deltas Num: %i, Size: %i\n", deltas.Num(), deltas.GetDataLength() );
  190. // Copy to delta buffer
  191. // NOTE - We don't need to save this delta off if peer has already ack'd this basestate.
  192. // This can happen due to the fact that we defer the processing of snap deltas on jobs.
  193. // When we start processing a delta, we use the currently ack'd basestate. If while we were processing
  194. // the delta, the client acks a new basestate, we can get into this situation. In this case, we simply don't
  195. // store the delta, since it will just take up space, and just get removed anyways during ApplySnapshotDelta.
  196. // (and cause lots of spam when it sees the delta's basestate doesn't match the current ack'd one)
  197. if ( deltaBaseSequence >= baseSequence ) {
  198. if ( !deltas.Append( snapSequence, deltaData, size ) ) {
  199. int resendLength = deltas.ItemLength( deltas.Num() - 1 );
  200. if ( !verify( resendLength <= maxLength ) ) {
  201. idLib::Error( "GetPendingSnapDelta: Size overflow for resend." );
  202. }
  203. memcpy( outBuffer, deltas.ItemData( deltas.Num() - 1 ), resendLength );
  204. size = -resendLength;
  205. }
  206. }
  207. if ( jobMemory->lzwInOutData.fullSnap ) {
  208. // We sent the full snap, we can stop sending this pending snap now...
  209. NET_VERBOSESNAPSHOT_PRINT_LEVEL( 5, va( " wrote enough deltas to a full snapshot\n" ) ); // FIXME: peer number?
  210. hasPendingSnap = false;
  211. partialBaseSequence = -1;
  212. } else {
  213. partialBaseSequence = deltaBaseSequence;
  214. }
  215. return size;
  216. }
  217. /*
  218. ========================
  219. idSnapshotProcessor::IsBusyConfirmingPartialSnap
  220. ========================
  221. */
  222. bool idSnapshotProcessor::IsBusyConfirmingPartialSnap() {
  223. if ( partialBaseSequence != -1 && baseSequence <= partialBaseSequence ) {
  224. return true;
  225. }
  226. return false;
  227. }
  228. /*
  229. ========================
  230. idSnapshotProcessor::ReceiveSnapshotDelta
  231. NOTE: we use ReadDeltaForJob twice, once to build the same base as the server (based on server acks, down ApplySnapshotDelta), and another time to apply the snapshot we just received
  232. could we avoid the double apply by keeping outSnap cached in memory and avoid rebuilding it from a delta when the next one comes around?
  233. ========================
  234. */
  235. bool idSnapshotProcessor::ReceiveSnapshotDelta( const byte * deltaData, int deltaLength, int visIndex, int & outSeq, int & outBaseSeq, idSnapShot & outSnap, bool & fullSnap ) {
  236. fullSnap = false;
  237. int deltaSequence = 0;
  238. int deltaBaseSequence = 0;
  239. // Get the sequence of this delta, and the base sequence it is delta'd from
  240. PeekDeltaSequence( (const char *)deltaData, deltaLength, deltaSequence, deltaBaseSequence );
  241. //idLib::Printf("Incoming snapshot: %i, %i\n", deltaSequence, deltaBaseSequence );
  242. if ( deltaSequence <= snapSequence ) {
  243. NET_VERBOSESNAPSHOT_PRINT( "Rejecting old delta: %d (snapSequence: %d \n", deltaSequence, snapSequence );
  244. return false; // Completely reject older out of order deltas
  245. }
  246. // Bring the base state up to date with the basestate this delta was compared to
  247. ApplySnapshotDelta( visIndex, deltaBaseSequence );
  248. // Once we get here, our base state should be caught up to that of the server
  249. assert( baseSequence == deltaBaseSequence );
  250. // Save the new delta
  251. if ( net_skipClientDeltaAppend.GetBool() || !deltas.Append( deltaSequence, deltaData, deltaLength ) ) {
  252. // This can happen if the delta queues get desync'd between the server and client.
  253. // With recent fixes, this should be extremely rare, or impossible.
  254. // Just in case this happens, we can recover by assuming we didn't even receive this delta.
  255. idLib::Printf( "NET: ReceiveSnapshotDelta: No room to append delta %d/%d \n", deltaSequence, deltaBaseSequence );
  256. return false;
  257. }
  258. // Update our snapshot sequence number to the newer one we just got (now that it's safe)
  259. snapSequence = deltaSequence;
  260. if ( deltas.Num() > 10 ) {
  261. NET_VERBOSESNAPSHOT_PRINT("NET: ReceiveSnapshotDelta: deltas.Num() > 10: %d\n ", deltas.Num() );
  262. for ( int i=0; i < deltas.Num(); i++ ) {
  263. NET_VERBOSESNAPSHOT_PRINT( "%d ", deltas.ItemSequence( i ) );
  264. }
  265. NET_VERBOSESNAPSHOT_PRINT( "\n" );
  266. }
  267. if ( baseSequence != deltaBaseSequence ) {
  268. // NOTE - With recent fixes, this should no longer be possible unless the delta is trashed
  269. // We should probably disconnect from the server when this happens now.
  270. static bool failed = false;
  271. if ( !failed ) {
  272. idLib::Printf( "NET: incorrect base state? not sure how this can happen... baseSequence: %d deltaBaseSequence: %d \n", baseSequence, deltaBaseSequence );
  273. }
  274. failed = true;
  275. return false;
  276. }
  277. // Copy out the current deltas sequence values to caller
  278. outSeq = deltaSequence;
  279. outBaseSeq = deltaBaseSequence;
  280. if ( baseSequence < 50 && net_debugBaseStates.GetBool() ) {
  281. idLib::Printf( "NET: Proper basestate... baseSequence: %d deltaBaseSequence: %d \n", baseSequence, deltaBaseSequence );
  282. }
  283. // Make a copy of the basestate the server used to create this delta, and then apply and return it
  284. outSnap = baseState;
  285. fullSnap = ApplyDeltaToSnapshot( outSnap, (const char *)deltaData, deltaLength, visIndex );
  286. // We received a new delta
  287. return true;
  288. }
  289. /*
  290. ========================
  291. idSnapshotProcessor::ApplySnapshotDelta
  292. Apply a snapshot delta to our current basestate, and make that the new base.
  293. We can remove all deltas that refer to the basetate we just removed.
  294. ========================
  295. */
  296. bool idSnapshotProcessor::ApplySnapshotDelta( int visIndex, int snapshotNumber ) {
  297. NET_VERBOSESNAPSHOT_PRINT_LEVEL( 6, va( "idSnapshotProcessor::ApplySnapshotDelta snapshotNumber: %d\n", snapshotNumber ) );
  298. // Sanity check deltas
  299. SanityCheckDeltas();
  300. // dump any deltas older than the acknoweledged snapshot, which should only happen if there is packet loss
  301. deltas.RemoveOlderThan( snapshotNumber );
  302. if ( deltas.Num() == 0 || deltas.ItemSequence( 0 ) != snapshotNumber ) {
  303. // this means the snapshot was either already acknowledged or came out of order
  304. // On the server, this can happen because the client is continuously/redundantly sending acks
  305. // Once the server has ack'd a certain base sequence, it will need to ignore all the redundant ones.
  306. // On the client, this will only happen due to out of order, or dropped packets.
  307. if ( !common->IsServer() ) {
  308. // these should be printed every time on the clients
  309. // printing on server is not useful / results in tons of spam
  310. if ( deltas.Num() == 0 ) {
  311. NET_VERBOSESNAPSHOT_PRINT("NET: Got snapshot but ignored... deltas.Num(): %d snapshotNumber: %d \n", deltas.Num(), snapshotNumber );
  312. } else {
  313. NET_VERBOSESNAPSHOT_PRINT("NET: Got snapshot but ignored... deltas.ItemSequence( 0 ): %d != snapshotNumber: %d \n ", deltas.ItemSequence( 0 ), snapshotNumber );
  314. for ( int i=0; i < deltas.Num(); i++ ) {
  315. NET_VERBOSESNAPSHOT_PRINT("%d ", deltas.ItemSequence(i) );
  316. }
  317. NET_VERBOSESNAPSHOT_PRINT("\n");
  318. }
  319. }
  320. return false;
  321. }
  322. int deltaSequence = 0;
  323. int deltaBaseSequence = 0;
  324. PeekDeltaSequence( (const char *)deltas.ItemData( 0 ), deltas.ItemLength( 0 ), deltaSequence, deltaBaseSequence );
  325. assert( deltaSequence == snapshotNumber ); // Make sure compressed sequence number matches that in data queue
  326. assert( baseSequence == deltaBaseSequence ); // If this delta isn't based off of our currently ack'd basestate, something is trashed...
  327. assert( deltaSequence > baseSequence );
  328. if ( baseSequence != deltaBaseSequence ) {
  329. // NOTE - This should no longer happen with recent fixes.
  330. // We should probably disconnect from the server if this happens. (packets are trashed most likely)
  331. NET_VERBOSESNAPSHOT_PRINT( "NET: Got snapshot %d but baseSequence does not match. baseSequence: %d deltaBaseSequence: %d. \n", snapshotNumber, baseSequence, deltaBaseSequence );
  332. return false;
  333. }
  334. // Apply this delta to our base state
  335. if ( ApplyDeltaToSnapshot( baseState, (const char *)deltas.ItemData( 0 ), deltas.ItemLength( 0 ), visIndex ) ) {
  336. lastFullSnapBaseSequence = deltaSequence;
  337. }
  338. baseSequence = deltaSequence; // This is now our new base sequence
  339. // Remove deltas that we no longer need
  340. RemoveDeltasForOldBaseSequence();
  341. // Sanity check deltas
  342. SanityCheckDeltas();
  343. return true;
  344. }
  345. /*
  346. ========================
  347. idSnapshotProcessor::RemoveDeltasForOldBaseSequence
  348. Remove deltas for basestate we no longer have. We know we can remove them, because we will never
  349. be able to apply them, since the basestate needed to generate a full snap from these deltas is gone.
  350. Ways we can get deltas based on basestate we no longer have:
  351. 1. Server sends sequence 50 based on 49. It then sends sequence 51 based on 49.
  352. Client acks 50, server applies it to 49, 50 is new base state.
  353. Server now has a delta sequence 51 based on 49 that it won't ever be able to apply (50 is new basestate).
  354. This is annoying, because it makes a lot of our sanity checks incorrectly fire off for benign issues.
  355. Here is a series of events that make the old ( baseSequence != deltaBaseSequence ) assert:
  356. Server:
  357. 49->50, 49->51, ack 50, 50->52, ack 51 (bam), 50->53
  358. Client
  359. 49->50, ack 50, 49->51, ack 51 (bam), 50->52, 50->53
  360. The client above will ack 51, even though he can't even apply that delta. To get around this, we simply don't
  361. allow delta to exist in the list, unless their basestate is the current basestate we maintain.
  362. This allows us to put sanity checks in place that don't fire off during benign conditions, and allow us
  363. to truly check for trashed conditions.
  364. ========================
  365. */
  366. void idSnapshotProcessor::RemoveDeltasForOldBaseSequence() {
  367. // Remove any deltas that would apply to the old base we no longer maintain
  368. // (we will never be able to apply these, since we don't have that base anymore)
  369. for ( int i = deltas.Num() - 1; i >= 0; i-- ) {
  370. int deltaSequence = 0;
  371. int deltaBaseSequence = 0;
  372. baseState.PeekDeltaSequence( (const char *)deltas.ItemData( i ), deltas.ItemLength( i ), deltaSequence, deltaBaseSequence );
  373. if ( deltaBaseSequence < baseSequence ) {
  374. // Remove this delta, and all deltas before this one
  375. deltas.RemoveOlderThan( deltas.ItemSequence( i ) + 1 );
  376. break;
  377. }
  378. }
  379. }
  380. /*
  381. ========================
  382. idSnapshotProcessor::SanityCheckDeltas
  383. Make sure delta sequence and basesequence values are valid, and in order, etc
  384. ========================
  385. */
  386. void idSnapshotProcessor::SanityCheckDeltas() {
  387. int deltaSequence = 0;
  388. int deltaBaseSequence = 0;
  389. int lastDeltaSequence = -1;
  390. int lastDeltaBaseSequence = -1;
  391. for ( int i = 0; i < deltas.Num(); i++ ) {
  392. baseState.PeekDeltaSequence( (const char *)deltas.ItemData( i ), deltas.ItemLength( i ), deltaSequence, deltaBaseSequence );
  393. assert( deltaSequence == deltas.ItemSequence( i ) ); // Make sure delta stored in compressed form matches the one stored in the data queue
  394. assert( deltaSequence > lastDeltaSequence ); // Make sure they are in order (we reject out of order sequences in ApplysnapshotDelta)
  395. assert( deltaBaseSequence >= lastDeltaBaseSequence ); // Make sure they are in order (they can be the same, since base sequences don't change until they've been ack'd)
  396. assert( deltaBaseSequence >= baseSequence ); // We should have removed old delta's that can no longer be applied
  397. assert( deltaBaseSequence == baseSequence || deltaBaseSequence == lastDeltaSequence ); // Make sure we still have a base (or eventually will have) that we can apply this delta to
  398. lastDeltaSequence = deltaSequence;
  399. lastDeltaBaseSequence = deltaBaseSequence;
  400. }
  401. }
  402. /*
  403. ========================
  404. idSnapshotProcessor::AddSnapObjTemplate
  405. ========================
  406. */
  407. void idSnapshotProcessor::AddSnapObjTemplate( int objID, idBitMsg & msg ) {
  408. extern idCVar net_ssTemplateDebug;
  409. idSnapShot::objectState_t * state = templateStates.S_AddObject( objID, MAX_UNSIGNED_TYPE( uint32 ), msg );
  410. if ( verify( state != NULL ) ) {
  411. if ( net_ssTemplateDebug.GetBool() ) {
  412. idLib::PrintfIf( net_ssTemplateDebug.GetBool(), "InjectingSnapObjBaseState[%d] size: %d\n", objID, state->buffer.Size() );
  413. state->Print( "BASE STATE" );
  414. }
  415. state->expectedSequence = snapSequence;
  416. }
  417. }