2011-12-14 00:23:17 +03:00
|
|
|
#include "StdInc.h"
|
2008-07-03 18:03:32 +03:00
|
|
|
#include "Connection.h"
|
2009-10-06 03:32:33 +03:00
|
|
|
|
2013-01-03 15:19:20 +03:00
|
|
|
#include "RegisterTypes.h"
|
2009-10-06 03:32:33 +03:00
|
|
|
|
2011-12-14 00:23:17 +03:00
|
|
|
#include <boost/asio.hpp>
|
2009-10-06 03:32:33 +03:00
|
|
|
|
2009-04-15 17:03:31 +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
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2008-07-03 18:03:32 +03:00
|
|
|
using namespace boost;
|
|
|
|
using namespace boost::asio::ip;
|
2013-02-23 15:22:23 +03:00
|
|
|
|
|
|
|
extern template void registerTypes<CISer<CConnection> >(CISer<CConnection>& s);
|
|
|
|
extern template void registerTypes<COSer<CConnection> >(COSer<CConnection>& s);
|
|
|
|
extern template void registerTypes<CSaveFile>(CSaveFile & s);
|
|
|
|
extern template void registerTypes<CLoadFile>(CLoadFile & s);
|
|
|
|
extern template void registerTypes<CTypeList>(CTypeList & s);
|
|
|
|
extern template void registerTypes<CLoadIntegrityValidator>(CLoadIntegrityValidator & s);
|
2008-07-03 18:03:32 +03:00
|
|
|
|
2009-03-07 00:11:17 +02:00
|
|
|
CTypeList typeList;
|
|
|
|
|
2008-07-03 18:03:32 +03:00
|
|
|
#define LOG(a) \
|
|
|
|
if(logging)\
|
|
|
|
out << a
|
|
|
|
#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
|
|
|
|
|
2008-07-09 20:22:28 +03:00
|
|
|
void CConnection::init()
|
2008-07-03 18:03:32 +03:00
|
|
|
{
|
2012-09-25 21:00:55 +03:00
|
|
|
enableSmartPointerSerializatoin();
|
|
|
|
disableStackSendingByID();
|
2009-03-07 17:54:12 +02:00
|
|
|
registerTypes(static_cast<CISer<CConnection>&>(*this));
|
|
|
|
registerTypes(static_cast<COSer<CConnection>&>(*this));
|
2008-07-03 18:03:32 +03:00
|
|
|
#ifdef LIL_ENDIAN
|
|
|
|
myEndianess = true;
|
|
|
|
#else
|
|
|
|
myEndianess = false;
|
|
|
|
#endif
|
2008-07-09 20:22:28 +03:00
|
|
|
connected = true;
|
|
|
|
std::string pom;
|
|
|
|
//we got connection
|
|
|
|
(*this) << std::string("Aiya!\n") << name << myEndianess; //identify ourselves
|
|
|
|
(*this) >> pom >> pom >> contactEndianess;
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->infoStream() << "Established connection with "<<pom;
|
2008-07-26 16:57:32 +03:00
|
|
|
wmx = new boost::mutex;
|
|
|
|
rmx = new boost::mutex;
|
2010-10-24 14:35:14 +03:00
|
|
|
|
2013-06-26 14:18:27 +03:00
|
|
|
handler = nullptr;
|
2010-10-24 14:35:14 +03:00
|
|
|
receivedStop = sendStop = false;
|
|
|
|
static int cid = 1;
|
|
|
|
connectionID = cid++;
|
2008-07-09 20:22:28 +03:00
|
|
|
}
|
|
|
|
|
2009-01-11 00:08:18 +02:00
|
|
|
CConnection::CConnection(std::string host, std::string port, std::string Name)
|
|
|
|
:io_service(new asio::io_service), name(Name)
|
2008-07-09 20:22:28 +03:00
|
|
|
{
|
2008-09-10 00:10:24 +03:00
|
|
|
int i;
|
2008-08-04 12:05:52 +03:00
|
|
|
boost::system::error_code error = asio::error::host_not_found;
|
2008-07-03 18:03:32 +03:00
|
|
|
socket = new tcp::socket(*io_service);
|
|
|
|
tcp::resolver resolver(*io_service);
|
2008-09-07 06:38:37 +03:00
|
|
|
tcp::resolver::iterator end, pom, endpoint_iterator = resolver.resolve(tcp::resolver::query(host,port),error);
|
|
|
|
if(error)
|
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->errorStream() << "Problem with resolving: \n" << error;
|
2008-09-07 06:38:37 +03:00
|
|
|
goto connerror1;
|
|
|
|
}
|
|
|
|
pom = endpoint_iterator;
|
|
|
|
if(pom != end)
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->infoStream()<<"Found endpoints:";
|
2008-09-07 06:38:37 +03:00
|
|
|
else
|
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->errorStream() << "Critical problem: No endpoints found!";
|
2008-09-07 06:38:37 +03:00
|
|
|
goto connerror1;
|
|
|
|
}
|
2008-09-10 00:10:24 +03:00
|
|
|
i=0;
|
2008-09-07 06:38:37 +03:00
|
|
|
while(pom != end)
|
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->infoStream() << "\t" << i << ": " << (boost::asio::ip::tcp::endpoint&)*pom;
|
2008-09-07 06:38:37 +03:00
|
|
|
pom++;
|
|
|
|
}
|
2008-09-09 10:05:02 +03:00
|
|
|
i=0;
|
2008-09-07 06:38:37 +03:00
|
|
|
while(endpoint_iterator != end)
|
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->infoStream() << "Trying connection to " << (boost::asio::ip::tcp::endpoint&)*endpoint_iterator << " (" << i++ << ")";
|
2008-09-07 06:38:37 +03:00
|
|
|
socket->connect(*endpoint_iterator, error);
|
|
|
|
if(!error)
|
|
|
|
{
|
|
|
|
init();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->errorStream() << "Problem with connecting: " << error;
|
2008-09-07 06:38:37 +03:00
|
|
|
}
|
2008-09-09 10:05:02 +03:00
|
|
|
endpoint_iterator++;
|
2008-09-07 06:38:37 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
//we shouldn't be here - error handling
|
|
|
|
connerror1:
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->errorStream() << "Something went wrong... checking for error info";
|
2008-09-07 06:38:37 +03:00
|
|
|
if(error)
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->errorStream() << error;
|
2008-09-07 06:38:37 +03:00
|
|
|
else
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->errorStream() << "No error info. ";
|
2008-09-07 06:38:37 +03:00
|
|
|
delete io_service;
|
2008-09-09 10:05:02 +03:00
|
|
|
//delete socket;
|
2012-04-22 10:32:45 +03:00
|
|
|
throw std::runtime_error("Can't establish connection :(");
|
2008-07-03 18:03:32 +03:00
|
|
|
}
|
2010-05-15 05:26:49 +03:00
|
|
|
CConnection::CConnection(TSocket * Socket, std::string Name )
|
2011-08-05 02:49:32 +03:00
|
|
|
:socket(Socket),io_service(&Socket->get_io_service()), name(Name)//, send(this), rec(this)
|
2008-07-03 18:03:32 +03:00
|
|
|
{
|
2008-07-09 20:22:28 +03:00
|
|
|
init();
|
|
|
|
}
|
2010-05-15 05:26:49 +03:00
|
|
|
CConnection::CConnection(TAcceptor * acceptor, boost::asio::io_service *Io_service, std::string Name)
|
2009-01-11 00:08:18 +02:00
|
|
|
: name(Name)//, send(this), rec(this)
|
2008-07-09 20:22:28 +03:00
|
|
|
{
|
2008-08-04 12:05:52 +03:00
|
|
|
boost::system::error_code error = asio::error::host_not_found;
|
2008-07-09 20:22:28 +03:00
|
|
|
socket = new tcp::socket(*io_service);
|
|
|
|
acceptor->accept(*socket,error);
|
2008-09-09 10:05:02 +03:00
|
|
|
if (error)
|
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->errorStream() << "Error on accepting: " << error;
|
2008-09-09 10:05:02 +03:00
|
|
|
delete socket;
|
2012-04-22 10:32:45 +03:00
|
|
|
throw std::runtime_error("Can't establish connection :(");
|
2008-09-09 10:05:02 +03:00
|
|
|
}
|
2008-07-09 20:22:28 +03:00
|
|
|
init();
|
2008-07-03 18:03:32 +03:00
|
|
|
}
|
|
|
|
int CConnection::write(const void * data, unsigned size)
|
|
|
|
{
|
2008-09-17 13:18:22 +03:00
|
|
|
//LOG("Sending " << size << " byte(s) of data" <<std::endl);
|
2010-01-29 22:52:45 +02:00
|
|
|
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;
|
|
|
|
}
|
2008-07-03 18:03:32 +03:00
|
|
|
}
|
|
|
|
int CConnection::read(void * data, unsigned size)
|
|
|
|
{
|
2008-09-17 13:18:22 +03:00
|
|
|
//LOG("Receiving " << size << " byte(s) of data" <<std::endl);
|
2010-01-29 22:52:45 +02:00
|
|
|
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;
|
|
|
|
}
|
2008-07-03 18:03:32 +03:00
|
|
|
}
|
|
|
|
CConnection::~CConnection(void)
|
|
|
|
{
|
2010-10-24 14:35:14 +03:00
|
|
|
if(handler)
|
|
|
|
handler->join();
|
|
|
|
|
|
|
|
delete handler;
|
|
|
|
|
2008-08-25 13:25:16 +03:00
|
|
|
close();
|
2010-01-29 22:52:45 +02:00
|
|
|
delete io_service;
|
|
|
|
delete wmx;
|
|
|
|
delete rmx;
|
2008-07-03 18:03:32 +03:00
|
|
|
}
|
2008-08-04 12:05:52 +03:00
|
|
|
|
2009-03-16 00:57:04 +02:00
|
|
|
template<class T>
|
|
|
|
CConnection & CConnection::operator&(const T &t) {
|
2012-11-20 20:53:45 +03:00
|
|
|
throw std::exception();
|
2009-03-16 00:57:04 +02:00
|
|
|
//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
|
2012-11-20 20:53:45 +03:00
|
|
|
return *this;
|
2009-03-16 00:57:04 +02:00
|
|
|
}
|
|
|
|
|
2008-08-25 13:25:16 +03:00
|
|
|
void CConnection::close()
|
|
|
|
{
|
|
|
|
if(socket)
|
|
|
|
{
|
|
|
|
socket->close();
|
|
|
|
delete socket;
|
2013-06-26 14:18:27 +03:00
|
|
|
socket = nullptr;
|
2008-08-25 13:25:16 +03:00
|
|
|
}
|
|
|
|
}
|
2008-11-16 03:06:15 +02:00
|
|
|
|
2010-01-29 22:52:45 +02:00
|
|
|
bool CConnection::isOpen() const
|
|
|
|
{
|
|
|
|
return socket && connected;
|
|
|
|
}
|
|
|
|
|
2013-04-11 18:58:01 +03:00
|
|
|
void CConnection::reportState(CLogger * out)
|
2010-07-14 05:53:21 +03:00
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
out->debugStream() << "CConnection";
|
2010-07-14 05:53:21 +03:00
|
|
|
if(socket && socket->is_open())
|
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
out->debugStream() << "\tWe have an open and valid socket";
|
|
|
|
out->debugStream() << "\t" << socket->available() <<" bytes awaiting";
|
2010-07-14 05:53:21 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-03 21:42:54 +03:00
|
|
|
CPack * CConnection::retreivePack()
|
|
|
|
{
|
2013-06-26 14:18:27 +03:00
|
|
|
CPack *ret = nullptr;
|
2010-09-03 21:42:54 +03:00
|
|
|
boost::unique_lock<boost::mutex> lock(*rmx);
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->traceStream() << "Listening... ";
|
2010-09-03 21:42:54 +03:00
|
|
|
*this >> ret;
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->traceStream() << "\treceived server message of type " << typeid(*ret).name();
|
2010-09-03 21:42:54 +03:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-03-03 20:06:03 +03:00
|
|
|
void CConnection::sendPackToServer(const CPack &pack, PlayerColor player, ui32 requestID)
|
2011-08-25 18:24:37 +03:00
|
|
|
{
|
|
|
|
boost::unique_lock<boost::mutex> lock(*wmx);
|
2013-04-09 17:31:36 +03:00
|
|
|
logNetwork->traceStream() << "Sending to server a pack of type " << typeid(pack).name();
|
2012-03-26 01:46:14 +03:00
|
|
|
*this << player << requestID << &pack; //packs has to be sent as polymorphic pointers!
|
2011-08-25 18:24:37 +03:00
|
|
|
}
|
|
|
|
|
2012-09-25 21:00:55 +03:00
|
|
|
void CConnection::disableStackSendingByID()
|
|
|
|
{
|
|
|
|
CISer<CConnection>::sendStackInstanceByIds = false;
|
|
|
|
COSer<CConnection>::sendStackInstanceByIds = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::enableStackSendingByID()
|
|
|
|
{
|
|
|
|
CISer<CConnection>::sendStackInstanceByIds = true;
|
|
|
|
COSer<CConnection>::sendStackInstanceByIds = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::disableSmartPointerSerialization()
|
|
|
|
{
|
|
|
|
CISer<CConnection>::smartPointerSerialization = false;
|
|
|
|
COSer<CConnection>::smartPointerSerialization = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::enableSmartPointerSerializatoin()
|
|
|
|
{
|
|
|
|
CISer<CConnection>::smartPointerSerialization = true;
|
|
|
|
COSer<CConnection>::smartPointerSerialization = true;
|
|
|
|
}
|
|
|
|
|
2013-02-09 21:18:55 +03:00
|
|
|
void CConnection::prepareForSendingHeroes()
|
|
|
|
{
|
|
|
|
loadedPointers.clear();
|
|
|
|
savedPointers.clear();
|
|
|
|
CISer<CConnection>::smartVectorMembersSerialization = false;
|
|
|
|
COSer<CConnection>::smartVectorMembersSerialization = false;
|
2013-02-14 18:19:35 +03:00
|
|
|
enableSmartPointerSerializatoin();
|
|
|
|
}
|
|
|
|
|
|
|
|
void CConnection::enterPregameConnectionMode()
|
|
|
|
{
|
|
|
|
loadedPointers.clear();
|
|
|
|
savedPointers.clear();
|
|
|
|
CISer<CConnection>::smartVectorMembersSerialization = false;
|
|
|
|
COSer<CConnection>::smartVectorMembersSerialization = false;
|
|
|
|
disableSmartPointerSerialization();
|
2013-02-09 21:18:55 +03:00
|
|
|
}
|
|
|
|
|
2008-11-16 03:06:15 +02:00
|
|
|
CSaveFile::CSaveFile( const std::string &fname )
|
2008-08-04 12:05:52 +03:00
|
|
|
{
|
2009-03-07 00:11:17 +02:00
|
|
|
registerTypes(*this);
|
2010-02-26 13:18:09 +02:00
|
|
|
openNextFile(fname);
|
|
|
|
}
|
|
|
|
|
|
|
|
CSaveFile::~CSaveFile()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
int CSaveFile::write( const void * data, unsigned size )
|
|
|
|
{
|
|
|
|
sfile->write((char *)data,size);
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CSaveFile::openNextFile(const std::string &fname)
|
|
|
|
{
|
2010-07-14 05:53:21 +03:00
|
|
|
fName = fname;
|
2012-06-09 22:58:17 +03:00
|
|
|
try
|
2009-06-02 01:31:11 +03:00
|
|
|
{
|
2012-06-09 22:58:17 +03:00
|
|
|
sfile = make_unique<std::ofstream>(fname.c_str(), std::ios::binary);
|
|
|
|
sfile->exceptions(std::ifstream::failbit | std::ifstream::badbit); //we throw a lot anyway
|
|
|
|
|
|
|
|
if(!(*sfile))
|
|
|
|
THROW_FORMAT("Error: cannot open to write %s!", fname);
|
|
|
|
|
2009-06-02 01:31:11 +03:00
|
|
|
sfile->write("VCMI",4); //write magic identifier
|
|
|
|
*this << version; //write format version
|
|
|
|
}
|
2012-06-09 22:58:17 +03:00
|
|
|
catch(...)
|
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
logGlobal->errorStream() << "Failed to save to " << fname;
|
2012-06-09 22:58:17 +03:00
|
|
|
clear();
|
|
|
|
throw;
|
|
|
|
}
|
2008-08-04 12:05:52 +03:00
|
|
|
}
|
|
|
|
|
2013-04-11 18:58:01 +03:00
|
|
|
void CSaveFile::reportState(CLogger * out)
|
2010-07-14 05:53:21 +03:00
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
out->debugStream() << "CSaveFile";
|
2012-04-08 04:15:18 +03:00
|
|
|
if(sfile.get() && *sfile)
|
2010-07-14 05:53:21 +03:00
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
out->debugStream() << "\tOpened " << fName << "\n\tPosition: " << sfile->tellp();
|
2010-07-14 05:53:21 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-09 22:58:17 +03:00
|
|
|
void CSaveFile::clear()
|
|
|
|
{
|
|
|
|
fName.clear();
|
|
|
|
sfile = nullptr;
|
|
|
|
}
|
|
|
|
|
2013-02-19 01:37:22 +03:00
|
|
|
void CSaveFile::putMagicBytes( const std::string &text )
|
|
|
|
{
|
|
|
|
write(text.c_str(), text.length());
|
|
|
|
}
|
|
|
|
|
2010-10-24 14:35:14 +03:00
|
|
|
CLoadFile::CLoadFile(const std::string &fname, int minimalVersion /*= version*/)
|
2010-02-26 13:18:09 +02:00
|
|
|
{
|
|
|
|
registerTypes(*this);
|
2010-10-24 14:35:14 +03:00
|
|
|
openNextFile(fname, minimalVersion);
|
2010-02-26 13:18:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
CLoadFile::~CLoadFile()
|
2008-08-04 12:05:52 +03:00
|
|
|
{
|
|
|
|
}
|
2008-11-16 03:06:15 +02:00
|
|
|
|
2010-02-26 13:18:09 +02:00
|
|
|
int CLoadFile::read( const void * data, unsigned size )
|
2008-11-16 03:06:15 +02:00
|
|
|
{
|
2012-06-08 03:12:31 +03:00
|
|
|
sfile->read((char *)data,size);
|
2008-11-16 03:06:15 +02:00
|
|
|
return size;
|
2009-01-06 20:42:20 +02:00
|
|
|
}
|
|
|
|
|
2010-10-24 14:35:14 +03:00
|
|
|
void CLoadFile::openNextFile(const std::string &fname, int minimalVersion)
|
2010-02-26 13:18:09 +02:00
|
|
|
{
|
2012-06-08 03:12:31 +03:00
|
|
|
assert(!reverseEndianess);
|
|
|
|
assert(minimalVersion <= version);
|
|
|
|
|
|
|
|
try
|
2009-06-02 01:31:11 +03:00
|
|
|
{
|
2012-06-08 03:12:31 +03:00
|
|
|
fName = fname;
|
|
|
|
sfile = make_unique<std::ifstream>(fname, std::ios::binary);
|
|
|
|
sfile->exceptions(std::ifstream::failbit | std::ifstream::badbit); //we throw a lot anyway
|
|
|
|
|
|
|
|
if(!(*sfile))
|
|
|
|
THROW_FORMAT("Error: cannot open to read %s!", fname);
|
|
|
|
|
|
|
|
//we can read
|
2009-06-02 01:31:11 +03:00
|
|
|
char buffer[4];
|
|
|
|
sfile->read(buffer, 4);
|
|
|
|
if(std::memcmp(buffer,"VCMI",4))
|
2012-06-08 03:12:31 +03:00
|
|
|
THROW_FORMAT("Error: not a VCMI file(%s)!", fname);
|
2009-06-02 01:31:11 +03:00
|
|
|
|
2012-06-08 03:12:31 +03:00
|
|
|
*this >> fileVersion;
|
|
|
|
if(fileVersion < minimalVersion)
|
|
|
|
THROW_FORMAT("Error: too old file format (%s)!", fname);
|
|
|
|
|
|
|
|
if(fileVersion > version)
|
2012-04-17 15:46:21 +03:00
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
logGlobal->warnStream() << boost::format("Warning format version mismatch: found %d when current is %d! (file %s)\n") % fileVersion % version % fname;
|
2012-06-08 03:12:31 +03:00
|
|
|
|
|
|
|
auto versionptr = (char*)&fileVersion;
|
2012-06-01 18:59:26 +03:00
|
|
|
std::reverse(versionptr, versionptr + 4);
|
2013-04-09 17:31:36 +03:00
|
|
|
logGlobal->warnStream() << "Version number reversed is " << fileVersion << ", checking...";
|
2012-06-08 03:12:31 +03:00
|
|
|
|
|
|
|
if(fileVersion == version)
|
2012-06-01 18:59:26 +03:00
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
logGlobal->warnStream() << fname << " seems to have different endianess! Entering reversing mode.";
|
2012-06-01 18:59:26 +03:00
|
|
|
reverseEndianess = true;
|
|
|
|
}
|
|
|
|
else
|
2012-06-08 03:12:31 +03:00
|
|
|
THROW_FORMAT("Error: too new file format (%s)!", fname);
|
2009-06-02 01:31:11 +03:00
|
|
|
}
|
|
|
|
}
|
2012-06-08 03:12:31 +03:00
|
|
|
catch(...)
|
|
|
|
{
|
|
|
|
clear(); //if anything went wrong, we delete file and rethrow
|
|
|
|
throw;
|
|
|
|
}
|
2009-01-06 20:42:20 +02:00
|
|
|
}
|
|
|
|
|
2013-04-11 18:58:01 +03:00
|
|
|
void CLoadFile::reportState(CLogger * out)
|
2010-07-14 05:53:21 +03:00
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
out->debugStream() << "CLoadFile";
|
2012-04-08 04:15:18 +03:00
|
|
|
if(!!sfile && *sfile)
|
2010-07-14 05:53:21 +03:00
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
out->debugStream() << "\tOpened " << fName << "\n\tPosition: " << sfile->tellg();
|
2010-07-14 05:53:21 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-08 03:12:31 +03:00
|
|
|
void CLoadFile::clear()
|
|
|
|
{
|
|
|
|
sfile = nullptr;
|
|
|
|
fName.clear();
|
|
|
|
fileVersion = 0;
|
|
|
|
}
|
|
|
|
|
2013-02-19 01:37:22 +03:00
|
|
|
void CLoadFile::checkMagicBytes( const std::string &text )
|
|
|
|
{
|
|
|
|
std::string loaded = text;
|
|
|
|
read(loaded.c_str(), text.length());
|
|
|
|
if(loaded != text)
|
|
|
|
throw std::runtime_error("Magic bytes doesn't match!");
|
|
|
|
}
|
|
|
|
|
2009-03-07 00:11:17 +02:00
|
|
|
CTypeList::CTypeList()
|
|
|
|
{
|
2009-03-12 01:25:59 +02:00
|
|
|
registerTypes(*this);
|
2009-03-07 00:11:17 +02:00
|
|
|
}
|
|
|
|
|
2009-03-14 20:19:07 +02:00
|
|
|
ui16 CTypeList::registerType( const std::type_info *type )
|
2009-03-07 00:11:17 +02:00
|
|
|
{
|
2009-03-07 17:54:12 +02:00
|
|
|
TTypeMap::const_iterator i = types.find(type);
|
2009-03-07 00:11:17 +02:00
|
|
|
if(i != types.end())
|
2009-03-07 17:54:12 +02:00
|
|
|
return i->second; //type found, return ID
|
2009-03-07 00:11:17 +02:00
|
|
|
|
2009-03-07 17:54:12 +02:00
|
|
|
//type not found - add it to the list and return given ID
|
2009-03-07 00:11:17 +02:00
|
|
|
ui16 id = types.size() + 1;
|
2009-03-07 17:54:12 +02:00
|
|
|
types.insert(std::make_pair(type,id));
|
2009-03-07 00:11:17 +02:00
|
|
|
return id;
|
|
|
|
}
|
|
|
|
|
2009-03-14 20:19:07 +02:00
|
|
|
ui16 CTypeList::getTypeID( const std::type_info *type )
|
2009-03-07 00:11:17 +02:00
|
|
|
{
|
2009-03-07 17:54:12 +02:00
|
|
|
TTypeMap::const_iterator i = types.find(type);
|
|
|
|
if(i != types.end())
|
|
|
|
return i->second;
|
2009-03-07 00:11:17 +02:00
|
|
|
else
|
|
|
|
return 0;
|
2009-10-04 05:02:45 +03:00
|
|
|
}
|
2010-06-26 19:02:10 +03:00
|
|
|
|
2010-10-24 14:35:14 +03:00
|
|
|
std::ostream & operator<<(std::ostream &str, const CConnection &cpc)
|
|
|
|
{
|
|
|
|
return str << "Connection with " << cpc.name << " (ID: " << cpc.connectionID << /*", " << (cpc.host ? "host" : "guest") <<*/ ")";
|
|
|
|
}
|
|
|
|
|
2010-06-26 19:02:10 +03:00
|
|
|
CSerializer::~CSerializer()
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
CSerializer::CSerializer()
|
|
|
|
{
|
|
|
|
smartVectorMembersSerialization = false;
|
2012-04-14 05:20:22 +03:00
|
|
|
sendStackInstanceByIds = false;
|
2010-06-26 19:02:10 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void CSerializer::addStdVecItems(CGameState *gs, LibClasses *lib)
|
|
|
|
{
|
2013-05-19 01:30:48 +03:00
|
|
|
registerVectoredType<CGObjectInstance, ObjectInstanceID>(&gs->map->objects,
|
|
|
|
[](const CGObjectInstance &obj){ return obj.id; });
|
|
|
|
registerVectoredType<CHero, HeroTypeID>(&lib->heroh->heroes,
|
|
|
|
[](const CHero &h){ return h.ID; });
|
|
|
|
registerVectoredType<CGHeroInstance, HeroTypeID>(&gs->map->allHeroes,
|
|
|
|
[](const CGHeroInstance &h){ return h.type->ID; });
|
|
|
|
registerVectoredType<CCreature, CreatureID>(&lib->creh->creatures,
|
|
|
|
[](const CCreature &cre){ return cre.idNumber; });
|
|
|
|
registerVectoredType<CArtifact, ArtifactID>(&lib->arth->artifacts,
|
|
|
|
[](const CArtifact &art){ return art.id; });
|
|
|
|
registerVectoredType<CArtifactInstance, ArtifactInstanceID>(&gs->map->artInstances,
|
|
|
|
[](const CArtifactInstance &artInst){ return artInst.id; });
|
|
|
|
registerVectoredType<CQuest, si32>(&gs->map->quests,
|
|
|
|
[](const CQuest &q){ return q.qid; });
|
|
|
|
|
2010-06-26 19:02:10 +03:00
|
|
|
smartVectorMembersSerialization = true;
|
2010-08-21 03:37:19 +03:00
|
|
|
}
|
2013-02-19 01:37:22 +03:00
|
|
|
|
|
|
|
CLoadIntegrityValidator::CLoadIntegrityValidator( const std::string &primaryFileName, const std::string &controlFileName, int minimalVersion /*= version*/ )
|
|
|
|
: foundDesync(false)
|
|
|
|
{
|
|
|
|
registerTypes(*this);
|
|
|
|
primaryFile = make_unique<CLoadFile>(primaryFileName, minimalVersion);
|
|
|
|
controlFile = make_unique<CLoadFile>(controlFileName, minimalVersion);
|
|
|
|
}
|
|
|
|
|
|
|
|
int CLoadIntegrityValidator::read( const void * data, unsigned size )
|
|
|
|
{
|
|
|
|
assert(primaryFile);
|
|
|
|
assert(controlFile);
|
|
|
|
|
|
|
|
if(!size)
|
|
|
|
return size;
|
|
|
|
|
|
|
|
std::vector<ui8> controlData(size);
|
|
|
|
auto ret = primaryFile->read(data, size);
|
2013-02-19 15:24:17 +03:00
|
|
|
|
|
|
|
if(!foundDesync)
|
2013-02-19 01:37:22 +03:00
|
|
|
{
|
2013-02-19 15:24:17 +03:00
|
|
|
controlFile->read(controlData.data(), size);
|
|
|
|
if(std::memcmp(data, controlData.data(), size))
|
|
|
|
{
|
2013-04-09 17:31:36 +03:00
|
|
|
logGlobal->errorStream() << "Desync found! Position: " << primaryFile->sfile->tellg();
|
2013-02-19 15:24:17 +03:00
|
|
|
foundDesync = true;
|
|
|
|
//throw std::runtime_error("Savegame dsynchronized!");
|
|
|
|
}
|
2013-02-19 01:37:22 +03:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
unique_ptr<CLoadFile> CLoadIntegrityValidator::decay()
|
|
|
|
{
|
2013-05-09 14:09:23 +03:00
|
|
|
primaryFile->loadedPointers = this->loadedPointers;
|
2013-02-19 01:37:22 +03:00
|
|
|
return std::move(primaryFile);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CLoadIntegrityValidator::checkMagicBytes( const std::string &text )
|
|
|
|
{
|
|
|
|
assert(primaryFile);
|
|
|
|
assert(controlFile);
|
|
|
|
|
|
|
|
primaryFile->checkMagicBytes(text);
|
|
|
|
controlFile->checkMagicBytes(text);
|
|
|
|
}
|