//----------------------------------------------------------------------------- // Copyright (C) Proxmark3 contributors. See AUTHORS.md for details. // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // See LICENSE.txt for the text of the license. //----------------------------------------------------------------------------- // utilities requiring Posix library functions //----------------------------------------------------------------------------- // ensure availability even with -std=c99; must be included before #if !defined(_WIN32) #define _POSIX_C_SOURCE 200112L // need localtime_r() #else #include #endif #include "util_posix.h" #include #include // Timer functions #if !defined (_WIN32) #include static void nsleep(uint64_t n) { struct timespec timeout; timeout.tv_sec = n / 1000000000; timeout.tv_nsec = n % 1000000000; while (nanosleep(&timeout, &timeout) && errno == EINTR); } void msleep(uint32_t n) { nsleep(1000000 * (uint64_t)n); } #endif // _WIN32 #ifdef __APPLE__ #ifndef CLOCK_MONOTONIC #define CLOCK_MONOTONIC (1) #endif #ifndef CLOCK_REALTIME #define CLOCK_REALTIME (2) #endif #include #include #include #include /* clock_gettime is not implemented on OSX prior to 10.12 */ int _civet_clock_gettime(int clk_id, struct timespec *t); int _civet_clock_gettime(int clk_id, struct timespec *t) { memset(t, 0, sizeof(*t)); if (clk_id == CLOCK_REALTIME) { struct timeval now; int rv = gettimeofday(&now, NULL); if (rv) { return rv; } t->tv_sec = now.tv_sec; t->tv_nsec = now.tv_usec * 1000; return 0; } else if (clk_id == CLOCK_MONOTONIC) { static uint64_t clock_start_time = 0; static mach_timebase_info_data_t timebase_info = {0, 0}; uint64_t now = mach_absolute_time(); if (clock_start_time == 0) { mach_timebase_info(&timebase_info); clock_start_time = now; } now = (uint64_t)((double)(now - clock_start_time) * (double)timebase_info.numer / (double)timebase_info.denom); t->tv_sec = now / 1000000000; t->tv_nsec = now % 1000000000; return 0; } return -1; // EINVAL - Clock ID is unknown } /* if clock_gettime is declared, then __CLOCK_AVAILABILITY will be defined */ #ifdef __CLOCK_AVAILABILITY /* If we compiled with Mac OSX 10.12 or later, then clock_gettime will be declared * but it may be NULL at runtime. So we need to check before using it. */ int _civet_safe_clock_gettime(int clk_id, struct timespec *t); int _civet_safe_clock_gettime(int clk_id, struct timespec *t) { if (clock_gettime) { return clock_gettime(clk_id, t); } return _civet_clock_gettime(clk_id, t); } #define clock_gettime _civet_safe_clock_gettime #else #define clock_gettime _civet_clock_gettime #endif #endif // a milliseconds timer for performance measurement uint64_t msclock(void) { #if defined(_WIN32) #include // WORKAROUND FOR MinGW (some versions - use if normal code does not compile) // It has no _ftime_s and needs explicit inclusion of timeb.h #include struct _timeb t; _ftime(&t); return 1000 * (uint64_t)t.time + t.millitm; // NORMAL CODE (use _ftime_s) //struct _timeb t; //if (_ftime_s(&t)) { // return 0; //} else { // return 1000 * t.time + t.millitm; //} #else struct timespec t; clock_gettime(CLOCK_MONOTONIC, &t); return (1000 * (uint64_t)t.tv_sec + t.tv_nsec / 1000000); #endif } // a micro seconds timer for performance measurement uint64_t usclock(void) { #if defined(_WIN32) #include // WORKAROUND FOR MinGW (some versions - use if normal code does not compile) // It has no _ftime_s and needs explicit inclusion of timeb.h #include struct _timeb t; _ftime(&t); return 1000 * (uint64_t)t.time + t.millitm; // NORMAL CODE (use _ftime_s) //struct _timeb t; //if (_ftime_s(&t)) { // return 0; //} else { // return 1000 * t.time + t.millitm; //} #else struct timespec t; clock_gettime(CLOCK_MONOTONIC, &t); //return (1000 * (uint64_t)t.tv_sec + t.tv_nsec / 1000); return (1000 * (uint64_t)t.tv_sec + (t.tv_nsec / 1000)); #endif }