summaryrefslogtreecommitdiffstats
path: root/modules/gdscript
diff options
context:
space:
mode:
authorrune-scape <spartacrafter@gmail.com>2024-07-23 12:38:30 -0700
committerrune-scape <spartacrafter@gmail.com>2024-07-23 12:54:45 -0700
commit86b23d042071cc9ad2ea1b3e7901c35acd423a9a (patch)
tree1ee0abcfd6116f15a8cc89bcfac38f0db4725571 /modules/gdscript
parent8e36f98ea5b5ab4c1fb5249f94b61a7bbfb379a1 (diff)
downloadredot-engine-86b23d042071cc9ad2ea1b3e7901c35acd423a9a.tar.gz
GDScriptCache: Clear abandoned parser refs
Diffstat (limited to 'modules/gdscript')
-rw-r--r--modules/gdscript/gdscript_cache.cpp33
-rw-r--r--modules/gdscript/gdscript_cache.h2
2 files changed, 34 insertions, 1 deletions
diff --git a/modules/gdscript/gdscript_cache.cpp b/modules/gdscript/gdscript_cache.cpp
index 7c9fba799d..2b1184bcb9 100644
--- a/modules/gdscript/gdscript_cache.cpp
+++ b/modules/gdscript/gdscript_cache.cpp
@@ -135,7 +135,9 @@ void GDScriptParserRef::clear() {
GDScriptParserRef::~GDScriptParserRef() {
clear();
- GDScriptCache::remove_parser(path);
+ if (!abandoned) {
+ GDScriptCache::remove_parser(path);
+ }
}
GDScriptCache *GDScriptCache::singleton = nullptr;
@@ -183,6 +185,17 @@ void GDScriptCache::remove_script(const String &p_path) {
return;
}
+ if (HashMap<String, Vector<ObjectID>>::Iterator E = singleton->abandoned_parser_map.find(p_path)) {
+ for (ObjectID parser_ref_id : E->value) {
+ Ref<GDScriptParserRef> parser_ref{ ObjectDB::get_instance(parser_ref_id) };
+ if (parser_ref.is_valid()) {
+ parser_ref->clear();
+ }
+ }
+ }
+
+ singleton->abandoned_parser_map.erase(p_path);
+
if (singleton->parser_map.has(p_path)) {
singleton->parser_map[p_path]->clear();
}
@@ -229,6 +242,13 @@ bool GDScriptCache::has_parser(const String &p_path) {
void GDScriptCache::remove_parser(const String &p_path) {
MutexLock lock(singleton->mutex);
+
+ if (singleton->parser_map.has(p_path)) {
+ GDScriptParserRef *parser_ref = singleton->parser_map[p_path];
+ parser_ref->abandoned = true;
+ singleton->abandoned_parser_map[p_path].push_back(parser_ref->get_instance_id());
+ }
+
// Can't clear the parser because some other parser might be currently using it in the chain of calls.
singleton->parser_map.erase(p_path);
@@ -432,6 +452,17 @@ void GDScriptCache::clear() {
singleton->parser_inverse_dependencies.clear();
+ for (const KeyValue<String, Vector<ObjectID>> &KV : singleton->abandoned_parser_map) {
+ for (ObjectID parser_ref_id : KV.value) {
+ Ref<GDScriptParserRef> parser_ref{ ObjectDB::get_instance(parser_ref_id) };
+ if (parser_ref.is_valid()) {
+ parser_ref->clear();
+ }
+ }
+ }
+
+ singleton->abandoned_parser_map.clear();
+
RBSet<Ref<GDScriptParserRef>> parser_map_refs;
for (KeyValue<String, GDScriptParserRef *> &E : singleton->parser_map) {
parser_map_refs.insert(E.value);
diff --git a/modules/gdscript/gdscript_cache.h b/modules/gdscript/gdscript_cache.h
index c927317e19..f7f2cd90e9 100644
--- a/modules/gdscript/gdscript_cache.h
+++ b/modules/gdscript/gdscript_cache.h
@@ -59,6 +59,7 @@ private:
String path;
uint32_t source_hash = 0;
bool clearing = false;
+ bool abandoned = false;
friend class GDScriptCache;
friend class GDScript;
@@ -79,6 +80,7 @@ public:
class GDScriptCache {
// String key is full path.
HashMap<String, GDScriptParserRef *> parser_map;
+ HashMap<String, Vector<ObjectID>> abandoned_parser_map;
HashMap<String, Ref<GDScript>> shallow_gdscript_cache;
HashMap<String, Ref<GDScript>> full_gdscript_cache;
HashMap<String, Ref<GDScript>> static_gdscript_cache;