From: Daniel Stenberg Date: Mon, 1 Sep 2025 21:21:58 +0000 (+0200) Subject: tool_filetime: correct the conditions X-Git-Tag: curl-8_16_0~51 X-Git-Url: http://git.ipfire.org/?a=commitdiff_plain;h=9598ccee59789716b92c885f63d82e12259b6f29;p=thirdparty%2Fcurl.git tool_filetime: correct the conditions The libcurl API for CURLINFO_FILETIME_T clearly says it contains -1 if not set. Everything else is a valid time stamp so use that. Follow-up to 54f1ef05d672453d75a5fc60 Closes #18446 --- diff --git a/src/tool_filetime.c b/src/tool_filetime.c index 424b8cc7ea..c818fe3ada 100644 --- a/src/tool_filetime.c +++ b/src/tool_filetime.c @@ -88,65 +88,63 @@ int getfiletime(const char *filename, curl_off_t *stamp) #if defined(HAVE_UTIME) || defined(HAVE_UTIMES) || defined(_WIN32) void setfiletime(curl_off_t filetime, const char *filename) { - if(filetime) { /* Windows utime() may attempt to adjust the Unix GMT file time by a daylight saving time offset and since it is GMT that is bad behavior. When we have access to a 64-bit type we can bypass utime and set the times directly. */ #if defined(_WIN32) && !defined(CURL_WINDOWS_UWP) - HANDLE hfile; - TCHAR *tchar_filename = curlx_convert_UTF8_to_tchar(filename); - - /* 910670515199 is the maximum Unix filetime that can be used as a - Windows FILETIME without overflow: 30827-12-31T23:59:59. */ - if(filetime > 910670515199) { - warnf("Failed to set filetime %" CURL_FORMAT_CURL_OFF_T - " on outfile: overflow", filetime); - curlx_unicodefree(tchar_filename); - return; - } + HANDLE hfile; + TCHAR *tchar_filename = curlx_convert_UTF8_to_tchar(filename); - hfile = CreateFile(tchar_filename, FILE_WRITE_ATTRIBUTES, - (FILE_SHARE_READ | FILE_SHARE_WRITE | - FILE_SHARE_DELETE), - NULL, OPEN_EXISTING, 0, NULL); + /* 910670515199 is the maximum Unix filetime that can be used as a + Windows FILETIME without overflow: 30827-12-31T23:59:59. */ + if(filetime > 910670515199) { + warnf("Failed to set filetime %" CURL_FORMAT_CURL_OFF_T + " on outfile: overflow", filetime); curlx_unicodefree(tchar_filename); - if(hfile != INVALID_HANDLE_VALUE) { - curl_off_t converted = ((curl_off_t)filetime * 10000000) + - 116444736000000000; - FILETIME ft; - ft.dwLowDateTime = (DWORD)(converted & 0xFFFFFFFF); - ft.dwHighDateTime = (DWORD)(converted >> 32); - if(!SetFileTime(hfile, NULL, &ft, &ft)) { - warnf("Failed to set filetime %" CURL_FORMAT_CURL_OFF_T - " on outfile: SetFileTime failed: GetLastError %u", - filetime, (unsigned int)GetLastError()); - } - CloseHandle(hfile); - } - else { + return; + } + + hfile = CreateFile(tchar_filename, FILE_WRITE_ATTRIBUTES, + (FILE_SHARE_READ | FILE_SHARE_WRITE | + FILE_SHARE_DELETE), + NULL, OPEN_EXISTING, 0, NULL); + curlx_unicodefree(tchar_filename); + if(hfile != INVALID_HANDLE_VALUE) { + curl_off_t converted = ((curl_off_t)filetime * 10000000) + + 116444736000000000; + FILETIME ft; + ft.dwLowDateTime = (DWORD)(converted & 0xFFFFFFFF); + ft.dwHighDateTime = (DWORD)(converted >> 32); + if(!SetFileTime(hfile, NULL, &ft, &ft)) { warnf("Failed to set filetime %" CURL_FORMAT_CURL_OFF_T - " on outfile: CreateFile failed: GetLastError %u", + " on outfile: SetFileTime failed: GetLastError %u", filetime, (unsigned int)GetLastError()); } + CloseHandle(hfile); + } + else { + warnf("Failed to set filetime %" CURL_FORMAT_CURL_OFF_T + " on outfile: CreateFile failed: GetLastError %u", + filetime, (unsigned int)GetLastError()); + } #elif defined(HAVE_UTIMES) - struct timeval times[2]; - times[0].tv_sec = times[1].tv_sec = (time_t)filetime; - times[0].tv_usec = times[1].tv_usec = 0; - if(utimes(filename, times)) { - warnf("Failed to set filetime %" CURL_FORMAT_CURL_OFF_T - " on '%s': %s", filetime, filename, strerror(errno)); - } + struct timeval times[2]; + times[0].tv_sec = times[1].tv_sec = (time_t)filetime; + times[0].tv_usec = times[1].tv_usec = 0; + if(utimes(filename, times)) { + warnf("Failed to set filetime %" CURL_FORMAT_CURL_OFF_T + " on '%s': %s", filetime, filename, strerror(errno)); + } #elif defined(HAVE_UTIME) - struct utimbuf times; - times.actime = (time_t)filetime; - times.modtime = (time_t)filetime; - if(utime(filename, ×)) { - warnf("Failed to set filetime %" CURL_FORMAT_CURL_OFF_T - " on '%s': %s", filetime, filename, strerror(errno)); - } -#endif + struct utimbuf times; + times.actime = (time_t)filetime; + times.modtime = (time_t)filetime; + if(utime(filename, ×)) { + warnf("Failed to set filetime %" CURL_FORMAT_CURL_OFF_T + " on '%s': %s", filetime, filename, strerror(errno)); } +#endif } #endif /* HAVE_UTIME || HAVE_UTIMES || _WIN32 */ diff --git a/src/tool_operate.c b/src/tool_operate.c index e58b7bd075..2c3030096f 100644 --- a/src/tool_operate.c +++ b/src/tool_operate.c @@ -679,7 +679,8 @@ static CURLcode post_per_transfer(struct per_transfer *per, /* Ask libcurl if we got a remote file time */ curl_off_t filetime = -1; curl_easy_getinfo(curl, CURLINFO_FILETIME_T, &filetime); - setfiletime(filetime, outs->filename); + if(filetime != -1) + setfiletime(filetime, outs->filename); } skip: /* Write the --write-out data before cleanup but after result is final */