// 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 .
#ifndef FRONTEND_SERVICE_H
#define FRONTEND_SERVICE_H
#include "nel/misc/time_nl.h"
#include "nel/misc/command.h"
#include "nel/misc/variable.h"
#include "nel/misc/file.h"
#include "nel/misc/stop_watch.h"
#include "nel/misc/command.h"
#include "nel/misc/hierarchical_timer.h"
#include "nel/net/login_cookie.h"
#include "nel/net/service.h"
#include "nel/net/udp_sock.h"
#include "fe_receive_sub.h"
#include "fe_send_sub.h"
#include "fe_types.h"
#include "history.h"
#include "prio_sub.h"
#include "client_id_lookup.h"
#include
#include
#define DECLARE_CF_CALLBACK( varname ) \
friend void cfcb##varname( NLMISC::CConfigFile::CVar& var )
class CModuleManager;
namespace CLFECOMMON
{
class CAction;
}
typedef CHashMap< TDataSetIndex, std::string> TEntityNamesMap;
extern NLMISC::CVariable UseWebPatchServer;
extern NLMISC::CVariable AcceptClientsAtStartup;
extern NLMISC::CVariable PatchingURLFooter;
// Return the name of an entity (if previously retrieved or "" if no name)
std::string getEntityName( const TDataSetRow& entityIndex );
// Conditional beep
void beepIfAllowed( uint freq, uint duration );
// Disconnection by UserId (requested by the login system)
void cbDisconnectClient (TUid userId, const std::string &reqServiceName);
// send an impulsion to a client
void sendImpulsion( TClientId clientid, NLNET::CMessage& msgin, uint8 channel, const char *extendedMsg, bool forceSingleShot );
//#define MEASURE_SENDING
//extern CDebugDisplayer FEDebugDisplayer;
/**
* CFrontEndService, based on IService5
*/
class CFrontEndService : public NLNET::IService, public NLMISC::ICommandsHandler
{
public:
/// Return the instance of the service
static CFrontEndService *instance() { return (CFrontEndService*)IService::getInstance(); }
/// Initialization
virtual void init();
/// Release
virtual void release();
/// Update
virtual bool update();
/// After mirror system is ready
void postInit();
CFeReceiveSub *receiveSub() { return &_ReceiveSub; }
CFeSendSub *sendSub() { return &_SendSub; }
CHistory *history() { return &_History; }
CEntityContainer& entityContainer() { return _EntityContainer; }
/// Priority Subsystem
CPrioSub PrioSub;
/// Constructor
CFrontEndService() :
ReceiveWatch(10),
SendWatch(10),
UserLWatch(10),
CycleWatch(10),
UserDurationPAverage(10),
ProcessVisionWatch(10),
BackEndRecvWatch1(5),
BackEndRecvWatch2(5),
BackEndRecvWatch3(5),
//HeadWatch(),
//FillWatch(),
//SndtWatch(),
SentActionsLastCycle(0),
//ScannedPropsLastCycle(0),
VisibilityDistance(250000), // 250 m
MonitoredClient(0),
AcceptClients(false),
PrioSub(),
_UpdateDuration(100),
_DgramLength(0),
_ClientLagTime(3000),
_ClientTimeOut(10000),
_ReceiveSub(),
_SendSub(),
_History(),
_GCCount(0),
_GCRatio(1)
{}
/// Called when there is a tick
void onTick();
/// returns the number of bit that the level can manage (biggest action it can manage)
uint getImpulseMaxBitSize(uint level) { return CImpulseEncoder::maxBitSize (level); }
/// Send an impulse to a given client (0<=level<=3)
void addImpulseToClient(TClientId client, CLFECOMMON::CActionImpulsion *action, uint level);
/// Send an impulse to a given entity, provided it is a client (0<=level<=3)
void addImpulseToEntity( const TEntityIndex& entity, CLFECOMMON::CActionImpulsion *action, uint level);
/// Remove clients that do not send datagrams anymore
void updateClientsStates();
/// Set clients to stalled or "server down" mode (and send stalled msg immediately)
void sendServerProblemStateToClients( uint connectionState );
/// Set clients to synchronize mode
void setClientsToSynchronizeState();
/// Monitor client (disabled if 0)
void monitorClient( TClientId id ) { MonitoredClient = id; }
/// Set game cycle ratio
void setGameCycleRatio( sint gcratio ) { _GCRatio = gcratio; _GCCount = 0; }
/// Callback called when a cookie become acceptable (a new player as logged
/// in and will connect here soon).
static void newCookieCallback(const NLNET::CLoginCookie &cookie);
/// StopWatch value for stats
NLMISC::CStopWatch ReceiveWatch; // All Receive Sub
NLMISC::CStopWatch SendWatch; // All Send Sub
NLMISC::CStopWatch UserLWatch;
NLMISC::CStopWatch CycleWatch;
NLMISC::TMsDuration UserDurationPAverage; // Userloop
//NLMISC::CStopWatch HeadWatch; // Sending: Setup header
//NLMISC::CStopWatch FillWatch; // Sending: Filling impulse and prioritized
//NLMISC::CStopWatch SndtWatch; // Sending: Flushing
NLMISC::CStopWatch ProcessVisionWatch; // PrioSub: Process Vision
NLMISC::CStopWatch BackEndRecvWatch1; // Netloop: cbDeltaUpdate
NLMISC::CStopWatch BackEndRecvWatch2; // Netloop: cbDeltaUpdateRemove
NLMISC::CStopWatch BackEndRecvWatch3; // Netlopp: cbDeltaNewVision
uint32 SentActionsLastCycle;
NLMISC::TTime LastTickTime;
bool StalledMode;
NLMISC::TTime LastStallTime;
/// Visibility distance
CLFECOMMON::TCoord VisibilityDistance;
/// Flag set at init and when the shard is known as down by a disconnection callback (EGS/IOS)
bool ShardDown;
/// Entity names (debugging purpose)
TEntityNamesMap EntityNames;
/// If not null, output stats about this client (debugging purpose)
TClientId MonitoredClient;
/// Accept Clients
bool AcceptClients;
protected:
/// Initialises module callback and module manager
void initModuleManagers();
DECLARE_CF_CALLBACK( PriorityMode );
DECLARE_CF_CALLBACK( TotalBandwidth );
DECLARE_CF_CALLBACK( ClientBandwidth );
DECLARE_CF_CALLBACK( LimboTimeOut );
DECLARE_CF_CALLBACK( ClientTimeOut );
DECLARE_CF_CALLBACK( AllowBeep );
DECLARE_CF_CALLBACK( GameCycleRatio );
DECLARE_CF_CALLBACK( CalcDistanceExecutionPeriod );
DECLARE_CF_CALLBACK( SortPrioExecutionPeriod );
DECLARE_CF_CALLBACK( DistanceDeltaRatioForPos );
/*DECLARE_CF_CALLBACK( PositionPrioExecutionPeriod );
DECLARE_CF_CALLBACK( OrientationPrioExecutionPeriod );
DECLARE_CF_CALLBACK( DiscreetPrioExecutionPeriod );*/
private:
CEntityContainer _EntityContainer;
/// Update duration (ms)
uint32 _UpdateDuration;
/// Length of datagrams
uint32 _DgramLength;
/// Lag time to probe for a client that does not send datagrams anymore (in ms)
uint32 _ClientLagTime;
/// Time-out to "disconnect" a client that does not send datagrams anymore (in ms)
uint32 _ClientTimeOut;
/// Time-out to "disconnect" a client that is in limbo mode and that doesn't ack
uint32 _LimboTimeOut;
/// Receive Subsystem
CFeReceiveSub _ReceiveSub;
/// Send Subsystem
CFeSendSub _SendSub;
/// Packet History
CHistory _History;
/// Module managers
std::vector< ::CModuleManager*> _ModuleManagers;
/// Counter for game cycle ratio
sint _GCCount;
/// Game cycle ratio
sint _GCRatio;
virtual const std::string &getCommandHandlerName() const
{
static std::string name("fe");
return name;
}
NLMISC_COMMAND_HANDLER_TABLE_BEGIN(CFrontEndService)
NLMISC_COMMAND_HANDLER_ADD(CFrontEndService, dump, "dump the frontend internal state", "no param");
NLMISC_COMMAND_HANDLER_TABLE_END
NLMISC_CLASS_COMMAND_DECL(dump);
};
#endif