-
-
Notifications
You must be signed in to change notification settings - Fork 21.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Make navigation map spatial queries thread-safe #79577
Conversation
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Looks good.
My only concern is with map_update_id
, which is checked in ERR_FAIL_COND
when the read lock is still not held.
@RandomShaper I don't think |
My concern is with the following situation: just after the main thread increases Making it a On a side note, zero seems to be a valid value for |
Included the
Now that I think about it, I think it should be changed so that in case of a wrap it does not trigger all the query error checks that fire when E.g. from |
efafdae
to
7e3a246
Compare
I'd suggest using RAII locking, safer, and none of the locking here is very complex so would be better either way, like so: diff --git a/modules/navigation/nav_map.cpp b/modules/navigation/nav_map.cpp
index fa591b6a44..78aba52854 100644
--- a/modules/navigation/nav_map.cpp
+++ b/modules/navigation/nav_map.cpp
@@ -116,9 +116,8 @@ gd::PointKey NavMap::get_point_key(const Vector3 &p_pos) const {
}
Vector<Vector3> NavMap::get_path(Vector3 p_origin, Vector3 p_destination, bool p_optimize, uint32_t p_navigation_layers, Vector<int32_t> *r_path_types, TypedArray<RID> *r_path_rids, Vector<int64_t> *r_path_owners) const {
- map_rwlock.read_lock();
+ RWLockRead read_lock(map_rwlock);
if (map_update_id == 0) {
- map_rwlock.read_unlock();
ERR_FAIL_V_MSG(Vector<Vector3>(), "NavigationServer map query failed because it was made before first map synchronization.");
}
@@ -171,7 +170,6 @@ Vector<Vector3> NavMap::get_path(Vector3 p_origin, Vector3 p_destination, bool p
// Check for trivial cases
if (!begin_poly || !end_poly) {
- map_rwlock.read_unlock();
return Vector<Vector3>();
}
if (begin_poly == end_poly) {
@@ -198,8 +196,6 @@ Vector<Vector3> NavMap::get_path(Vector3 p_origin, Vector3 p_destination, bool p
path.write[0] = begin_point;
path.write[1] = end_point;
- map_rwlock.read_unlock();
-
return path;
}
@@ -348,8 +344,6 @@ Vector<Vector3> NavMap::get_path(Vector3 p_origin, Vector3 p_destination, bool p
path.write[0] = begin_point;
path.write[1] = end_point;
- map_rwlock.read_unlock();
-
return path;
}
@@ -436,8 +430,6 @@ Vector<Vector3> NavMap::get_path(Vector3 p_origin, Vector3 p_destination, bool p
path.write[0] = begin_point;
path.write[1] = end_point;
- map_rwlock.read_unlock();
-
return path;
}
@@ -582,8 +574,6 @@ Vector<Vector3> NavMap::get_path(Vector3 p_origin, Vector3 p_destination, bool p
}
}
- map_rwlock.read_unlock();
-
// Ensure post conditions (path arrays MUST match in size).
CRASH_COND(r_path_types && path.size() != r_path_types->size());
CRASH_COND(r_path_rids && path.size() != r_path_rids->size());
@@ -593,9 +583,8 @@ Vector<Vector3> NavMap::get_path(Vector3 p_origin, Vector3 p_destination, bool p
}
Vector3 NavMap::get_closest_point_to_segment(const Vector3 &p_from, const Vector3 &p_to, const bool p_use_collision) const {
- map_rwlock.read_lock();
+ RWLockRead read_lock(map_rwlock);
if (map_update_id == 0) {
- map_rwlock.read_unlock();
ERR_FAIL_V_MSG(Vector3(), "NavigationServer map query failed because it was made before first map synchronization.");
}
@@ -642,46 +631,38 @@ Vector3 NavMap::get_closest_point_to_segment(const Vector3 &p_from, const Vector
}
}
- map_rwlock.read_unlock();
-
return closest_point;
}
Vector3 NavMap::get_closest_point(const Vector3 &p_point) const {
- map_rwlock.read_lock();
+ RWLockRead read_lock(map_rwlock);
if (map_update_id == 0) {
- map_rwlock.read_unlock();
ERR_FAIL_V_MSG(Vector3(), "NavigationServer map query failed because it was made before first map synchronization.");
}
gd::ClosestPointQueryResult cp = get_closest_point_info(p_point);
- map_rwlock.read_unlock();
return cp.point;
}
Vector3 NavMap::get_closest_point_normal(const Vector3 &p_point) const {
- map_rwlock.read_lock();
+ RWLockRead read_lock(map_rwlock);
if (map_update_id == 0) {
- map_rwlock.read_unlock();
ERR_FAIL_V_MSG(Vector3(), "NavigationServer map query failed because it was made before first map synchronization.");
}
gd::ClosestPointQueryResult cp = get_closest_point_info(p_point);
- map_rwlock.read_unlock();
return cp.normal;
}
RID NavMap::get_closest_point_owner(const Vector3 &p_point) const {
- map_rwlock.read_lock();
+ RWLockRead read_lock(map_rwlock);
if (map_update_id == 0) {
- map_rwlock.read_unlock();
ERR_FAIL_V_MSG(RID(), "NavigationServer map query failed because it was made before first map synchronization.");
}
gd::ClosestPointQueryResult cp = get_closest_point_info(p_point);
- map_rwlock.read_unlock();
return cp.owner;
}
gd::ClosestPointQueryResult NavMap::get_closest_point_info(const Vector3 &p_point) const {
- map_rwlock.read_lock();
+ RWLockRead read_lock(map_rwlock);
gd::ClosestPointQueryResult result;
real_t closest_point_ds = FLT_MAX;
@@ -701,8 +682,6 @@ gd::ClosestPointQueryResult NavMap::get_closest_point_info(const Vector3 &p_poin
}
}
- map_rwlock.read_unlock();
-
return result;
}
@@ -813,7 +792,7 @@ void NavMap::remove_agent_as_controlled(NavAgent *agent) {
}
void NavMap::sync() {
- map_rwlock.write_lock();
+ RWLockWrite write_lock(map_rwlock);
// Performance Monitor
int _new_pm_region_count = regions.size();
@@ -1141,8 +1120,6 @@ void NavMap::sync() {
pm_edge_merge_count = _new_pm_edge_merge_count;
pm_edge_connection_count = _new_pm_edge_connection_count;
pm_edge_free_count = _new_pm_edge_free_count;
-
- map_rwlock.write_unlock();
}
void NavMap::_update_rvo_obstacles_tree_2d() { |
Makes navigation map spatial queries thread-safe by adding a readers–writer lock.
7e3a246
to
4cc8748
Compare
@RandomShaper Could you check the new version? |
Thanks! |
Makes navigation map spatial queries thread-safe by adding a readers–writer lock for the NavMap.
This avoids subtle bugs and sometimes even crashes related to the navigation map synchronization while users used threads to do spatial queries on the navigation map.
Tested with spawning endless number of agents each having its own thread doing those spatial queries e.g. map_get_path() or map_get_closest_point(), in a while loop while an AnimationPlayer did change the navigation map every single frame. While Godot managed to eventually collapse my entire system at around 1000+ agents from having spawned so many threads the navigation map and pathfinding stood strong till the end.
Production edit: closes godotengine/godot-roadmap#55