// Ryzom - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
#include "stdpch.h"
#include "game_share/action_factory.h"
#include "game_share/action_position.h"
#include "game_share/action_sint64.h"
#include "game_share/mode_and_behaviour.h"
#include "game_share/entity_types.h"
#include "distance_prioritizer.h"
#include "client_host.h"
#include "vision_provider.h"
#include
#include "frontend_service.h"
#ifdef TEST_LOST_PACKET
#include
#endif
using namespace CLFECOMMON;
using namespace NLMISC;
using namespace NLNET;
// Test feature to simulate a lost packet when sending a new sheet. Works with 1 client only.
#ifdef TEST_LOST_PACKET
CVariable TestPacketLost( "test", "TestPacketLost", "", true, 0, 1 );
TGameCycle TestPacketLostTimer = 0;
TCLEntityId TestPacketLostSlot = 0;
#endif
TClientId verbosePropertiesSent = INVALID_CLIENT;
// TODO-Minimal visual bandwith
/*sint32 NbMinimalVisualBits;
void cbNbMinimalVisualBytesChanged( IVariable& var );
CVariable NbMinimalVisualBytes( "fe", "NbMinimalVisualBytes", "Min number of bytes for vision per msg", 50, 0, true, cbNbMinimalVisualBytesChanged, true );
void cbNbMinimalVisualBytesChanged( IVariable& var )
{
NbMinimalVisualBits = NbMinimalVisualBytes.get() * 8;
}*/
/// Init
void CDistancePrioritizer::init( CVisionArray *va, CVisionProvider *vp, CHistory *hy )
{
_VisionArray = va;
_VisionProvider = vp;
_History = hy;
SortSpreader.init();
for ( sint i=0; i!=NB_VISUAL_PROPERTIES; ++i )
_DistThresholdTable[i] = -1;
_VisualPropertyTreeRoot = (TVPNodeServer*)NewNode();
sint nbProp = _VisualPropertyTreeRoot->buildTree();
// Check that the last property matches the number of properties-1
if ( nbProp != NB_VISUAL_PROPERTIES )
{
nlwarning( "Found %hu properties in the sheets, the prioritizer knows %hu properties", NB_VISUAL_PROPERTIES, nbProp );
}
// Init distance thresholds
CLFECOMMON::initThresholdTable( _DistThresholdTable );
}
/*
* Calculate the priorities
*/
void CDistancePrioritizer::calculatePriorities()
{
// Sort
if ( SortSpreader.mustProcessNow() )
{
THostMap::iterator icm;
sint clientmapindex, outerBoundIndex;
SortSpreader.getProcessingBounds( icm, clientmapindex, outerBoundIndex );
while ( clientmapindex < outerBoundIndex )
{
CClientHost *clienthost = GETCLIENTA(icm);
// Prioritize only at the opposite time of sending for a particular client
if ( ! clienthost->whenToSend() )
{
// Update priorities
updatePriorityOfEntitiesSeenByClient( clienthost->clientId() );
// Sort entities by decreasing priority
sortEntitiesOfClient( clienthost->clientId() );
}
++clientmapindex;
++icm;
}
SortSpreader.endProcessing( icm );
}
SortSpreader.incCycle();
}
/*
#define arbitrateDiscreetProperty( name ) \
get##name##node()->BranchHasPayload = entityIsWithinDistanceThreshold( PROPERTY_##name ) && discreetPropertyHasChanged( PROPERTY_##name, (TYPE_##name*)NULL )
#define arbitrateDiscreetPropertyWithoutThreshold( name ) \
get##name##node()->BranchHasPayload = discreetPropertyHasChanged( PROPERTY_##name, (TYPE_##name*)NULL )
#define arbitrateTargetList( name ) \
get##name##node()->BranchHasPayload = targetListHasChanged( PROPERTY_##name, (TYPE_##name*)NULL )
*/
#ifdef STORE_MIRROR_VP_IN_CLASS
#define arbitrateDiscreetProperty( entry, name ) \
GET_VP_NODE(name)->BranchHasPayload = \
entityIsWithinDistanceThreshold( PROPERTY_##name ) \
&& discreetPropertyHasChanged( entry.Properties[PROPERTY_##name], sentity->VP_##name, PROPERTY_##name, (TYPE_##name*)NULL )
#define arbitrateDiscreetPropertyWithoutThreshold( entry, name ) \
GET_VP_NODE(name)->BranchHasPayload = \
discreetPropertyHasChanged( entry.Properties[PROPERTY_##name], sentity->VP_##name, PROPERTY_##name, (TYPE_##name*)NULL )
#else // STORE_MIRROR_VP_IN_CLASS
#define arbitrateDiscreetProperty( entry, name ) \
GET_VP_NODE(name)->BranchHasPayload = \
entityIsWithinDistanceThreshold( PROPERTY_##name ) \
&& discreetPropertyHasChanged( entry.Properties[PROPERTY_##name], PROPERTY_##name, (TYPE_##name*)NULL )
#define arbitrateDiscreetPropertyWithoutThreshold( entry, name ) \
GET_VP_NODE(name)->BranchHasPayload = \
discreetPropertyHasChanged( entry.Properties[PROPERTY_##name], PROPERTY_##name, (TYPE_##name*)NULL )
#define arbitrateTargetList( entry, name ) \
#endif // STORE_MIRROR_VP_IN_CLASS
#define arbitrateNeverSendProperty( name ) \
GET_VP_NODE(name)->BranchHasPayload = false
#define arbitrateTargetList( entry, name ) \
GET_VP_NODE(name)->BranchHasPayload = \
targetListHasChanged( entry.Properties[PROPERTY_##name], PROPERTY_##name, (TYPE_##name*)NULL )
#define DECLARE_AP(name) CActionSint64 *ap = CActionFactory::getInstance()->getVolatilePropAction( TVPNodeServer::PrioContext.Slot, PROPERTY_##name )
#define DECLARE_AP_INDEX(index) CActionSint64 *ap = CActionFactory::getInstance()->getVolatilePropAction( TVPNodeServer::PrioContext.Slot, index )
#define REMOVE_AP()
/*
#define DECLARE_AP(name) CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_##name )
#define DECLARE_AP_INDEX(index) CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, index )
#define REMOVE_AP() CActionFactory::getInstance()->remove( (CAction*&)ap )
*/
/*
*
*/
void CDistancePrioritizer::fillOutBox( CClientHost& client, TOutBox& outbox )
{
H_AUTO(FillOutBox)
TClientId clientId = client.clientId();
initDispatchingCycle( clientId );
#ifdef NL_DEBUG
client.MoveNumber = 0;
#endif
if (!client.entityIndex().isValid())
return;
// initialize common context part
TVPNodeServer::PrioContext.Prioritizer = this;
TVPNodeServer::PrioContext.ClientHost = &client;
TVPNodeServer::PrioContext.ClientId = clientId;
CMirrorPropValueRO targetIndexOfClient( TheDataset, client.entityIndex(), DSPropertyTARGET_ID );
// TODO-Minimal visual bandwith
/*sint32 initialPosInBit = outbox.getPosInBit();*/
sint32 currentPosInBit;
while ( true )
{
currentPosInBit = outbox.getPosInBit();
// Browse the seen entities, sorted by distance
TCLEntityId slot = getNextEntityToStudy( clientId );
if ( slot == INVALID_SLOT )
{
// Exit when all the pairs have been successfully filled
#ifdef TEST_LOST_PACKET
if ( (TestPacketLostTimer!=0) && (CTickEventHandler::getGameCycle() >= TestPacketLostTimer) )
{
nldebug( "Negative ack for %hu %hu", client.clientId(), TestPacketLostSlot );
_History->_PacketHistory.negativeAck( client.clientId(), TestPacketLostSlot, PROPERTY_SHEET, 0 );
TestPacketLostTimer = 0;
}
#endif
return;
}
// TODO-Minimal visual bandwith
// Always allow to write a minimal amount of visual properties
/*if ( currentPosInBit - initialPosInBit >= NbMinimalVisualBits )*/
{
// Don't fill if the free space is lower than 32 bits (checked once per pair, not once per property)
if ( currentPosInBit + 32 > client.getCurrentThrottle() )
{
// Exit when the size limit has been reached before all the pairs have been filled
#ifdef NL_DEBUG
uint nbRemainingPairs = _PrioritizedEntitiesByClient[clientId].size() - _CurrentEntitiesToStudy[clientId];
if ( nbRemainingPairs > 0 )
LOG_WHAT_IS_SENT( "%u: C%hu S%hu: %u pairs remaining", CTickEventHandler::getGameCycle(), clientId, (uint16)slot, nbRemainingPairs );
LOG_WHAT_IS_SENT( "C%hu: outbox full (%d bits)", clientId, currentPosInBit );
#endif
#ifdef TEST_LOST_PACKET
if ( (TestPacketLostTimer!=0) && (CTickEventHandler::getGameCycle() >= TestPacketLostTimer) )
{
nldebug( "Negative ack for %hu %hu", client.clientId(), TestPacketLostSlot );
_History->_PacketHistory.negativeAck( client.clientId(), TestPacketLostSlot, PROPERTY_SHEET, 0 );
TestPacketLostTimer = 0;
}
#endif
return;
}
}
// Get the entity corresponding to the the client/slot pair
TPairState& pairState = _VisionArray->getPairState( clientId, slot );
CEntity* sentity = NULL;
TEntityIndex entityIndex = pairState.EntityIndex;
TVPNodeServer::PrioContext.EntityIndex = entityIndex;
if ( entityIndex.isValid() )
sentity = TheEntityContainer->getEntity( entityIndex );
if ( pairState.associationSuppressed() )
{
// Pure unassociation case: we must send an empty block
serialSlotHeader( client, NULL, pairState, slot, outbox );
uint32 bits = 0;
outbox.serialAndLog2( bits, 2 ); // 2 bits for pos & other properties
//nldebug( "Pure unassociation of C%hu S%hu", clientId, (uint16)slot );
// The first time, sentity is non-null. If this is a resending (after a neg-ack), it's null.
if ( sentity )
_VisionProvider->postRemovePair( clientId, slot );
}
else
{
if ( sentity == NULL )
continue;
H_BEFORE(OneSlotArbitrate)
H_BEFORE(OneSlotArbitrateInit)
// Initialize the context
TVPNodeServer::PrioContext.Slot = slot;
TVPNodeServer::PrioContext.EntityIndex = entityIndex;
TVPNodeServer::PrioContext.Sentity = sentity;
//TVPNodeServer::PrioContext.PairState = &pairState;
TVPNodeServer::PrioContext.DistanceCE = pairState.DistanceCE;
TVPNodeServer::PrioContext.Timestamp = 0;
TVPNodeServer::PrioContext.IsTarget = (targetIndexOfClient() == entityIndex);
TVPNodeServer::PrioContext.PositionAlreadySent = (_History->getMileage( clientId, slot ) != 0);
TVPNodeServer::PrioContext.ZCache = sentity->z(entityIndex); // setup Z cache for all later mirror access to entity->z()
const CPropertyHistory::CEntityEntry& entry = _History->getEntityEntry(TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.Slot);
// Debug display
//nlinfo( "Preparing a block to client %hu (%s) for slot %hu", clientId, client.eId().toString().c_str(), (uint16)slot );
// Continuous properties
uint8 seenEntityType = TheDataset.getEntityId( entityIndex ).getType();
H_AFTER(OneSlotArbitrateInit)
bool thetaIntMode = false;
// Arbitrate all discreet properties
switch (seenEntityType)
{
default:
arbitrateAllDiscreetProperties(entry);
break;
case RYZOMID::npc:
arbitrateNPCDiscreetProperties(entry);
break;
case RYZOMID::creature:
arbitrateCreatureDiscreetProperties(entry);
break;
case RYZOMID::forageSource:
arbitrateForageSourceDiscreetProperties(entry);
thetaIntMode = true;
break;
}
H_BEFORE(OneSlotArbitratePropagate)
// Propagate back BranchHasPayload flags
//_VisualPropertyTreeRoot->propagateBackBranchHasPayload();
TVPNodeServer::fastPropagateBackBranchHasPayload();
H_AFTER(OneSlotArbitratePropagate)
H_AFTER(OneSlotArbitrate)
// ******** Fill the buffer with the properties ********
// Fill the header
if ( _VisualPropertyTreeRoot->BranchHasPayload )
{
H_AUTO(OneSlotFill)
//nldebug( "Filling for C%hu, pass %u, BEFORE HEADER: bitpos: %d", clientId, ++i, outbox.getPosInBit() );
//nlinfo( "C%hu S%hu AB%hu", clientId, (uint16)slot, (uint32)pairState.AssociationChangeBits );
serialSlotHeader( client, sentity, pairState, slot, outbox );
//nldebug( "AFTER HEADER: pos: %d", outbox.getPosInBit() );
// Fill the position if required
//TVPNodeServer *currentNode = _VisualPropertyTreeRoot;
outbox.serialBitAndLog( GET_VP_NODE(POSITION)->BranchHasPayload );
if ( GET_VP_NODE(POSITION)->BranchHasPayload )
{
//CActionPosition *ap = (CActionPosition*)(CActionFactory::getInstance()->createByPropIndex( slot, PROPERTY_POSITION ));
//ap->PropertyCode = PROPERTY_POSITION;
CActionPosition* ap = CActionFactory::getInstance()->getVolatilePositionAction(slot);
// When the mode is transmitted or the entity is not in local mode (first bit of Z), transmit the *absolute* position
if ( (! (TVPNodeServer::PrioContext.ZCache & 0x1)) ||
(GET_VP_NODE(POSITION)->BranchHasPayload && GET_VP_NODE(MODE)->BranchHasPayload) )
{
ap->Position[0] = sentity->X();
ap->Position[1] = sentity->Y();
//ap->Position[2] = sentity->z( entityIndex );
ap->Position[2] = TVPNodeServer::PrioContext.ZCache;
ap->IsRelative = false;
LOG_WHAT_IS_SENT( "%u: C%hu S%hu: Filling ABSPOS: %d %d %d m", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, sentity->posXm( entityIndex ), sentity->posYm( entityIndex ), sentity->posZm( entityIndex ) );
}
else
{
CMirrorPropValueRO propLX( TheDataset, entityIndex, DSPropertyLocalX );
CMirrorPropValueRO propLY( TheDataset, entityIndex, DSPropertyLocalY );
CMirrorPropValueRO propLZ( TheDataset, entityIndex, DSPropertyLocalZ );
ap->Position[0] = propLX();
ap->Position[1] = propLY();
ap->Position[2] = propLZ();
ap->IsRelative = true;
LOG_WHAT_IS_SENT( "%u: C%hu S%hu: Filling RELPOS: %d %d %d mm", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, ap->Position[0], ap->Position[1], ap->Position[2] );
}
CActionFactory::getInstance()->packFast( ap, outbox );
_History->storePosition( clientId, client.sendNumber(), ap, sentity->Mileage, TVPNodeServer::PrioContext.IsTarget, TVPNodeServer::PrioContext.Timestamp );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
++client.NbActionsSentAtCycle;
}
// Fill the orientation if required
//currentNode = currentNode->B;
TVPNodeServer *currentNode = _VisualPropertyTreeRoot->B;
outbox.serialBitAndLog( currentNode->BranchHasPayload );
if ( currentNode->BranchHasPayload )
{
outbox.serialBitAndLog( currentNode->A->BranchHasPayload );
if ( currentNode->A->BranchHasPayload )
{
//CActionSint64 *ap = (CActionSint64*)(CActionFactory::getInstance()->createByPropIndex( slot, PROPERTY_ORIENTATION ));
//ap->PropertyCode = PROPERTY_ORIENTATION; // useless: already set by createByPropIndex
DECLARE_AP(ORIENTATION);
CMirrorPropValueRO prop( TheDataset, entityIndex, DSPropertyORIENTATION );
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu ORIENT (P%hu) : %.1f", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, prop() );
uint32 value = ( thetaIntMode ? (uint32)(prop()) : *((uint32*)&(prop())) );
ap->setAndPackValue( value, outbox );
_History->store( clientId, client.sendNumber(), ap );
// TODO: send less bits for forage source entity
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++client.NbActionsSentAtCycle;
}
// Fill the required discreet properties if required
currentNode->B->fillDiscreetProperties( outbox );
//nlinfo("end fill");
//TVPNodeServer::fastFillDiscreetProperties( outbox );
//nlinfo("end fastfill");
}
}
#ifdef NL_DEBUG
if ( _VisualPropertyTreeRoot->A->BranchHasPayload )
{
//CClientEntityIdTranslator::CEntityInfo& info = client.IdTranslator.getInfo( slot ); // CHANGED BEN
//if ( info.AssociationState == CClientEntityIdTranslator::CEntityInfo::AwaitingAssAck ) // CHANGED BEN
if ( _VisionArray->getAssociationState(clientId, slot) == TPairState::AwaitingAssAck )
{
nlwarning( "C%hu S%hu: Sending position but sheet id not sent", clientId, (uint16)slot );
client.displaySlotProperties( slot );
}
}
//if ( (verbosePropertiesSent==9999) || (verbosePropertiesSent==TVPNodeServer::PrioContext.ClientId) )
//{
// nldebug( "To C%hu", clientId );
// outbox.displayStream();
//}
#endif
// ******** End of iteration ********
//_VisualPropertyTreeRoot->displayStatesOfTreeLeaves();
//nldebug( "C%hu S%hu: end of pair block at bitpos %d", clientId, (uint16)slot, outbox.getPosInBit() );
}
// Reset the priority of the pair, even if nothing was filled
if ( slot != 0 )
pairState.resetPrio(); // note: nothing can remain to be filled, as we allow to exceed the size limit
}
}
/*
* Test the criterion for the position of the entity 'slot' seen by 'clientId'
*/
bool CDistancePrioritizer::positionHasChangedEnough()
{
// TEMP: do not send if local mode (because the local position is useless for the mektoub)
if ( TVPNodeServer::PrioContext.ZCache & 0x1 )
return false;
uint32 lastSentMileage = _History->getMileage( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.Slot );
if ( lastSentMileage != 0 )
{
// Calculate difference distance between current and lastsent mileage (unsigned to allow mileage overflow)
if ( (TVPNodeServer::PrioContext.Sentity->Mileage - lastSentMileage) * _DistanceDeltaRatio > (uint32)(TVPNodeServer::PrioContext.DistanceCE) )
{
#ifdef NL_DEBUG
++(TVPNodeServer::PrioContext.ClientHost->MoveNumber);
#endif
return true;
}
else
{
if (verbosePropertiesSent==0)
{
nldebug("Ignoring move: Mileage=%u, lastSentMileage=%u, _DistanceDeltaRatio=%u, DistanceCE=%u",
(uint32) TVPNodeServer::PrioContext.Sentity->Mileage,
(uint32) lastSentMileage,
(uint32) _DistanceDeltaRatio,
(uint32) TVPNodeServer::PrioContext.DistanceCE );
}
return false;
}
}
else
{
// Not sent yet
return true;
}
}
/*
* Test the criterion for thetaIntMode
*/
bool CDistancePrioritizer::thetaIntModehasChanged(const CPropertyHistory::CPropertyEntry& entry)
{
if ( ! entityIsWithinDistanceThreshold( PROPERTY_ORIENTATION ) )
return false;
if ( entry.HasValue )
{
CMirrorPropValueRO currentTheta( TheDataset, TVPNodeServer::PrioContext.EntityIndex, DSPropertyORIENTATION );
return ( (uint32)currentTheta != (uint32)entry.LastSent );
}
else
{
return TVPNodeServer::PrioContext.Sentity->propertyIsInitialized( PROPERTY_ORIENTATION, DSPropertyORIENTATION, TVPNodeServer::PrioContext.EntityIndex, (TYPE_ORIENTATION*)NULL );
}
}
/*
* Test the criterion for the orientation of the entity 'slot' seen by 'clientId' + initialized
*/
bool CDistancePrioritizer::orientationHasChangedEnough(const CPropertyHistory::CPropertyEntry& entry, float angleRatio )
{
if ( ! entityIsWithinDistanceThreshold( PROPERTY_ORIENTATION ) )
return false;
if ( entry.HasValue )
{
CMirrorPropValueRO currentAngle( TheDataset, TVPNodeServer::PrioContext.EntityIndex, DSPropertyORIENTATION );
/*
// Orientation is a float angle in radian
const float& oldangle = *((float*)&(entry.LastSent));
float deltaAngle = (float)fabs( (float)(currentAngle() - oldangle) );
deltaAngle = (float)fmod( deltaAngle+(2*Pi), (2*Pi) );
//nldebug( "getDelta(theta) : dA=%g", deltaAngle );
return ( deltaAngle > (float)Pi/angleRatio ); // the orientation is useful only when the pos does not change
*/
float oldangle;
entry.getValue(oldangle);
float deltaAngle = (float)( Pi - fabs(fmod(currentAngle()-oldangle+4*Pi, 2*Pi)-Pi) ); // deltaAngle is in [0, 2*Pi]
//nldebug( "getDelta(theta) : dA=%g", deltaAngle );
return ( deltaAngle*angleRatio > (float)Pi ); // the orientation is useful only when the pos does not change
}
else
{
// Not sent yet => always sent theta, even if it's zero (anyway, it's unlikely to be exactly 0.0: remind that this initial float is determined, for bots, by leveldesigners with the mouse)
return true;
}
}
/*
* Test the criterion for the specified property of the entity 'slot' seen by 'clientId'
*/
inline bool CDistancePrioritizer::entityIsWithinDistanceThreshold( TPropIndex propIndex )
{
// Compare distance with the threshold
//nldebug( "C%hu - slot %hu - prop %hu: DISTANCE=%d THRESHOL=%d", TVPNodeServer::PrioContext.ClientHost->clientId(), (uint16)TVPNodeServer::PrioContext.Slot, propIndex, TVPNodeServer::PrioContext.DistanceCE, TheEntityTranslator->getDistThreshold( propertyid ) );
return ( TVPNodeServer::PrioContext.DistanceCE < getDistThreshold( propIndex ) );
}
/*
* Special arbitrate case for BEHAVIOUR property
* Threshold in this case should be the same than the target list threshold when
* the behaviour is a range attack or projectile behaviour
*/
#ifndef NL_DEBUG
inline
#endif
void CDistancePrioritizer::arbitrateDiscreetBehaviourProperty(const CPropertyHistory::CEntityEntry& entry, CEntity* sentity)
{
#ifdef STORE_MIRROR_VP_IN_CLASS
const CMirrorPropValueRO& propBehav = sentity->VP_BEHAVIOUR;
#else
CMirrorPropValueRO propBehav( TheDataset, TVPNodeServer::PrioContext.EntityIndex, DSPropertyBEHAVIOUR );
#endif
if (!discreetPropertyHasChanged( entry.Properties[PROPERTY_BEHAVIOUR], propBehav, PROPERTY_BEHAVIOUR, (TYPE_BEHAVIOUR*)NULL ))
{
GET_VP_NODE(BEHAVIOUR)->BranchHasPayload = false;
return;
}
const MBEHAV::CBehaviour &behav = propBehav();
TPropIndex refDistanceProperty;
switch(behav.Behaviour)
{
case MBEHAV::CAST_OFF_SUCCESS:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_OFF_LINK:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_CUR_SUCCESS:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_CUR_LINK:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_MIX_SUCCESS:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_MIX_LINK:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::RANGE_ATTACK:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_ACID:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_BLIND:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_COLD:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_ELEC:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_FEAR:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_FIRE:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_HEALHP:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_MAD:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_POISON:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_ROOT:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_ROT:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_SHOCK:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_SLEEP:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_SLOW:
refDistanceProperty = PROPERTY_TARGET_LIST;
break;
case MBEHAV::CAST_STUN:
refDistanceProperty = PROPERTY_TARGET_LIST; // valid distance should be the same than the target list
// because target list and behaviour are sent together
break;
default:
refDistanceProperty = PROPERTY_BEHAVIOUR;
break;
}
//
GET_VP_NODE(BEHAVIOUR)->BranchHasPayload = entityIsWithinDistanceThreshold(refDistanceProperty);
}
/*
* Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent
*/
void CDistancePrioritizer::arbitrateAllDiscreetProperties(const CPropertyHistory::CEntityEntry& entry)
{
H_AUTO(arbitrateAllDiscreetProperties);
CEntity* sentity = TVPNodeServer::PrioContext.Sentity;
arbitrateDiscreetPropertyWithoutThreshold( entry, SHEET );
arbitrateDiscreetBehaviourProperty( entry, sentity );
// Don't limit to the distance threshold when triggering sending of the name of the target
if ( TVPNodeServer::PrioContext.IsTarget )
arbitrateDiscreetPropertyWithoutThreshold( entry, NAME_STRING_ID );
else
arbitrateDiscreetProperty( entry, NAME_STRING_ID );
//arbitrateDiscreetProperty( entry, TARGET_ID ); // now done in fillOutBox() in "mode switch"
arbitrateDiscreetProperty( entry, CONTEXTUAL );
arbitrateDiscreetPropertyWithoutThreshold( entry, MODE );
arbitrateDiscreetProperty( entry, VPA );
arbitrateDiscreetProperty( entry, VPB );
arbitrateDiscreetProperty( entry, VPC );
arbitrateDiscreetPropertyWithoutThreshold( entry, ENTITY_MOUNTED_ID );
arbitrateDiscreetPropertyWithoutThreshold( entry, RIDER_ENTITY_ID );
arbitrateTargetList( entry, TARGET_LIST );
arbitrateDiscreetProperty( entry, VISUAL_FX );
arbitrateDiscreetProperty( entry, GUILD_SYMBOL );
arbitrateDiscreetProperty( entry, GUILD_NAME_ID );
arbitrateDiscreetProperty( entry, EVENT_FACTION_ID );
arbitrateDiscreetProperty( entry, PVP_MODE );
arbitrateDiscreetProperty( entry, PVP_CLAN );
arbitrateNeverSendProperty( OWNER_PEOPLE );
arbitrateDiscreetProperty( entry, OUTPOST_INFOS );
if (TVPNodeServer::PrioContext.Slot != 0)
{
arbitrateCommonPosAndMode(entry);
arbitrateDiscreetProperty( entry, TARGET_ID );
arbitrateDiscreetProperty( entry, BARS );
}
else
{
arbitrateSlot0PosAndMode(entry);
arbitrateDiscreetPropertyWithoutThreshold( entry, TARGET_ID ); // no need of threshold
// never send BARS for User Player (no need since sent with the USER:BARS message)
arbitrateNeverSendProperty( BARS );
}
}
/*
* Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent
*/
void CDistancePrioritizer::arbitrateNPCDiscreetProperties(const CPropertyHistory::CEntityEntry& entry)
{
H_AUTO(arbitrateNPCDiscreetProperties);
CEntity* sentity = TVPNodeServer::PrioContext.Sentity;
arbitrateDiscreetPropertyWithoutThreshold( entry, SHEET );
arbitrateDiscreetBehaviourProperty(entry, sentity);
// Don't limit to the distance threshold when triggering sending of the name of the target
if ( TVPNodeServer::PrioContext.IsTarget )
arbitrateDiscreetPropertyWithoutThreshold( entry, NAME_STRING_ID );
else
arbitrateDiscreetProperty( entry, NAME_STRING_ID );
arbitrateDiscreetProperty( entry, TARGET_ID ); // NPC can never be in Slot 0
// Specific distance for NPCs' contextual property
GET_VP_NODE(CONTEXTUAL)->BranchHasPayload =
TVPNodeServer::PrioContext.DistanceCE < THRESHOLD_CONTEXTUAL_NPC
&& discreetPropertyHasChanged( entry.Properties[PROPERTY_CONTEXTUAL], sentity->VP_CONTEXTUAL, PROPERTY_CONTEXTUAL, (TYPE_CONTEXTUAL*)NULL );
arbitrateDiscreetPropertyWithoutThreshold( entry, MODE );
arbitrateDiscreetProperty( entry, BARS );
arbitrateDiscreetProperty( entry, VPA );
arbitrateDiscreetProperty( entry, VPB );
arbitrateDiscreetProperty( entry, VPC );
arbitrateDiscreetPropertyWithoutThreshold( entry, ENTITY_MOUNTED_ID );
arbitrateNeverSendProperty( RIDER_ENTITY_ID );
arbitrateTargetList( entry, TARGET_LIST );
arbitrateDiscreetProperty( entry, VISUAL_FX );
arbitrateDiscreetProperty( entry, GUILD_SYMBOL );
arbitrateDiscreetProperty( entry, GUILD_NAME_ID );
arbitrateNeverSendProperty( EVENT_FACTION_ID );
arbitrateNeverSendProperty( PVP_MODE );
arbitrateNeverSendProperty( PVP_CLAN );
arbitrateNeverSendProperty( OWNER_PEOPLE );
arbitrateDiscreetProperty( entry, OUTPOST_INFOS );
arbitrateCommonPosAndMode(entry);
}
/*
* Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent
*/
void CDistancePrioritizer::arbitrateCreatureDiscreetProperties(const CPropertyHistory::CEntityEntry& entry)
{
H_AUTO(arbitrateCreatureDiscreetProperties);
CEntity* sentity = TVPNodeServer::PrioContext.Sentity;
arbitrateDiscreetPropertyWithoutThreshold( entry, SHEET );
arbitrateDiscreetBehaviourProperty(entry, sentity);
// Don't limit to the distance threshold when triggering sending of the name of the target
if ( TVPNodeServer::PrioContext.IsTarget )
arbitrateDiscreetPropertyWithoutThreshold( entry, NAME_STRING_ID );
else
arbitrateDiscreetProperty( entry, NAME_STRING_ID );
arbitrateDiscreetProperty( entry, TARGET_ID ); // Creature can never be in Slot 0
arbitrateDiscreetProperty( entry, CONTEXTUAL );
arbitrateDiscreetPropertyWithoutThreshold( entry, MODE );
arbitrateDiscreetProperty( entry, BARS );
arbitrateNeverSendProperty( VPA );
arbitrateDiscreetProperty( entry, VPB );
arbitrateNeverSendProperty( VPC );
arbitrateNeverSendProperty( ENTITY_MOUNTED_ID );
arbitrateDiscreetPropertyWithoutThreshold( entry, RIDER_ENTITY_ID );
arbitrateTargetList( entry, TARGET_LIST );
arbitrateNeverSendProperty( VISUAL_FX );
arbitrateNeverSendProperty( GUILD_SYMBOL );
arbitrateNeverSendProperty( GUILD_NAME_ID );
arbitrateNeverSendProperty( EVENT_FACTION_ID );
arbitrateNeverSendProperty( PVP_MODE );
arbitrateNeverSendProperty( PVP_CLAN );
arbitrateDiscreetProperty( entry, OWNER_PEOPLE );
arbitrateNeverSendProperty( OUTPOST_INFOS );
arbitrateCommonPosAndMode(entry);
}
/*
* Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent
*/
void CDistancePrioritizer::arbitrateForageSourceDiscreetProperties(const CPropertyHistory::CEntityEntry& entry)
{
H_AUTO(arbitrateForageSourceDiscreetProperties);
CEntity* sentity = TVPNodeServer::PrioContext.Sentity;
arbitrateDiscreetPropertyWithoutThreshold( entry, SHEET );
arbitrateNeverSendProperty( BEHAVIOUR );
arbitrateDiscreetProperty( entry, NAME_STRING_ID );
arbitrateDiscreetProperty( entry, TARGET_ID ); // ForageSource can never be in Slot 0
arbitrateDiscreetProperty( entry, CONTEXTUAL );
arbitrateNeverSendProperty( MODE );
arbitrateDiscreetProperty( entry, BARS );
arbitrateNeverSendProperty( VPA );
arbitrateNeverSendProperty( VPB );
arbitrateNeverSendProperty( VPC );
arbitrateNeverSendProperty( ENTITY_MOUNTED_ID );
arbitrateNeverSendProperty( RIDER_ENTITY_ID );
arbitrateTargetList( entry, TARGET_LIST );
arbitrateDiscreetProperty( entry, VISUAL_FX );
arbitrateNeverSendProperty( GUILD_SYMBOL );
arbitrateNeverSendProperty( GUILD_NAME_ID );
arbitrateNeverSendProperty( EVENT_FACTION_ID );
arbitrateNeverSendProperty( PVP_MODE );
arbitrateNeverSendProperty( PVP_CLAN );
arbitrateNeverSendProperty( OWNER_PEOPLE );
arbitrateNeverSendProperty( OUTPOST_INFOS );
GET_VP_NODE(POSITION)->BranchHasPayload = sentity->positionIsInitialized() && positionHasChangedEnough();
GET_VP_NODE(ORIENTATION)->BranchHasPayload = thetaIntModehasChanged(entry.Properties[PROPERTY_ORIENTATION]);
}
/*
* Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent
*/
inline void CDistancePrioritizer::arbitrateCommonPosAndMode(const CPropertyHistory::CEntityEntry& entry)
{
// Position if changed enough (if not carried by mode)
// Orientation if changed > 60° (it's head angle only) or in first block (useful for static entities such as bot objects)
bool modeIsChanging = GET_VP_NODE(MODE)->BranchHasPayload;
bool sheetIsChanging = GET_VP_NODE(SHEET)->BranchHasPayload;
bool posIsReady = TVPNodeServer::PrioContext.Sentity->positionIsInitialized();
GET_VP_NODE(POSITION)->BranchHasPayload = (!modeIsChanging) && posIsReady && positionHasChangedEnough();
GET_VP_NODE(ORIENTATION)->BranchHasPayload = (sheetIsChanging && posIsReady) || orientationHasChangedEnough( entry.Properties[PROPERTY_ORIENTATION], 36.0f ); // 5°
}
/*
* Fill the BranchHasPayload flags of the tree, using the rules deciding if a discreet property need to be sent
*/
inline void CDistancePrioritizer::arbitrateSlot0PosAndMode(const CPropertyHistory::CEntityEntry& entry)
{
arbitrateNeverSendProperty(POSITION);
arbitrateNeverSendProperty(ORIENTATION);
}
/*
*
*/
inline void CDistancePrioritizer::serialSlotHeader( CClientHost& client, CEntity *sentity, TPairState& pairState, CLFECOMMON::TCLEntityId slot, TOutBox& outbox )
{
// Slot (8 bits)
#ifdef NL_DEBUG
sint beginbitpos = outbox.getPosInBit();
#endif
outbox.serialAndLog1( slot );
// Association change bits (2 bits)
uint32 associationBits = (uint32)pairState.AssociationChangeBits;
outbox.serialAndLog2( associationBits, 2 );
if ( pairState.AssociationChangeBits != pairState.PrevAssociationBits )
{
//LOG_WHAT_IS_SENT( "slot %hu ab %u beginpos %d endpos %d beginbitpos %d endbitpos %d", (uint16)slot, associationBits, beginbitpos/8, outbox.getPosInBit()/8, beginbitpos, outbox.getPosInBit() );
pairState.PrevAssociationBits = pairState.AssociationChangeBits; // & 0x3;
_History->storeDisassociation( client.clientId(), slot, client.sendNumber(), pairState.AssociationChangeBits );
// pairState.AssociationChangeBits &= 0x7F;
}
// Timestamp (1 or 5 bits, depending on the type of entity) (TVPNodeServer::PrioContext.Timestamp initialized to 0)
uint32 timestampDelta = 0;
if ( sentity )
{
const CEntityId& seenEntityId = TheDataset.getEntityId( TVPNodeServer::PrioContext.EntityIndex );
if ( seenEntityId.getType() == RYZOMID::player )
{
// For players, always set the timestamp delta, using TickPosition
// Note: discreet property change times won't be accurate
TVPNodeServer::PrioContext.Timestamp = sentity->TickPosition;
timestampDelta = CTickEventHandler::getGameCycle() - sentity->TickPosition;
if ( timestampDelta > 15 ) // clamp to 4bit
timestampDelta = 15;
timestampDelta |= 0x10; // 'timestampIsThere bit': first bit is bit 5 (high to low order)
}
else if ( seenEntityId.getType() >= RYZOMID::object )
{
// For non-players/non-bots types (e.g. bags), set the timestamp delta if entity is being spawned to the client
//if ( _VisualPropertyTreeRoot->B->B->getSHEETnode()->BranchHasPayload ) // assumes this is done after arbitrateDiscreetProperties() // CHANGED BEN
if ( GET_VP_NODE(SHEET)->BranchHasPayload ) // assumes this is done after arbitrateDiscreetProperties()
{
TVPNodeServer::PrioContext.Timestamp = TheDataset.getOnlineTimestamp( TVPNodeServer::PrioContext.EntityIndex );
timestampDelta = CTickEventHandler::getGameCycle() - TVPNodeServer::PrioContext.Timestamp;
if ( timestampDelta > 15 ) // clamp to 4bit
timestampDelta = 15;
timestampDelta |= 0x10; // 'timestampIsThere bit': first bit is bit 5 (high to low order)
}
}
// For bots, the timestamp is not needed, the client will take _ServerGameCycle
}
outbox.serialAndLog2( timestampDelta, (timestampDelta!=0) ? 5 : 1 );
}
#ifdef STORE_MIRROR_VP_IN_CLASS
#define caseFillAction( name ) ap->setValue64( TVPNodeServer::PrioContext.Sentity->VP_##name() );
#else // STORE_MIRROR_VP_IN_CLASS
#define caseFillAction( name ) \
CMirrorPropValueRO prop( TheDataset, TVPNodeServer::PrioContext.EntityIndex, DSProperty##name ); \
ap->setValue64( prop() );
#endif // STORE_MIRROR_VP_IN_CLASS
/*
* SHEET
*/
void fillSHEET( TOutBox& outbox, TPropIndex )
{
#ifdef NL_DEBUG
// In non-debug cases, it is done in CVisionProvider::addPair()
//CClientEntityIdTranslator::CEntityInfo& info = TVPNodeServer::PrioContext.ClientHost->IdTranslator.getInfo( TVPNodeServer::PrioContext.Slot ); // CHANGED BEN
//info.AssociationState = CClientEntityIdTranslator::CEntityInfo::NormalAssociation; // CHANGED BEN
TVPNodeServer::PrioContext.Prioritizer->getVisionArray()->setAssociationState( TVPNodeServer::PrioContext.ClientId,
TVPNodeServer::PrioContext.Slot,
TPairState::NormalAssociation);
#endif
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_SHEET ); // CHANGED BEN
DECLARE_AP(SHEET);
// Pack sheet and compressed row into the action
#ifdef STORE_MIRROR_VP_IN_CLASS
CMirrorPropValueRO& prop = TVPNodeServer::PrioContext.Sentity->VP_SHEET;
#else
CMirrorPropValueRO prop( TheDataset, TVPNodeServer::PrioContext.EntityIndex, DSPropertySHEET );
#endif
uint32 sheetValue = prop();
TDataSetIndex compressedRow = TVPNodeServer::PrioContext.EntityIndex.getCompressedIndex();
uint64 value = (uint64)sheetValue | (((uint64)compressedRow) << 32);
ap->setValue64( value );
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu SHEET at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_SHEET, outbox.getPosInBit(), ap->getValue() );
// Add row into the action
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
// Include alias if non-null in mirror (only for mission giver NPCs)
CMirrorPropValueRO aliasProp( TheDataset, TVPNodeServer::PrioContext.EntityIndex, DSPropertyNPC_ALIAS );
if (aliasProp() != 0)
{
bool aliasBit = true;
outbox.serialBitAndLog( aliasBit );
outbox.serialAndLog1( const_cast(aliasProp()) ); // no need to store in history, alias never changes for an entity
}
else
{
bool aliasBit = false;
outbox.serialBitAndLog( aliasBit );
}
#ifdef TEST_LOST_PACKET
if ( TestPacketLost.get() )
{
nldebug( "This SHEET sending will be dropped..." );
TestPacketLostTimer = CTickEventHandler::getGameCycle() + 10;
TestPacketLost = false;
TestPacketLostSlot = TVPNodeServer::PrioContext.Slot;
}
#endif
}
/*
* BEHAVIOUR
*/
void fillBEHAVIOUR( TOutBox& outbox, TPropIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_BEHAVIOUR );
DECLARE_AP(BEHAVIOUR);
caseFillAction( BEHAVIOUR )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu BEHAVIOUR at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_BEHAVIOUR, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* NAME_STRING_ID
*/
void fillNAME_STRING_ID( TOutBox& outbox, TPropIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_NAME_STRING_ID );
DECLARE_AP(NAME_STRING_ID);
caseFillAction( NAME_STRING_ID )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu NAME_STRING_ID at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_NAME_STRING_ID, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* TARGET_LIST
*/
static vector TargetSlotsList(256);
NLMISC_COMMAND(displayTargetList,"Display the target list of an entity","")
{
if ( args.size() > 1 )
return false;
CEntityId entity;
entity.fromString(args[0].c_str());
CMirrorPropValueList targets(TheDataset,
entity,
"TargetList");
CMirrorPropValueList::iterator it;
it = targets.begin();
if (it != targets.end())
{
const uint32 *effectCycle = &((*it)());
log.displayNL("TargetStamp: %d", *effectCycle);
++it;
}
for (; it!=targets.end(); ++it)
{
uint32 index = (*it)();
log.displayNL("Target: %d", TDataSetRow::createFromRawIndex(index).getIndex());
}
return true;
}
void fillTARGET_LIST( TOutBox& outbox, TPropIndex )
{
CClientHost *client = TVPNodeServer::PrioContext.ClientHost;
CMirrorPropValueList targets(TheDataset,
TVPNodeServer::PrioContext.EntityIndex,
DSPropertyTARGET_LIST );
CMirrorPropValueList::iterator it;
TargetSlotsList.clear();
it = targets.begin();
for (; it!=targets.end(); )
{
TDataSetRow index = TDataSetRow::createFromRawIndex((*it)());
++it;
// check list overflow
if (it == targets.end())
break;
// distance to target (in 1/127 of 100m)
uint32 dt = *(&((*it)()));
// translate slot
TCLEntityId slot = client->IdTranslator.getCEId(index);
if (slot != INVALID_SLOT)
{
TargetSlotsList.push_back(slot);
TargetSlotsList.push_back((uint8)dt);
}
++it;
}
// serialises branch has payload
bool payLoad = true;
outbox.serialBitAndLog(payLoad);
// restricts to 256 entities
uint longListSize = TargetSlotsList.size();
if (longListSize > 32)
longListSize = 32;
uint8 listSize = (uint8)longListSize;
// serialises short size
outbox.serialAndLog1(listSize);
// serialises slot list
if (listSize > 0)
outbox.serialBuffer(&(TargetSlotsList[0]), listSize);
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_TARGET_LIST );
DECLARE_AP(TARGET_LIST);
ap->setValue64( TheDataset.getChangeTimestamp( DSPropertyTARGET_LIST, TVPNodeServer::PrioContext.EntityIndex ) );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu TARGET_LIST at bitpos %d", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_TARGET_LIST, outbox.getPosInBit() );
}
/*
* BARS
*/
void fillBARS( TOutBox& outbox, TPropIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_BARS );
DECLARE_AP(BARS);
caseFillAction( BARS )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu BARS at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_BARS, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* VPA, VPB, VPC
* Assumes:
* - They have the same mirror size
* - There mirror dataset property index is contiguous (VPC = VPB + 1 = VPA + 2)
*/
void fillVisualPropertyABC( TOutBox& outbox, TPropIndex propIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, propIndex );
DECLARE_AP_INDEX(propIndex);
CMirrorPropValueRO prop( TheDataset, TVPNodeServer::PrioContext.EntityIndex, propIndex-PROPERTY_VPA+DSPropertyVPA ); \
ap->setValue64( prop() );
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu %s at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, propIndex, CLFECOMMON::getPropText( propIndex ), outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* CONTEXTUAL
*/
void fillCONTEXTUAL( TOutBox& outbox, TPropIndex propIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_CONTEXTUAL );
DECLARE_AP(CONTEXTUAL);
caseFillAction( CONTEXTUAL )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu CONTEXTUAL at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_CONTEXTUAL, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* VISUAL_FX
*/
void fillVISUAL_FX( TOutBox& outbox, TPropIndex propIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_VISUAL_FX );
DECLARE_AP(VISUAL_FX);
caseFillAction( VISUAL_FX )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu VISUAL_FX at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_VISUAL_FX, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* MODE
*/
void fillMODE( TOutBox& outbox, TPropIndex )
{
// Fill for mode special case
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_MODE );
DECLARE_AP(MODE);
uint64 modeLong; // uses 8+4+16+16 = 44 bits
// Mode value (on 8 bits)
CMirrorPropValue prop( TheDataset, TVPNodeServer::PrioContext.EntityIndex, DSPropertyMODE );
// Store in history
ap->setValue64( prop().RawModeAndParam );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
// Game cycle when mode changed (4 bits -> 1.6 second max)
uint32 tickModeDelta = CTickEventHandler::getGameCycle() - prop.getTimestamp();
if ( tickModeDelta > 15 ) tickModeDelta = 15;
// Pack all with position 2D when mode changed, or combat angle (16 bits * 2)
/*
// OBSOLETE: MBEHAV::COMBAT_FLOAT no longer used
if ( prop().Mode == MBEHAV::COMBAT_FLOAT )
{
uint64 theta = (uint64)(*(uint32*)&(prop().Theta));
modeLong = ((uint64)(prop().Mode)) | ((uint64)(tickModeDelta << 8)) | (theta << 12);
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu MODE at bitpos %d : %u [theta=%g dt=%u]", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_MODE, outbox.getPosInBit(), prop().Mode, prop().Theta, tickModeDelta );
}
else
*/
{
uint64 posModeX16, posModeY16;
if ( TVPNodeServer::PrioContext.PositionAlreadySent )
{
// Take the pos from the mode change
posModeX16 = prop().Pos.X16;
posModeY16 = prop().Pos.Y16;
}
else
{
// Take the current pos
posModeX16 = TVPNodeServer::PrioContext.Sentity->X() >> 4; // TODO: make a method for this formula
posModeY16 = TVPNodeServer::PrioContext.Sentity->Y() >> 4;
tickModeDelta = 1;
}
modeLong = ((uint64)((uint8)(prop().Mode))) | ((uint64)(tickModeDelta << 8)) | (posModeX16 << 12) | (posModeY16 << 28);
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu MODE at bitpos %d : %u [x16=%hu y16=%hu dt=%u] %s", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_MODE, outbox.getPosInBit(), prop().Mode, prop().Pos.X16, prop().Pos.Y16, tickModeDelta, TVPNodeServer::PrioContext.PositionAlreadySent?"":" WithFirstPos" );
// The pos might be null (if mode set before 1st pos). The client has to handle the case.
/*#ifdef NL_DEBUG
if ( posModeX16==0 || posModeY16==0 )
nlwarning( "E%d: The pos16 in the mode is %hu %hu", TVPNodeServer::PrioContext.EntityIndex, (uint16)posModeX16, (uint16)posModeY16 );
#endif*/
}
// Fill
ap->setAndPackValue( modeLong, outbox );
//CActionFactory::getInstance()->remove( (CAction*&)act );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* GUILD_NAME_ID
*/
void fillGUILD_NAME_ID( TOutBox& outbox, TPropIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_GUILD_NAME_ID );
DECLARE_AP(GUILD_NAME_ID);
caseFillAction( GUILD_NAME_ID )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu GUILD_NAME_ID at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_GUILD_NAME_ID, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* GUILD_SYMBOL
*/
void fillGUILD_SYMBOL( TOutBox& outbox, TPropIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_GUILD_SYMBOL );
DECLARE_AP(GUILD_SYMBOL);
caseFillAction( GUILD_SYMBOL )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu GUILD_SYMBOL at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_GUILD_SYMBOL, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* EVENT_FACTION_ID
*/
void fillEVENT_FACTION_ID( TOutBox& outbox, TPropIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_EVENT_FACTION_ID );
DECLARE_AP(EVENT_FACTION_ID);
caseFillAction( EVENT_FACTION_ID )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu EVENT_FACTION_ID at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_EVENT_FACTION_ID, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* PVP_MODE
*/
void fillPVP_MODE( TOutBox& outbox, TPropIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_PVP_MODE );
DECLARE_AP(PVP_MODE);
caseFillAction( PVP_MODE )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu PVP_MODE at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_PVP_MODE, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* PVP_CLAN
*/
void fillPVP_CLAN( TOutBox& outbox, TPropIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, PROPERTY_PVP_CLAN );
DECLARE_AP(PVP_CLAN);
caseFillAction( PVP_CLAN )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu PVP_CLAN at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_PVP_CLAN, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* OWNER_PEOPLE
*/
void fillOWNER_PEOPLE( TOutBox& outbox, TPropIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
DECLARE_AP(OWNER_PEOPLE);
caseFillAction( OWNER_PEOPLE )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu OWNER_PEOPLE at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_OWNER_PEOPLE, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* OUTPOST_INFOS
*/
void fillOUTPOST_INFOS( TOutBox& outbox, TPropIndex )
{
bool payloadBit = true;
outbox.serialBitAndLog( payloadBit );
DECLARE_AP(OUTPOST_INFOS);
caseFillAction( OUTPOST_INFOS )
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu OUTPOST_INFOS at bitpos %d - value %"NL_I64"u", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, PROPERTY_OUTPOST_INFOS, outbox.getPosInBit(), ap->getValue() );
ap->packFast( outbox );
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
/*
* TARGET_ID, ENTITY_MOUNTED, RIDER_ENTITY
*/
void fillRowProperty( TOutBox& outbox, TPropIndex propIndex )
{
bool payloadBit = true;
TCLEntityId slot;
CMirrorPropValueRO prop( TheDataset, TVPNodeServer::PrioContext.EntityIndex, CEntityContainer::propertyIndexInDataSetToVisualPropIndex( propIndex ) );
TEntityIndex targetindex(prop());
if ( !targetindex.isValid() )
{
// No target
slot = INVALID_SLOT;
}
else
{
TEntityIndex seenEntityIndex = TVPNodeServer::PrioContext.EntityIndex;
const char *propName = getPropText( propIndex );
LOG_WHAT_IS_SENT( "%u: About to send client %hu (%s) the %s '%u --> %u'", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientHost->clientId(), TVPNodeServer::PrioContext.ClientHost->eId().toString().c_str(), propName, seenEntityIndex.getIndex(), targetindex.getIndex() );
if ( targetindex == TVPNodeServer::PrioContext.ClientHost->entityIndex() )
{
// The entity targets the client
slot = 0;
}
else if ( targetindex == seenEntityIndex )
{
// The entity targets itself
slot = INVALID_SLOT;
}
else
{
// Translate CEntityId to slot: get entityid, then slot
TCLEntityId result = TVPNodeServer::PrioContext.ClientHost->IdTranslator.getCEId( targetindex );
if ( result != INVALID_SLOT )
{
slot = result;
}
else
{
LOG_WHAT_IS_SENT( "%s slot not found: E%u", propName, targetindex.getIndex() );
payloadBit = false;
}
}
}
// Fill for property target/mount special case
outbox.serialBitAndLog( payloadBit );
if ( payloadBit )
{
LOG_WHAT_IS_SENT( "%u: Filling buffer for C%hu S%hu P%hu %s at bitpos %d - slot %hu", CTickEventHandler::getGameCycle(), TVPNodeServer::PrioContext.ClientId, (uint16)TVPNodeServer::PrioContext.Slot, propIndex, CLFECOMMON::getPropText( propIndex ), outbox.getPosInBit(), (uint16)slot );
//CActionSint64 *ap = (CActionSint64*)CActionFactory::getInstance()->createByPropIndex( TVPNodeServer::PrioContext.Slot, propIndex );
DECLARE_AP_INDEX(propIndex);
ap->setValue64( prop().getIndex() );
// Store the entity index into the history
CFrontEndService::instance()->history()->store( TVPNodeServer::PrioContext.ClientId, TVPNodeServer::PrioContext.ClientHost->sendNumber(), ap );
// The value that will be sent is the slot, not the entity index
ap->setAndPackValue( slot, outbox );
//CActionFactory::getInstance()->remove( (CAction*&)ap );
REMOVE_AP();
++(TVPNodeServer::PrioContext.ClientHost->NbActionsSentAtCycle);
}
}
TVPNodeServer::TPrioContext TVPNodeServer::PrioContext;
// Flattened Node Tree
TVPNodeServer* TVPNodeServer::FlatVPTree[CLFECOMMON::NB_VISUAL_PROPERTIES];
// Reordered Node tree (used by fastPropagateBackBranchHasPayload())
std::vector TVPNodeServer::SortedFlatVPTree;
const bool TVPNodeServer::FalseBoolPayLoad = false;
// Reordered Node tree (used by fastFillDiscreetProperties)
std::vector TVPNodeServer::SortedFlatVPTreeFill;
void TVPNodeServer::initSortedFlatVPTree()
{
if (isLeaf())
return;
CSortedFlatVPTreeItem item;
item.Node = this;
if (a())
{
item.APayLoad = &(a()->BranchHasPayload);
a()->initSortedFlatVPTree();
}
if (b())
{
item.BPayLoad = &(b()->BranchHasPayload);
b()->initSortedFlatVPTree();
}
SortedFlatVPTree.push_back(item);
}
void TVPNodeServer::initSortedFlatVPTreeFill()
{
uint thisItem = SortedFlatVPTreeFill.size();
SortedFlatVPTreeFill.push_back(CSortedFlatVPTreeFillItem());
SortedFlatVPTreeFill[thisItem].Node = this;
if (a()) a()->initSortedFlatVPTreeFill();
if (b()) b()->initSortedFlatVPTreeFill();
SortedFlatVPTreeFill[thisItem].NextIfNoPayload = SortedFlatVPTreeFill.size();
}
namespace CLFECOMMON
{
// Factory for TVPNodeBase::buildTree()
TVPNodeBase *NewNode()
{
return (TVPNodeBase*) new TVPNodeServer();
}
};
#ifdef NL_DEBUG
NLMISC_DYNVARIABLE( uint32, MoveNumber, "MoveNumber of entities seen by monitored client" )
{
if ( get )
{
CFrontEndService *fe = CFrontEndService::instance();
//nlassert( fe->MonitoredClient <= MAX_NB_CLIENTS );
CClientHost *client = fe->receiveSub()->clientIdCont()[fe->MonitoredClient];
if ( client )
{
*pointer = client->MoveNumber;
}
else
{
*pointer = 9999;
}
}
}
#endif
NLMISC_COMMAND(verbosePropertiesSent,"Turn on or off or check the state of verbose logging of what is sent"," | all | off")
{
if ( args.size() > 1 )
return false;
if ( args.size() == 1 )
{
if ( args[0] == string("all") )
verbosePropertiesSent = 0;
else if ( args[0] == string("off") )
verbosePropertiesSent = INVALID_CLIENT;
else
verbosePropertiesSent = atoi(args[0].c_str());
}
log.displayNL( "verbosePropertiesSent is %s", (verbosePropertiesSent==INVALID_CLIENT)?"off":((verbosePropertiesSent==0)?"all":toString("C%hu", verbosePropertiesSent).c_str()) );
return true;
}