diff options
Diffstat (limited to 'net/bridge/br_forward.c')
| -rw-r--r-- | net/bridge/br_forward.c | 57 | 
1 files changed, 31 insertions, 26 deletions
diff --git a/net/bridge/br_forward.c b/net/bridge/br_forward.c index 2bd11ec6d16..056b67b0e27 100644 --- a/net/bridge/br_forward.c +++ b/net/bridge/br_forward.c @@ -26,32 +26,25 @@ static int deliver_clone(const struct net_bridge_port *prev,  			 void (*__packet_hook)(const struct net_bridge_port *p,  					       struct sk_buff *skb)); -/* Don't forward packets to originating port or forwarding diasabled */ +/* Don't forward packets to originating port or forwarding disabled */  static inline int should_deliver(const struct net_bridge_port *p,  				 const struct sk_buff *skb)  { -	return (((p->flags & BR_HAIRPIN_MODE) || skb->dev != p->dev) && -		p->state == BR_STATE_FORWARDING); -} - -static inline unsigned packet_length(const struct sk_buff *skb) -{ -	return skb->len - (skb->protocol == htons(ETH_P_8021Q) ? VLAN_HLEN : 0); +	return ((p->flags & BR_HAIRPIN_MODE) || skb->dev != p->dev) && +		br_allowed_egress(p->br, nbp_get_vlan_info(p), skb) && +		p->state == BR_STATE_FORWARDING;  }  int br_dev_queue_push_xmit(struct sk_buff *skb)  { -	/* drop mtu oversized packets except gso */ -	if (packet_length(skb) > skb->dev->mtu && !skb_is_gso(skb)) +	/* ip_fragment doesn't copy the MAC header */ +	if (nf_bridge_maybe_copy_header(skb) || +	    !is_skb_forwardable(skb->dev, skb)) {  		kfree_skb(skb); -	else { -		/* ip_fragment doesn't copy the MAC header */ -		if (nf_bridge_maybe_copy_header(skb)) -			kfree_skb(skb); -		else { -			skb_push(skb, ETH_HLEN); -			dev_queue_xmit(skb); -		} +	} else { +		skb_push(skb, ETH_HLEN); +		br_drop_fake_rtable(skb); +		dev_queue_xmit(skb);  	}  	return 0; @@ -66,10 +59,14 @@ int br_forward_finish(struct sk_buff *skb)  static void __br_deliver(const struct net_bridge_port *to, struct sk_buff *skb)  { +	skb = br_handle_vlan(to->br, nbp_get_vlan_info(to), skb); +	if (!skb) +		return; +  	skb->dev = to->dev; -	if (unlikely(netpoll_tx_running(to->dev))) { -		if (packet_length(skb) > skb->dev->mtu && !skb_is_gso(skb)) +	if (unlikely(netpoll_tx_running(to->br->dev))) { +		if (!is_skb_forwardable(skb->dev, skb))  			kfree_skb(skb);  		else {  			skb_push(skb, ETH_HLEN); @@ -91,6 +88,10 @@ static void __br_forward(const struct net_bridge_port *to, struct sk_buff *skb)  		return;  	} +	skb = br_handle_vlan(to->br, nbp_get_vlan_info(to), skb); +	if (!skb) +		return; +  	indev = skb->dev;  	skb->dev = to->dev;  	skb_forward_csum(skb); @@ -102,7 +103,7 @@ static void __br_forward(const struct net_bridge_port *to, struct sk_buff *skb)  /* called with rcu_read_lock */  void br_deliver(const struct net_bridge_port *to, struct sk_buff *skb)  { -	if (should_deliver(to, skb)) { +	if (to && should_deliver(to, skb)) {  		__br_deliver(to, skb);  		return;  	} @@ -168,7 +169,8 @@ out:  static void br_flood(struct net_bridge *br, struct sk_buff *skb,  		     struct sk_buff *skb0,  		     void (*__packet_hook)(const struct net_bridge_port *p, -					   struct sk_buff *skb)) +					   struct sk_buff *skb), +		     bool unicast)  {  	struct net_bridge_port *p;  	struct net_bridge_port *prev; @@ -176,6 +178,9 @@ static void br_flood(struct net_bridge *br, struct sk_buff *skb,  	prev = NULL;  	list_for_each_entry_rcu(p, &br->port_list, list) { +		/* Do not flood unicast traffic to ports that turn it off */ +		if (unicast && !(p->flags & BR_FLOOD)) +			continue;  		prev = maybe_deliver(prev, p, skb, __packet_hook);  		if (IS_ERR(prev))  			goto out; @@ -197,16 +202,16 @@ out:  /* called with rcu_read_lock */ -void br_flood_deliver(struct net_bridge *br, struct sk_buff *skb) +void br_flood_deliver(struct net_bridge *br, struct sk_buff *skb, bool unicast)  { -	br_flood(br, skb, NULL, __br_deliver); +	br_flood(br, skb, NULL, __br_deliver, unicast);  }  /* called under bridge lock */  void br_flood_forward(struct net_bridge *br, struct sk_buff *skb, -		      struct sk_buff *skb2) +		      struct sk_buff *skb2, bool unicast)  { -	br_flood(br, skb, skb2, __br_forward); +	br_flood(br, skb, skb2, __br_forward, unicast);  }  #ifdef CONFIG_BRIDGE_IGMP_SNOOPING  | 
