net: sched: enable per cpu qstats
authorJohn Fastabend <john.fastabend@gmail.com>
Sun, 28 Sep 2014 18:54:24 +0000 (11:54 -0700)
committerDavid S. Miller <davem@davemloft.net>
Tue, 30 Sep 2014 05:02:26 +0000 (01:02 -0400)
After previous patches to simplify qstats the qstats can be
made per cpu with a packed union in Qdisc struct.

Signed-off-by: John Fastabend <john.r.fastabend@intel.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
17 files changed:
include/net/gen_stats.h
include/net/sch_generic.h
net/core/gen_stats.c
net/sched/act_api.c
net/sched/sch_api.c
net/sched/sch_atm.c
net/sched/sch_cbq.c
net/sched/sch_drr.c
net/sched/sch_fq_codel.c
net/sched/sch_hfsc.c
net/sched/sch_htb.c
net/sched/sch_mq.c
net/sched/sch_mqprio.c
net/sched/sch_multiq.c
net/sched/sch_prio.c
net/sched/sch_qfq.c
net/sched/sch_sfq.c

index de9b3dd5750e006a926092917e582c9f57676341..cbafa3768d48d1e1cae3e068e86f9f0a01813c9e 100644 (file)
@@ -41,7 +41,8 @@ int gnet_stats_copy_rate_est(struct gnet_dump *d,
                             const struct gnet_stats_basic_packed *b,
                             struct gnet_stats_rate_est64 *r);
 int gnet_stats_copy_queue(struct gnet_dump *d,
-                         struct gnet_stats_queue *q, __u32 len);
+                         struct gnet_stats_queue __percpu *cpu_q,
+                         struct gnet_stats_queue *q, __u32 qlen);
 int gnet_stats_copy_app(struct gnet_dump *d, void *st, int len);
 
 int gnet_stats_finish_copy(struct gnet_dump *d);
index 23a0f0fc83d884572562c7a08914195c568775cf..f12669819d1a566b0d8d00eb9be6be9dcc872999 100644 (file)
@@ -90,7 +90,10 @@ struct Qdisc {
                struct gnet_stats_basic_cpu __percpu *cpu_bstats;
        } __packed;
        unsigned int            __state;
-       struct gnet_stats_queue qstats;
+       union {
+               struct gnet_stats_queue qstats;
+               struct gnet_stats_queue __percpu *cpu_qstats;
+       } __packed;
        struct rcu_head         rcu_head;
        int                     padded;
        atomic_t                refcnt;
@@ -543,6 +546,13 @@ static inline void qdisc_qstats_drop(struct Qdisc *sch)
        sch->qstats.drops++;
 }
 
