123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323 |
- /** @file
- * @brief Virtual Network Interface
- */
- /*
- * Copyright (c) 2021 Intel Corporation
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #ifndef ZEPHYR_INCLUDE_NET_VIRTUAL_H_
- #define ZEPHYR_INCLUDE_NET_VIRTUAL_H_
- #include <kernel.h>
- #include <zephyr/types.h>
- #include <stdbool.h>
- #include <sys/atomic.h>
- #include <net/net_ip.h>
- #include <net/net_pkt.h>
- #include <sys/util.h>
- #include <net/net_if.h>
- #ifdef __cplusplus
- extern "C" {
- #endif
- /**
- * @brief Virtual network interface support functions
- * @defgroup virtual Virtual Network Interface Support Functions
- * @ingroup networking
- * @{
- */
- /** Virtual interface capabilities */
- enum virtual_interface_caps {
- /** IPIP tunnel */
- VIRTUAL_INTERFACE_IPIP = BIT(1),
- /** @cond INTERNAL_HIDDEN */
- /* Marker for capabilities - must be at the end of the enum.
- * It is here because the capability list cannot be empty.
- */
- VIRTUAL_INTERFACE_NUM_CAPS
- /** @endcond */
- };
- /** @cond INTERNAL_HIDDEN */
- enum virtual_interface_config_type {
- VIRTUAL_INTERFACE_CONFIG_TYPE_PEER_ADDRESS,
- VIRTUAL_INTERFACE_CONFIG_TYPE_MTU,
- };
- struct virtual_interface_config {
- sa_family_t family;
- union {
- struct in_addr peer4addr;
- struct in6_addr peer6addr;
- int mtu;
- };
- };
- #if defined(CONFIG_NET_L2_VIRTUAL)
- #define VIRTUAL_MAX_NAME_LEN CONFIG_NET_L2_VIRTUAL_MAX_NAME_LEN
- #else
- #define VIRTUAL_MAX_NAME_LEN 0
- #endif
- /** @endcond */
- struct virtual_interface_api {
- /**
- * The net_if_api must be placed in first position in this
- * struct so that we are compatible with network interface API.
- */
- struct net_if_api iface_api;
- /** Get the virtual interface capabilities */
- enum virtual_interface_caps (*get_capabilities)(struct net_if *iface);
- /** Start the device */
- int (*start)(const struct device *dev);
- /** Stop the device */
- int (*stop)(const struct device *dev);
- /** Send a network packet */
- int (*send)(struct net_if *iface, struct net_pkt *pkt);
- /** Receive a network packet */
- enum net_verdict (*recv)(struct net_if *iface, struct net_pkt *pkt);
- /** Check if this received network packet is for this interface.
- * The callback returns NET_CONTINUE if this interface will accept the
- * packet and pass it upper layers, NET_DROP if the packet is to be
- * dropped and NET_OK to pass it to next interface.
- */
- enum net_verdict (*input)(struct net_if *input_iface,
- struct net_if *iface,
- struct net_addr *remote_addr,
- struct net_pkt *pkt);
- /** Pass the attachment information to virtual interface */
- int (*attach)(struct net_if *virtual_iface, struct net_if *iface);
- /** Set specific L2 configuration */
- int (*set_config)(struct net_if *iface,
- enum virtual_interface_config_type type,
- const struct virtual_interface_config *config);
- /** Get specific L2 configuration */
- int (*get_config)(struct net_if *iface,
- enum virtual_interface_config_type type,
- struct virtual_interface_config *config);
- };
- /* Make sure that the network interface API is properly setup inside
- * Virtual API struct (it is the first one).
- */
- BUILD_ASSERT(offsetof(struct virtual_interface_api, iface_api) == 0);
- /** Virtual L2 context that is needed to binding to the real network interface
- */
- struct virtual_interface_context {
- /** @cond INTERNAL_HIDDEN */
- /* Keep track of contexts */
- sys_snode_t node;
- /* My virtual network interface */
- struct net_if *virtual_iface;
- /** @endcond */
- /**
- * Other network interface this virtual network interface is
- * attached to. These values can be chained so virtual network
- * interfaces can run on top of other virtual interfaces.
- */
- struct net_if *iface;
- /**
- * This tells what L2 features does virtual support.
- */
- enum net_l2_flags virtual_l2_flags;
- /** Is this context already initialized */
- bool is_init;
- /** Link address for this network interface */
- struct net_linkaddr_storage lladdr;
- /** User friendly name of this L2 layer. */
- char name[VIRTUAL_MAX_NAME_LEN];
- };
- /**
- * @brief Attach virtual network interface to the given network interface.
- *
- * @param virtual_iface Virtual network interface.
- * @param iface Network interface we are attached to. This can be NULL,
- * if we want to detach.
- *
- * @return 0 if ok, <0 if attaching failed
- */
- int net_virtual_interface_attach(struct net_if *virtual_iface,
- struct net_if *iface);
- /**
- * @brief Return network interface related to this virtual network interface.
- * The returned network interface is below this virtual network interface.
- *
- * @param iface Virtual network interface.
- *
- * @return Network interface related to this virtual interface or
- * NULL if no such interface exists.
- */
- struct net_if *net_virtual_get_iface(struct net_if *iface);
- /**
- * @brief Return the name of the virtual network interface L2.
- *
- * @param iface Virtual network interface.
- * @param buf Buffer to store the name
- * @param len Max buffer length
- *
- * @return Name of the virtual network interface.
- */
- char *net_virtual_get_name(struct net_if *iface, char *buf, size_t len);
- /**
- * @brief Set the name of the virtual network interface L2.
- *
- * @param iface Virtual network interface.
- * @param name Name of the virtual L2 layer.
- */
- void net_virtual_set_name(struct net_if *iface, const char *name);
- /**
- * @brief Set the L2 flags of the virtual network interface.
- *
- * @param iface Virtual network interface.
- * @param flags L2 flags to set.
- *
- * @return Previous flags that were set.
- */
- enum net_l2_flags net_virtual_set_flags(struct net_if *iface,
- enum net_l2_flags flags);
- /**
- * @brief Feed the IP pkt to stack if tunneling is enabled.
- *
- * @param input_iface Network interface receiving the pkt.
- * @param remote_addr IP address of the sender.
- * @param pkt Network packet.
- *
- * @return Verdict what to do with the packet.
- */
- enum net_verdict net_virtual_input(struct net_if *input_iface,
- struct net_addr *remote_addr,
- struct net_pkt *pkt);
- /** @cond INTERNAL_HIDDEN */
- /**
- * @brief Initialize the network interface so that a virtual
- * interface can be attached to it.
- *
- * @param iface Network interface
- */
- #if defined(CONFIG_NET_L2_VIRTUAL)
- void net_virtual_init(struct net_if *iface);
- #else
- static inline void net_virtual_init(struct net_if *iface)
- {
- ARG_UNUSED(iface);
- }
- #endif
- /**
- * @brief Take virtual network interface down. This is called
- * if the underlying interface is going down.
- *
- * @param iface Network interface
- */
- #if defined(CONFIG_NET_L2_VIRTUAL)
- void net_virtual_disable(struct net_if *iface);
- #else
- static inline void net_virtual_disable(struct net_if *iface)
- {
- ARG_UNUSED(iface);
- }
- #endif
- #define VIRTUAL_L2_CTX_TYPE struct virtual_interface_context
- /**
- * @brief Return virtual device hardware capability information.
- *
- * @param iface Network interface
- *
- * @return Hardware capabilities
- */
- static inline enum virtual_interface_caps
- net_virtual_get_iface_capabilities(struct net_if *iface)
- {
- const struct virtual_interface_api *virt =
- (struct virtual_interface_api *)net_if_get_device(iface)->api;
- if (!virt->get_capabilities) {
- return (enum virtual_interface_caps)0;
- }
- return virt->get_capabilities(iface);
- }
- #define Z_NET_VIRTUAL_INTERFACE_INIT(node_id, dev_name, drv_name, \
- init_fn, pm_control_fn, data, cfg, \
- prio, api, mtu) \
- Z_NET_DEVICE_INIT(node_id, dev_name, drv_name, init_fn, \
- pm_control_fn, data, cfg, prio, api, \
- VIRTUAL_L2, NET_L2_GET_CTX_TYPE(VIRTUAL_L2), \
- mtu)
- /** @endcond */
- /**
- * @def NET_VIRTUAL_INTERFACE_INIT
- *
- * @brief Create a virtual network interface. Binding to another interface
- * is done at runtime by calling net_virtual_interface_attach().
- * The attaching is done automatically when setting up tunneling
- * when peer IP address is set in IP tunneling driver.
- *
- * @param dev_name Network device name.
- * @param drv_name The name this instance of the driver exposes to
- * the system.
- * @param init_fn Address to the init function of the driver.
- * @param pm_control_fn Pointer to pm_control function.
- * Can be NULL if not implemented.
- * @param data Pointer to the device's private data.
- * @param cfg The address to the structure containing the
- * configuration information for this instance of the driver.
- * @param prio The initialization level at which configuration occurs.
- * @param api Provides an initial pointer to the API function struct
- * used by the driver. Can be NULL.
- * @param mtu Maximum transfer unit in bytes for this network interface.
- * This is the default value and its value can be tweaked at runtime.
- */
- #define NET_VIRTUAL_INTERFACE_INIT(dev_name, drv_name, init_fn, \
- pm_control_fn, \
- data, cfg, prio, api, mtu) \
- Z_NET_VIRTUAL_INTERFACE_INIT(DT_INVALID_NODE, dev_name, \
- drv_name, init_fn, pm_control_fn, \
- data, cfg, prio, api, mtu)
- /**
- * @}
- */
- #ifdef __cplusplus
- }
- #endif
- #endif /* ZEPHYR_INCLUDE_NET_VIRTUAL_H_ */
|