summaryrefslogtreecommitdiffstats
path: root/editor/editor_file_system.cpp
diff options
context:
space:
mode:
authorRémi Verschelde <rverschelde@gmail.com>2024-09-26 12:45:31 +0200
committerRémi Verschelde <rverschelde@gmail.com>2024-09-26 12:45:31 +0200
commitef7547384c962f536b9020b74cf0984a534007bb (patch)
treede60dcfd8b818aa67e0eb6d21d13d60d6f917d7e /editor/editor_file_system.cpp
parent9d5b59b2ae456a1f9a98c918cef6ef217f52d31c (diff)
parent21f7c8a25f23c40541925c95d7d25f52f81c9811 (diff)
downloadredot-engine-ef7547384c962f536b9020b74cf0984a534007bb.tar.gz
Merge pull request #95678 from Hilderin/fix-slow-load-on-large-project-v2
Fix slow editor load on large projects (v2)
Diffstat (limited to 'editor/editor_file_system.cpp')
-rw-r--r--editor/editor_file_system.cpp298
1 files changed, 202 insertions, 96 deletions
diff --git a/editor/editor_file_system.cpp b/editor/editor_file_system.cpp
index 2b51071b15..c4c7a2cbb8 100644
--- a/editor/editor_file_system.cpp
+++ b/editor/editor_file_system.cpp
@@ -282,6 +282,21 @@ void EditorFileSystem::_first_scan_process_scripts(const ScannedDirectory *p_sca
}
for (const String &scan_file : p_scan_dir->files) {
+ // Optimization to skip the ResourceLoader::get_resource_type for files
+ // that are not scripts. Some loader get_resource_type methods read the file
+ // which can be very slow on large projects.
+ String ext = scan_file.get_extension().to_lower();
+ bool is_script = false;
+ for (int i = 0; i < ScriptServer::get_language_count(); i++) {
+ if (ScriptServer::get_language(i)->get_extension() == ext) {
+ is_script = true;
+ break;
+ }
+ }
+ if (!is_script) {
+ continue; // Not a script.
+ }
+
String path = p_scan_dir->full_path.path_join(scan_file);
String type = ResourceLoader::get_resource_type(path);
@@ -371,6 +386,11 @@ void EditorFileSystem::_scan_filesystem() {
fc.script_class_name = split[7].get_slice("<>", 0);
fc.script_class_extends = split[7].get_slice("<>", 1);
fc.script_class_icon_path = split[7].get_slice("<>", 2);
+ fc.import_md5 = split[7].get_slice("<>", 3);
+ String dest_paths = split[7].get_slice("<>", 4);
+ if (!dest_paths.is_empty()) {
+ fc.import_dest_paths = dest_paths.split("<*>");
+ }
String deps = split[8].strip_edges();
if (deps.length()) {
@@ -463,12 +483,33 @@ void EditorFileSystem::_thread_func(void *_userdata) {
sd->_scan_filesystem();
}
-bool EditorFileSystem::_test_for_reimport(const String &p_path, bool p_only_imported_files) {
- if (!reimport_on_missing_imported_files && p_only_imported_files) {
- return false;
+bool EditorFileSystem::_is_test_for_reimport_needed(const String &p_path, uint64_t p_last_modification_time, uint64_t p_modification_time, uint64_t p_last_import_modification_time, uint64_t p_import_modification_time, const Vector<String> &p_import_dest_paths) {
+ // The idea here is to trust the cache. If the last modification times in the cache correspond
+ // to the last modification times of the files on disk, it means the files have not changed since
+ // the last import, and the files in .godot/imported (p_import_dest_paths) should all be valid.
+ if (p_last_modification_time != p_modification_time) {
+ return true;
+ }
+ if (p_last_import_modification_time != p_import_modification_time) {
+ return true;
}
+ if (reimport_on_missing_imported_files) {
+ for (const String &path : p_import_dest_paths) {
+ if (!FileAccess::exists(path)) {
+ return true;
+ }
+ }
+ }
+ return false;
+}
- if (!FileAccess::exists(p_path + ".import")) {
+bool EditorFileSystem::_test_for_reimport(const String &p_path, const String &p_expected_import_md5) {
+ if (p_expected_import_md5.is_empty()) {
+ // Marked as reimportation needed.
+ return true;
+ }
+ String new_md5 = FileAccess::get_md5(p_path + ".import");
+ if (p_expected_import_md5 != new_md5) {
return true;
}
@@ -489,7 +530,7 @@ bool EditorFileSystem::_test_for_reimport(const String &p_path, bool p_only_impo
int lines = 0;
String error_text;
- List<String> to_check;
+ Vector<String> to_check;
String importer_name;
String source_file = "";
@@ -498,6 +539,7 @@ bool EditorFileSystem::_test_for_reimport(const String &p_path, bool p_only_impo
String dest_md5 = "";
int version = 0;
bool found_uid = false;
+ Variant meta;
while (true) {
assign = Variant();
@@ -522,8 +564,8 @@ bool EditorFileSystem::_test_for_reimport(const String &p_path, bool p_only_impo
to_check.push_back(value);
} else if (assign == "files") {
Array fa = value;
- for (int i = 0; i < fa.size(); i++) {
- to_check.push_back(fa[i]);
+ for (const Variant &check_path : fa) {
+ to_check.push_back(check_path);
}
} else if (assign == "importer_version") {
version = value;
@@ -531,12 +573,12 @@ bool EditorFileSystem::_test_for_reimport(const String &p_path, bool p_only_impo
importer_name = value;
} else if (assign == "uid") {
found_uid = true;
- } else if (!p_only_imported_files) {
- if (assign == "source_file") {
- source_file = value;
- } else if (assign == "dest_files") {
- dest_files = value;
- }
+ } else if (assign == "source_file") {
+ source_file = value;
+ } else if (assign == "dest_files") {
+ dest_files = value;
+ } else if (assign == "metadata") {
+ meta = value;
}
} else if (next_tag.name != "remap" && next_tag.name != "deps") {
@@ -544,33 +586,40 @@ bool EditorFileSystem::_test_for_reimport(const String &p_path, bool p_only_impo
}
}
- if (!ResourceFormatImporter::get_singleton()->are_import_settings_valid(p_path)) {
- // Reimport settings are out of sync with project settings, reimport.
- return true;
- }
-
if (importer_name == "keep" || importer_name == "skip") {
- return false; //keep mode, do not reimport
+ return false; // Keep mode, do not reimport.
}
if (!found_uid) {
- return true; //UID not found, old format, reimport.
+ return true; // UID not found, old format, reimport.
+ }
+
+ // Imported files are gone, reimport.
+ for (const String &E : to_check) {
+ if (!FileAccess::exists(E)) {
+ return true;
+ }
}
Ref<ResourceImporter> importer = ResourceFormatImporter::get_singleton()->get_importer_by_name(importer_name);
if (importer.is_null()) {
- return true; // the importer has possibly changed, try to reimport.
+ return true; // The importer has possibly changed, try to reimport.
}
if (importer->get_format_version() > version) {
- return true; // version changed, reimport
+ return true; // Version changed, reimport.
+ }
+
+ if (!importer->are_import_settings_valid(p_path, meta)) {
+ // Reimport settings are out of sync with project settings, reimport.
+ return true;
}
- // Read the md5's from a separate file (so the import parameters aren't dependent on the file version
+ // Read the md5's from a separate file (so the import parameters aren't dependent on the file version).
String base_path = ResourceFormatImporter::get_singleton()->get_import_base_path(p_path);
Ref<FileAccess> md5s = FileAccess::open(base_path + ".md5", FileAccess::READ, &err);
- if (md5s.is_null()) { // No md5's stored for this resource
+ if (md5s.is_null()) { // No md5's stored for this resource.
return true;
}
@@ -588,50 +637,101 @@ bool EditorFileSystem::_test_for_reimport(const String &p_path, bool p_only_impo
break;
} else if (err != OK) {
ERR_PRINT("ResourceFormatImporter::load - '" + p_path + ".import.md5:" + itos(lines) + "' error '" + error_text + "'.");
- return false; // parse error
+ return false; // Parse error.
}
if (!assign.is_empty()) {
- if (!p_only_imported_files) {
- if (assign == "source_md5") {
- source_md5 = value;
- } else if (assign == "dest_md5") {
- dest_md5 = value;
- }
+ if (assign == "source_md5") {
+ source_md5 = value;
+ } else if (assign == "dest_md5") {
+ dest_md5 = value;
}
}
}
- //imported files are gone, reimport
- for (const String &E : to_check) {
- if (!FileAccess::exists(E)) {
+ // Check source md5 matching.
+ if (!source_file.is_empty() && source_file != p_path) {
+ return true; // File was moved, reimport.
+ }
+
+ if (source_md5.is_empty()) {
+ return true; // Lacks md5, so just reimport.
+ }
+
+ String md5 = FileAccess::get_md5(p_path);
+ if (md5 != source_md5) {
+ return true;
+ }
+
+ if (!dest_files.is_empty() && !dest_md5.is_empty()) {
+ md5 = FileAccess::get_multiple_md5(dest_files);
+ if (md5 != dest_md5) {
return true;
}
}
- //check source md5 matching
- if (!p_only_imported_files) {
- if (!source_file.is_empty() && source_file != p_path) {
- return true; //file was moved, reimport
- }
+ return false; // Nothing changed.
+}
- if (source_md5.is_empty()) {
- return true; //lacks md5, so just reimport
- }
+Vector<String> EditorFileSystem::_get_import_dest_paths(const String &p_path) {
+ Error err;
+ Ref<FileAccess> f = FileAccess::open(p_path + ".import", FileAccess::READ, &err);
- String md5 = FileAccess::get_md5(p_path);
- if (md5 != source_md5) {
- return true;
+ if (f.is_null()) { // No import file, reimport.
+ return Vector<String>();
+ }
+
+ VariantParser::StreamFile stream;
+ stream.f = f;
+
+ String assign;
+ Variant value;
+ VariantParser::Tag next_tag;
+
+ int lines = 0;
+ String error_text;
+
+ Vector<String> dest_paths;
+ String importer_name;
+
+ while (true) {
+ assign = Variant();
+ next_tag.fields.clear();
+ next_tag.name = String();
+
+ err = VariantParser::parse_tag_assign_eof(&stream, lines, error_text, next_tag, assign, value, nullptr, true);
+ if (err == ERR_FILE_EOF) {
+ break;
+ } else if (err != OK) {
+ ERR_PRINT("ResourceFormatImporter::load - '" + p_path + ".import:" + itos(lines) + "' error '" + error_text + "'.");
+ // Parse error, skip and let user attempt manual reimport to avoid reimport loop.
+ return Vector<String>();
}
- if (dest_files.size() && !dest_md5.is_empty()) {
- md5 = FileAccess::get_multiple_md5(dest_files);
- if (md5 != dest_md5) {
- return true;
+ if (!assign.is_empty()) {
+ if (assign == "valid" && value.operator bool() == false) {
+ // Invalid import (failed previous import), skip and let user attempt manual reimport to avoid reimport loop.
+ return Vector<String>();
}
+ if (assign.begins_with("path")) {
+ dest_paths.push_back(value);
+ } else if (assign == "files") {
+ Array fa = value;
+ for (const Variant &dest_path : fa) {
+ dest_paths.push_back(dest_path);
+ }
+ } else if (assign == "importer") {
+ importer_name = value;
+ }
+ } else if (next_tag.name != "remap" && next_tag.name != "deps") {
+ break;
}
}
- return false; //nothing changed
+ if (importer_name == "keep" || importer_name == "skip") {
+ return Vector<String>();
+ }
+
+ return dest_paths;
}
bool EditorFileSystem::_scan_import_support(const Vector<String> &reimports) {
@@ -774,20 +874,9 @@ bool EditorFileSystem::_update_scan_actions() {
ERR_CONTINUE(idx == -1);
String full_path = ia.dir->get_file_path(idx);
- bool need_reimport = _test_for_reimport(full_path, false);
- // Workaround GH-94416 for the Android editor for now.
- // `import_mt` seems to always be 0 and force a reimport on any fs scan.
-#ifndef ANDROID_ENABLED
- if (!need_reimport && FileAccess::exists(full_path + ".import")) {
- uint64_t import_mt = ia.dir->get_file_import_modified_time(idx);
- if (import_mt != FileAccess::get_modified_time(full_path + ".import")) {
- need_reimport = true;
- }
- }
-#endif
-
+ bool need_reimport = _test_for_reimport(full_path, ia.dir->files[idx]->import_md5);
if (need_reimport) {
- //must reimport
+ // Must reimport.
reimports.push_back(full_path);
Vector<String> dependencies = _get_dependencies(full_path);
for (const String &dep : dependencies) {
@@ -797,10 +886,14 @@ bool EditorFileSystem::_update_scan_actions() {
}
}
} else {
- //must not reimport, all was good
- //update modified times, to avoid reimport
+ // Must not reimport, all was good.
+ // Update modified times, md5 and destination paths, to avoid reimport.
ia.dir->files[idx]->modified_time = FileAccess::get_modified_time(full_path);
ia.dir->files[idx]->import_modified_time = FileAccess::get_modified_time(full_path + ".import");
+ if (ia.dir->files[idx]->import_md5.is_empty()) {
+ ia.dir->files[idx]->import_md5 = FileAccess::get_md5(full_path + ".import");
+ }
+ ia.dir->files[idx]->import_dest_paths = _get_import_dest_paths(full_path);
}
fs_changed = true;
@@ -1029,26 +1122,36 @@ void EditorFileSystem::_process_file_system(const ScannedDirectory *p_scan_dir,
if (import_extensions.has(ext)) {
//is imported
- uint64_t import_mt = 0;
- if (FileAccess::exists(path + ".import")) {
- import_mt = FileAccess::get_modified_time(path + ".import");
- }
+ uint64_t import_mt = FileAccess::get_modified_time(path + ".import");
- if (fc && fc->modification_time == mt && fc->import_modification_time == import_mt && !_test_for_reimport(path, true)) {
+ if (fc) {
fi->type = fc->type;
fi->resource_script_class = fc->resource_script_class;
fi->uid = fc->uid;
fi->deps = fc->deps;
- fi->modified_time = fc->modification_time;
- fi->import_modified_time = fc->import_modification_time;
-
+ fi->modified_time = mt;
+ fi->import_modified_time = import_mt;
+ fi->import_md5 = fc->import_md5;
+ fi->import_dest_paths = fc->import_dest_paths;
fi->import_valid = fc->import_valid;
fi->script_class_name = fc->script_class_name;
fi->import_group_file = fc->import_group_file;
fi->script_class_extends = fc->script_class_extends;
fi->script_class_icon_path = fc->script_class_icon_path;
- if (revalidate_import_files && !ResourceFormatImporter::get_singleton()->are_import_settings_valid(path)) {
+ // Ensures backward compatibility when the project is loaded for the first time with the added import_md5
+ // and import_dest_paths properties in the file cache.
+ if (fc->import_md5.is_empty()) {
+ fi->import_md5 = FileAccess::get_md5(path + ".import");
+ fi->import_dest_paths = _get_import_dest_paths(path);
+ }
+
+ // The method _is_test_for_reimport_needed checks if the files were modified and ensures that
+ // all the destination files still exist without reading the .import file.
+ // If something is different, we will queue a test for reimportation that will check
+ // the md5 of all files and import settings and, if necessary, execute a reimportation.
+ if (_is_test_for_reimport_needed(path, fc->modification_time, mt, fc->import_modification_time, import_mt, fi->import_dest_paths) ||
+ (revalidate_import_files && !ResourceFormatImporter::get_singleton()->are_import_settings_valid(path))) {
ItemAction ia;
ia.action = ItemAction::ACTION_FILE_TEST_REIMPORT;
ia.dir = p_dir;
@@ -1075,6 +1178,8 @@ void EditorFileSystem::_process_file_system(const ScannedDirectory *p_scan_dir,
fi->script_class_name = _get_global_script_class(fi->type, path, &fi->script_class_extends, &fi->script_class_icon_path);
fi->modified_time = 0;
fi->import_modified_time = 0;
+ fi->import_md5 = "";
+ fi->import_dest_paths = Vector<String>();
fi->import_valid = (fi->type == "TextFile" || fi->type == "OtherFile") ? true : ResourceLoader::is_import_valid(path);
ItemAction ia;
@@ -1089,9 +1194,11 @@ void EditorFileSystem::_process_file_system(const ScannedDirectory *p_scan_dir,
fi->type = fc->type;
fi->resource_script_class = fc->resource_script_class;
fi->uid = fc->uid;
- fi->modified_time = fc->modification_time;
+ fi->modified_time = mt;
fi->deps = fc->deps;
fi->import_modified_time = 0;
+ fi->import_md5 = "";
+ fi->import_dest_paths = Vector<String>();
fi->import_valid = true;
fi->script_class_name = fc->script_class_name;
fi->script_class_extends = fc->script_class_extends;
@@ -1126,6 +1233,8 @@ void EditorFileSystem::_process_file_system(const ScannedDirectory *p_scan_dir,
fi->deps = _get_dependencies(path);
fi->modified_time = mt;
fi->import_modified_time = 0;
+ fi->import_md5 = "";
+ fi->import_dest_paths = Vector<String>();
fi->import_valid = true;
// Files in dep_update_list are forced for rescan to update dependencies. They don't need other updates.
@@ -1269,6 +1378,8 @@ void EditorFileSystem::_scan_fs_changes(EditorFileSystemDirectory *p_dir, ScanPr
String path = cd.path_join(fi->file);
fi->modified_time = FileAccess::get_modified_time(path);
fi->import_modified_time = 0;
+ fi->import_md5 = "";
+ fi->import_dest_paths = Vector<String>();
fi->type = ResourceLoader::get_resource_type(path);
fi->resource_script_class = ResourceLoader::get_resource_script_class(path);
if (fi->type == "" && textfile_extensions.has(ext)) {
@@ -1323,26 +1434,13 @@ void EditorFileSystem::_scan_fs_changes(EditorFileSystemDirectory *p_dir, ScanPr
String path = cd.path_join(p_dir->files[i]->file);
if (import_extensions.has(p_dir->files[i]->file.get_extension().to_lower())) {
- //check here if file must be imported or not
-
+ // Check here if file must be imported or not.
+ // Same logic as in _process_file_system, the last modifications dates
+ // needs to be trusted to prevent reading all the .import files and the md5
+ // each time the user switch back to Godot.
uint64_t mt = FileAccess::get_modified_time(path);
-
- bool reimport = false;
-
- if (mt != p_dir->files[i]->modified_time) {
- reimport = true; //it was modified, must be reimported.
- } else if (!FileAccess::exists(path + ".import")) {
- reimport = true; //no .import file, obviously reimport
- } else {
- uint64_t import_mt = FileAccess::get_modified_time(path + ".import");
- if (import_mt != p_dir->files[i]->import_modified_time) {
- reimport = true;
- } else if (_test_for_reimport(path, true)) {
- reimport = true;
- }
- }
-
- if (reimport) {
+ uint64_t import_mt = FileAccess::get_modified_time(path + ".import");
+ if (_is_test_for_reimport_needed(path, p_dir->files[i]->modified_time, mt, p_dir->files[i]->import_modified_time, import_mt, p_dir->files[i]->import_dest_paths)) {
ItemAction ia;
ia.action = ItemAction::ACTION_FILE_TEST_REIMPORT;
ia.dir = p_dir;
@@ -1625,7 +1723,7 @@ void EditorFileSystem::_save_filesystem_cache(EditorFileSystemDirectory *p_dir,
cache_string.append(itos(file_info->import_modified_time));
cache_string.append(itos(file_info->import_valid));
cache_string.append(file_info->import_group_file);
- cache_string.append(String("<>").join({ file_info->script_class_name, file_info->script_class_extends, file_info->script_class_icon_path }));
+ cache_string.append(String("<>").join({ file_info->script_class_name, file_info->script_class_extends, file_info->script_class_icon_path, file_info->import_md5, String("<*>").join(file_info->import_dest_paths) }));
cache_string.append(String("<>").join(file_info->deps));
p_file->store_line(String("::").join(cache_string));
@@ -2183,6 +2281,8 @@ void EditorFileSystem::update_files(const Vector<String> &p_script_paths) {
fi->file = file_name;
fi->import_modified_time = 0;
fi->import_valid = (type == "TextFile" || type == "OtherFile") ? true : ResourceLoader::is_import_valid(file);
+ fi->import_md5 = "";
+ fi->import_dest_paths = Vector<String>();
if (idx == fs->files.size()) {
fs->files.push_back(fi);
@@ -2458,6 +2558,8 @@ Error EditorFileSystem::_reimport_group(const String &p_group_file, const Vector
//update modified times, to avoid reimport
fs->files[cpos]->modified_time = FileAccess::get_modified_time(file);
fs->files[cpos]->import_modified_time = FileAccess::get_modified_time(file + ".import");
+ fs->files[cpos]->import_md5 = FileAccess::get_md5(file + ".import");
+ fs->files[cpos]->import_dest_paths = dest_paths;
fs->files[cpos]->deps = _get_dependencies(file);
fs->files[cpos]->uid = uid;
fs->files[cpos]->type = importer->get_resource_type();
@@ -2559,11 +2661,13 @@ Error EditorFileSystem::_reimport_file(const String &p_file, const HashMap<Strin
if (p_update_file_system) {
fs->files[cpos]->modified_time = FileAccess::get_modified_time(p_file);
fs->files[cpos]->import_modified_time = FileAccess::get_modified_time(p_file + ".import");
+ fs->files[cpos]->import_md5 = FileAccess::get_md5(p_file + ".import");
+ fs->files[cpos]->import_dest_paths = Vector<String>();
fs->files[cpos]->deps.clear();
fs->files[cpos]->type = "";
fs->files[cpos]->import_valid = false;
+ EditorResourcePreview::get_singleton()->check_for_invalidation(p_file);
}
- EditorResourcePreview::get_singleton()->check_for_invalidation(p_file);
return OK;
}
Ref<ResourceImporter> importer;
@@ -2729,6 +2833,8 @@ Error EditorFileSystem::_reimport_file(const String &p_file, const HashMap<Strin
// Update modified times, to avoid reimport.
fs->files[cpos]->modified_time = FileAccess::get_modified_time(p_file);
fs->files[cpos]->import_modified_time = FileAccess::get_modified_time(p_file + ".import");
+ fs->files[cpos]->import_md5 = FileAccess::get_md5(p_file + ".import");
+ fs->files[cpos]->import_dest_paths = dest_paths;
fs->files[cpos]->deps = _get_dependencies(p_file);
fs->files[cpos]->type = importer->get_resource_type();
fs->files[cpos]->uid = uid;