blob: 593e6094ddd40c19d21ece418b038a8de5d54d93 [file] [log] [blame]
David Ahern4e3c8992015-08-13 14:59:00 -06001/*
2 * include/net/net_vrf.h - adds vrf dev structure definitions
3 * Copyright (c) 2015 Cumulus Networks
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
9 */
10
11#ifndef __LINUX_NET_VRF_H
12#define __LINUX_NET_VRF_H
13
14struct net_vrf_dev {
15 struct rcu_head rcu;
16 int ifindex; /* ifindex of master dev */
17 u32 tb_id; /* table id for VRF */
18};
19
20struct slave {
21 struct list_head list;
22 struct net_device *dev;
23};
24
25struct slave_queue {
26 struct list_head all_slaves;
David Ahern4e3c8992015-08-13 14:59:00 -060027};
28
29struct net_vrf {
30 struct slave_queue queue;
31 struct rtable *rth;
32 u32 tb_id;
33};
34
35
36#if IS_ENABLED(CONFIG_NET_VRF)
37/* called with rcu_read_lock() */
38static inline int vrf_master_ifindex_rcu(const struct net_device *dev)
39{
40 struct net_vrf_dev *vrf_ptr;
41 int ifindex = 0;
42
43 if (!dev)
44 return 0;
45
Nikolay Aleksandrov18041e32015-08-18 21:40:16 +030046 if (netif_is_vrf(dev)) {
David Ahern4e3c8992015-08-13 14:59:00 -060047 ifindex = dev->ifindex;
Nikolay Aleksandrov18041e32015-08-18 21:40:16 +030048 } else {
David Ahern4e3c8992015-08-13 14:59:00 -060049 vrf_ptr = rcu_dereference(dev->vrf_ptr);
50 if (vrf_ptr)
51 ifindex = vrf_ptr->ifindex;
52 }
53
54 return ifindex;
55}
56
Nikolay Aleksandrov18041e32015-08-18 21:40:16 +030057static inline int vrf_master_ifindex(const struct net_device *dev)
58{
59 int ifindex;
60
61 rcu_read_lock();
62 ifindex = vrf_master_ifindex_rcu(dev);
63 rcu_read_unlock();
64
65 return ifindex;
66}
67
David Ahern4e3c8992015-08-13 14:59:00 -060068/* called with rcu_read_lock */
David Ahern9b8ff512015-09-01 14:26:35 -060069static inline u32 vrf_dev_table_rcu(const struct net_device *dev)
David Ahern4e3c8992015-08-13 14:59:00 -060070{
David Ahern9b8ff512015-09-01 14:26:35 -060071 u32 tb_id = 0;
David Ahern4e3c8992015-08-13 14:59:00 -060072
73 if (dev) {
74 struct net_vrf_dev *vrf_ptr;
75
76 vrf_ptr = rcu_dereference(dev->vrf_ptr);
77 if (vrf_ptr)
78 tb_id = vrf_ptr->tb_id;
79 }
80 return tb_id;
81}
82
David Ahern9b8ff512015-09-01 14:26:35 -060083static inline u32 vrf_dev_table(const struct net_device *dev)
David Ahern4e3c8992015-08-13 14:59:00 -060084{
David Ahern9b8ff512015-09-01 14:26:35 -060085 u32 tb_id;
David Ahern4e3c8992015-08-13 14:59:00 -060086
87 rcu_read_lock();
88 tb_id = vrf_dev_table_rcu(dev);
89 rcu_read_unlock();
90
91 return tb_id;
92}
93
David Ahern9b8ff512015-09-01 14:26:35 -060094static inline u32 vrf_dev_table_ifindex(struct net *net, int ifindex)
David Aherndc028da2015-08-16 17:13:27 -060095{
96 struct net_device *dev;
David Ahern9b8ff512015-09-01 14:26:35 -060097 u32 tb_id = 0;
David Aherndc028da2015-08-16 17:13:27 -060098
99 if (!ifindex)
100 return 0;
101
102 rcu_read_lock();
103
104 dev = dev_get_by_index_rcu(net, ifindex);
105 if (dev)
106 tb_id = vrf_dev_table_rcu(dev);
107
108 rcu_read_unlock();
109
110 return tb_id;
111}
112
David Ahern4e3c8992015-08-13 14:59:00 -0600113/* called with rtnl */
David Ahern9b8ff512015-09-01 14:26:35 -0600114static inline u32 vrf_dev_table_rtnl(const struct net_device *dev)
David Ahern4e3c8992015-08-13 14:59:00 -0600115{
David Ahern9b8ff512015-09-01 14:26:35 -0600116 u32 tb_id = 0;
David Ahern4e3c8992015-08-13 14:59:00 -0600117
118 if (dev) {
119 struct net_vrf_dev *vrf_ptr;
120
121 vrf_ptr = rtnl_dereference(dev->vrf_ptr);
122 if (vrf_ptr)
123 tb_id = vrf_ptr->tb_id;
124 }
125 return tb_id;
126}
127
128/* caller has already checked netif_is_vrf(dev) */
129static inline struct rtable *vrf_dev_get_rth(const struct net_device *dev)
130{
131 struct rtable *rth = ERR_PTR(-ENETUNREACH);
132 struct net_vrf *vrf = netdev_priv(dev);
133
134 if (vrf) {
135 rth = vrf->rth;
136 atomic_inc(&rth->dst.__refcnt);
137 }
138 return rth;
139}
140
141#else
142static inline int vrf_master_ifindex_rcu(const struct net_device *dev)
143{
144 return 0;
145}
146
Nikolay Aleksandrov18041e32015-08-18 21:40:16 +0300147static inline int vrf_master_ifindex(const struct net_device *dev)
148{
149 return 0;
150}
151
David Ahern9b8ff512015-09-01 14:26:35 -0600152static inline u32 vrf_dev_table_rcu(const struct net_device *dev)
David Ahern4e3c8992015-08-13 14:59:00 -0600153{
154 return 0;
155}
156
David Ahern9b8ff512015-09-01 14:26:35 -0600157static inline u32 vrf_dev_table(const struct net_device *dev)
David Ahern4e3c8992015-08-13 14:59:00 -0600158{
159 return 0;
160}
161
David Ahern9b8ff512015-09-01 14:26:35 -0600162static inline u32 vrf_dev_table_ifindex(struct net *net, int ifindex)
David Aherndc028da2015-08-16 17:13:27 -0600163{
164 return 0;
165}
166
David Ahern9b8ff512015-09-01 14:26:35 -0600167static inline u32 vrf_dev_table_rtnl(const struct net_device *dev)
David Ahern4e3c8992015-08-13 14:59:00 -0600168{
169 return 0;
170}
171
172static inline struct rtable *vrf_dev_get_rth(const struct net_device *dev)
173{
174 return ERR_PTR(-ENETUNREACH);
175}
176#endif
177
178#endif /* __LINUX_NET_VRF_H */