|  | @@ -19,27 +19,24 @@
 | 
	
		
			
				|  |  |  #include "../../../lib/PathfinderUtil.h"
 | 
	
		
			
				|  |  |  #include "../../../lib/CPlayerState.h"
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +namespace NKAI
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  std::shared_ptr<boost::multi_array<AIPathNode, 5>> AISharedStorage::shared;
 | 
	
		
			
				|  |  |  std::set<int3> commitedTiles;
 | 
	
		
			
				|  |  |  std::set<int3> commitedTilesInitial;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -#ifdef ENVIRONMENT64
 | 
	
		
			
				|  |  | -const int BUCKET_COUNT = 11;
 | 
	
		
			
				|  |  | -#else
 | 
	
		
			
				|  |  | -const int BUCKET_COUNT = 7;
 | 
	
		
			
				|  |  | -#endif // ENVIRONMENT64
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  const uint64_t FirstActorMask = 1;
 | 
	
		
			
				|  |  | -const int BUCKET_SIZE = GameConstants::MAX_HEROES_PER_PLAYER;
 | 
	
		
			
				|  |  | -const int NUM_CHAINS = BUCKET_COUNT * BUCKET_SIZE;
 | 
	
		
			
				|  |  |  const uint64_t MIN_ARMY_STRENGTH_FOR_CHAIN = 5000;
 | 
	
		
			
				|  |  |  const uint64_t MIN_ARMY_STRENGTH_FOR_NEXT_ACTOR = 1000;
 | 
	
		
			
				|  |  | +const uint64_t CHAIN_MAX_DEPTH = 4;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  AISharedStorage::AISharedStorage(int3 sizes)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  |  	if(!shared){
 | 
	
		
			
				|  |  |  		shared.reset(new boost::multi_array<AIPathNode, 5>(
 | 
	
		
			
				|  |  | -			boost::extents[EPathfindingLayer::NUM_LAYERS][sizes.z][sizes.x][sizes.y][NUM_CHAINS]));
 | 
	
		
			
				|  |  | +			boost::extents[EPathfindingLayer::NUM_LAYERS][sizes.z][sizes.x][sizes.y][AIPathfinding::NUM_CHAINS]));
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	nodes = shared;
 | 
	
	
		
			
				|  | @@ -88,11 +85,11 @@ void AINodeStorage::initialize(const PathfinderOptions & options, const CGameSta
 | 
	
		
			
				|  |  |  			{
 | 
	
		
			
				|  |  |  				for(pos.y = 0; pos.y < sizes.y; ++pos.y)
 | 
	
		
			
				|  |  |  				{
 | 
	
		
			
				|  |  | -					const TerrainTile* tile = &gs->map->getTile(pos);
 | 
	
		
			
				|  |  | -					if (!tile->terType.isPassable())
 | 
	
		
			
				|  |  | +					const TerrainTile & tile = gs->map->getTile(pos);
 | 
	
		
			
				|  |  | +					if (!tile.terType->isPassable())
 | 
	
		
			
				|  |  |  						continue;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -					if (tile->terType.isWater())
 | 
	
		
			
				|  |  | +					if (tile.terType->isWater())
 | 
	
		
			
				|  |  |  					{
 | 
	
		
			
				|  |  |  						resetTile(pos, ELayer::SAIL, PathfinderUtil::evaluateAccessibility<ELayer::SAIL>(pos, tile, fow, player, gs));
 | 
	
		
			
				|  |  |  						if (useFlying)
 | 
	
	
		
			
				|  | @@ -139,16 +136,16 @@ boost::optional<AIPathNode *> AINodeStorage::getOrCreateNode(
 | 
	
		
			
				|  |  |  	const EPathfindingLayer layer, 
 | 
	
		
			
				|  |  |  	const ChainActor * actor)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  | -	int bucketIndex = ((uintptr_t)actor) % BUCKET_COUNT;
 | 
	
		
			
				|  |  | -	int bucketOffset = bucketIndex * BUCKET_SIZE;
 | 
	
		
			
				|  |  | -	auto chains = nodes.get(pos, layer); //FIXME: chain was the innermost layer
 | 
	
		
			
				|  |  | +	int bucketIndex = ((uintptr_t)actor) % AIPathfinding::BUCKET_COUNT;
 | 
	
		
			
				|  |  | +	int bucketOffset = bucketIndex * AIPathfinding::BUCKET_SIZE;
 | 
	
		
			
				|  |  | +	auto chains = nodes.get(pos, layer);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	if(chains[0].blocked())
 | 
	
		
			
				|  |  |  	{
 | 
	
		
			
				|  |  |  		return boost::none;
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -	for(auto i = BUCKET_SIZE - 1; i >= 0; i--)
 | 
	
		
			
				|  |  | +	for(auto i = AIPathfinding::BUCKET_SIZE - 1; i >= 0; i--)
 | 
	
		
			
				|  |  |  	{
 | 
	
		
			
				|  |  |  		AIPathNode & node = chains[i + bucketOffset];
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -171,8 +168,9 @@ boost::optional<AIPathNode *> AINodeStorage::getOrCreateNode(
 | 
	
		
			
				|  |  |  std::vector<CGPathNode *> AINodeStorage::getInitialNodes()
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  |  	if(heroChainPass)
 | 
	
		
			
				|  |  | -{
 | 
	
		
			
				|  |  | -		calculateTownPortalTeleportations(heroChain);
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		if(heroChainTurn == 0)
 | 
	
		
			
				|  |  | +			calculateTownPortalTeleportations(heroChain);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  		return heroChain;
 | 
	
		
			
				|  |  |  	}
 | 
	
	
		
			
				|  | @@ -207,7 +205,8 @@ std::vector<CGPathNode *> AINodeStorage::getInitialNodes()
 | 
	
		
			
				|  |  |  		}
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -	calculateTownPortalTeleportations(initialNodes);
 | 
	
		
			
				|  |  | +	if(heroChainTurn == 0)
 | 
	
		
			
				|  |  | +		calculateTownPortalTeleportations(initialNodes);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	return initialNodes;
 | 
	
		
			
				|  |  |  }
 | 
	
	
		
			
				|  | @@ -265,7 +264,7 @@ void AINodeStorage::commit(
 | 
	
		
			
				|  |  |  	destination->theNodeBefore = source->theNodeBefore;
 | 
	
		
			
				|  |  |  	destination->chainOther = nullptr;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  	logAi->trace(
 | 
	
		
			
				|  |  |  		"Commited %s -> %s, cost: %f, turn: %s, mp: %d, hero: %s, mask: %x, army: %lld",
 | 
	
		
			
				|  |  |  		source->coord.toString(),
 | 
	
	
		
			
				|  | @@ -406,8 +405,8 @@ public:
 | 
	
		
			
				|  |  |  		AINodeStorage & storage, AISharedStorage & nodes, const std::vector<int3> & tiles, uint64_t chainMask, int heroChainTurn)
 | 
	
		
			
				|  |  |  		:existingChains(), newChains(), delayedWork(), nodes(nodes), storage(storage), chainMask(chainMask), heroChainTurn(heroChainTurn), heroChain(), tiles(tiles)
 | 
	
		
			
				|  |  |  	{
 | 
	
		
			
				|  |  | -		existingChains.reserve(NUM_CHAINS);
 | 
	
		
			
				|  |  | -		newChains.reserve(NUM_CHAINS);
 | 
	
		
			
				|  |  | +		existingChains.reserve(AIPathfinding::NUM_CHAINS);
 | 
	
		
			
				|  |  | +		newChains.reserve(AIPathfinding::NUM_CHAINS);
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	void execute(const blocked_range<size_t>& r)
 | 
	
	
		
			
				|  | @@ -593,7 +592,7 @@ void HeroChainCalculationTask::cleanupInefectiveChains(std::vector<ExchangeCandi
 | 
	
		
			
				|  |  |  		auto isNotEffective = storage.hasBetterChain(chainInfo.carrierParent, &chainInfo, chains)
 | 
	
		
			
				|  |  |  			|| storage.hasBetterChain(chainInfo.carrierParent, &chainInfo, result);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  		if(isNotEffective)
 | 
	
		
			
				|  |  |  		{
 | 
	
		
			
				|  |  |  			logAi->trace(
 | 
	
	
		
			
				|  | @@ -623,6 +622,9 @@ void HeroChainCalculationTask::calculateHeroChain(
 | 
	
		
			
				|  |  |  		if((node->actor->chainMask & chainMask) == 0 && (srcNode->actor->chainMask & chainMask) == 0)
 | 
	
		
			
				|  |  |  			continue;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +		if(node->actor->actorExchangeCount + srcNode->actor->actorExchangeCount > CHAIN_MAX_DEPTH)
 | 
	
		
			
				|  |  | +			continue;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  		if(node->action == CGPathNode::ENodeAction::BATTLE
 | 
	
		
			
				|  |  |  			|| node->action == CGPathNode::ENodeAction::TELEPORT_BATTLE
 | 
	
		
			
				|  |  |  			|| node->action == CGPathNode::ENodeAction::TELEPORT_NORMAL
 | 
	
	
		
			
				|  | @@ -635,7 +637,7 @@ void HeroChainCalculationTask::calculateHeroChain(
 | 
	
		
			
				|  |  |  			|| (node->action == CGPathNode::ENodeAction::UNKNOWN && node->actor->hero)
 | 
	
		
			
				|  |  |  			|| (node->actor->chainMask & srcNode->actor->chainMask) != 0)
 | 
	
		
			
				|  |  |  		{
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  			logAi->trace(
 | 
	
		
			
				|  |  |  				"Skip exchange %s[%x] -> %s[%x] at %s because of %s",
 | 
	
		
			
				|  |  |  				node->actor->toString(),
 | 
	
	
		
			
				|  | @@ -652,7 +654,7 @@ void HeroChainCalculationTask::calculateHeroChain(
 | 
	
		
			
				|  |  |  			continue;
 | 
	
		
			
				|  |  |  		}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  		logAi->trace(
 | 
	
		
			
				|  |  |  			"Thy exchange %s[%x] -> %s[%x] at %s",
 | 
	
		
			
				|  |  |  			node->actor->toString(),
 | 
	
	
		
			
				|  | @@ -676,7 +678,7 @@ void HeroChainCalculationTask::calculateHeroChain(
 | 
	
		
			
				|  |  |  		&& carrier->action != CGPathNode::BLOCKING_VISIT
 | 
	
		
			
				|  |  |  		&& (other->armyLoss == 0 || other->armyLoss < other->actor->armyValue))
 | 
	
		
			
				|  |  |  	{
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  		logAi->trace(
 | 
	
		
			
				|  |  |  			"Exchange allowed %s[%x] -> %s[%x] at %s",
 | 
	
		
			
				|  |  |  			other->actor->toString(),
 | 
	
	
		
			
				|  | @@ -693,7 +695,7 @@ void HeroChainCalculationTask::calculateHeroChain(
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  			if(hasLessMp && hasLessExperience)
 | 
	
		
			
				|  |  |  			{
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  				logAi->trace("Exchange at %s is ineficient. Blocked.", carrier->coord.toString());
 | 
	
		
			
				|  |  |  #endif
 | 
	
		
			
				|  |  |  				return;
 | 
	
	
		
			
				|  | @@ -718,7 +720,7 @@ void HeroChainCalculationTask::addHeroChain(const std::vector<ExchangeCandidate>
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  		if(!chainNodeOptional)
 | 
	
		
			
				|  |  |  		{
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  			logAi->trace("Exchange at %s can not allocate node. Blocked.", carrier->coord.toString());
 | 
	
		
			
				|  |  |  #endif
 | 
	
		
			
				|  |  |  			continue;
 | 
	
	
		
			
				|  | @@ -728,7 +730,7 @@ void HeroChainCalculationTask::addHeroChain(const std::vector<ExchangeCandidate>
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  		if(exchangeNode->action != CGPathNode::ENodeAction::UNKNOWN)
 | 
	
		
			
				|  |  |  		{
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  			logAi->trace(
 | 
	
		
			
				|  |  |  				"Skip exchange %s[%x] -> %s[%x] at %s because node is in use",
 | 
	
		
			
				|  |  |  				other->actor->toString(),
 | 
	
	
		
			
				|  | @@ -742,7 +744,7 @@ void HeroChainCalculationTask::addHeroChain(const std::vector<ExchangeCandidate>
 | 
	
		
			
				|  |  |  		
 | 
	
		
			
				|  |  |  		if(exchangeNode->turns != 0xFF && exchangeNode->getCost() < chainInfo.getCost())
 | 
	
		
			
				|  |  |  		{
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  			logAi->trace(
 | 
	
		
			
				|  |  |  				"Skip exchange %s[%x] -> %s[%x] at %s because not effective enough. %f < %f",
 | 
	
		
			
				|  |  |  				other->actor->toString(),
 | 
	
	
		
			
				|  | @@ -773,7 +775,7 @@ void HeroChainCalculationTask::addHeroChain(const std::vector<ExchangeCandidate>
 | 
	
		
			
				|  |  |  		exchangeNode->chainOther = other;
 | 
	
		
			
				|  |  |  		exchangeNode->armyLoss = chainInfo.armyLoss;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  		logAi->trace(
 | 
	
		
			
				|  |  |  			"Chain accepted at %s %s -> %s, mask %x, cost %f, turn: %s, mp: %d, army %i", 
 | 
	
		
			
				|  |  |  			exchangeNode->coord.toString(), 
 | 
	
	
		
			
				|  | @@ -994,8 +996,6 @@ struct TowmPortalFinder
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	CGPathNode * getBestInitialNodeForTownPortal(const CGTownInstance * targetTown)
 | 
	
		
			
				|  |  |  	{
 | 
	
		
			
				|  |  | -		CGPathNode * bestNode = nullptr;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  |  		for(CGPathNode * node : initialNodes)
 | 
	
		
			
				|  |  |  		{
 | 
	
		
			
				|  |  |  			auto aiNode = nodeStorage->getAINode(node);
 | 
	
	
		
			
				|  | @@ -1018,11 +1018,10 @@ struct TowmPortalFinder
 | 
	
		
			
				|  |  |  					continue;
 | 
	
		
			
				|  |  |  			}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -			if(!bestNode || bestNode->getCost() > node->getCost())
 | 
	
		
			
				|  |  | -				bestNode = node;
 | 
	
		
			
				|  |  | +			return node;
 | 
	
		
			
				|  |  |  		}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -		return bestNode;
 | 
	
		
			
				|  |  | +		return nullptr;
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	boost::optional<AIPathNode *> createTownPortalNode(const CGTownInstance * targetTown)
 | 
	
	
		
			
				|  | @@ -1060,6 +1059,55 @@ struct TowmPortalFinder
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  };
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | +template<class TVector>
 | 
	
		
			
				|  |  | +void AINodeStorage::calculateTownPortal(
 | 
	
		
			
				|  |  | +	const ChainActor * actor,
 | 
	
		
			
				|  |  | +	const std::map<const CGHeroInstance *, int> & maskMap,
 | 
	
		
			
				|  |  | +	const std::vector<CGPathNode *> & initialNodes,
 | 
	
		
			
				|  |  | +	TVector & output)
 | 
	
		
			
				|  |  | +{
 | 
	
		
			
				|  |  | +	auto towns = cb->getTownsInfo(false);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	vstd::erase_if(towns, [&](const CGTownInstance * t) -> bool
 | 
	
		
			
				|  |  | +		{
 | 
	
		
			
				|  |  | +			return cb->getPlayerRelations(actor->hero->tempOwner, t->tempOwner) == PlayerRelations::ENEMIES;
 | 
	
		
			
				|  |  | +		});
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	if(!towns.size())
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		return; // no towns no need to run loop further
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	TowmPortalFinder townPortalFinder(actor, initialNodes, towns, this);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +	if(townPortalFinder.actorCanCastTownPortal())
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		for(const CGTownInstance * targetTown : towns)
 | 
	
		
			
				|  |  | +		{
 | 
	
		
			
				|  |  | +			// TODO: allow to hide visiting hero in garrison
 | 
	
		
			
				|  |  | +			if(targetTown->visitingHero)
 | 
	
		
			
				|  |  | +			{
 | 
	
		
			
				|  |  | +				auto basicMask = maskMap.at(targetTown->visitingHero.get());
 | 
	
		
			
				|  |  | +				bool heroIsInChain = (actor->chainMask & basicMask) != 0;
 | 
	
		
			
				|  |  | +				bool sameActorInTown = actor->chainMask == basicMask;
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +				if(sameActorInTown || !heroIsInChain)
 | 
	
		
			
				|  |  | +					continue;
 | 
	
		
			
				|  |  | +			}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +			auto nodeOptional = townPortalFinder.createTownPortalNode(targetTown);
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +			if(nodeOptional)
 | 
	
		
			
				|  |  | +			{
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 1
 | 
	
		
			
				|  |  | +				logAi->trace("Adding town portal node at %s", targetTown->name);
 | 
	
		
			
				|  |  | +#endif
 | 
	
		
			
				|  |  | +				output.push_back(nodeOptional.get());
 | 
	
		
			
				|  |  | +			}
 | 
	
		
			
				|  |  | +		}
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +}
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  |  void AINodeStorage::calculateTownPortalTeleportations(std::vector<CGPathNode *> & initialNodes)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  |  	std::set<const ChainActor *> actorsOfInitial;
 | 
	
	
		
			
				|  | @@ -1068,7 +1116,8 @@ void AINodeStorage::calculateTownPortalTeleportations(std::vector<CGPathNode *>
 | 
	
		
			
				|  |  |  	{
 | 
	
		
			
				|  |  |  		auto aiNode = getAINode(node);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -		actorsOfInitial.insert(aiNode->actor->baseActor);
 | 
	
		
			
				|  |  | +		if(aiNode->actor->hero)
 | 
	
		
			
				|  |  | +			actorsOfInitial.insert(aiNode->actor->baseActor);
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	std::map<const CGHeroInstance *, int> maskMap;
 | 
	
	
		
			
				|  | @@ -1079,50 +1128,28 @@ void AINodeStorage::calculateTownPortalTeleportations(std::vector<CGPathNode *>
 | 
	
		
			
				|  |  |  			maskMap[basicActor->hero] = basicActor->chainMask;
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -	for(const ChainActor * actor : actorsOfInitial)
 | 
	
		
			
				|  |  | -	{
 | 
	
		
			
				|  |  | -		if(!actor->hero)
 | 
	
		
			
				|  |  | -			continue;
 | 
	
		
			
				|  |  | +	boost::sort(initialNodes, NodeComparer<CGPathNode>());
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -		auto towns = cb->getTownsInfo(false);
 | 
	
		
			
				|  |  | +	std::vector<const ChainActor *> actorsVector(actorsOfInitial.begin(), actorsOfInitial.end());
 | 
	
		
			
				|  |  | +	tbb::concurrent_vector<CGPathNode *> output;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -		vstd::erase_if(towns, [&](const CGTownInstance * t) -> bool
 | 
	
		
			
				|  |  | -		{
 | 
	
		
			
				|  |  | -			return cb->getPlayerRelations(actor->hero->tempOwner, t->tempOwner) == PlayerRelations::ENEMIES;
 | 
	
		
			
				|  |  | -		});
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -		if(!towns.size())
 | 
	
		
			
				|  |  | -		{
 | 
	
		
			
				|  |  | -			return; // no towns no need to run loop further
 | 
	
		
			
				|  |  | -		}
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -		TowmPortalFinder townPortalFinder(actor, initialNodes, towns, this);
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -		if(townPortalFinder.actorCanCastTownPortal())
 | 
	
		
			
				|  |  | -		{
 | 
	
		
			
				|  |  | -			for(const CGTownInstance * targetTown : towns)
 | 
	
		
			
				|  |  | +	if(actorsVector.size() * initialNodes.size() > 1000)
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		parallel_for(blocked_range<size_t>(0, actorsVector.size()), [&](const blocked_range<size_t> & r)
 | 
	
		
			
				|  |  |  			{
 | 
	
		
			
				|  |  | -				// TODO: allow to hide visiting hero in garrison
 | 
	
		
			
				|  |  | -				if(targetTown->visitingHero)
 | 
	
		
			
				|  |  | +				for(int i = r.begin(); i != r.end(); i++)
 | 
	
		
			
				|  |  |  				{
 | 
	
		
			
				|  |  | -					auto basicMask = maskMap[targetTown->visitingHero.get()];
 | 
	
		
			
				|  |  | -					bool heroIsInChain = (actor->chainMask & basicMask) != 0;
 | 
	
		
			
				|  |  | -					bool sameActorInTown = actor->chainMask == basicMask;
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -					if(sameActorInTown || !heroIsInChain)
 | 
	
		
			
				|  |  | -						continue;
 | 
	
		
			
				|  |  | +					calculateTownPortal(actorsVector[i], maskMap, initialNodes, output);
 | 
	
		
			
				|  |  |  				}
 | 
	
		
			
				|  |  | +			});
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -				auto nodeOptional = townPortalFinder.createTownPortalNode(targetTown);
 | 
	
		
			
				|  |  | -
 | 
	
		
			
				|  |  | -				if(nodeOptional)
 | 
	
		
			
				|  |  | -				{
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 1
 | 
	
		
			
				|  |  | -					logAi->trace("Adding town portal node at %s", targetTown->name);
 | 
	
		
			
				|  |  | -#endif
 | 
	
		
			
				|  |  | -					initialNodes.push_back(nodeOptional.get());
 | 
	
		
			
				|  |  | -				}
 | 
	
		
			
				|  |  | -			}
 | 
	
		
			
				|  |  | +		std::copy(output.begin(), output.end(), std::back_inserter(initialNodes));
 | 
	
		
			
				|  |  | +	}
 | 
	
		
			
				|  |  | +	else
 | 
	
		
			
				|  |  | +	{
 | 
	
		
			
				|  |  | +		for(auto actor : actorsVector)
 | 
	
		
			
				|  |  | +		{
 | 
	
		
			
				|  |  | +			calculateTownPortal(actor, maskMap, initialNodes, initialNodes);
 | 
	
		
			
				|  |  |  		}
 | 
	
		
			
				|  |  |  	}
 | 
	
		
			
				|  |  |  }
 | 
	
	
		
			
				|  | @@ -1156,7 +1183,7 @@ bool AINodeStorage::hasBetterChain(
 | 
	
		
			
				|  |  |  		{
 | 
	
		
			
				|  |  |  			if(node.getCost() < candidateNode->getCost())
 | 
	
		
			
				|  |  |  			{
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  				logAi->trace(
 | 
	
		
			
				|  |  |  					"Block ineficient battle move %s->%s, hero: %s[%X], army %lld, mp diff: %i",
 | 
	
		
			
				|  |  |  					source->coord.toString(),
 | 
	
	
		
			
				|  | @@ -1180,7 +1207,7 @@ bool AINodeStorage::hasBetterChain(
 | 
	
		
			
				|  |  |  		if(nodeArmyValue > candidateArmyValue
 | 
	
		
			
				|  |  |  			&& node.getCost() <= candidateNode->getCost())
 | 
	
		
			
				|  |  |  		{
 | 
	
		
			
				|  |  | -#if PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  			logAi->trace(
 | 
	
		
			
				|  |  |  				"Block ineficient move because of stronger army %s->%s, hero: %s[%X], army %lld, mp diff: %i",
 | 
	
		
			
				|  |  |  				source->coord.toString(),
 | 
	
	
		
			
				|  | @@ -1206,7 +1233,7 @@ bool AINodeStorage::hasBetterChain(
 | 
	
		
			
				|  |  |  					continue;
 | 
	
		
			
				|  |  |  				}
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -#if AI_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  | +#if NKAI_PATHFINDER_TRACE_LEVEL >= 2
 | 
	
		
			
				|  |  |  				logAi->trace(
 | 
	
		
			
				|  |  |  					"Block ineficient move because of stronger hero %s->%s, hero: %s[%X], army %lld, mp diff: %i",
 | 
	
		
			
				|  |  |  					source->coord.toString(),
 | 
	
	
		
			
				|  | @@ -1244,7 +1271,7 @@ std::vector<AIPath> AINodeStorage::getChainInfo(const int3 & pos, bool isOnLand)
 | 
	
		
			
				|  |  |  {
 | 
	
		
			
				|  |  |  	std::vector<AIPath> paths;
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  | -	paths.reserve(NUM_CHAINS / 4);
 | 
	
		
			
				|  |  | +	paths.reserve(AIPathfinding::NUM_CHAINS / 4);
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	auto chains = nodes.get(pos, isOnLand ? EPathfindingLayer::LAND : EPathfindingLayer::SAIL);
 | 
	
		
			
				|  |  |  
 | 
	
	
		
			
				|  | @@ -1428,3 +1455,5 @@ std::string AIPath::toString() const
 | 
	
		
			
				|  |  |  
 | 
	
		
			
				|  |  |  	return str.str();
 | 
	
		
			
				|  |  |  }
 | 
	
		
			
				|  |  | +
 | 
	
		
			
				|  |  | +}
 |