diff options
author | Rémi Verschelde <rverschelde@gmail.com> | 2023-08-30 12:16:06 +0200 |
---|---|---|
committer | Rémi Verschelde <rverschelde@gmail.com> | 2023-08-30 12:16:06 +0200 |
commit | 56b13b6865475c7a42ddd5337402552b60476513 (patch) | |
tree | 2524896a1875365bb951447e2ddf3521f52bf534 /drivers | |
parent | a0d12cf93f822dbe870c9a50bff0127e236e725b (diff) | |
parent | 49177b6eeb9cff7cc48dbdf8f4d31164c1440b1c (diff) | |
download | redot-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.cpp | 39 |
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) { |