ieee802154_radio.h 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  1. /*
  2. * Copyright (c) 2016 Intel Corporation.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Public IEEE 802.15.4 Radio API
  9. */
  10. #ifndef ZEPHYR_INCLUDE_NET_IEEE802154_RADIO_H_
  11. #define ZEPHYR_INCLUDE_NET_IEEE802154_RADIO_H_
  12. #include <device.h>
  13. #include <net/net_if.h>
  14. #include <net/net_pkt.h>
  15. #include <net/ieee802154.h>
  16. #ifdef __cplusplus
  17. extern "C" {
  18. #endif
  19. /**
  20. * @addtogroup ieee802154
  21. * @{
  22. */
  23. /**
  24. * @brief IEEE 802.15.4 Channel assignments
  25. *
  26. * Channel numbering for 868 MHz, 915 MHz, and 2450 MHz bands.
  27. *
  28. * - Channel 0 is for 868.3 MHz.
  29. * - Channels 1-10 are for 906 to 924 MHz with 2 MHz channel spacing.
  30. * - Channels 11-26 are for 2405 to 2530 MHz with 5 MHz channel spacing.
  31. *
  32. * For more information, please refer to 802.15.4-2015 Section 10.1.2.2.
  33. */
  34. enum ieee802154_channel {
  35. IEEE802154_SUB_GHZ_CHANNEL_MIN = 0,
  36. IEEE802154_SUB_GHZ_CHANNEL_MAX = 10,
  37. IEEE802154_2_4_GHZ_CHANNEL_MIN = 11,
  38. IEEE802154_2_4_GHZ_CHANNEL_MAX = 26,
  39. };
  40. enum ieee802154_hw_caps {
  41. IEEE802154_HW_FCS = BIT(0), /* Frame Check-Sum supported */
  42. IEEE802154_HW_PROMISC = BIT(1), /* Promiscuous mode supported */
  43. IEEE802154_HW_FILTER = BIT(2), /* Filter PAN ID, long/short addr */
  44. IEEE802154_HW_CSMA = BIT(3), /* CSMA-CA supported */
  45. IEEE802154_HW_2_4_GHZ = BIT(4), /* 2.4Ghz radio supported */
  46. IEEE802154_HW_TX_RX_ACK = BIT(5), /* Handles ACK request on TX */
  47. IEEE802154_HW_SUB_GHZ = BIT(6), /* Sub-GHz radio supported */
  48. IEEE802154_HW_ENERGY_SCAN = BIT(7), /* Energy scan supported */
  49. IEEE802154_HW_TXTIME = BIT(8), /* TX at specified time supported */
  50. IEEE802154_HW_SLEEP_TO_TX = BIT(9), /* TX directly from sleep supported */
  51. IEEE802154_HW_TX_SEC = BIT(10), /* TX security hadling supported */
  52. IEEE802154_HW_RXTIME = BIT(11), /* RX at specified time supported */
  53. };
  54. enum ieee802154_filter_type {
  55. IEEE802154_FILTER_TYPE_IEEE_ADDR,
  56. IEEE802154_FILTER_TYPE_SHORT_ADDR,
  57. IEEE802154_FILTER_TYPE_PAN_ID,
  58. IEEE802154_FILTER_TYPE_SRC_IEEE_ADDR,
  59. IEEE802154_FILTER_TYPE_SRC_SHORT_ADDR,
  60. };
  61. enum ieee802154_event {
  62. IEEE802154_EVENT_TX_STARTED, /* Data transmission started */
  63. IEEE802154_EVENT_RX_FAILED, /* Data reception failed */
  64. IEEE802154_EVENT_SLEEP, /* Sleep pending */
  65. };
  66. enum ieee802154_rx_fail_reason {
  67. IEEE802154_RX_FAIL_NOT_RECEIVED, /* Nothing received */
  68. IEEE802154_RX_FAIL_INVALID_FCS, /* Frame had invalid checksum */
  69. IEEE802154_RX_FAIL_ADDR_FILTERED, /* Address did not match */
  70. IEEE802154_RX_FAIL_OTHER /* General reason */
  71. };
  72. typedef void (*energy_scan_done_cb_t)(const struct device *dev,
  73. int16_t max_ed);
  74. typedef void (*ieee802154_event_cb_t)(const struct device *dev,
  75. enum ieee802154_event evt,
  76. void *event_params);
  77. struct ieee802154_filter {
  78. /** @cond ignore */
  79. union {
  80. uint8_t *ieee_addr;
  81. uint16_t short_addr;
  82. uint16_t pan_id;
  83. };
  84. /* @endcond */
  85. };
  86. struct ieee802154_key {
  87. uint8_t *key_value;
  88. uint32_t key_frame_counter;
  89. bool frame_counter_per_key;
  90. uint8_t key_id_mode;
  91. uint8_t key_index;
  92. };
  93. /** IEEE802.15.4 Transmission mode. */
  94. enum ieee802154_tx_mode {
  95. /** Transmit packet immediately, no CCA. */
  96. IEEE802154_TX_MODE_DIRECT,
  97. /** Perform CCA before packet transmission. */
  98. IEEE802154_TX_MODE_CCA,
  99. /** Perform full CSMA CA procedure before packet transmission. */
  100. IEEE802154_TX_MODE_CSMA_CA,
  101. /** Transmit packet in the future, at specified time, no CCA. */
  102. IEEE802154_TX_MODE_TXTIME,
  103. /** Transmit packet in the future, perform CCA before transmission. */
  104. IEEE802154_TX_MODE_TXTIME_CCA,
  105. };
  106. /** IEEE802.15.4 Frame Pending Bit table address matching mode. */
  107. enum ieee802154_fpb_mode {
  108. /** The pending bit shall be set only for addresses found in the list.
  109. */
  110. IEEE802154_FPB_ADDR_MATCH_THREAD,
  111. /** The pending bit shall be cleared for short addresses found in
  112. * the list.
  113. */
  114. IEEE802154_FPB_ADDR_MATCH_ZIGBEE,
  115. };
  116. /** IEEE802.15.4 driver configuration types. */
  117. enum ieee802154_config_type {
  118. /** Indicates how radio driver should set Frame Pending bit in ACK
  119. * responses for Data Requests. If enabled, radio driver should
  120. * determine whether to set the bit or not based on the information
  121. * provided with ``IEEE802154_CONFIG_ACK_FPB`` config and FPB address
  122. * matching mode specified. Otherwise, Frame Pending bit should be set
  123. * to ``1``(see IEEE Std 802.15.4-2006, 7.2.2.3.1).
  124. */
  125. IEEE802154_CONFIG_AUTO_ACK_FPB,
  126. /** Indicates whether to set ACK Frame Pending bit for specific address
  127. * or not. Disabling the Frame Pending bit with no address provided
  128. * (NULL pointer) should disable it for all enabled addresses.
  129. */
  130. IEEE802154_CONFIG_ACK_FPB,
  131. /** Indicates whether the device is a PAN coordinator. */
  132. IEEE802154_CONFIG_PAN_COORDINATOR,
  133. /** Enable/disable promiscuous mode. */
  134. IEEE802154_CONFIG_PROMISCUOUS,
  135. /** Specifies new radio event handler. Specifying NULL as a handler
  136. * will disable radio events notification.
  137. */
  138. IEEE802154_CONFIG_EVENT_HANDLER,
  139. /** Updates MAC keys and key index for radios supporting transmit security. */
  140. IEEE802154_CONFIG_MAC_KEYS,
  141. /** Sets the current MAC frame counter value for radios supporting transmit security. */
  142. IEEE802154_CONFIG_FRAME_COUNTER,
  143. /** Configure a radio reception slot. This can be used for any scheduler reception, e.g.:
  144. * Zigbee GP device, CSL, TSCH, etc.
  145. *
  146. * In order to configure a CSL receiver the upper layer should combine several
  147. * configuration options in the following way:
  148. * 1. Use ``IEEE802154_CONFIG_ENH_ACK_HEADER_IE`` once to inform the radio driver of the
  149. * short and extended addresses of the peer to which it should inject CSL IEs.
  150. * 2. Use ``IEEE802154_CONFIG_CSL_RX_TIME`` periodically, before each use of
  151. * ``IEEE802154_CONFIG_CSL_PERIOD`` setting parameters of the nearest CSL RX window, and
  152. * before each use of IEEE_CONFIG_RX_SLOT setting parameters of the following (not the
  153. * nearest one) CSL RX window, to allow the radio driver to calculate the proper CSL Phase
  154. * to the nearest CSL window to inject in the CSL IEs for both transmitted data and ack
  155. * frames.
  156. * 3. Use ``IEEE802154_CONFIG_CSL_PERIOD`` on each value change to update the current CSL
  157. * period value which will be injected in the CSL IEs together with the CSL Phase based on
  158. * ``IEEE802154_CONFIG_CSL_RX_TIME``.
  159. * 4. Use ``IEEE802154_CONFIG_RX_SLOT`` periodically to schedule the immediate receive
  160. * window earlier enough before the expected window start time, taking into account
  161. * possible clock drifts and scheduling uncertainties.
  162. *
  163. * This diagram shows the usage of the four options over time:
  164. * Start CSL Schedule CSL window
  165. *
  166. * ENH_ACK_HEADER_IE CSL_RX_TIME (following window)
  167. * | |
  168. * | CSL_RX_TIME (nearest window) | RX_SLOT (nearest window)
  169. * | | | |
  170. * | | CSL_PERIOD | |
  171. * | | | | |
  172. * v v v v v
  173. * ----------------------------------------------------------[ CSL window ]-----+
  174. * ^ |
  175. * | |
  176. * +--------------------- loop ---------+
  177. */
  178. IEEE802154_CONFIG_RX_SLOT,
  179. /** Configure CSL receiver (Endpoint) period */
  180. IEEE802154_CONFIG_CSL_PERIOD,
  181. /** Configure the next CSL receive window center, in units of microseconds,
  182. * based on the radio time.
  183. */
  184. IEEE802154_CONFIG_CSL_RX_TIME,
  185. /** Indicates whether to inject IE into ENH ACK Frame for specific address
  186. * or not. Disabling the ENH ACK with no address provided (NULL pointer)
  187. * should disable it for all enabled addresses.
  188. */
  189. IEEE802154_CONFIG_ENH_ACK_HEADER_IE,
  190. };
  191. /** IEEE802.15.4 driver configuration data. */
  192. struct ieee802154_config {
  193. /** Configuration data. */
  194. union {
  195. /** ``IEEE802154_CONFIG_AUTO_ACK_FPB`` */
  196. struct {
  197. bool enabled;
  198. enum ieee802154_fpb_mode mode;
  199. } auto_ack_fpb;
  200. /** ``IEEE802154_CONFIG_ACK_FPB`` */
  201. struct {
  202. uint8_t *addr;
  203. bool extended;
  204. bool enabled;
  205. } ack_fpb;
  206. /** ``IEEE802154_CONFIG_PAN_COORDINATOR`` */
  207. bool pan_coordinator;
  208. /** ``IEEE802154_CONFIG_PROMISCUOUS`` */
  209. bool promiscuous;
  210. /** ``IEEE802154_CONFIG_EVENT_HANDLER`` */
  211. ieee802154_event_cb_t event_handler;
  212. /** ``IEEE802154_CONFIG_MAC_KEYS``
  213. * Pointer to an array containing a list of keys used
  214. * for MAC encryption. Refer to secKeyIdLookupDescriptor and
  215. * secKeyDescriptor in IEEE 802.15.4
  216. *
  217. * key_value field points to a buffer containing the 16 byte
  218. * key. The buffer is copied by the callee.
  219. *
  220. * The variable length array is terminated by key_value field
  221. * set to NULL.
  222. */
  223. struct ieee802154_key *mac_keys;
  224. /** ``IEEE802154_CONFIG_FRAME_COUNTER`` */
  225. uint32_t frame_counter;
  226. /** ``IEEE802154_CONFIG_RX_SLOT`` */
  227. struct {
  228. uint8_t channel;
  229. uint32_t start;
  230. uint32_t duration;
  231. } rx_slot;
  232. /** ``IEEE802154_CONFIG_CSL_PERIOD`` */
  233. uint32_t csl_period;
  234. /** ``IEEE802154_CONFIG_CSL_RX_TIME`` */
  235. uint32_t csl_rx_time;
  236. /** ``IEEE802154_CONFIG_ENH_ACK_HEADER_IE`` */
  237. struct {
  238. const uint8_t *data;
  239. uint16_t data_len;
  240. uint16_t short_addr;
  241. /**
  242. * The extended address is expected to be passed starting
  243. * with the leftmost octet and ending with the rightmost octet.
  244. * A device with an extended address 01:23:45:67:89:ab:cd:ef
  245. * should provide a pointer to array containing values in the
  246. * same exact order.
  247. */
  248. const uint8_t *ext_addr;
  249. } ack_ie;
  250. };
  251. };
  252. /**
  253. * @brief IEEE 802.15.4 radio interface API.
  254. *
  255. */
  256. struct ieee802154_radio_api {
  257. /**
  258. * Mandatory to get in first position.
  259. * A network device should indeed provide a pointer on such
  260. * net_if_api structure. So we make current structure pointer
  261. * that can be casted to a net_if_api structure pointer.
  262. */
  263. struct net_if_api iface_api;
  264. /** Get the device capabilities */
  265. enum ieee802154_hw_caps (*get_capabilities)(const struct device *dev);
  266. /** Clear Channel Assesment - Check channel's activity */
  267. int (*cca)(const struct device *dev);
  268. /** Set current channel */
  269. int (*set_channel)(const struct device *dev, uint16_t channel);
  270. /** Set/Unset filters (for IEEE802154_HW_FILTER ) */
  271. int (*filter)(const struct device *dev,
  272. bool set,
  273. enum ieee802154_filter_type type,
  274. const struct ieee802154_filter *filter);
  275. /** Set TX power level in dbm */
  276. int (*set_txpower)(const struct device *dev, int16_t dbm);
  277. /** Transmit a packet fragment */
  278. int (*tx)(const struct device *dev, enum ieee802154_tx_mode mode,
  279. struct net_pkt *pkt, struct net_buf *frag);
  280. /** Start the device */
  281. int (*start)(const struct device *dev);
  282. /** Stop the device */
  283. int (*stop)(const struct device *dev);
  284. /** Set specific radio driver configuration. */
  285. int (*configure)(const struct device *dev,
  286. enum ieee802154_config_type type,
  287. const struct ieee802154_config *config);
  288. /** Get the available amount of Sub-GHz channels */
  289. uint16_t (*get_subg_channel_count)(const struct device *dev);
  290. /** Run an energy detection scan.
  291. * Note: channel must be set prior to request this function.
  292. * duration parameter is in ms.
  293. */
  294. int (*ed_scan)(const struct device *dev,
  295. uint16_t duration,
  296. energy_scan_done_cb_t done_cb);
  297. /** Get the current radio time in microseconds */
  298. uint64_t (*get_time)(const struct device *dev);
  299. /** Get the current accuracy, in units of ± ppm, of the clock used for
  300. * scheduling delayed receive or transmit radio operations.
  301. * Note: Implementations may optimize this value based on operational
  302. * conditions (i.e.: temperature).
  303. */
  304. uint8_t (*get_sch_acc)(const struct device *dev);
  305. };
  306. /* Make sure that the network interface API is properly setup inside
  307. * IEEE 802154 radio API struct (it is the first one).
  308. */
  309. BUILD_ASSERT(offsetof(struct ieee802154_radio_api, iface_api) == 0);
  310. #define IEEE802154_AR_FLAG_SET (0x20)
  311. /**
  312. * @brief Check if AR flag is set on the frame inside given net_pkt
  313. *
  314. * @param frag A valid pointer on a net_buf structure, must not be NULL,
  315. * and its length should be at least made of 1 byte (ACK frames
  316. * are the smallest frames on 15.4 and made of 3 bytes, not
  317. * not counting the FCS part).
  318. *
  319. * @return True if AR flag is set, False otherwise
  320. */
  321. static inline bool ieee802154_is_ar_flag_set(struct net_buf *frag)
  322. {
  323. return (*frag->data & IEEE802154_AR_FLAG_SET);
  324. }
  325. /**
  326. * @brief Radio driver ACK handling function that hw drivers should use
  327. *
  328. * @details ACK handling requires fast handling and thus such function
  329. * helps to hook directly the hw drivers to the radio driver.
  330. *
  331. * @param iface A valid pointer on a network interface that received the packet
  332. * @param pkt A valid pointer on a packet to check
  333. *
  334. * @return NET_OK if it was handled, NET_CONTINUE otherwise
  335. */
  336. extern enum net_verdict ieee802154_radio_handle_ack(struct net_if *iface,
  337. struct net_pkt *pkt);
  338. /**
  339. * @brief Initialize L2 stack for a given interface
  340. *
  341. * @param iface A valid pointer on a network interface
  342. */
  343. #ifndef CONFIG_IEEE802154_RAW_MODE
  344. void ieee802154_init(struct net_if *iface);
  345. #else
  346. #define ieee802154_init(_iface_)
  347. #endif /* CONFIG_IEEE802154_RAW_MODE */
  348. #ifdef __cplusplus
  349. }
  350. #endif
  351. /**
  352. * @}
  353. */
  354. #endif /* ZEPHYR_INCLUDE_NET_IEEE802154_RADIO_H_ */