gptp.h 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. /*
  2. * Copyright (c) 2017 Intel Corporation.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Public functions for the Precision Time Protocol Stack.
  9. *
  10. */
  11. #ifndef ZEPHYR_INCLUDE_NET_GPTP_H_
  12. #define ZEPHYR_INCLUDE_NET_GPTP_H_
  13. /**
  14. * @brief generic Precision Time Protocol (gPTP) support
  15. * @defgroup gptp gPTP support
  16. * @ingroup networking
  17. * @{
  18. */
  19. #include <net/net_core.h>
  20. #include <net/ptp_time.h>
  21. #include <stdbool.h>
  22. #ifdef __cplusplus
  23. extern "C" {
  24. #endif
  25. /** @cond INTERNAL_HIDDEN */
  26. #define GPTP_OFFSET_SCALED_LOG_VAR_UNKNOWN 0x436A
  27. #define GPTP_PRIORITY1_NON_GM_CAPABLE 255
  28. #define GPTP_PRIORITY1_GM_CAPABLE 248
  29. #if defined(CONFIG_NET_GPTP_BMCA_PRIORITY2)
  30. #define GPTP_PRIORITY2_DEFAULT CONFIG_NET_GPTP_BMCA_PRIORITY2
  31. #else
  32. #define GPTP_PRIORITY2_DEFAULT 248
  33. #endif
  34. /** @endcond */
  35. /**
  36. * @brief Scaled Nanoseconds.
  37. */
  38. struct gptp_scaled_ns {
  39. /** High half. */
  40. int32_t high;
  41. /** Low half. */
  42. int64_t low;
  43. } __packed;
  44. /**
  45. * @brief UScaled Nanoseconds.
  46. */
  47. struct gptp_uscaled_ns {
  48. /** High half. */
  49. uint32_t high;
  50. /** Low half. */
  51. uint64_t low;
  52. } __packed;
  53. /** @cond INTERNAL_HIDDEN */
  54. #if defined(CONFIG_NEWLIB_LIBC)
  55. #include <math.h>
  56. #define GPTP_POW2(exp) pow(2, exp)
  57. #else
  58. static inline double gptp_pow2(int exp)
  59. {
  60. double res;
  61. if (exp >= 0) {
  62. res = 1 << exp;
  63. } else {
  64. res = 1.0;
  65. while (exp++) {
  66. res /= 2;
  67. }
  68. }
  69. return res;
  70. }
  71. #define GPTP_POW2(exp) gptp_pow2(exp)
  72. #endif
  73. /* Pre-calculated constants */
  74. /* 2^16 */
  75. #define GPTP_POW2_16 65536.0
  76. /* 2^41 */
  77. #define GPTP_POW2_41 2199023255552.0
  78. /* Message types. Event messages have BIT(3) set to 0, and general messages
  79. * have that bit set to 1. IEEE 802.1AS chapter 10.5.2.2.2
  80. */
  81. #define GPTP_SYNC_MESSAGE 0x00
  82. #define GPTP_DELAY_REQ_MESSAGE 0x01
  83. #define GPTP_PATH_DELAY_REQ_MESSAGE 0x02
  84. #define GPTP_PATH_DELAY_RESP_MESSAGE 0x03
  85. #define GPTP_FOLLOWUP_MESSAGE 0x08
  86. #define GPTP_DELAY_RESP_MESSAGE 0x09
  87. #define GPTP_PATH_DELAY_FOLLOWUP_MESSAGE 0x0a
  88. #define GPTP_ANNOUNCE_MESSAGE 0x0b
  89. #define GPTP_SIGNALING_MESSAGE 0x0c
  90. #define GPTP_MANAGEMENT_MESSAGE 0x0d
  91. #define GPTP_IS_EVENT_MSG(msg_type) (!((msg_type) & BIT(3)))
  92. #define GPTP_CLOCK_ID_LEN 8
  93. /** @endcond */
  94. /**
  95. * @brief Port Identity.
  96. */
  97. struct gptp_port_identity {
  98. /** Clock identity of the port. */
  99. uint8_t clk_id[GPTP_CLOCK_ID_LEN];
  100. /** Number of the port. */
  101. uint16_t port_number;
  102. } __packed;
  103. struct gptp_flags {
  104. union {
  105. /** Byte access. */
  106. uint8_t octets[2];
  107. /** Whole field access. */
  108. uint16_t all;
  109. };
  110. } __packed;
  111. struct gptp_hdr {
  112. /** Type of the message. */
  113. uint8_t message_type:4;
  114. /** Transport specific, always 1. */
  115. uint8_t transport_specific:4;
  116. /** Version of the PTP, always 2. */
  117. uint8_t ptp_version:4;
  118. /** Reserved field. */
  119. uint8_t reserved0:4;
  120. /** Total length of the message from the header to the last TLV. */
  121. uint16_t message_length;
  122. /** Domain number, always 0. */
  123. uint8_t domain_number;
  124. /** Reserved field. */
  125. uint8_t reserved1;
  126. /** Message flags. */
  127. struct gptp_flags flags;
  128. /** Correction Field. The content depends of the message type. */
  129. int64_t correction_field;
  130. /** Reserved field. */
  131. uint32_t reserved2;
  132. /** Port Identity of the sender. */
  133. struct gptp_port_identity port_id;
  134. /** Sequence Id. */
  135. uint16_t sequence_id;
  136. /** Control value. Sync: 0, Follow-up: 2, Others: 5. */
  137. uint8_t control;
  138. /** Message Interval in Log2 for Sync and Announce messages. */
  139. int8_t log_msg_interval;
  140. } __packed;
  141. /** @cond INTERNAL_HIDDEN */
  142. #define GPTP_GET_CURRENT_TIME_USCALED_NS(port, uscaled_ns_ptr) \
  143. do { \
  144. (uscaled_ns_ptr)->low = \
  145. gptp_get_current_time_nanosecond(port) << 16; \
  146. (uscaled_ns_ptr)->high = 0; \
  147. } while (false)
  148. /** @endcond */
  149. /**
  150. * @typedef gptp_phase_dis_callback_t
  151. * @brief Define callback that is called after a phase discontinuity has been
  152. * sent by the grandmaster.
  153. * @param gm_identity A pointer to first element of a ClockIdentity array.
  154. * The size of the array is GPTP_CLOCK_ID_LEN.
  155. * @param time_base A pointer to the value of timeBaseIndicator of the current
  156. * grandmaster.
  157. * @param last_gm_ph_change A pointer to the value of lastGmPhaseChange received
  158. * from grandmaster.
  159. * @param last_gm_freq_change A pointer to the value of lastGmFreqChange
  160. * received from the grandmaster.
  161. */
  162. typedef void (*gptp_phase_dis_callback_t)(
  163. uint8_t *gm_identity,
  164. uint16_t *time_base,
  165. struct gptp_scaled_ns *last_gm_ph_change,
  166. double *last_gm_freq_change);
  167. /**
  168. * @brief Phase discontinuity callback structure.
  169. *
  170. * Stores the phase discontinuity callback information. Caller must make sure
  171. * that the variable pointed by this is valid during the lifetime of
  172. * registration. Typically this means that the variable cannot be
  173. * allocated from stack.
  174. */
  175. struct gptp_phase_dis_cb {
  176. /** Node information for the slist. */
  177. sys_snode_t node;
  178. /** Phase discontinuity callback. */
  179. gptp_phase_dis_callback_t cb;
  180. };
  181. /**
  182. * @brief ClockSourceTime.invoke function parameters
  183. *
  184. * Parameters passed by ClockSourceTime.invoke function.
  185. */
  186. struct gptp_clk_src_time_invoke_params {
  187. /** Frequency change on the last Time Base Indicator Change. */
  188. double last_gm_freq_change;
  189. /** The time this function is invoked. */
  190. struct net_ptp_extended_time src_time;
  191. /** Phase change on the last Time Base Indicator Change. */
  192. struct gptp_scaled_ns last_gm_phase_change;
  193. /** Time Base - changed only if Phase or Frequency changes. */
  194. uint16_t time_base_indicator;
  195. };
  196. /**
  197. * @brief Register a phase discontinuity callback.
  198. *
  199. * @param phase_dis Caller specified handler for the callback.
  200. * @param cb Callback to register.
  201. */
  202. void gptp_register_phase_dis_cb(struct gptp_phase_dis_cb *phase_dis,
  203. gptp_phase_dis_callback_t cb);
  204. /**
  205. * @brief Unregister a phase discontinuity callback.
  206. *
  207. * @param phase_dis Caller specified handler for the callback.
  208. */
  209. void gptp_unregister_phase_dis_cb(struct gptp_phase_dis_cb *phase_dis);
  210. /**
  211. * @brief Call a phase discontinuity callback function.
  212. */
  213. void gptp_call_phase_dis_cb(void);
  214. /**
  215. * @brief Get gPTP time.
  216. *
  217. * @param slave_time A pointer to structure where timestamp will be saved.
  218. * @param gm_present A pointer to a boolean where status of the
  219. * presence of a grand master will be saved.
  220. *
  221. * @return Error code. 0 if no error.
  222. */
  223. int gptp_event_capture(struct net_ptp_time *slave_time, bool *gm_present);
  224. /**
  225. * @brief Utility function to print clock id to a user supplied buffer.
  226. *
  227. * @param clk_id Clock id
  228. * @param output Output buffer
  229. * @param output_len Output buffer len
  230. *
  231. * @return Pointer to output buffer
  232. */
  233. char *gptp_sprint_clock_id(const uint8_t *clk_id, char *output,
  234. size_t output_len);
  235. /**
  236. * @typedef gptp_port_cb_t
  237. * @brief Callback used while iterating over gPTP ports
  238. *
  239. * @param port Port number
  240. * @param iface Pointer to network interface
  241. * @param user_data A valid pointer to user data or NULL
  242. */
  243. typedef void (*gptp_port_cb_t)(int port, struct net_if *iface,
  244. void *user_data);
  245. /**
  246. * @brief Go through all the gPTP ports and call callback for each of them.
  247. *
  248. * @param cb User-supplied callback function to call
  249. * @param user_data User specified data
  250. */
  251. void gptp_foreach_port(gptp_port_cb_t cb, void *user_data);
  252. /**
  253. * @brief Get gPTP domain.
  254. * @details This contains all the configuration / status of the gPTP domain.
  255. *
  256. * @return Pointer to domain or NULL if not found.
  257. */
  258. struct gptp_domain *gptp_get_domain(void);
  259. /**
  260. * @brief This interface is used by the ClockSource entity to provide time to
  261. * the ClockMaster entity of a time-aware system.
  262. *
  263. * @param arg Current state and parameters of the ClockSource entity.
  264. */
  265. void gptp_clk_src_time_invoke(struct gptp_clk_src_time_invoke_params *arg);
  266. /**
  267. * @brief Return pointer to gPTP packet header in network packet.
  268. *
  269. * @param pkt Network packet (received or sent)
  270. *
  271. * @return Pointer to gPTP header.
  272. */
  273. struct gptp_hdr *gptp_get_hdr(struct net_pkt *pkt);
  274. #ifdef __cplusplus
  275. }
  276. #endif
  277. /**
  278. * @}
  279. */
  280. #endif /* ZEPHYR_INCLUDE_NET_GPTP_H_ */