From 120f15efb77c374a4ebb1b50120c95e2fab3c2f3 Mon Sep 17 00:00:00 2001 From: baptr <1522777+baptr@users.noreply.github.com> Date: Sun, 21 Jan 2024 15:07:53 -0800 Subject: [PATCH] Mutex guard tile_map_layer dirty.cell_list. TilesEditorUtils calls tile_map->set_pattern from a thread (to refresh tile pattern preview images). set_pattern sets each of the cells into the temporary map, marking them dirty via set_cell. This can race against the tree invoking the TileMapLayer's internal_update, which proceses dirty.cell_list and eventually clears it. If a cell's shared dirty entry ends up losing the race, clear can get stuck in an infinite loop where remove refuses to change it because the entry doesn't think it's a member of the list, but the list head still points to it. This spams ERROR: Condition "p_elem->_root != this" is true. at: remove (./core/templates/self_list.h:80) And eventually leads to a crash when the error message buffer fills. Guarding the list mutations avoids the race that sets up for this failure, fixing https://www.github.com/godotengine/godot/issues/86226 --- scene/2d/tile_map_layer.cpp | 5 ++++- scene/2d/tile_map_layer.h | 1 + 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/scene/2d/tile_map_layer.cpp b/scene/2d/tile_map_layer.cpp index df79b3fee658..bba0f84be7bf 100644 --- a/scene/2d/tile_map_layer.cpp +++ b/scene/2d/tile_map_layer.cpp @@ -2111,6 +2111,8 @@ void TileMapLayer::notify_tile_map_change(DirtyFlags p_what) { } void TileMapLayer::internal_update() { + MutexLock lock(dirty.mutex); + // Find TileData that need a runtime modification. // This may add cells to the dirty list is a runtime modification has been notified. _build_runtime_update_tile_data(); @@ -2187,6 +2189,7 @@ void TileMapLayer::set_cell(const Vector2i &p_coords, int p_source_id, const Vec c.alternative_tile = alternative_tile; // Make the given cell dirty. + MutexLock lock(dirty.mutex); if (!E->value.dirty_list_element.in_list()) { dirty.cell_list.add(&(E->value.dirty_list_element)); } @@ -2939,4 +2942,4 @@ TerrainConstraint::TerrainConstraint(Ref p_tile_set, const Vector2i &p_ } } terrain = p_terrain; -} \ No newline at end of file +} diff --git a/scene/2d/tile_map_layer.h b/scene/2d/tile_map_layer.h index 2cc57f50d1b6..aa4232da229a 100644 --- a/scene/2d/tile_map_layer.h +++ b/scene/2d/tile_map_layer.h @@ -262,6 +262,7 @@ class TileMapLayer : public Node2D { // Dirty flag. Allows knowing what was modified since the last update. struct { + Mutex mutex; bool flags[DIRTY_FLAGS_MAX] = { false }; SelfList::List cell_list; } dirty;