virtual.h 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. /** @file
  2. * @brief Virtual Network Interface
  3. */
  4. /*
  5. * Copyright (c) 2021 Intel Corporation
  6. *
  7. * SPDX-License-Identifier: Apache-2.0
  8. */
  9. #ifndef ZEPHYR_INCLUDE_NET_VIRTUAL_H_
  10. #define ZEPHYR_INCLUDE_NET_VIRTUAL_H_
  11. #include <kernel.h>
  12. #include <zephyr/types.h>
  13. #include <stdbool.h>
  14. #include <sys/atomic.h>
  15. #include <net/net_ip.h>
  16. #include <net/net_pkt.h>
  17. #include <sys/util.h>
  18. #include <net/net_if.h>
  19. #ifdef __cplusplus
  20. extern "C" {
  21. #endif
  22. /**
  23. * @brief Virtual network interface support functions
  24. * @defgroup virtual Virtual Network Interface Support Functions
  25. * @ingroup networking
  26. * @{
  27. */
  28. /** Virtual interface capabilities */
  29. enum virtual_interface_caps {
  30. /** IPIP tunnel */
  31. VIRTUAL_INTERFACE_IPIP = BIT(1),
  32. /** @cond INTERNAL_HIDDEN */
  33. /* Marker for capabilities - must be at the end of the enum.
  34. * It is here because the capability list cannot be empty.
  35. */
  36. VIRTUAL_INTERFACE_NUM_CAPS
  37. /** @endcond */
  38. };
  39. /** @cond INTERNAL_HIDDEN */
  40. enum virtual_interface_config_type {
  41. VIRTUAL_INTERFACE_CONFIG_TYPE_PEER_ADDRESS,
  42. VIRTUAL_INTERFACE_CONFIG_TYPE_MTU,
  43. };
  44. struct virtual_interface_config {
  45. sa_family_t family;
  46. union {
  47. struct in_addr peer4addr;
  48. struct in6_addr peer6addr;
  49. int mtu;
  50. };
  51. };
  52. #if defined(CONFIG_NET_L2_VIRTUAL)
  53. #define VIRTUAL_MAX_NAME_LEN CONFIG_NET_L2_VIRTUAL_MAX_NAME_LEN
  54. #else
  55. #define VIRTUAL_MAX_NAME_LEN 0
  56. #endif
  57. /** @endcond */
  58. struct virtual_interface_api {
  59. /**
  60. * The net_if_api must be placed in first position in this
  61. * struct so that we are compatible with network interface API.
  62. */
  63. struct net_if_api iface_api;
  64. /** Get the virtual interface capabilities */
  65. enum virtual_interface_caps (*get_capabilities)(struct net_if *iface);
  66. /** Start the device */
  67. int (*start)(const struct device *dev);
  68. /** Stop the device */
  69. int (*stop)(const struct device *dev);
  70. /** Send a network packet */
  71. int (*send)(struct net_if *iface, struct net_pkt *pkt);
  72. /** Receive a network packet */
  73. enum net_verdict (*recv)(struct net_if *iface, struct net_pkt *pkt);
  74. /** Check if this received network packet is for this interface.
  75. * The callback returns NET_CONTINUE if this interface will accept the
  76. * packet and pass it upper layers, NET_DROP if the packet is to be
  77. * dropped and NET_OK to pass it to next interface.
  78. */
  79. enum net_verdict (*input)(struct net_if *input_iface,
  80. struct net_if *iface,
  81. struct net_addr *remote_addr,
  82. struct net_pkt *pkt);
  83. /** Pass the attachment information to virtual interface */
  84. int (*attach)(struct net_if *virtual_iface, struct net_if *iface);
  85. /** Set specific L2 configuration */
  86. int (*set_config)(struct net_if *iface,
  87. enum virtual_interface_config_type type,
  88. const struct virtual_interface_config *config);
  89. /** Get specific L2 configuration */
  90. int (*get_config)(struct net_if *iface,
  91. enum virtual_interface_config_type type,
  92. struct virtual_interface_config *config);
  93. };
  94. /* Make sure that the network interface API is properly setup inside
  95. * Virtual API struct (it is the first one).
  96. */
  97. BUILD_ASSERT(offsetof(struct virtual_interface_api, iface_api) == 0);
  98. /** Virtual L2 context that is needed to binding to the real network interface
  99. */
  100. struct virtual_interface_context {
  101. /** @cond INTERNAL_HIDDEN */
  102. /* Keep track of contexts */
  103. sys_snode_t node;
  104. /* My virtual network interface */
  105. struct net_if *virtual_iface;
  106. /** @endcond */
  107. /**
  108. * Other network interface this virtual network interface is
  109. * attached to. These values can be chained so virtual network
  110. * interfaces can run on top of other virtual interfaces.
  111. */
  112. struct net_if *iface;
  113. /**
  114. * This tells what L2 features does virtual support.
  115. */
  116. enum net_l2_flags virtual_l2_flags;
  117. /** Is this context already initialized */
  118. bool is_init;
  119. /** Link address for this network interface */
  120. struct net_linkaddr_storage lladdr;
  121. /** User friendly name of this L2 layer. */
  122. char name[VIRTUAL_MAX_NAME_LEN];
  123. };
  124. /**
  125. * @brief Attach virtual network interface to the given network interface.
  126. *
  127. * @param virtual_iface Virtual network interface.
  128. * @param iface Network interface we are attached to. This can be NULL,
  129. * if we want to detach.
  130. *
  131. * @return 0 if ok, <0 if attaching failed
  132. */
  133. int net_virtual_interface_attach(struct net_if *virtual_iface,
  134. struct net_if *iface);
  135. /**
  136. * @brief Return network interface related to this virtual network interface.
  137. * The returned network interface is below this virtual network interface.
  138. *
  139. * @param iface Virtual network interface.
  140. *
  141. * @return Network interface related to this virtual interface or
  142. * NULL if no such interface exists.
  143. */
  144. struct net_if *net_virtual_get_iface(struct net_if *iface);
  145. /**
  146. * @brief Return the name of the virtual network interface L2.
  147. *
  148. * @param iface Virtual network interface.
  149. * @param buf Buffer to store the name
  150. * @param len Max buffer length
  151. *
  152. * @return Name of the virtual network interface.
  153. */
  154. char *net_virtual_get_name(struct net_if *iface, char *buf, size_t len);
  155. /**
  156. * @brief Set the name of the virtual network interface L2.
  157. *
  158. * @param iface Virtual network interface.
  159. * @param name Name of the virtual L2 layer.
  160. */
  161. void net_virtual_set_name(struct net_if *iface, const char *name);
  162. /**
  163. * @brief Set the L2 flags of the virtual network interface.
  164. *
  165. * @param iface Virtual network interface.
  166. * @param flags L2 flags to set.
  167. *
  168. * @return Previous flags that were set.
  169. */
  170. enum net_l2_flags net_virtual_set_flags(struct net_if *iface,
  171. enum net_l2_flags flags);
  172. /**
  173. * @brief Feed the IP pkt to stack if tunneling is enabled.
  174. *
  175. * @param input_iface Network interface receiving the pkt.
  176. * @param remote_addr IP address of the sender.
  177. * @param pkt Network packet.
  178. *
  179. * @return Verdict what to do with the packet.
  180. */
  181. enum net_verdict net_virtual_input(struct net_if *input_iface,
  182. struct net_addr *remote_addr,
  183. struct net_pkt *pkt);
  184. /** @cond INTERNAL_HIDDEN */
  185. /**
  186. * @brief Initialize the network interface so that a virtual
  187. * interface can be attached to it.
  188. *
  189. * @param iface Network interface
  190. */
  191. #if defined(CONFIG_NET_L2_VIRTUAL)
  192. void net_virtual_init(struct net_if *iface);
  193. #else
  194. static inline void net_virtual_init(struct net_if *iface)
  195. {
  196. ARG_UNUSED(iface);
  197. }
  198. #endif
  199. /**
  200. * @brief Take virtual network interface down. This is called
  201. * if the underlying interface is going down.
  202. *
  203. * @param iface Network interface
  204. */
  205. #if defined(CONFIG_NET_L2_VIRTUAL)
  206. void net_virtual_disable(struct net_if *iface);
  207. #else
  208. static inline void net_virtual_disable(struct net_if *iface)
  209. {
  210. ARG_UNUSED(iface);
  211. }
  212. #endif
  213. #define VIRTUAL_L2_CTX_TYPE struct virtual_interface_context
  214. /**
  215. * @brief Return virtual device hardware capability information.
  216. *
  217. * @param iface Network interface
  218. *
  219. * @return Hardware capabilities
  220. */
  221. static inline enum virtual_interface_caps
  222. net_virtual_get_iface_capabilities(struct net_if *iface)
  223. {
  224. const struct virtual_interface_api *virt =
  225. (struct virtual_interface_api *)net_if_get_device(iface)->api;
  226. if (!virt->get_capabilities) {
  227. return (enum virtual_interface_caps)0;
  228. }
  229. return virt->get_capabilities(iface);
  230. }
  231. #define Z_NET_VIRTUAL_INTERFACE_INIT(node_id, dev_name, drv_name, \
  232. init_fn, pm_control_fn, data, cfg, \
  233. prio, api, mtu) \
  234. Z_NET_DEVICE_INIT(node_id, dev_name, drv_name, init_fn, \
  235. pm_control_fn, data, cfg, prio, api, \
  236. VIRTUAL_L2, NET_L2_GET_CTX_TYPE(VIRTUAL_L2), \
  237. mtu)
  238. /** @endcond */
  239. /**
  240. * @def NET_VIRTUAL_INTERFACE_INIT
  241. *
  242. * @brief Create a virtual network interface. Binding to another interface
  243. * is done at runtime by calling net_virtual_interface_attach().
  244. * The attaching is done automatically when setting up tunneling
  245. * when peer IP address is set in IP tunneling driver.
  246. *
  247. * @param dev_name Network device name.
  248. * @param drv_name The name this instance of the driver exposes to
  249. * the system.
  250. * @param init_fn Address to the init function of the driver.
  251. * @param pm_control_fn Pointer to pm_control function.
  252. * Can be NULL if not implemented.
  253. * @param data Pointer to the device's private data.
  254. * @param cfg The address to the structure containing the
  255. * configuration information for this instance of the driver.
  256. * @param prio The initialization level at which configuration occurs.
  257. * @param api Provides an initial pointer to the API function struct
  258. * used by the driver. Can be NULL.
  259. * @param mtu Maximum transfer unit in bytes for this network interface.
  260. * This is the default value and its value can be tweaked at runtime.
  261. */
  262. #define NET_VIRTUAL_INTERFACE_INIT(dev_name, drv_name, init_fn, \
  263. pm_control_fn, \
  264. data, cfg, prio, api, mtu) \
  265. Z_NET_VIRTUAL_INTERFACE_INIT(DT_INVALID_NODE, dev_name, \
  266. drv_name, init_fn, pm_control_fn, \
  267. data, cfg, prio, api, mtu)
  268. /**
  269. * @}
  270. */
  271. #ifdef __cplusplus
  272. }
  273. #endif
  274. #endif /* ZEPHYR_INCLUDE_NET_VIRTUAL_H_ */