aboutsummaryrefslogtreecommitdiffstats
path: root/net/bridge/br_private.h
diff options
context:
space:
mode:
authorHerbert Xu <herbert@gondor.apana.org.au>2010-02-27 14:41:45 -0500
committerDavid S. Miller <davem@davemloft.net>2010-02-28 03:48:45 -0500
commiteb1d16414339a6e113d89e2cca2556005d7ce919 (patch)
treeb6c49f870cc36241a894a8c723a38e224d41464b /net/bridge/br_private.h
parent025d89c27f54c69cd0e51666d88aada33666bb6c (diff)
bridge: Add core IGMP snooping support
This patch adds the core functionality of IGMP snooping support without actually hooking it up. So this patch should be a no-op as far as the bridge's external behaviour is concerned. All the new code and data is controlled by the Kconfig option BRIDGE_IGMP_SNOOPING. A run-time toggle is also available. The multicast switching is done using an hash table that is lockless on the read-side through RCU. On the write-side the new multicast_lock is used for all operations. The hash table supports dynamic growth/rehashing. The hash table will be rehashed if any chain length exceeds a preset limit. If rehashing does not reduce the maximum chain length then snooping will be disabled. These features may be added in future (in no particular order): * IGMPv3 source support * Non-querier router detection * IPv6 Signed-off-by: Herbert Xu <herbert@gondor.apana.org.au> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/bridge/br_private.h')
-rw-r--r--net/bridge/br_private.h139
1 files changed, 139 insertions, 0 deletions
diff --git a/net/bridge/br_private.h b/net/bridge/br_private.h
index fad5a2669d3..44345c9afdd 100644
--- a/net/bridge/br_private.h
+++ b/net/bridge/br_private.h
@@ -57,6 +57,41 @@ struct net_bridge_fdb_entry
57 unsigned char is_static; 57 unsigned char is_static;
58}; 58};
59 59
60struct net_bridge_port_group {
61 struct net_bridge_port *port;
62 struct net_bridge_port_group *next;
63 struct hlist_node mglist;
64 struct rcu_head rcu;
65 struct timer_list timer;
66 struct timer_list query_timer;
67 __be32 addr;
68 u32 queries_sent;
69};
70
71struct net_bridge_mdb_entry
72{
73 struct hlist_node hlist[2];
74 struct hlist_node mglist;
75 struct net_bridge *br;
76 struct net_bridge_port_group *ports;
77 struct rcu_head rcu;
78 struct timer_list timer;
79 struct timer_list query_timer;
80 __be32 addr;
81 u32 queries_sent;
82};
83
84struct net_bridge_mdb_htable
85{
86 struct hlist_head *mhash;
87 struct rcu_head rcu;
88 struct net_bridge_mdb_htable *old;
89 u32 size;
90 u32 max;
91 u32 secret;
92 u32 ver;
93};
94
60struct net_bridge_port 95struct net_bridge_port
61{ 96{
62 struct net_bridge *br; 97 struct net_bridge *br;
@@ -84,6 +119,15 @@ struct net_bridge_port
84 119
85 unsigned long flags; 120 unsigned long flags;
86#define BR_HAIRPIN_MODE 0x00000001 121#define BR_HAIRPIN_MODE 0x00000001
122
123#ifdef CONFIG_BRIDGE_IGMP_SNOOPING
124 u32 multicast_startup_queries_sent;
125 unsigned char multicast_router;
126 struct timer_list multicast_router_timer;
127 struct timer_list multicast_query_timer;
128 struct hlist_head mglist;
129 struct hlist_node rlist;
130#endif
87}; 131};
88 132
89struct net_bridge 133struct net_bridge
@@ -124,6 +168,35 @@ struct net_bridge
124 unsigned char topology_change; 168 unsigned char topology_change;
125 unsigned char topology_change_detected; 169 unsigned char topology_change_detected;
126 170
171#ifdef CONFIG_BRIDGE_IGMP_SNOOPING
172 unsigned char multicast_router;
173
174 u8 multicast_disabled:1;
175
176 u32 hash_elasticity;
177 u32 hash_max;
178
179 u32 multicast_last_member_count;
180 u32 multicast_startup_queries_sent;
181 u32 multicast_startup_query_count;
182
183 unsigned long multicast_last_member_interval;
184 unsigned long multicast_membership_interval;
185 unsigned long multicast_querier_interval;
186 unsigned long multicast_query_interval;
187 unsigned long multicast_query_response_interval;
188 unsigned long multicast_startup_query_interval;
189
190 spinlock_t multicast_lock;
191 struct net_bridge_mdb_htable *mdb;
192 struct hlist_head router_list;
193 struct hlist_head mglist;
194
195 struct timer_list multicast_router_timer;
196 struct timer_list multicast_querier_timer;
197 struct timer_list multicast_query_timer;
198#endif
199
127 struct timer_list hello_timer; 200 struct timer_list hello_timer;
128 struct timer_list tcn_timer; 201 struct timer_list tcn_timer;
129 struct timer_list topology_change_timer; 202 struct timer_list topology_change_timer;
@@ -133,6 +206,8 @@ struct net_bridge
133 206
134struct br_input_skb_cb { 207struct br_input_skb_cb {
135 struct net_device *brdev; 208 struct net_device *brdev;
209 int igmp;
210 int mrouters_only;
136}; 211};
137 212
138#define BR_INPUT_SKB_CB(__skb) ((struct br_input_skb_cb *)(__skb)->cb) 213#define BR_INPUT_SKB_CB(__skb) ((struct br_input_skb_cb *)(__skb)->cb)
@@ -204,6 +279,70 @@ extern struct sk_buff *br_handle_frame(struct net_bridge_port *p,
204extern int br_dev_ioctl(struct net_device *dev, struct ifreq *rq, int cmd); 279extern int br_dev_ioctl(struct net_device *dev, struct ifreq *rq, int cmd);
205extern int br_ioctl_deviceless_stub(struct net *net, unsigned int cmd, void __user *arg); 280extern int br_ioctl_deviceless_stub(struct net *net, unsigned int cmd, void __user *arg);
206 281
282/* br_multicast.c */
283#ifdef CONFIG_BRIDGE_IGMP_SNOOPING
284extern int br_multicast_rcv(struct net_bridge *br,
285 struct net_bridge_port *port,
286 struct sk_buff *skb);
287extern struct net_bridge_mdb_entry *br_mdb_get(struct net_bridge *br,
288 struct sk_buff *skb);
289extern void br_multicast_add_port(struct net_bridge_port *port);
290extern void br_multicast_del_port(struct net_bridge_port *port);
291extern void br_multicast_enable_port(struct net_bridge_port *port);
292extern void br_multicast_disable_port(struct net_bridge_port *port);
293extern void br_multicast_init(struct net_bridge *br);
294extern void br_multicast_open(struct net_bridge *br);
295extern void br_multicast_stop(struct net_bridge *br);
296#else
297static inline int br_multicast_rcv(struct net_bridge *br,
298 struct net_bridge_port *port,
299 struct sk_buff *skb)
300{
301 return 0;
302}
303
304static inline struct net_bridge_mdb_entry *br_mdb_get(struct net_bridge *br,
305 struct sk_buff *skb)
306{
307 return NULL;
308}
309
310static inline void br_multicast_add_port(struct net_bridge_port *port)
311{
312}
313
314static inline void br_multicast_del_port(struct net_bridge_port *port)
315{
316}
317
318static inline void br_multicast_enable_port(struct net_bridge_port *port)
319{
320}
321
322static inline void br_multicast_disable_port(struct net_bridge_port *port)
323{
324}
325
326static inline void br_multicast_init(struct net_bridge *br)
327{
328}
329
330static inline void br_multicast_open(struct net_bridge *br)
331{
332}
333
334static inline void br_multicast_stop(struct net_bridge *br)
335{
336}
337#endif
338
339static inline bool br_multicast_is_router(struct net_bridge *br)
340{
341 return br->multicast_router == 2 ||
342 (br->multicast_router == 1 &&
343 timer_pending(&br->multicast_router_timer));
344}
345
207/* br_netfilter.c */ 346/* br_netfilter.c */
208#ifdef CONFIG_BRIDGE_NETFILTER 347#ifdef CONFIG_BRIDGE_NETFILTER
209extern int br_netfilter_init(void); 348extern int br_netfilter_init(void);