1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161
|
/* This file is part of the Spring engine (GPL v2 or later), see LICENSE.html */
#include <boost/scoped_ptr.hpp>
#include <boost/shared_ptr.hpp>
// NOTE: these _must_ be included before NetProtocol.h due to some ambiguity in
// Boost hash_float.hpp ("call of overloaded ‘ldexp(float&, int&)’ is ambiguous")
#include "System/Net/UDPConnection.h"
#include "System/Net/LocalConnection.h"
#include "NetProtocol.h"
#include "Game/GameData.h"
#include "Game/GlobalUnsynced.h"
#include "Sim/Misc/GlobalConstants.h"
#include "System/Net/UnpackPacket.h"
#include "System/LoadSave/DemoRecorder.h"
#include "System/Platform/Threading.h"
#include "System/Config/ConfigHandler.h"
#include "System/GlobalConfig.h"
#include "System/Log/ILog.h"
CONFIG(int, SourcePort).defaultValue(0);
CNetProtocol::CNetProtocol() : keepUpdating(false)
{
demoRecorder.reset(NULL);
}
CNetProtocol::~CNetProtocol()
{
Send(CBaseNetProtocol::Get().SendQuit(""));
Close();
LOG("%s", serverConn->Statistics().c_str());
}
void CNetProtocol::InitClient(const char* server_addr, unsigned portnum, const std::string& myName, const std::string& myPasswd, const std::string& myVersion)
{
userName = myName;
userPasswd = myPasswd;
netcode::UDPConnection* conn = new netcode::UDPConnection(configHandler->GetInt("SourcePort"), server_addr, portnum);
conn->Unmute();
serverConn.reset(conn);
serverConn->SendData(CBaseNetProtocol::Get().SendAttemptConnect(userName, userPasswd, myVersion, globalConfig->networkLossFactor));
serverConn->Flush(true);
LOG("Connecting to %s:%i using name %s", server_addr, portnum, myName.c_str());
}
void CNetProtocol::AttemptReconnect(const std::string& myVersion)
{
netcode::UDPConnection* conn = new netcode::UDPConnection(*serverConn);
conn->Unmute();
conn->SendData(CBaseNetProtocol::Get().SendAttemptConnect(userName, userPasswd, myVersion, globalConfig->networkLossFactor, true));
conn->Flush(true);
LOG("Reconnecting to server... %ds", dynamic_cast<netcode::UDPConnection&>(*serverConn).GetReconnectSecs());
delete conn;
}
bool CNetProtocol::NeedsReconnect() {
return serverConn->NeedsReconnect();
}
void CNetProtocol::InitLocalClient()
{
serverConn.reset(new netcode::CLocalConnection);
serverConn->Flush();
LOG("Connecting to local server");
}
bool CNetProtocol::CheckTimeout(int nsecs, bool initial) const {
return serverConn->CheckTimeout(nsecs, initial);
}
bool CNetProtocol::Connected() const
{
return (serverConn->GetDataReceived() > 0);
}
std::string CNetProtocol::ConnectionStr() const
{
return serverConn->GetFullAddress();
}
boost::shared_ptr<const netcode::RawPacket> CNetProtocol::Peek(unsigned ahead) const
{
return serverConn->Peek(ahead);
}
void CNetProtocol::DeleteBufferPacketAt(unsigned index)
{
return serverConn->DeleteBufferPacketAt(index);
}
float CNetProtocol::GetPacketTime(int frameNum) const
{
if (frameNum == 0)
return gu->gameTime;
return (gu->startTime + (float)frameNum / (float)GAME_SPEED);
}
boost::shared_ptr<const netcode::RawPacket> CNetProtocol::GetData(int frameNum)
{
boost::shared_ptr<const netcode::RawPacket> ret = serverConn->GetData();
if (ret.get() == NULL) { return ret; }
if (ret->data[0] == NETMSG_GAMEDATA) { return ret; }
if (demoRecorder.get() != NULL) {
demoRecorder->SaveToDemo(ret->data, ret->length, GetPacketTime(frameNum));
}
return ret;
}
void CNetProtocol::Send(boost::shared_ptr<const netcode::RawPacket> pkt)
{
serverConn->SendData(pkt);
}
void CNetProtocol::Send(const netcode::RawPacket* pkt)
{
boost::shared_ptr<const netcode::RawPacket> ptr(pkt);
Send(ptr);
}
__FORCE_ALIGN_STACK__
void CNetProtocol::UpdateLoop()
{
Threading::SetThreadName("heartbeat");
while (keepUpdating) {
Update();
spring_msecs(100).sleep();
}
}
void CNetProtocol::Update()
{
serverConn->Update();
}
void CNetProtocol::Close(bool flush)
{
serverConn->Close(flush);
}
void CNetProtocol::SetDemoRecorder(CDemoRecorder* r) { demoRecorder.reset(r); }
CDemoRecorder* CNetProtocol::GetDemoRecorder() const { return demoRecorder.get(); }
unsigned int CNetProtocol::GetNumWaitingServerPackets() const { return (serverConn.get())->GetPacketQueueSize(); }
CNetProtocol* net = NULL;
|