net: Introduce ndo_get_port_parent_id()
authorFlorian Fainelli <f.fainelli@gmail.com>
Wed, 6 Feb 2019 17:45:35 +0000 (09:45 -0800)
committerDavid S. Miller <davem@davemloft.net>
Wed, 6 Feb 2019 22:16:11 +0000 (14:16 -0800)
In preparation for getting rid of switchdev_ops, create a dedicated NDO
operation for getting the port's parent identifier. There are
essentially two classes of drivers that need to implement getting the
port's parent ID which are VF/PF drivers with a built-in switch, and
pure switchdev drivers such as mlxsw, ocelot, dsa etc.

We introduce a helper function: dev_get_port_parent_id() which supports
recursion into the lower devices to obtain the first port's parent ID.

Convert the bridge, core and ipv4 multicast routing code to check for
such ndo_get_port_parent_id() and call the helper function when valid
before falling back to switchdev_port_attr_get(). This will allow us to
convert all relevant drivers in one go instead of having to implement
both switchdev_port_attr_get() and ndo_get_port_parent_id() operations,
then get rid of switchdev_port_attr_get().

Acked-by: Jiri Pirko <jiri@mellanox.com>
Signed-off-by: Florian Fainelli <f.fainelli@gmail.com>
Reviewed-by: Ido Schimmel <idosch@mellanox.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
include/linux/netdevice.h
net/bridge/br_switchdev.c
net/core/dev.c
net/core/net-sysfs.c
net/core/rtnetlink.c
net/ipv4/ipmr.c

