| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280 | /* * CGameInterface.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 * */#include "StdInc.h"#include "CGameInterface.h"#include "CStack.h"#include "VCMIDirs.h"#include "serializer/BinaryDeserializer.h"#include "serializer/BinarySerializer.h"#ifdef STATIC_AI# include "AI/VCAI/VCAI.h"# include "AI/Nullkiller/AIGateway.h"# include "AI/BattleAI/BattleAI.h"# include "AI/StupidAI/StupidAI.h"# include "AI/EmptyAI/CEmptyAI.h"#else# ifdef VCMI_WINDOWS#  include <windows.h> //for .dll libs# else#  include <dlfcn.h># endif // VCMI_WINDOWS#endif // STATIC_AIVCMI_LIB_NAMESPACE_BEGINtemplate<typename rett>std::shared_ptr<rett> createAny(const boost::filesystem::path & libpath, const std::string & methodName){#ifdef STATIC_AI	// android currently doesn't support loading libs dynamically, so the access to the known libraries	// is possible only via specializations of this template	throw std::runtime_error("Could not resolve ai library " + libpath.generic_string());#else	using TGetAIFun = void (*)(std::shared_ptr<rett> &);	using TGetNameFun = void (*)(char *);	char temp[150];	TGetAIFun getAI = nullptr;	TGetNameFun getName = nullptr;#ifdef VCMI_WINDOWS#ifdef __MINGW32__#pragma GCC diagnostic push#pragma GCC diagnostic ignored "-Wcast-function-type"#endif	HMODULE dll = LoadLibraryW(libpath.c_str());	if (dll)	{		getName = reinterpret_cast<TGetNameFun>(GetProcAddress(dll, "GetAiName"));		getAI = reinterpret_cast<TGetAIFun>(GetProcAddress(dll, methodName.c_str()));	}#ifdef __MINGW32__#pragma GCC diagnostic pop#endif#else // !VCMI_WINDOWS	void *dll = dlopen(libpath.string().c_str(), RTLD_LOCAL | RTLD_LAZY);	if (dll)	{		getName = reinterpret_cast<TGetNameFun>(dlsym(dll, "GetAiName"));		getAI = reinterpret_cast<TGetAIFun>(dlsym(dll, methodName.c_str()));	}#endif // VCMI_WINDOWS	if (!dll)	{		logGlobal->error("Cannot open dynamic library (%s). Throwing...", libpath.string());		throw std::runtime_error("Cannot open dynamic library");	}	else if(!getName || !getAI)	{		logGlobal->error("%s does not export method %s", libpath.string(), methodName);#ifdef VCMI_WINDOWS		FreeLibrary(dll);#else		dlclose(dll);#endif		throw std::runtime_error("Cannot find method " + methodName);	}	getName(temp);	logGlobal->info("Loaded %s", temp);	std::shared_ptr<rett> ret;	getAI(ret);	if(!ret)		logGlobal->error("Cannot get AI!");	return ret;#endif // STATIC_AI}#ifdef STATIC_AItemplate<>std::shared_ptr<CGlobalAI> createAny(const boost::filesystem::path & libpath, const std::string & methodName){	if(libpath.stem() == "libNullkiller") {		return std::make_shared<NKAI::AIGateway>();	}	else{		return std::make_shared<VCAI>();	}}template<>std::shared_ptr<CBattleGameInterface> createAny(const boost::filesystem::path & libpath, const std::string & methodName){	if(libpath.stem() == "libBattleAI")		return std::make_shared<CBattleAI>();	else if(libpath.stem() == "libStupidAI")		return std::make_shared<CStupidAI>();	return std::make_shared<CEmptyAI>();}#endif // STATIC_AItemplate<typename rett>std::shared_ptr<rett> createAnyAI(const std::string & dllname, const std::string & methodName){	logGlobal->info("Opening %s", dllname);	const boost::filesystem::path filePath = VCMIDirs::get().fullLibraryPath("AI", dllname);	auto ret = createAny<rett>(filePath, methodName);	ret->dllName = dllname;	return ret;}std::shared_ptr<CGlobalAI> CDynLibHandler::getNewAI(const std::string & dllname){	return createAnyAI<CGlobalAI>(dllname, "GetNewAI");}std::shared_ptr<CBattleGameInterface> CDynLibHandler::getNewBattleAI(const std::string & dllname){	return createAnyAI<CBattleGameInterface>(dllname, "GetNewBattleAI");}#if SCRIPTING_ENABLEDstd::shared_ptr<scripting::Module> CDynLibHandler::getNewScriptingModule(const boost::filesystem::path & dllname){	return createAny<scripting::Module>(dllname, "GetNewModule");}#endifvoid CGlobalAI::activeStack(const CStack * stack){	BattleAction ba;	ba.actionType = EActionType::DEFEND;	ba.stackNumber = stack->unitId();	assert(0);}CGlobalAI::CGlobalAI(){	human = false;}void CAdventureAI::battleNewRound(int round){	battleAI->battleNewRound(round);}void CAdventureAI::battleCatapultAttacked(const CatapultAttack & ca){	battleAI->battleCatapultAttacked(ca);}void CAdventureAI::battleStart(const CCreatureSet * army1, const CCreatureSet * army2, int3 tile,							   const CGHeroInstance * hero1, const CGHeroInstance * hero2, bool side){	assert(!battleAI);	assert(cbc);	battleAI = CDynLibHandler::getNewBattleAI(getBattleAIName());	battleAI->initBattleInterface(env, cbc);	battleAI->battleStart(army1, army2, tile, hero1, hero2, side);}void CAdventureAI::battleStacksAttacked(const std::vector<BattleStackAttacked> & bsa, bool ranged){	battleAI->battleStacksAttacked(bsa, ranged);}void CAdventureAI::actionStarted(const BattleAction & action){	battleAI->actionStarted(action);}void CAdventureAI::battleNewRoundFirst(int round){	battleAI->battleNewRoundFirst(round);}void CAdventureAI::actionFinished(const BattleAction & action){	battleAI->actionFinished(action);}void CAdventureAI::battleStacksEffectsSet(const SetStackEffect & sse){	battleAI->battleStacksEffectsSet(sse);}void CAdventureAI::battleObstaclesChanged(const std::vector<ObstacleChanges> & obstacles){	battleAI->battleObstaclesChanged(obstacles);}void CAdventureAI::battleStackMoved(const CStack * stack, std::vector<BattleHex> dest, int distance, bool teleport){	battleAI->battleStackMoved(stack, dest, distance, teleport);}void CAdventureAI::battleAttack(const BattleAttack * ba){	battleAI->battleAttack(ba);}void CAdventureAI::battleSpellCast(const BattleSpellCast * sc){	battleAI->battleSpellCast(sc);}void CAdventureAI::battleEnd(const BattleResult * br, QueryID queryID){	battleAI->battleEnd(br, queryID);	battleAI.reset();}void CAdventureAI::battleUnitsChanged(const std::vector<UnitChanges> & units){	battleAI->battleUnitsChanged(units);}void CAdventureAI::activeStack(const CStack * stack){	battleAI->activeStack(stack);}void CAdventureAI::yourTacticPhase(int distance){	battleAI->yourTacticPhase(distance);}void CAdventureAI::saveGame(BinarySerializer & h, const int version) /*saving */{	LOG_TRACE_PARAMS(logAi, "version '%i'", version);	bool hasBattleAI = static_cast<bool>(battleAI);	h & hasBattleAI;	if(hasBattleAI)	{		h & battleAI->dllName;	}}void CAdventureAI::loadGame(BinaryDeserializer & h, const int version) /*loading */{	LOG_TRACE_PARAMS(logAi, "version '%i'", version);	bool hasBattleAI = false;	h & hasBattleAI;	if(hasBattleAI)	{		std::string dllName;		h & dllName;		battleAI = CDynLibHandler::getNewBattleAI(dllName);		assert(cbc); //it should have been set by the one who new'ed us		battleAI->initBattleInterface(env, cbc);	}}VCMI_LIB_NAMESPACE_END
 |