123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591 |
- #pragma hdrstop
- #include "../idLib/precompiled.h"
- idCVar net_maxRate( "net_maxRate", "50", CVAR_INTEGER, "max send rate in kilobytes per second" );
- idCVar net_showReliableCompression( "net_showReliableCompression", "0", CVAR_BOOL, "Show reliable compression ratio." );
- void idPacketProcessor::QueueReliableAck( int lastReliable ) {
-
- if ( lastReliable >= reliableSequenceRecv ) {
- queuedReliableAck = lastReliable;
- reliableSequenceRecv = lastReliable;
- }
- }
- int idPacketProcessor::FinalizeRead( idBitMsg & inMsg, idBitMsg & outMsg, int & userValue ) {
- userValue = 0;
-
- idInnerPacketHeader header;
- header.ReadFromMsg( inMsg );
-
- if ( !verify( header.Type() != PACKET_TYPE_FRAGMENTED ) ) {
- idLib::Printf("Received invalid fragmented packet.\n" );
- return RETURN_TYPE_NONE;
- }
- if ( header.Type() == PACKET_TYPE_RELIABLE_ACK ) {
-
- int reliableSequence = inMsg.ReadLong();
- reliable.RemoveOlderThan( reliableSequence + 1 );
- header.ReadFromMsg( inMsg );
- }
- if ( header.Type() == PACKET_TYPE_OOB ) {
-
- userValue = header.Value();
- } else {
-
- if ( !verify( header.Type() == PACKET_TYPE_INBAND ) ) {
- idLib::Printf("In-band packet expected, received type %i instead.\n", header.Type() );
- return RETURN_TYPE_NONE;
- }
-
-
- numReliable = 0;
-
-
- int numReliableRecv = header.Value();
- int bufferPos = 0;
-
- if ( numReliableRecv > 0 ) {
-
- inMsg.ReadByteAlign();
- int compressedSize = inMsg.ReadShort();
- lzwCompressionData_t lzwData;
- idLZWCompressor lzwCompressor( &lzwData );
-
- lzwCompressor.Start( (uint8*)inMsg.GetReadData() + inMsg.GetReadCount(), compressedSize );
- int reliableSequence = 0;
- lzwCompressor.ReadAgnostic< int >( reliableSequence );
- for ( int r = 0; r < numReliableRecv; r++ ) {
- uint8 uncompMem[ MAX_MSG_SIZE ];
- uint16 reliableDataLength = 0;
- lzwCompressor.ReadAgnostic< uint16 >( reliableDataLength );
- lzwCompressor.Read( uncompMem, reliableDataLength );
- if ( reliableSequence + r > reliableSequenceRecv ) {
- if ( !verify( bufferPos + reliableDataLength <= sizeof( reliableBuffer ) ) ) {
- idLib::Printf( "Reliable msg size overflow.\n" );
- return RETURN_TYPE_NONE;
- }
- if ( !verify( numReliable < MAX_RELIABLE_QUEUE ) ) {
- idLib::Printf( "Reliable msg count overflow.\n" );
- return RETURN_TYPE_NONE;
- }
- memcpy( reliableBuffer + bufferPos, uncompMem, reliableDataLength );
- reliableMsgSize[ numReliable ] = reliableDataLength;
- reliableMsgPtrs[ numReliable++ ] = &reliableBuffer[ bufferPos ];
- bufferPos += reliableDataLength;
- } else {
- extern idCVar net_verboseReliable;
- if ( net_verboseReliable.GetBool() ) {
- idLib::Printf( "Ignoring reliable msg %i because %i was already acked\n", ( reliableSequence + r ), reliableSequenceRecv );
- }
- }
-
- if ( !verify( lzwCompressor.IsOverflowed() == false ) ) {
- idLib::Printf( "lzwCompressor.IsOverflowed() == true.\n" );
- return RETURN_TYPE_NONE;
- }
- }
-
- inMsg.SetReadCount( inMsg.GetReadCount() + compressedSize );
- QueueReliableAck( reliableSequence + numReliableRecv - 1 );
- }
- }
-
-
- outMsg.BeginWriting();
- outMsg.WriteData( inMsg.GetReadData() + inMsg.GetReadCount(), inMsg.GetRemainingData() );
- outMsg.SetSize( inMsg.GetRemainingData() );
-
- return ( header.Type() == PACKET_TYPE_OOB ) ? RETURN_TYPE_OOB : RETURN_TYPE_INBAND;
- }
- bool idPacketProcessor::QueueReliableMessage( byte type, const byte * data, int dataLen ) {
- return reliable.Append( reliableSequenceSend++, &type, 1, data, dataLen );
- }
- bool idPacketProcessor::CanSendMoreData() const {
- if ( net_maxRate.GetInteger() == 0 ) {
- return true;
- }
- return ( outgoingRateBytes <= net_maxRate.GetInteger() * 1024 );
- }
- void idPacketProcessor::UpdateOutgoingRate( const int time, const int size ) {
- outgoingBytes += size;
-
- if ( time > outgoingRateTime ) {
- outgoingRateBytes -= outgoingRateBytes * (float)( time - outgoingRateTime ) / 1000.0f;
- if ( outgoingRateBytes < 0.0f ) {
- outgoingRateBytes = 0.0f;
- }
- }
-
- outgoingRateTime = time;
- outgoingRateBytes += size;
-
- if ( time - lastOutgoingRateTime > BANDWIDTH_AVERAGE_PERIOD ) {
- currentOutgoingRate = 1000 * ( outgoingBytes - lastOutgoingBytes ) / ( time - lastOutgoingRateTime );
- lastOutgoingBytes = outgoingBytes;
- lastOutgoingRateTime = time;
- }
- }
- void idPacketProcessor::UpdateIncomingRate( const int time, const int size ) {
- incomingBytes += size;
-
- if ( time > incomingRateTime ) {
- incomingRateBytes -= incomingRateBytes * (float)( time - incomingRateTime ) / 1000.0f;
- if ( incomingRateBytes < 0.0f ) {
- incomingRateBytes = 0.0f;
- }
- }
- incomingRateTime = time;
- incomingRateBytes += size;
-
- if ( time - lastIncomingRateTime > BANDWIDTH_AVERAGE_PERIOD ) {
- currentIncomingRate = 1000 * ( incomingBytes - lastIncomingBytes ) / ( time - lastIncomingRateTime );
- lastIncomingBytes = incomingBytes;
- lastIncomingRateTime = time;
- }
- }
- bool idPacketProcessor::ProcessOutgoing( const int time, const idBitMsg & msg, bool isOOB, int userData ) {
-
-
- if ( !verify( fragmentedSend == false ) ) {
- idLib::Warning( "ProcessOutgoing: fragmentedSend == true!");
- return false;
- }
-
- if ( !verify( unsentMsg.GetRemainingData() == 0 ) ) {
- idLib::Warning( "ProcessOutgoing: unsentMsg.GetRemainingData() > 0!");
- return false;
- }
-
-
- unsentMsg.InitWrite( unsentBuffer, sizeof( unsentBuffer ) );
- unsentMsg.BeginWriting();
-
-
- if ( queuedReliableAck >= 0 ) {
- idInnerPacketHeader header( PACKET_TYPE_RELIABLE_ACK, 0 );
- header.WriteToMsg( unsentMsg );
- unsentMsg.WriteLong( queuedReliableAck );
- queuedReliableAck = -1;
- }
- if ( isOOB ) {
- if ( msg.GetSize() + unsentMsg.GetSize() > MAX_OOB_MSG_SIZE ) {
- idLib::Printf("Out-of-band packet too large %i\n", unsentMsg.GetSize() );
- assert( 0 );
- return false;
- }
-
- idInnerPacketHeader header( PACKET_TYPE_OOB, userData );
- header.WriteToMsg( unsentMsg );
- } else {
-
- idInnerPacketHeader header( PACKET_TYPE_INBAND, reliable.Num() );
- header.WriteToMsg( unsentMsg );
- if ( reliable.Num() > 0 ) {
-
- unsentMsg.WriteByteAlign();
-
- lzwCompressionData_t lzwData;
- idLZWCompressor lzwCompressor( &lzwData );
- lzwCompressor.Start( unsentMsg.GetWriteData() + unsentMsg.GetSize() + 2, unsentMsg.GetRemainingSpace() - 2 );
- int uncompressedSize = 4;
- lzwCompressor.WriteAgnostic< int >( reliable.ItemSequence( 0 ) );
- for ( int i = 0; i < reliable.Num(); i++ ) {
- lzwCompressor.WriteAgnostic< uint16 >( reliable.ItemLength( i ) );
- lzwCompressor.Write( reliable.ItemData( i ), reliable.ItemLength( i ) );
- uncompressedSize += 2 + reliable.ItemLength( i );
- }
- lzwCompressor.End();
- if ( lzwCompressor.IsOverflowed() ) {
- idLib::Error( "reliable msg compressor overflow." );
- }
- unsentMsg.WriteShort( lzwCompressor.Length() );
- unsentMsg.SetSize( unsentMsg.GetSize() + lzwCompressor.Length() );
- if ( net_showReliableCompression.GetBool() ) {
- static int totalUncompressed = 0;
- static int totalCompressed = 0;
- totalUncompressed += uncompressedSize;
- totalCompressed += lzwCompressor.Length();
- float ratio1 = (float)lzwCompressor.Length() / (float)uncompressedSize;
- float ratio2 = (float)totalCompressed / (float)totalUncompressed;
- idLib::Printf( "Uncompressed: %i, Compressed: %i, TotalUncompressed: %i, TotalCompressed: %i, (%2.2f / %2.2f )\n", uncompressedSize, lzwCompressor.Length(), totalUncompressed, totalCompressed, ratio1, ratio2 );
- }
- }
- }
-
-
- unsentMsg.WriteData( msg.GetReadData(), msg.GetSize() );
- if ( unsentMsg.GetSize() > MAX_PACKET_SIZE ) {
- if ( isOOB ) {
- idLib::Error( "oob msg's cannot fragment" );
- }
- fragmentedSend = true;
- }
-
- return true;
- }
- bool idPacketProcessor::GetSendFragment( const int time, sessionId_t sessionID, idBitMsg & outMsg ) {
- lastSendTime = time;
- if ( unsentMsg.GetRemainingData() <= 0 ) {
- return false;
- }
-
- outMsg.BeginWriting();
- idOuterPacketHeader outerHeader( sessionID );
-
- outerHeader.WriteToMsg( outMsg );
-
- if ( !fragmentedSend ) {
-
- outMsg.WriteData( unsentMsg.GetReadData(), unsentMsg.GetSize() );
- unsentMsg.SetSize( 0 );
- } else {
- int currentSize = idMath::ClampInt( 0, MAX_PACKET_SIZE, unsentMsg.GetRemainingData() );
- assert( currentSize > 0 );
- assert( unsentMsg.GetRemainingData() - currentSize >= 0 );
-
-
- bool moreFragments = ( unsentMsg.GetRemainingData() - currentSize > 0 ) ? true : false;
-
- if ( !unsentMsg.GetReadCount() ) {
- assert( moreFragments );
- idInnerPacketHeader header( PACKET_TYPE_FRAGMENTED, FRAGMENT_START );
- header.WriteToMsg( outMsg );
- } else {
- idInnerPacketHeader header( PACKET_TYPE_FRAGMENTED, moreFragments ? FRAGMENT_MIDDLE : FRAGMENT_END );
- header.WriteToMsg( outMsg );
- }
-
- outMsg.WriteLong( fragmentSequence );
- outMsg.WriteData( unsentMsg.GetReadData() + unsentMsg.GetReadCount(), currentSize );
- unsentMsg.ReadData( NULL, currentSize );
- assert( moreFragments == unsentMsg.GetRemainingData() > 0 );
- fragmentedSend = moreFragments;
-
- fragmentSequence++;
- fragmentAccumulator++;
- }
-
- UpdateOutgoingRate( time, outMsg.GetSize() );
-
- return true;
- }
- int idPacketProcessor::ProcessIncoming( int time, sessionId_t expectedSessionID, idBitMsg & msg, idBitMsg & out, int & userData, const int peerNum ) {
- assert( msg.GetSize() <= MAX_FINAL_PACKET_SIZE );
-
- UpdateIncomingRate( time, msg.GetSize() );
- idOuterPacketHeader outerHeader;
- outerHeader.ReadFromMsg( msg );
- sessionId_t sessionID = outerHeader.GetSessionID();
- assert( sessionID == expectedSessionID );
-
- if ( !verify( sessionID != SESSION_ID_CONNECTIONLESS_PARTY && sessionID != SESSION_ID_CONNECTIONLESS_GAME && sessionID != SESSION_ID_CONNECTIONLESS_GAME_STATE ) ) {
- idLib::Printf( "Expected non connectionless ID, but got a connectionless one\n" );
- return RETURN_TYPE_NONE;
- }
-
- if ( sessionID != expectedSessionID ) {
- idLib::Printf( "Expected session id: %8x but got %8x instead\n", expectedSessionID, sessionID );
- return RETURN_TYPE_NONE;
- }
- int c,b;
- msg.SaveReadState( c, b );
- idInnerPacketHeader header;
- header.ReadFromMsg( msg );
- if ( header.Type() != PACKET_TYPE_FRAGMENTED ) {
-
- msg.RestoreReadState( c, b );
- return FinalizeRead( msg, out, userData );
- }
-
- int readSequence = msg.ReadLong();
- if ( header.Value() == FRAGMENT_START ) {
- msgWritePos = 0;
- } else if ( fragmentSequence == -1 || readSequence != fragmentSequence + 1 ) {
- droppedFrags++;
- idLib::Printf( "Dropped Fragments - PeerNum: %i FragmentSeq: %i, ReadSeq: %i, Total: %i\n", peerNum, fragmentSequence, readSequence, droppedFrags );
-
- fragmentSequence = -1;
- return RETURN_TYPE_NONE;
- }
- fragmentSequence = readSequence;
- assert( msg.GetRemainingData() > 0 );
- if ( !verify( msgWritePos + msg.GetRemainingData() < sizeof( msgBuffer ) ) ) {
- idLib::Error( "ProcessIncoming: Fragmented msg buffer overflow." );
- }
- memcpy( msgBuffer + msgWritePos, msg.GetReadData() + msg.GetReadCount(), msg.GetRemainingData() );
- msgWritePos += msg.GetRemainingData();
-
- if ( header.Value() == FRAGMENT_END ) {
-
- idBitMsg msg( msgBuffer, sizeof( msgBuffer ) );
- msg.SetSize( msgWritePos );
- return FinalizeRead( msg, out, userData );
- }
-
- if ( !verify( header.Value() == FRAGMENT_START || header.Value() == FRAGMENT_MIDDLE ) ) {
- idLib::Printf( "ProcessIncoming: Invalid packet.\n" );
- }
-
-
- return RETURN_TYPE_NONE;
- }
-
- bool idPacketProcessor::ProcessConnectionlessOutgoing( idBitMsg & msg, idBitMsg & out, int lobbyType, int userData ) {
- sessionId_t sessionID = lobbyType + 1;
-
-
- idOuterPacketHeader outerHeader( sessionID );
- outerHeader.WriteToMsg( out );
-
- idInnerPacketHeader header( PACKET_TYPE_OOB, userData );
- header.WriteToMsg( out );
-
- out.WriteData( msg.GetReadData(), msg.GetSize() );
- return true;
- }
- bool idPacketProcessor::ProcessConnectionlessIncoming( idBitMsg & msg, idBitMsg & out, int & userData ) {
- idOuterPacketHeader outerHeader;
- outerHeader.ReadFromMsg( msg );
- sessionId_t sessionID = outerHeader.GetSessionID();
- if ( sessionID != SESSION_ID_CONNECTIONLESS_PARTY && sessionID != SESSION_ID_CONNECTIONLESS_GAME && sessionID != SESSION_ID_CONNECTIONLESS_GAME_STATE ) {
-
- idLib::Printf( "ProcessConnectionlessIncoming: Invalid session ID - %d\n", sessionID );
- return false;
- }
- idInnerPacketHeader header;
- header.ReadFromMsg( msg );
- if ( header.Type() != PACKET_TYPE_OOB ) {
- idLib::Printf( "ProcessConnectionlessIncoming: header.Type() != PACKET_TYPE_OOB\n" );
- return false;
- }
-
- userData = header.Value();
-
- out.BeginWriting();
- out.WriteData( msg.GetReadData() + msg.GetReadCount(), msg.GetRemainingData() );
- out.SetSize( msg.GetRemainingData() );
- return true;
- }
- idPacketProcessor::sessionId_t idPacketProcessor::GetSessionID( idBitMsg & msg ) {
- sessionId_t sessionID;
- int c,b;
- msg.SaveReadState( c, b );
-
- idOuterPacketHeader outerHeader;
- outerHeader.ReadFromMsg( msg );
-
-
- sessionID = outerHeader.GetSessionID();
- msg.RestoreReadState( c, b );
- return sessionID;
- }
- idCVar net_verifyReliableQueue( "net_verifyReliableQueue", "2", CVAR_INTEGER, "0: warn only, 1: error, 2: fixup, 3: fixup and verbose, 4: force test" );
- #define RELIABLE_VERBOSE if ( net_verifyReliableQueue.GetInteger() >= 3 ) idLib::Printf
- void idPacketProcessor::VerifyEmptyReliableQueue( byte keepMsgBelowThis, byte replaceWithThisMsg ) {
- if ( net_verifyReliableQueue.GetInteger() == 4 ) {
- RELIABLE_VERBOSE( "pushing a fake game reliable\n" );
- const char * garbage = "garbage";
- QueueReliableMessage( keepMsgBelowThis + 4, (const byte *)garbage, 8 );
- QueueReliableMessage( replaceWithThisMsg, NULL, 0 );
- }
- if ( reliable.Num() == 0 ) {
- return;
- }
- if ( net_verifyReliableQueue.GetInteger() == 1 ) {
- idLib::Error( "reliable queue is not empty: %d messages", reliable.Num() );
- return;
- }
- idLib::Warning( "reliable queue is not empty: %d messages", reliable.Num() );
- if ( net_verifyReliableQueue.GetInteger() == 0 ) {
- return;
- }
-
- idDataQueue< MAX_RELIABLE_QUEUE, MAX_MSG_SIZE > clean;
- RELIABLE_VERBOSE( "rollback send sequence from %d to %d\n", reliableSequenceSend, reliable.ItemSequence( 0 ) );
- for ( int i = 0; i < reliable.Num(); i++ ) {
- byte peek = reliable.ItemData( i )[0];
- if ( peek < keepMsgBelowThis ) {
- RELIABLE_VERBOSE( "keeping %d\n", peek );
- clean.Append( reliable.ItemSequence( i ), reliable.ItemData( i ), reliable.ItemLength( i ) );
- } else {
-
-
-
-
- clean.Append( reliable.ItemSequence( i ), &replaceWithThisMsg, 1 );
- RELIABLE_VERBOSE( "dropping %d\n", peek );
- }
- }
- assert( reliable.Num() == clean.Num() );
- reliable = clean;
- }
|