index ba57d0ba425ec894e868b22a8901b292aca62b21..1d95e634f3fe079e4e6fce9b9e30efed6fd522eb 100644 (file)
@@ -1188,6 +1188,10 @@ struct dev_ifalias {
  *     not implement this, it is assumed that the hw is not able to have
  *     multiple net devices on single physical port.
  *
+ * int (*ndo_get_port_parent_id)(struct net_device *dev,
+ *                              struct netdev_phys_item_id *ppid)
+ *     Called to get the parent ID of the physical port of this device.
+ *
  * void (*ndo_udp_tunnel_add)(struct net_device *dev,
  *                           struct udp_tunnel_info *ti);
  *     Called by UDP tunnel to notify a driver about the UDP port and socket
@@ -1412,6 +1416,8 @@ struct net_device_ops {
                                                      bool new_carrier);
        int                     (*ndo_get_phys_port_id)(struct net_device *dev,
                                                        struct netdev_phys_item_id *ppid);
+       int                     (*ndo_get_port_parent_id)(struct net_device *dev,
+                                                         struct netdev_phys_item_id *ppid);
        int                     (*ndo_get_phys_port_name)(struct net_device *dev,
                                                          char *name, size_t len);
        void                    (*ndo_udp_tunnel_add)(struct net_device *dev,
@@ -3651,6 +3657,9 @@ int dev_get_phys_port_id(struct net_device *dev,
                         struct netdev_phys_item_id *ppid);
 int dev_get_phys_port_name(struct net_device *dev,
                           char *name, size_t len);
+int dev_get_port_parent_id(struct net_device *dev,
+                          struct netdev_phys_item_id *ppid, bool recurse);
+bool netdev_port_same_parent_id(struct net_device *a, struct net_device *b);
 int dev_change_proto_down(struct net_device *dev, bool proto_down);
 struct sk_buff *validate_xmit_skb_list(struct sk_buff *skb, struct net_device *dev, bool *again);
 struct sk_buff *dev_hard_start_xmit(struct sk_buff *skb, struct net_device *dev,
index 4d2b9eb7604a7571c44ff5e4aa1af46059fdbcbc..06b0ae44585f39d6ff2e27c610c085138993907e 100644 (file)
@@ -14,7 +14,8 @@ static int br_switchdev_mark_get(struct net_bridge *br, struct net_device *dev)
 
        /* dev is yet to be added to the port list. */
        list_for_each_entry(p, &br->port_list, list) {
-               if (switchdev_port_same_parent_id(dev, p->dev))
+               if (netdev_port_same_parent_id(dev, p->dev) ||
+                   switchdev_port_same_parent_id(dev, p->dev))
                        return p->offload_fwd_mark;
        }
 
@@ -23,6 +24,7 @@ static int br_switchdev_mark_get(struct net_bridge *br, struct net_device *dev)
 
 int nbp_switchdev_mark_set(struct net_bridge_port *p)
 {
+       const struct net_device_ops *ops = p->dev->netdev_ops;
        struct switchdev_attr attr = {
                .orig_dev = p->dev,
                .id = SWITCHDEV_ATTR_ID_PORT_PARENT_ID,
@@ -31,7 +33,10 @@ int nbp_switchdev_mark_set(struct net_bridge_port *p)
 
        ASSERT_RTNL();
 
-       err = switchdev_port_attr_get(p->dev, &attr);
+       if (ops->ndo_get_port_parent_id)
+               err = dev_get_port_parent_id(p->dev, &attr.u.ppid, true);
+       else
+               err = switchdev_port_attr_get(p->dev, &attr);
        if (err) {
                if (err == -EOPNOTSUPP)
                        return 0;
index bfa4be42afff7e2cb2340d098873765e61ef67a5..8c6d5cf8a308448cf30ecbb796f8f777d4f7994c 100644 (file)
@@ -7877,6 +7877,63 @@ int dev_get_phys_port_name(struct net_device *dev,
 }
 EXPORT_SYMBOL(dev_get_phys_port_name);
 
+/**
+ *     dev_get_port_parent_id - Get the device's port parent identifier
+ *     @dev: network device
+ *     @ppid: pointer to a storage for the port's parent identifier
+ *     @recurse: allow/disallow recursion to lower devices
+ *
+ *     Get the devices's port parent identifier
+ */
+int dev_get_port_parent_id(struct net_device *dev,
+                          struct netdev_phys_item_id *ppid,
+                          bool recurse)
+{
+       const struct net_device_ops *ops = dev->netdev_ops;
+       struct netdev_phys_item_id first = { };
+       struct net_device *lower_dev;
+       struct list_head *iter;
+       int err = -EOPNOTSUPP;
+
+       if (ops->ndo_get_port_parent_id)
+               return ops->ndo_get_port_parent_id(dev, ppid);
+
+       if (!recurse)
+               return err;
+
+       netdev_for_each_lower_dev(dev, lower_dev, iter) {
+               err = dev_get_port_parent_id(lower_dev, ppid, recurse);
+               if (err)
+                       break;
+               if (!first.id_len)
+                       first = *ppid;
+               else if (memcmp(&first, ppid, sizeof(*ppid)))
+                       return -ENODATA;
+       }
+
+       return err;
+}
+EXPORT_SYMBOL(dev_get_port_parent_id);
+
+/**
+ *     netdev_port_same_parent_id - Indicate if two network devices have
+ *     the same port parent identifier
+ *     @a: first network device
+ *     @b: second network device
+ */
+bool netdev_port_same_parent_id(struct net_device *a, struct net_device *b)
+{
+       struct netdev_phys_item_id a_id = { };
+       struct netdev_phys_item_id b_id = { };
+
+       if (dev_get_port_parent_id(a, &a_id, true) ||
+           dev_get_port_parent_id(b, &b_id, true))
+               return false;
+
+       return netdev_phys_item_id_same(&a_id, &b_id);
+}
+EXPORT_SYMBOL(netdev_port_same_parent_id);
+
 /**
  *     dev_change_proto_down - update protocol port state information
  *     @dev: device
index ff9fd2bb4ce438119b6fac982aa47c626bd43d7d..4eace9f1dcf92fe6d7a7f01b7af1eb140d44a205 100644 (file)
@@ -495,6 +495,7 @@ static ssize_t phys_switch_id_show(struct device *dev,
                                   struct device_attribute *attr, char *buf)
 {
        struct net_device *netdev = to_net_dev(dev);
+       const struct net_device_ops *ops = netdev->netdev_ops;
        ssize_t ret = -EINVAL;
 
        if (!rtnl_trylock())
@@ -507,7 +508,11 @@ static ssize_t phys_switch_id_show(struct device *dev,
                        .flags = SWITCHDEV_F_NO_RECURSE,
                };
 
-               ret = switchdev_port_attr_get(netdev, &attr);
+               if (ops->ndo_get_port_parent_id)
+                       ret = dev_get_port_parent_id(netdev, &attr.u.ppid,
+                                                    false);
+               else
+                       ret = switchdev_port_attr_get(netdev, &attr);
                if (!ret)
                        ret = sprintf(buf, "%*phN\n", attr.u.ppid.id_len,
                                      attr.u.ppid.id);
index f5a98082ac7a1c6ca47447f490a2b3618fb569e8..90dd02c1f561567ef7636292427e1de97daa241e 100644 (file)
@@ -1146,6 +1146,7 @@ static int rtnl_phys_port_name_fill(struct sk_buff *skb, struct net_device *dev)
 
 static int rtnl_phys_switch_id_fill(struct sk_buff *skb, struct net_device *dev)
 {
+       const struct net_device_ops *ops = dev->netdev_ops;
        int err;
        struct switchdev_attr attr = {
                .orig_dev = dev,
@@ -1153,7 +1154,10 @@ static int rtnl_phys_switch_id_fill(struct sk_buff *skb, struct net_device *dev)
                .flags = SWITCHDEV_F_NO_RECURSE,
        };
 
-       err = switchdev_port_attr_get(dev, &attr);
+       if (ops->ndo_get_port_parent_id)
+               err = dev_get_port_parent_id(dev, &attr.u.ppid, false);
+       else
+               err = switchdev_port_attr_get(dev, &attr);
        if (err) {
                if (err == -EOPNOTSUPP)
                        return 0;
index fb99002c3d4e8523f2aa7926fee41ea8272bdb76..c71bcc42d66d7c78778d472d821413db0feedfb1 100644 (file)
@@ -837,6 +837,7 @@ static void ipmr_update_thresholds(struct mr_table *mrt, struct mr_mfc *cache,
 static int vif_add(struct net *net, struct mr_table *mrt,
                   struct vifctl *vifc, int mrtsock)
 {
+       const struct net_device_ops *ops;
        int vifi = vifc->vifc_vifi;
        struct switchdev_attr attr = {
                .id = SWITCHDEV_ATTR_ID_PORT_PARENT_ID,
@@ -920,7 +921,12 @@ static int vif_add(struct net *net, struct mr_table *mrt,
                        (VIFF_TUNNEL | VIFF_REGISTER));
 
        attr.orig_dev = dev;
-       if (!switchdev_port_attr_get(dev, &attr)) {
+       ops = dev->netdev_ops;
+       if (ops->ndo_get_port_parent_id &&
+           !dev_get_port_parent_id(dev, &attr.u.ppid, true)) {
+               memcpy(v->dev_parent_id.id, attr.u.ppid.id, attr.u.ppid.id_len);
+               v->dev_parent_id.id_len = attr.u.ppid.id_len;
+       } else if (!switchdev_port_attr_get(dev, &attr)) {
                memcpy(v->dev_parent_id.id, attr.u.ppid.id, attr.u.ppid.id_len);
                v->dev_parent_id.id_len = attr.u.ppid.id_len;
        } else {