522 lines
13 KiB
C
Raw Normal View History

2020-04-22 12:56:21 -04:00
//========= Copyright Valve Corporation, All rights reserved. ============//
//
// Purpose:
//
// $NoKeywords: $
//=============================================================================//
#ifndef DATATABLE_H
#define DATATABLE_H
#ifdef _WIN32
#pragma once
#endif
#include "dt_common.h"
#include "dt_recv_eng.h"
#include "dt_send_eng.h"
#include "utlvector.h"
#include "dt_encode.h"
#include "utlmap.h"
#include "tier1/bitbuf.h"
class SendTable;
class RecvTable;
class CDTISendTable;
#define MAX_EXCLUDE_PROPS 512
// Bit counts used to encode the information about a property.
#define PROPINFOBITS_NUMPROPS 10
#define PROPINFOBITS_TYPE 5
#define PROPINFOBITS_FLAGS SPROP_NUMFLAGBITS_NETWORKED
#define PROPINFOBITS_STRINGBUFFERLEN 10
#define PROPINFOBITS_NUMBITS 7
#define PROPINFOBITS_RIGHTSHIFT 6
#define PROPINFOBITS_NUMELEMENTS 10 // For arrays.
class ExcludeProp
{
public:
char const *m_pTableName;
char const *m_pPropName;
};
// ------------------------------------------------------------------------------------ //
// CDeltaBitsReader.
// ------------------------------------------------------------------------------------ //
class CDeltaBitsReader
{
public:
CDeltaBitsReader( bf_read *pBuf );
~CDeltaBitsReader();
// Write the next property index. Returns the number of bits used.
unsigned int ReadNextPropIndex();
unsigned int ReadNextPropIndex_Continued();
void SkipPropData( const SendProp *pProp );
int ComparePropData( CDeltaBitsReader* pOut, const SendProp *pProp );
void CopyPropData( bf_write* pOut, const SendProp *pProp );
// If you know you're done but you're not at the end (you haven't called until
// ReadNextPropIndex returns -1), call this so it won't assert in its destructor.
void ForceFinished();
private:
bf_read *m_pBuf;
int m_iLastProp;
};
FORCEINLINE CDeltaBitsReader::CDeltaBitsReader( bf_read *pBuf )
{
m_pBuf = pBuf;
m_iLastProp = -1;
}
FORCEINLINE CDeltaBitsReader::~CDeltaBitsReader()
{
// Make sure they read to the end unless they specifically said they don't care.
Assert( !m_pBuf );
}
FORCEINLINE void CDeltaBitsReader::ForceFinished()
{
#ifdef DBGFLAG_ASSERT
m_pBuf = NULL;
#endif
}
FORCEINLINE unsigned int CDeltaBitsReader::ReadNextPropIndex()
{
Assert( m_pBuf );
// Expanded and optimized version of
// if ( m_pBuf->ReadOneBit() )
// {
// m_iLastProp += 1 + m_pBuf->ReadUBitVar();
// return m_iLastProp;
// }
// return ~0u;
if ( m_pBuf->GetNumBitsLeft() >= 7 )
{
uint bits = m_pBuf->ReadUBitLong( 7 );
if ( bits & 1 )
{
uint delta = bits >> 3;
if ( bits & 6 )
{
delta = m_pBuf->ReadUBitVarInternal( (bits & 6) >> 1 );
}
m_iLastProp = m_iLastProp + 1 + delta;
Assert( m_iLastProp < MAX_DATATABLE_PROPS );
return m_iLastProp;
}
m_pBuf->m_iCurBit -= 6; // Unread six bits we shouldn't have looked at
}
else
{
// Not enough bits for a property index.
if ( m_pBuf->ReadOneBit() )
{
// Expected a zero bit! Force an overflow!
m_pBuf->Seek(-1);
}
}
ForceFinished();
return ~0u;
}
FORCEINLINE void CDeltaBitsReader::SkipPropData( const SendProp *pProp )
{
g_PropTypeFns[ pProp->GetType() ].SkipProp( pProp, m_pBuf );
}
FORCEINLINE void CDeltaBitsReader::CopyPropData( bf_write* pOut, const SendProp *pProp )
{
int start = m_pBuf->GetNumBitsRead();
g_PropTypeFns[ pProp->GetType() ].SkipProp( pProp, m_pBuf );
int len = m_pBuf->GetNumBitsRead() - start;
m_pBuf->Seek( start );
pOut->WriteBitsFromBuffer( m_pBuf, len );
}
FORCEINLINE int CDeltaBitsReader::ComparePropData( CDeltaBitsReader *pInReader, const SendProp *pProp )
{
bf_read *pIn = pInReader->m_pBuf;
return g_PropTypeFns[pProp->m_Type].CompareDeltas( pProp, m_pBuf, pIn );
}
// ------------------------------------------------------------------------------------ //
// CDeltaBitsWriter.
// ------------------------------------------------------------------------------------ //
class CDeltaBitsWriter
{
public:
CDeltaBitsWriter( bf_write *pBuf );
~CDeltaBitsWriter();
// Write the next property index. Returns the number of bits used.
void WritePropIndex( int iProp );
// Access the buffer it's outputting to.
bf_write* GetBitBuf();
private:
bf_write *m_pBuf;
int m_iLastProp;
};
inline CDeltaBitsWriter::CDeltaBitsWriter( bf_write *pBuf )
{
m_pBuf = pBuf;
m_iLastProp = -1;
}
inline bf_write* CDeltaBitsWriter::GetBitBuf()
{
return m_pBuf;
}
FORCEINLINE void CDeltaBitsWriter::WritePropIndex( int iProp )
{
Assert( iProp >= 0 && iProp < MAX_DATATABLE_PROPS );
unsigned int diff = iProp - m_iLastProp;
m_iLastProp = iProp;
Assert( diff > 0 && diff <= MAX_DATATABLE_PROPS );
// Expanded inline for maximum efficiency.
//m_pBuf->WriteOneBit( 1 );
//m_pBuf->WriteUBitVar( diff - 1 );
COMPILE_TIME_ASSERT( MAX_DATATABLE_PROPS <= 0x1000u );
int n = ((diff < 0x11u) ? -1 : 0) + ((diff < 0x101u) ? -1 : 0);
m_pBuf->WriteUBitLong( diff*8 - 8 + 4 + n*2 + 1, 8 + n*4 + 4 + 2 + 1 );
}
inline CDeltaBitsWriter::~CDeltaBitsWriter()
{
m_pBuf->WriteOneBit( 0 );
}
// ----------------------------------------------------------------------------- //
//
// CSendNode
//
// Each datatable gets a tree of CSendNodes. There is one CSendNode
// for each datatable property that was in the original SendTable.
//
// ----------------------------------------------------------------------------- //
class CSendNode
{
public:
CSendNode();
~CSendNode();
int GetNumChildren() const;
CSendNode* GetChild( int i ) const;
// Returns true if the specified prop is in this node or any of its children.
bool IsPropInRecursiveProps( int i ) const;
// Each datatable property (without SPROP_PROXY_ALWAYS_YES set) gets a unique index here.
// The engine stores arrays of CSendProxyRecipients with the results of the proxies and indexes the results
// with this index.
//
// Returns DATATABLE_PROXY_INDEX_NOPROXY if the property has SPROP_PROXY_ALWAYS_YES set.
unsigned short GetDataTableProxyIndex() const;
void SetDataTableProxyIndex( unsigned short val );
// Similar to m_DataTableProxyIndex, but doesn't use DATATABLE_PROXY_INDEX_INVALID,
// so this can be used to index CDataTableStack::m_pProxies.
unsigned short GetRecursiveProxyIndex() const;
void SetRecursiveProxyIndex( unsigned short val );
public:
// Child datatables.
CUtlVector<CSendNode*> m_Children;
// The datatable property that leads us to this CSendNode.
// This indexes the CSendTablePrecalc or CRecvDecoder's m_DatatableProps list.
// The root CSendNode sets this to -1.
short m_iDatatableProp;
// The SendTable that this node represents.
// ALL CSendNodes have this.
const SendTable *m_pTable;
//
// Properties in this table.
//
// m_iFirstRecursiveProp to m_nRecursiveProps defines the list of propertise
// of this node and all its children.
unsigned short m_iFirstRecursiveProp;
unsigned short m_nRecursiveProps;
// See GetDataTableProxyIndex().
unsigned short m_DataTableProxyIndex;
// See GetRecursiveProxyIndex().
unsigned short m_RecursiveProxyIndex;
};
inline int CSendNode::GetNumChildren() const
{
return m_Children.Count();
}
inline CSendNode* CSendNode::GetChild( int i ) const
{
return m_Children[i];
}
inline bool CSendNode::IsPropInRecursiveProps( int i ) const
{
int index = i - (int)m_iFirstRecursiveProp;
return index >= 0 && index < m_nRecursiveProps;
}
inline unsigned short CSendNode::GetDataTableProxyIndex() const
{
Assert( m_DataTableProxyIndex != DATATABLE_PROXY_INDEX_INVALID ); // Make sure it's been set before.
return m_DataTableProxyIndex;
}
inline void CSendNode::SetDataTableProxyIndex( unsigned short val )
{
m_DataTableProxyIndex = val;
}
inline unsigned short CSendNode::GetRecursiveProxyIndex() const
{
return m_RecursiveProxyIndex;
}
inline void CSendNode::SetRecursiveProxyIndex( unsigned short val )
{
m_RecursiveProxyIndex = val;
}
class CFastLocalTransferPropInfo
{
public:
unsigned short m_iRecvOffset;
unsigned short m_iSendOffset;
unsigned short m_iProp;
};
class CFastLocalTransferInfo
{
public:
CUtlVector<CFastLocalTransferPropInfo> m_FastInt32;
CUtlVector<CFastLocalTransferPropInfo> m_FastInt16;
CUtlVector<CFastLocalTransferPropInfo> m_FastInt8;
CUtlVector<CFastLocalTransferPropInfo> m_FastVector;
CUtlVector<CFastLocalTransferPropInfo> m_OtherProps; // Props that must be copied slowly (proxies and all).
};
// ----------------------------------------------------------------------------- //
// CSendTablePrecalc
// ----------------------------------------------------------------------------- //
class CSendTablePrecalc
{
public:
CSendTablePrecalc();
virtual ~CSendTablePrecalc();
// This function builds the flat property array given a SendTable.
bool SetupFlatPropertyArray();
int GetNumProps() const;
const SendProp* GetProp( int i ) const;
int GetNumDatatableProps() const;
const SendProp* GetDatatableProp( int i ) const;
SendTable* GetSendTable() const;
CSendNode* GetRootNode();
int GetNumDataTableProxies() const;
void SetNumDataTableProxies( int count );
public:
class CProxyPathEntry
{
public:
unsigned short m_iDatatableProp; // Lookup into CSendTablePrecalc or CRecvDecoder::m_DatatableProps.
unsigned short m_iProxy;
};
class CProxyPath
{
public:
unsigned short m_iFirstEntry; // Index into m_ProxyPathEntries.
unsigned short m_nEntries;
};
CUtlVector<CProxyPathEntry> m_ProxyPathEntries; // For each proxy index, this is all the DT proxies that generate it.
CUtlVector<CProxyPath> m_ProxyPaths; // CProxyPathEntries lookup into this.
// These are what CSendNodes reference.
// These are actual data properties (ints, floats, etc).
CUtlVector<const SendProp*> m_Props;
// Each datatable in a SendTable's tree gets a proxy index, and its properties reference that.
CUtlVector<unsigned char> m_PropProxyIndices;
// CSendNode::m_iDatatableProp indexes this.
// These are the datatable properties (SendPropDataTable).
CUtlVector<const SendProp*> m_DatatableProps;
// This is the property hierarchy, with the nodes indexing m_Props.
CSendNode m_Root;
// From whence we came.
SendTable *m_pSendTable;
// For instrumentation.
CDTISendTable *m_pDTITable;
// This is precalculated in single player to allow faster direct copying of the entity data
// from the server entity to the client entity.
CFastLocalTransferInfo m_FastLocalTransfer;
// This tells how many data table properties there are without SPROP_PROXY_ALWAYS_YES.
// Arrays allocated with this size can be indexed by CSendNode::GetDataTableProxyIndex().
int m_nDataTableProxies;
// Map prop offsets to indices for properties that can use it.
CUtlMap<unsigned short, unsigned short> m_PropOffsetToIndexMap;
};
inline int CSendTablePrecalc::GetNumProps() const
{
return m_Props.Count();
}
inline const SendProp* CSendTablePrecalc::GetProp( int i ) const
{
return m_Props[i];
}
inline int CSendTablePrecalc::GetNumDatatableProps() const
{
return m_DatatableProps.Count();
}
inline const SendProp* CSendTablePrecalc::GetDatatableProp( int i ) const
{
return m_DatatableProps[i];
}
inline SendTable* CSendTablePrecalc::GetSendTable() const
{
return m_pSendTable;
}
inline CSendNode* CSendTablePrecalc::GetRootNode()
{
return &m_Root;
}
inline int CSendTablePrecalc::GetNumDataTableProxies() const
{
return m_nDataTableProxies;
}
inline void CSendTablePrecalc::SetNumDataTableProxies( int count )
{
m_nDataTableProxies = count;
}
// ------------------------------------------------------------------------ //
// Helpers.
// ------------------------------------------------------------------------ //
// Used internally by various datatable modules.
void DataTable_Warning( PRINTF_FORMAT_STRING const char *pInMessage, ... ) FMTFUNCTION( 1, 2 );
bool ShouldWatchThisProp( const SendTable *pTable, int objectID, const char *pPropName );
// Same as AreBitArraysEqual but does a trivial test to make sure the
// two arrays are equally sized.
bool CompareBitArrays(
void const *pPacked1,
void const *pPacked2,
int nBits1,
int nBits2
);
// to skip of a Property we just IsEncodedZero to read over it
// this is faster then doing a full Decode()
inline void SkipPropData( bf_read *pIn, const SendProp *pProp )
{
g_PropTypeFns[ pProp->GetType() ].SkipProp( pProp, pIn );
}
// This is to be called on SendTables and RecvTables to setup array properties
// to point at their property templates and to set the SPROP_INSIDEARRAY flag
// on the properties inside arrays.
// We make the proptype an explicit template parameter because
// gcc templating cannot deduce typedefs from classes in templates properly
template< class TableType, class PropType >
void SetupArrayProps_R( TableType *pTable )
{
// If this table has already been initialized in here, then jump out.
if ( pTable->IsInitialized() )
return;
pTable->SetInitialized( true );
for ( int i=0; i < pTable->GetNumProps(); i++ )
{
PropType *pProp = pTable->GetProp( i );
if ( pProp->GetType() == DPT_Array )
{
ErrorIfNot( i >= 1,
("SetupArrayProps_R: array prop '%s' is at index zero.", pProp->GetName())
);
// Get the property defining the elements in the array.
PropType *pArrayProp = pTable->GetProp( i-1 );
pArrayProp->SetInsideArray();
pProp->SetArrayProp( pArrayProp );
}
else if ( pProp->GetType() == DPT_DataTable )
{
// Recurse into children datatables.
SetupArrayProps_R<TableType,PropType>( pProp->GetDataTable() );
}
}
}
#endif // DATATABLE_H