xref: /freebsd/bin/ps/print.c (revision 119656bc26959f207d464174393d67e198d239be)
1 /*-
2  * SPDX-License-Identifier: BSD-3-Clause
3  *
4  * Copyright (c) 1990, 1993, 1994
5  *	The Regents of the University of California.  All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer.
12  * 2. Redistributions in binary form must reproduce the above copyright
13  *    notice, this list of conditions and the following disclaimer in the
14  *    documentation and/or other materials provided with the distribution.
15  * 3. Neither the name of the University nor the names of its contributors
16  *    may be used to endorse or promote products derived from this software
17  *    without specific prior written permission.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
20  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
21  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
22  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
23  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
24  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
25  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
26  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
27  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
28  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29  * SUCH DAMAGE.
30  */
31 
32 #if 0
33 #ifndef lint
34 static char sccsid[] = "@(#)print.c	8.6 (Berkeley) 4/16/94";
35 #endif /* not lint */
36 #endif
37 
38 #include <sys/cdefs.h>
39 __FBSDID("$FreeBSD$");
40 
41 #include <sys/param.h>
42 #include <sys/time.h>
43 #include <sys/resource.h>
44 #include <sys/proc.h>
45 #include <sys/stat.h>
46 
47 #include <sys/mac.h>
48 #include <sys/user.h>
49 #include <sys/sysctl.h>
50 #include <sys/vmmeter.h>
51 
52 #include <err.h>
53 #include <grp.h>
54 #include <jail.h>
55 #include <langinfo.h>
56 #include <locale.h>
57 #include <math.h>
58 #include <nlist.h>
59 #include <pwd.h>
60 #include <stddef.h>
61 #include <stdint.h>
62 #include <stdio.h>
63 #include <stdlib.h>
64 #include <string.h>
65 #include <unistd.h>
66 #include <vis.h>
67 #include <libxo/xo.h>
68 
69 #include "ps.h"
70 
71 #define	COMMAND_WIDTH	16
72 #define	ARGUMENTS_WIDTH	16
73 
74 #define	ps_pgtok(a)	(((a) * getpagesize()) / 1024)
75 
76 void
77 printheader(void)
78 {
79 	VAR *v;
80 	struct varent *vent;
81 
82 	STAILQ_FOREACH(vent, &varlist, next_ve)
83 		if (*vent->header != '\0')
84 			break;
85 	if (!vent)
86 		return;
87 
88 	STAILQ_FOREACH(vent, &varlist, next_ve) {
89 		v = vent->var;
90 		if (v->flag & LJUST) {
91 			if (STAILQ_NEXT(vent, next_ve) == NULL)	/* last one */
92 				xo_emit("{T:/%hs}", vent->header);
93 			else
94 				xo_emit("{T:/%-*hs}", v->width, vent->header);
95 		} else
96 			xo_emit("{T:/%*hs}", v->width, vent->header);
97 		if (STAILQ_NEXT(vent, next_ve) != NULL)
98 			xo_emit("{P: }");
99 	}
100 	xo_emit("\n");
101 }
102 
103 char *
104 arguments(KINFO *k, VARENT *ve)
105 {
106 	char *vis_args;
107 
108 	if ((vis_args = malloc(strlen(k->ki_args) * 4 + 1)) == NULL)
109 		xo_errx(1, "malloc failed");
110 	strvis(vis_args, k->ki_args, VIS_TAB | VIS_NL | VIS_NOSLASH);
111 
112 	if (STAILQ_NEXT(ve, next_ve) != NULL && strlen(vis_args) > ARGUMENTS_WIDTH)
113 		vis_args[ARGUMENTS_WIDTH] = '\0';
114 
115 	return (vis_args);
116 }
117 
118 char *
119 command(KINFO *k, VARENT *ve)
120 {
121 	char *vis_args, *vis_env, *str;
122 
123 	if (cflag) {
124 		/* If it is the last field, then don't pad */
125 		if (STAILQ_NEXT(ve, next_ve) == NULL) {
126 			asprintf(&str, "%s%s%s%s%s",
127 			    k->ki_d.prefix ? k->ki_d.prefix : "",
128 			    k->ki_p->ki_comm,
129 			    (showthreads && k->ki_p->ki_numthreads > 1) ? "/" : "",
130 			    (showthreads && k->ki_p->ki_numthreads > 1) ? k->ki_p->ki_tdname : "",
131 			    (showthreads && k->ki_p->ki_numthreads > 1) ? k->ki_p->ki_moretdname : "");
132 		} else
133 			str = strdup(k->ki_p->ki_comm);
134 
135 		return (str);
136 	}
137 	if ((vis_args = malloc(strlen(k->ki_args) * 4 + 1)) == NULL)
138 		xo_errx(1, "malloc failed");
139 	strvis(vis_args, k->ki_args, VIS_TAB | VIS_NL | VIS_NOSLASH);
140 
141 	if (STAILQ_NEXT(ve, next_ve) == NULL) {
142 		/* last field */
143 
144 		if (k->ki_env) {
145 			if ((vis_env = malloc(strlen(k->ki_env) * 4 + 1))
146 			    == NULL)
147 				xo_errx(1, "malloc failed");
148 			strvis(vis_env, k->ki_env,
149 			    VIS_TAB | VIS_NL | VIS_NOSLASH);
150 		} else
151 			vis_env = NULL;
152 
153 		asprintf(&str, "%s%s%s%s",
154 		    k->ki_d.prefix ? k->ki_d.prefix : "",
155 		    vis_env ? vis_env : "",
156 		    vis_env ? " " : "",
157 		    vis_args);
158 
159 		if (vis_env != NULL)
160 			free(vis_env);
161 		free(vis_args);
162 	} else {
163 		/* ki_d.prefix & ki_env aren't shown for interim fields */
164 		str = vis_args;
165 
166 		if (strlen(str) > COMMAND_WIDTH)
167 			str[COMMAND_WIDTH] = '\0';
168 	}
169 
170 	return (str);
171 }
172 
173 char *
174 ucomm(KINFO *k, VARENT *ve)
175 {
176 	char *str;
177 
178 	if (STAILQ_NEXT(ve, next_ve) == NULL) {	/* last field, don't pad */
179 		asprintf(&str, "%s%s%s%s%s",
180 		    k->ki_d.prefix ? k->ki_d.prefix : "",
181 		    k->ki_p->ki_comm,
182 		    (showthreads && k->ki_p->ki_numthreads > 1) ? "/" : "",
183 		    (showthreads && k->ki_p->ki_numthreads > 1) ? k->ki_p->ki_tdname : "",
184 		    (showthreads && k->ki_p->ki_numthreads > 1) ? k->ki_p->ki_moretdname : "");
185 	} else {
186 		if (showthreads && k->ki_p->ki_numthreads > 1)
187 			asprintf(&str, "%s/%s%s", k->ki_p->ki_comm,
188 			    k->ki_p->ki_tdname, k->ki_p->ki_moretdname);
189 		else
190 			str = strdup(k->ki_p->ki_comm);
191 	}
192 	return (str);
193 }
194 
195 char *
196 tdnam(KINFO *k, VARENT *ve __unused)
197 {
198 	char *str;
199 
200 	if (showthreads && k->ki_p->ki_numthreads > 1)
201 		asprintf(&str, "%s%s", k->ki_p->ki_tdname,
202 		    k->ki_p->ki_moretdname);
203 	else
204 		str = strdup("      ");
205 
206 	return (str);
207 }
208 
209 char *
210 logname(KINFO *k, VARENT *ve __unused)
211 {
212 
213 	if (*k->ki_p->ki_login == '\0')
214 		return (NULL);
215 	return (strdup(k->ki_p->ki_login));
216 }
217 
218 char *
219 state(KINFO *k, VARENT *ve __unused)
220 {
221 	long flag, tdflags;
222 	char *cp, *buf;
223 
224 	buf = malloc(16);
225 	if (buf == NULL)
226 		xo_errx(1, "malloc failed");
227 
228 	flag = k->ki_p->ki_flag;
229 	tdflags = k->ki_p->ki_tdflags;	/* XXXKSE */
230 	cp = buf;
231 
232 	switch (k->ki_p->ki_stat) {
233 
234 	case SSTOP:
235 		*cp = 'T';
236 		break;
237 
238 	case SSLEEP:
239 		if (tdflags & TDF_SINTR)	/* interruptable (long) */
240 			*cp = k->ki_p->ki_slptime >= MAXSLP ? 'I' : 'S';
241 		else
242 			*cp = 'D';
243 		break;
244 
245 	case SRUN:
246 	case SIDL:
247 		*cp = 'R';
248 		break;
249 
250 	case SWAIT:
251 		*cp = 'W';
252 		break;
253 
254 	case SLOCK:
255 		*cp = 'L';
256 		break;
257 
258 	case SZOMB:
259 		*cp = 'Z';
260 		break;
261 
262 	default:
263 		*cp = '?';
264 	}
265 	cp++;
266 	if (!(flag & P_INMEM))
267 		*cp++ = 'W';
268 	if (k->ki_p->ki_nice < NZERO || k->ki_p->ki_pri.pri_class == PRI_REALTIME)
269 		*cp++ = '<';
270 	else if (k->ki_p->ki_nice > NZERO || k->ki_p->ki_pri.pri_class == PRI_IDLE)
271 		*cp++ = 'N';
272 	if (flag & P_TRACED)
273 		*cp++ = 'X';
274 	if (flag & P_WEXIT && k->ki_p->ki_stat != SZOMB)
275 		*cp++ = 'E';
276 	if (flag & P_PPWAIT)
277 		*cp++ = 'V';
278 	if ((flag & P_SYSTEM) || k->ki_p->ki_lock > 0)
279 		*cp++ = 'L';
280 	if ((k->ki_p->ki_cr_flags & CRED_FLAG_CAPMODE) != 0)
281 		*cp++ = 'C';
282 	if (k->ki_p->ki_kiflag & KI_SLEADER)
283 		*cp++ = 's';
284 	if ((flag & P_CONTROLT) && k->ki_p->ki_pgid == k->ki_p->ki_tpgid)
285 		*cp++ = '+';
286 	if (flag & P_JAILED)
287 		*cp++ = 'J';
288 	*cp = '\0';
289 	return (buf);
290 }
291 
292 #define	scalepri(x)	((x) - PZERO)
293 
294 char *
295 pri(KINFO *k, VARENT *ve __unused)
296 {
297 	char *str;
298 
299 	asprintf(&str, "%d", scalepri(k->ki_p->ki_pri.pri_level));
300 	return (str);
301 }
302 
303 char *
304 upr(KINFO *k, VARENT *ve __unused)
305 {
306 	char *str;
307 
308 	asprintf(&str, "%d", scalepri(k->ki_p->ki_pri.pri_user));
309 	return (str);
310 }
311 #undef scalepri
312 
313 char *
314 uname(KINFO *k, VARENT *ve __unused)
315 {
316 
317 	return (strdup(user_from_uid(k->ki_p->ki_uid, 0)));
318 }
319 
320 char *
321 egroupname(KINFO *k, VARENT *ve __unused)
322 {
323 
324 	return (strdup(group_from_gid(k->ki_p->ki_groups[0], 0)));
325 }
326 
327 char *
328 rgroupname(KINFO *k, VARENT *ve __unused)
329 {
330 
331 	return (strdup(group_from_gid(k->ki_p->ki_rgid, 0)));
332 }
333 
334 char *
335 runame(KINFO *k, VARENT *ve __unused)
336 {
337 
338 	return (strdup(user_from_uid(k->ki_p->ki_ruid, 0)));
339 }
340 
341 char *
342 tdev(KINFO *k, VARENT *ve __unused)
343 {
344 	dev_t dev;
345 	char *str;
346 
347 	dev = k->ki_p->ki_tdev;
348 	if (dev == NODEV)
349 		str = strdup("-");
350 	else
351 		asprintf(&str, "%#jx", (uintmax_t)dev);
352 
353 	return (str);
354 }
355 
356 char *
357 tname(KINFO *k, VARENT *ve __unused)
358 {
359 	dev_t dev;
360 	char *ttname, *str;
361 
362 	dev = k->ki_p->ki_tdev;
363 	if (dev == NODEV || (ttname = devname(dev, S_IFCHR)) == NULL)
364 		str = strdup("- ");
365 	else {
366 		if (strncmp(ttname, "tty", 3) == 0 ||
367 		    strncmp(ttname, "cua", 3) == 0)
368 			ttname += 3;
369 		if (strncmp(ttname, "pts/", 4) == 0)
370 			ttname += 4;
371 		asprintf(&str, "%s%c", ttname,
372 		    k->ki_p->ki_kiflag & KI_CTTY ? ' ' : '-');
373 	}
374 
375 	return (str);
376 }
377 
378 char *
379 longtname(KINFO *k, VARENT *ve __unused)
380 {
381 	dev_t dev;
382 	const char *ttname;
383 
384 	dev = k->ki_p->ki_tdev;
385 	if (dev == NODEV || (ttname = devname(dev, S_IFCHR)) == NULL)
386 		ttname = "-";
387 
388 	return (strdup(ttname));
389 }
390 
391 char *
392 started(KINFO *k, VARENT *ve __unused)
393 {
394 	time_t then;
395 	struct tm *tp;
396 	size_t buflen = 100;
397 	char *buf;
398 
399 	if (!k->ki_valid)
400 		return (NULL);
401 
402 	buf = malloc(buflen);
403 	if (buf == NULL)
404 		xo_errx(1, "malloc failed");
405 
406 	then = k->ki_p->ki_start.tv_sec;
407 	tp = localtime(&then);
408 	if (now - k->ki_p->ki_start.tv_sec < 24 * 3600) {
409 		(void)strftime(buf, buflen, "%H:%M  ", tp);
410 	} else if (now - k->ki_p->ki_start.tv_sec < 7 * 86400) {
411 		(void)strftime(buf, buflen, "%a%H  ", tp);
412 	} else
413 		(void)strftime(buf, buflen, "%e%b%y", tp);
414 	return (buf);
415 }
416 
417 char *
418 lstarted(KINFO *k, VARENT *ve __unused)
419 {
420 	time_t then;
421 	char *buf;
422 	size_t buflen = 100;
423 
424 	if (!k->ki_valid)
425 		return (NULL);
426 
427 	buf = malloc(buflen);
428 	if (buf == NULL)
429 		xo_errx(1, "malloc failed");
430 
431 	then = k->ki_p->ki_start.tv_sec;
432 	(void)strftime(buf, buflen, "%c", localtime(&then));
433 	return (buf);
434 }
435 
436 char *
437 lockname(KINFO *k, VARENT *ve __unused)
438 {
439 	char *str;
440 
441 	if (k->ki_p->ki_kiflag & KI_LOCKBLOCK) {
442 		if (k->ki_p->ki_lockname[0] != 0)
443 			str = strdup(k->ki_p->ki_lockname);
444 		else
445 			str = strdup("???");
446 	} else
447 		str = NULL;
448 
449 	return (str);
450 }
451 
452 char *
453 wchan(KINFO *k, VARENT *ve __unused)
454 {
455 	char *str;
456 
457 	if (k->ki_p->ki_wchan) {
458 		if (k->ki_p->ki_wmesg[0] != 0)
459 			str = strdup(k->ki_p->ki_wmesg);
460 		else
461 			asprintf(&str, "%lx", (long)k->ki_p->ki_wchan);
462 	} else
463 		str = NULL;
464 
465 	return (str);
466 }
467 
468 char *
469 nwchan(KINFO *k, VARENT *ve __unused)
470 {
471 	char *str;
472 
473 	if (k->ki_p->ki_wchan)
474 		asprintf(&str, "%0lx", (long)k->ki_p->ki_wchan);
475 	else
476 		str = NULL;
477 
478 	return (str);
479 }
480 
481 char *
482 mwchan(KINFO *k, VARENT *ve __unused)
483 {
484 	char *str;
485 
486 	if (k->ki_p->ki_wchan) {
487 		if (k->ki_p->ki_wmesg[0] != 0)
488 			str = strdup(k->ki_p->ki_wmesg);
489 		else
490                         asprintf(&str, "%lx", (long)k->ki_p->ki_wchan);
491 	} else if (k->ki_p->ki_kiflag & KI_LOCKBLOCK) {
492 		if (k->ki_p->ki_lockname[0]) {
493 			str = strdup(k->ki_p->ki_lockname);
494 		} else
495 			str = strdup("???");
496 	} else
497 		str = NULL;
498 
499 	return (str);
500 }
501 
502 char *
503 vsize(KINFO *k, VARENT *ve __unused)
504 {
505 	char *str;
506 
507 	asprintf(&str, "%lu", (u_long)(k->ki_p->ki_size / 1024));
508 	return (str);
509 }
510 
511 static char *
512 printtime(KINFO *k, VARENT *ve __unused, long secs, long psecs)
513 /* psecs is "parts" of a second. first micro, then centi */
514 {
515 	static char decimal_point;
516 	char *str;
517 
518 	if (decimal_point == '\0')
519 		decimal_point = localeconv()->decimal_point[0];
520 	if (!k->ki_valid) {
521 		secs = 0;
522 		psecs = 0;
523 	} else {
524 		/* round and scale to 100's */
525 		psecs = (psecs + 5000) / 10000;
526 		secs += psecs / 100;
527 		psecs = psecs % 100;
528 	}
529 	asprintf(&str, "%ld:%02ld%c%02ld",
530 	    secs / 60, secs % 60, decimal_point, psecs);
531 	return (str);
532 }
533 
534 char *
535 cputime(KINFO *k, VARENT *ve)
536 {
537 	long secs, psecs;
538 
539 	/*
540 	 * This counts time spent handling interrupts.  We could
541 	 * fix this, but it is not 100% trivial (and interrupt
542 	 * time fractions only work on the sparc anyway).	XXX
543 	 */
544 	secs = k->ki_p->ki_runtime / 1000000;
545 	psecs = k->ki_p->ki_runtime % 1000000;
546 	if (sumrusage) {
547 		secs += k->ki_p->ki_childtime.tv_sec;
548 		psecs += k->ki_p->ki_childtime.tv_usec;
549 	}
550 	return (printtime(k, ve, secs, psecs));
551 }
552 
553 char *
554 cpunum(KINFO *k, VARENT *ve __unused)
555 {
556 	char *cpu;
557 
558 	if (k->ki_p->ki_stat == SRUN && k->ki_p->ki_oncpu != NOCPU) {
559 		asprintf(&cpu, "%d", k->ki_p->ki_oncpu);
560 	} else {
561 		asprintf(&cpu, "%d", k->ki_p->ki_lastcpu);
562 	}
563 	return (cpu);
564 }
565 
566 char *
567 systime(KINFO *k, VARENT *ve)
568 {
569 	long secs, psecs;
570 
571 	secs = k->ki_p->ki_rusage.ru_stime.tv_sec;
572 	psecs = k->ki_p->ki_rusage.ru_stime.tv_usec;
573 	if (sumrusage) {
574 		secs += k->ki_p->ki_childstime.tv_sec;
575 		psecs += k->ki_p->ki_childstime.tv_usec;
576 	}
577 	return (printtime(k, ve, secs, psecs));
578 }
579 
580 char *
581 usertime(KINFO *k, VARENT *ve)
582 {
583 	long secs, psecs;
584 
585 	secs = k->ki_p->ki_rusage.ru_utime.tv_sec;
586 	psecs = k->ki_p->ki_rusage.ru_utime.tv_usec;
587 	if (sumrusage) {
588 		secs += k->ki_p->ki_childutime.tv_sec;
589 		psecs += k->ki_p->ki_childutime.tv_usec;
590 	}
591 	return (printtime(k, ve, secs, psecs));
592 }
593 
594 char *
595 elapsed(KINFO *k, VARENT *ve __unused)
596 {
597 	time_t val;
598 	int days, hours, mins, secs;
599 	char *str;
600 
601 	if (!k->ki_valid)
602 		return (NULL);
603 	val = now - k->ki_p->ki_start.tv_sec;
604 	days = val / (24 * 60 * 60);
605 	val %= 24 * 60 * 60;
606 	hours = val / (60 * 60);
607 	val %= 60 * 60;
608 	mins = val / 60;
609 	secs = val % 60;
610 	if (days != 0)
611 		asprintf(&str, "%3d-%02d:%02d:%02d", days, hours, mins, secs);
612 	else if (hours != 0)
613 		asprintf(&str, "%02d:%02d:%02d", hours, mins, secs);
614 	else
615 		asprintf(&str, "%02d:%02d", mins, secs);
616 
617 	return (str);
618 }
619 
620 char *
621 elapseds(KINFO *k, VARENT *ve __unused)
622 {
623 	time_t val;
624 	char *str;
625 
626 	if (!k->ki_valid)
627 		return (NULL);
628 	val = now - k->ki_p->ki_start.tv_sec;
629 	asprintf(&str, "%jd", (intmax_t)val);
630 	return (str);
631 }
632 
633 double
634 getpcpu(const KINFO *k)
635 {
636 	static int failure;
637 
638 	if (!nlistread)
639 		failure = donlist();
640 	if (failure)
641 		return (0.0);
642 
643 #define	fxtofl(fixpt)	((double)(fixpt) / fscale)
644 
645 	/* XXX - I don't like this */
646 	if (k->ki_p->ki_swtime == 0 || (k->ki_p->ki_flag & P_INMEM) == 0)
647 		return (0.0);
648 	if (rawcpu)
649 		return (100.0 * fxtofl(k->ki_p->ki_pctcpu));
650 	return (100.0 * fxtofl(k->ki_p->ki_pctcpu) /
651 		(1.0 - exp(k->ki_p->ki_swtime * log(fxtofl(ccpu)))));
652 }
653 
654 char *
655 pcpu(KINFO *k, VARENT *ve __unused)
656 {
657 	char *str;
658 
659 	asprintf(&str, "%.1f", getpcpu(k));
660 	return (str);
661 }
662 
663 static double
664 getpmem(KINFO *k)
665 {
666 	static int failure;
667 	double fracmem;
668 
669 	if (!nlistread)
670 		failure = donlist();
671 	if (failure)
672 		return (0.0);
673 
674 	if ((k->ki_p->ki_flag & P_INMEM) == 0)
675 		return (0.0);
676 	/* XXX want pmap ptpages, segtab, etc. (per architecture) */
677 	/* XXX don't have info about shared */
678 	fracmem = ((double)k->ki_p->ki_rssize) / mempages;
679 	return (100.0 * fracmem);
680 }
681 
682 char *
683 pmem(KINFO *k, VARENT *ve __unused)
684 {
685 	char *str;
686 
687 	asprintf(&str, "%.1f", getpmem(k));
688 	return (str);
689 }
690 
691 char *
692 pagein(KINFO *k, VARENT *ve __unused)
693 {
694 	char *str;
695 
696 	asprintf(&str, "%ld", k->ki_valid ? k->ki_p->ki_rusage.ru_majflt : 0);
697 	return (str);
698 }
699 
700 /* ARGSUSED */
701 char *
702 maxrss(KINFO *k __unused, VARENT *ve __unused)
703 {
704 
705 	/* XXX not yet */
706 	return (NULL);
707 }
708 
709 char *
710 priorityr(KINFO *k, VARENT *ve __unused)
711 {
712 	struct priority *lpri;
713 	char *str;
714 	unsigned class, level;
715 
716 	lpri = &k->ki_p->ki_pri;
717 	class = lpri->pri_class;
718 	level = lpri->pri_level;
719 	switch (class) {
720 	case RTP_PRIO_REALTIME:
721 	/* alias for PRI_REALTIME */
722 		asprintf(&str, "real:%u", level - PRI_MIN_REALTIME);
723 		break;
724 	case RTP_PRIO_NORMAL:
725 	/* alias for PRI_TIMESHARE */
726 		if (level >= PRI_MIN_TIMESHARE)
727 			asprintf(&str, "normal:%u", level - PRI_MIN_TIMESHARE);
728 		else
729 			asprintf(&str, "kernel:%u", level - PRI_MIN_KERN);
730 		break;
731 	case RTP_PRIO_IDLE:
732 	/* alias for PRI_IDLE */
733 		asprintf(&str, "idle:%u", level - PRI_MIN_IDLE);
734 		break;
735 	case RTP_PRIO_ITHD:
736 	/* alias for PRI_ITHD */
737 		asprintf(&str, "intr:%u", level - PRI_MIN_ITHD);
738 		break;
739 	default:
740 		asprintf(&str, "%u:%u", class, level);
741 		break;
742 	}
743 	return (str);
744 }
745 
746 /*
747  * Generic output routines.  Print fields from various prototype
748  * structures.
749  */
750 static char *
751 printval(void *bp, VAR *v)
752 {
753 	static char ofmt[32] = "%";
754 	const char *fcp;
755 	char *cp, *str;
756 
757 	cp = ofmt + 1;
758 	fcp = v->fmt;
759 	while ((*cp++ = *fcp++));
760 
761 #define	CHKINF127(n)	(((n) > 127) && (v->flag & INF127) ? 127 : (n))
762 
763 	switch (v->type) {
764 	case CHAR:
765 		(void)asprintf(&str, ofmt, *(char *)bp);
766 		break;
767 	case UCHAR:
768 		(void)asprintf(&str, ofmt, *(u_char *)bp);
769 		break;
770 	case SHORT:
771 		(void)asprintf(&str, ofmt, *(short *)bp);
772 		break;
773 	case USHORT:
774 		(void)asprintf(&str, ofmt, *(u_short *)bp);
775 		break;
776 	case INT:
777 		(void)asprintf(&str, ofmt, *(int *)bp);
778 		break;
779 	case UINT:
780 		(void)asprintf(&str, ofmt, CHKINF127(*(u_int *)bp));
781 		break;
782 	case LONG:
783 		(void)asprintf(&str, ofmt, *(long *)bp);
784 		break;
785 	case ULONG:
786 		(void)asprintf(&str, ofmt, *(u_long *)bp);
787 		break;
788 	case KPTR:
789 		(void)asprintf(&str, ofmt, *(u_long *)bp);
790 		break;
791 	case PGTOK:
792 		(void)asprintf(&str, ofmt, ps_pgtok(*(u_long *)bp));
793 		break;
794 	}
795 
796 	return (str);
797 }
798 
799 char *
800 kvar(KINFO *k, VARENT *ve)
801 {
802 	VAR *v;
803 
804 	v = ve->var;
805 	return (printval((char *)((char *)k->ki_p + v->off), v));
806 }
807 
808 char *
809 rvar(KINFO *k, VARENT *ve)
810 {
811 	VAR *v;
812 
813 	v = ve->var;
814 	if (!k->ki_valid)
815 		return (NULL);
816 	return (printval((char *)((char *)(&k->ki_p->ki_rusage) + v->off), v));
817 }
818 
819 char *
820 emulname(KINFO *k, VARENT *ve __unused)
821 {
822 
823 	return (strdup(k->ki_p->ki_emul));
824 }
825 
826 char *
827 label(KINFO *k, VARENT *ve __unused)
828 {
829 	char *string;
830 	mac_t proclabel;
831 	int error;
832 
833 	string = NULL;
834 	if (mac_prepare_process_label(&proclabel) == -1) {
835 		xo_warn("mac_prepare_process_label");
836 		goto out;
837 	}
838 	error = mac_get_pid(k->ki_p->ki_pid, proclabel);
839 	if (error == 0) {
840 		if (mac_to_text(proclabel, &string) == -1)
841 			string = NULL;
842 	}
843 	mac_free(proclabel);
844 out:
845 	return (string);
846 }
847 
848 char *
849 loginclass(KINFO *k, VARENT *ve __unused)
850 {
851 
852 	/*
853 	 * Don't display login class for system processes;
854 	 * login classes are used for resource limits,
855 	 * and limits don't apply to system processes.
856 	 */
857 	if (k->ki_p->ki_flag & P_SYSTEM) {
858 		return (strdup("-"));
859 	}
860 	return (strdup(k->ki_p->ki_loginclass));
861 }
862 
863 char *
864 jailname(KINFO *k, VARENT *ve __unused)
865 {
866 	char *name;
867 
868 	if (k->ki_p->ki_jid == 0)
869 		return (strdup("-"));
870 	name = jail_getname(k->ki_p->ki_jid);
871 	if (name == NULL)
872 		return (strdup("-"));
873 	return (name);
874 }
875