From 41c07856361d7cf2bcbda6d84386b1a0d3969f6a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pedro=20J=2E=20Est=C3=A9banez?= Date: Mon, 3 Jul 2023 13:54:55 +0200 Subject: [PATCH] Workaround resource loading crashes due to buggy TLS --- core/io/resource_loader.cpp | 21 +++++++++++++-------- core/io/resource_loader.h | 2 +- 2 files changed, 14 insertions(+), 9 deletions(-) diff --git a/core/io/resource_loader.cpp b/core/io/resource_loader.cpp index 525c41cf872..a82e574505e 100644 --- a/core/io/resource_loader.cpp +++ b/core/io/resource_loader.cpp @@ -239,15 +239,15 @@ ResourceLoader::LoadToken::~LoadToken() { Ref ResourceLoader::_load(const String &p_path, const String &p_original_path, const String &p_type_hint, ResourceFormatLoader::CacheMode p_cache_mode, Error *r_error, bool p_use_sub_threads, float *r_progress) { load_nesting++; - if (load_paths_stack.size()) { + if (load_paths_stack->size()) { thread_load_mutex.lock(); - HashMap::Iterator E = thread_load_tasks.find(load_paths_stack[load_paths_stack.size() - 1]); + HashMap::Iterator E = thread_load_tasks.find(load_paths_stack->get(load_paths_stack->size() - 1)); if (E) { E->value.sub_tasks.insert(p_path); } thread_load_mutex.unlock(); } - load_paths_stack.push_back(p_path); + load_paths_stack->push_back(p_path); // Try all loaders and pick the first match for the type hint bool found = false; @@ -263,7 +263,7 @@ Ref ResourceLoader::_load(const String &p_path, const String &p_origin } } - load_paths_stack.resize(load_paths_stack.size() - 1); + load_paths_stack->resize(load_paths_stack->size() - 1); load_nesting--; if (!res.is_null()) { @@ -296,8 +296,10 @@ void ResourceLoader::_thread_load_function(void *p_userdata) { // Thread-safe either if it's the current thread or a brand new one. CallQueue *mq_override = nullptr; if (load_nesting == 0) { + load_paths_stack = memnew(Vector); + if (!load_task.dependent_path.is_empty()) { - load_paths_stack.push_back(load_task.dependent_path); + load_paths_stack->push_back(load_task.dependent_path); } if (!Thread::is_main_thread()) { mq_override = memnew(CallQueue); @@ -356,8 +358,11 @@ void ResourceLoader::_thread_load_function(void *p_userdata) { thread_load_mutex.unlock(); - if (load_nesting == 0 && mq_override) { - memdelete(mq_override); + if (load_nesting == 0) { + if (mq_override) { + memdelete(mq_override); + } + memdelete(load_paths_stack); } } @@ -1166,7 +1171,7 @@ bool ResourceLoader::timestamp_on_load = false; thread_local int ResourceLoader::load_nesting = 0; thread_local WorkerThreadPool::TaskID ResourceLoader::caller_task_id = 0; -thread_local Vector ResourceLoader::load_paths_stack; +thread_local Vector *ResourceLoader::load_paths_stack; template <> thread_local uint32_t SafeBinaryMutex::count = 0; diff --git a/core/io/resource_loader.h b/core/io/resource_loader.h index 592befb6039..2701caa3f47 100644 --- a/core/io/resource_loader.h +++ b/core/io/resource_loader.h @@ -182,7 +182,7 @@ class ResourceLoader { static thread_local int load_nesting; static thread_local WorkerThreadPool::TaskID caller_task_id; - static thread_local Vector load_paths_stack; + static thread_local Vector *load_paths_stack; // A pointer to avoid broken TLS implementations from double-running the destructor. static SafeBinaryMutex thread_load_mutex; static HashMap thread_load_tasks; static bool cleaning_tasks;