1571 lines
38 KiB
C++
1571 lines
38 KiB
C++
//========= Copyright Valve Corporation, All rights reserved. ============//
|
|
//
|
|
// Purpose:
|
|
//
|
|
// $NoKeywords: $
|
|
//=============================================================================//
|
|
|
|
#include "quakedef.h"
|
|
#include "dt.h"
|
|
#include "dt_encode.h"
|
|
#include "coordsize.h"
|
|
|
|
// memdbgon must be the last include file in a .cpp file!!!
|
|
#include "strtools.h"
|
|
#include "sysexternal.h"
|
|
#include "tier0/memdbgon.h"
|
|
|
|
extern void DataTable_Warning( const char *pInMessage, ... );
|
|
extern bool ShouldWatchThisProp( const SendTable *pTable, int objectID, const char *pPropName );
|
|
|
|
// The engine implements this.
|
|
extern const char* GetObjectClassName( int objectID );
|
|
|
|
// Check for special flags like SPROP_COORD, SPROP_NOSCALE, and SPROP_NORMAL.
|
|
// Returns true if it encoded the float.
|
|
static inline bool EncodeSpecialFloat( const SendProp *pProp, float fVal, bf_write *pOut )
|
|
{
|
|
int flags = pProp->GetFlags();
|
|
if ( flags & SPROP_COORD )
|
|
{
|
|
pOut->WriteBitCoord( fVal );
|
|
return true;
|
|
}
|
|
else if ( flags & SPROP_COORD_MP )
|
|
{
|
|
pOut->WriteBitCoordMP( fVal, false, false );
|
|
return true;
|
|
}
|
|
else if ( flags & SPROP_COORD_MP_LOWPRECISION )
|
|
{
|
|
pOut->WriteBitCoordMP( fVal, false, true );
|
|
return true;
|
|
}
|
|
else if ( flags & SPROP_COORD_MP_INTEGRAL )
|
|
{
|
|
pOut->WriteBitCoordMP( fVal, true, false );
|
|
return true;
|
|
}
|
|
else if ( flags & SPROP_NOSCALE )
|
|
{
|
|
pOut->WriteBitFloat( fVal );
|
|
return true;
|
|
}
|
|
else if ( flags & SPROP_NORMAL )
|
|
{
|
|
pOut->WriteBitNormal( fVal );
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// TODO_ENHANCED:
|
|
// Encoding works by sending a normalized float (from range 0.0 to 1.0)
|
|
// This works because we know the range we encode the float to
|
|
static inline void EncodeFloat( const SendProp* pProp, float fVal, bf_write* pOut, int objectID )
|
|
{
|
|
const auto WriteNormalizedFloat = [&]( double dblValue )
|
|
{
|
|
bool isPerfectOne = dblValue >= 1.0;
|
|
|
|
if ( isPerfectOne )
|
|
{
|
|
pOut->WriteOneBit( true );
|
|
}
|
|
else
|
|
{
|
|
pOut->WriteOneBit( false );
|
|
|
|
char strNumber[16];
|
|
V_memset( strNumber, 0, sizeof( strNumber ) );
|
|
V_sprintf_safe( strNumber, "%f", static_cast< float >( dblValue ) );
|
|
|
|
auto sixDigitsValue = V_atoi( &strNumber[2] );
|
|
|
|
if ( sixDigitsValue > 999999 )
|
|
{
|
|
Sys_Error( "EncodeFloat: tell to xutaxkamay that he's "
|
|
"dumb\n" );
|
|
}
|
|
|
|
pOut->WriteUBitLong( sixDigitsValue, 20 );
|
|
}
|
|
};
|
|
|
|
// Check for special flags like SPROP_COORD, SPROP_NOSCALE, and
|
|
// SPROP_NORMAL.
|
|
if ( EncodeSpecialFloat( pProp, fVal, pOut ) )
|
|
{
|
|
return;
|
|
}
|
|
|
|
if ( ( pProp->m_fHighValue == 0.0f && pProp->m_fLowValue == 0.0f ) || pProp->m_fHighValue == pProp->m_fLowValue )
|
|
{
|
|
pOut->WriteBitFloat( fVal );
|
|
return;
|
|
}
|
|
|
|
if ( fVal > pProp->m_fHighValue )
|
|
{
|
|
Sys_Error( "EncodeFloat: %s has value %f which is too big "
|
|
"(%f)",
|
|
pProp->GetName(),
|
|
fVal,
|
|
pProp->m_fHighValue );
|
|
}
|
|
|
|
if ( fVal < pProp->m_fLowValue )
|
|
{
|
|
Sys_Error( "EncodeFloat: %s has value %f which is too low "
|
|
"(%f)",
|
|
pProp->GetName(),
|
|
fVal,
|
|
pProp->m_fLowValue );
|
|
}
|
|
|
|
if ( pProp->m_fLowValue > pProp->m_fHighValue )
|
|
{
|
|
Sys_Error( "EncodeFloat: %s(%f) low value %f is higher than "
|
|
"%f",
|
|
pProp->GetName(),
|
|
fVal,
|
|
pProp->m_fLowValue,
|
|
pProp->m_fHighValue );
|
|
}
|
|
|
|
double dblVal = static_cast< double >( fVal );
|
|
|
|
// If low value bigger than zero, we need to substract to the
|
|
// lowest value possible, so we can recover it later.
|
|
if ( ( pProp->m_fLowValue > 0.0f && pProp->m_fHighValue > 0.0f )
|
|
|| ( pProp->m_fHighValue < 0.0f && pProp->m_fLowValue < 0.0f ) )
|
|
{
|
|
dblVal -= pProp->m_fLowValue;
|
|
}
|
|
|
|
double dblRange = pProp->m_fHighValue - pProp->m_fLowValue;
|
|
|
|
bool sign = false;
|
|
|
|
if ( dblVal < 0.0 )
|
|
{
|
|
sign = true;
|
|
dblVal = -dblVal;
|
|
}
|
|
|
|
pOut->WriteOneBit( sign );
|
|
|
|
WriteNormalizedFloat( dblRange > 1.0 ? ( dblVal / dblRange ) : dblVal );
|
|
}
|
|
|
|
// Look for special flags like SPROP_COORD, SPROP_NOSCALE, and SPROP_NORMAL and
|
|
// decode if they're there. Fills in fVal and returns true if it decodes anything.
|
|
static inline bool DecodeSpecialFloat( SendProp const *pProp, bf_read *pIn, float &fVal )
|
|
{
|
|
int flags = pProp->GetFlags();
|
|
|
|
if ( flags & SPROP_COORD )
|
|
{
|
|
fVal = pIn->ReadBitCoord();
|
|
return true;
|
|
}
|
|
else if ( flags & SPROP_COORD_MP )
|
|
{
|
|
fVal = pIn->ReadBitCoordMP( false, false );
|
|
return true;
|
|
}
|
|
else if ( flags & SPROP_COORD_MP_LOWPRECISION )
|
|
{
|
|
fVal = pIn->ReadBitCoordMP( false, true );
|
|
return true;
|
|
}
|
|
else if ( flags & SPROP_COORD_MP_INTEGRAL )
|
|
{
|
|
fVal = pIn->ReadBitCoordMP( true, false );
|
|
return true;
|
|
}
|
|
if ( flags & SPROP_NOSCALE )
|
|
{
|
|
fVal = pIn->ReadBitFloat();
|
|
return true;
|
|
}
|
|
else if ( flags & SPROP_NORMAL )
|
|
{
|
|
fVal = pIn->ReadBitNormal();
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
static inline float DecodeNormalizedFloat( const SendProp* pProp, bf_read* pIn )
|
|
{
|
|
bool sign = pIn->ReadOneBit();
|
|
double dblVal;
|
|
|
|
if ( !pIn->ReadOneBit() )
|
|
{
|
|
const auto fractionPart = pIn->ReadUBitLong( 20 );
|
|
|
|
char strNumber[8];
|
|
V_memset( strNumber, 0, sizeof( strNumber ) );
|
|
V_sprintf_safe( strNumber, "%i", fractionPart );
|
|
|
|
int countDigits = 0;
|
|
while ( strNumber[countDigits] )
|
|
{
|
|
countDigits++;
|
|
}
|
|
|
|
int digitLeft = 6 - countDigits;
|
|
|
|
if ( digitLeft < 0 )
|
|
{
|
|
Sys_Error( "DecodeNormalizedFloat: digitLeft < 0\n" );
|
|
}
|
|
|
|
char strFraction[16];
|
|
strFraction[0] = '0';
|
|
strFraction[1] = '.';
|
|
|
|
for ( int i = 0; i < digitLeft; i++ )
|
|
{
|
|
strFraction[2 + i] = '0';
|
|
}
|
|
|
|
for ( int i = 0; i < countDigits; i++ )
|
|
{
|
|
strFraction[2 + digitLeft + i] = strNumber[i];
|
|
}
|
|
|
|
dblVal = V_atof( strFraction );
|
|
}
|
|
else
|
|
{
|
|
dblVal = 1.0;
|
|
}
|
|
|
|
if ( dblVal <= 0.0 )
|
|
{
|
|
return 0.0f;
|
|
}
|
|
|
|
double dblRange = pProp->m_fHighValue - pProp->m_fLowValue;
|
|
|
|
if ( dblRange > 1.0 )
|
|
{
|
|
dblVal *= dblRange;
|
|
}
|
|
|
|
if ( ( pProp->m_fLowValue > 0.0f && pProp->m_fHighValue > 0.0f )
|
|
|| ( pProp->m_fHighValue < 0.0f && pProp->m_fLowValue < 0.0f ) )
|
|
{
|
|
dblVal += pProp->m_fLowValue;
|
|
}
|
|
|
|
dblVal = sign ? -dblVal : dblVal;
|
|
return static_cast< float >( dblVal );
|
|
}
|
|
|
|
static inline float DecodeFloat(const SendProp* pProp, bf_read* pIn)
|
|
{
|
|
float fVal;
|
|
|
|
// Check for special flags..
|
|
if (DecodeSpecialFloat(pProp, pIn, fVal))
|
|
{
|
|
return fVal;
|
|
}
|
|
|
|
if ((pProp->m_fHighValue == 0.0f && pProp->m_fLowValue == 0.0f)
|
|
|| pProp->m_fHighValue == pProp->m_fLowValue)
|
|
{
|
|
fVal = pIn->ReadBitFloat();
|
|
return fVal;
|
|
}
|
|
|
|
if (pProp->m_fLowValue > pProp->m_fHighValue)
|
|
{
|
|
Sys_Error("EncodeFloat: %s low value %f is higher than "
|
|
"%f",
|
|
pProp->GetName(),
|
|
pProp->m_fLowValue,
|
|
pProp->m_fHighValue);
|
|
}
|
|
|
|
fVal = DecodeNormalizedFloat(pProp, pIn);
|
|
|
|
return fVal;
|
|
}
|
|
|
|
static inline void DecodeVector(SendProp const *pProp, bf_read *pIn, float *v)
|
|
{
|
|
v[0] = DecodeFloat(pProp, pIn);
|
|
v[1] = DecodeFloat(pProp, pIn);
|
|
|
|
// Don't read in the third component for normals
|
|
if ((pProp->GetFlags() & SPROP_NORMAL) == 0)
|
|
{
|
|
v[2] = DecodeFloat(pProp, pIn);
|
|
}
|
|
else
|
|
{
|
|
int signbit = pIn->ReadOneBit();
|
|
|
|
float v0v0v1v1 = v[0] * v[0] +
|
|
v[1] * v[1];
|
|
if (v0v0v1v1 < 1.0f)
|
|
v[2] = sqrtf( 1.0f - v0v0v1v1 );
|
|
else
|
|
v[2] = 0.0f;
|
|
|
|
if (signbit)
|
|
v[2] *= -1.0f;
|
|
}
|
|
}
|
|
|
|
static inline void DecodeQuaternion(SendProp const *pProp, bf_read *pIn, float *v)
|
|
{
|
|
v[0] = DecodeFloat(pProp, pIn);
|
|
v[1] = DecodeFloat(pProp, pIn);
|
|
v[2] = DecodeFloat(pProp, pIn);
|
|
v[3] = DecodeFloat(pProp, pIn);
|
|
}
|
|
|
|
int DecodeBits( DecodeInfo *pInfo, unsigned char *pOut )
|
|
{
|
|
bf_read temp;
|
|
|
|
// Read the property in (note: we don't return the bits from here because Decode returns
|
|
// the decoded bits.. we're interested in getting the encoded bits).
|
|
temp = *pInfo->m_pIn;
|
|
pInfo->m_pRecvProp = NULL;
|
|
pInfo->m_pData = NULL;
|
|
g_PropTypeFns[pInfo->m_pProp->m_Type].Decode( pInfo );
|
|
|
|
// Return the encoded bits.
|
|
int nBits = pInfo->m_pIn->GetNumBitsRead() - temp.GetNumBitsRead();
|
|
temp.ReadBits(pOut, nBits);
|
|
return nBits;
|
|
}
|
|
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// Most of the prop types can use this generic FastCopy version. Arrays are a bit of a pain.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
inline void Generic_FastCopy(
|
|
const SendProp *pSendProp,
|
|
const RecvProp *pRecvProp,
|
|
const unsigned char *pSendData,
|
|
unsigned char *pRecvData,
|
|
int objectID )
|
|
{
|
|
// Get the data out of the ent.
|
|
CRecvProxyData recvProxyData;
|
|
|
|
pSendProp->GetProxyFn()(
|
|
pSendProp,
|
|
pSendData,
|
|
pSendData + pSendProp->GetOffset(),
|
|
&recvProxyData.m_Value,
|
|
0,
|
|
objectID
|
|
);
|
|
|
|
// Fill in the data for the recv proxy.
|
|
recvProxyData.m_pRecvProp = pRecvProp;
|
|
recvProxyData.m_iElement = 0;
|
|
recvProxyData.m_ObjectID = objectID;
|
|
pRecvProp->GetProxyFn()( &recvProxyData, pRecvData, pRecvData + pRecvProp->GetOffset() );
|
|
}
|
|
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// DecodeInfo implementation.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
void DecodeInfo::CopyVars( const DecodeInfo *pOther )
|
|
{
|
|
m_pStruct = pOther->m_pStruct;
|
|
m_pData = pOther->m_pData;
|
|
|
|
m_pRecvProp = pOther->m_pRecvProp;
|
|
m_pProp = pOther->m_pProp;
|
|
m_pIn = pOther->m_pIn;
|
|
m_ObjectID = pOther->m_ObjectID;
|
|
m_iElement = pOther->m_iElement;
|
|
}
|
|
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// Int property type abstraction.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
void Int_Encode( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp, bf_write *pOut, int objectID )
|
|
{
|
|
int nValue = pVar->m_Int;
|
|
|
|
if ( pProp->GetFlags() & SPROP_VARINT)
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_UNSIGNED )
|
|
{
|
|
pOut->WriteVarInt32( nValue );
|
|
}
|
|
else
|
|
{
|
|
pOut->WriteSignedVarInt32( nValue );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// If signed, preserve lower bits and then re-extend sign if nValue < 0;
|
|
// if unsigned, preserve all 32 bits no matter what. Bonus: branchless.
|
|
int nPreserveBits = ( 0x7FFFFFFF >> ( 32 - pProp->m_nBits ) );
|
|
nPreserveBits |= ( pProp->GetFlags() & SPROP_UNSIGNED ) ? 0xFFFFFFFF : 0;
|
|
int nSignExtension = ( nValue >> 31 ) & ~nPreserveBits;
|
|
|
|
nValue &= nPreserveBits;
|
|
nValue |= nSignExtension;
|
|
|
|
#ifdef DBGFLAG_ASSERT
|
|
// Assert that either the property is unsigned and in valid range,
|
|
// or signed with a consistent sign extension in the high bits
|
|
if ( pProp->m_nBits < 32 )
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_UNSIGNED )
|
|
{
|
|
AssertMsg3( nValue == pVar->m_Int, "Unsigned prop %s needs more bits? Expected %i == %i", pProp->GetName(), nValue, pVar->m_Int );
|
|
}
|
|
else
|
|
{
|
|
AssertMsg3( nValue == pVar->m_Int, "Signed prop %s needs more bits? Expected %i == %i", pProp->GetName(), nValue, pVar->m_Int );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// This should never trigger, but I'm leaving it in for old-time's sake.
|
|
Assert( nValue == pVar->m_Int );
|
|
}
|
|
#endif
|
|
|
|
pOut->WriteUBitLong( nValue, pProp->m_nBits, false );
|
|
}
|
|
}
|
|
|
|
|
|
void Int_Decode( DecodeInfo *pInfo )
|
|
{
|
|
const SendProp *pProp = pInfo->m_pProp;
|
|
int flags = pProp->GetFlags();
|
|
|
|
if ( flags & SPROP_VARINT )
|
|
{
|
|
if ( flags & SPROP_UNSIGNED )
|
|
{
|
|
pInfo->m_Value.m_Int = (int)pInfo->m_pIn->ReadVarInt32();
|
|
}
|
|
else
|
|
{
|
|
pInfo->m_Value.m_Int = pInfo->m_pIn->ReadSignedVarInt32();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int bits = pProp->m_nBits;
|
|
pInfo->m_Value.m_Int = pInfo->m_pIn->ReadUBitLong(bits);
|
|
|
|
if( bits != 32 && (flags & SPROP_UNSIGNED) == 0 )
|
|
{
|
|
unsigned int highbit = 1ul << (pProp->m_nBits - 1);
|
|
if ( pInfo->m_Value.m_Int & highbit )
|
|
{
|
|
pInfo->m_Value.m_Int -= highbit; // strip high bit...
|
|
pInfo->m_Value.m_Int -= highbit; // ... then put it back with sign extension
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( pInfo->m_pRecvProp )
|
|
{
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
}
|
|
|
|
|
|
int Int_CompareDeltas( const SendProp *pProp, bf_read *p1, bf_read *p2 )
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_VARINT)
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_UNSIGNED )
|
|
{
|
|
return p1->ReadVarInt32() != p2->ReadVarInt32();
|
|
}
|
|
return p1->ReadSignedVarInt32() != p2->ReadSignedVarInt32();
|
|
}
|
|
|
|
return p1->CompareBits(p2, pProp->m_nBits);
|
|
}
|
|
|
|
const char* Int_GetTypeNameString()
|
|
{
|
|
return "DPT_Int";
|
|
}
|
|
|
|
|
|
bool Int_IsZero( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp )
|
|
{
|
|
return (pVar->m_Int == 0);
|
|
}
|
|
|
|
|
|
void Int_DecodeZero( DecodeInfo *pInfo )
|
|
{
|
|
pInfo->m_Value.m_Int = 0;
|
|
|
|
if ( pInfo->m_pRecvProp )
|
|
{
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
}
|
|
|
|
bool Int_IsEncodedZero( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_VARINT)
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_UNSIGNED )
|
|
{
|
|
return pIn->ReadVarInt32() == 0;
|
|
}
|
|
return pIn->ReadSignedVarInt32() == 0;
|
|
}
|
|
return pIn->ReadUBitLong( pProp->m_nBits ) == 0;
|
|
}
|
|
|
|
void Int_SkipProp( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_VARINT)
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_UNSIGNED )
|
|
{
|
|
pIn->ReadVarInt32();
|
|
}
|
|
else
|
|
{
|
|
pIn->ReadSignedVarInt32();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pIn->SeekRelative( pProp->m_nBits );
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// Float type abstraction.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
void Float_Encode( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp, bf_write *pOut, int objectID )
|
|
{
|
|
EncodeFloat( pProp, pVar->m_Float, pOut, objectID );
|
|
}
|
|
|
|
|
|
void Float_Decode( DecodeInfo *pInfo )
|
|
{
|
|
pInfo->m_Value.m_Float = DecodeFloat(pInfo->m_pProp, pInfo->m_pIn);
|
|
|
|
if ( pInfo->m_pRecvProp )
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
|
|
|
|
int Float_CompareDeltas( const SendProp *pProp, bf_read *p1, bf_read *p2 )
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_COORD )
|
|
{
|
|
return p1->ReadBitCoord() != p2->ReadBitCoord();
|
|
}
|
|
else if ( pProp->GetFlags() & SPROP_COORD_MP )
|
|
{
|
|
return p1->ReadBitCoordMP( false, false ) != p2->ReadBitCoordMP( false, false );
|
|
}
|
|
else if ( pProp->GetFlags() & SPROP_COORD_MP_LOWPRECISION )
|
|
{
|
|
return p1->ReadBitCoordMP( false, true ) != p2->ReadBitCoordMP( false, true );
|
|
}
|
|
else if ( pProp->GetFlags() & SPROP_COORD_MP_INTEGRAL )
|
|
{
|
|
return p1->ReadBitCoordMP( true, false ) != p2->ReadBitCoordMP( true, false );
|
|
}
|
|
else if ( pProp->GetFlags() & SPROP_NOSCALE )
|
|
{
|
|
return p1->ReadUBitLong( 32 ) != p2->ReadUBitLong( 32 );
|
|
}
|
|
else if ( pProp->GetFlags() & SPROP_NORMAL )
|
|
{
|
|
return p1->ReadUBitLong( NORMAL_FRACTIONAL_BITS+1 ) != p2->ReadUBitLong( NORMAL_FRACTIONAL_BITS+1 );
|
|
}
|
|
else
|
|
{
|
|
return DecodeNormalizedFloat(pProp, p1) != DecodeNormalizedFloat(pProp, p2);
|
|
}
|
|
}
|
|
|
|
const char* Float_GetTypeNameString()
|
|
{
|
|
return "DPT_Float";
|
|
}
|
|
|
|
|
|
bool Float_IsZero( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp )
|
|
{
|
|
return (pVar->m_Float == 0);
|
|
}
|
|
|
|
|
|
void Float_DecodeZero( DecodeInfo *pInfo )
|
|
{
|
|
pInfo->m_Value.m_Float = 0;
|
|
if ( pInfo->m_pRecvProp )
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
|
|
bool Float_IsEncodedZero( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
return DecodeFloat( pProp, pIn ) == 0.0f;
|
|
}
|
|
|
|
void Float_SkipProp( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
// Check for special flags..
|
|
if(pProp->GetFlags() & SPROP_COORD)
|
|
{
|
|
// Read the required integer and fraction flags
|
|
unsigned int val = pIn->ReadUBitLong(2);
|
|
// this reads two bits, the first bit (bit0 in this word) indicates integer part
|
|
// the second bit (bit1 in this word) indicates the fractional part
|
|
|
|
// If we got either parse them, otherwise it's a zero.
|
|
if ( val )
|
|
{
|
|
// sign bit
|
|
int seekDist = 1;
|
|
|
|
// If there's an integer, read it in
|
|
if ( val & 1 )
|
|
seekDist += COORD_INTEGER_BITS;
|
|
|
|
if ( val & 2 )
|
|
seekDist += COORD_FRACTIONAL_BITS;
|
|
|
|
pIn->SeekRelative( seekDist );
|
|
}
|
|
}
|
|
else if ( pProp->GetFlags() & SPROP_COORD_MP )
|
|
{
|
|
pIn->ReadBitCoordMP( false, false );
|
|
}
|
|
else if ( pProp->GetFlags() & SPROP_COORD_MP_LOWPRECISION )
|
|
{
|
|
pIn->ReadBitCoordMP( false, true );
|
|
}
|
|
else if ( pProp->GetFlags() & SPROP_COORD_MP_INTEGRAL )
|
|
{
|
|
pIn->ReadBitCoordMP( true, false );
|
|
}
|
|
else if(pProp->GetFlags() & SPROP_NOSCALE)
|
|
{
|
|
pIn->SeekRelative( 32 );
|
|
}
|
|
else if(pProp->GetFlags() & SPROP_NORMAL)
|
|
{
|
|
pIn->SeekRelative( NORMAL_FRACTIONAL_BITS + 1 );
|
|
}
|
|
else
|
|
{
|
|
DecodeNormalizedFloat(pProp, pIn);
|
|
}
|
|
}
|
|
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// Vector type abstraction.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
void Vector_Encode( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp, bf_write *pOut, int objectID )
|
|
{
|
|
EncodeFloat(pProp, pVar->m_Vector[0], pOut, objectID);
|
|
EncodeFloat(pProp, pVar->m_Vector[1], pOut, objectID);
|
|
// Don't write out the third component for normals
|
|
if ((pProp->GetFlags() & SPROP_NORMAL) == 0)
|
|
{
|
|
EncodeFloat(pProp, pVar->m_Vector[2], pOut, objectID);
|
|
}
|
|
else
|
|
{
|
|
// Write a sign bit for z instead!
|
|
int signbit = (pVar->m_Vector[2] <= -NORMAL_RESOLUTION);
|
|
pOut->WriteOneBit( signbit );
|
|
}
|
|
}
|
|
|
|
|
|
void Vector_Decode(DecodeInfo *pInfo)
|
|
{
|
|
DecodeVector( pInfo->m_pProp, pInfo->m_pIn, pInfo->m_Value.m_Vector );
|
|
|
|
if( pInfo->m_pRecvProp )
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
|
|
|
|
int Vector_CompareDeltas( const SendProp *pProp, bf_read *p1, bf_read *p2 )
|
|
{
|
|
int c1 = Float_CompareDeltas( pProp, p1, p2 );
|
|
int c2 = Float_CompareDeltas( pProp, p1, p2 );
|
|
int c3;
|
|
|
|
if ( pProp->GetFlags() & SPROP_NORMAL )
|
|
{
|
|
c3 = p1->ReadOneBit() != p2->ReadOneBit();
|
|
}
|
|
else
|
|
{
|
|
c3 = Float_CompareDeltas( pProp, p1, p2 );
|
|
}
|
|
|
|
return c1 | c2 | c3;
|
|
}
|
|
|
|
const char* Vector_GetTypeNameString()
|
|
{
|
|
return "DPT_Vector";
|
|
}
|
|
|
|
|
|
bool Vector_IsZero( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp )
|
|
{
|
|
return ( pVar->m_Vector[0] == 0 ) && ( pVar->m_Vector[1] == 0 ) && ( pVar->m_Vector[2] == 0 );
|
|
}
|
|
|
|
|
|
void Vector_DecodeZero( DecodeInfo *pInfo )
|
|
{
|
|
pInfo->m_Value.m_Vector[0] = 0;
|
|
pInfo->m_Value.m_Vector[1] = 0;
|
|
pInfo->m_Value.m_Vector[2] = 0;
|
|
if ( pInfo->m_pRecvProp )
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
|
|
bool Vector_IsEncodedZero( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
float v[3];
|
|
|
|
DecodeVector( pProp, pIn, v );
|
|
|
|
return ( v[0] == 0 ) && ( v[1] == 0 ) && ( v[2] == 0 );
|
|
}
|
|
|
|
void Vector_SkipProp( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
Float_SkipProp(pProp, pIn);
|
|
Float_SkipProp(pProp, pIn);
|
|
|
|
// Don't read in the third component for normals
|
|
if ( pProp->GetFlags() & SPROP_NORMAL )
|
|
{
|
|
pIn->SeekRelative( 1 );
|
|
}
|
|
else
|
|
{
|
|
Float_SkipProp(pProp, pIn);
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// VectorXY type abstraction.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
void VectorXY_Encode( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp, bf_write *pOut, int objectID )
|
|
{
|
|
EncodeFloat(pProp, pVar->m_Vector[0], pOut, objectID);
|
|
EncodeFloat(pProp, pVar->m_Vector[1], pOut, objectID);
|
|
}
|
|
|
|
|
|
void VectorXY_Decode(DecodeInfo *pInfo)
|
|
{
|
|
pInfo->m_Value.m_Vector[0] = DecodeFloat(pInfo->m_pProp, pInfo->m_pIn);
|
|
pInfo->m_Value.m_Vector[1] = DecodeFloat(pInfo->m_pProp, pInfo->m_pIn);
|
|
|
|
if( pInfo->m_pRecvProp )
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
|
|
|
|
int VectorXY_CompareDeltas( const SendProp *pProp, bf_read *p1, bf_read *p2 )
|
|
{
|
|
int c1 = Float_CompareDeltas( pProp, p1, p2 );
|
|
int c2 = Float_CompareDeltas( pProp, p1, p2 );
|
|
|
|
return c1 | c2;
|
|
}
|
|
|
|
const char* VectorXY_GetTypeNameString()
|
|
{
|
|
return "DPT_VectorXY";
|
|
}
|
|
|
|
|
|
bool VectorXY_IsZero( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp )
|
|
{
|
|
return ( pVar->m_Vector[0] == 0 ) && ( pVar->m_Vector[1] == 0 );
|
|
}
|
|
|
|
|
|
void VectorXY_DecodeZero( DecodeInfo *pInfo )
|
|
{
|
|
pInfo->m_Value.m_Vector[0] = 0;
|
|
pInfo->m_Value.m_Vector[1] = 0;
|
|
if ( pInfo->m_pRecvProp )
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
|
|
bool VectorXY_IsEncodedZero( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
float v[2];
|
|
|
|
v[0] = DecodeFloat(pProp, pIn);
|
|
v[1] = DecodeFloat(pProp, pIn);
|
|
|
|
return ( v[0] == 0 ) && ( v[1] == 0 );
|
|
}
|
|
|
|
void VectorXY_SkipProp( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
Float_SkipProp(pProp, pIn);
|
|
Float_SkipProp(pProp, pIn);
|
|
}
|
|
|
|
#if 0 // We can't ship this since it changes the size of DTVariant to be 20 bytes instead of 16 and that breaks MODs!!!
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// Quaternion type abstraction.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
void Quaternion_Encode( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp, bf_write *pOut, int objectID )
|
|
{
|
|
EncodeFloat(pProp, pVar->m_Vector[0], pOut, objectID);
|
|
EncodeFloat(pProp, pVar->m_Vector[1], pOut, objectID);
|
|
EncodeFloat(pProp, pVar->m_Vector[2], pOut, objectID);
|
|
EncodeFloat(pProp, pVar->m_Vector[3], pOut, objectID);
|
|
}
|
|
|
|
|
|
void Quaternion_Decode(DecodeInfo *pInfo)
|
|
{
|
|
DecodeQuaternion( pInfo->m_pProp, pInfo->m_pIn, pInfo->m_Value.m_Vector );
|
|
|
|
if( pInfo->m_pRecvProp )
|
|
{
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
}
|
|
|
|
|
|
int Quaternion_CompareDeltas( const SendProp *pProp, bf_read *p1, bf_read *p2 )
|
|
{
|
|
int c1 = Float_CompareDeltas( pProp, p1, p2 );
|
|
int c2 = Float_CompareDeltas( pProp, p1, p2 );
|
|
int c3 = Float_CompareDeltas( pProp, p1, p2 );
|
|
int c4 = Float_CompareDeltas( pProp, p1, p2 );
|
|
return c1 | c2 | c3 | c4;
|
|
}
|
|
|
|
const char* Quaternion_GetTypeNameString()
|
|
{
|
|
return "DPT_Quaternion";
|
|
}
|
|
|
|
|
|
bool Quaternion_IsZero( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp )
|
|
{
|
|
return ( pVar->m_Vector[0] == 0 ) && ( pVar->m_Vector[1] == 0 ) && ( pVar->m_Vector[2] == 0 ) && ( pVar->m_Vector[3] == 0 );
|
|
}
|
|
|
|
|
|
void Quaternion_DecodeZero( DecodeInfo *pInfo )
|
|
{
|
|
pInfo->m_Value.m_Vector[0] = 0;
|
|
pInfo->m_Value.m_Vector[1] = 0;
|
|
pInfo->m_Value.m_Vector[2] = 0;
|
|
pInfo->m_Value.m_Vector[3] = 0;
|
|
if ( pInfo->m_pRecvProp )
|
|
{
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
}
|
|
|
|
bool Quaternion_IsEncodedZero( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
float v[4];
|
|
|
|
DecodeQuaternion( pProp, pIn, v );
|
|
|
|
return ( v[0] == 0 ) && ( v[1] == 0 ) && ( v[2] == 0 ) && ( v[3] == 0 );
|
|
}
|
|
|
|
void Quaternion_SkipProp( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
Float_SkipProp(pProp, pIn);
|
|
Float_SkipProp(pProp, pIn);
|
|
Float_SkipProp(pProp, pIn);
|
|
Float_SkipProp(pProp, pIn);
|
|
}
|
|
#endif
|
|
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// String type abstraction.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
void String_Encode( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp, bf_write *pOut, int objectID )
|
|
{
|
|
// First count the string length, then do one WriteBits call.
|
|
int len;
|
|
for ( len=0; len < DT_MAX_STRING_BUFFERSIZE-1; len++ )
|
|
{
|
|
if( pVar->m_pString[len] == 0 )
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Optionally write the length here so deltas can be compared faster.
|
|
pOut->WriteUBitLong( len, DT_MAX_STRING_BITS );
|
|
pOut->WriteBits( pVar->m_pString, len * 8 );
|
|
}
|
|
|
|
|
|
void String_Decode(DecodeInfo *pInfo)
|
|
{
|
|
// Read it in.
|
|
int len = pInfo->m_pIn->ReadUBitLong( DT_MAX_STRING_BITS );
|
|
|
|
char *tempStr = pInfo->m_TempStr;
|
|
|
|
if ( len >= DT_MAX_STRING_BUFFERSIZE )
|
|
{
|
|
Warning( "String_Decode( %s ) invalid length (%d)\n", pInfo->m_pRecvProp->GetName(), len );
|
|
len = DT_MAX_STRING_BUFFERSIZE - 1;
|
|
}
|
|
|
|
pInfo->m_pIn->ReadBits( tempStr, len*8 );
|
|
tempStr[len] = 0;
|
|
|
|
pInfo->m_Value.m_pString = tempStr;
|
|
|
|
// Give it to the RecvProxy.
|
|
if ( pInfo->m_pRecvProp )
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
|
|
|
|
// Compare the bits in pBuf1 and pBuf2 and return 1 if they are different.
|
|
// This must always seek both buffers to wherever they start at + nBits.
|
|
static inline int AreBitsDifferent( bf_read *pBuf1, bf_read *pBuf2, int nBits )
|
|
{
|
|
int nDWords = nBits >> 5;
|
|
|
|
int diff = 0;
|
|
for ( int iDWord=0; iDWord < nDWords; iDWord++ )
|
|
{
|
|
diff |= (pBuf1->ReadUBitLong(32) != pBuf2->ReadUBitLong(32));
|
|
}
|
|
|
|
int nRemainingBits = nBits - (nDWords<<5);
|
|
if (nRemainingBits > 0)
|
|
diff |= pBuf1->ReadUBitLong( nRemainingBits ) != pBuf2->ReadUBitLong( nRemainingBits );
|
|
|
|
return diff;
|
|
}
|
|
|
|
|
|
int String_CompareDeltas( const SendProp *pProp, bf_read *p1, bf_read *p2 )
|
|
{
|
|
int len1 = p1->ReadUBitLong( DT_MAX_STRING_BITS );
|
|
int len2 = p2->ReadUBitLong( DT_MAX_STRING_BITS );
|
|
|
|
if ( len1 == len2 )
|
|
{
|
|
// check if both strings are empty
|
|
if (len1 == 0)
|
|
return false;
|
|
|
|
// Ok, they're short and fast.
|
|
return AreBitsDifferent( p1, p2, len1*8 );
|
|
}
|
|
else
|
|
{
|
|
p1->SeekRelative( len1 * 8 );
|
|
p2->SeekRelative( len2 * 8 );
|
|
return true;
|
|
}
|
|
}
|
|
|
|
const char* String_GetTypeNameString()
|
|
{
|
|
return "DPT_String";
|
|
}
|
|
|
|
|
|
bool String_IsZero( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp )
|
|
{
|
|
return ( pVar->m_pString[0] == 0 );
|
|
}
|
|
|
|
|
|
void String_DecodeZero( DecodeInfo *pInfo )
|
|
{
|
|
pInfo->m_Value.m_pString = pInfo->m_TempStr;
|
|
pInfo->m_TempStr[0] = 0;
|
|
if ( pInfo->m_pRecvProp )
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
|
|
bool String_IsEncodedZero( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
// Read it in.
|
|
int len = pIn->ReadUBitLong( DT_MAX_STRING_BITS );
|
|
|
|
pIn->SeekRelative( len*8 );
|
|
|
|
return len == 0;
|
|
}
|
|
|
|
void String_SkipProp( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
int len = pIn->ReadUBitLong( DT_MAX_STRING_BITS );
|
|
pIn->SeekRelative( len*8 );
|
|
}
|
|
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// Array abstraction.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
int Array_GetLength( const unsigned char *pStruct, const SendProp *pProp, int objectID )
|
|
{
|
|
// Get the array length from the proxy.
|
|
ArrayLengthSendProxyFn proxy = pProp->GetArrayLengthProxy();
|
|
|
|
if ( proxy )
|
|
{
|
|
int nElements = proxy( pStruct, objectID );
|
|
|
|
// Make sure it's not too big.
|
|
if ( nElements > pProp->GetNumElements() )
|
|
{
|
|
Assert( false );
|
|
nElements = pProp->GetNumElements();
|
|
}
|
|
|
|
return nElements;
|
|
}
|
|
else
|
|
{
|
|
return pProp->GetNumElements();
|
|
}
|
|
}
|
|
|
|
|
|
void Array_Encode( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp, bf_write *pOut, int objectID )
|
|
{
|
|
SendProp *pArrayProp = pProp->GetArrayProp();
|
|
AssertMsg( pArrayProp, "Array_Encode: missing m_pArrayProp for SendProp '%s'.", pProp->m_pVarName );
|
|
|
|
int nElements = Array_GetLength( pStruct, pProp, objectID );
|
|
|
|
// Write the number of elements.
|
|
pOut->WriteUBitLong( nElements, pProp->GetNumArrayLengthBits() );
|
|
|
|
unsigned char *pCurStructOffset = (unsigned char*)pStruct + pArrayProp->GetOffset();
|
|
for ( int iElement=0; iElement < nElements; iElement++ )
|
|
{
|
|
DVariant var;
|
|
|
|
// Call the proxy to get the value, then encode.
|
|
pArrayProp->GetProxyFn()( pArrayProp, pStruct, pCurStructOffset, &var, iElement, objectID );
|
|
g_PropTypeFns[pArrayProp->GetType()].Encode( pStruct, &var, pArrayProp, pOut, objectID );
|
|
|
|
pCurStructOffset += pProp->GetElementStride();
|
|
}
|
|
}
|
|
|
|
|
|
void Array_Decode( DecodeInfo *pInfo )
|
|
{
|
|
SendProp *pArrayProp = pInfo->m_pProp->GetArrayProp();
|
|
AssertMsg( pArrayProp, ("Array_Decode: missing m_pArrayProp for a property.") );
|
|
|
|
// Setup a DecodeInfo that is used to decode each of the child properties.
|
|
DecodeInfo subDecodeInfo;
|
|
subDecodeInfo.CopyVars( pInfo );
|
|
subDecodeInfo.m_pProp = pArrayProp;
|
|
|
|
int elementStride = 0;
|
|
ArrayLengthRecvProxyFn lengthProxy = 0;
|
|
if ( pInfo->m_pRecvProp )
|
|
{
|
|
RecvProp *pArrayRecvProp = pInfo->m_pRecvProp->GetArrayProp();
|
|
subDecodeInfo.m_pRecvProp = pArrayRecvProp;
|
|
|
|
// Note we get the OFFSET from the array element property and the STRIDE from the array itself.
|
|
subDecodeInfo.m_pData = (char*)pInfo->m_pData + pArrayRecvProp->GetOffset();
|
|
elementStride = pInfo->m_pRecvProp->GetElementStride();
|
|
Assert( elementStride != -1 ); // (Make sure it was set..)
|
|
|
|
lengthProxy = pInfo->m_pRecvProp->GetArrayLengthProxy();
|
|
}
|
|
|
|
int nElements = pInfo->m_pIn->ReadUBitLong( pInfo->m_pProp->GetNumArrayLengthBits() );
|
|
|
|
if ( lengthProxy )
|
|
lengthProxy( pInfo->m_pStruct, pInfo->m_ObjectID, nElements );
|
|
|
|
for ( subDecodeInfo.m_iElement=0; subDecodeInfo.m_iElement < nElements; subDecodeInfo.m_iElement++ )
|
|
{
|
|
g_PropTypeFns[pArrayProp->GetType()].Decode( &subDecodeInfo );
|
|
subDecodeInfo.m_pData = (char*)subDecodeInfo.m_pData + elementStride;
|
|
}
|
|
}
|
|
|
|
|
|
int Array_CompareDeltas( const SendProp *pProp, bf_read *p1, bf_read *p2 )
|
|
{
|
|
SendProp *pArrayProp = pProp->GetArrayProp();
|
|
AssertMsg( pArrayProp, "Array_CompareDeltas: missing m_pArrayProp for SendProp '%s'.", pProp->m_pVarName );
|
|
|
|
int nLengthBits = pProp->GetNumArrayLengthBits();
|
|
int length1 = p1->ReadUBitLong( nLengthBits );
|
|
int length2 = p2->ReadUBitLong( nLengthBits );
|
|
|
|
int bDifferent = length1 != length2;
|
|
|
|
// Compare deltas on the props that are the same.
|
|
int nSame = min( length1, length2 );
|
|
for ( int iElement=0; iElement < nSame; iElement++ )
|
|
{
|
|
bDifferent |= g_PropTypeFns[pArrayProp->GetType()].CompareDeltas( pArrayProp, p1, p2 );
|
|
}
|
|
|
|
// Now just eat up the remaining properties in whichever buffer was larger.
|
|
if ( length1 != length2 )
|
|
{
|
|
bf_read *buffer = (length1 > length2) ? p1 : p2;
|
|
|
|
int nExtra = max( length1, length2 ) - nSame;
|
|
for ( int iEatUp=0; iEatUp < nExtra; iEatUp++ )
|
|
{
|
|
SkipPropData( buffer, pArrayProp );
|
|
}
|
|
}
|
|
|
|
return bDifferent;
|
|
}
|
|
|
|
|
|
void Array_FastCopy(
|
|
const SendProp *pSendProp,
|
|
const RecvProp *pRecvProp,
|
|
const unsigned char *pSendData,
|
|
unsigned char *pRecvData,
|
|
int objectID )
|
|
{
|
|
const RecvProp *pArrayRecvProp = pRecvProp->GetArrayProp();
|
|
const SendProp *pArraySendProp = pSendProp->GetArrayProp();
|
|
|
|
CRecvProxyData recvProxyData;
|
|
recvProxyData.m_pRecvProp = pArrayRecvProp;
|
|
recvProxyData.m_ObjectID = objectID;
|
|
|
|
// Find out the array length and call the RecvProp's array-length proxy.
|
|
int nElements = Array_GetLength( pSendData, pSendProp, objectID );
|
|
ArrayLengthRecvProxyFn lengthProxy = pRecvProp->GetArrayLengthProxy();
|
|
if ( lengthProxy )
|
|
lengthProxy( pRecvData, objectID, nElements );
|
|
|
|
const unsigned char *pCurSendPos = pSendData + pArraySendProp->GetOffset();
|
|
unsigned char *pCurRecvPos = pRecvData + pArrayRecvProp->GetOffset();
|
|
for ( recvProxyData.m_iElement=0; recvProxyData.m_iElement < nElements; recvProxyData.m_iElement++ )
|
|
{
|
|
// Get this array element out of the sender's data.
|
|
pArraySendProp->GetProxyFn()( pArraySendProp, pSendData, pCurSendPos, &recvProxyData.m_Value, recvProxyData.m_iElement, objectID );
|
|
pCurSendPos += pSendProp->GetElementStride();
|
|
|
|
// Write it into the receiver.
|
|
pArrayRecvProp->GetProxyFn()( &recvProxyData, pRecvData, pCurRecvPos );
|
|
pCurRecvPos += pRecvProp->GetElementStride();
|
|
}
|
|
}
|
|
|
|
const char* Array_GetTypeNameString()
|
|
{
|
|
return "DPT_Array";
|
|
}
|
|
|
|
|
|
bool Array_IsZero( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp )
|
|
{
|
|
int nElements = Array_GetLength( pStruct, pProp, -1 );
|
|
return ( nElements == 0 );
|
|
}
|
|
|
|
|
|
void Array_DecodeZero( DecodeInfo *pInfo )
|
|
{
|
|
ArrayLengthRecvProxyFn lengthProxy = pInfo->m_pRecvProp->GetArrayLengthProxy();
|
|
if ( lengthProxy )
|
|
lengthProxy( pInfo->m_pStruct, pInfo->m_ObjectID, 0 );
|
|
}
|
|
|
|
bool Array_IsEncodedZero( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
SendProp *pArrayProp = pProp->GetArrayProp();
|
|
AssertMsg( pArrayProp, ("Array_IsEncodedZero: missing m_pArrayProp for a property.") );
|
|
|
|
int nElements = pIn->ReadUBitLong( pProp->GetNumArrayLengthBits() );
|
|
|
|
for ( int i=0; i < nElements; i++ )
|
|
{
|
|
// skip over data
|
|
g_PropTypeFns[pArrayProp->GetType()].IsEncodedZero( pArrayProp, pIn );
|
|
}
|
|
|
|
return nElements == 0;;
|
|
}
|
|
|
|
void Array_SkipProp( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
SendProp *pArrayProp = pProp->GetArrayProp();
|
|
AssertMsg( pArrayProp, ("Array_SkipProp: missing m_pArrayProp for a property.") );
|
|
|
|
int nElements = pIn->ReadUBitLong( pProp->GetNumArrayLengthBits() );
|
|
|
|
for ( int i=0; i < nElements; i++ )
|
|
{
|
|
// skip over data
|
|
g_PropTypeFns[pArrayProp->GetType()].SkipProp( pArrayProp, pIn );
|
|
}
|
|
}
|
|
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// Datatable type abstraction.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
const char* DataTable_GetTypeNameString()
|
|
{
|
|
return "DPT_DataTable";
|
|
}
|
|
|
|
|
|
// ---------------------------------------------------------------------------------------- //
|
|
// Int 64 property type abstraction.
|
|
// ---------------------------------------------------------------------------------------- //
|
|
|
|
void Int64_Encode( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp, bf_write *pOut, int objectID )
|
|
{
|
|
#ifdef SUPPORTS_INT64
|
|
if ( pProp->GetFlags() & SPROP_VARINT)
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_UNSIGNED )
|
|
{
|
|
pOut->WriteVarInt64( pVar->m_Int64 );
|
|
}
|
|
else
|
|
{
|
|
pOut->WriteSignedVarInt32( pVar->m_Int64 );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
bool bNeg = pVar->m_Int64 < 0;
|
|
int64 iCopy = bNeg ? -pVar->m_Int64 : pVar->m_Int64;
|
|
uint32 *pInt = (uint32*)&iCopy;
|
|
uint32 lowInt = *pInt++;
|
|
uint32 highInt = *pInt;
|
|
if( pProp->IsSigned() )
|
|
{
|
|
pOut->WriteOneBit( bNeg );
|
|
pOut->WriteUBitLong( (unsigned int)lowInt, 32 );
|
|
pOut->WriteUBitLong( (unsigned int)highInt, pProp->m_nBits - 32 - 1 ); // For the sign bit
|
|
}
|
|
else
|
|
{
|
|
pOut->WriteUBitLong( (unsigned int)lowInt, 32 );
|
|
pOut->WriteUBitLong( (unsigned int)highInt, pProp->m_nBits - 32 );
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
void Int64_Decode( DecodeInfo *pInfo )
|
|
{
|
|
#ifdef SUPPORTS_INT64
|
|
if ( pInfo->m_pProp->GetFlags() & SPROP_VARINT )
|
|
{
|
|
if ( pInfo->m_pProp->GetFlags() & SPROP_UNSIGNED )
|
|
{
|
|
pInfo->m_Value.m_Int64 = (int64)pInfo->m_pIn->ReadVarInt64();
|
|
}
|
|
else
|
|
{
|
|
pInfo->m_Value.m_Int64 = pInfo->m_pIn->ReadSignedVarInt64();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
uint32 highInt = 0;
|
|
uint32 lowInt = 0;
|
|
bool bNeg = false;
|
|
if(pInfo->m_pProp->IsSigned())
|
|
{
|
|
bNeg = pInfo->m_pIn->ReadOneBit() != 0;
|
|
lowInt = pInfo->m_pIn->ReadUBitLong( 32 );
|
|
highInt = pInfo->m_pIn->ReadUBitLong( pInfo->m_pProp->m_nBits - 32 - 1 );
|
|
}
|
|
else
|
|
{
|
|
lowInt = pInfo->m_pIn->ReadUBitLong( 32 );
|
|
highInt = pInfo->m_pIn->ReadUBitLong( pInfo->m_pProp->m_nBits - 32 );
|
|
}
|
|
|
|
uint32 *pInt = (uint32*)&pInfo->m_Value.m_Int64;
|
|
*pInt++ = lowInt;
|
|
*pInt = highInt;
|
|
|
|
if ( bNeg )
|
|
{
|
|
pInfo->m_Value.m_Int64 = -pInfo->m_Value.m_Int64;
|
|
}
|
|
}
|
|
|
|
if ( pInfo->m_pRecvProp )
|
|
{
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
int Int64_CompareDeltas( const SendProp *pProp, bf_read *p1, bf_read *p2 )
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_VARINT)
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_UNSIGNED )
|
|
{
|
|
return p1->ReadVarInt64() != p2->ReadVarInt64();
|
|
}
|
|
return p1->ReadSignedVarInt64() != p2->ReadSignedVarInt64();
|
|
}
|
|
|
|
uint32 highInt1 = p1->ReadUBitLong( pProp->m_nBits - 32 );
|
|
uint32 lowInt1 = p1->ReadUBitLong( 32 );
|
|
uint32 highInt2 = p2->ReadUBitLong( pProp->m_nBits - 32 );
|
|
uint32 lowInt2 = p2->ReadUBitLong( 32 );
|
|
return highInt1 != highInt2 || lowInt1 != lowInt2;
|
|
}
|
|
|
|
const char* Int64_GetTypeNameString()
|
|
{
|
|
return "DPT_Int64";
|
|
}
|
|
|
|
|
|
bool Int64_IsZero( const unsigned char *pStruct, DVariant *pVar, const SendProp *pProp )
|
|
{
|
|
#ifdef SUPPORTS_INT64
|
|
return (pVar->m_Int64 == 0);
|
|
#else
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
|
|
void Int64_DecodeZero( DecodeInfo *pInfo )
|
|
{
|
|
#ifdef SUPPORTS_INT64
|
|
pInfo->m_Value.m_Int64 = 0;
|
|
|
|
if ( pInfo->m_pRecvProp )
|
|
{
|
|
pInfo->m_pRecvProp->GetProxyFn()( pInfo, pInfo->m_pStruct, pInfo->m_pData );
|
|
}
|
|
#endif
|
|
}
|
|
|
|
bool Int64_IsEncodedZero( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_VARINT)
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_UNSIGNED )
|
|
{
|
|
return pIn->ReadVarInt64() == 0;
|
|
}
|
|
return pIn->ReadSignedVarInt64() == 0;
|
|
}
|
|
|
|
uint32 highInt1 = pIn->ReadUBitLong( pProp->m_nBits - 32 );
|
|
uint32 lowInt1 = pIn->ReadUBitLong( 32 );
|
|
return (highInt1 == 0 && lowInt1 == 0);
|
|
}
|
|
|
|
void Int64_SkipProp( const SendProp *pProp, bf_read *pIn )
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_VARINT)
|
|
{
|
|
if ( pProp->GetFlags() & SPROP_UNSIGNED )
|
|
{
|
|
pIn->ReadVarInt64();
|
|
}
|
|
else
|
|
{
|
|
pIn->ReadSignedVarInt64();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
pIn->SeekRelative( pProp->m_nBits );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
PropTypeFns g_PropTypeFns[DPT_NUMSendPropTypes] =
|
|
{
|
|
// DPT_Int
|
|
{
|
|
Int_Encode,
|
|
Int_Decode,
|
|
Int_CompareDeltas,
|
|
Generic_FastCopy,
|
|
Int_GetTypeNameString,
|
|
Int_IsZero,
|
|
Int_DecodeZero,
|
|
Int_IsEncodedZero,
|
|
Int_SkipProp,
|
|
},
|
|
|
|
// DPT_Float
|
|
{
|
|
Float_Encode,
|
|
Float_Decode,
|
|
Float_CompareDeltas,
|
|
Generic_FastCopy,
|
|
Float_GetTypeNameString,
|
|
Float_IsZero,
|
|
Float_DecodeZero,
|
|
Float_IsEncodedZero,
|
|
Float_SkipProp,
|
|
},
|
|
|
|
// DPT_Vector
|
|
{
|
|
Vector_Encode,
|
|
Vector_Decode,
|
|
Vector_CompareDeltas,
|
|
Generic_FastCopy,
|
|
Vector_GetTypeNameString,
|
|
Vector_IsZero,
|
|
Vector_DecodeZero,
|
|
Vector_IsEncodedZero,
|
|
Vector_SkipProp,
|
|
},
|
|
|
|
// DPT_VectorXY
|
|
{
|
|
VectorXY_Encode,
|
|
VectorXY_Decode,
|
|
VectorXY_CompareDeltas,
|
|
Generic_FastCopy,
|
|
VectorXY_GetTypeNameString,
|
|
VectorXY_IsZero,
|
|
VectorXY_DecodeZero,
|
|
VectorXY_IsEncodedZero,
|
|
VectorXY_SkipProp,
|
|
},
|
|
|
|
// DPT_String
|
|
{
|
|
String_Encode,
|
|
String_Decode,
|
|
String_CompareDeltas,
|
|
Generic_FastCopy,
|
|
String_GetTypeNameString,
|
|
String_IsZero,
|
|
String_DecodeZero,
|
|
String_IsEncodedZero,
|
|
String_SkipProp,
|
|
},
|
|
|
|
// DPT_Array
|
|
{
|
|
Array_Encode,
|
|
Array_Decode,
|
|
Array_CompareDeltas,
|
|
Array_FastCopy,
|
|
Array_GetTypeNameString,
|
|
Array_IsZero,
|
|
Array_DecodeZero,
|
|
Array_IsEncodedZero,
|
|
Array_SkipProp,
|
|
},
|
|
|
|
// DPT_DataTable
|
|
{
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
DataTable_GetTypeNameString,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
},
|
|
#if 0 // We can't ship this since it changes the size of DTVariant to be 20 bytes instead of 16 and that breaks MODs!!!
|
|
|
|
// DPT_Quaternion
|
|
{
|
|
Quaternion_Encode,
|
|
Quaternion_Decode,
|
|
Quaternion_CompareDeltas,
|
|
Generic_FastCopy,
|
|
Quaternion_GetTypeNameString,
|
|
Quaternion_IsZero,
|
|
Quaternion_DecodeZero,
|
|
Quaternion_IsEncodedZero,
|
|
Quaternion_SkipProp,
|
|
},
|
|
#endif
|
|
|
|
#ifdef SUPPORTS_INT64
|
|
// DPT_Int64
|
|
{
|
|
Int64_Encode,
|
|
Int64_Decode,
|
|
Int64_CompareDeltas,
|
|
Generic_FastCopy,
|
|
Int64_GetTypeNameString,
|
|
Int64_IsZero,
|
|
Int64_DecodeZero,
|
|
Int64_IsEncodedZero,
|
|
Int64_SkipProp,
|
|
},
|
|
#endif
|
|
|
|
};
|