Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/bin/ps Calculate CPU usage (pcpu) once per process if it is ...



details:   https://anonhg.NetBSD.org/src/rev/ffb1cbef9d2f
branches:  trunk
changeset: 349192:ffb1cbef9d2f
user:      rin <rin%NetBSD.org@localhost>
date:      Mon Nov 28 08:21:10 2016 +0000

description:
Calculate CPU usage (pcpu) once per process if it is required. This change
significantly improves performance for slow machines when output is sorted
by pcpu.

ok martin

diffstat:

 bin/ps/extern.h |   4 +-
 bin/ps/print.c  |  25 ++---------------
 bin/ps/ps.c     |  82 +++++++++++++++++++++++++++++++++++++++++++++-----------
 bin/ps/ps.h     |  11 ++++++-
 4 files changed, 79 insertions(+), 43 deletions(-)

diffs (truncated from 323 to 300 lines):

diff -r d838d09c0599 -r ffb1cbef9d2f bin/ps/extern.h
--- a/bin/ps/extern.h   Mon Nov 28 08:19:23 2016 +0000
+++ b/bin/ps/extern.h   Mon Nov 28 08:21:10 2016 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: extern.h,v 1.36 2016/11/28 08:19:23 rin Exp $  */
+/*     $NetBSD: extern.h,v 1.37 2016/11/28 08:21:10 rin Exp $  */
 
 /*-
  * Copyright (c) 1991, 1993, 1994
@@ -37,7 +37,7 @@
  */
 
 extern double log_ccpu;
-extern int eval, fscale, mempages, nlistread, rawcpu, maxslp, uspace;
+extern int eval, fscale, mempages, nlistread, maxslp, uspace;
 extern int sumrusage, termwidth, totwidth;
 extern int needenv, needcomm, commandonly;
 extern uid_t myuid;
diff -r d838d09c0599 -r ffb1cbef9d2f bin/ps/print.c
--- a/bin/ps/print.c    Mon Nov 28 08:19:23 2016 +0000
+++ b/bin/ps/print.c    Mon Nov 28 08:21:10 2016 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: print.c,v 1.124 2016/11/28 08:19:23 rin Exp $  */
+/*     $NetBSD: print.c,v 1.125 2016/11/28 08:21:10 rin Exp $  */
 
 /*
  * Copyright (c) 2000, 2007 The NetBSD Foundation, Inc.
@@ -63,7 +63,7 @@
 #if 0
 static char sccsid[] = "@(#)print.c    8.6 (Berkeley) 4/16/94";
 #else
-__RCSID("$NetBSD: print.c,v 1.124 2016/11/28 08:19:23 rin Exp $");
+__RCSID("$NetBSD: print.c,v 1.125 2016/11/28 08:21:10 rin Exp $");
 #endif
 #endif /* not lint */
 
@@ -1091,33 +1091,14 @@
        cputime1(secs, psecs, v, mode);
 }
 
-double
-getpcpu(const struct kinfo_proc2 *k)
-{
-
-       if (!nlistread)
-               donlist();
-
-#define        fxtofl(fixpt)   ((double)(fixpt) / fscale)
-
-       if (k->p_swtime == 0 || k->p_realstat == SZOMB)
-               return (0.0);
-       if (rawcpu)
-               return (100.0 * fxtofl(k->p_pctcpu));
-       return (100.0 * fxtofl(k->p_pctcpu) /
-               (1.0 - exp(k->p_swtime * log_ccpu)));
-}
-
 void
 pcpu(void *arg, VARENT *ve, enum mode mode)
 {
-       struct kinfo_proc2 *k;
        VAR *v;
        double dbl;
 
-       k = arg;
        v = ve->var;
-       dbl = getpcpu(k);
+       dbl = ((struct pinfo *)arg)->pcpu;
        doubleprintorsetwidth(v, dbl, (dbl >= 99.95) ? 0 : 1, mode);
 }
 
diff -r d838d09c0599 -r ffb1cbef9d2f bin/ps/ps.c
--- a/bin/ps/ps.c       Mon Nov 28 08:19:23 2016 +0000
+++ b/bin/ps/ps.c       Mon Nov 28 08:21:10 2016 +0000
@@ -1,4 +1,4 @@
-/*     $NetBSD: ps.c,v 1.85 2016/11/28 08:18:27 rin Exp $      */
+/*     $NetBSD: ps.c,v 1.86 2016/11/28 08:21:10 rin Exp $      */
 
 /*
  * Copyright (c) 2000-2008 The NetBSD Foundation, Inc.
@@ -68,7 +68,7 @@
 #if 0
 static char sccsid[] = "@(#)ps.c       8.4 (Berkeley) 4/2/94";
 #else
-__RCSID("$NetBSD: ps.c,v 1.85 2016/11/28 08:18:27 rin Exp $");
+__RCSID("$NetBSD: ps.c,v 1.86 2016/11/28 08:21:10 rin Exp $");
 #endif
 #endif /* not lint */
 
