/* * Copyright 2022-2024 The OpenSSL Project Authors. All Rights Reserved. * * Licensed under the Apache License 2.0 (the "License"). You may not use * this file except in compliance with the License. You can obtain a copy * in the file LICENSE in the source distribution or at * https://www.openssl.org/source/license.html */ #include #include "internal/e_os.h" /* system-specific variants defining OSSL_sleep() */ #if defined(OPENSSL_SYS_UNIX) || defined(__DJGPP__) # if defined(OPENSSL_USE_USLEEP) \ || defined(__DJGPP__) \ || (defined(__TANDEM) && defined(_REENTRANT)) /* * usleep() was made obsolete by POSIX.1-2008, and nanosleep() * should be used instead. However, nanosleep() isn't implemented * on the platforms given above, so we still use it for those. * Also, OPENSSL_USE_USLEEP can be defined to enable the use of * usleep, if it turns out that nanosleep() is unavailable. */ # include void OSSL_sleep(uint64_t millis) { unsigned int s = (unsigned int)(millis / 1000); unsigned int us = (unsigned int)((millis % 1000) * 1000); if (s > 0) sleep(s); /* * On NonStop with the PUT thread model, thread context switch is * cooperative, with usleep() being a "natural" context switch point. * We avoid checking us > 0 here, to allow that context switch to * happen. */ usleep(us); } # elif defined(__TANDEM) && !defined(_REENTRANT) # include void OSSL_sleep(uint64_t millis) { /* HPNS does not support usleep for non threaded apps */ PROCESS_DELAY_(millis * 1000); } # else /* nanosleep is defined by POSIX.1-2001 */ # include void OSSL_sleep(uint64_t millis) { struct timespec ts; ts.tv_sec = (long int) (millis / 1000); ts.tv_nsec = (long int) (millis % 1000) * 1000000ul; nanosleep(&ts, NULL); } # endif #elif defined(_WIN32) && !defined(OPENSSL_SYS_UEFI) # include void OSSL_sleep(uint64_t millis) { /* * Windows' Sleep() takes a DWORD argument, which is smaller than * a uint64_t, so we need to limit it to 49 days, which should be enough. */ DWORD limited_millis = (DWORD)-1; if (millis < limited_millis) limited_millis = (DWORD)millis; Sleep(limited_millis); } #else /* Fallback to a busy wait */ # include "internal/time.h" static void ossl_sleep_secs(uint64_t secs) { /* * sleep() takes an unsigned int argument, which is smaller than * a uint64_t, so it needs to be limited to 136 years which * should be enough even for Sleeping Beauty. */ unsigned int limited_secs = UINT_MAX; if (secs < limited_secs) limited_secs = (unsigned int)secs; sleep(limited_secs); } static void ossl_sleep_millis(uint64_t millis) { const OSSL_TIME finish = ossl_time_add(ossl_time_now(), ossl_ms2time(millis)); while (ossl_time_compare(ossl_time_now(), finish) < 0) /* busy wait */ ; } void OSSL_sleep(uint64_t millis) { ossl_sleep_secs(millis / 1000); ossl_sleep_millis(millis % 1000); } #endif /* defined(OPENSSL_SYS_UNIX) || defined(__DJGPP__) */