mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
timeUtils.hpp
Go to the documentation of this file.
1/** \file timeUtils.hpp
2 * \brief Utilities for working with time
3 *
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 * \ingroup utils_files
7 *
8 */
9
10//***********************************************************************//
11// Copyright 2015-2020 Jared R. Males (jaredmales@gmail.com)
12//
13// This file is part of mxlib.
14//
15// mxlib is free software: you can redistribute it and/or modify
16// it under the terms of the GNU General Public License as published by
17// the Free Software Foundation, either version 3 of the License, or
18// (at your option) any later version.
19//
20// mxlib is distributed in the hope that it will be useful,
21// but WITHOUT ANY WARRANTY; without even the implied warranty of
22// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
23// GNU General Public License for more details.
24//
25// You should have received a copy of the GNU General Public License
26// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
27//***********************************************************************//
28
29#ifndef timeUtils_hpp
30#define timeUtils_hpp
31
32#include <time.h>
33#include <sys/time.h>
34#include <cmath>
35
36#include <thread>
37#include <chrono>
38
39#include <iostream>
40
41#include "../ioutils/stringUtils.hpp"
42#include "../astro/sofa.hpp"
43
44namespace mx
45{
46namespace sys
47{
48
49/// Get the current system time in seconds.
50/** Uses timespec, so nanosecond resolution is possible.
51 *
52 * \tparam typeT is the type to return the time in [default=double].
53 * must have cast from integer types, and + and / operators defined.
54 * \tparam clk_id is the sys/time.h clock identifier [default=CLOCK_REALTIME]
55 *
56 * \retval typeT containing the current time in seconds
57 *
58 * \test Verify operation of get_curr_time. \ref tests_sys_timeUtils_get_curr_time "[test doc]"
59 *
60 * \ingroup timeutils
61 */
62template <typename typeT = double, clockid_t clk_id = CLOCK_REALTIME>
63typeT get_curr_time( timespec &tsp /**<[out] a timespec to populate with the current time */ )
64{
65 clock_gettime( clk_id, &tsp );
66
67 return ( (typeT)tsp.tv_sec ) + ( (typeT)tsp.tv_nsec ) / 1e9;
68}
69
70// Specialization for most common use case.
71template <>
72double get_curr_time<double, CLOCK_REALTIME>( timespec &tsp );
73
74/// Get the current system time in seconds.
75/** Uses timespec, so nanosecond resolution is possible. This version creates a timespec internally.
76 *
77 * \overload
78 *
79 * \tparam typeT is the type to return the time in [default=double].
80 * must have cast from integer types, and + and / operators defined.
81 * \tparam clk_id is the sys/time.h clock identifier [default=CLOCK_REALTIME]
82 *
83 * \retval typeT containing the current time in seconds
84 *
85 * \test Verify operation of get_curr_time. \ref tests_sys_timeUtils_get_curr_time "[test doc]"
86 *
87 * \ingroup timeutils
88 */
89template <typename typeT = double, clockid_t clk_id = CLOCK_REALTIME>
91{
92 struct timespec tsp;
93 return get_curr_time<typeT, clk_id>( tsp );
94}
95
96// Specialization for most common use case.
97template <>
98double get_curr_time<double, CLOCK_REALTIME>();
99
100/// Sleep for a specified period in seconds.
101/**
102 * \test Verify operation of thread sleep functions. \ref tests_sys_timeUtils_sleep "[test doc]"
103 *
104 * \ingroup timeutils_sleep
105 */
106void sleep( unsigned sec /**< [in] the number of seconds to sleep. */ );
107
108/// Sleep for a specified period in milliseconds.
109/**
110 * \test Verify operation of thread sleep functions. \ref tests_sys_timeUtils_sleep "[test doc]"
111 *
112 * \ingroup timeutils_sleep
113 */
114void milliSleep( unsigned msec /**< [in] the number of milliseconds to sleep. */ );
115
116/// Sleep for a specified period in microseconds.
117/**
118 * \test Verify operation of thread sleep functions. \ref tests_sys_timeUtils_sleep "[test doc]"
119 *
120 * \ingroup timeutils_sleep
121 */
122void microSleep( unsigned usec /**< [in] the number of microseconds to sleep. */ );
123
124/// Sleep for a specified period in nanoseconds.
125/**
126 * \test Verify operation of thread sleep functions. \ref tests_sys_timeUtils_sleep "[test doc]"
127 *
128 * \ingroup timeutils_sleep
129 */
130void nanoSleep( unsigned nsec /**< [in] the number of microseconds to sleep. */ );
131
132/// Adds a time offset to an existing timespec
133/** The offset is specified in nanoseconds, which can be greater than 1e9.
134 *
135 * \test Verify operation of timespecAddNsec. \ref tests_sys_timeUtils_timespecAddNsec "[test doc]"
136 *
137 * \ingroup timeutils
138 */
139void timespecAddNsec( timespec &ts, ///< [in.out] the time to add to
140 unsigned nsec ///< [in] the number of nanoseconds to add to ts.
141);
142
143/** Parse a string of format hh:mm:ss.s
144 * Breaks a time string into constituent parts. Handles -h by distributing the sign to m and s.
145 *
146 * \tparam floatT is a floating point type
147 *
148 * \test Verify parsing of a formatted time string. \ref tests_sys_timeUtils_parse_hms "[test doc]"
149 *
150 * \ingroup timeutils
151 */
152template <typename floatT>
153void parse_hms( floatT &h, ///< [out] the hour component coverted to floatT
154 floatT &m, ///< [out] the minute component converted to floatT
155 floatT &s, ///< [out] the second component converted to floatT
156 const std::string &hmsstr ///< [in] a string of format hh:mm:ss.s where ss.s can be of any precision
157)
158{
159 int st, en;
160
161 int sgn = 1;
162
163 st = 0;
164 en = hmsstr.find( ':', st );
165
166 h = ioutils::convertFromString<floatT>( hmsstr.substr( st, en - st ).c_str() );
167
168 // Check for negative
169 if( std::signbit( h ) )
170 sgn = -1;
171
172 st = en + 1;
173
174 en = hmsstr.find( ':', st );
175
176 m = sgn * ioutils::convertFromString<floatT>( hmsstr.substr( st, en - st ) );
177
178 st = en + 1;
179
180 s = sgn * ioutils::convertFromString<floatT>( hmsstr.substr( st, hmsstr.length() - st ).c_str() );
181}
182
183/// Converts a Gregorian calendar date into modified Julian date (MJD).
184/** Uses the SOFA function iauCal2jd. This is not a template in floating point
185 * because SOFA is always double precision.
186 *
187 * \retval double containing the MJD
188 * \retval <0 on error (-1 = bad year, -2 = bad month, -3 = bad day)
189 *
190 * \test Verify calculation of MJD. \ref tests_sys_timeUtils_Cal2mjd "[test doc]"
191 *
192 * \ingroup timeutils
193 */
194double Cal2mjd( int yr, ///< [in] Gregorian calendar year
195 int mon, ///< [in] Gregorian calendar month
196 int day, ///< [in] Gregorian calendar day
197 int hr, ///< [in] Gregorian calendar hour
198 int min, ///< [in] Gregorian calendar minute
199 double sec ///< [in] Gregorian calendar second
200);
201
202/// Parse an ISO8601 date of the form "YYYY-MM-DDTHH:MM:SS.S" into the individual components.
203/** Parsing is currently only for the exact form above, which is the form in a FITS file.
204 * See https://en.wikipedia.org/?title=ISO_8601.
205 *
206 * \returns 0 on success
207 * \returns -4 if fdate is not long enough
208 *
209 * \test Verify parsing of an ISO 8601 time string \ref tests_sys_timeUtils_ISO8601dateBreakdown "[test doc]"
210 *
211 * \ingroup timeutils
212 */
213int ISO8601dateBreakdown( int &yr, ///< [out] Gregorian calendar year
214 int &mon, ///< [out] Gregorian calendar month
215 int &day, ///< [out] Gregorian calendar day
216 int &hr, ///< [out] Gregorian calendar hour
217 int &min, ///< [out] Gregorian calendar minute
218 double &sec, ///< [out] Gregorian calendar second
219 const std::string &fdate ///< [in] is a standard ISO8601 date string
220);
221
222/// Parse an ISO8601 date of the form "YYYY-MM-DDTHH:MM:SS.S" and return the modified Julian date (MJD)
223/** Parsing is currently only for the exact form above, which is the form in a FITS file.
224 * See https://en.wikipedia.org/?title=ISO_8601.
225 * After parsing calls Cal2mjd.
226 *
227 * \test Verify conversion of an ISO 8601 time string to MJD. \ref tests_sys_timeUtils_ISO8601date2mjd "[test doc]"
228 *
229 * \ingroup timeutils
230 */
231double ISO8601date2mjd( const std::string &fdate /**<[in] a standard ISO8601 date string*/ );
232
233/// Get a date-time string in ISO 8601 format
234/** For recognized time types, returns a string in the ISO 8601 format:
235 * YYYY-MM-DDYHH:MM:SS.SS, with optional timezone designation such as Z or +00:00.
236 *
237 * \tparam timeT is the time type
238 *
239 * \retval std::string containing the format date/time
240 *
241 * \ingroup timeutils
242 */
243template <typename timeT>
244std::string ISO8601DateTimeStr( const timeT &timeIn, ///< [in] the input time
245 int timeZone = 0 ///< [in] [optional] specifies whether to include a timezone
246 ///< designation. 0=> none, 1=> letter, 2=>offset.
247);
248
249/// Get a date-time string in ISO 8601 format for time_t
250/** Returns a string in the ISO 8601 format:
251 * YYYY-MM-DDYHH:MM:SS, with optional timezone designation such as Z or +00:00.
252 *
253 * \retval std::string containing the format date/time
254 *
255 * \ingroup timeutils
256 */
257template <>
258std::string ISO8601DateTimeStr<time_t>( const time_t &timeIn, ///< [in] the input time
259 int timeZone ///< [in] [optional] specifies whether to include a timezone
260 ///< designation. 0=> none, 1=> letter, 2=>offset.
261);
262
263/// Get a date-time string in ISO 8601 format for timespec
264/** Returns a string in the ISO 8601 format:
265 * YYYY-MM-DDYHH:MM:SS.SSSSSSSSS, with optional timezone designation such as Z or +00:00.
266 *
267 * \retval std::string containing the format date/time
268 *
269 * \ingroup timeutils
270 */
271template <>
272std::string ISO8601DateTimeStr<timespec>( const timespec &timeIn, ///< [in] the input time
273 int timeZone ///< [in] [optional] specifies whether to include a timezone
274 ///< designation. 0=> none, 1=> letter, 2=>offset.
275);
276
277/// Get a date-time string in ISO 8601 format for the current UTC time
278/** Returns a string in the ISO 8601 format:
279 * YYYY-MM-DDYHH:MM:SS.SS, with optional timezone designation such as Z or +00:00.
280 *
281 * \overload
282 *
283 * \retval std::string containing the format date/time
284 *
285 * \ingroup timeutils
286 */
287
288std::string ISO8601DateTimeStr(int timeZone = 0 /**< [in] [optional] specifies whether to include a timezone designation. 0=> none, 1=> letter, 2=>offset.*/);
289
290/// Get a date-time string in ISO 8601 format for an MJD
291/** Returns a string in the ISO 8601 format:
292 * YYYY-MM-DDYHH:MM:SS.SSSSSSSSS, with optional timezone designation such as Z or +00:00.
293 *
294 * \retval std::string containing the format date/time
295 *
296 * \ingroup timeutils
297 */
298std::string ISO8601DateTimeStrMJD(
299 const double &timeIn, ///< [in] the input time
300 int timeZone = 0 ///< [in] specifies whether to include a timezone designation. 0=> none, 1=> letter, 2=>offset.
301);
302/// Get a timestamp string in the form YYYYMMDDHHMMSS.SSSSSSSSS
303/** Assumes the input timespec is in UTC.
304 *
305 * \returns 0 on success
306 * \returns -1 on error.
307 *
308 * \ingroup timeutils
309 */
310int timeStamp( std::string &tstamp, ///< [out] the string to hold the formatted time
311 timespec &ts ///< [in] the timespec from which to produce the timestamp string
312);
313
314/// Convert a UTC timespec to TAI modified Julian date
315/** Converts a timespec assumed to be in <a href="https://en.wikipedia.org/wiki/Coordinated_Universal_Time">Coordinated
316 * Universal Time (UTC)</a> to a <a href="https://en.wikipedia.org/wiki/Julian_day">
317 * Modified Julian Date (MJD)</a> in
318 * <a href="https://en.wikipedia.org/wiki/International_Atomic_Time">International Atomic Time (TAI)</a>.
319 *
320 * \retval 1 SOFA dubious year [see SOFA documentation for iauDat]
321 * \retval 0 success
322 * \retval -1 SOFA bad year [see SOFA documentation for iauDat and iauCal2jd]
323 * \retval -2 SOFA bad month [see SOFA documentation for iauDat and iauCal2jd]
324 * \retval -3 SOFA bad day [see SOFA documentation for iauDat and iauCal2jd]
325 * \retval -4 SOFA bad fractional day [see SOFA documentation for iauDat and iauCal2jd]
326 * \retval -5 SOFA internal error [see SOFA documentation for iauDat and iauCal2jd]
327 * \retval -10 gmtime_r returned error, check errno
328 *
329 * \ingroup timeutils
330 */
331int timespecUTC2TAIMJD( double &djm, ///< [out] the modified Julian day number
332 double &djmf, ///< [out] the fraction of the day
333 const timespec &tsp, ///< [in] contains the UTC time
334 tm *tm0 ///< [out] [optional] will be filled with the broken down UTC time
335);
336
337/// Calculate the mean time of two times given by timespecs
338/**
339 * \returns the mean value of the inputs.
340 */
341timespec meanTimespec( timespec ts1, ///< [in] one of the times to average
342 timespec ts2 ///< [in] the other time to average
343);
344
345namespace tscomp
346{
347
348/// Timespec comparison operator < (see caveats)
349/** Caveats:
350 * - If the inputs are in UTC (or similar scale) this does not account for leap seconds
351 * - Assumes that the `tv_nsec` field does not exceed 999999999 nanoseconds
352 *
353 * \returns true if tsL is earlier than tsR
354 * \returns false otherwise
355 *
356 * \ingroup timeutils_tscomp
357 */
358bool operator<( timespec const &tsL, ///< [in] the left hand side of the comparison
359 timespec const &tsR ///< [in] the right hand side of the comparison
360);
361
362/// Timespec comparison operator > (see caveats)
363/** Caveats:
364 * - If the inputs are in UTC (or similar scale) this does not account for leap seconds
365 * - Assumes that the `tv_nsec` field does not exceed 999999999 nanoseconds
366 *
367 * \returns true if tsL is later than tsR
368 * \returns false otherwise
369 *
370 * \ingroup timeutils_tscomp
371 */
372bool operator>( timespec const &tsL, ///< [in] the left hand side of the comparison
373 timespec const &tsR ///< [in] the right hand side of the comparison
374);
375
376/// Timespec comparison operator == (see caveats)
377/** Caveats:
378 * - If the inputs are in UTC (or similar scale) this does not account for leap seconds
379 * - Assumes that the `tv_nsec` field does not exceed 999999999 nanoseconds
380 *
381 * \returns true if tsL is exactly the same as tsR
382 * \returns false otherwise
383 *
384 * \ingroup timeutils_tscomp
385 */
386bool operator==( timespec const &tsL, ///< [in] the left hand side of the comparison
387 timespec const &tsR ///< [in] the right hand side of the comparison
388);
389
390/// Timespec comparison operator <= (see caveats)
391/** Caveats:
392 * - If the inputs are in UTC (or similar scale) this does not account for leap seconds
393 * - Assumes that the `tv_nsec` field does not exceed 999999999 nanoseconds.
394 *
395 * \returns true if tsL is earlier than or exactly equal to tsR
396 * \returns false otherwise
397 *
398 * \ingroup timeutils_tscomp
399 */
400bool operator<=( timespec const &tsL, ///< [in] the left hand side of the comparison
401 timespec const &tsR ///< [in] the right hand side of the comparison
402);
403
404/// Timespec comparison operator >= (see caveats)
405/** Caveats:
406 * - If the inputs are in UTC (or similar scale) this does not account for leap seconds
407 * - Assumes that the `tv_nsec` field does not exceed 999999999 nanoseconds
408 *
409 * \returns true if tsL is exactly equal to or is later than tsR
410 * \returns false otherwise
411 *
412 * \ingroup timeutils_tscomp
413 */
414bool operator>=( timespec const &tsL, ///< [in] the left hand side of the comparison
415 timespec const &tsR ///< [in] the right hand side of the comparison
416);
417
418} // namespace tscomp
419
420namespace tsop
421{
422
423/// Add an amount of time specified in seconds to a timespec
424/**
425 * \returns the resulting timespec after addition
426 */
427template <typename arithT>
428timespec operator+( timespec ts, ///< [in] the timespec to add to
429 arithT add ///< [in] the seconds to add
430)
431{
432 ts.tv_sec += floor( add );
433
434 ts.tv_nsec += ( add - floor( add ) ) * 1e9;
435
436 if( ts.tv_nsec >= 1e9 )
437 {
438 ts.tv_sec += 1;
439 ts.tv_nsec -= 1e9;
440 }
441
442 return ts;
443}
444
445/// Subtract an amount of time specified in seconds from a timespec
446/**
447 * \returns the resulting timespec after subtraction
448 */
449template <typename arithT>
450timespec operator-( timespec ts, ///< [in] the timespec to subtract from
451 arithT sub ///< [in] the seconds to subtract
452)
453{
454 ts.tv_sec -= floor( sub );
455
456 ts.tv_nsec -= ( sub - floor( sub ) ) * 1e9;
457
458 if( ts.tv_nsec < 0 )
459 {
460 ts.tv_sec -= 1;
461 ts.tv_nsec += 1e9;
462 }
463
464 return ts;
465}
466
467} // namespace tsop
468
469} // namespace sys
470} // namespace mx
471
472#endif // timeUtils_hpp
void nanoSleep(unsigned nsec)
Sleep for a specified period in nanoseconds.
Definition timeUtils.cpp:66
void sleep(unsigned sec)
Sleep for a specified period in seconds.
Definition timeUtils.cpp:51
void milliSleep(unsigned msec)
Sleep for a specified period in milliseconds.
Definition timeUtils.cpp:56
void microSleep(unsigned usec)
Sleep for a specified period in microseconds.
Definition timeUtils.cpp:61
bool operator<(timespec const &tsL, timespec const &tsR)
Timespec comparison operator < (see caveats)
bool operator==(timespec const &tsL, timespec const &tsR)
Timespec comparison operator == (see caveats)
bool operator>(timespec const &tsL, timespec const &tsR)
Timespec comparison operator > (see caveats)
bool operator>=(timespec const &tsL, timespec const &tsR)
Timespec comparison operator >= (see caveats)
bool operator<=(timespec const &tsL, timespec const &tsR)
Timespec comparison operator <= (see caveats)
int ISO8601dateBreakdown(int &yr, int &mon, int &day, int &hr, int &min, double &sec, const std::string &fdate)
Parse an ISO8601 date of the form "YYYY-MM-DDTHH:MM:SS.S" into the individual components.
Definition timeUtils.cpp:98
std::string ISO8601DateTimeStr(const timeT &timeIn, int timeZone=0)
Get a date-time string in ISO 8601 format.
std::string ISO8601DateTimeStrMJD(const double &timeIn, int timeZone=0)
Get a date-time string in ISO 8601 format for an MJD.
int timespecUTC2TAIMJD(double &djm, double &djmf, const timespec &tsp, tm *tm0)
Convert a UTC timespec to TAI modified Julian date.
std::string ISO8601DateTimeStr< timespec >(const timespec &timeIn, int timeZone)
Get a date-time string in ISO 8601 format for timespec.
void timespecAddNsec(timespec &ts, unsigned nsec)
Adds a time offset to an existing timespec.
Definition timeUtils.cpp:71
void parse_hms(floatT &h, floatT &m, floatT &s, const std::string &hmsstr)
typeT get_curr_time()
Get the current system time in seconds.
Definition timeUtils.hpp:90
int timeStamp(std::string &tstamp, timespec &ts)
Get a timestamp string in the form YYYYMMDDHHMMSS.SSSSSSSSS.
double Cal2mjd(int yr, int mon, int day, int hr, int min, double sec)
Converts a Gregorian calendar date into modified Julian date (MJD).
Definition timeUtils.cpp:83
double ISO8601date2mjd(const std::string &fdate)
Parse an ISO8601 date of the form "YYYY-MM-DDTHH:MM:SS.S" and return the modified Julian date (MJD)
std::string ISO8601DateTimeStr< time_t >(const time_t &timeIn, int timeZone)
Get a date-time string in ISO 8601 format for time_t.
The mxlib c++ namespace.
Definition mxError.hpp:106
timespec operator-(timespec ts, arithT sub)
Subtract an amount of time specified in seconds from a timespec.
timespec operator+(timespec ts, arithT add)
Add an amount of time specified in seconds to a timespec.
timespec meanTimespec(timespec ts1, timespec ts2)
Calculate the mean time of two times given by timespecs.