@@ -89,6 +89,7 @@
 #include <kvm.h>
 #include <limits.h>
 #include <locale.h>
+#include <math.h>
 #include <nlist.h>
 #include <paths.h>
 #include <pwd.h>
@@ -106,12 +107,10 @@
 #define        GETOPTSTR       "aAcCeghjk:LlM:mN:O:o:p:rSsTt:U:uvW:wx"
 #define        ARGOPTS         "kMNOopUW"
 
-struct kinfo_proc2 *kinfo;
 struct varlist displaylist = SIMPLEQ_HEAD_INITIALIZER(displaylist);
 struct varlist sortlist = SIMPLEQ_HEAD_INITIALIZER(sortlist);
 
 int    eval;                   /* exit value */
-int    rawcpu;                 /* -C */
 int    sumrusage;              /* -S */
 int    termwidth;              /* width of screen (0 == infinity) */
 int    totwidth;               /* calculated width of requested variables */
@@ -124,6 +123,8 @@
                    struct kinfo_lwp *, int);
 static struct kinfo_proc2
                *getkinfo_kvm(kvm_t *, int, int, int *);
+static struct pinfo
+               *setpinfo(struct kinfo_proc2 *, int, int, int);
 static char    *kludge_oldps_options(char *);
 static int      pscomp(const void *, const void *);
 static void     scanvars(void);