+static inline void qdisc_qstats_drop_cpu(struct Qdisc *sch)
+{
+       struct gnet_stats_queue *qstats = this_cpu_ptr(sch->cpu_qstats);
+
+       qstats->drops++;
+}
+
 static inline void qdisc_qstats_overlimit(struct Qdisc *sch)
 {
        sch->qstats.overlimits++;
index ad3ecb6ba835ceffecbe1a89e53df7387a8011df..14681b97a4f381b4eff4efcd9dce3430e57921aa 100644 (file)
@@ -215,33 +215,74 @@ gnet_stats_copy_rate_est(struct gnet_dump *d,
 }
 EXPORT_SYMBOL(gnet_stats_copy_rate_est);
 
+static void
+__gnet_stats_copy_queue_cpu(struct gnet_stats_queue *qstats,
+                           const struct gnet_stats_queue __percpu *q)
+{
+       int i;
+
+       for_each_possible_cpu(i) {
+               const struct gnet_stats_queue *qcpu = per_cpu_ptr(q, i);
+
+               qstats->qlen = 0;
+               qstats->backlog += qcpu->backlog;
+               qstats->drops += qcpu->drops;
+               qstats->requeues += qcpu->requeues;
+               qstats->overlimits += qcpu->overlimits;
+       }
+}
+
+static void __gnet_stats_copy_queue(struct gnet_stats_queue *qstats,
+                                   const struct gnet_stats_queue __percpu *cpu,
+                                   const struct gnet_stats_queue *q,
+                                   __u32 qlen)
+{
+       if (cpu) {
+               __gnet_stats_copy_queue_cpu(qstats, cpu);
+       } else {
+               qstats->qlen = q->qlen;
+               qstats->backlog = q->backlog;
+               qstats->drops = q->drops;
+               qstats->requeues = q->requeues;
+               qstats->overlimits = q->overlimits;
+       }
+
+       qstats->qlen = qlen;
+}
+
 /**
  * gnet_stats_copy_queue - copy queue statistics into statistics TLV
  * @d: dumping handle
+ * @cpu_q: per cpu queue statistics
  * @q: queue statistics
  * @qlen: queue length statistics
  *
  * Appends the queue statistics to the top level TLV created by
- * gnet_stats_start_copy().
+ * gnet_stats_start_copy(). Using per cpu queue statistics if
+ * they are available.
  *
  * Returns 0 on success or -1 with the statistic lock released
  * if the room in the socket buffer was not sufficient.
  */
 int
 gnet_stats_copy_queue(struct gnet_dump *d,
+                     struct gnet_stats_queue __percpu *cpu_q,
                      struct gnet_stats_queue *q, __u32 qlen)
 {
-       q->qlen = qlen;
+       struct gnet_stats_queue qstats = {0};
+
+       __gnet_stats_copy_queue(&qstats, cpu_q, q, qlen);
 
        if (d->compat_tc_stats) {
-               d->tc_stats.drops = q->drops;
-               d->tc_stats.qlen = q->qlen;
-               d->tc_stats.backlog = q->backlog;
-               d->tc_stats.overlimits = q->overlimits;
+               d->tc_stats.drops = qstats.drops;
+               d->tc_stats.qlen = qstats.qlen;
+               d->tc_stats.backlog = qstats.backlog;
+               d->tc_stats.overlimits = qstats.overlimits;
        }
 
        if (d->tail)
-               return gnet_stats_copy(d, TCA_STATS_QUEUE, q, sizeof(*q));
+               return gnet_stats_copy(d, TCA_STATS_QUEUE,
+                                      &qstats, sizeof(qstats));
 
        return 0;
 }
index 2e134093b8eccca95d5540da742021f209ef7847..3d43e4979f27c8dca3083c863549592216f741fe 100644 (file)
@@ -623,7 +623,7 @@ int tcf_action_copy_stats(struct sk_buff *skb, struct tc_action *a,
        if (gnet_stats_copy_basic(&d, NULL, &p->tcfc_bstats) < 0 ||
            gnet_stats_copy_rate_est(&d, &p->tcfc_bstats,
                                     &p->tcfc_rate_est) < 0 ||
-           gnet_stats_copy_queue(&d,
+           gnet_stats_copy_queue(&d, NULL,
                                  &p->tcfc_qstats,
                                  p->tcfc_qstats.qlen) < 0)
                goto errout;
index ca00ea8e84dc1f04e5b65be5656d30a018557030..aa8329508dba380c84fa3665572514f38294e9e9 100644 (file)
@@ -947,6 +947,10 @@ qdisc_create(struct net_device *dev, struct netdev_queue *dev_queue,
                                alloc_percpu(struct gnet_stats_basic_cpu);
                        if (!sch->cpu_bstats)
                                goto err_out4;
+
+                       sch->cpu_qstats = alloc_percpu(struct gnet_stats_queue);
+                       if (!sch->cpu_qstats)
+                               goto err_out4;
                }
 
                if (tca[TCA_STAB]) {
@@ -995,6 +999,7 @@ err_out:
 
 err_out4:
        free_percpu(sch->cpu_bstats);
+       free_percpu(sch->cpu_qstats);
        /*
         * Any broken qdiscs that would require a ops->reset() here?
         * The qdisc was never in action so it shouldn't be necessary.
@@ -1313,6 +1318,7 @@ static int tc_fill_qdisc(struct sk_buff *skb, struct Qdisc *q, u32 clid,
                         u32 portid, u32 seq, u16 flags, int event)
 {
        struct gnet_stats_basic_cpu __percpu *cpu_bstats = NULL;
+       struct gnet_stats_queue __percpu *cpu_qstats = NULL;
        struct tcmsg *tcm;
        struct nlmsghdr  *nlh;
        unsigned char *b = skb_tail_pointer(skb);
@@ -1349,12 +1355,14 @@ static int tc_fill_qdisc(struct sk_buff *skb, struct Qdisc *q, u32 clid,
        if (q->ops->dump_stats && q->ops->dump_stats(q, &d) < 0)
                goto nla_put_failure;
 
-       if (qdisc_is_percpu_stats(q))
+       if (qdisc_is_percpu_stats(q)) {
                cpu_bstats = q->cpu_bstats;
+               cpu_qstats = q->cpu_qstats;
+       }
 
        if (gnet_stats_copy_basic(&d, cpu_bstats, &q->bstats) < 0 ||
            gnet_stats_copy_rate_est(&d, &q->bstats, &q->rate_est) < 0 ||
-           gnet_stats_copy_queue(&d, &q->qstats, qlen) < 0)
+           gnet_stats_copy_queue(&d, cpu_qstats, &q->qstats, qlen) < 0)
                goto nla_put_failure;
 
        if (gnet_stats_finish_copy(&d) < 0)
index c145eb6279cc323768bcafe7537799d0e430403f..e3e2cc5fd0689e6ac5b6226ab7ca0987601ad733 100644 (file)
@@ -638,7 +638,7 @@ atm_tc_dump_class_stats(struct Qdisc *sch, unsigned long arg,
        struct atm_flow_data *flow = (struct atm_flow_data *)arg;
 
        if (gnet_stats_copy_basic(d, NULL, &flow->bstats) < 0 ||
-           gnet_stats_copy_queue(d, &flow->qstats, flow->q->q.qlen) < 0)
+           gnet_stats_copy_queue(d, NULL, &flow->qstats, flow->q->q.qlen) < 0)
                return -1;
 
        return 0;
index c610081ffba53846a3585bc80d898e0506567d88..beeb75f80fdb970139b030fbb9ac268983b617e9 100644 (file)
@@ -1602,7 +1602,7 @@ cbq_dump_class_stats(struct Qdisc *sch, unsigned long arg,
 
        if (gnet_stats_copy_basic(d, NULL, &cl->bstats) < 0 ||
            gnet_stats_copy_rate_est(d, &cl->bstats, &cl->rate_est) < 0 ||
-           gnet_stats_copy_queue(d, &cl->qstats, cl->q->q.qlen) < 0)
+           gnet_stats_copy_queue(d, NULL, &cl->qstats, cl->q->q.qlen) < 0)
                return -1;
 
        return gnet_stats_copy_app(d, &cl->xstats, sizeof(cl->xstats));
index 5835a93905b16cc7684a1e1266e6df71257fd70b..338706092c27d1255a5c16bb2cc4c5564f79e558 100644 (file)
@@ -284,7 +284,7 @@ static int drr_dump_class_stats(struct Qdisc *sch, unsigned long arg,
 
        if (gnet_stats_copy_basic(d, NULL, &cl->bstats) < 0 ||
            gnet_stats_copy_rate_est(d, &cl->bstats, &cl->rate_est) < 0 ||
-           gnet_stats_copy_queue(d, &cl->qdisc->qstats, qlen) < 0)
+           gnet_stats_copy_queue(d, NULL, &cl->qdisc->qstats, qlen) < 0)
                return -1;
 
        return gnet_stats_copy_app(d, &xstats, sizeof(xstats));
index 226d7359753998b26fccdb01fdaffaeed8b940e9..b9ca32ebc1de0922499379dda1735d75abdd26bc 100644 (file)
@@ -550,7 +550,7 @@ static int fq_codel_dump_class_stats(struct Qdisc *sch, unsigned long cl,
                qs.backlog = q->backlogs[idx];
                qs.drops = flow->dropped;
        }
-       if (gnet_stats_copy_queue(d, &qs, 0) < 0)
+       if (gnet_stats_copy_queue(d, NULL, &qs, 0) < 0)
                return -1;
        if (idx < q->flows_cnt)
                return gnet_stats_copy_app(d, &xstats, sizeof(xstats));
index d364acb2ad07ff6a1791aa2dcfb177bb7a1894b0..e6c7416d033219a66932bc8b2208a2046ffdd8eb 100644 (file)
@@ -1378,7 +1378,7 @@ hfsc_dump_class_stats(struct Qdisc *sch, unsigned long arg,
 
        if (gnet_stats_copy_basic(d, NULL, &cl->bstats) < 0 ||
            gnet_stats_copy_rate_est(d, &cl->bstats, &cl->rate_est) < 0 ||
-           gnet_stats_copy_queue(d, &cl->qstats, cl->qdisc->q.qlen) < 0)
+           gnet_stats_copy_queue(d, NULL, &cl->qstats, cl->qdisc->q.qlen) < 0)
                return -1;
 
        return gnet_stats_copy_app(d, &xstats, sizeof(xstats));
index 3a691fd88f9935bf0f9a0e4f009babdfcedbd06f..f1acb0f60dc35b724289f98498b5cf2162583ce9 100644 (file)
@@ -1147,7 +1147,7 @@ htb_dump_class_stats(struct Qdisc *sch, unsigned long arg, struct gnet_dump *d)
 
        if (gnet_stats_copy_basic(d, NULL, &cl->bstats) < 0 ||
            gnet_stats_copy_rate_est(d, NULL, &cl->rate_est) < 0 ||
-           gnet_stats_copy_queue(d, &cl->qstats, qlen) < 0)
+           gnet_stats_copy_queue(d, NULL, &cl->qstats, qlen) < 0)
                return -1;
 
        return gnet_stats_copy_app(d, &cl->xstats, sizeof(cl->xstats));
index 6416a69420620c564d40cfd3db5fa82df9bd3010..f3cbaecd283af4ce5a78ad90d135d008d2b542d7 100644 (file)
@@ -200,7 +200,7 @@ static int mq_dump_class_stats(struct Qdisc *sch, unsigned long cl,
 
        sch = dev_queue->qdisc_sleeping;
        if (gnet_stats_copy_basic(d, NULL, &sch->bstats) < 0 ||
-           gnet_stats_copy_queue(d, &sch->qstats, sch->q.qlen) < 0)
+           gnet_stats_copy_queue(d, NULL, &sch->qstats, sch->q.qlen) < 0)
                return -1;
        return 0;
 }
index 03dbeb5e81816d8f0d224c264a7c035125c3af4f..3811a745452cf402cd498ed47058fdf655d18cbd 100644 (file)
@@ -356,14 +356,15 @@ static int mqprio_dump_class_stats(struct Qdisc *sch, unsigned long cl,
                /* Reclaim root sleeping lock before completing stats */
                spin_lock_bh(d->lock);
                if (gnet_stats_copy_basic(d, NULL, &bstats) < 0 ||
-                   gnet_stats_copy_queue(d, &qstats, qlen) < 0)
+                   gnet_stats_copy_queue(d, NULL, &qstats, qlen) < 0)
                        return -1;
        } else {
                struct netdev_queue *dev_queue = mqprio_queue_get(sch, cl);
 
                sch = dev_queue->qdisc_sleeping;
                if (gnet_stats_copy_basic(d, NULL, &sch->bstats) < 0 ||
-                   gnet_stats_copy_queue(d, &sch->qstats, sch->q.qlen) < 0)
+                   gnet_stats_copy_queue(d, NULL,
+                                         &sch->qstats, sch->q.qlen) < 0)
                        return -1;
        }
        return 0;
index 53357b368bffde6041eff145b87261c346e018f5..42dd218871e09a3d6cce471a41afecde8fdfdb05 100644 (file)
@@ -361,7 +361,7 @@ static int multiq_dump_class_stats(struct Qdisc *sch, unsigned long cl,
 
        cl_q = q->queues[cl - 1];
        if (gnet_stats_copy_basic(d, NULL, &cl_q->bstats) < 0 ||
-           gnet_stats_copy_queue(d, &cl_q->qstats, cl_q->q.qlen) < 0)
+           gnet_stats_copy_queue(d, NULL, &cl_q->qstats, cl_q->q.qlen) < 0)
                return -1;
 
        return 0;
index 4644f55242d2f8aa648f10e48f4b26caa2126255..8e5cd34aaa74dbaddc4b38be4916bb79526e90fb 100644 (file)
@@ -325,7 +325,7 @@ static int prio_dump_class_stats(struct Qdisc *sch, unsigned long cl,
 
        cl_q = q->queues[cl - 1];
        if (gnet_stats_copy_basic(d, NULL, &cl_q->bstats) < 0 ||
-           gnet_stats_copy_queue(d, &cl_q->qstats, cl_q->q.qlen) < 0)
+           gnet_stats_copy_queue(d, NULL, &cl_q->qstats, cl_q->q.qlen) < 0)
                return -1;
 
        return 0;
index 66df9d9e301afa82a17b315b98413c9a22c2da5e..3ec7e88a43cab2df5e012d3c5216bc31b75eafe3 100644 (file)
@@ -670,7 +670,8 @@ static int qfq_dump_class_stats(struct Qdisc *sch, unsigned long arg,
 
        if (gnet_stats_copy_basic(d, NULL, &cl->bstats) < 0 ||
            gnet_stats_copy_rate_est(d, &cl->bstats, &cl->rate_est) < 0 ||
-           gnet_stats_copy_queue(d, &cl->qdisc->qstats, cl->qdisc->q.qlen) < 0)
+           gnet_stats_copy_queue(d, NULL,
+                                 &cl->qdisc->qstats, cl->qdisc->q.qlen) < 0)
                return -1;
 
        return gnet_stats_copy_app(d, &xstats, sizeof(xstats));
index d4afcbc1c6f7d6fe2d7bbbe7fc586695ab3f5e87..b877140beda5573b8d2dbe4a701bf24355ba7229 100644 (file)
@@ -871,7 +871,7 @@ static int sfq_dump_class_stats(struct Qdisc *sch, unsigned long cl,
                qs.qlen = slot->qlen;
                qs.backlog = slot->backlog;
        }
-       if (gnet_stats_copy_queue(d, &qs, qs.qlen) < 0)
+       if (gnet_stats_copy_queue(d, NULL, &qs, qs.qlen) < 0)
                return -1;
        return gnet_stats_copy_app(d, &xstats, sizeof(xstats));
 }