net/dcb: Add dcbnl buffer attribute
authorHuy Nguyen <huyn@mellanox.com>
Thu, 22 Feb 2018 17:57:10 +0000 (11:57 -0600)
committerSaeed Mahameed <saeedm@mellanox.com>
Thu, 24 May 2018 21:22:59 +0000 (14:22 -0700)
In this patch, we add dcbnl buffer attribute to allow user
change the NIC's buffer configuration such as priority
to buffer mapping and buffer size of individual buffer.

This attribute combined with pfc attribute allows advanced user to
fine tune the qos setting for specific priority queue. For example,
user can give dedicated buffer for one or more priorities or user
can give large buffer to certain priorities.

The dcb buffer configuration will be controlled by lldptool.
lldptool -T -i eth2 -V BUFFER prio 0,2,5,7,1,2,3,6
  maps priorities 0,1,2,3,4,5,6,7 to receive buffer 0,2,5,7,1,2,3,6
lldptool -T -i eth2 -V BUFFER size 87296,87296,0,87296,0,0,0,0
  sets receive buffer size for buffer 0,1,2,3,4,5,6,7 respectively

After discussion on mailing list with Jakub, Jiri, Ido and John, we agreed to
choose dcbnl over devlink interface since this feature is intended to set
port attributes which are governed by the netdev instance of that port, where
devlink API is more suitable for global ASIC configurations.

We present an use case scenario where dcbnl buffer attribute configured
by advance user helps reduce the latency of messages of different sizes.

Scenarios description:
On ConnectX-5, we run latency sensitive traffic with
small/medium message sizes ranging from 64B to 256KB and bandwidth sensitive
traffic with large messages sizes 512KB and 1MB. We group small, medium,
and large message sizes to their own pfc enables priorities as follow.
  Priorities 1 & 2 (64B, 256B and 1KB)
  Priorities 3 & 4 (4KB, 8KB, 16KB, 64KB, 128KB and 256KB)
  Priorities 5 & 6 (512KB and 1MB)

By default, ConnectX-5 maps all pfc enabled priorities to a single
lossless fixed buffer size of 50% of total available buffer space. The
other 50% is assigned to lossy buffer. Using dcbnl buffer attribute,
we create three equal size lossless buffers. Each buffer has 25% of total
available buffer space. Thus, the lossy buffer size reduces to 25%. Priority
to lossless  buffer mappings are set as follow.
  Priorities 1 & 2 on lossless buffer #1
  Priorities 3 & 4 on lossless buffer #2
  Priorities 5 & 6 on lossless buffer #3

We observe improvements in latency for small and medium message sizes
as follows. Please note that the large message sizes bandwidth performance is
reduced but the total bandwidth remains the same.
  256B message size (42 % latency reduction)
  4K message size (21% latency reduction)
  64K message size (16% latency reduction)

CC: Ido Schimmel <idosch@idosch.org>
CC: Jakub Kicinski <jakub.kicinski@netronome.com>
CC: Jiri Pirko <jiri@resnulli.us>
CC: Or Gerlitz <gerlitz.or@gmail.com>
CC: Parav Pandit <parav@mellanox.com>
CC: Aron Silverton <aron.silverton@oracle.com>
Signed-off-by: Huy Nguyen <huyn@mellanox.com>
Reviewed-by: Parav Pandit <parav@mellanox.com>
Signed-off-by: Saeed Mahameed <saeedm@mellanox.com>
include/net/dcbnl.h
include/uapi/linux/dcbnl.h
net/dcb/dcbnl.c

index 207d9ba1f92c8ae1789cff6fdc99ad15f256488b..0e5e91be2d30cf59e3e856e4e3dca2960971a39d 100644 (file)
@@ -101,6 +101,10 @@ struct dcbnl_rtnl_ops {
        /* CEE peer */
        int (*cee_peer_getpg) (struct net_device *, struct cee_pg *);
        int (*cee_peer_getpfc) (struct net_device *, struct cee_pfc *);
+
+       /* buffer settings */
+       int (*dcbnl_getbuffer)(struct net_device *, struct dcbnl_buffer *);
+       int (*dcbnl_setbuffer)(struct net_device *, struct dcbnl_buffer *);
 };
 
 #endif /* __NET_DCBNL_H__ */
