Commit 81d35307 authored by Stephen Hemminger's avatar Stephen Hemminger Committed by David S. Miller

[BRIDGE]: set features based on enslaved devices

Make features of the bridge pseudo-device be a subset of the underlying
devices.  Motivated by Xen and others who use bridging to do failover.

Signed-off-by: Catalin BOIE <catab at umrella.ro>
Signed-off-by: default avatarStephen Hemminger <shemminger@osdl.org>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 81e81575
...@@ -21,10 +21,7 @@ ...@@ -21,10 +21,7 @@
static struct net_device_stats *br_dev_get_stats(struct net_device *dev) static struct net_device_stats *br_dev_get_stats(struct net_device *dev)
{ {
struct net_bridge *br; struct net_bridge *br = netdev_priv(dev);
br = dev->priv;
return &br->statistics; return &br->statistics;
} }
...@@ -54,9 +51,11 @@ int br_dev_xmit(struct sk_buff *skb, struct net_device *dev) ...@@ -54,9 +51,11 @@ int br_dev_xmit(struct sk_buff *skb, struct net_device *dev)
static int br_dev_open(struct net_device *dev) static int br_dev_open(struct net_device *dev)
{ {
netif_start_queue(dev); struct net_bridge *br = netdev_priv(dev);
br_stp_enable_bridge(dev->priv); br_features_recompute(br);
netif_start_queue(dev);
br_stp_enable_bridge(br);
return 0; return 0;
} }
...@@ -67,7 +66,7 @@ static void br_dev_set_multicast_list(struct net_device *dev) ...@@ -67,7 +66,7 @@ static void br_dev_set_multicast_list(struct net_device *dev)
static int br_dev_stop(struct net_device *dev) static int br_dev_stop(struct net_device *dev)
{ {
br_stp_disable_bridge(dev->priv); br_stp_disable_bridge(netdev_priv(dev));
netif_stop_queue(dev); netif_stop_queue(dev);
...@@ -76,7 +75,7 @@ static int br_dev_stop(struct net_device *dev) ...@@ -76,7 +75,7 @@ static int br_dev_stop(struct net_device *dev)
static int br_change_mtu(struct net_device *dev, int new_mtu) static int br_change_mtu(struct net_device *dev, int new_mtu)
{ {
if ((new_mtu < 68) || new_mtu > br_min_mtu(dev->priv)) if (new_mtu < 68 || new_mtu > br_min_mtu(netdev_priv(dev)))
return -EINVAL; return -EINVAL;
dev->mtu = new_mtu; dev->mtu = new_mtu;
......
...@@ -314,6 +314,28 @@ int br_min_mtu(const struct net_bridge *br) ...@@ -314,6 +314,28 @@ int br_min_mtu(const struct net_bridge *br)
return mtu; return mtu;
} }
/*
* Recomputes features using slave's features
*/
void br_features_recompute(struct net_bridge *br)
{
struct net_bridge_port *p;
unsigned long features, checksum;
features = NETIF_F_SG | NETIF_F_FRAGLIST
| NETIF_F_HIGHDMA | NETIF_F_TSO;
checksum = NETIF_F_IP_CSUM; /* least commmon subset */
list_for_each_entry(p, &br->port_list, list) {
if (!(p->dev->features
& (NETIF_F_IP_CSUM|NETIF_F_NO_CSUM|NETIF_F_HW_CSUM)))
checksum = 0;
features &= p->dev->features;
}
br->dev->features = features | checksum | NETIF_F_LLTX;
}
/* called with RTNL */ /* called with RTNL */
int br_add_if(struct net_bridge *br, struct net_device *dev) int br_add_if(struct net_bridge *br, struct net_device *dev)
{ {
...@@ -368,6 +390,7 @@ int br_del_if(struct net_bridge *br, struct net_device *dev) ...@@ -368,6 +390,7 @@ int br_del_if(struct net_bridge *br, struct net_device *dev)
spin_lock_bh(&br->lock); spin_lock_bh(&br->lock);
br_stp_recalculate_bridge_id(br); br_stp_recalculate_bridge_id(br);
br_features_recompute(br);
spin_unlock_bh(&br->lock); spin_unlock_bh(&br->lock);
return 0; return 0;
......
...@@ -65,6 +65,15 @@ static int br_device_event(struct notifier_block *unused, unsigned long event, v ...@@ -65,6 +65,15 @@ static int br_device_event(struct notifier_block *unused, unsigned long event, v
} }
break; break;
case NETDEV_FEAT_CHANGE:
if (br->dev->flags & IFF_UP)
br_features_recompute(br);
/* could do recursive feature change notification
* but who would care??
*/
break;
case NETDEV_DOWN: case NETDEV_DOWN:
if (br->dev->flags & IFF_UP) if (br->dev->flags & IFF_UP)
br_stp_disable_port(p); br_stp_disable_port(p);
......
...@@ -174,6 +174,7 @@ extern int br_add_if(struct net_bridge *br, ...@@ -174,6 +174,7 @@ extern int br_add_if(struct net_bridge *br,
extern int br_del_if(struct net_bridge *br, extern int br_del_if(struct net_bridge *br,
struct net_device *dev); struct net_device *dev);
extern int br_min_mtu(const struct net_bridge *br); extern int br_min_mtu(const struct net_bridge *br);
extern void br_features_recompute(struct net_bridge *br);
/* br_input.c */ /* br_input.c */
extern int br_handle_frame_finish(struct sk_buff *skb); extern int br_handle_frame_finish(struct sk_buff *skb);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment