mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
timeUtils.cpp
Go to the documentation of this file.
1/** \file timeUtils.cpp
2 * \brief Definitions for 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 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#include "sys/timeUtils.hpp"
30
31namespace mx
32{
33namespace sys
34{
35
36template <>
37double get_curr_time<double, CLOCK_REALTIME>( timespec &tsp )
38{
39 clock_gettime( CLOCK_REALTIME, &tsp );
40
41 return ( (double)tsp.tv_sec ) + ( (double)tsp.tv_nsec ) / 1e9;
42}
43
44template <>
45double get_curr_time<double, CLOCK_REALTIME>()
46{
47 struct timespec tsp;
48 return get_curr_time<double, CLOCK_REALTIME>( tsp );
49}
50
51void sleep( unsigned sec )
52{
53 std::this_thread::sleep_for( std::chrono::seconds( sec ) );
54}
55
56void milliSleep( unsigned msec )
57{
58 std::this_thread::sleep_for( std::chrono::milliseconds( msec ) );
59}
60
61void microSleep( unsigned usec )
62{
63 std::this_thread::sleep_for( std::chrono::microseconds( usec ) );
64}
65
66void nanoSleep( unsigned nsec )
67{
68 std::this_thread::sleep_for( std::chrono::nanoseconds( nsec ) );
69}
70
71void timespecAddNsec( timespec &ts, unsigned nsec )
72{
73 ts.tv_nsec += nsec % 1000000000;
74 ts.tv_sec += nsec / 1000000000;
75
76 if( ts.tv_nsec > 999999999 )
77 {
78 ts.tv_nsec -= 1000000000;
79 ts.tv_sec += 1;
80 }
81}
82
83double Cal2mjd( int yr, int mon, int day, int hr, int min, double sec )
84{
85 double djm0;
86 double djm;
87
88 int rv = sofa::iauCal2jd( yr, mon, day, &djm0, &djm );
89
90 if( rv < 0 )
91 return (double)rv;
92
93 djm0 = djm + ( ( (double)hr ) / ( 24.0 ) + ( (double)min ) / ( 60.0 * 24.0 ) + sec / ( 3600.0 * 24.0 ) );
94
95 return djm0;
96}
97
98int ISO8601dateBreakdown( int &yr, int &mon, int &day, int &hr, int &min, double &sec, const std::string &fdate )
99{
100 if( fdate.length() < 19 )
101 return -4;
102
103 yr = atoi( fdate.substr( 0, 4 ).c_str() );
104 mon = atoi( fdate.substr( 5, 2 ).c_str() );
105 day = atoi( fdate.substr( 8, 2 ).c_str() );
106
107 double _hr, _min;
108 parse_hms( _hr, _min, sec, fdate.substr( 11 ) );
109
110 hr = floor( _hr );
111 min = floor( _min );
112
113 return 0;
114}
115
116double ISO8601date2mjd( const std::string &fdate )
117{
118 if( fdate.length() < 19 )
119 return -4;
120
121 int yr, mon, day, hr, min;
122 // double hr, min, sec;
123 double sec;
124
125 ISO8601dateBreakdown( yr, mon, day, hr, min, sec, fdate );
126
127 return Cal2mjd( yr, mon, day, hr, min, sec );
128}
129
130template <>
131std::string ISO8601DateTimeStr<time_t>( const time_t &timeIn, int timeZone )
132{
133 tm bdt;
134 gmtime_r( &timeIn, &bdt );
135
136 char tstr[25];
137
138 strftime( tstr, 25, "%FT%H:%M:%S", &bdt );
139
140 std::string result = tstr;
141
142 if( timeZone == 1 )
143 result += "Z";
144 if( timeZone == 2 )
145 result += "+00:00";
146
147 return result;
148}
149
150template <>
151std::string ISO8601DateTimeStr<timespec>( const timespec &timeIn, int timeZone )
152{
153 std::string result = ISO8601DateTimeStr<time_t>( timeIn.tv_sec, 0 );
154
155 char tstr[20];
156
157 snprintf( tstr, 20, ".%09ld", timeIn.tv_nsec );
158
159 result += tstr;
160
161 if( timeZone == 1 )
162 result += "Z";
163 if( timeZone == 2 )
164 result += "+00:00";
165
166 return result;
167}
168
169std::string ISO8601DateTimeStr( int timeZone )
170{
171 return ISO8601DateTimeStr<time_t>( ::time( 0 ), timeZone );
172}
173
174std::string ISO8601DateTimeStrMJD( const double &timeIn, int timeZone )
175{
176 int iy, im, id;
177 double fd;
178
179 sofa::iauJd2cal( DJM0, timeIn, &iy, &im, &id, &fd );
180
181 int hr, mn;
182
183 hr = floor( fd * 24.0 );
184 fd = ( fd - hr / 24.0 ) * 24.0;
185
186 mn = floor( fd * 60. );
187
188 fd = ( fd - mn / 60.0 ) * 3600.0;
189
190 char tstr[32];
191
192 snprintf( tstr, 32, "%04d-%02d-%02dT%02d:%02d:%012.9f", iy, im, id, hr, mn, fd );
193
194 std::string result = tstr;
195
196 if( timeZone == 1 )
197 result += "Z";
198 if( timeZone == 2 )
199 result += "+00:00";
200
201 return result;
202}
203
204int timeStamp( std::string &tstamp, timespec &ts )
205{
206 tm uttime; // The broken down time.
207
208 time_t t0 = ts.tv_sec;
209
210 if( gmtime_r( &t0, &uttime ) == 0 )
211 {
212 std::cerr << "Error getting UT time (gmtime_r returned 0). At: " << __FILE__ << " " << __LINE__ << "\n";
213 return -1;
214 }
215
216 char buffer[48];
217
218 snprintf( buffer,
219 sizeof( buffer ),
220 "%04i%02i%02i%02i%02i%02i%09i",
221 uttime.tm_year + 1900,
222 uttime.tm_mon + 1,
223 uttime.tm_mday,
224 uttime.tm_hour,
225 uttime.tm_min,
226 uttime.tm_sec,
227 static_cast<int>( ts.tv_nsec ) ); // casting in case we switch type of time_ns.
228
229 tstamp = buffer;
230
231 return 0;
232}
233
234int timespecUTC2TAIMJD( double &djm, double &djmf, const timespec &tsp, tm *tm0 )
235{
236 double dat, djm0;
237 tm *tmrv;
238 int rv1, rv2;
239
240 // Get the broken down time corresponding to tsp0
241 tmrv = gmtime_r( &tsp.tv_sec, tm0 );
242 if( tmrv == 0 )
243 return -10;
244
245 // Then determine deltaAT = TAI-UTC
246 rv1 = sofa::iauDat( 1900 + tm0->tm_year, 1 + tm0->tm_mon, tm0->tm_mday, 0.0, &dat );
247 if( rv1 < 0 )
248 return rv1;
249
250 // And get the MJD
251 rv2 = sofa::iauCal2jd( 1900 + tm0->tm_year, 1 + tm0->tm_mon, tm0->tm_mday, &djm0, &djm );
252 if( rv2 < 0 )
253 return rv2;
254
255 // Finally calculate the day fraction
256 djmf = ( (double)tm0->tm_hour ) / 24.0 + ( (double)tm0->tm_min ) / ( 24.0 * 60. ) +
257 ( ( (double)tm0->tm_sec ) + ( (double)tsp.tv_nsec / 1e9 ) + dat ) / ( 24.0 * 3600.0 );
258
259 if( djmf >= 1.0 )
260 {
261 djmf -= 1.0;
262 djm += 1.0;
263 }
264
265 if( rv1 )
266 return rv1;
267 return 0;
268}
269
270timespec meanTimespec( timespec ts1, timespec ts2 )
271{
272 double means = ( ts1.tv_sec + ts2.tv_sec ) / 2.0;
273 double meanns = ( ts1.tv_nsec + ts2.tv_nsec ) / 2.0;
274
275 ts1.tv_sec = floor( means );
276 ts1.tv_nsec = round( meanns );
277
278 if( means != floor( means ) )
279 {
280 ts1.tv_nsec += 5e8;
281
282 if( ts1.tv_nsec >= 1e9 )
283 {
284 ts1.tv_sec += 1;
285 ts1.tv_nsec -= 1e9;
286 }
287 }
288
289 return ts1;
290}
291
292namespace tscomp
293{
294
295bool operator<( timespec const &tsL, timespec const &tsR )
296{
297 return ( ( ( tsL.tv_sec == tsR.tv_sec ) && ( tsL.tv_nsec < tsR.tv_nsec ) ) || ( tsL.tv_sec < tsR.tv_sec ) );
298}
299
300bool operator>( timespec const &tsL, timespec const &tsR )
301{
302 return ( ( ( tsL.tv_sec == tsR.tv_sec ) && ( tsL.tv_nsec > tsR.tv_nsec ) ) || ( tsL.tv_sec > tsR.tv_sec ) );
303}
304
305bool operator==( timespec const &tsL, timespec const &tsR )
306{
307 return ( ( tsL.tv_sec == tsR.tv_sec ) && ( tsL.tv_nsec == tsR.tv_nsec ) );
308}
309
310bool operator<=( timespec const &tsL, timespec const &tsR )
311{
312 return ( tsL < tsR || tsL == tsR );
313}
314
315bool operator>=( timespec const &tsL, timespec const &tsR )
316{
317 return ( tsL > tsR || tsL == tsR );
318}
319
320} // namespace tscomp
321
322} // namespace sys
323} // namespace mx
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)
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
Utilities for working with time.
timespec meanTimespec(timespec ts1, timespec ts2)
Calculate the mean time of two times given by timespecs.