index 2c0c6453c3f40542cd8d4e4f5538e994b8e71b64..60aa2e4466981ab63027b5092c81d4e4e2010d77 100644 (file)
@@ -163,6 +163,16 @@ struct ieee_pfc {
        __u64   indications[IEEE_8021QAZ_MAX_TCS];
 };
 
+#define IEEE_8021Q_MAX_PRIORITIES 8
+#define DCBX_MAX_BUFFERS  8
+struct dcbnl_buffer {
+       /* priority to buffer mapping */
+       __u8    prio2buffer[IEEE_8021Q_MAX_PRIORITIES];
+       /* buffer size in Bytes */
+       __u32   buffer_size[DCBX_MAX_BUFFERS];
+       __u32   total_size;
+};
+
 /* CEE DCBX std supported values */
 #define CEE_DCBX_MAX_PGS       8
 #define CEE_DCBX_MAX_PRIO      8
@@ -406,6 +416,7 @@ enum ieee_attrs {
        DCB_ATTR_IEEE_MAXRATE,
        DCB_ATTR_IEEE_QCN,
        DCB_ATTR_IEEE_QCN_STATS,
+       DCB_ATTR_DCB_BUFFER,
        __DCB_ATTR_IEEE_MAX
 };
 #define DCB_ATTR_IEEE_MAX (__DCB_ATTR_IEEE_MAX - 1)
index bae7d78aa06895dd9237abc6f1bc2ae3db23f38b..d2f4e0c1faafa985f67bda48b55ea1bf409be05f 100644 (file)
@@ -176,6 +176,7 @@ static const struct nla_policy dcbnl_ieee_policy[DCB_ATTR_IEEE_MAX + 1] = {
        [DCB_ATTR_IEEE_MAXRATE]   = {.len = sizeof(struct ieee_maxrate)},
        [DCB_ATTR_IEEE_QCN]         = {.len = sizeof(struct ieee_qcn)},
        [DCB_ATTR_IEEE_QCN_STATS]   = {.len = sizeof(struct ieee_qcn_stats)},
+       [DCB_ATTR_DCB_BUFFER]       = {.len = sizeof(struct dcbnl_buffer)},
 };
 
 /* DCB number of traffic classes nested attributes. */
@@ -1094,6 +1095,16 @@ static int dcbnl_ieee_fill(struct sk_buff *skb, struct net_device *netdev)
                        return -EMSGSIZE;
        }
 
+       if (ops->dcbnl_getbuffer) {
+               struct dcbnl_buffer buffer;
+
+               memset(&buffer, 0, sizeof(buffer));
+               err = ops->dcbnl_getbuffer(netdev, &buffer);
+               if (!err &&
+                   nla_put(skb, DCB_ATTR_DCB_BUFFER, sizeof(buffer), &buffer))
+                       return -EMSGSIZE;
+       }
+
        app = nla_nest_start(skb, DCB_ATTR_IEEE_APP_TABLE);
        if (!app)
                return -EMSGSIZE;
@@ -1453,6 +1464,15 @@ static int dcbnl_ieee_set(struct net_device *netdev, struct nlmsghdr *nlh,
                        goto err;
        }
 
+       if (ieee[DCB_ATTR_DCB_BUFFER] && ops->dcbnl_setbuffer) {
+               struct dcbnl_buffer *buffer =
+                       nla_data(ieee[DCB_ATTR_DCB_BUFFER]);
+
+               err = ops->dcbnl_setbuffer(netdev, buffer);
+               if (err)
+                       goto err;
+       }
+
        if (ieee[DCB_ATTR_IEEE_APP_TABLE]) {
                struct nlattr *attr;
                int rem;