mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-20 08:53:52 +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.
181 lines
5.2 KiB
C++
181 lines
5.2 KiB
C++
#ifndef OPENMW_COMPONENTS_DETOURNAVIGATOR_NAVMESHCACHEITEM_H
|
|
#define OPENMW_COMPONENTS_DETOURNAVIGATOR_NAVMESHCACHEITEM_H
|
|
|
|
#include "sharednavmesh.hpp"
|
|
#include "tileposition.hpp"
|
|
#include "navmeshtilescache.hpp"
|
|
#include "dtstatus.hpp"
|
|
#include "navmeshdata.hpp"
|
|
#include "version.hpp"
|
|
|
|
#include <components/misc/guarded.hpp>
|
|
|
|
#include <map>
|
|
#include <ostream>
|
|
#include <set>
|
|
|
|
struct dtMeshTile;
|
|
|
|
namespace DetourNavigator
|
|
{
|
|
enum class UpdateNavMeshStatus : unsigned
|
|
{
|
|
ignored = 0,
|
|
removed = 1 << 0,
|
|
added = 1 << 1,
|
|
replaced = removed | added,
|
|
failed = 1 << 2,
|
|
lost = removed | failed,
|
|
cached = 1 << 3,
|
|
unchanged = replaced | cached,
|
|
restored = added | cached,
|
|
};
|
|
|
|
inline bool isSuccess(UpdateNavMeshStatus value)
|
|
{
|
|
return (static_cast<unsigned>(value) & static_cast<unsigned>(UpdateNavMeshStatus::failed)) == 0;
|
|
}
|
|
|
|
inline std::ostream& operator <<(std::ostream& stream, UpdateNavMeshStatus value)
|
|
{
|
|
switch (value)
|
|
{
|
|
case UpdateNavMeshStatus::ignored:
|
|
return stream << "ignore";
|
|
case UpdateNavMeshStatus::removed:
|
|
return stream << "removed";
|
|
case UpdateNavMeshStatus::added:
|
|
return stream << "add";
|
|
case UpdateNavMeshStatus::replaced:
|
|
return stream << "replaced";
|
|
case UpdateNavMeshStatus::failed:
|
|
return stream << "failed";
|
|
case UpdateNavMeshStatus::lost:
|
|
return stream << "lost";
|
|
case UpdateNavMeshStatus::cached:
|
|
return stream << "cached";
|
|
case UpdateNavMeshStatus::unchanged:
|
|
return stream << "unchanged";
|
|
case UpdateNavMeshStatus::restored:
|
|
return stream << "restored";
|
|
}
|
|
return stream << "unknown(" << static_cast<unsigned>(value) << ")";
|
|
}
|
|
|
|
class UpdateNavMeshStatusBuilder
|
|
{
|
|
public:
|
|
UpdateNavMeshStatusBuilder() = default;
|
|
|
|
explicit UpdateNavMeshStatusBuilder(UpdateNavMeshStatus value)
|
|
: mResult(value) {}
|
|
|
|
UpdateNavMeshStatusBuilder removed(bool value)
|
|
{
|
|
if (value)
|
|
set(UpdateNavMeshStatus::removed);
|
|
else
|
|
unset(UpdateNavMeshStatus::removed);
|
|
return *this;
|
|
}
|
|
|
|
UpdateNavMeshStatusBuilder added(bool value)
|
|
{
|
|
if (value)
|
|
set(UpdateNavMeshStatus::added);
|
|
else
|
|
unset(UpdateNavMeshStatus::added);
|
|
return *this;
|
|
}
|
|
|
|
UpdateNavMeshStatusBuilder failed(bool value)
|
|
{
|
|
if (value)
|
|
set(UpdateNavMeshStatus::failed);
|
|
else
|
|
unset(UpdateNavMeshStatus::failed);
|
|
return *this;
|
|
}
|
|
|
|
UpdateNavMeshStatusBuilder cached(bool value)
|
|
{
|
|
if (value)
|
|
set(UpdateNavMeshStatus::cached);
|
|
else
|
|
unset(UpdateNavMeshStatus::cached);
|
|
return *this;
|
|
}
|
|
|
|
UpdateNavMeshStatus getResult() const
|
|
{
|
|
return mResult;
|
|
}
|
|
|
|
private:
|
|
UpdateNavMeshStatus mResult = UpdateNavMeshStatus::ignored;
|
|
|
|
void set(UpdateNavMeshStatus value)
|
|
{
|
|
mResult = static_cast<UpdateNavMeshStatus>(static_cast<unsigned>(mResult) | static_cast<unsigned>(value));
|
|
}
|
|
|
|
void unset(UpdateNavMeshStatus value)
|
|
{
|
|
mResult = static_cast<UpdateNavMeshStatus>(static_cast<unsigned>(mResult) & ~static_cast<unsigned>(value));
|
|
}
|
|
};
|
|
|
|
const dtMeshTile* getTile(const dtNavMesh& navMesh, const TilePosition& position);
|
|
|
|
class NavMeshCacheItem
|
|
{
|
|
public:
|
|
NavMeshCacheItem(const NavMeshPtr& impl, std::size_t generation)
|
|
: mImpl(impl)
|
|
, mVersion {generation, 0}
|
|
{
|
|
}
|
|
|
|
const dtNavMesh& getImpl() const
|
|
{
|
|
return *mImpl;
|
|
}
|
|
|
|
const Version& getVersion() const { return mVersion; }
|
|
|
|
UpdateNavMeshStatus updateTile(const TilePosition& position, NavMeshTilesCache::Value&& cached,
|
|
NavMeshData&& navMeshData);
|
|
|
|
UpdateNavMeshStatus removeTile(const TilePosition& position);
|
|
|
|
UpdateNavMeshStatus markAsEmpty(const TilePosition& position);
|
|
|
|
bool isEmptyTile(const TilePosition& position) const;
|
|
|
|
template <class Function>
|
|
void forEachUsedTile(Function&& function) const
|
|
{
|
|
for (const auto& [position, tile] : mUsedTiles)
|
|
if (const dtMeshTile* meshTile = getTile(*mImpl, position))
|
|
function(position, tile.mVersion, *meshTile);
|
|
}
|
|
|
|
private:
|
|
struct Tile
|
|
{
|
|
Version mVersion;
|
|
NavMeshTilesCache::Value mCached;
|
|
NavMeshData mData;
|
|
};
|
|
|
|
NavMeshPtr mImpl;
|
|
Version mVersion;
|
|
std::map<TilePosition, Tile> mUsedTiles;
|
|
std::set<TilePosition> mEmptyTiles;
|
|
};
|
|
|
|
using GuardedNavMeshCacheItem = Misc::ScopeGuarded<NavMeshCacheItem>;
|
|
using SharedNavMeshCacheItem = std::shared_ptr<GuardedNavMeshCacheItem>;
|
|
}
|
|
|
|
#endif
|