113498266Sopenharmony_ci/*************************************************************************** 213498266Sopenharmony_ci * _ _ ____ _ 313498266Sopenharmony_ci * Project ___| | | | _ \| | 413498266Sopenharmony_ci * / __| | | | |_) | | 513498266Sopenharmony_ci * | (__| |_| | _ <| |___ 613498266Sopenharmony_ci * \___|\___/|_| \_\_____| 713498266Sopenharmony_ci * 813498266Sopenharmony_ci * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al. 913498266Sopenharmony_ci * 1013498266Sopenharmony_ci * This software is licensed as described in the file COPYING, which 1113498266Sopenharmony_ci * you should have received as part of this distribution. The terms 1213498266Sopenharmony_ci * are also available at https://curl.se/docs/copyright.html. 1313498266Sopenharmony_ci * 1413498266Sopenharmony_ci * You may opt to use, copy, modify, merge, publish, distribute and/or sell 1513498266Sopenharmony_ci * copies of the Software, and permit persons to whom the Software is 1613498266Sopenharmony_ci * furnished to do so, under the terms of the COPYING file. 1713498266Sopenharmony_ci * 1813498266Sopenharmony_ci * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY 1913498266Sopenharmony_ci * KIND, either express or implied. 2013498266Sopenharmony_ci * 2113498266Sopenharmony_ci * SPDX-License-Identifier: curl 2213498266Sopenharmony_ci * 2313498266Sopenharmony_ci ***************************************************************************/ 2413498266Sopenharmony_ci#include "curl_setup.h" 2513498266Sopenharmony_ci#include <curl/curl.h> 2613498266Sopenharmony_ci#include "testutil.h" 2713498266Sopenharmony_ci#include "memdebug.h" 2813498266Sopenharmony_ci 2913498266Sopenharmony_ci#if defined(_WIN32) 3013498266Sopenharmony_ci 3113498266Sopenharmony_cistruct timeval tutil_tvnow(void) 3213498266Sopenharmony_ci{ 3313498266Sopenharmony_ci /* 3413498266Sopenharmony_ci ** GetTickCount() is available on _all_ Windows versions from W95 up 3513498266Sopenharmony_ci ** to nowadays. Returns milliseconds elapsed since last system boot, 3613498266Sopenharmony_ci ** increases monotonically and wraps once 49.7 days have elapsed. 3713498266Sopenharmony_ci */ 3813498266Sopenharmony_ci struct timeval now; 3913498266Sopenharmony_ci DWORD milliseconds = GetTickCount(); 4013498266Sopenharmony_ci now.tv_sec = (long)(milliseconds / 1000); 4113498266Sopenharmony_ci now.tv_usec = (long)((milliseconds % 1000) * 1000); 4213498266Sopenharmony_ci return now; 4313498266Sopenharmony_ci} 4413498266Sopenharmony_ci 4513498266Sopenharmony_ci#elif defined(HAVE_CLOCK_GETTIME_MONOTONIC) 4613498266Sopenharmony_ci 4713498266Sopenharmony_cistruct timeval tutil_tvnow(void) 4813498266Sopenharmony_ci{ 4913498266Sopenharmony_ci /* 5013498266Sopenharmony_ci ** clock_gettime() is granted to be increased monotonically when the 5113498266Sopenharmony_ci ** monotonic clock is queried. Time starting point is unspecified, it 5213498266Sopenharmony_ci ** could be the system start-up time, the Epoch, or something else, 5313498266Sopenharmony_ci ** in any case the time starting point does not change once that the 5413498266Sopenharmony_ci ** system has started up. 5513498266Sopenharmony_ci */ 5613498266Sopenharmony_ci struct timeval now; 5713498266Sopenharmony_ci struct timespec tsnow; 5813498266Sopenharmony_ci if(0 == clock_gettime(CLOCK_MONOTONIC, &tsnow)) { 5913498266Sopenharmony_ci now.tv_sec = tsnow.tv_sec; 6013498266Sopenharmony_ci now.tv_usec = (int)(tsnow.tv_nsec / 1000); 6113498266Sopenharmony_ci } 6213498266Sopenharmony_ci /* 6313498266Sopenharmony_ci ** Even when the configure process has truly detected monotonic clock 6413498266Sopenharmony_ci ** availability, it might happen that it is not actually available at 6513498266Sopenharmony_ci ** run-time. When this occurs simply fallback to other time source. 6613498266Sopenharmony_ci */ 6713498266Sopenharmony_ci#ifdef HAVE_GETTIMEOFDAY 6813498266Sopenharmony_ci else 6913498266Sopenharmony_ci (void)gettimeofday(&now, NULL); 7013498266Sopenharmony_ci#else 7113498266Sopenharmony_ci else { 7213498266Sopenharmony_ci now.tv_sec = time(NULL); 7313498266Sopenharmony_ci now.tv_usec = 0; 7413498266Sopenharmony_ci } 7513498266Sopenharmony_ci#endif 7613498266Sopenharmony_ci return now; 7713498266Sopenharmony_ci} 7813498266Sopenharmony_ci 7913498266Sopenharmony_ci#elif defined(HAVE_GETTIMEOFDAY) 8013498266Sopenharmony_ci 8113498266Sopenharmony_cistruct timeval tutil_tvnow(void) 8213498266Sopenharmony_ci{ 8313498266Sopenharmony_ci /* 8413498266Sopenharmony_ci ** gettimeofday() is not granted to be increased monotonically, due to 8513498266Sopenharmony_ci ** clock drifting and external source time synchronization it can jump 8613498266Sopenharmony_ci ** forward or backward in time. 8713498266Sopenharmony_ci */ 8813498266Sopenharmony_ci struct timeval now; 8913498266Sopenharmony_ci (void)gettimeofday(&now, NULL); 9013498266Sopenharmony_ci return now; 9113498266Sopenharmony_ci} 9213498266Sopenharmony_ci 9313498266Sopenharmony_ci#else 9413498266Sopenharmony_ci 9513498266Sopenharmony_cistruct timeval tutil_tvnow(void) 9613498266Sopenharmony_ci{ 9713498266Sopenharmony_ci /* 9813498266Sopenharmony_ci ** time() returns the value of time in seconds since the Epoch. 9913498266Sopenharmony_ci */ 10013498266Sopenharmony_ci struct timeval now; 10113498266Sopenharmony_ci now.tv_sec = time(NULL); 10213498266Sopenharmony_ci now.tv_usec = 0; 10313498266Sopenharmony_ci return now; 10413498266Sopenharmony_ci} 10513498266Sopenharmony_ci 10613498266Sopenharmony_ci#endif 10713498266Sopenharmony_ci 10813498266Sopenharmony_ci/* 10913498266Sopenharmony_ci * Make sure that the first argument is the more recent time, as otherwise 11013498266Sopenharmony_ci * we'll get a weird negative time-diff back... 11113498266Sopenharmony_ci * 11213498266Sopenharmony_ci * Returns: the time difference in number of milliseconds. 11313498266Sopenharmony_ci */ 11413498266Sopenharmony_cilong tutil_tvdiff(struct timeval newer, struct timeval older) 11513498266Sopenharmony_ci{ 11613498266Sopenharmony_ci return (long)(newer.tv_sec-older.tv_sec)*1000+ 11713498266Sopenharmony_ci (long)(newer.tv_usec-older.tv_usec)/1000; 11813498266Sopenharmony_ci} 11913498266Sopenharmony_ci 12013498266Sopenharmony_ci/* 12113498266Sopenharmony_ci * Same as tutil_tvdiff but with full usec resolution. 12213498266Sopenharmony_ci * 12313498266Sopenharmony_ci * Returns: the time difference in seconds with subsecond resolution. 12413498266Sopenharmony_ci */ 12513498266Sopenharmony_cidouble tutil_tvdiff_secs(struct timeval newer, struct timeval older) 12613498266Sopenharmony_ci{ 12713498266Sopenharmony_ci if(newer.tv_sec != older.tv_sec) 12813498266Sopenharmony_ci return (double)(newer.tv_sec-older.tv_sec)+ 12913498266Sopenharmony_ci (double)(newer.tv_usec-older.tv_usec)/1000000.0; 13013498266Sopenharmony_ci return (double)(newer.tv_usec-older.tv_usec)/1000000.0; 13113498266Sopenharmony_ci} 13213498266Sopenharmony_ci 13313498266Sopenharmony_ci#ifdef _WIN32 13413498266Sopenharmony_ciHMODULE win32_load_system_library(const TCHAR *filename) 13513498266Sopenharmony_ci{ 13613498266Sopenharmony_ci size_t filenamelen = _tcslen(filename); 13713498266Sopenharmony_ci size_t systemdirlen = GetSystemDirectory(NULL, 0); 13813498266Sopenharmony_ci size_t written; 13913498266Sopenharmony_ci TCHAR *path; 14013498266Sopenharmony_ci 14113498266Sopenharmony_ci if(!filenamelen || filenamelen > 32768 || 14213498266Sopenharmony_ci !systemdirlen || systemdirlen > 32768) 14313498266Sopenharmony_ci return NULL; 14413498266Sopenharmony_ci 14513498266Sopenharmony_ci /* systemdirlen includes null character */ 14613498266Sopenharmony_ci path = malloc(sizeof(TCHAR) * (systemdirlen + 1 + filenamelen)); 14713498266Sopenharmony_ci if(!path) 14813498266Sopenharmony_ci return NULL; 14913498266Sopenharmony_ci 15013498266Sopenharmony_ci /* if written >= systemdirlen then nothing was written */ 15113498266Sopenharmony_ci written = GetSystemDirectory(path, (unsigned int)systemdirlen); 15213498266Sopenharmony_ci if(!written || written >= systemdirlen) 15313498266Sopenharmony_ci return NULL; 15413498266Sopenharmony_ci 15513498266Sopenharmony_ci if(path[written - 1] != _T('\\')) 15613498266Sopenharmony_ci path[written++] = _T('\\'); 15713498266Sopenharmony_ci 15813498266Sopenharmony_ci _tcscpy(path + written, filename); 15913498266Sopenharmony_ci 16013498266Sopenharmony_ci return LoadLibrary(path); 16113498266Sopenharmony_ci} 16213498266Sopenharmony_ci#endif 163