2016-09-10 03:28:11 +03:00
|
|
|
/*
|
|
|
|
* Connection.cpp, part of VCMI engine
|
|
|
|
*
|
|
|
|
* Authors: listed in file AUTHORS in main folder
|
|
|
|
*
|
|
|
|
* License: GNU General Public License v2.0 or later
|
|
|
|
* Full text of license available in license.txt file, in main folder
|
|
|
|
*
|
|
|
|
*/
|
2017-07-13 11:26:03 +03:00
|
|
|
#include "StdInc.h"
|
|
|
|
#include "Connection.h"
|
|
|
|
|
|
|
|
#include "../registerTypes/RegisterTypes.h"
|
|
|
|
#include "../mapping/CMap.h"
|
|
|
|
#include "../CGameState.h"
|
|
|
|
|
|
|
|
#include <boost/asio.hpp>
|
2016-09-10 03:28:11 +03:00
|
|
|
|
|
|
|
using namespace boost;
|
|
|
|
using namespace boost::asio::ip;
|
|
|
|
|
|
|
|
#if defined(__hppa__) || \
|
|
|
|
defined(__m68k__) || defined(mc68000) || defined(_M_M68K) || \
|
|
|
|
(defined(__MIPS__) && defined(__MISPEB__)) || \
|
|
|
|
defined(__ppc__) || defined(__POWERPC__) || defined(_M_PPC) || \
|
|
|
|
defined(__sparc__)
|
|
|
|
#define BIG_ENDIAN
|
|
|
|
#else
|
|
|
|
#define LIL_ENDIAN
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
void CConnection::init()
|
|
|
|
{
|
|
|
|
boost::asio::ip::tcp::no_delay option(true);
|
|
|
|
socket->set_option(option);
|
|
|
|
|
2016-11-13 12:44:46 +03:00
|
|
|
enableSmartPointerSerialization();
|
2016-09-10 03:28:11 +03:00
|
|
|
disableStackSendingByID();
|
|
|
|
registerTypes(iser);
|
|
|
|
registerTypes(oser);
|
|
|
|
#ifdef LIL_ENDIAN
|
|
|
|
myEndianess = true;
|
|
|
|
#else
|
|
|
|
myEndianess = false;
|
|
|
|
#endif
|
|
|
|
connected = true;
|
|
|
|
std::string pom;
|
|
|
|
//we got connection
|
|
|
|
oser & std::string("Aiya!\n") & name & myEndianess; //identify ourselves
|
|
|
|
iser & pom & pom & contactEndianess;
|
2017-08-11 20:03:05 +03:00
|
|
|
logNetwork->info("Established connection with %s", pom);
|
2017-07-16 12:58:05 +03:00
|
|
|
wmx = new boost::mutex();
|
|
|
|
rmx = new boost::mutex();
|
2016-09-10 03:28:11 +03:00
|
|
|
|
|
|
|
handler = nullptr;
|
|
|
|
receivedStop = sendStop = false;
|
|
|
|
static int cid = 1;
|
|
|
|
connectionID = cid++;
|
2016-10-29 19:52:19 +03:00
|
|
|
iser.fileVersion = SERIALIZATION_VERSION;
|
2016-09-10 03:28:11 +03:00
|
|
|
}
|
|
|
|
|
2017-06-04 08:49:23 +03:00
|
|
|
CConnection::CConnection(std::string host, ui16 port, std::string Name)
|
2016-09-10 03:28:11 +03:00
|
|
|
:iser(this), oser(this), io_service(new asio::io_service), name(Name)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
boost::system::error_code error = asio::error::host_not_found;
|
|
|
|
socket = new tcp::socket(*io_service);
|
|
|
|
tcp::resolver resolver(*io_service);
|
2017-06-04 16:48:04 +03:00
|
|
|
tcp::resolver::iterator end, pom, endpoint_iterator = resolver.resolve(tcp::resolver::query(host, std::to_string(port)),error);
|
2016-09-10 03:28:11 +03:00
|
|
|
if(error)
|
|
|
|
{
|
2017-08-11 20:03:05 +03:00
|
|
|
logNetwork->error("Problem with resolving: \n%s", error.message());
|
2016-09-10 03:28:11 +03:00
|
|
|
goto connerror1;
|
|
|
|
}
|
|
|
|
pom = endpoint_iterator;
|
|
|
|
if(pom != end)
|
2017-08-10 19:39:27 +03:00
|
|
|
logNetwork->info("Found endpoints:");
|
2016-09-10 03:28:11 +03:00
|
|
|
else
|
|
|
|
{
|
2017-08-10 19:39:27 +03:00
|
|
|
logNetwork->error("Critical problem: No endpoints found!");
|
2016-09-10 03:28:11 +03:00
|
|
|
goto connerror1;
|
|
|
|
}
|
|
|
|
i=0;
|
|
|
|
while(pom != end)
|
|
|
|
{
|
2017-08-11 20:03:05 +03:00
|
|
|
logNetwork->info("\t%d:%s", i, (boost::asio::ip::tcp::endpoint&)*pom);
|
2016-09-10 03:28:11 +03:00
|
|
|
pom++;
|
|
|
|
}
|
|
|
|
i=0;
|
|
|
|
while(endpoint_iterator != end)
|
|
|
|
{
|
2017-08-11 20:03:05 +03:00
|
|
|
logNetwork->info("Trying connection to %s(%d)", (boost::asio::ip::tcp::endpoint&)*endpoint_iterator, i++);
|
2016-09-10 03:28:11 +03:00
|
|
|
socket->connect(*endpoint_iterator, error);
|
|
|
|
if(!error)
|
|
|
|
{
|
|
|
|
init();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-08-11 20:03:05 +03:00
|
|
|
logNetwork->error("Problem with connecting: %s", error.message());
|
2016-09-10 03:28:11 +03:00
|
|
|
}
|
|
|
|
endpoint_iterator++;
|
|
|
|
}
|
|
|
|
|
|
|
|
//we shouldn't be here - error handling
|
|
|
|
connerror1:
|
2017-08-10 19:39:27 +03:00
|
|
|
logNetwork->error("Something went wrong... checking for error info");
|
2016-09-10 03:28:11 +03:00
|
|
|
if(error)
|
2017-08-11 20:03:05 +03:00
|
|
|
logNetwork->error(error.message());
|
2016-09-10 03:28:11 +03:00
|
|
|
else
|
2017-08-10 19:39:27 +03:00
|
|
|
logNetwork->error("No error info. ");
|
2016-09-10 03:28:11 +03:00
|
|
|
delete io_service;
|
|
|
|
//delete socket;
|
|
|
|
throw std::runtime_error("Can't establish connection :(");
|
|
|
|
}
|
|
|
|
CConnection::CConnection(TSocket * Socket, std::string Name )
|
|
|
|
:iser(this), oser(this), socket(Socket),io_service(&Socket->get_io_service()), name(Name)//, send(this), rec(this)
|
|
|
|
{
|
|
|
|
init();
|
|
|
|
}
|
|
|
|
CConnection::CConnection(TAcceptor * acceptor, boost::asio::io_service *Io_service, std::string Name)
|
|
|
|
: iser(this), oser(this), name(Name)//, send(this), rec(this)
|
|
|
|
{
|
|
|
|
boost::system::error_code error = asio::error::host_not_found;
|
|
|
|
socket = new tcp::socket(*io_service);
|
|
|
|
acceptor->accept(*socket,error);
|
|
|
|
if (error)
|
|
|
|
{
|
2017-08-11 20:03:05 +03:00
|
|
|
logNetwork->error("Error on accepting: %s", error.message());
|
2016-09-10 03:28:11 +03:00
|
|
|
delete socket;
|
|
|
|
throw std::runtime_error("Can't establish connection :(");
|
|
|
|
}
|
|
|
|
init();
|
|
|
|
}
|
|
|
|
int CConnection::write(const void * data, unsigned size)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
ret = asio::write(*socket,asio::const_buffers_1(asio::const_buffer(data,size)));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
//connection has been lost
|
|
|
|
connected = false;
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int CConnection::read(void * data, unsigned size)
|
|
|
|
{
|
|
|
|
try
|
|
|
|
{
|
|
|
|
int ret = asio::read(*socket,asio::mutable_buffers_1(asio::mutable_buffer(data,size)));
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
//connection has been lost
|
|
|
|
connected = false;
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
CConnection::~CConnection(void)
|
|
|
|
{
|
|
|
|
if(handler)
|
|
|
|
handler->join();
|
|
|
|
|
|
|
|
delete handler;
|
|
|
|
|
|
|
|
close();
|
|
|
|
delete io_service;
|
|
|
|
delete wmx;
|
|
|
|
delete rmx;
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
CConnection & CConnection::operator&(const T &t) {
|
|
|
|
// throw std::exception();
|
|
|
|
//XXX this is temporaly ? solution to fix gcc (4.3.3, other?) compilation
|
|
|
|
// problem for more details contact t0@czlug.icis.pcz.pl or impono@gmail.com
|
|
|
|
// do not remove this exception it shoudnt be called
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::close()
|
|
|
|
{
|
|
|
|
if(socket)
|
|
|
|
{
|
|
|
|
socket->close();
|
2016-12-04 12:54:26 +03:00
|
|
|
vstd::clear_pointer(socket);
|
2016-09-10 03:28:11 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CConnection::isOpen() const
|
|
|
|
{
|
|
|
|
return socket && connected;
|
|
|
|
}
|
|
|
|
|
2016-10-30 02:43:06 +03:00
|
|
|
bool CConnection::isHost() const
|
|
|
|
{
|
|
|
|
return connectionID == 1;
|
|
|
|
}
|
|
|
|
|
2017-08-11 20:03:05 +03:00
|
|
|
void CConnection::reportState(vstd::CLoggerBase * out)
|
2016-09-10 03:28:11 +03:00
|
|
|
{
|
2017-08-10 19:39:27 +03:00
|
|
|
out->debug("CConnection");
|
2016-09-10 03:28:11 +03:00
|
|
|
if(socket && socket->is_open())
|
|
|
|
{
|
2017-08-10 19:39:27 +03:00
|
|
|
out->debug("\tWe have an open and valid socket");
|
2017-08-11 20:03:05 +03:00
|
|
|
out->debug("\t %d bytes awaiting", socket->available());
|
2016-09-10 03:28:11 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CPack * CConnection::retreivePack()
|
|
|
|
{
|
|
|
|
CPack *ret = nullptr;
|
|
|
|
boost::unique_lock<boost::mutex> lock(*rmx);
|
2017-08-10 19:39:27 +03:00
|
|
|
logNetwork->trace("Listening... ");
|
2016-09-10 03:28:11 +03:00
|
|
|
iser & ret;
|
2017-08-11 20:03:05 +03:00
|
|
|
logNetwork->trace("\treceived server message of type %s", (ret? typeid(*ret).name() : "nullptr"));
|
2016-09-10 03:28:11 +03:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::sendPackToServer(const CPack &pack, PlayerColor player, ui32 requestID)
|
|
|
|
{
|
|
|
|
boost::unique_lock<boost::mutex> lock(*wmx);
|
2017-08-11 20:03:05 +03:00
|
|
|
logNetwork->trace("Sending to server a pack of type %s", typeid(pack).name());
|
2016-09-10 03:28:11 +03:00
|
|
|
oser & player & requestID & &pack; //packs has to be sent as polymorphic pointers!
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::disableStackSendingByID()
|
|
|
|
{
|
|
|
|
CSerializer::sendStackInstanceByIds = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::enableStackSendingByID()
|
|
|
|
{
|
|
|
|
CSerializer::sendStackInstanceByIds = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::disableSmartPointerSerialization()
|
|
|
|
{
|
|
|
|
iser.smartPointerSerialization = oser.smartPointerSerialization = false;
|
|
|
|
}
|
|
|
|
|
2016-11-13 12:44:46 +03:00
|
|
|
void CConnection::enableSmartPointerSerialization()
|
2016-09-10 03:28:11 +03:00
|
|
|
{
|
|
|
|
iser.smartPointerSerialization = oser.smartPointerSerialization = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::prepareForSendingHeroes()
|
|
|
|
{
|
|
|
|
iser.loadedPointers.clear();
|
|
|
|
oser.savedPointers.clear();
|
|
|
|
disableSmartVectorMemberSerialization();
|
2016-11-13 12:44:46 +03:00
|
|
|
enableSmartPointerSerialization();
|
2016-09-10 03:28:11 +03:00
|
|
|
disableStackSendingByID();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::enterPregameConnectionMode()
|
|
|
|
{
|
|
|
|
iser.loadedPointers.clear();
|
|
|
|
oser.savedPointers.clear();
|
|
|
|
disableSmartVectorMemberSerialization();
|
|
|
|
disableSmartPointerSerialization();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::disableSmartVectorMemberSerialization()
|
|
|
|
{
|
|
|
|
CSerializer::smartVectorMembersSerialization = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::enableSmartVectorMemberSerializatoin()
|
|
|
|
{
|
|
|
|
CSerializer::smartVectorMembersSerialization = true;
|
|
|
|
}
|
|
|
|
|
2017-08-11 16:50:00 +03:00
|
|
|
std::string CConnection::toString() const
|
|
|
|
{
|
|
|
|
boost::format fmt("Connection with %s (ID: %d)");
|
|
|
|
fmt % name % connectionID;
|
|
|
|
return fmt.str();
|
|
|
|
}
|