MagAO-X
Operations Applications Utilities Source
timespecX.hpp
Go to the documentation of this file.
1 /** \file timespecX.hpp
2  * \brief A fixed-width timespec structure and utilities.
3  * \author Jared R. Males (jaredmales@gmail.com)
4  *
5  * \ingroup flatlogs_files
6  *
7  * History:
8  * - 2017-06-27 created by JRM
9  * - 2018-08-17 moved to flatlogs
10  */
11 
12 #ifndef flatlogs_timespecX_hpp
13 #define flatlogs_timespecX_hpp
14 
15 #include <cstdint>
16 
17 #include "logDefs.hpp"
18 
19 namespace flatlogs
20 {
21 
22 ///A fixed-width timespec structure.
23 /** To ensure that binary encoding of time is stable regardless of environment, we use a custom timespec
24  * composed of fixed-width types.
25  *
26  * \note This is NOT binary compatible with plain timespec. Use the provided conversions.
27  *
28  * \ingroup flatlogs_time
29  *
30  */
31 struct timespecX
32 {
33  secT time_s; ///< Time since the Unix epoch
34  nanosecT time_ns; ///< Nanoseconds.
35 
36 
37  ///Convert a native timespec to a timespecX.
38  /**
39  * \returns this reference, if values are 0 and 0 then the input was too big or negative.
40  */
41  timespecX & operator=( const timespec & ts /**< [in] the native timespec from which to get values */)
42  {
43  if(ts.tv_sec < 0 || ts.tv_sec > 4294967295) ///\todo make this use minval and maxval
44  {
45  time_s = 0;
46  time_ns = 0;
47  }
48  else
49  {
50  time_s = ts.tv_sec;
51  time_ns = ts.tv_nsec;
52  }
53 
54  return *this;
55  }
56 
57  ///Get a native timespec from this custom one.
58  timespec getTimespec()
59  {
60  struct timespec ts;
61 
62  ts.tv_sec = time_s;
63  ts.tv_nsec = time_ns;
64 
65  return ts;
66  }
67 
68  ///Fill the the timespecX with the current time.
69  /** This is based on the usual clock_gettime. clockid_t is a template parameter
70  * since we probaby always want CLOCK_REALTIME, but if we don't for some reason
71  * it will be passed in the same order as in clock_gettime.
72  *
73  * \tparam clk_id specifies the type.
74  */
75  template<clockid_t clk_id=CLOCK_REALTIME>
76  void gettime()
77  {
78  struct timespec ts;
79  clock_gettime(clk_id, &ts);
80  (*this) = ts; //see operator=
81  }
82 
83  ///Get the filename timestamp for this timespecX.
84  /** Fills in a string with the timestamp encoded as
85  * \verbatim
86  YYYYMMDDHHMMSSNNNNNNNNN
87  \endverbatim
88  *
89  */
90  int timeStamp(std::string & tstamp /**< [out] the string to hold the formatted time */)
91  {
92  tm uttime;//The broken down time.
93 
94  time_t t0 = time_s;
95 
96  if(gmtime_r(&t0, &uttime) == 0)
97  {
98  std::cerr << "Error getting UT time (gmtime_r returned 0). At: " << __FILE__ << " " << __LINE__ << "\n";
99  return -1;
100  }
101 
102  char buffer[24];
103 
104  snprintf(buffer, 24, "%04i%02i%02i%02i%02i%02i%09i", uttime.tm_year+1900, uttime.tm_mon+1, uttime.tm_mday, uttime.tm_hour, uttime.tm_min, uttime.tm_sec, static_cast<int>(time_ns)); //casting in case we switch type of time_ns.
105 
106  tstamp = buffer;
107 
108  return 0;
109  }
110 
111  ///Get the filname timestamp for this timespecX.
112  /** Returns a string with the timestamp encoded as
113  * \verbatim
114  YYYYMMDDHHMMSSNNNNNNNNN
115  \endverbatim
116  *
117  */
118  std::string timeStamp()
119  {
120  std::string tstamp;
121  timeStamp(tstamp);
122  return tstamp;
123  }
124 
125  /// Get a date-time string in ISO 8601 format for timespecX
126  /** Returns a string in the ISO 8601 format:
127  * \verbatim
128  YYYY-MM-DDTHH:MM:SS.SSSSSSSSS
129  \endverbatim
130  *
131  *
132  * \retval std::string containing the formated date/time
133  *
134  */
135  std::string ISO8601DateTimeStrX()
136  {
137  tm bdt; //broken down time
138  time_t tt = time_s;
139  gmtime_r( &tt, &bdt);
140 
141  char tstr1[25];
142 
143  strftime(tstr1, 25, "%FT%H:%M:%S", &bdt);
144 
145  char tstr2[11];
146 
147  snprintf(tstr2, 11, ".%09i", static_cast<int>(time_ns)); //casting in case we switch to int64_t
148 
149  return std::string(tstr1) + std::string(tstr2);
150  }
151 } __attribute__((packed));
152 
153 ///Convert a timespecX to a native timespec
154 /**
155  * \ingroup flatlogs_time
156  */
157 inline
158 void timespecFromX ( timespec & ts, ///< [out] the native timespec to set
159  const timespecX & tsX ///< [in] the fixed-width timespec from which to get values
160  )
161 {
162  ts.tv_sec = tsX.time_s;
163  ts.tv_nsec = tsX.time_ns;
164 
165 }
166 
167 ///Fill in a timespecX with the current time.
168 /** This is based on the usual clock_gettime. clockid_t is a template parameter
169  * since we probaby always want CLOCK_REALTIME, but if we don't for some reason
170  * it will be passed in the same order as in clock_gettime.
171  *
172  * \tparam clk_id specifies the type.
173  *
174  * \ingroup flatlogs_time
175  *
176  */
177 template<clockid_t clk_id=CLOCK_REALTIME>
178 void clock_gettimeX( timespecX & tsX /**< [out] the fixed-width timespec to populate */)
179 {
180  tsX.gettime<clk_id>();
181 }
182 
183 }//namespace flatlogs
184 
185 
186 #endif //flatlogs_timespecX_hpp
187 
std::string ISO8601DateTimeStrX()
Get a date-time string in ISO 8601 format for timespecX.
Definition: timespecX.hpp:135
int timeStamp(std::string &tstamp)
Get the filename timestamp for this timespecX.
Definition: timespecX.hpp:90
void clock_gettimeX(timespecX &tsX)
Fill in a timespecX with the current time.
Definition: timespecX.hpp:178
void timespecFromX(timespec &ts, const timespecX &tsX)
Convert a timespecX to a native timespec.
Definition: timespecX.hpp:158
uint32_t secT
The type used for seconds.
Definition: logDefs.hpp:27
timespecX & operator=(const timespec &ts)
Convert a native timespec to a timespecX.
Definition: timespecX.hpp:41
secT time_s
Time since the Unix epoch.
Definition: timespecX.hpp:33
nanosecT time_ns
Nanoseconds.
Definition: timespecX.hpp:34
void gettime()
Fill the the timespecX with the current time.
Definition: timespecX.hpp:76
Type definitions for the flatlogs format.
uint32_t nanosecT
The type used for nanoseconds.
Definition: logDefs.hpp:32
timespec getTimespec()
Get a native timespec from this custom one.
Definition: timespecX.hpp:58
std::string timeStamp()
Get the filname timestamp for this timespecX.
Definition: timespecX.hpp:118
A fixed-width timespec structure.
Definition: timespecX.hpp:31