1
0
mirror of https://github.com/vcmi/vcmi.git synced 2024-12-14 10:12:59 +02:00
vcmi/AI/Nullkiller/Engine/Nullkiller.cpp

374 lines
8.7 KiB
C++
Raw Normal View History

/*
* Nullkiller.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 "Nullkiller.h"
2021-05-16 14:39:38 +02:00
#include "../AIGateway.h"
#include "../Behaviors/CaptureObjectsBehavior.h"
#include "../Behaviors/RecruitHeroBehavior.h"
#include "../Behaviors/BuyArmyBehavior.h"
#include "../Behaviors/StartupBehavior.h"
2021-05-16 12:53:32 +02:00
#include "../Behaviors/DefenceBehavior.h"
#include "../Behaviors/BuildingBehavior.h"
2021-05-16 13:19:07 +02:00
#include "../Behaviors/GatherArmyBehavior.h"
2021-05-16 13:45:12 +02:00
#include "../Behaviors/ClusterBehavior.h"
#include "../Goals/Invalid.h"
#include "../Goals/Composition.h"
2022-09-26 20:01:07 +02:00
namespace NKAI
{
using namespace Goals;
2022-09-26 20:01:07 +02:00
#if NKAI_TRACE_LEVEL >= 1
#define MAXPASS 1000000
#else
#define MAXPASS 30
#endif
Nullkiller::Nullkiller()
{
memory.reset(new AIMemory());
}
void Nullkiller::init(std::shared_ptr<CCallback> cb, PlayerColor playerID)
{
this->cb = cb;
this->playerID = playerID;
priorityEvaluator.reset(new PriorityEvaluator(this));
priorityEvaluators.reset(
new SharedPool<PriorityEvaluator>(
[&]()->std::unique_ptr<PriorityEvaluator>
{
2022-12-07 23:36:20 +02:00
return std::make_unique<PriorityEvaluator>(this);
}));
dangerHitMap.reset(new DangerHitMapAnalyzer(this));
2021-05-16 13:57:33 +02:00
buildAnalyzer.reset(new BuildAnalyzer(this));
objectClusterizer.reset(new ObjectClusterizer(this));
dangerEvaluator.reset(new FuzzyHelper(this));
pathfinder.reset(new AIPathfinder(cb.get(), this));
armyManager.reset(new ArmyManager(cb.get(), this));
heroManager.reset(new HeroManager(cb.get(), this));
decomposer.reset(new DeepDecomposer());
armyFormation.reset(new ArmyFormation(cb, this));
}
Goals::TTask Nullkiller::choseBestTask(Goals::TTaskVec & tasks) const
{
Goals::TTask bestTask = *vstd::maxElementByFun(tasks, [](Goals::TTask task) -> float{
return task->priority;
});
return bestTask;
}
Goals::TTask Nullkiller::choseBestTask(Goals::TSubgoal behavior, int decompositionMaxDepth) const
{
boost::this_thread::interruption_point();
2021-05-15 20:57:44 +02:00
logAi->debug("Checking behavior %s", behavior->toString());
2021-05-15 20:27:22 +02:00
2021-11-23 08:41:03 +02:00
auto start = std::chrono::high_resolution_clock::now();
Goals::TGoalVec elementarGoals = decomposer->decompose(behavior, decompositionMaxDepth);
Goals::TTaskVec tasks;
2021-05-16 13:57:33 +02:00
boost::this_thread::interruption_point();
for(auto goal : elementarGoals)
{
Goals::TTask task = Goals::taskptr(*goal);
2021-05-16 13:38:53 +02:00
if(task->priority <= 0)
task->priority = priorityEvaluator->evaluate(goal);
tasks.push_back(task);
}
2021-05-16 13:38:53 +02:00
if(tasks.empty())
{
logAi->debug("Behavior %s found no tasks. Time taken %ld", behavior->toString(), timeElapsed(start));
2021-05-16 13:38:53 +02:00
return Goals::taskptr(Goals::Invalid());
}
auto task = choseBestTask(tasks);
logAi->debug(
"Behavior %s returns %s, priority %f. Time taken %ld",
behavior->toString(),
task->toString(),
task->priority,
timeElapsed(start));
return task;
}
void Nullkiller::resetAiState()
{
2021-05-16 14:39:38 +02:00
lockedResources = TResources();
2023-07-27 14:58:49 +02:00
scanDepth = ScanDepth::MAIN_FULL;
playerID = ai->playerID;
lockedHeroes.clear();
dangerHitMap->reset();
2022-09-06 20:14:22 +02:00
useHeroChain = true;
}
2022-09-06 20:14:22 +02:00
void Nullkiller::updateAiState(int pass, bool fast)
{
2021-05-16 13:57:33 +02:00
boost::this_thread::interruption_point();
2021-11-23 08:41:03 +02:00
auto start = std::chrono::high_resolution_clock::now();
activeHero = nullptr;
2023-02-28 09:07:59 +02:00
setTargetObject(-1);
decomposer->reset();
buildAnalyzer->update();
2022-09-06 20:14:22 +02:00
if(!fast)
{
memory->removeInvisibleObjects(cb.get());
2022-09-06 20:14:22 +02:00
dangerHitMap->updateHitMap();
dangerHitMap->calculateTileOwners();
2022-09-06 20:14:22 +02:00
boost::this_thread::interruption_point();
2021-05-16 13:57:33 +02:00
2022-09-06 20:14:22 +02:00
heroManager->update();
logAi->trace("Updating paths");
2022-09-06 20:14:22 +02:00
std::map<const CGHeroInstance *, HeroRole> activeHeroes;
2022-09-06 20:14:22 +02:00
for(auto hero : cb->getHeroesInfo())
{
if(getHeroLockedReason(hero) == HeroLockedReason::DEFENCE)
continue;
2022-09-06 20:14:22 +02:00
activeHeroes[hero] = heroManager->getHeroRole(hero);
}
2022-09-06 20:14:22 +02:00
PathfinderSettings cfg;
cfg.useHeroChain = useHeroChain;
2023-07-27 14:58:49 +02:00
if(scanDepth == ScanDepth::SMALL)
2022-09-06 20:14:22 +02:00
{
2023-07-27 14:58:49 +02:00
cfg.mainTurnDistanceLimit = MAIN_TURN_DISTANCE_LIMIT;
}
if(scanDepth != ScanDepth::ALL_FULL)
{
cfg.scoutTurnDistanceLimit = SCOUT_TURN_DISTANCE_LIMIT;
2022-09-06 20:14:22 +02:00
}
2021-05-16 14:01:34 +02:00
boost::this_thread::interruption_point();
2022-09-06 20:14:22 +02:00
pathfinder->updatePaths(activeHeroes, cfg);
boost::this_thread::interruption_point();
2022-09-06 20:14:22 +02:00
objectClusterizer->clusterize();
}
2022-09-06 20:14:22 +02:00
armyManager->update();
logAi->debug("AI state updated in %ld", timeElapsed(start));
}
bool Nullkiller::isHeroLocked(const CGHeroInstance * hero) const
{
return getHeroLockedReason(hero) != HeroLockedReason::NOT_LOCKED;
}
bool Nullkiller::arePathHeroesLocked(const AIPath & path) const
{
if(getHeroLockedReason(path.targetHero) == HeroLockedReason::STARTUP)
{
2022-09-26 20:01:07 +02:00
#if NKAI_TRACE_LEVEL >= 1
2023-02-28 09:07:59 +02:00
logAi->trace("Hero %s is locked by STARTUP. Discarding %s", path.targetHero->getObjectName(), path.toString());
#endif
return true;
}
for(auto & node : path.nodes)
{
auto lockReason = getHeroLockedReason(node.targetHero);
if(lockReason != HeroLockedReason::NOT_LOCKED)
{
2022-09-26 20:01:07 +02:00
#if NKAI_TRACE_LEVEL >= 1
2023-02-28 09:07:59 +02:00
logAi->trace("Hero %s is locked by STARTUP. Discarding %s", path.targetHero->getObjectName(), path.toString());
#endif
return true;
}
}
return false;
}
HeroLockedReason Nullkiller::getHeroLockedReason(const CGHeroInstance * hero) const
{
auto found = lockedHeroes.find(hero);
return found != lockedHeroes.end() ? found->second : HeroLockedReason::NOT_LOCKED;
}
void Nullkiller::makeTurn()
{
boost::lock_guard<boost::mutex> sharedStorageLock(AISharedStorage::locker);
const int MAX_DEPTH = 10;
const float FAST_TASK_MINIMAL_PRIORITY = 0.7f;
resetAiState();
for(int i = 1; i <= MAXPASS; i++)
{
updateAiState(i);
2021-05-15 18:23:42 +02:00
2022-09-06 20:14:22 +02:00
Goals::TTask bestTask = taskptr(Goals::Invalid());
2023-07-27 14:58:49 +02:00
for(;i <= MAXPASS; i++)
2022-09-06 20:14:22 +02:00
{
Goals::TTaskVec fastTasks = {
choseBestTask(sptr(BuyArmyBehavior()), 1),
choseBestTask(sptr(BuildingBehavior()), 1)
};
bestTask = choseBestTask(fastTasks);
2023-02-28 09:07:59 +02:00
if(bestTask->priority >= FAST_TASK_MINIMAL_PRIORITY)
2022-09-06 20:14:22 +02:00
{
executeTask(bestTask);
updateAiState(i, true);
}
2023-07-27 14:58:49 +02:00
else
{
break;
}
}
2022-09-06 20:14:22 +02:00
Goals::TTaskVec bestTasks = {
2022-09-06 20:14:22 +02:00
bestTask,
choseBestTask(sptr(RecruitHeroBehavior()), 1),
choseBestTask(sptr(CaptureObjectsBehavior()), 1),
choseBestTask(sptr(ClusterBehavior()), MAX_DEPTH),
choseBestTask(sptr(DefenceBehavior()), MAX_DEPTH),
choseBestTask(sptr(GatherArmyBehavior()), MAX_DEPTH)
};
2021-05-15 21:02:52 +02:00
if(cb->getDate(Date::DAY) == 1)
{
bestTasks.push_back(choseBestTask(sptr(StartupBehavior()), 1));
2021-05-15 21:02:52 +02:00
}
2022-09-06 20:14:22 +02:00
bestTask = choseBestTask(bestTasks);
2021-05-16 14:01:34 +02:00
HeroPtr hero = bestTask->getHero();
2022-09-06 20:14:22 +02:00
HeroRole heroRole = HeroRole::MAIN;
if(hero.validAndSet())
heroRole = heroManager->getHeroRole(hero);
if(heroRole != HeroRole::MAIN || bestTask->getHeroExchangeCount() <= 1)
useHeroChain = false;
2023-07-27 14:58:49 +02:00
// TODO: better to check turn distance here instead of priority
2023-02-28 09:07:59 +02:00
if((heroRole != HeroRole::MAIN || bestTask->priority < SMALL_SCAN_MIN_PRIORITY)
2023-07-27 14:58:49 +02:00
&& scanDepth == ScanDepth::MAIN_FULL)
2021-05-16 14:01:34 +02:00
{
2023-02-28 09:07:59 +02:00
useHeroChain = false;
scanDepth = ScanDepth::SMALL;
2023-02-28 09:07:59 +02:00
logAi->trace(
2023-07-27 14:58:49 +02:00
"Goal %s has low priority %f so decreasing scan depth to gain performance.",
2023-02-28 09:07:59 +02:00
bestTask->toString(),
bestTask->priority);
2021-05-16 14:01:34 +02:00
}
if(bestTask->priority < MIN_PRIORITY)
{
2023-07-27 14:58:49 +02:00
auto heroes = cb->getHeroesInfo();
auto hasMp = vstd::contains_if(heroes, [](const CGHeroInstance * h) -> bool
{
return h->movementPointsRemaining() > 100;
});
if(hasMp && scanDepth != ScanDepth::ALL_FULL)
{
logAi->trace(
"Goal %s has too low priority %f so increasing scan depth to full.",
bestTask->toString(),
bestTask->priority);
scanDepth = ScanDepth::ALL_FULL;
useHeroChain = false;
continue;
}
logAi->trace("Goal %s has too low priority. It is not worth doing it. Ending turn.", bestTask->toString());
return;
}
2022-09-06 20:14:22 +02:00
executeTask(bestTask);
2023-08-05 12:49:49 +02:00
if(i == MAXPASS)
{
logAi->error("Goal %s exceeded maxpass. Terminating AI turn.", bestTask->toString());
}
2022-09-06 20:14:22 +02:00
}
}
2021-05-16 14:08:39 +02:00
2022-09-06 20:14:22 +02:00
void Nullkiller::executeTask(Goals::TTask task)
{
auto start = std::chrono::high_resolution_clock::now();
2022-09-06 20:14:22 +02:00
std::string taskDescr = task->toString();
2022-09-06 20:14:22 +02:00
boost::this_thread::interruption_point();
logAi->debug("Trying to realize %s (value %2.3f)", taskDescr, task->priority);
2022-09-06 20:14:22 +02:00
try
{
task->accept(ai);
logAi->trace("Task %s completed in %lld", taskDescr, timeElapsed(start));
2022-09-06 20:14:22 +02:00
}
catch(goalFulfilledException &)
{
logAi->trace("Task %s completed in %lld", taskDescr, timeElapsed(start));
2022-09-06 20:14:22 +02:00
}
2022-11-03 21:16:49 +02:00
catch(cannotFulfillGoalException & e)
2022-09-06 20:14:22 +02:00
{
2023-02-28 09:07:59 +02:00
logAi->error("Failed to realize subgoal of type %s, I will stop.", taskDescr);
logAi->error("The error message was: %s", e.what());
2022-09-06 20:14:22 +02:00
2023-02-28 09:07:59 +02:00
#if NKAI_TRACE_LEVEL == 0
throw; // will be recatched and AI turn ended
#endif
}
2021-05-16 14:39:38 +02:00
}
TResources Nullkiller::getFreeResources() const
{
auto freeRes = cb->getResourceAmount() - lockedResources;
freeRes.positive();
return freeRes;
}
void Nullkiller::lockResources(const TResources & res)
{
lockedResources += res;
2021-11-23 08:41:03 +02:00
}
2022-09-26 20:01:07 +02:00
}