#include "navmeshtilescache.hpp" #include "exceptions.hpp" #include namespace DetourNavigator { namespace { inline std::string makeNavMeshKey(const RecastMesh& recastMesh, const std::vector& offMeshConnections) { std::string result; result.reserve( recastMesh.getIndices().size() * sizeof(int) + recastMesh.getVertices().size() * sizeof(float) + recastMesh.getAreaTypes().size() * sizeof(AreaType) + recastMesh.getWater().size() * sizeof(RecastMesh::Water) + offMeshConnections.size() * sizeof(OffMeshConnection) ); std::copy( reinterpret_cast(recastMesh.getIndices().data()), reinterpret_cast(recastMesh.getIndices().data() + recastMesh.getIndices().size()), std::back_inserter(result) ); std::copy( reinterpret_cast(recastMesh.getVertices().data()), reinterpret_cast(recastMesh.getVertices().data() + recastMesh.getVertices().size()), std::back_inserter(result) ); std::copy( reinterpret_cast(recastMesh.getAreaTypes().data()), reinterpret_cast(recastMesh.getAreaTypes().data() + recastMesh.getAreaTypes().size()), std::back_inserter(result) ); std::copy( reinterpret_cast(recastMesh.getWater().data()), reinterpret_cast(recastMesh.getWater().data() + recastMesh.getWater().size()), std::back_inserter(result) ); std::copy( reinterpret_cast(offMeshConnections.data()), reinterpret_cast(offMeshConnections.data() + offMeshConnections.size()), std::back_inserter(result) ); return result; } } NavMeshTilesCache::NavMeshTilesCache(const std::size_t maxNavMeshDataSize) : mMaxNavMeshDataSize(maxNavMeshDataSize), mUsedNavMeshDataSize(0), mFreeNavMeshDataSize(0) {} NavMeshTilesCache::Value NavMeshTilesCache::get(const osg::Vec3f& agentHalfExtents, const TilePosition& changedTile, const RecastMesh& recastMesh, const std::vector& offMeshConnections) { const std::lock_guard lock(mMutex); const auto agentValues = mValues.find(agentHalfExtents); if (agentValues == mValues.end()) return Value(); const auto tileValues = agentValues->second.find(changedTile); if (tileValues == agentValues->second.end()) return Value(); const auto tile = tileValues->second.mMap.find(RecastMeshKeyView(recastMesh, offMeshConnections)); if (tile == tileValues->second.mMap.end()) return Value(); acquireItemUnsafe(tile->second); return Value(*this, tile->second); } NavMeshTilesCache::Value NavMeshTilesCache::set(const osg::Vec3f& agentHalfExtents, const TilePosition& changedTile, const RecastMesh& recastMesh, const std::vector& offMeshConnections, NavMeshData&& value) { const auto navMeshSize = static_cast(value.mSize); const std::lock_guard lock(mMutex); if (navMeshSize > mMaxNavMeshDataSize) return Value(); if (navMeshSize > mFreeNavMeshDataSize + (mMaxNavMeshDataSize - mUsedNavMeshDataSize)) return Value(); auto navMeshKey = makeNavMeshKey(recastMesh, offMeshConnections); const auto itemSize = navMeshSize + 2 * navMeshKey.size(); if (itemSize > mFreeNavMeshDataSize + (mMaxNavMeshDataSize - mUsedNavMeshDataSize)) return Value(); while (!mFreeItems.empty() && mUsedNavMeshDataSize + itemSize > mMaxNavMeshDataSize) removeLeastRecentlyUsed(); const auto iterator = mFreeItems.emplace(mFreeItems.end(), agentHalfExtents, changedTile, std::move(navMeshKey)); const auto emplaced = mValues[agentHalfExtents][changedTile].mMap.emplace(iterator->mNavMeshKey, iterator); if (!emplaced.second) { mFreeItems.erase(iterator); throw InvalidArgument("Set existing cache value"); } iterator->mNavMeshData = std::move(value); mUsedNavMeshDataSize += itemSize; mFreeNavMeshDataSize += itemSize; acquireItemUnsafe(iterator); return Value(*this, iterator); } void NavMeshTilesCache::removeLeastRecentlyUsed() { const auto& item = mFreeItems.back(); const auto agentValues = mValues.find(item.mAgentHalfExtents); if (agentValues == mValues.end()) return; const auto tileValues = agentValues->second.find(item.mChangedTile); if (tileValues == agentValues->second.end()) return; const auto value = tileValues->second.mMap.find(item.mNavMeshKey); if (value == tileValues->second.mMap.end()) return; mUsedNavMeshDataSize -= getSize(item); mFreeNavMeshDataSize -= getSize(item); tileValues->second.mMap.erase(value); mFreeItems.pop_back(); if (!tileValues->second.mMap.empty()) return; agentValues->second.erase(tileValues); if (!agentValues->second.empty()) return; mValues.erase(agentValues); } void NavMeshTilesCache::acquireItemUnsafe(ItemIterator iterator) { if (++iterator->mUseCount > 1) return; mBusyItems.splice(mBusyItems.end(), mFreeItems, iterator); mFreeNavMeshDataSize -= getSize(*iterator); } void NavMeshTilesCache::releaseItem(ItemIterator iterator) { if (--iterator->mUseCount > 0) return; const std::lock_guard lock(mMutex); mFreeItems.splice(mFreeItems.begin(), mBusyItems, iterator); mFreeNavMeshDataSize += getSize(*iterator); } namespace { struct CompareBytes { const char* mRhsIt; const char* mRhsEnd; template int operator ()(const std::vector& lhs) { const auto lhsBegin = reinterpret_cast(lhs.data()); const auto lhsEnd = reinterpret_cast(lhs.data() + lhs.size()); const auto lhsSize = static_cast(lhsEnd - lhsBegin); const auto rhsSize = static_cast(mRhsEnd - mRhsIt); const auto size = std::min(lhsSize, rhsSize); if (const auto result = std::memcmp(lhsBegin, mRhsIt, size)) return result; if (lhsSize > rhsSize) return 1; mRhsIt += size; return 0; } }; } int NavMeshTilesCache::RecastMeshKeyView::compare(const std::string& other) const { CompareBytes compareBytes {other.data(), other.data() + other.size()}; if (const auto result = compareBytes(mRecastMesh.get().getIndices())) return result; if (const auto result = compareBytes(mRecastMesh.get().getVertices())) return result; if (const auto result = compareBytes(mRecastMesh.get().getAreaTypes())) return result; if (const auto result = compareBytes(mRecastMesh.get().getWater())) return result; if (const auto result = compareBytes(mOffMeshConnections.get())) return result; if (compareBytes.mRhsIt < compareBytes.mRhsEnd) return -1; return 0; } }