@@ -197,12 +198,14 @@
 int
 main(int argc, char *argv[])
 {
+       struct kinfo_proc2 *kinfo;
+       struct pinfo *pinfo;
        struct varent *vent;
        struct winsize ws;
        struct kinfo_lwp *kl, *l;
        int ch, i, j, fmt, lineno, nentries, nlwps;
        long long flag;
-       int prtheader, wflag, what, xflg, showlwps;
+       int calc_pcpu, prtheader, wflag, what, xflg, rawcpu, showlwps;
        char *nlistf, *memf, *swapf, errbuf[_POSIX2_LINE_MAX];
        char *ttname;
 
@@ -394,11 +397,20 @@
 
        /* Add default sort criteria */
        parsesort("tdev,pid");
+       calc_pcpu = 0;
        SIMPLEQ_FOREACH(vent, &sortlist, next) {
                if (vent->var->flag & LWP || vent->var->type == UNSPECIFIED)
                        warnx("Cannot sort on %s, sort key ignored",
                                vent->var->name);
+               if (vent->var->type == PCPU)
+                       calc_pcpu = 1;
        }
+       if (!calc_pcpu)
+               SIMPLEQ_FOREACH(vent, &displaylist, next)
+                       if (vent->var->type == PCPU) {
+                               calc_pcpu = 1;
+                               break;
+                       }
 
        /*
         * scan requested variables, noting what structures are needed.
@@ -414,10 +426,12 @@
                printheader();
                return 1;
        }
+       pinfo = setpinfo(kinfo, nentries, calc_pcpu, rawcpu);
+
        /*
         * sort proc list
         */
-       qsort(kinfo, nentries, sizeof(struct kinfo_proc2), pscomp);
+       qsort(pinfo, nentries, sizeof(struct pinfo), pscomp);
        /*
         * For each proc, call each variable output function in
         * "setwidth" mode to determine the widest element of
@@ -425,7 +439,8 @@
         */
 
        for (i = 0; i < nentries; i++) {
-               struct kinfo_proc2 *ki = &kinfo[i];
+               struct pinfo *pi = &pinfo[i];
+               struct kinfo_proc2 *ki = pi->ki;
 
                if (xflg == 0 && (ki->p_tdev == (uint32_t)NODEV ||
                    (ki->p_flag & P_CONTROLT) == 0))
@@ -438,7 +453,7 @@
                if (showlwps == 0) {
                        l = pick_representative_lwp(ki, kl, nlwps);
                        SIMPLEQ_FOREACH(vent, &displaylist, next)
-                               OUTPUT(vent, ki, l, WIDTHMODE);
+                               OUTPUT(vent, l, pi, ki, WIDTHMODE);
                } else {
                        /* The printing is done with the loops
                         * reversed, but here we don't need that,
@@ -446,7 +461,7 @@
                         */
                        SIMPLEQ_FOREACH(vent, &displaylist, next)
                                for (j = 0; j < nlwps; j++)
-                                       OUTPUT(vent, ki, &kl[j], WIDTHMODE);
+                                       OUTPUT(vent, &kl[j], pi, ki, WIDTHMODE);
                }
        }
        /*
@@ -460,7 +475,8 @@
         * print mode.
         */
        for (i = lineno = 0; i < nentries; i++) {
-               struct kinfo_proc2 *ki = &kinfo[i];
+               struct pinfo *pi = &pinfo[i];
+               struct kinfo_proc2 *ki = pi->ki;
 
                if (xflg == 0 && (ki->p_tdev == (uint32_t)NODEV ||
                    (ki->p_flag & P_CONTROLT ) == 0))
@@ -472,7 +488,7 @@
                if (showlwps == 0) {
                        l = pick_representative_lwp(ki, kl, nlwps);
                        SIMPLEQ_FOREACH(vent, &displaylist, next) {
-                               OUTPUT(vent, ki, l, PRINTMODE);
+                               OUTPUT(vent, l, pi, ki, PRINTMODE);
                                if (SIMPLEQ_NEXT(vent, next) != NULL)
                                        (void)putchar(' ');
                        }
@@ -485,7 +501,7 @@
                } else {
                        for (j = 0; j < nlwps; j++) {
                                SIMPLEQ_FOREACH(vent, &displaylist, next) {
-                                       OUTPUT(vent, ki, &kl[j], PRINTMODE);
+                                       OUTPUT(vent, &kl[j], pi, ki, PRINTMODE);
                                        if (SIMPLEQ_NEXT(vent, next) != NULL)
                                                (void)putchar(' ');
                                }
@@ -567,6 +583,36 @@
            nentriesp));
 }
 
+static struct pinfo *
+setpinfo(struct kinfo_proc2 *ki, int nentries, int calc_pcpu, int rawcpu)
+{
+       struct pinfo *pi;
+       int i;
+
+       pi = malloc(nentries * sizeof(struct pinfo));
+       if (pi == NULL)
+               err(1, "malloc");
+
+       if (calc_pcpu && !nlistread)
+               donlist();
+
+       for (i = 0; i < nentries; i++) {
+               pi[i].ki = &ki[i];
+               if (!calc_pcpu)
+                       continue;
+               if (ki[i].p_realstat == SZOMB ||
+                   (!rawcpu && ki[i].p_swtime == 0)) {
+                       pi[i].pcpu = 0.0;
+                       continue;
+               }
+               pi[i].pcpu = 100.0 * (double)ki[i].p_pctcpu / fscale;
+               if (!rawcpu)
+                       pi[i].pcpu /= 1.0 - exp(ki[i].p_swtime * log_ccpu);
+       }
+
+       return pi;
+}
+
 static void
 scanvars(void)
 {
@@ -585,8 +631,10 @@
 static int
 pscomp(const void *a, const void *b)
 {
-       const struct kinfo_proc2 *ka = (const struct kinfo_proc2 *)a;
-       const struct kinfo_proc2 *kb = (const struct kinfo_proc2 *)b;
+       const struct pinfo *pa = (const struct pinfo *)a;
+       const struct pinfo *pb = (const struct pinfo *)b;
+       const struct kinfo_proc2 *ka = pa->ki;
+       const struct kinfo_proc2 *kb = pb->ki;
 
        int i;
        int64_t i64;
@@ -631,8 +679,8 @@
                case UINT32:
                        RDIFF(uint32_t);
                case SIGLIST:
-                       sa = (const void *)((const char *)a + v->off);
-                       sb = (const void *)((const char *)b + v->off);
+                       sa = (const void *)((const char *)ka + v->off);
+                       sb = (const void *)((const char *)kb + v->off);
                        i = 0;
                        do {
                                if (sa->__bits[i] > sb->__bits[i])
@@ -666,7 +714,7 @@
                                return i64 > 0 ? 1 : -1;
                        continue;
                case PCPU:
-                       i = getpcpu(kb) - getpcpu(ka);
+                       i = pb->pcpu - pa->pcpu;
                        if (i != 0)
                                return i;
                        continue;
diff -r d838d09c0599 -r ffb1cbef9d2f bin/ps/ps.h
--- a/bin/ps/ps.h       Mon Nov 28 08:19:23 2016 +0000
+++ b/bin/ps/ps.h       Mon Nov 28 08:21:10 2016 +0000
@@ -1,4 +1,4 @@



Home | Main Index | Thread Index | Old Index