net: dsa: add support for BRIDGE_MROUTER attribute
authorVivien Didelot <vivien.didelot@gmail.com>
Tue, 9 Jul 2019 03:31:13 +0000 (23:31 -0400)
committerDavid S. Miller <davem@davemloft.net>
Tue, 9 Jul 2019 21:49:34 +0000 (14:49 -0700)
This patch adds support for enabling or disabling the flooding of
unknown multicast traffic on the CPU ports, depending on the value
of the switchdev SWITCHDEV_ATTR_ID_BRIDGE_MROUTER attribute.

The current behavior is kept unchanged but a user can now prevent
the CPU conduit to be flooded with a lot of unregistered traffic that
the network stack needs to filter in software with e.g.:

    echo 0 > /sys/class/net/br0/multicast_router

Signed-off-by: Vivien Didelot <vivien.didelot@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
net/dsa/dsa_priv.h
net/dsa/port.c
net/dsa/slave.c

index b2be53a13aa0584db06bcc25e10a0cc69805309c..12f8c7ee4dd8aaef3e602a9bcc969f8351720807 100644 (file)
@@ -150,6 +150,8 @@ int dsa_port_pre_bridge_flags(const struct dsa_port *dp, unsigned long flags,
                              struct switchdev_trans *trans);
 int dsa_port_bridge_flags(const struct dsa_port *dp, unsigned long flags,
                          struct switchdev_trans *trans);
+int dsa_port_mrouter(struct dsa_port *dp, bool mrouter,
+                    struct switchdev_trans *trans);
 int dsa_port_vlan_add(struct dsa_port *dp,
                      const struct switchdev_obj_port_vlan *vlan,
                      struct switchdev_trans *trans);
index d2b65e8dc60cb2258ac56c41e7b56acb34dce4d3..f071acf2842bdb8b9b29eb521d00bf61b5dc8ba2 100644 (file)
@@ -261,6 +261,18 @@ int dsa_port_bridge_flags(const struct dsa_port *dp, unsigned long flags,
        return err;
 }
 
+int dsa_port_mrouter(struct dsa_port *dp, bool mrouter,
+                    struct switchdev_trans *trans)
+{
+       struct dsa_switch *ds = dp->ds;
+       int port = dp->index;
+
+       if (switchdev_trans_ph_prepare(trans))
+               return ds->ops->port_egress_floods ? 0 : -EOPNOTSUPP;
+
+       return ds->ops->port_egress_floods(ds, port, true, mrouter);
+}
+
 int dsa_port_fdb_add(struct dsa_port *dp, const unsigned char *addr,
                     u16 vid)
 {
index 9bcb598fc84037d737e84621ccedc8f3819cbd4d..614c38ece104307a2c54dc76a8014402061ae240 100644 (file)
@@ -301,6 +301,9 @@ static int dsa_slave_port_attr_set(struct net_device *dev,
        case SWITCHDEV_ATTR_ID_PORT_BRIDGE_FLAGS:
                ret = dsa_port_bridge_flags(dp, attr->u.brport_flags, trans);
                break;
+       case SWITCHDEV_ATTR_ID_BRIDGE_MROUTER:
+               ret = dsa_port_mrouter(dp->cpu_dp, attr->u.mrouter, trans);
+               break;
        default:
                ret = -EOPNOTSUPP;
                break;