2021-08-05 22:05:09 +00:00
|
|
|
#include "settings.hpp"
|
|
|
|
|
|
|
|
#include <components/detournavigator/asyncnavmeshupdater.hpp>
|
|
|
|
#include <components/detournavigator/makenavmesh.hpp>
|
|
|
|
#include <components/detournavigator/serialization.hpp>
|
|
|
|
#include <components/loadinglistener/loadinglistener.hpp>
|
|
|
|
#include <components/detournavigator/navmeshdbutils.hpp>
|
|
|
|
#include <components/detournavigator/dbrefgeometryobject.hpp>
|
|
|
|
|
|
|
|
#include <DetourNavMesh.h>
|
|
|
|
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
|
|
|
|
#include <map>
|
2022-03-10 17:34:35 +00:00
|
|
|
#include <limits>
|
2021-08-05 22:05:09 +00:00
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
using namespace testing;
|
|
|
|
using namespace DetourNavigator;
|
|
|
|
using namespace DetourNavigator::Tests;
|
|
|
|
|
2022-03-10 17:34:35 +00:00
|
|
|
void addHeightFieldPlane(TileCachedRecastMeshManager& recastMeshManager, const osg::Vec2i cellPosition = osg::Vec2i(0, 0))
|
2021-08-05 22:05:09 +00:00
|
|
|
{
|
|
|
|
const int cellSize = 8192;
|
|
|
|
recastMeshManager.addHeightfield(cellPosition, cellSize, HeightfieldPlane {0});
|
|
|
|
}
|
|
|
|
|
2021-12-16 21:40:02 +00:00
|
|
|
void addObject(const btBoxShape& shape, TileCachedRecastMeshManager& recastMeshManager)
|
|
|
|
{
|
|
|
|
const ObjectId id(&shape);
|
|
|
|
osg::ref_ptr<Resource::BulletShape> bulletShape(new Resource::BulletShape);
|
|
|
|
bulletShape->mFileName = "test.nif";
|
|
|
|
bulletShape->mFileHash = "test_hash";
|
|
|
|
ObjectTransform objectTransform;
|
|
|
|
std::fill(std::begin(objectTransform.mPosition.pos), std::end(objectTransform.mPosition.pos), 0.1f);
|
|
|
|
std::fill(std::begin(objectTransform.mPosition.rot), std::end(objectTransform.mPosition.rot), 0.2f);
|
|
|
|
objectTransform.mScale = 3.14f;
|
|
|
|
const CollisionShape collisionShape(
|
|
|
|
osg::ref_ptr<Resource::BulletShapeInstance>(new Resource::BulletShapeInstance(bulletShape)),
|
|
|
|
shape, objectTransform
|
|
|
|
);
|
2022-02-01 00:21:47 +00:00
|
|
|
recastMeshManager.addObject(id, collisionShape, btTransform::getIdentity(), AreaType_ground, [] (auto) {});
|
2021-12-16 21:40:02 +00:00
|
|
|
}
|
|
|
|
|
2021-08-05 22:05:09 +00:00
|
|
|
struct DetourNavigatorAsyncNavMeshUpdaterTest : Test
|
|
|
|
{
|
|
|
|
Settings mSettings = makeSettings();
|
|
|
|
TileCachedRecastMeshManager mRecastMeshManager {mSettings.mRecast};
|
|
|
|
OffMeshConnectionsManager mOffMeshConnectionsManager {mSettings.mRecast};
|
|
|
|
const osg::Vec3f mAgentHalfExtents {29, 29, 66};
|
|
|
|
const TilePosition mPlayerTile {0, 0};
|
|
|
|
const std::string mWorldspace = "sys::default";
|
2021-12-16 21:40:02 +00:00
|
|
|
const btBoxShape mBox {btVector3(100, 100, 20)};
|
2021-08-05 22:05:09 +00:00
|
|
|
Loading::Listener mListener;
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, for_all_jobs_done_when_empty_wait_should_terminate)
|
|
|
|
{
|
|
|
|
AsyncNavMeshUpdater updater {mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr};
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, for_required_tiles_present_when_empty_wait_should_terminate)
|
|
|
|
{
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
|
|
|
updater.wait(mListener, WaitConditionType::requiredTilesPresent);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_should_generate_navmesh_tile)
|
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace);
|
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(makeEmptyNavMesh(mSettings), 1);
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles {{TilePosition {0, 0}, ChangeType::add}};
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
EXPECT_NE(navMeshCacheItem->lockConst()->getImpl().getTileRefAt(0, 0, 0), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, repeated_post_should_lead_to_cache_hit)
|
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace);
|
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(makeEmptyNavMesh(mSettings), 1);
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles {{TilePosition {0, 0}, ChangeType::add}};
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
ASSERT_EQ(stats.mCache.mGetCount, 1);
|
|
|
|
ASSERT_EQ(stats.mCache.mHitCount, 0);
|
|
|
|
}
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
EXPECT_EQ(stats.mCache.mGetCount, 2);
|
|
|
|
EXPECT_EQ(stats.mCache.mHitCount, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_for_update_change_type_should_not_update_cache)
|
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace);
|
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(makeEmptyNavMesh(mSettings), 1);
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles {{TilePosition {0, 0}, ChangeType::update}};
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
ASSERT_EQ(stats.mCache.mGetCount, 1);
|
|
|
|
ASSERT_EQ(stats.mCache.mHitCount, 0);
|
|
|
|
}
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
EXPECT_EQ(stats.mCache.mGetCount, 2);
|
|
|
|
EXPECT_EQ(stats.mCache.mHitCount, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_should_write_generated_tile_to_db)
|
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace);
|
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
2021-12-16 21:40:02 +00:00
|
|
|
addObject(mBox, mRecastMeshManager);
|
2022-03-10 17:34:35 +00:00
|
|
|
auto db = std::make_unique<NavMeshDb>(":memory:", std::numeric_limits<std::uint64_t>::max());
|
2021-08-05 22:05:09 +00:00
|
|
|
NavMeshDb* const dbPtr = db.get();
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, std::move(db));
|
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(makeEmptyNavMesh(mSettings), 1);
|
|
|
|
const TilePosition tilePosition {0, 0};
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles {{tilePosition, ChangeType::add}};
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
2022-03-10 17:34:35 +00:00
|
|
|
updater.stop();
|
2021-08-05 22:05:09 +00:00
|
|
|
const auto recastMesh = mRecastMeshManager.getMesh(mWorldspace, tilePosition);
|
|
|
|
ASSERT_NE(recastMesh, nullptr);
|
|
|
|
ShapeId nextShapeId {1};
|
|
|
|
const std::vector<DbRefGeometryObject> objects = makeDbRefGeometryObjects(recastMesh->getMeshSources(),
|
|
|
|
[&] (const MeshSource& v) { return resolveMeshSource(*dbPtr, v, nextShapeId); });
|
|
|
|
const auto tile = dbPtr->findTile(mWorldspace, tilePosition, serialize(mSettings.mRecast, *recastMesh, objects));
|
|
|
|
ASSERT_TRUE(tile.has_value());
|
|
|
|
EXPECT_EQ(tile->mTileId, 1);
|
|
|
|
EXPECT_EQ(tile->mVersion, mSettings.mNavMeshVersion);
|
|
|
|
}
|
|
|
|
|
2021-12-16 21:40:02 +00:00
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_when_writing_to_db_disabled_should_not_write_tiles)
|
2021-08-05 22:05:09 +00:00
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace);
|
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
2021-12-16 21:40:02 +00:00
|
|
|
addObject(mBox, mRecastMeshManager);
|
2022-03-10 17:34:35 +00:00
|
|
|
auto db = std::make_unique<NavMeshDb>(":memory:", std::numeric_limits<std::uint64_t>::max());
|
2021-08-05 22:05:09 +00:00
|
|
|
NavMeshDb* const dbPtr = db.get();
|
|
|
|
mSettings.mWriteToNavMeshDb = false;
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, std::move(db));
|
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(makeEmptyNavMesh(mSettings), 1);
|
|
|
|
const TilePosition tilePosition {0, 0};
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles {{tilePosition, ChangeType::add}};
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
2022-03-10 17:34:35 +00:00
|
|
|
updater.stop();
|
2021-08-05 22:05:09 +00:00
|
|
|
const auto recastMesh = mRecastMeshManager.getMesh(mWorldspace, tilePosition);
|
|
|
|
ASSERT_NE(recastMesh, nullptr);
|
|
|
|
ShapeId nextShapeId {1};
|
|
|
|
const std::vector<DbRefGeometryObject> objects = makeDbRefGeometryObjects(recastMesh->getMeshSources(),
|
|
|
|
[&] (const MeshSource& v) { return resolveMeshSource(*dbPtr, v, nextShapeId); });
|
|
|
|
const auto tile = dbPtr->findTile(mWorldspace, tilePosition, serialize(mSettings.mRecast, *recastMesh, objects));
|
|
|
|
ASSERT_FALSE(tile.has_value());
|
|
|
|
}
|
|
|
|
|
2021-12-16 21:40:02 +00:00
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_when_writing_to_db_disabled_should_not_write_shapes)
|
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace);
|
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
addObject(mBox, mRecastMeshManager);
|
2022-03-10 17:34:35 +00:00
|
|
|
auto db = std::make_unique<NavMeshDb>(":memory:", std::numeric_limits<std::uint64_t>::max());
|
2021-12-16 21:40:02 +00:00
|
|
|
NavMeshDb* const dbPtr = db.get();
|
|
|
|
mSettings.mWriteToNavMeshDb = false;
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, std::move(db));
|
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(makeEmptyNavMesh(mSettings), 1);
|
|
|
|
const TilePosition tilePosition {0, 0};
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles {{tilePosition, ChangeType::add}};
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
2022-03-10 17:34:35 +00:00
|
|
|
updater.stop();
|
2021-12-16 21:40:02 +00:00
|
|
|
const auto recastMesh = mRecastMeshManager.getMesh(mWorldspace, tilePosition);
|
|
|
|
ASSERT_NE(recastMesh, nullptr);
|
|
|
|
const auto objects = makeDbRefGeometryObjects(recastMesh->getMeshSources(),
|
|
|
|
[&] (const MeshSource& v) { return resolveMeshSource(*dbPtr, v); });
|
|
|
|
EXPECT_FALSE(objects.has_value());
|
|
|
|
}
|
|
|
|
|
2021-08-05 22:05:09 +00:00
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, post_should_read_from_db_on_cache_miss)
|
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace);
|
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
mSettings.mMaxNavMeshTilesCacheSize = 0;
|
2022-03-10 17:34:35 +00:00
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager,
|
|
|
|
std::make_unique<NavMeshDb>(":memory:", std::numeric_limits<std::uint64_t>::max()));
|
2021-08-05 22:05:09 +00:00
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(makeEmptyNavMesh(mSettings), 1);
|
|
|
|
const std::map<TilePosition, ChangeType> changedTiles {{TilePosition {0, 0}, ChangeType::add}};
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
ASSERT_EQ(stats.mCache.mGetCount, 1);
|
|
|
|
ASSERT_EQ(stats.mCache.mHitCount, 0);
|
|
|
|
ASSERT_TRUE(stats.mDb.has_value());
|
|
|
|
ASSERT_EQ(stats.mDb->mGetTileCount, 1);
|
|
|
|
ASSERT_EQ(stats.mDbGetTileHits, 0);
|
|
|
|
}
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
{
|
|
|
|
const auto stats = updater.getStats();
|
|
|
|
EXPECT_EQ(stats.mCache.mGetCount, 2);
|
|
|
|
EXPECT_EQ(stats.mCache.mHitCount, 0);
|
|
|
|
ASSERT_TRUE(stats.mDb.has_value());
|
|
|
|
EXPECT_EQ(stats.mDb->mGetTileCount, 2);
|
|
|
|
EXPECT_EQ(stats.mDbGetTileHits, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, on_changing_player_tile_post_should_remove_tiles_out_of_range)
|
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace);
|
|
|
|
addHeightFieldPlane(mRecastMeshManager);
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, nullptr);
|
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(makeEmptyNavMesh(mSettings), 1);
|
|
|
|
const std::map<TilePosition, ChangeType> changedTilesAdd {{TilePosition {0, 0}, ChangeType::add}};
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTilesAdd);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
ASSERT_NE(navMeshCacheItem->lockConst()->getImpl().getTileRefAt(0, 0, 0), 0);
|
|
|
|
const std::map<TilePosition, ChangeType> changedTilesRemove {{TilePosition {0, 0}, ChangeType::remove}};
|
|
|
|
const TilePosition playerTile(100, 100);
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, playerTile, mWorldspace, changedTilesRemove);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
EXPECT_EQ(navMeshCacheItem->lockConst()->getImpl().getTileRefAt(0, 0, 0), 0);
|
|
|
|
}
|
2022-03-10 17:34:35 +00:00
|
|
|
|
|
|
|
TEST_F(DetourNavigatorAsyncNavMeshUpdaterTest, should_stop_writing_to_db_when_size_limit_is_reached)
|
|
|
|
{
|
|
|
|
mRecastMeshManager.setWorldspace(mWorldspace);
|
|
|
|
for (int x = -1; x <= 1; ++x)
|
|
|
|
for (int y = -1; y <= 1; ++y)
|
|
|
|
addHeightFieldPlane(mRecastMeshManager, osg::Vec2i(x, y));
|
|
|
|
addObject(mBox, mRecastMeshManager);
|
|
|
|
auto db = std::make_unique<NavMeshDb>(":memory:", 4097);
|
|
|
|
NavMeshDb* const dbPtr = db.get();
|
|
|
|
AsyncNavMeshUpdater updater(mSettings, mRecastMeshManager, mOffMeshConnectionsManager, std::move(db));
|
|
|
|
const auto navMeshCacheItem = std::make_shared<GuardedNavMeshCacheItem>(makeEmptyNavMesh(mSettings), 1);
|
|
|
|
std::map<TilePosition, ChangeType> changedTiles;
|
|
|
|
for (int x = -5; x <= 5; ++x)
|
|
|
|
for (int y = -5; y <= 5; ++y)
|
|
|
|
changedTiles.emplace(TilePosition {x, y}, ChangeType::add);
|
|
|
|
updater.post(mAgentHalfExtents, navMeshCacheItem, mPlayerTile, mWorldspace, changedTiles);
|
|
|
|
updater.wait(mListener, WaitConditionType::allJobsDone);
|
|
|
|
updater.stop();
|
2022-05-08 19:28:43 +00:00
|
|
|
const std::set<TilePosition> present {
|
|
|
|
TilePosition(-2, 0),
|
|
|
|
TilePosition(-1, -1),
|
|
|
|
TilePosition(-1, 0),
|
|
|
|
TilePosition(-1, 1),
|
|
|
|
TilePosition(0, -2),
|
|
|
|
TilePosition(0, -1),
|
|
|
|
TilePosition(0, 0),
|
|
|
|
TilePosition(0, 1),
|
|
|
|
TilePosition(0, 2),
|
|
|
|
TilePosition(1, -1),
|
|
|
|
TilePosition(1, 0),
|
|
|
|
TilePosition(1, 1),
|
|
|
|
};
|
2022-03-10 17:34:35 +00:00
|
|
|
for (int x = -5; x <= 5; ++x)
|
|
|
|
for (int y = -5; y <= 5; ++y)
|
|
|
|
{
|
|
|
|
const TilePosition tilePosition(x, y);
|
|
|
|
const auto recastMesh = mRecastMeshManager.getMesh(mWorldspace, tilePosition);
|
|
|
|
ASSERT_NE(recastMesh, nullptr);
|
|
|
|
const std::optional<std::vector<DbRefGeometryObject>> objects = makeDbRefGeometryObjects(recastMesh->getMeshSources(),
|
|
|
|
[&] (const MeshSource& v) { return resolveMeshSource(*dbPtr, v); });
|
|
|
|
if (!objects.has_value())
|
|
|
|
continue;
|
2022-05-08 19:28:43 +00:00
|
|
|
EXPECT_EQ(dbPtr->findTile(mWorldspace, tilePosition, serialize(mSettings.mRecast, *recastMesh, *objects)).has_value(),
|
|
|
|
present.find(tilePosition) != present.end())
|
|
|
|
<< tilePosition.x() << " " << tilePosition.y() << " present=" << (present.find(tilePosition) != present.end());
|
2022-03-10 17:34:35 +00:00
|
|
|
}
|
|
|
|
}
|
2021-08-05 22:05:09 +00:00
|
|
|
}
|