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 162 163 164 165 166 167 168 169 170 171 172 173
|
/* This file is part of the Spring engine (GPL v2 or later), see LICENSE.html */
#include "System/Net/UDPConnection.h"
#include <SDL_timer.h>
#include <boost/scoped_ptr.hpp>
#include <boost/shared_ptr.hpp>
#include "System/mmgr.h"
// 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/LocalConnection.h"
#include "System/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/Config/ConfigHandler.h"
#include "System/GlobalConfig.h"
#include "System/Log/ILog.h"
#include "lib/gml/gmlmut.h"
CONFIG(int, SourcePort).defaultValue(0);
CNetProtocol::CNetProtocol() : loading(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)
{
GML_STDMUTEX_LOCK(net); // InitClient
netcode::UDPConnection* conn = new netcode::UDPConnection(configHandler->GetInt("SourcePort"), server_addr, portnum);
conn->Unmute();
serverConn.reset(conn);
serverConn->SendData(CBaseNetProtocol::Get().SendAttemptConnect(myName, myPasswd, 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& myName, const std::string& myPasswd, const std::string& myVersion) {
GML_STDMUTEX_LOCK(net); // AttemptReconnect
netcode::UDPConnection* conn = new netcode::UDPConnection(*serverConn);
conn->Unmute();
conn->SendData(CBaseNetProtocol::Get().SendAttemptConnect(myName, myPasswd, 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()
{
GML_STDMUTEX_LOCK(net); // 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
{
GML_STDMUTEX_LOCK(net); // Peek
return serverConn->Peek(ahead);
}
void CNetProtocol::DeleteBufferPacketAt(unsigned index)
{
GML_STDMUTEX_LOCK(net); // DeleteBufferPacketAt
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)
{
GML_STDMUTEX_LOCK(net); // GetData
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)
{
GML_STDMUTEX_LOCK(net); // Send
serverConn->SendData(pkt);
}
void CNetProtocol::Send(const netcode::RawPacket* pkt)
{
boost::shared_ptr<const netcode::RawPacket> ptr(pkt);
Send(ptr);
}
void CNetProtocol::UpdateLoop()
{
loading = true;
while (loading) {
Update();
SDL_Delay(400);
}
}
void CNetProtocol::Update()
{
GML_STDMUTEX_LOCK(net); // Update
serverConn->Update();
}
void CNetProtocol::Close(bool flush) {
GML_STDMUTEX_LOCK(net); // Close
serverConn->Close(flush);
}
void CNetProtocol::SetDemoRecorder(CDemoRecorder* r) { demoRecorder.reset(r); }
CDemoRecorder* CNetProtocol::GetDemoRecorder() const { return demoRecorder.get(); }
CNetProtocol* net = NULL;
|