timeutil.h 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. /*
  2. * Copyright (c) 2019 Peter Bigot Consulting, LLC
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Utilities supporting operation on time data structures.
  9. *
  10. * POSIX defines gmtime() to convert from time_t to struct tm, but all
  11. * inverse transformations are non-standard or require access to time
  12. * zone information. timeutil_timegm() implements the functionality
  13. * of the GNU extension timegm() function, but changes the error value
  14. * as @c EOVERFLOW is not a standard C error identifier.
  15. *
  16. * timeutil_timegm64() is provided to support full precision
  17. * conversion on platforms where @c time_t is limited to 32 bits.
  18. */
  19. #ifndef ZEPHYR_INCLUDE_SYS_TIMEUTIL_H_
  20. #define ZEPHYR_INCLUDE_SYS_TIMEUTIL_H_
  21. #include <time.h>
  22. #include <zephyr/types.h>
  23. #ifdef __cplusplus
  24. extern "C" {
  25. #endif
  26. /**
  27. * @defgroup timeutil_apis Time Utility APIs
  28. * @defgroup timeutil_repr_apis Time Representation APIs
  29. * @ingroup timeutil_apis
  30. * @{
  31. */
  32. /**
  33. * @brief Convert broken-down time to a POSIX epoch offset in seconds.
  34. *
  35. * @param tm pointer to broken down time.
  36. *
  37. * @return the corresponding time in the POSIX epoch time scale.
  38. *
  39. * @see http://man7.org/linux/man-pages/man3/timegm.3.html
  40. */
  41. int64_t timeutil_timegm64(const struct tm *tm);
  42. /**
  43. * @brief Convert broken-down time to a POSIX epoch offset in seconds.
  44. *
  45. * @param tm pointer to broken down time.
  46. *
  47. * @return the corresponding time in the POSIX epoch time scale. If
  48. * the time cannot be represented then @c (time_t)-1 is returned and
  49. * @c errno is set to @c ERANGE`.
  50. *
  51. * @see http://man7.org/linux/man-pages/man3/timegm.3.html
  52. */
  53. time_t timeutil_timegm(const struct tm *tm);
  54. /**
  55. * @}
  56. * @defgroup timeutil_sync_apis Time Synchronization APIs
  57. * @ingroup timeutil_apis
  58. * @{
  59. */
  60. /**
  61. * @brief Immutable state for synchronizing two clocks.
  62. *
  63. * Values required to convert durations between two time scales.
  64. *
  65. * @note The accuracy of the translation and calculated skew between sources
  66. * depends on the resolution of these frequencies. A reference frequency with
  67. * microsecond or nanosecond resolution would produce the most accurate
  68. * tracking when the local reference is the Zephyr tick counter. A reference
  69. * source like an RTC chip with 1 Hz resolution requires a much larger
  70. * interval between sampled instants to detect relative clock drift.
  71. */
  72. struct timeutil_sync_config {
  73. /** The nominal instance counter rate in Hz.
  74. *
  75. * This value is assumed to be precise, but may drift depending on
  76. * the reference clock source.
  77. *
  78. * The value must be positive.
  79. */
  80. uint32_t ref_Hz;
  81. /** The nominal local counter rate in Hz.
  82. *
  83. * This value is assumed to be inaccurate but reasonably stable. For
  84. * a local clock driven by a crystal oscillator an error of 25 ppm is
  85. * common; for an RC oscillator larger errors should be expected. The
  86. * timeutil_sync infrastructure can calculate the skew between the
  87. * local and reference clocks and apply it when converting between
  88. * time scales.
  89. *
  90. * The value must be positive.
  91. */
  92. uint32_t local_Hz;
  93. };
  94. /**
  95. * @brief Representation of an instant in two time scales.
  96. *
  97. * Capturing the same instant in two time scales provides a
  98. * registration point that can be used to convert between those time
  99. * scales.
  100. */
  101. struct timeutil_sync_instant {
  102. /** An instant in the reference time scale.
  103. *
  104. * This must never be zero in an initialized timeutil_sync_instant
  105. * object.
  106. */
  107. uint64_t ref;
  108. /** The corresponding instance in the local time scale.
  109. *
  110. * This may be zero in a valid timeutil_sync_instant object.
  111. */
  112. uint64_t local;
  113. };
  114. /**
  115. * @brief State required to convert instants between time scales.
  116. *
  117. * This state in conjunction with functions that manipulate it capture
  118. * the offset information necessary to convert between two timescales
  119. * along with information that corrects for skew due to inaccuracies
  120. * in clock rates.
  121. *
  122. * State objects should be zero-initialized before use.
  123. */
  124. struct timeutil_sync_state {
  125. /** Pointer to reference and local rate information. */
  126. const struct timeutil_sync_config *cfg;
  127. /** The base instant in both time scales. */
  128. struct timeutil_sync_instant base;
  129. /** The most recent instant in both time scales.
  130. *
  131. * This is captured here to provide data for skew calculation.
  132. */
  133. struct timeutil_sync_instant latest;
  134. /** The scale factor used to correct for clock skew.
  135. *
  136. * The nominal rate for the local counter is assumed to be
  137. * inaccurate but stable, i.e. it will generally be some
  138. * parts-per-million faster or slower than specified.
  139. *
  140. * A duration in observed local clock ticks must be multiplied by
  141. * this value to produce a duration in ticks of a clock operating at
  142. * the nominal local rate.
  143. *
  144. * A zero value indicates that the skew has not been initialized.
  145. * If the value is zero when #base is initialized the skew will be
  146. * set to 1. Otherwise the skew is assigned through
  147. * timeutil_sync_state_set_skew().
  148. */
  149. float skew;
  150. };
  151. /**
  152. * @brief Record a new instant in the time synchronization state.
  153. *
  154. * Note that this updates only the latest persisted instant. The skew
  155. * is not adjusted automatically.
  156. *
  157. * @param tsp pointer to a timeutil_sync_state object.
  158. *
  159. * @param inst the new instant to be recorded. This becomes the base
  160. * instant if there is no base instant, otherwise the value must be
  161. * strictly after the base instant in both the reference and local
  162. * time scales.
  163. *
  164. * @retval 0 if installation succeeded in providing a new base
  165. * @retval 1 if installation provided a new latest instant
  166. * @retval -EINVAL if the new instant is not compatible with the base instant
  167. */
  168. int timeutil_sync_state_update(struct timeutil_sync_state *tsp,
  169. const struct timeutil_sync_instant *inst);
  170. /**
  171. * @brief Update the state with a new skew and possibly base value.
  172. *
  173. * Set the skew from a value retrieved from persistent storage, or
  174. * calculated based on recent skew estimations including from
  175. * timeutil_sync_estimate_skew().
  176. *
  177. * Optionally update the base timestamp. If the base is replaced the
  178. * latest instant will be cleared until timeutil_sync_state_update() is
  179. * invoked.
  180. *
  181. * @param tsp pointer to a time synchronization state.
  182. *
  183. * @param skew the skew to be used. The value must be positive and
  184. * shouldn't be too far away from 1.
  185. *
  186. * @param base optional new base to be set. If provided this becomes
  187. * the base timestamp that will be used along with skew to convert
  188. * between reference and local timescale instants. Setting the base
  189. * clears the captured latest value.
  190. *
  191. * @return 0 if skew was updated
  192. * @return -EINVAL if skew was not valid
  193. */
  194. int timeutil_sync_state_set_skew(struct timeutil_sync_state *tsp, float skew,
  195. const struct timeutil_sync_instant *base);
  196. /**
  197. * @brief Estimate the skew based on current state.
  198. *
  199. * Using the base and latest syncpoints from the state determine the
  200. * skew of the local clock relative to the reference clock. See
  201. * timeutil_sync_state::skew.
  202. *
  203. * @param tsp pointer to a time synchronization state. The base and latest
  204. * syncpoints must be present and the latest syncpoint must be after
  205. * the base point in the local time scale.
  206. *
  207. * @return the estimated skew, or zero if skew could not be estimated.
  208. */
  209. float timeutil_sync_estimate_skew(const struct timeutil_sync_state *tsp);
  210. /**
  211. * @brief Interpolate a reference timescale instant from a local
  212. * instant.
  213. *
  214. * @param tsp pointer to a time synchronization state. This must have a base
  215. * and a skew installed.
  216. *
  217. * @param local an instant measured in the local timescale. This may
  218. * be before or after the base instant.
  219. *
  220. * @param refp where the corresponding instant in the reference
  221. * timescale should be stored. A negative interpolated reference time
  222. * produces an error. If interpolation fails the referenced object is
  223. * not modified.
  224. *
  225. * @retval 0 if interpolated using a skew of 1
  226. * @retval 1 if interpolated using a skew not equal to 1
  227. * @retval -EINVAL
  228. * * the times synchronization state is not adequately initialized
  229. * * @p refp is null
  230. * @retval -ERANGE the interpolated reference time would be negative
  231. */
  232. int timeutil_sync_ref_from_local(const struct timeutil_sync_state *tsp,
  233. uint64_t local, uint64_t *refp);
  234. /**
  235. * @brief Interpolate a local timescale instant from a reference
  236. * instant.
  237. *
  238. * @param tsp pointer to a time synchronization state. This must have a base
  239. * and a skew installed.
  240. *
  241. * @param ref an instant measured in the reference timescale. This
  242. * may be before or after the base instant.
  243. *
  244. * @param localp where the corresponding instant in the local
  245. * timescale should be stored. An interpolated value before local
  246. * time 0 is provided without error. If interpolation fails the
  247. * referenced object is not modified.
  248. *
  249. * @retval 0 if successful with a skew of 1
  250. * @retval 1 if successful with a skew not equal to 1
  251. * @retval -EINVAL
  252. * * the time synchronization state is not adequately initialized
  253. * * @p refp is null
  254. */
  255. int timeutil_sync_local_from_ref(const struct timeutil_sync_state *tsp,
  256. uint64_t ref, int64_t *localp);
  257. /**
  258. * @brief Convert from a skew to an error in parts-per-billion.
  259. *
  260. * A skew of 1.0 has zero error. A skew less than 1 has a positive
  261. * error (clock is faster than it should be). A skew greater than one
  262. * has a negative error (clock is slower than it should be).
  263. *
  264. * Note that due to the limited precision of @c float compared with @c
  265. * double the smallest error that can be represented is about 120 ppb.
  266. * A "precise" time source may have error on the order of 2000 ppb.
  267. *
  268. * A skew greater than 3.14748 may underflow the 32-bit
  269. * representation; this represents a clock running at less than 1/3
  270. * its nominal rate.
  271. *
  272. * @return skew error represented as parts-per-billion, or INT32_MIN
  273. * if the skew cannot be represented in the return type.
  274. */
  275. int32_t timeutil_sync_skew_to_ppb(float skew);
  276. #ifdef __cplusplus
  277. }
  278. #endif
  279. /**
  280. * @}
  281. */
  282. #endif /* ZEPHYR_INCLUDE_SYS_TIMEUTIL_H_ */