/* clock.h: Definitions for clock calculations This file is part of Cygwin. This software is a copyrighted work licensed under the terms of the Cygwin license. Please consult the file "CYGWIN_LICENSE" for details. */ #ifndef __CLOCK_H__ #define __CLOCK_H__ #include /* Must be a power of 2. */ #define MAX_CLOCKS (8) /* Conversions for per-process and per-thread clocks */ #define CLOCKID(cid) \ ((cid) % MAX_CLOCKS) #define PID_TO_CLOCKID(pid) \ ((pid) * MAX_CLOCKS + CLOCK_PROCESS_CPUTIME_ID) #define CLOCKID_TO_PID(cid) \ (((cid) - CLOCK_PROCESS_CPUTIME_ID) / MAX_CLOCKS) #define CLOCKID_IS_PROCESS(cid) \ (CLOCKID(cid) == CLOCK_PROCESS_CPUTIME_ID) #define THREADID_TO_CLOCKID(tid) \ ((tid) * MAX_CLOCKS + CLOCK_THREAD_CPUTIME_ID) #define CLOCKID_TO_THREADID(cid) \ (((cid) - CLOCK_THREAD_CPUTIME_ID) / MAX_CLOCKS) #define CLOCKID_IS_THREAD(cid) \ (CLOCKID(cid) == CLOCK_THREAD_CPUTIME_ID) /* Largest delay in ms for sleep and alarm calls. Allow actual delay to exceed requested delay by 10 s. Express as multiple of 1000 (i.e. seconds) + max resolution The tv_sec argument in timeval structures cannot exceed CLOCK_DELAY_MAX / 1000 - 1, so that adding fractional part and rounding won't exceed CLOCK_DELAY_MAX */ #define CLOCK_DELAY_MAX ((((UINT_MAX - 10000) / 1000) * 1000) + 10) /* 100ns difference between Windows and UNIX timebase. */ #define FACTOR (0x19db1ded53e8000LL) /* # of nanosecs per second. */ #define NSPERSEC (1000000000LL) /* # of 100ns intervals per second. */ #define NS100PERSEC (10000000LL) /* # of microsecs per second. */ #define USPERSEC (1000000LL) /* # of millisecs per second. */ #define MSPERSEC (1000L) class clk_t { protected: /* Some values are returned as ticks/s, some as 100ns period of a single tick. Store the original value and use a computation method making the most sense for the value given, to avoid rounding issues. */ union { LONGLONG ticks_per_sec; LONGLONG period; }; void init (); virtual int now (clockid_t, struct timespec *) = 0; public: int nsecs (clockid_t _id, struct timespec *ts) { return now (_id, ts); } virtual void resolution (struct timespec *); /* shortcuts for non-process/thread clocks */ void nsecs (struct timespec *ts) { now (0, ts); } ULONGLONG nsecs () { struct timespec ts; now (0, &ts); return (ULONGLONG) ts.tv_sec * NSPERSEC + ts.tv_nsec; } LONGLONG n100secs () { struct timespec ts; now (0, &ts); return ts.tv_sec * NS100PERSEC + ts.tv_nsec / (NSPERSEC/NS100PERSEC); } LONGLONG usecs () { struct timespec ts; now (0, &ts); return ts.tv_sec * USPERSEC + ts.tv_nsec / (NSPERSEC/USPERSEC); } LONGLONG msecs () { struct timespec ts; now (0, &ts); return ts.tv_sec * MSPERSEC + ts.tv_nsec / (NSPERSEC/MSPERSEC); } }; class clk_realtime_coarse_t : public clk_t { virtual int now (clockid_t, struct timespec *); }; class clk_realtime_t : public clk_t { void init (); virtual int now (clockid_t, struct timespec *); public: virtual void resolution (struct timespec *); }; class clk_process_t : public clk_t { virtual int now (clockid_t, struct timespec *); }; class clk_thread_t : public clk_t { virtual int now (clockid_t, struct timespec *); }; class clk_monotonic_t : public clk_t { protected: void init (); private: virtual int now (clockid_t, struct timespec *); public: virtual void resolution (struct timespec *); /* Under strace 1st call is so early that vtable is NULL. */ LONGLONG strace_usecs () { struct timespec ts; clk_monotonic_t::now (0, &ts); return ts.tv_sec * USPERSEC + ts.tv_nsec / (NSPERSEC/USPERSEC); } }; class clk_monotonic_coarse_t : public clk_t { virtual int now (clockid_t, struct timespec *); }; class clk_boottime_t : public clk_monotonic_t { virtual int now (clockid_t, struct timespec *); }; clk_t *get_clock (clockid_t clk_id); /* Compute interval between two timespec timestamps: ts1 = ts1 - ts0. */ static inline void ts_diff (const struct timespec &ts0, struct timespec &ts1) { ts1.tv_nsec -= ts0.tv_nsec; if (ts1.tv_nsec < 0) { ts1.tv_nsec += NSPERSEC; --ts1.tv_sec; } ts1.tv_sec -= ts0.tv_sec; } static inline bool valid_timespec (const timespec& ts) { if (ts.tv_nsec < 0 || ts.tv_nsec >= NSPERSEC || ts.tv_sec < 0) return false; return true; } #endif /*__CLOCK_H__*/