mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-23 14:54:00 +00:00
96eb8d7be9
Perform all request to db in a single thread to avoid blocking navmesh generator threads due to slow write operations. Write to db navmesh for all changes except update as it done for memory cache. Batch multiple db operations into a single transaction to speed up writing by not executing fsync after each insert/update query. All reads are performed in the same transaction so they see uncommited data.
383 lines
18 KiB
C++
383 lines
18 KiB
C++
#include "operators.hpp"
|
|
#include "generate.hpp"
|
|
|
|
#include <components/detournavigator/navmeshtilescache.hpp>
|
|
#include <components/detournavigator/exceptions.hpp>
|
|
#include <components/detournavigator/recastmesh.hpp>
|
|
#include <components/detournavigator/preparednavmeshdata.hpp>
|
|
#include <components/detournavigator/ref.hpp>
|
|
#include <components/detournavigator/preparednavmeshdatatuple.hpp>
|
|
#include <components/detournavigator/recast.hpp>
|
|
|
|
#include <osg/Vec3f>
|
|
|
|
#include <RecastAlloc.h>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <random>
|
|
#include <stdexcept>
|
|
|
|
namespace
|
|
{
|
|
using namespace testing;
|
|
using namespace DetourNavigator;
|
|
using namespace DetourNavigator::Tests;
|
|
|
|
template <class T, class Random>
|
|
void generateRecastArray(T*& values, int size, Random& random)
|
|
{
|
|
values = static_cast<T*>(permRecastAlloc(size * sizeof(T)));
|
|
generateRange(values, values + static_cast<std::ptrdiff_t>(size), random);
|
|
}
|
|
|
|
template <class Random>
|
|
void generate(rcPolyMesh& value, int size, Random& random)
|
|
{
|
|
value.nverts = size;
|
|
value.maxpolys = size;
|
|
value.nvp = size;
|
|
value.npolys = size;
|
|
rcVcopy(value.bmin, osg::Vec3f(-1, -2, -3).ptr());
|
|
rcVcopy(value.bmax, osg::Vec3f(3, 2, 1).ptr());
|
|
generateValue(value.cs, random);
|
|
generateValue(value.ch, random);
|
|
generateValue(value.borderSize, random);
|
|
generateValue(value.maxEdgeError, random);
|
|
generateRecastArray(value.verts, getVertsLength(value), random);
|
|
generateRecastArray(value.polys, getPolysLength(value), random);
|
|
generateRecastArray(value.regs, getRegsLength(value), random);
|
|
generateRecastArray(value.flags, getFlagsLength(value), random);
|
|
generateRecastArray(value.areas, getAreasLength(value), random);
|
|
}
|
|
|
|
template <class Random>
|
|
void generate(rcPolyMeshDetail& value, int size, Random& random)
|
|
{
|
|
value.nmeshes = size;
|
|
value.nverts = size;
|
|
value.ntris = size;
|
|
generateRecastArray(value.meshes, getMeshesLength(value), random);
|
|
generateRecastArray(value.verts, getVertsLength(value), random);
|
|
generateRecastArray(value.tris, getTrisLength(value), random);
|
|
}
|
|
|
|
template <class Random>
|
|
void generate(PreparedNavMeshData& value, int size, Random& random)
|
|
{
|
|
generateValue(value.mUserId, random);
|
|
generateValue(value.mCellHeight, random);
|
|
generateValue(value.mCellSize, random);
|
|
generate(value.mPolyMesh, size, random);
|
|
generate(value.mPolyMeshDetail, size, random);
|
|
}
|
|
|
|
std::unique_ptr<PreparedNavMeshData> makePeparedNavMeshData(int size)
|
|
{
|
|
std::minstd_rand random;
|
|
auto result = std::make_unique<PreparedNavMeshData>();
|
|
generate(*result, size, random);
|
|
return result;
|
|
}
|
|
|
|
std::unique_ptr<PreparedNavMeshData> clone(const PreparedNavMeshData& value)
|
|
{
|
|
return std::make_unique<PreparedNavMeshData>(value);
|
|
}
|
|
|
|
Mesh makeMesh()
|
|
{
|
|
std::vector<int> indices {{0, 1, 2}};
|
|
std::vector<float> vertices {{0, 0, 0, 1, 0, 0, 1, 1, 0}};
|
|
std::vector<AreaType> areaTypes {1, AreaType_ground};
|
|
return Mesh(std::move(indices), std::move(vertices), std::move(areaTypes));
|
|
}
|
|
|
|
struct DetourNavigatorNavMeshTilesCacheTest : Test
|
|
{
|
|
const osg::Vec3f mAgentHalfExtents {1, 2, 3};
|
|
const TilePosition mTilePosition {0, 0};
|
|
const std::size_t mGeneration = 0;
|
|
const std::size_t mRevision = 0;
|
|
const Mesh mMesh {makeMesh()};
|
|
const std::vector<CellWater> mWater {};
|
|
const std::vector<Heightfield> mHeightfields {};
|
|
const std::vector<FlatHeightfield> mFlatHeightfields {};
|
|
const std::vector<MeshSource> mSources {};
|
|
const RecastMesh mRecastMesh {mGeneration, mRevision, mMesh, mWater, mHeightfields, mFlatHeightfields, mSources};
|
|
std::unique_ptr<PreparedNavMeshData> mPreparedNavMeshData {makePeparedNavMeshData(3)};
|
|
|
|
const std::size_t mRecastMeshSize = sizeof(mRecastMesh) + getSize(mRecastMesh);
|
|
const std::size_t mRecastMeshWithWaterSize = mRecastMeshSize + sizeof(CellWater);
|
|
const std::size_t mPreparedNavMeshDataSize = sizeof(*mPreparedNavMeshData) + getSize(*mPreparedNavMeshData);
|
|
};
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, get_for_empty_cache_should_return_empty_value)
|
|
{
|
|
const std::size_t maxSize = 0;
|
|
NavMeshTilesCache cache(maxSize);
|
|
|
|
EXPECT_FALSE(cache.get(mAgentHalfExtents, mTilePosition, mRecastMesh));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, set_for_not_enought_cache_size_should_return_empty_value)
|
|
{
|
|
const std::size_t maxSize = 0;
|
|
NavMeshTilesCache cache(maxSize);
|
|
|
|
EXPECT_FALSE(cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData)));
|
|
EXPECT_NE(mPreparedNavMeshData, nullptr);
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, set_should_return_cached_value)
|
|
{
|
|
const std::size_t maxSize = mRecastMeshSize + mPreparedNavMeshDataSize;
|
|
NavMeshTilesCache cache(maxSize);
|
|
const auto copy = clone(*mPreparedNavMeshData);
|
|
ASSERT_EQ(*mPreparedNavMeshData, *copy);
|
|
|
|
const auto result = cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData));
|
|
ASSERT_TRUE(result);
|
|
EXPECT_EQ(result.get(), *copy);
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, set_existing_element_should_return_cached_element)
|
|
{
|
|
const std::size_t maxSize = 2 * (mRecastMeshSize + mPreparedNavMeshDataSize);
|
|
NavMeshTilesCache cache(maxSize);
|
|
auto copy = clone(*mPreparedNavMeshData);
|
|
const auto sameCopy = clone(*mPreparedNavMeshData);
|
|
|
|
cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData));
|
|
EXPECT_EQ(mPreparedNavMeshData, nullptr);
|
|
const auto result = cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(copy));
|
|
ASSERT_TRUE(result);
|
|
EXPECT_EQ(result.get(), *sameCopy);
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, get_should_return_cached_value)
|
|
{
|
|
const std::size_t maxSize = mRecastMeshSize + mPreparedNavMeshDataSize;
|
|
NavMeshTilesCache cache(maxSize);
|
|
const auto copy = clone(*mPreparedNavMeshData);
|
|
|
|
cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData));
|
|
const auto result = cache.get(mAgentHalfExtents, mTilePosition, mRecastMesh);
|
|
ASSERT_TRUE(result);
|
|
EXPECT_EQ(result.get(), *copy);
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, get_for_cache_miss_by_agent_half_extents_should_return_empty_value)
|
|
{
|
|
const std::size_t maxSize = 1;
|
|
NavMeshTilesCache cache(maxSize);
|
|
const osg::Vec3f unexsistentAgentHalfExtents {1, 1, 1};
|
|
|
|
cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData));
|
|
EXPECT_FALSE(cache.get(unexsistentAgentHalfExtents, mTilePosition, mRecastMesh));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, get_for_cache_miss_by_tile_position_should_return_empty_value)
|
|
{
|
|
const std::size_t maxSize = 1;
|
|
NavMeshTilesCache cache(maxSize);
|
|
const TilePosition unexistentTilePosition {1, 1};
|
|
|
|
cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData));
|
|
EXPECT_FALSE(cache.get(mAgentHalfExtents, unexistentTilePosition, mRecastMesh));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, get_for_cache_miss_by_recast_mesh_should_return_empty_value)
|
|
{
|
|
const std::size_t maxSize = 1;
|
|
NavMeshTilesCache cache(maxSize);
|
|
const std::vector<CellWater> water(1, CellWater {osg::Vec2i(), Water {1, 0.0f}});
|
|
const RecastMesh unexistentRecastMesh(mGeneration, mRevision, mMesh, water, mHeightfields, mFlatHeightfields, mSources);
|
|
|
|
cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData));
|
|
EXPECT_FALSE(cache.get(mAgentHalfExtents, mTilePosition, unexistentRecastMesh));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, set_should_replace_unused_value)
|
|
{
|
|
const std::size_t maxSize = mRecastMeshWithWaterSize + mPreparedNavMeshDataSize;
|
|
NavMeshTilesCache cache(maxSize);
|
|
|
|
const std::vector<CellWater> water(1, CellWater {osg::Vec2i(), Water {1, 0.0f}});
|
|
const RecastMesh anotherRecastMesh(mGeneration, mRevision, mMesh, water, mHeightfields, mFlatHeightfields, mSources);
|
|
auto anotherPreparedNavMeshData = makePeparedNavMeshData(3);
|
|
const auto copy = clone(*anotherPreparedNavMeshData);
|
|
|
|
cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData));
|
|
const auto result = cache.set(mAgentHalfExtents, mTilePosition, anotherRecastMesh,
|
|
std::move(anotherPreparedNavMeshData));
|
|
ASSERT_TRUE(result);
|
|
EXPECT_EQ(result.get(), *copy);
|
|
EXPECT_FALSE(cache.get(mAgentHalfExtents, mTilePosition, mRecastMesh));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, set_should_not_replace_used_value)
|
|
{
|
|
const std::size_t maxSize = mRecastMeshWithWaterSize + mPreparedNavMeshDataSize;
|
|
NavMeshTilesCache cache(maxSize);
|
|
|
|
const std::vector<CellWater> water(1, CellWater {osg::Vec2i(), Water {1, 0.0f}});
|
|
const RecastMesh anotherRecastMesh(mGeneration, mRevision, mMesh, water, mHeightfields, mFlatHeightfields, mSources);
|
|
auto anotherPreparedNavMeshData = makePeparedNavMeshData(3);
|
|
|
|
const auto value = cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh,
|
|
std::move(mPreparedNavMeshData));
|
|
EXPECT_FALSE(cache.set(mAgentHalfExtents, mTilePosition, anotherRecastMesh,
|
|
std::move(anotherPreparedNavMeshData)));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, set_should_replace_unused_least_recently_set_value)
|
|
{
|
|
const std::size_t maxSize = 2 * (mRecastMeshWithWaterSize + mPreparedNavMeshDataSize);
|
|
NavMeshTilesCache cache(maxSize);
|
|
const auto copy = clone(*mPreparedNavMeshData);
|
|
|
|
const std::vector<CellWater> leastRecentlySetWater(1, CellWater {osg::Vec2i(), Water {1, 0.0f}});
|
|
const RecastMesh leastRecentlySetRecastMesh(mGeneration, mRevision, mMesh, leastRecentlySetWater,
|
|
mHeightfields, mFlatHeightfields, mSources);
|
|
auto leastRecentlySetData = makePeparedNavMeshData(3);
|
|
|
|
const std::vector<CellWater> mostRecentlySetWater(1, CellWater {osg::Vec2i(), Water {2, 0.0f}});
|
|
const RecastMesh mostRecentlySetRecastMesh(mGeneration, mRevision, mMesh, mostRecentlySetWater,
|
|
mHeightfields, mFlatHeightfields, mSources);
|
|
auto mostRecentlySetData = makePeparedNavMeshData(3);
|
|
|
|
ASSERT_TRUE(cache.set(mAgentHalfExtents, mTilePosition, leastRecentlySetRecastMesh,
|
|
std::move(leastRecentlySetData)));
|
|
ASSERT_TRUE(cache.set(mAgentHalfExtents, mTilePosition, mostRecentlySetRecastMesh,
|
|
std::move(mostRecentlySetData)));
|
|
|
|
const auto result = cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh,
|
|
std::move(mPreparedNavMeshData));
|
|
EXPECT_EQ(result.get(), *copy);
|
|
|
|
EXPECT_FALSE(cache.get(mAgentHalfExtents, mTilePosition, leastRecentlySetRecastMesh));
|
|
EXPECT_TRUE(cache.get(mAgentHalfExtents, mTilePosition, mostRecentlySetRecastMesh));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, set_should_replace_unused_least_recently_used_value)
|
|
{
|
|
const std::size_t maxSize = 2 * (mRecastMeshWithWaterSize + mPreparedNavMeshDataSize);
|
|
NavMeshTilesCache cache(maxSize);
|
|
|
|
const std::vector<CellWater> leastRecentlyUsedWater(1, CellWater {osg::Vec2i(), Water {1, 0.0f}});
|
|
const RecastMesh leastRecentlyUsedRecastMesh(mGeneration, mRevision, mMesh, leastRecentlyUsedWater,
|
|
mHeightfields, mFlatHeightfields, mSources);
|
|
auto leastRecentlyUsedData = makePeparedNavMeshData(3);
|
|
const auto leastRecentlyUsedCopy = clone(*leastRecentlyUsedData);
|
|
|
|
const std::vector<CellWater> mostRecentlyUsedWater(1, CellWater {osg::Vec2i(), Water {2, 0.0f}});
|
|
const RecastMesh mostRecentlyUsedRecastMesh(mGeneration, mRevision, mMesh, mostRecentlyUsedWater,
|
|
mHeightfields, mFlatHeightfields, mSources);
|
|
auto mostRecentlyUsedData = makePeparedNavMeshData(3);
|
|
const auto mostRecentlyUsedCopy = clone(*mostRecentlyUsedData);
|
|
|
|
cache.set(mAgentHalfExtents, mTilePosition, leastRecentlyUsedRecastMesh, std::move(leastRecentlyUsedData));
|
|
cache.set(mAgentHalfExtents, mTilePosition, mostRecentlyUsedRecastMesh, std::move(mostRecentlyUsedData));
|
|
|
|
{
|
|
const auto value = cache.get(mAgentHalfExtents, mTilePosition, leastRecentlyUsedRecastMesh);
|
|
ASSERT_TRUE(value);
|
|
ASSERT_EQ(value.get(), *leastRecentlyUsedCopy);
|
|
}
|
|
|
|
{
|
|
const auto value = cache.get(mAgentHalfExtents, mTilePosition, mostRecentlyUsedRecastMesh);
|
|
ASSERT_TRUE(value);
|
|
ASSERT_EQ(value.get(), *mostRecentlyUsedCopy);
|
|
}
|
|
|
|
const auto copy = clone(*mPreparedNavMeshData);
|
|
const auto result = cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh,
|
|
std::move(mPreparedNavMeshData));
|
|
EXPECT_EQ(result.get(), *copy);
|
|
|
|
EXPECT_FALSE(cache.get(mAgentHalfExtents, mTilePosition, leastRecentlyUsedRecastMesh));
|
|
EXPECT_TRUE(cache.get(mAgentHalfExtents, mTilePosition, mostRecentlyUsedRecastMesh));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, set_should_not_replace_unused_least_recently_used_value_when_item_does_not_not_fit_cache_max_size)
|
|
{
|
|
const std::size_t maxSize = 2 * (mRecastMeshWithWaterSize + mPreparedNavMeshDataSize);
|
|
NavMeshTilesCache cache(maxSize);
|
|
|
|
const std::vector<CellWater> water(1, CellWater {osg::Vec2i(), Water {1, 0.0f}});
|
|
const RecastMesh tooLargeRecastMesh(mGeneration, mRevision, mMesh, water,
|
|
mHeightfields, mFlatHeightfields, mSources);
|
|
auto tooLargeData = makePeparedNavMeshData(10);
|
|
|
|
cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData));
|
|
EXPECT_FALSE(cache.set(mAgentHalfExtents, mTilePosition, tooLargeRecastMesh, std::move(tooLargeData)));
|
|
EXPECT_TRUE(cache.get(mAgentHalfExtents, mTilePosition, mRecastMesh));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, set_should_not_replace_unused_least_recently_used_value_when_item_does_not_not_fit_size_of_unused_items)
|
|
{
|
|
const std::size_t maxSize = 2 * (mRecastMeshWithWaterSize + mPreparedNavMeshDataSize);
|
|
NavMeshTilesCache cache(maxSize);
|
|
|
|
const std::vector<CellWater> anotherWater(1, CellWater {osg::Vec2i(), Water {1, 0.0f}});
|
|
const RecastMesh anotherRecastMesh(mGeneration, mRevision, mMesh, anotherWater,
|
|
mHeightfields, mFlatHeightfields, mSources);
|
|
auto anotherData = makePeparedNavMeshData(3);
|
|
|
|
const std::vector<CellWater> tooLargeWater(1, CellWater {osg::Vec2i(), Water {2, 0.0f}});
|
|
const RecastMesh tooLargeRecastMesh(mGeneration, mRevision, mMesh, tooLargeWater,
|
|
mHeightfields, mFlatHeightfields, mSources);
|
|
auto tooLargeData = makePeparedNavMeshData(10);
|
|
|
|
const auto value = cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh,
|
|
std::move(mPreparedNavMeshData));
|
|
ASSERT_TRUE(value);
|
|
ASSERT_TRUE(cache.set(mAgentHalfExtents, mTilePosition, anotherRecastMesh,
|
|
std::move(anotherData)));
|
|
EXPECT_FALSE(cache.set(mAgentHalfExtents, mTilePosition, tooLargeRecastMesh,
|
|
std::move(tooLargeData)));
|
|
EXPECT_TRUE(cache.get(mAgentHalfExtents, mTilePosition, mRecastMesh));
|
|
EXPECT_TRUE(cache.get(mAgentHalfExtents, mTilePosition, anotherRecastMesh));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, release_used_after_set_then_used_by_get_item_should_left_this_item_available)
|
|
{
|
|
const std::size_t maxSize = mRecastMeshWithWaterSize + mPreparedNavMeshDataSize;
|
|
NavMeshTilesCache cache(maxSize);
|
|
|
|
const std::vector<CellWater> water(1, CellWater {osg::Vec2i(), Water {1, 0.0f}});
|
|
const RecastMesh anotherRecastMesh(mGeneration, mRevision, mMesh, water, mHeightfields, mFlatHeightfields, mSources);
|
|
auto anotherData = makePeparedNavMeshData(3);
|
|
|
|
const auto firstCopy = cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData));
|
|
ASSERT_TRUE(firstCopy);
|
|
{
|
|
const auto secondCopy = cache.get(mAgentHalfExtents, mTilePosition, mRecastMesh);
|
|
ASSERT_TRUE(secondCopy);
|
|
}
|
|
EXPECT_FALSE(cache.set(mAgentHalfExtents, mTilePosition, anotherRecastMesh, std::move(anotherData)));
|
|
EXPECT_TRUE(cache.get(mAgentHalfExtents, mTilePosition, mRecastMesh));
|
|
}
|
|
|
|
TEST_F(DetourNavigatorNavMeshTilesCacheTest, release_twice_used_item_should_left_this_item_available)
|
|
{
|
|
const std::size_t maxSize = mRecastMeshWithWaterSize + mPreparedNavMeshDataSize;
|
|
NavMeshTilesCache cache(maxSize);
|
|
|
|
const std::vector<CellWater> water(1, CellWater {osg::Vec2i(), Water {1, 0.0f}});
|
|
const RecastMesh anotherRecastMesh(mGeneration, mRevision, mMesh, water, mHeightfields, mFlatHeightfields, mSources);
|
|
auto anotherData = makePeparedNavMeshData(3);
|
|
|
|
cache.set(mAgentHalfExtents, mTilePosition, mRecastMesh, std::move(mPreparedNavMeshData));
|
|
const auto firstCopy = cache.get(mAgentHalfExtents, mTilePosition, mRecastMesh);
|
|
ASSERT_TRUE(firstCopy);
|
|
{
|
|
const auto secondCopy = cache.get(mAgentHalfExtents, mTilePosition, mRecastMesh);
|
|
ASSERT_TRUE(secondCopy);
|
|
}
|
|
EXPECT_FALSE(cache.set(mAgentHalfExtents, mTilePosition, anotherRecastMesh, std::move(anotherData)));
|
|
EXPECT_TRUE(cache.get(mAgentHalfExtents, mTilePosition, mRecastMesh));
|
|
}
|
|
}
|