diff options
Diffstat (limited to 'net/bridge/br_forward.c')
| -rw-r--r-- | net/bridge/br_forward.c | 39 | 
1 files changed, 29 insertions, 10 deletions
diff --git a/net/bridge/br_forward.c b/net/bridge/br_forward.c index 92fb3293a21..a98ef139309 100644 --- a/net/bridge/br_forward.c +++ b/net/bridge/br_forward.c @@ -15,6 +15,7 @@  #include <linux/slab.h>  #include <linux/kernel.h>  #include <linux/netdevice.h> +#include <linux/netpoll.h>  #include <linux/skbuff.h>  #include <linux/if_vlan.h>  #include <linux/netfilter_bridge.h> @@ -50,7 +51,13 @@ int br_dev_queue_push_xmit(struct sk_buff *skb)  		else {  			skb_push(skb, ETH_HLEN); -			dev_queue_xmit(skb); +#ifdef CONFIG_NET_POLL_CONTROLLER +			if (unlikely(skb->dev->priv_flags & IFF_IN_NETPOLL)) { +				netpoll_send_skb(skb->dev->npinfo->netpoll, skb); +				skb->dev->priv_flags &= ~IFF_IN_NETPOLL; +			} else +#endif +				dev_queue_xmit(skb);  		}  	} @@ -66,9 +73,23 @@ int br_forward_finish(struct sk_buff *skb)  static void __br_deliver(const struct net_bridge_port *to, struct sk_buff *skb)  { +#ifdef CONFIG_NET_POLL_CONTROLLER +	struct net_bridge *br = to->br; +	if (unlikely(br->dev->priv_flags & IFF_IN_NETPOLL)) { +		struct netpoll *np; +		to->dev->npinfo = skb->dev->npinfo; +		np = skb->dev->npinfo->netpoll; +		np->real_dev = np->dev = to->dev; +		to->dev->priv_flags |= IFF_IN_NETPOLL; +	} +#endif  	skb->dev = to->dev;  	NF_HOOK(NFPROTO_BRIDGE, NF_BR_LOCAL_OUT, skb, NULL, skb->dev,  		br_forward_finish); +#ifdef CONFIG_NET_POLL_CONTROLLER +	if (skb->dev->npinfo) +		skb->dev->npinfo->netpoll->dev = br->dev; +#endif  }  static void __br_forward(const struct net_bridge_port *to, struct sk_buff *skb) @@ -208,17 +229,15 @@ static void br_multicast_flood(struct net_bridge_mdb_entry *mdst,  {  	struct net_device *dev = BR_INPUT_SKB_CB(skb)->brdev;  	struct net_bridge *br = netdev_priv(dev); -	struct net_bridge_port *port; -	struct net_bridge_port *lport, *rport; -	struct net_bridge_port *prev; +	struct net_bridge_port *prev = NULL;  	struct net_bridge_port_group *p;  	struct hlist_node *rp; -	prev = NULL; - -	rp = br->router_list.first; -	p = mdst ? mdst->ports : NULL; +	rp = rcu_dereference(br->router_list.first); +	p = mdst ? rcu_dereference(mdst->ports) : NULL;  	while (p || rp) { +		struct net_bridge_port *port, *lport, *rport; +  		lport = p ? p->port : NULL;  		rport = rp ? hlist_entry(rp, struct net_bridge_port, rlist) :  			     NULL; @@ -231,9 +250,9 @@ static void br_multicast_flood(struct net_bridge_mdb_entry *mdst,  			goto out;  		if ((unsigned long)lport >= (unsigned long)port) -			p = p->next; +			p = rcu_dereference(p->next);  		if ((unsigned long)rport >= (unsigned long)port) -			rp = rp->next; +			rp = rcu_dereference(rp->next);  	}  	if (!prev)  |