net_timeout.h 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. /** @file
  2. * @brief Network timer with wrap around
  3. *
  4. * Timer that runs longer than about 49 days needs to calculate wraps.
  5. */
  6. /*
  7. * Copyright (c) 2018 Intel Corporation
  8. * Copyright (c) 2020 Nordic Semiconductor ASA
  9. *
  10. * SPDX-License-Identifier: Apache-2.0
  11. */
  12. #ifndef ZEPHYR_INCLUDE_NET_NET_TIMEOUT_H_
  13. #define ZEPHYR_INCLUDE_NET_NET_TIMEOUT_H_
  14. /**
  15. * @brief Network long timeout primitives and helpers
  16. * @defgroup net_timeout Network long timeout primitives and helpers
  17. * @ingroup networking
  18. * @{
  19. */
  20. #include <string.h>
  21. #include <stdbool.h>
  22. #include <limits.h>
  23. #include <zephyr/types.h>
  24. #include <sys/slist.h>
  25. #ifdef __cplusplus
  26. extern "C" {
  27. #endif
  28. /** @brief Divisor used to support ms resolution timeouts.
  29. *
  30. * Because delays are processed in work queues which are not invoked
  31. * synchronously with clock changes we need to be able to detect timeouts
  32. * after they occur, which requires comparing "deadline" to "now" with enough
  33. * "slop" to handle any observable latency due to "now" advancing past
  34. * "deadline".
  35. *
  36. * The simplest solution is to use the native conversion of the well-defined
  37. * 32-bit unsigned difference to a 32-bit signed difference, which caps the
  38. * maximum delay at INT32_MAX. This is compatible with the standard mechanism
  39. * for detecting completion of deadlines that do not overflow their
  40. * representation.
  41. */
  42. #define NET_TIMEOUT_MAX_VALUE ((uint32_t)INT32_MAX)
  43. /** Generic struct for handling network timeouts.
  44. *
  45. * Except for the linking node, all access to state from these objects must go
  46. * through the defined API.
  47. */
  48. struct net_timeout {
  49. /** Used to link multiple timeouts that share a common timer infrastructure.
  50. *
  51. * For examples a set of related timers may use a single delayed work
  52. * structure, which is always scheduled at the shortest time to a
  53. * timeout event.
  54. */
  55. sys_snode_t node;
  56. /* Time at which the timer was last set.
  57. *
  58. * This usually corresponds to the low 32 bits of k_uptime_get(). */
  59. uint32_t timer_start;
  60. /* Portion of remaining timeout that does not exceed
  61. * NET_TIMEOUT_MAX_VALUE.
  62. *
  63. * This value is updated in parallel with timer_start and wrap_counter
  64. * by net_timeout_evaluate().
  65. */
  66. uint32_t timer_timeout;
  67. /* Timer wrap count.
  68. *
  69. * This tracks multiples of NET_TIMEOUT_MAX_VALUE milliseconds that
  70. * have yet to pass. It is also updated along with timer_start and
  71. * wrap_counter by net_timeout_evaluate().
  72. */
  73. uint32_t wrap_counter;
  74. };
  75. /** @brief Configure a network timeout structure.
  76. *
  77. * @param timeout a pointer to the timeout state.
  78. *
  79. * @param lifetime the duration of the timeout in seconds.
  80. *
  81. * @param now the time at which the timeout started counting down, in
  82. * milliseconds. This is generally a captured value of k_uptime_get_32().
  83. */
  84. void net_timeout_set(struct net_timeout *timeout,
  85. uint32_t lifetime,
  86. uint32_t now);
  87. /** @brief Return the 64-bit system time at which the timeout will complete.
  88. *
  89. * @note Correct behavior requires invocation of net_timeout_evaluate() at its
  90. * specified intervals.
  91. *
  92. * @param timeout state a pointer to the timeout state, initialized by
  93. * net_timeout_set() and maintained by net_timeout_evaluate().
  94. *
  95. * @param now the full-precision value of k_uptime_get() relative to which the
  96. * deadline will be calculated.
  97. *
  98. * @return the value of k_uptime_get() at which the timeout will expire.
  99. */
  100. int64_t net_timeout_deadline(const struct net_timeout *timeout,
  101. int64_t now);
  102. /** @brief Calculate the remaining time to the timeout in whole seconds.
  103. *
  104. * @note This function rounds the remaining time down, i.e. if the timeout
  105. * will occur in 3500 milliseconds the value 3 will be returned.
  106. *
  107. * @note Correct behavior requires invocation of net_timeout_evaluate() at its
  108. * specified intervals.
  109. *
  110. * @param timeout a pointer to the timeout state
  111. *
  112. * @param now the time relative to which the estimate of remaining time should
  113. * be calculated. This should be recently captured value from
  114. * k_uptime_get_32().
  115. *
  116. * @retval 0 if the timeout has completed.
  117. * @retval positive the remaining duration of the timeout, in seconds.
  118. */
  119. uint32_t net_timeout_remaining(const struct net_timeout *timeout,
  120. uint32_t now);
  121. /** @brief Update state to reflect elapsed time and get new delay.
  122. *
  123. * This function must be invoked periodically to (1) apply the effect of
  124. * elapsed time on what remains of a total delay that exceeded the maximum
  125. * representable delay, and (2) determine that either the timeout has
  126. * completed or that the infrastructure must wait a certain period before
  127. * checking again for completion.
  128. *
  129. * @param timeout a pointer to the timeout state
  130. *
  131. * @param now the time relative to which the estimate of remaining time should
  132. * be calculated. This should be recently captured value from
  133. * k_uptime_get_32().
  134. *
  135. * @retval 0 if the timeout has completed
  136. * @retval positive the maximum delay until the state of this timeout should
  137. * be re-evaluated, in milliseconds.
  138. */
  139. uint32_t net_timeout_evaluate(struct net_timeout *timeout,
  140. uint32_t now);
  141. #ifdef __cplusplus
  142. }
  143. #endif
  144. /**
  145. * @}
  146. */
  147. #endif /* ZEPHYR_INCLUDE_NET_NET_TIMEOUT_H_ */