bridge: allow forwarding some link local frames
authorstephen hemminger <shemminger@vyatta.com>
Mon, 3 Oct 2011 18:14:46 +0000 (18:14 +0000)
committerDavid S. Miller <davem@davemloft.net>
Thu, 6 Oct 2011 19:27:56 +0000 (15:27 -0400)
This is based on an earlier patch by Nick Carter with comments
by David Lamparter but with some refinements. Thanks for their patience
this is a confusing area with overlap of standards, user requirements,
and compatibility with earlier releases.

It adds a new sysfs attribute
   /sys/class/net/brX/bridge/group_fwd_mask
that controls forwarding of frames with address of: 01-80-C2-00-00-0X
The default setting has no forwarding to retain compatibility.

One change from earlier releases is that forwarding of group
addresses is not dependent on STP being enabled or disabled. This
choice was made based on interpretation of tie 802.1 standards.
I expect complaints will arise because of this, but better to follow
the standard than continue acting incorrectly by default.

The filtering mask is writeable, but only values that don't forward
known control frames are allowed. It intentionally blocks attempts
to filter control protocols. For example: writing a 8 allows
forwarding 802.1X PAE addresses which is the most common request.

Reported-by: David Lamparter <equinox@diac24.net>
Original-patch-by: Nick Carter <ncarter100@gmail.com>
Signed-off-by: Stephen Hemminger <shemminger@vyatta.com>
Tested-by: Benjamin Poirier <benjamin.poirier@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
net/bridge/br_device.c
net/bridge/br_input.c
net/bridge/br_private.h
net/bridge/br_sysfs_br.c

index ee68eee79e52570d73ad740e03dad1c8829ac364..28325d15773b5325dbb715b4ff8141c7456855ea 100644 (file)
@@ -361,6 +361,8 @@ void br_dev_setup(struct net_device *dev)
        memcpy(br->group_addr, br_group_address, ETH_ALEN);
 
        br->stp_enabled = BR_NO_STP;
+       br->group_fwd_mask = BR_GROUPFWD_DEFAULT;
+
        br->designated_root = br->bridge_id;
        br->bridge_max_age = br->max_age = 20 * HZ;
        br->bridge_hello_time = br->hello_time = 2 * HZ;
index f06ee39c73fd64dfe958e6f94e735e60e3ba0b78..6f9f8c014725dd74ddfdda99e0173fa3c5d2d16f 100644 (file)
@@ -162,14 +162,37 @@ rx_handler_result_t br_handle_frame(struct sk_buff **pskb)
        p = br_port_get_rcu(skb->dev);
 
        if (unlikely(is_link_local(dest))) {
-               /* Pause frames shouldn't be passed up by driver anyway */
-               if (skb->protocol == htons(ETH_P_PAUSE))
+               /*
+                * See IEEE 802.1D Table 7-10 Reserved addresses
+                *
+                * Assignment                           Value
+                * Bridge Group Address         01-80-C2-00-00-00
+                * (MAC Control) 802.3          01-80-C2-00-00-01
+                * (Link Aggregation) 802.3     01-80-C2-00-00-02
+                * 802.1X PAE address           01-80-C2-00-00-03
+                *
+                * 802.1AB LLDP                 01-80-C2-00-00-0E
+                *
+                * Others reserved for future standardization
+                */
+               switch (dest[5]) {
+               case 0x00:      /* Bridge Group Address */
+                       /* If STP is turned off,
+                          then must forward to keep loop detection */
+                       if (p->br->stp_enabled == BR_NO_STP)
+                               goto forward;
+                       break;
+
+               case 0x01:      /* IEEE MAC (Pause) */
                        goto drop;
 
-               /* If STP is turned off, then forward */
-               if (p->br->stp_enabled == BR_NO_STP && dest[5] == 0)
-                       goto forward;
+               default:
+                       /* Allow selective forwarding for most other protocols */
+                       if (p->br->group_fwd_mask & (1u << dest[5]))
+                               goto forward;
+               }
 
+               /* Deliver packet to local host only */
                if (NF_HOOK(NFPROTO_BRIDGE, NF_BR_LOCAL_IN, skb, skb->dev,
                            NULL, br_handle_local_finish)) {
                        return RX_HANDLER_CONSUMED; /* consumed by filter */
index 78cc364997d9888bc4a1e9869c7c64edb6378ee9..a248fe65b29a669c840a5a7cdf3d89c2a61d34d4 100644 (file)
 
 #define BR_VERSION     "2.3"
 
+/* Control of forwarding link local multicast */
+#define BR_GROUPFWD_DEFAULT    0
+/* Don't allow forwarding control protocols like STP and LLDP */
+#define BR_GROUPFWD_RESTRICTED 0x4007u
+
 /* Path to usermode spanning tree program */
 #define BR_STP_PROG    "/sbin/bridge-stp"
 
@@ -193,6 +198,8 @@ struct net_bridge
        unsigned long                   flags;
 #define BR_SET_MAC_ADDR                0x00000001
 
+       u16                             group_fwd_mask;
+
        /* STP */
        bridge_id                       designated_root;
        bridge_id                       bridge_id;
index 68b893ea8c3ab2d1da10c764e6466da77b74f0a9..c236c0e439848342f2d5c3825b33b6c8bdc6ab47 100644 (file)
@@ -149,6 +149,39 @@ static ssize_t store_stp_state(struct device *d,
 static DEVICE_ATTR(stp_state, S_IRUGO | S_IWUSR, show_stp_state,
                   store_stp_state);
 
+static ssize_t show_group_fwd_mask(struct device *d,
+                             struct device_attribute *attr, char *buf)
+{
+       struct net_bridge *br = to_bridge(d);
+       return sprintf(buf, "%#x\n", br->group_fwd_mask);
+}
+
+
+static ssize_t store_group_fwd_mask(struct device *d,
+                              struct device_attribute *attr, const char *buf,
+                              size_t len)
+{
+       struct net_bridge *br = to_bridge(d);
+       char *endp;
+       unsigned long val;
+
+       if (!capable(CAP_NET_ADMIN))
+               return -EPERM;
+
+       val = simple_strtoul(buf, &endp, 0);
+       if (endp == buf)
+               return -EINVAL;
+
+       if (val & BR_GROUPFWD_RESTRICTED)
+               return -EINVAL;
+
+       br->group_fwd_mask = val;
+
+       return len;
+}
+static DEVICE_ATTR(group_fwd_mask, S_IRUGO | S_IWUSR, show_group_fwd_mask,
+                  store_group_fwd_mask);
+
 static ssize_t show_priority(struct device *d, struct device_attribute *attr,
                             char *buf)
 {
@@ -652,6 +685,7 @@ static struct attribute *bridge_attrs[] = {
        &dev_attr_max_age.attr,
        &dev_attr_ageing_time.attr,
        &dev_attr_stp_state.attr,
+       &dev_attr_group_fwd_mask.attr,
        &dev_attr_priority.attr,
        &dev_attr_bridge_id.attr,
        &dev_attr_root_id.attr,