File: [local] / src / bin / ps / print.c (download)
Revision 1.32, Thu Jan 8 18:18:35 2004 UTC (20 years, 4 months ago) by millert
Branch: MAIN
CVS Tags: OPENBSD_3_6_BASE, OPENBSD_3_6, OPENBSD_3_5_BASE, OPENBSD_3_5 Changes since 1.31: +116 -190 lines
Convert to kinfo_proc2. The sort order for -u and -m will be different
due to a bug fix in the sort routine.
|
/* $OpenBSD: print.c,v 1.32 2004/01/08 18:18:35 millert Exp $ */
/* $NetBSD: print.c,v 1.27 1995/09/29 21:58:12 cgd Exp $ */
/*-
* Copyright (c) 1990, 1993, 1994
* The Regents of the University of California. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of the University nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
#ifndef lint
#if 0
static char sccsid[] = "@(#)print.c 8.6 (Berkeley) 4/16/94";
#else
static char rcsid[] = "$OpenBSD: print.c,v 1.32 2004/01/08 18:18:35 millert Exp $";
#endif
#endif /* not lint */
#include <sys/param.h>
#include <sys/time.h>
#include <sys/resource.h>
#include <sys/proc.h>
#include <sys/stat.h>
#include <sys/ucred.h>
#include <sys/sysctl.h>
#include <uvm/uvm_extern.h>
#include <err.h>
#include <grp.h>
#include <kvm.h>
#include <math.h>
#include <nlist.h>
#include <stddef.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <tzfile.h>
#include <unistd.h>
#include <pwd.h>
#include "ps.h"
extern kvm_t *kd;
extern int needenv, needcomm, commandonly;
static char *cmdpart(char *);
#define min(a,b) ((a) < (b) ? (a) : (b))
static char *
cmdpart(char *arg0)
{
char *cp;
return ((cp = strrchr(arg0, '/')) != NULL ? cp + 1 : arg0);
}
void
printheader(void)
{
VAR *v;
struct varent *vent;
for (vent = vhead; vent; vent = vent->next) {
v = vent->var;
if (v->flag & LJUST) {
if (vent->next == NULL) /* last one */
(void)printf("%s", v->header);
else
(void)printf("%-*s", v->width, v->header);
} else
(void)printf("%*s", v->width, v->header);
if (vent->next != NULL)
(void)putchar(' ');
}
(void)putchar('\n');
}
void
command(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
int left;
char **argv, **p;
v = ve->var;
if (ve->next != NULL || termwidth != UNLIMITED) {
if (ve->next == NULL) {
left = termwidth - (totwidth - v->width);
if (left < 1) /* already wrapped, just use std width */
left = v->width;
} else
left = v->width;
} else
left = -1;
if (needenv && kd != NULL) {
argv = kvm_getenvv2(kd, kp, termwidth);
if ((p = argv) != NULL) {
while (*p) {
fmt_puts(*p, &left);
p++;
fmt_putc(' ', &left);
}
}
} else
argv = NULL;
if (needcomm) {
if (!commandonly) {
if (kd != NULL) {
argv = kvm_getargv2(kd, kp, termwidth);
if ((p = argv) != NULL) {
while (*p) {
fmt_puts(*p, &left);
p++;
fmt_putc(' ', &left);
}
}
}
if (argv == NULL || argv[0] == '\0' ||
strcmp(cmdpart(argv[0]), kp->p_comm)) {
fmt_putc('(', &left);
fmt_puts(kp->p_comm, &left);
fmt_putc(')', &left);
}
} else {
fmt_puts(kp->p_comm, &left);
}
}
if (ve->next && left > 0)
printf("%*s", left, "");
}
void
ucomm(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%-*s", v->width, kp->p_comm);
}
void
logname(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
if (kp->p_login[0]) {
int n = min(v->width, MAXLOGNAME);
(void)printf("%-*.*s", n, n, kp->p_login);
if (v->width > n)
(void)printf("%*s", v->width - n, "");
} else
(void)printf("%-*s", v->width, "-");
}
#define pgtok(a) (((a)*getpagesize())/1024)
void
state(const struct kinfo_proc2 *kp, VARENT *ve)
{
int flag;
char *cp;
VAR *v;
char buf[16];
v = ve->var;
flag = kp->p_flag;
cp = buf;
switch (kp->p_stat) {
case SSTOP:
*cp = 'T';
break;
case SSLEEP:
if (flag & P_SINTR) /* interruptible (long) */
*cp = kp->p_slptime >= maxslp ? 'I' : 'S';
else
*cp = 'D';
break;
case SRUN:
case SIDL:
*cp = 'R';
break;
case SZOMB:
*cp = 'Z';
break;
default:
*cp = '?';
}
cp++;
if (flag & P_INMEM) {
} else
*cp++ = 'W';
if (kp->p_nice < NZERO)
*cp++ = '<';
else if (kp->p_nice > NZERO)
*cp++ = 'N';
if (flag & P_TRACED)
*cp++ = 'X';
if (flag & P_SYSTRACE)
*cp++ = 'x';
if (flag & P_WEXIT && kp->p_stat != SZOMB)
*cp++ = 'E';
if (flag & P_PPWAIT)
*cp++ = 'V';
if (flag & P_SYSTEM)
*cp++ = 'K';
/* XXX Since P_SYSTEM now shows a K, should L just be for holdcnt? */
if ((flag & P_SYSTEM) || kp->p_holdcnt)
*cp++ = 'L';
if ((flag & P_SYSTEM) == 0 &&
kp->p_rlim_rss_cur / 1024 < pgtok(kp->p_vm_rssize))
*cp++ = '>';
if (kp->p_eflag & EPROC_SLEADER)
*cp++ = 's';
if ((flag & P_CONTROLT) && kp->p__pgid == kp->p_tpgid)
*cp++ = '+';
*cp = '\0';
(void)printf("%-*s", v->width, buf);
}
void
pri(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%*d", v->width, kp->p_priority - PZERO);
}
void
uname(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%-*s",
(int)v->width, user_from_uid(kp->p_uid, 0));
}
void
runame(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%-*s",
(int)v->width, user_from_uid(kp->p_ruid, 0));
}
void
gname(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%-*s",
(int)v->width, group_from_gid(kp->p_gid, 0));
}
void
rgname(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%-*s",
(int)v->width, group_from_gid(kp->p_rgid, 0));
}
void
tdev(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
dev_t dev;
char buff[16];
v = ve->var;
dev = kp->p_tdev;
if (dev == NODEV)
(void)printf("%*s", v->width, "??");
else {
(void)snprintf(buff, sizeof(buff),
"%d/%d", major(dev), minor(dev));
(void)printf("%*s", v->width, buff);
}
}
void
tname(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
dev_t dev;
char *ttname;
v = ve->var;
dev = kp->p_tdev;
if (dev == NODEV || (ttname = devname(dev, S_IFCHR)) == NULL)
(void)printf("%-*s", v->width, "??");
else {
if (strncmp(ttname, "tty", 3) == 0)
ttname += 3;
(void)printf("%*.*s%c", v->width-1, v->width-1, ttname,
kp->p_eflag & EPROC_CTTY ? ' ' : '-');
}
}
void
longtname(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
dev_t dev;
char *ttname;
v = ve->var;
dev = kp->p_tdev;
if (dev == NODEV || (ttname = devname(dev, S_IFCHR)) == NULL)
(void)printf("%-*s", v->width, "??");
else
(void)printf("%-*s", v->width, ttname);
}
void
started(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
static time_t now;
time_t startt;
struct tm *tp;
char buf[100];
v = ve->var;
if (!kp->p_uvalid) {
(void)printf("%-*s", v->width, "-");
return;
}
startt = kp->p_ustart_sec;
tp = localtime(&startt);
if (!now)
(void)time(&now);
if (now - kp->p_ustart_sec < 24 * SECSPERHOUR) {
(void)strftime(buf, sizeof(buf) - 1, "%l:%M%p", tp);
} else if (now - kp->p_ustart_sec < 7 * SECSPERDAY) {
(void)strftime(buf, sizeof(buf) - 1, "%a%I%p", tp);
} else
(void)strftime(buf, sizeof(buf) - 1, "%e%b%y", tp);
(void)printf("%-*s", v->width, buf);
}
void
lstarted(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
time_t startt;
char buf[100];
v = ve->var;
if (!kp->p_uvalid) {
(void)printf("%-*s", v->width, "-");
return;
}
startt = kp->p_ustart_sec;
(void)strftime(buf, sizeof(buf) -1, "%c",
localtime(&startt));
(void)printf("%-*s", v->width, buf);
}
void
wchan(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
if (kp->p_wchan) {
int n;
if (kp->p_wmesg) {
n = min(v->width, WMESGLEN);
(void)printf("%-*.*s", n, n, kp->p_wmesg);
if (v->width > n)
(void)printf("%*s", v->width - n, "");
} else
(void)printf("%-*lx", v->width,
(long)kp->p_wchan &~ KERNBASE);
} else
(void)printf("%-*s", v->width, "-");
}
void
vsize(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%*d", v->width,
pgtok(kp->p_vm_dsize + kp->p_vm_ssize + kp->p_vm_tsize));
}
void
rssize(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
/* XXX don't have info about shared */
(void)printf("%*d", v->width, (kp->p_flag & P_SYSTEM) ? 0 :
pgtok(kp->p_vm_rssize));
}
void
p_rssize(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%*d", v->width, (kp->p_flag & P_SYSTEM) ? 0 :
pgtok(kp->p_vm_rssize));
}
void
cputime(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
long secs;
long psecs; /* "parts" of a second. first micro, then centi */
char obuff[128];
v = ve->var;
if (kp->p_stat == SZOMB || !kp->p_uvalid) {
secs = 0;
psecs = 0;
} else {
/*
* This counts time spent handling interrupts. We could
* fix this, but it is not 100% trivial (and interrupt
* time fractions only work on the sparc anyway). XXX
*/
secs = kp->p_rtime_sec;
psecs = kp->p_rtime_usec;
if (sumrusage) {
secs += kp->p_uctime_sec;
psecs += kp->p_uctime_usec;
}
/*
* round and scale to 100's
*/
psecs = (psecs + 5000) / 10000;
secs += psecs / 100;
psecs = psecs % 100;
}
(void)snprintf(obuff, sizeof(obuff),
"%3ld:%02ld.%02ld", secs/60, secs%60, psecs);
(void)printf("%*s", v->width, obuff);
}
double
getpcpu(const struct kinfo_proc2 *kp)
{
static int failure;
double d;
if (!nlistread)
failure = donlist();
if (failure)
return (0.0);
#define fxtofl(fixpt) ((double)(fixpt) / fscale)
/* XXX - I don't like this */
if (kp->p_swtime == 0 || (kp->p_flag & P_INMEM) == 0)
return (0.0);
if (rawcpu)
return (100.0 * fxtofl(kp->p_pctcpu));
d = kp->p_swtime * log(fxtofl(ccpu));
if (d < -700.0)
d = 0.0; /* avoid IEEE underflow */
else
d = exp(d);
if (d == 1.0)
return (0.0);
return (100.0 * fxtofl(kp->p_pctcpu) /
(1.0 - d));
}
void
pcpu(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%*.1f", v->width, getpcpu(kp));
}
double
getpmem(const struct kinfo_proc2 *kp)
{
static int failure;
double fracmem;
int szptudot;
if (!nlistread)
failure = donlist();
if (failure)
return (0.0);
if ((kp->p_flag & P_INMEM) == 0 || (kp->p_flag & P_SYSTEM))
return (0.0);
/* XXX want pmap ptpages, segtab, etc. (per architecture) */
szptudot = USPACE/getpagesize();
/* XXX don't have info about shared */
fracmem = ((float)kp->p_vm_rssize + szptudot)/mempages;
return (100.0 * fracmem);
}
void
pmem(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%*.1f", v->width, getpmem(kp));
}
void
pagein(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%*llu", v->width,
kp->p_uvalid ? kp->p_uru_majflt : 0);
}
void
maxrss(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%*lld", v->width, kp->p_rlim_rss_cur / 1024);
}
void
tsize(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%*d", v->width, pgtok(kp->p_vm_tsize));
}
void
dsize(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%*d", v->width, pgtok(kp->p_vm_dsize));
}
void
ssize(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%*d", v->width, pgtok(kp->p_vm_ssize));
}
/*
* Generic output routines. Print fields from various prototype
* structures.
*/
static void
printval(char *bp, VAR *v)
{
char ofmt[32];
snprintf(ofmt, sizeof(ofmt), "%%%s*%s", (v->flag & LJUST) ? "-" : "",
v->fmt);
/*
* Note that the "INF127" check is nonsensical for types
* that are or can be signed.
*/
#define GET(type) (*(type *)bp)
#define CHK_INF127(n) (((n) > 127) && (v->flag & INF127) ? 127 : (n))
switch (v->type) {
case INT8:
(void)printf(ofmt, v->width, GET(int8_t));
break;
case UINT8:
(void)printf(ofmt, v->width, CHK_INF127(GET(u_int8_t)));
break;
case INT16:
(void)printf(ofmt, v->width, GET(int16_t));
break;
case UINT16:
(void)printf(ofmt, v->width, CHK_INF127(GET(u_int16_t)));
break;
case INT32:
(void)printf(ofmt, v->width, GET(int32_t));
break;
case UINT32:
(void)printf(ofmt, v->width, CHK_INF127(GET(u_int32_t)));
break;
case INT64:
(void)printf(ofmt, v->width, GET(int64_t));
case UINT64:
(void)printf(ofmt, v->width, CHK_INF127(GET(u_int64_t)));
break;
default:
errx(1, "unknown type %d", v->type);
}
#undef GET
#undef CHK_INF127
}
void
pvar(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
if ((v->flag & USER) && !kp->p_uvalid)
(void)printf("%*s", v->width, "-");
else
printval((char *)kp + v->off, v);
}
void
emulname(const struct kinfo_proc2 *kp, VARENT *ve)
{
VAR *v;
v = ve->var;
(void)printf("%-*s", (int)v->width, kp->p_emul);
}