lldp.h 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. /** @file
  2. @brief LLDP definitions and handler
  3. This is not to be included by the application.
  4. */
  5. /*
  6. * Copyright (c) 2017 Intel Corporation
  7. *
  8. * SPDX-License-Identifier: Apache-2.0
  9. */
  10. #ifndef ZEPHYR_INCLUDE_NET_LLDP_H_
  11. #define ZEPHYR_INCLUDE_NET_LLDP_H_
  12. /**
  13. * @brief LLDP definitions and helpers
  14. * @defgroup lldp Link Layer Discovery Protocol definitions and helpers
  15. * @ingroup networking
  16. * @{
  17. */
  18. #ifdef __cplusplus
  19. extern "C" {
  20. #endif
  21. /** @cond INTERNAL_HIDDEN */
  22. #define LLDP_TLV_GET_LENGTH(type_length) (type_length & BIT_MASK(9))
  23. #define LLDP_TLV_GET_TYPE(type_length) ((uint8_t)(type_length >> 9))
  24. /* LLDP Definitions */
  25. /* According to the spec, End of LLDPDU TLV value is constant. */
  26. #define NET_LLDP_END_LLDPDU_VALUE 0x0000
  27. /*
  28. * For the Chassis ID TLV Value, if subtype is a MAC address then we must
  29. * use values from CONFIG_NET_LLDP_CHASSIS_ID_MAC0 through
  30. * CONFIG_NET_LLDP_CHASSIS_ID_MAC5. If not, we use CONFIG_NET_LLDP_CHASSIS_ID.
  31. *
  32. * FIXME: implement a similar scheme for subtype 5 (network address).
  33. */
  34. #if defined(CONFIG_NET_LLDP_CHASSIS_ID_SUBTYPE)
  35. #if (CONFIG_NET_LLDP_CHASSIS_ID_SUBTYPE == 4)
  36. #define NET_LLDP_CHASSIS_ID_VALUE \
  37. { \
  38. CONFIG_NET_LLDP_CHASSIS_ID_MAC0, \
  39. CONFIG_NET_LLDP_CHASSIS_ID_MAC1, \
  40. CONFIG_NET_LLDP_CHASSIS_ID_MAC2, \
  41. CONFIG_NET_LLDP_CHASSIS_ID_MAC3, \
  42. CONFIG_NET_LLDP_CHASSIS_ID_MAC4, \
  43. CONFIG_NET_LLDP_CHASSIS_ID_MAC5 \
  44. }
  45. #define NET_LLDP_CHASSIS_ID_VALUE_LEN (6)
  46. #else
  47. #define NET_LLDP_CHASSIS_ID_VALUE CONFIG_NET_LLDP_CHASSIS_ID
  48. #define NET_LLDP_CHASSIS_ID_VALUE_LEN (sizeof(CONFIG_NET_LLDP_CHASSIS_ID) - 1)
  49. #endif
  50. #else
  51. #define NET_LLDP_CHASSIS_ID_VALUE 0
  52. #define NET_LLDP_CHASSIS_ID_VALUE_LEN 0
  53. #endif
  54. /*
  55. * For the Port ID TLV Value, if subtype is a MAC address then we must
  56. * use values from CONFIG_NET_LLDP_PORT_ID_MAC0 through
  57. * CONFIG_NET_LLDP_PORT_ID_MAC5. If not, we use CONFIG_NET_LLDP_PORT_ID.
  58. *
  59. * FIXME: implement a similar scheme for subtype 4 (network address).
  60. */
  61. #if defined(CONFIG_NET_LLDP_PORT_ID_SUBTYPE)
  62. #if (CONFIG_NET_LLDP_PORT_ID_SUBTYPE == 3)
  63. #define NET_LLDP_PORT_ID_VALUE \
  64. { \
  65. CONFIG_NET_LLDP_PORT_ID_MAC0, \
  66. CONFIG_NET_LLDP_PORT_ID_MAC1, \
  67. CONFIG_NET_LLDP_PORT_ID_MAC2, \
  68. CONFIG_NET_LLDP_PORT_ID_MAC3, \
  69. CONFIG_NET_LLDP_PORT_ID_MAC4, \
  70. CONFIG_NET_LLDP_PORT_ID_MAC5 \
  71. }
  72. #define NET_LLDP_PORT_ID_VALUE_LEN (6)
  73. #else
  74. #define NET_LLDP_PORT_ID_VALUE CONFIG_NET_LLDP_PORT_ID
  75. #define NET_LLDP_PORT_ID_VALUE_LEN (sizeof(CONFIG_NET_LLDP_PORT_ID) - 1)
  76. #endif
  77. #else
  78. #define NET_LLDP_PORT_ID_VALUE 0
  79. #define NET_LLDP_PORT_ID_VALUE_LEN 0
  80. #endif
  81. /*
  82. * TLVs Length.
  83. * Note that TLVs that have a subtype must have a byte added to their length.
  84. */
  85. #define NET_LLDP_CHASSIS_ID_TLV_LEN (NET_LLDP_CHASSIS_ID_VALUE_LEN + 1)
  86. #define NET_LLDP_PORT_ID_TLV_LEN (NET_LLDP_PORT_ID_VALUE_LEN + 1)
  87. #define NET_LLDP_TTL_TLV_LEN (2)
  88. /*
  89. * Time to Live value.
  90. * Calculate based on section 9.2.5.22 from LLDP spec.
  91. *
  92. * FIXME: when the network interface is about to be ‘disabled’ TTL shall be set
  93. * to zero so LLDP Rx agents can invalidate the entry related to this node.
  94. */
  95. #if defined(CONFIG_NET_LLDP_TX_INTERVAL) && defined(CONFIG_NET_LLDP_TX_HOLD)
  96. #define NET_LLDP_TTL \
  97. MIN((CONFIG_NET_LLDP_TX_INTERVAL * CONFIG_NET_LLDP_TX_HOLD) + 1, 65535)
  98. #endif
  99. struct net_if;
  100. /** @endcond */
  101. /** TLV Types. Please refer to table 8-1 from IEEE 802.1AB standard. */
  102. enum net_lldp_tlv_type {
  103. LLDP_TLV_END_LLDPDU = 0, /**< End Of LLDPDU (optional) */
  104. LLDP_TLV_CHASSIS_ID = 1, /**< Chassis ID (mandatory) */
  105. LLDP_TLV_PORT_ID = 2, /**< Port ID (mandatory) */
  106. LLDP_TLV_TTL = 3, /**< Time To Live (mandatory) */
  107. LLDP_TLV_PORT_DESC = 4, /**< Port Description (optional) */
  108. LLDP_TLV_SYSTEM_NAME = 5, /**< System Name (optional) */
  109. LLDP_TLV_SYSTEM_DESC = 6, /**< System Description (optional) */
  110. LLDP_TLV_SYSTEM_CAPABILITIES = 7, /**< System Capability (optional) */
  111. LLDP_TLV_MANAGEMENT_ADDR = 8, /**< Management Address (optional) */
  112. /* Types 9 - 126 are reserved. */
  113. LLDP_TLV_ORG_SPECIFIC = 127, /**< Org specific TLVs (optional) */
  114. };
  115. /** Chassis ID TLV, see chapter 8.5.2 in IEEE 802.1AB */
  116. struct net_lldp_chassis_tlv {
  117. /** 7 bits for type, 9 bits for length */
  118. uint16_t type_length;
  119. /** ID subtype */
  120. uint8_t subtype;
  121. /** Chassis ID value */
  122. uint8_t value[NET_LLDP_CHASSIS_ID_VALUE_LEN];
  123. } __packed;
  124. /** Port ID TLV, see chapter 8.5.3 in IEEE 802.1AB */
  125. struct net_lldp_port_tlv {
  126. /** 7 bits for type, 9 bits for length */
  127. uint16_t type_length;
  128. /** ID subtype */
  129. uint8_t subtype;
  130. /** Port ID value */
  131. uint8_t value[NET_LLDP_PORT_ID_VALUE_LEN];
  132. } __packed;
  133. /** Time To Live TLV, see chapter 8.5.4 in IEEE 802.1AB */
  134. struct net_lldp_time_to_live_tlv {
  135. /** 7 bits for type, 9 bits for length */
  136. uint16_t type_length;
  137. /** Time To Live (TTL) value */
  138. uint16_t ttl;
  139. } __packed;
  140. /**
  141. * LLDP Data Unit (LLDPDU) shall contain the following ordered TLVs
  142. * as stated in "8.2 LLDPDU format" from the IEEE 802.1AB
  143. */
  144. struct net_lldpdu {
  145. struct net_lldp_chassis_tlv chassis_id; /**< Mandatory Chassis TLV */
  146. struct net_lldp_port_tlv port_id; /**< Mandatory Port TLV */
  147. struct net_lldp_time_to_live_tlv ttl; /**< Mandatory TTL TLV */
  148. } __packed;
  149. /**
  150. * @brief Set the LLDP data unit for a network interface.
  151. *
  152. * @param iface Network interface
  153. * @param lldpdu LLDP data unit struct
  154. *
  155. * @return 0 if ok, <0 if error
  156. */
  157. int net_lldp_config(struct net_if *iface, const struct net_lldpdu *lldpdu);
  158. /**
  159. * @brief Set the Optional LLDP TLVs for a network interface.
  160. *
  161. * @param iface Network interface
  162. * @param tlv LLDP optional TLVs following mandatory part
  163. * @param len Length of the optional TLVs
  164. *
  165. * @return 0 if ok, <0 if error
  166. */
  167. int net_lldp_config_optional(struct net_if *iface, const uint8_t *tlv,
  168. size_t len);
  169. /**
  170. * @brief Initialize LLDP engine.
  171. */
  172. void net_lldp_init(void);
  173. /**
  174. * @brief LLDP Receive packet callback
  175. *
  176. * Callback gets called upon receiving packet. It is responsible for
  177. * freeing packet or indicating to the stack that it needs to free packet
  178. * by returning correct net_verdict.
  179. *
  180. * Returns:
  181. * - NET_DROP, if packet was invalid, rejected or we want the stack to free it.
  182. * In this case the core stack will free the packet.
  183. * - NET_OK, if the packet was accepted, in this case the ownership of the
  184. * net_pkt goes to callback and core network stack will forget it.
  185. */
  186. typedef enum net_verdict (*net_lldp_recv_cb_t)(struct net_if *iface,
  187. struct net_pkt *pkt);
  188. /**
  189. * @brief Register LLDP Rx callback function
  190. *
  191. * @param iface Network interface
  192. * @param cb Callback function
  193. *
  194. * @return 0 if ok, < 0 if error
  195. */
  196. int net_lldp_register_callback(struct net_if *iface, net_lldp_recv_cb_t cb);
  197. /**
  198. * @brief Parse LLDP packet
  199. *
  200. * @param iface Network interface
  201. * @param pkt Network packet
  202. *
  203. * @return Return the policy for network buffer
  204. */
  205. enum net_verdict net_lldp_recv(struct net_if *iface, struct net_pkt *pkt);
  206. /**
  207. * @brief Set LLDP protocol data unit (LLDPDU) for the network interface.
  208. *
  209. * @param iface Network interface
  210. *
  211. * @return <0 if error, index in lldp array if iface is found there
  212. */
  213. #if defined(CONFIG_NET_LLDP)
  214. int net_lldp_set_lldpdu(struct net_if *iface);
  215. #else
  216. #define net_lldp_set_lldpdu(iface)
  217. #endif
  218. /**
  219. * @brief Unset LLDP protocol data unit (LLDPDU) for the network interface.
  220. *
  221. * @param iface Network interface
  222. */
  223. #if defined(CONFIG_NET_LLDP)
  224. void net_lldp_unset_lldpdu(struct net_if *iface);
  225. #else
  226. #define net_lldp_unset_lldpdu(iface)
  227. #endif
  228. #ifdef __cplusplus
  229. }
  230. #endif
  231. /**
  232. * @}
  233. */
  234. #endif /* ZEPHYR_INCLUDE_NET_LLDP_H_ */