/* * include/net/net_vrf.h - adds vrf dev structure definitions * Copyright (c) 2015 Cumulus Networks * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. */ #ifndef __LINUX_NET_VRF_H #define __LINUX_NET_VRF_H struct net_vrf_dev { struct rcu_head rcu; int ifindex; /* ifindex of master dev */ u32 tb_id; /* table id for VRF */ }; struct slave { struct list_head list; struct net_device *dev; }; struct slave_queue { struct list_head all_slaves; }; struct net_vrf { struct slave_queue queue; struct rtable *rth; u32 tb_id; }; #if IS_ENABLED(CONFIG_NET_VRF) /* called with rcu_read_lock() */ static inline int vrf_master_ifindex_rcu(const struct net_device *dev) { struct net_vrf_dev *vrf_ptr; int ifindex = 0; if (!dev) return 0; if (netif_is_vrf(dev)) { ifindex = dev->ifindex; } else { vrf_ptr = rcu_dereference(dev->vrf_ptr); if (vrf_ptr) ifindex = vrf_ptr->ifindex; } return ifindex; } static inline int vrf_master_ifindex(const struct net_device *dev) { int ifindex; rcu_read_lock(); ifindex = vrf_master_ifindex_rcu(dev); rcu_read_unlock(); return ifindex; } /* called with rcu_read_lock */ static inline u32 vrf_dev_table_rcu(const struct net_device *dev) { u32 tb_id = 0; if (dev) { struct net_vrf_dev *vrf_ptr; vrf_ptr = rcu_dereference(dev->vrf_ptr); if (vrf_ptr) tb_id = vrf_ptr->tb_id; } return tb_id; } static inline u32 vrf_dev_table(const struct net_device *dev) { u32 tb_id; rcu_read_lock(); tb_id = vrf_dev_table_rcu(dev); rcu_read_unlock(); return tb_id; } static inline u32 vrf_dev_table_ifindex(struct net *net, int ifindex) { struct net_device *dev; u32 tb_id = 0; if (!ifindex) return 0; rcu_read_lock(); dev = dev_get_by_index_rcu(net, ifindex); if (dev) tb_id = vrf_dev_table_rcu(dev); rcu_read_unlock(); return tb_id; } /* called with rtnl */ static inline u32 vrf_dev_table_rtnl(const struct net_device *dev) { u32 tb_id = 0; if (dev) { struct net_vrf_dev *vrf_ptr; vrf_ptr = rtnl_dereference(dev->vrf_ptr); if (vrf_ptr) tb_id = vrf_ptr->tb_id; } return tb_id; } /* caller has already checked netif_is_vrf(dev) */ static inline struct rtable *vrf_dev_get_rth(const struct net_device *dev) { struct rtable *rth = ERR_PTR(-ENETUNREACH); struct net_vrf *vrf = netdev_priv(dev); if (vrf) { rth = vrf->rth; atomic_inc(&rth->dst.__refcnt); } return rth; } #else static inline int vrf_master_ifindex_rcu(const struct net_device *dev) { return 0; } static inline int vrf_master_ifindex(const struct net_device *dev) { return 0; } static inline u32 vrf_dev_table_rcu(const struct net_device *dev) { return 0; } static inline u32 vrf_dev_table(const struct net_device *dev) { return 0; } static inline u32 vrf_dev_table_ifindex(struct net *net, int ifindex) { return 0; } static inline u32 vrf_dev_table_rtnl(const struct net_device *dev) { return 0; } static inline struct rtable *vrf_dev_get_rth(const struct net_device *dev) { return ERR_PTR(-ENETUNREACH); } #endif #endif /* __LINUX_NET_VRF_H */