summaryrefslogtreecommitdiffstats
path: root/drivers
diff options
context:
space:
mode:
authorRémi Verschelde <rverschelde@gmail.com>2023-08-30 12:16:06 +0200
committerRémi Verschelde <rverschelde@gmail.com>2023-08-30 12:16:06 +0200
commit56b13b6865475c7a42ddd5337402552b60476513 (patch)
tree2524896a1875365bb951447e2ddf3521f52bf534 /drivers
parenta0d12cf93f822dbe870c9a50bff0127e236e725b (diff)
parent49177b6eeb9cff7cc48dbdf8f4d31164c1440b1c (diff)
downloadredot-engine-56b13b6865475c7a42ddd5337402552b60476513.tar.gz
Merge pull request #81001 from RandomShaper/win_safe_save
Make Windows' safe save more resilient
Diffstat (limited to 'drivers')
-rw-r--r--drivers/windows/file_access_windows.cpp39
1 files changed, 17 insertions, 22 deletions
diff --git a/drivers/windows/file_access_windows.cpp b/drivers/windows/file_access_windows.cpp
index c1e0c3fb60..da2ce1c208 100644
--- a/drivers/windows/file_access_windows.cpp
+++ b/drivers/windows/file_access_windows.cpp
@@ -175,32 +175,27 @@ void FileAccessWindows::_close() {
f = nullptr;
if (!save_path.is_empty()) {
+ // This workaround of trying multiple times is added to deal with paranoid Windows
+ // antiviruses that love reading just written files even if they are not executable, thus
+ // locking the file and preventing renaming from happening.
+
bool rename_error = true;
- int attempts = 4;
- while (rename_error && attempts) {
- // This workaround of trying multiple times is added to deal with paranoid Windows
- // antiviruses that love reading just written files even if they are not executable, thus
- // locking the file and preventing renaming from happening.
-
-#ifdef UWP_ENABLED
- // UWP has no PathFileExists, so we check attributes instead
- DWORD fileAttr;
-
- fileAttr = GetFileAttributesW((LPCWSTR)(save_path.utf16().get_data()));
- if (INVALID_FILE_ATTRIBUTES == fileAttr) {
-#else
- if (!PathFileExistsW((LPCWSTR)(save_path.utf16().get_data()))) {
-#endif
- // Creating new file
- rename_error = _wrename((LPCWSTR)(path.utf16().get_data()), (LPCWSTR)(save_path.utf16().get_data())) != 0;
+ const Char16String &path_utf16 = path.utf16();
+ const Char16String &save_path_utf16 = save_path.utf16();
+ for (int i = 0; i < 1000; i++) {
+ if (ReplaceFileW((LPCWSTR)(save_path_utf16.get_data()), (LPCWSTR)(path_utf16.get_data()), nullptr, REPLACEFILE_IGNORE_MERGE_ERRORS | REPLACEFILE_IGNORE_ACL_ERRORS, nullptr, nullptr)) {
+ rename_error = false;
} else {
- // Atomic replace for existing file
- rename_error = !ReplaceFileW((LPCWSTR)(save_path.utf16().get_data()), (LPCWSTR)(path.utf16().get_data()), nullptr, 2 | 4, nullptr, nullptr);
+ // Either the target exists and is locked (temporarily, hopefully)
+ // or it doesn't exist; let's assume the latter before re-trying.
+ rename_error = _wrename((LPCWSTR)(path_utf16.get_data()), (LPCWSTR)(save_path_utf16.get_data())) != 0;
}
- if (rename_error) {
- attempts--;
- OS::get_singleton()->delay_usec(100000); // wait 100msec and try again
+
+ if (!rename_error) {
+ break;
}
+
+ OS::get_singleton()->delay_usec(1000);
}
if (rename_error) {