| /*************************************************************************** | 
 |  *                                  _   _ ____  _ | 
 |  *  Project                     ___| | | |  _ \| | | 
 |  *                             / __| | | | |_) | | | 
 |  *                            | (__| |_| |  _ <| |___ | 
 |  *                             \___|\___/|_| \_\_____| | 
 |  * | 
 |  * Copyright (C) 1998 - 2022, Daniel Stenberg, <daniel@haxx.se>, et al. | 
 |  * | 
 |  * This software is licensed as described in the file COPYING, which | 
 |  * you should have received as part of this distribution. The terms | 
 |  * are also available at https://curl.se/docs/copyright.html. | 
 |  * | 
 |  * You may opt to use, copy, modify, merge, publish, distribute and/or sell | 
 |  * copies of the Software, and permit persons to whom the Software is | 
 |  * furnished to do so, under the terms of the COPYING file. | 
 |  * | 
 |  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY | 
 |  * KIND, either express or implied. | 
 |  * | 
 |  * SPDX-License-Identifier: curl | 
 |  * | 
 |  ***************************************************************************/ | 
 | #include "tool_setup.h" | 
 |  | 
 | #if defined(HAVE_STRCASECMP) && defined(HAVE_STRINGS_H) | 
 | #include <strings.h> | 
 | #endif | 
 |  | 
 | #include "tool_util.h" | 
 |  | 
 | #include "memdebug.h" /* keep this as LAST include */ | 
 |  | 
 | #if defined(WIN32) && !defined(MSDOS) | 
 |  | 
 | /* set in win32_init() */ | 
 | extern LARGE_INTEGER tool_freq; | 
 | extern bool tool_isVistaOrGreater; | 
 |  | 
 | /* In case of bug fix this function has a counterpart in timeval.c */ | 
 | struct timeval tvnow(void) | 
 | { | 
 |   struct timeval now; | 
 |   if(tool_isVistaOrGreater) { /* QPC timer might have issues pre-Vista */ | 
 |     LARGE_INTEGER count; | 
 |     QueryPerformanceCounter(&count); | 
 |     now.tv_sec = (long)(count.QuadPart / tool_freq.QuadPart); | 
 |     now.tv_usec = (long)((count.QuadPart % tool_freq.QuadPart) * 1000000 / | 
 |                          tool_freq.QuadPart); | 
 |   } | 
 |   else { | 
 |     /* Disable /analyze warning that GetTickCount64 is preferred  */ | 
 | #if defined(_MSC_VER) | 
 | #pragma warning(push) | 
 | #pragma warning(disable:28159) | 
 | #endif | 
 |     DWORD milliseconds = GetTickCount(); | 
 | #if defined(_MSC_VER) | 
 | #pragma warning(pop) | 
 | #endif | 
 |  | 
 |     now.tv_sec = (long)(milliseconds / 1000); | 
 |     now.tv_usec = (long)((milliseconds % 1000) * 1000); | 
 |   } | 
 |   return now; | 
 | } | 
 |  | 
 | #elif defined(HAVE_CLOCK_GETTIME_MONOTONIC) | 
 |  | 
 | struct timeval tvnow(void) | 
 | { | 
 |   /* | 
 |   ** clock_gettime() is granted to be increased monotonically when the | 
 |   ** monotonic clock is queried. Time starting point is unspecified, it | 
 |   ** could be the system start-up time, the Epoch, or something else, | 
 |   ** in any case the time starting point does not change once that the | 
 |   ** system has started up. | 
 |   */ | 
 |   struct timeval now; | 
 |   struct timespec tsnow; | 
 |   if(0 == clock_gettime(CLOCK_MONOTONIC, &tsnow)) { | 
 |     now.tv_sec = tsnow.tv_sec; | 
 |     now.tv_usec = (int)(tsnow.tv_nsec / 1000); | 
 |   } | 
 |   /* | 
 |   ** Even when the configure process has truly detected monotonic clock | 
 |   ** availability, it might happen that it is not actually available at | 
 |   ** run-time. When this occurs simply fallback to other time source. | 
 |   */ | 
 | #ifdef HAVE_GETTIMEOFDAY | 
 |   else | 
 |     (void)gettimeofday(&now, NULL); | 
 | #else | 
 |   else { | 
 |     now.tv_sec = time(NULL); | 
 |     now.tv_usec = 0; | 
 |   } | 
 | #endif | 
 |   return now; | 
 | } | 
 |  | 
 | #elif defined(HAVE_GETTIMEOFDAY) | 
 |  | 
 | struct timeval tvnow(void) | 
 | { | 
 |   /* | 
 |   ** gettimeofday() is not granted to be increased monotonically, due to | 
 |   ** clock drifting and external source time synchronization it can jump | 
 |   ** forward or backward in time. | 
 |   */ | 
 |   struct timeval now; | 
 |   (void)gettimeofday(&now, NULL); | 
 |   return now; | 
 | } | 
 |  | 
 | #else | 
 |  | 
 | struct timeval tvnow(void) | 
 | { | 
 |   /* | 
 |   ** time() returns the value of time in seconds since the Epoch. | 
 |   */ | 
 |   struct timeval now; | 
 |   now.tv_sec = time(NULL); | 
 |   now.tv_usec = 0; | 
 |   return now; | 
 | } | 
 |  | 
 | #endif | 
 |  | 
 | /* | 
 |  * Make sure that the first argument is the more recent time, as otherwise | 
 |  * we'll get a weird negative time-diff back... | 
 |  * | 
 |  * Returns: the time difference in number of milliseconds. | 
 |  */ | 
 | long tvdiff(struct timeval newer, struct timeval older) | 
 | { | 
 |   return (long)(newer.tv_sec-older.tv_sec)*1000+ | 
 |     (long)(newer.tv_usec-older.tv_usec)/1000; | 
 | } | 
 |  | 
 | /* Case insensitive compare. Accept NULL pointers. */ | 
 | int struplocompare(const char *p1, const char *p2) | 
 | { | 
 |   if(!p1) | 
 |     return p2? -1: 0; | 
 |   if(!p2) | 
 |     return 1; | 
 | #ifdef HAVE_STRCASECMP | 
 |   return strcasecmp(p1, p2); | 
 | #elif defined(HAVE_STRCMPI) | 
 |   return strcmpi(p1, p2); | 
 | #elif defined(HAVE_STRICMP) | 
 |   return stricmp(p1, p2); | 
 | #else | 
 |   return strcmp(p1, p2); | 
 | #endif | 
 | } | 
 |  | 
 | /* Indirect version to use as qsort callback. */ | 
 | int struplocompare4sort(const void *p1, const void *p2) | 
 | { | 
 |   return struplocompare(* (char * const *) p1, * (char * const *) p2); | 
 | } |