The sch_mqprio qdisc creates a sub-qdisc per tx queue which are then
called independently for enqueue and dequeue operations. However
statistics are aggregated and pushed up to the "master" qdisc.

This patch adds support for any of the sub-qdiscs to be per cpu
statistic qdiscs. To handle this case add a check when calculating
stats and aggregate the per cpu stats if needed.

Signed-off-by: John Fastabend <john.r.fastab...@intel.com>
---
 net/sched/sch_mqprio.c |   61 +++++++++++++++++++++++++++++++-----------------
 1 file changed, 39 insertions(+), 22 deletions(-)

diff --git a/net/sched/sch_mqprio.c b/net/sched/sch_mqprio.c
index 549c663..24f0360 100644
--- a/net/sched/sch_mqprio.c
+++ b/net/sched/sch_mqprio.c
@@ -229,22 +229,32 @@ static int mqprio_dump(struct Qdisc *sch, struct sk_buff 
*skb)
        unsigned char *b = skb_tail_pointer(skb);
        struct tc_mqprio_qopt opt = { 0 };
        struct Qdisc *qdisc;
-       unsigned int i;
+       unsigned int ntx, tc;
 
        sch->q.qlen = 0;
        memset(&sch->bstats, 0, sizeof(sch->bstats));
        memset(&sch->qstats, 0, sizeof(sch->qstats));
 
-       for (i = 0; i < dev->num_tx_queues; i++) {
-               qdisc = rtnl_dereference(netdev_get_tx_queue(dev, i)->qdisc);
+       for (ntx = 0; ntx < dev->num_tx_queues; ntx++) {
+               struct gnet_stats_basic_cpu __percpu *cpu_bstats = NULL;
+               struct gnet_stats_queue __percpu *cpu_qstats = NULL;
+               __u32 qlen = 0;
+
+               qdisc = netdev_get_tx_queue(dev, ntx)->qdisc_sleeping;
                spin_lock_bh(qdisc_lock(qdisc));
-               sch->q.qlen             += qdisc->q.qlen;
-               sch->bstats.bytes       += qdisc->bstats.bytes;
-               sch->bstats.packets     += qdisc->bstats.packets;
-               sch->qstats.backlog     += qdisc->qstats.backlog;
-               sch->qstats.drops       += qdisc->qstats.drops;
-               sch->qstats.requeues    += qdisc->qstats.requeues;
-               sch->qstats.overlimits  += qdisc->qstats.overlimits;
+
+               if (qdisc_is_percpu_stats(qdisc)) {
+                       cpu_bstats = qdisc->cpu_bstats;
+                       cpu_qstats = qdisc->cpu_qstats;
+               }
+
+               qlen = qdisc_qlen_sum(qdisc);
+
+               __gnet_stats_copy_basic(NULL, &sch->bstats,
+                                       cpu_bstats, &qdisc->bstats);
+               __gnet_stats_copy_queue(&sch->qstats,
+                                       cpu_qstats, &qdisc->qstats, qlen);
+
                spin_unlock_bh(qdisc_lock(qdisc));
        }
 
@@ -252,9 +262,9 @@ static int mqprio_dump(struct Qdisc *sch, struct sk_buff 
*skb)
        memcpy(opt.prio_tc_map, dev->prio_tc_map, sizeof(opt.prio_tc_map));
        opt.hw = priv->hw_owned;
 
-       for (i = 0; i < netdev_get_num_tc(dev); i++) {
-               opt.count[i] = dev->tc_to_txq[i].count;
-               opt.offset[i] = dev->tc_to_txq[i].offset;
+       for (tc = 0; tc < netdev_get_num_tc(dev); tc++) {
+               opt.count[tc] = dev->tc_to_txq[tc].count;
+               opt.offset[tc] = dev->tc_to_txq[tc].offset;
        }
 
        if (nla_put(skb, TCA_OPTIONS, sizeof(opt), &opt))
@@ -332,7 +342,6 @@ static int mqprio_dump_class_stats(struct Qdisc *sch, 
unsigned long cl,
        if (cl <= netdev_get_num_tc(dev)) {
                int i;
                __u32 qlen = 0;
-               struct Qdisc *qdisc;
                struct gnet_stats_queue qstats = {0};
                struct gnet_stats_basic_packed bstats = {0};
                struct netdev_tc_txq tc = dev->tc_to_txq[cl - 1];
@@ -347,18 +356,26 @@ static int mqprio_dump_class_stats(struct Qdisc *sch, 
unsigned long cl,
 
                for (i = tc.offset; i < tc.offset + tc.count; i++) {
                        struct netdev_queue *q = netdev_get_tx_queue(dev, i);
+                       struct Qdisc *qdisc = rtnl_dereference(q->qdisc);
+                       struct gnet_stats_basic_cpu __percpu *cpu_bstats = NULL;
+                       struct gnet_stats_queue __percpu *cpu_qstats = NULL;
 
-                       qdisc = rtnl_dereference(q->qdisc);
                        spin_lock_bh(qdisc_lock(qdisc));
-                       qlen              += qdisc->q.qlen;
-                       bstats.bytes      += qdisc->bstats.bytes;
-                       bstats.packets    += qdisc->bstats.packets;
-                       qstats.backlog    += qdisc->qstats.backlog;
-                       qstats.drops      += qdisc->qstats.drops;
-                       qstats.requeues   += qdisc->qstats.requeues;
-                       qstats.overlimits += qdisc->qstats.overlimits;
+                       if (qdisc_is_percpu_stats(qdisc)) {
+                               cpu_bstats = qdisc->cpu_bstats;
+                               cpu_qstats = qdisc->cpu_qstats;
+                       }
+
+                       qlen = qdisc_qlen_sum(qdisc);
+                       __gnet_stats_copy_basic(NULL, &sch->bstats,
+                                               cpu_bstats, &qdisc->bstats);
+                       __gnet_stats_copy_queue(&sch->qstats,
+                                               cpu_qstats,
+                                               &qdisc->qstats,
+                                               qlen);
                        spin_unlock_bh(qdisc_lock(qdisc));
                }
+
                /* Reclaim root sleeping lock before completing stats */
                if (d->lock)
                        spin_lock_bh(d->lock);

Reply via email to