xref: /illumos-gate/usr/src/cmd/ps/ps.c (revision 82049ff560eed6fbdf4cf222d894467f5809f9b3)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /*
23  * Copyright (c) 2013 Gary Mills
24  *
25  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
26  * Use is subject to license terms.
27  */
28 
29 /*
30  * Copyright (c) 2018, Joyent, Inc.
31  */
32 
33 /*	Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T	*/
34 /*	  All Rights Reserved	*/
35 
36 /*
37  * ps -- print things about processes.
38  */
39 #include <stdio.h>
40 #include <ctype.h>
41 #include <string.h>
42 #include <errno.h>
43 #include <fcntl.h>
44 #include <pwd.h>
45 #include <grp.h>
46 #include <sys/types.h>
47 #include <sys/stat.h>
48 #include <sys/mkdev.h>
49 #include <unistd.h>
50 #include <stdlib.h>
51 #include <limits.h>
52 #include <dirent.h>
53 #include <sys/signal.h>
54 #include <sys/fault.h>
55 #include <sys/syscall.h>
56 #include <sys/time.h>
57 #include <procfs.h>
58 #include <locale.h>
59 #include <wctype.h>
60 #include <wchar.h>
61 #include <libw.h>
62 #include <stdarg.h>
63 #include <sys/proc.h>
64 #include <sys/pset.h>
65 #include <project.h>
66 #include <zone.h>
67 
68 #define	min(a, b)	((a) > (b) ? (b) : (a))
69 #define	max(a, b)	((a) < (b) ? (b) : (a))
70 
71 #define	NTTYS	20	/* initial size of table for -t option  */
72 #define	SIZ	30	/* initial size of tables for -p, -s, -g, -h and -z */
73 
74 /*
75  * Size of buffer holding args for t, p, s, g, u, U, G, z options.
76  * Set to ZONENAME_MAX, the minimum value needed to allow any
77  * zone to be specified.
78  */
79 #define	ARGSIZ ZONENAME_MAX
80 
81 /* Max chars in a user/group name or printed u/g id */
82 #define	MAXUGNAME (LOGNAME_MAX+2)
83 
84 /* Structure for storing user or group info */
85 struct ugdata {
86 	id_t	id;			/* numeric user-id or group-id */
87 	char	name[MAXUGNAME+1];	/* user/group name, null terminated */
88 };
89 
90 struct ughead {
91 	size_t	size;		/* number of ugdata structs allocated */
92 	size_t	nent;		/* number of active entries */
93 	struct ugdata *ent;	/* pointer to array of actual entries */
94 };
95 
96 enum fname {	/* enumeration of field names */
97 	F_USER,		/* effective user of the process */
98 	F_RUSER,	/* real user of the process */
99 	F_GROUP,	/* effective group of the process */
100 	F_RGROUP,	/* real group of the process */
101 	F_UID,		/* numeric effective uid of the process */
102 	F_RUID,		/* numeric real uid of the process */
103 	F_GID,		/* numeric effective gid of the process */
104 	F_RGID,		/* numeric real gid of the process */
105 	F_PID,		/* process id */
106 	F_PPID,		/* parent process id */
107 	F_PGID,		/* process group id */
108 	F_SID,		/* session id */
109 	F_PSR,		/* bound processor */
110 	F_LWP,		/* lwp-id */
111 	F_LWPNAME,	/* lwp name */
112 	F_NLWP,		/* number of lwps */
113 	F_OPRI,		/* old priority (obsolete) */
114 	F_PRI,		/* new priority */
115 	F_F,		/* process flags */
116 	F_S,		/* letter indicating the state */
117 	F_C,		/* processor utilization (obsolete) */
118 	F_PCPU,		/* percent of recently used cpu time */
119 	F_PMEM,		/* percent of physical memory used (rss) */
120 	F_OSZ,		/* virtual size of the process in pages */
121 	F_VSZ,		/* virtual size of the process in kilobytes */
122 	F_RSS,		/* resident set size of the process in kilobytes */
123 	F_NICE,		/* "nice" value of the process */
124 	F_CLASS,	/* scheduler class */
125 	F_STIME,	/* start time of the process, hh:mm:ss or Month Day */
126 	F_ETIME,	/* elapsed time of the process, [[dd-]hh:]mm:ss */
127 	F_TIME,		/* cpu time of the process, [[dd-]hh:]mm:ss */
128 	F_TTY,		/* name of the controlling terminal */
129 	F_ADDR,		/* address of the process (obsolete) */
130 	F_WCHAN,	/* wait channel (sleep condition variable) */
131 	F_FNAME,	/* file name of command */
132 	F_COMM,		/* name of command (argv[0] value) */
133 	F_ARGS,		/* name of command plus all its arguments */
134 	F_TASKID,	/* task id */
135 	F_PROJID,	/* project id */
136 	F_PROJECT,	/* project name of the process */
137 	F_PSET,		/* bound processor set */
138 	F_ZONE,		/* zone name */
139 	F_ZONEID,	/* zone id */
140 	F_CTID,		/* process contract id */
141 	F_LGRP,		/* process home lgroup */
142 	F_DMODEL	/* process data model */
143 };
144 
145 struct field {
146 	struct field	*next;		/* linked list */
147 	int		fname;		/* field index */
148 	const char	*header;	/* header to use */
149 	int		width;		/* width of field */
150 };
151 
152 static	struct field *fields = NULL;	/* fields selected via -o */
153 static	struct field *last_field = NULL;
154 static	int do_header = 0;
155 static	struct timeval now;
156 
157 /* array of defined fields, in fname order */
158 struct def_field {
159 	const char *fname;
160 	const char *header;
161 	int width;
162 	int minwidth;
163 };
164 
165 static struct def_field fname[] = {
166 	/* fname	header		width	minwidth */
167 	{ "user",	"USER",		8,	8	},
168 	{ "ruser",	"RUSER",	8,	8	},
169 	{ "group",	"GROUP",	8,	8	},
170 	{ "rgroup",	"RGROUP",	8,	8	},
171 	{ "uid",	"UID",		5,	5	},
172 	{ "ruid",	"RUID",		5,	5	},
173 	{ "gid",	"GID",		5,	5	},
174 	{ "rgid",	"RGID",		5,	5	},
175 	{ "pid",	"PID",		5,	5	},
176 	{ "ppid",	"PPID",		5,	5	},
177 	{ "pgid",	"PGID",		5,	5	},
178 	{ "sid",	"SID",		5,	5	},
179 	{ "psr",	"PSR",		3,	2	},
180 	{ "lwp",	"LWP",		6,	2	},
181 	{ "lwpname",	"LWPNAME",	32,	8	},
182 	{ "nlwp",	"NLWP",		4,	2	},
183 	{ "opri",	"PRI",		3,	2	},
184 	{ "pri",	"PRI",		3,	2	},
185 	{ "f",		"F",		2,	2	},
186 	{ "s",		"S",		1,	1	},
187 	{ "c",		"C",		2,	2	},
188 	{ "pcpu",	"%CPU",		4,	4	},
189 	{ "pmem",	"%MEM",		4,	4	},
190 	{ "osz",	"SZ",		4,	4	},
191 	{ "vsz",	"VSZ",		4,	4	},
192 	{ "rss",	"RSS",		4,	4	},
193 	{ "nice",	"NI",		2,	2	},
194 	{ "class",	"CLS",		4,	2	},
195 	{ "stime",	"STIME",	8,	8	},
196 	{ "etime",	"ELAPSED",	11,	7	},
197 	{ "time",	"TIME",		11,	5	},
198 	{ "tty",	"TT",		7,	7	},
199 #ifdef _LP64
200 	{ "addr",	"ADDR",		16,	8	},
201 	{ "wchan",	"WCHAN",	16,	8	},
202 #else
203 	{ "addr",	"ADDR",		8,	8	},
204 	{ "wchan",	"WCHAN",	8,	8	},
205 #endif
206 	{ "fname",	"COMMAND",	8,	8	},
207 	{ "comm",	"COMMAND",	80,	8	},
208 	{ "args",	"COMMAND",	80,	80	},
209 	{ "taskid",	"TASKID",	5,	5	},
210 	{ "projid",	"PROJID",	5,	5	},
211 	{ "project",	"PROJECT",	8,	8	},
212 	{ "pset",	"PSET",		3,	3	},
213 	{ "zone",	"ZONE",		8,	8	},
214 	{ "zoneid",	"ZONEID",	5,	5	},
215 	{ "ctid",	"CTID",		5,	5	},
216 	{ "lgrp",	"LGRP",		4,	2	},
217 	{ "dmodel",	"DMODEL",	6,	6	},
218 };
219 
220 #define	NFIELDS	(sizeof (fname) / sizeof (fname[0]))
221 
222 static	int	retcode = 1;
223 static	int	lflg;
224 static	int	Aflg;
225 static	int	uflg;
226 static	int	Uflg;
227 static	int	Gflg;
228 static	int	aflg;
229 static	int	dflg;
230 static	int	Lflg;
231 static	int	Pflg;
232 static	int	Wflg;
233 static	int	yflg;
234 static	int	pflg;
235 static	int	fflg;
236 static	int	cflg;
237 static	int	jflg;
238 static	int	gflg;
239 static	int	sflg;
240 static	int	tflg;
241 static	int	zflg;
242 static	int	Zflg;
243 static	int	hflg;
244 static	int	Hflg;
245 static	uid_t	tuid = (uid_t)-1;
246 static	int	errflg;
247 
248 static	int	ndev;		/* number of devices */
249 static	int	maxdev;		/* number of devl structures allocated */
250 
251 #define	DNINCR	100
252 #define	DNSIZE	14
253 static struct devl {		/* device list   */
254 	char	dname[DNSIZE];	/* device name   */
255 	dev_t	ddev;		/* device number */
256 } *devl;
257 
258 static	struct tty {
259 	char *tname;
260 	dev_t tdev;
261 } *tty = NULL;			/* for t option */
262 static	size_t	ttysz = 0;
263 static	int	ntty = 0;
264 
265 static	pid_t	*pid = NULL;	/* for p option */
266 static	size_t	pidsz = 0;
267 static	size_t	npid = 0;
268 
269 static	int	*lgrps = NULL;	/* list of lgroup IDs for for h option */
270 static	size_t	lgrps_size = 0;	/* size of the lgrps list */
271 static	size_t	nlgrps = 0;	/* number elements in the list */
272 
273 /* Maximum possible lgroup ID value */
274 #define	MAX_LGRP_ID 256
275 
276 static	pid_t	*grpid = NULL;	/* for g option */
277 static	size_t	grpidsz = 0;
278 static	int	ngrpid = 0;
279 
280 static	pid_t	*sessid = NULL;	/* for s option */
281 static	size_t	sessidsz = 0;
282 static	int	nsessid = 0;
283 
284 static	zoneid_t *zoneid = NULL; /* for z option */
285 static	size_t	zoneidsz = 0;
286 static	int	nzoneid = 0;
287 
288 static	int	kbytes_per_page;
289 static	int	pidwidth;
290 
291 static	char	*procdir = "/proc";	/* standard /proc directory */
292 
293 static struct ughead	euid_tbl;	/* table to store selected euid's */
294 static struct ughead	ruid_tbl;	/* table to store selected real uid's */
295 static struct ughead	egid_tbl;	/* table to store selected egid's */
296 static struct ughead	rgid_tbl;	/* table to store selected real gid's */
297 static prheader_t *lpsinfobuf;		/* buffer to contain lpsinfo */
298 static size_t	lpbufsize;
299 
300 /*
301  * This constant defines the sentinal number of process IDs below which we
302  * only examine individual entries in /proc rather than scanning through
303  * /proc. This optimization is a huge win in the common case.
304  */
305 #define	PTHRESHOLD	40
306 
307 #define	UCB_OPTS	"-aceglnrtuvwxSU"
308 
309 static	void	usage(void);
310 static	char	*getarg(char **);
311 static	char	*parse_format(char *);
312 static	char	*gettty(psinfo_t *);
313 static	int	prfind(int, psinfo_t *, char **);
314 static	void	prcom(psinfo_t *, char *);
315 static	void	prtpct(ushort_t, int);
316 static	void	print_time(time_t, int);
317 static	void	print_field(psinfo_t *, struct field *, const char *);
318 static	void	print_zombie_field(psinfo_t *, struct field *, const char *);
319 static	void	pr_fields(psinfo_t *, const char *,
320 		void (*print_fld)(psinfo_t *, struct field *, const char *));
321 static	int	search(pid_t *, int, pid_t);
322 static	void	add_ugentry(struct ughead *, char *);
323 static	int	uconv(struct ughead *);
324 static	int	gconv(struct ughead *);
325 static	int	ugfind(id_t, struct ughead *);
326 static	void	prtime(timestruc_t, int, int);
327 static	void	przom(psinfo_t *);
328 static	int	namencnt(char *, int, int);
329 static	char	*err_string(int);
330 static	int	print_proc(char *pname);
331 static	time_t	delta_secs(const timestruc_t *);
332 static	int	str2id(const char *, pid_t *, long, long);
333 static	int	str2uid(const char *,  uid_t *, unsigned long, unsigned long);
334 static	void	*Realloc(void *, size_t);
335 static	int	pidcmp(const void *p1, const void *p2);
336 
337 extern	int	ucbmain(int, char **);
338 static	int	stdmain(int, char **);
339 
340 int
341 main(int argc, char **argv)
342 {
343 	const char *me;
344 
345 	/*
346 	 * The original two ps'es are linked in a single binary;
347 	 * their main()s are renamed to stdmain for /usr/bin/ps and
348 	 * ucbmain for /usr/ucb/ps.
349 	 * We try to figure out which instance of ps the user wants to run.
350 	 * Traditionally, the UCB variant doesn't require the flag argument
351 	 * start with a "-".  If the first argument doesn't start with a
352 	 * "-", we call "ucbmain".
353 	 * If there's a first argument and it starts with a "-", we check
354 	 * whether any of the options isn't acceptable to "ucbmain"; in that
355 	 * case we run "stdmain".
356 	 * If we can't tell from the options which main to call, we check
357 	 * the binary we are running.  We default to "stdmain" but
358 	 * any mention in the executable name of "ucb" causes us to call
359 	 * ucbmain.
360 	 */
361 	if (argv[1] != NULL) {
362 		if (argv[1][0] != '-')
363 			return (ucbmain(argc, argv));
364 		else if (argv[1][strspn(argv[1], UCB_OPTS)] != '\0')
365 			return (stdmain(argc, argv));
366 	}
367 
368 	me = getexecname();
369 
370 	if (me != NULL && strstr(me, "ucb") != NULL)
371 		return (ucbmain(argc, argv));
372 	else
373 		return (stdmain(argc, argv));
374 }
375 
376 static int
377 stdmain(int argc, char **argv)
378 {
379 	char	*p;
380 	char	*p1;
381 	char	*parg;
382 	int	c;
383 	int	i;
384 	int	pgerrflg = 0;	/* err flg: non-numeric arg w/p & g options */
385 	size_t	size, len;
386 	DIR	*dirp;
387 	struct dirent *dentp;
388 	pid_t	maxpid;
389 	pid_t	id;
390 	int	ret;
391 	char	loc_stime_str[32];
392 
393 	(void) setlocale(LC_ALL, "");
394 #if !defined(TEXT_DOMAIN)	/* Should be defined by cc -D */
395 #define	TEXT_DOMAIN	"SYS_TEST"	/* Use this only if it weren't */
396 #endif
397 	(void) textdomain(TEXT_DOMAIN);
398 
399 	(void) memset(&euid_tbl, 0, sizeof (euid_tbl));
400 	(void) memset(&ruid_tbl, 0, sizeof (ruid_tbl));
401 	(void) memset(&egid_tbl, 0, sizeof (egid_tbl));
402 	(void) memset(&rgid_tbl, 0, sizeof (rgid_tbl));
403 
404 	kbytes_per_page = sysconf(_SC_PAGESIZE) / 1024;
405 
406 	(void) gettimeofday(&now, NULL);
407 
408 	/*
409 	 * calculate width of pid fields based on configured MAXPID
410 	 * (must be at least 5 to retain output format compatibility)
411 	 */
412 	id = maxpid = (pid_t)sysconf(_SC_MAXPID);
413 	pidwidth = 1;
414 	while ((id /= 10) > 0)
415 		++pidwidth;
416 	pidwidth = pidwidth < 5 ? 5 : pidwidth;
417 
418 	fname[F_PID].width = fname[F_PPID].width = pidwidth;
419 	fname[F_PGID].width = fname[F_SID].width = pidwidth;
420 
421 	/*
422 	 * TRANSLATION_NOTE
423 	 * Specify the printf format with width and precision for
424 	 * the STIME field.
425 	 */
426 	len = snprintf(loc_stime_str, sizeof (loc_stime_str),
427 	    dcgettext(NULL, "%8.8s", LC_TIME), "STIME");
428 	if (len >= sizeof (loc_stime_str))
429 		len = sizeof (loc_stime_str) - 1;
430 
431 	fname[F_STIME].width = fname[F_STIME].minwidth = len;
432 
433 	while ((c = getopt(argc, argv, "jlfceAadLPWyZHh:t:p:g:u:U:G:n:s:o:z:"))
434 	    != EOF)
435 		switch (c) {
436 		case 'H':		/* Show home lgroups */
437 			Hflg++;
438 			break;
439 		case 'h':
440 			/*
441 			 * Show processes/threads with given home lgroups
442 			 */
443 			hflg++;
444 			p1 = optarg;
445 			do {
446 				int id;
447 
448 				/*
449 				 * Get all IDs in the list, verify for
450 				 * correctness and place in lgrps array.
451 				 */
452 				parg = getarg(&p1);
453 				/* Convert string to integer */
454 				ret = str2id(parg, (pid_t *)&id, 0,
455 				    MAX_LGRP_ID);
456 				/* Complain if ID didn't parse correctly */
457 				if (ret != 0) {
458 					pgerrflg++;
459 					(void) fprintf(stderr,
460 					    gettext("ps: %s "), parg);
461 					if (ret == EINVAL)
462 						(void) fprintf(stderr,
463 						    gettext("is an invalid "
464 						    "non-numeric argument"));
465 					else
466 						(void) fprintf(stderr,
467 						    gettext("exceeds valid "
468 						    "range"));
469 					(void) fprintf(stderr,
470 					    gettext(" for -h option\n"));
471 					continue;
472 				}
473 
474 				/* Extend lgrps array if needed */
475 				if (nlgrps == lgrps_size) {
476 					/* Double the size of the lgrps array */
477 					if (lgrps_size == 0)
478 						lgrps_size = SIZ;
479 					lgrps_size *= 2;
480 					lgrps = Realloc(lgrps,
481 					    lgrps_size * sizeof (int));
482 				}
483 				/* place the id in the lgrps table */
484 				lgrps[nlgrps++] = id;
485 			} while (*p1);
486 			break;
487 		case 'l':		/* long listing */
488 			lflg++;
489 			break;
490 		case 'f':		/* full listing */
491 			fflg++;
492 			break;
493 		case 'j':
494 			jflg++;
495 			break;
496 		case 'c':
497 			/*
498 			 * Format output to reflect scheduler changes:
499 			 * high numbers for high priorities and don't
500 			 * print nice or p_cpu values.  'c' option only
501 			 * effective when used with 'l' or 'f' options.
502 			 */
503 			cflg++;
504 			break;
505 		case 'A':		/* list every process */
506 		case 'e':		/* (obsolete) list every process */
507 			Aflg++;
508 			tflg = Gflg = Uflg = uflg = pflg = gflg = sflg = 0;
509 			zflg = hflg = 0;
510 			break;
511 		case 'a':
512 			/*
513 			 * Same as 'e' except no session group leaders
514 			 * and no non-terminal processes.
515 			 */
516 			aflg++;
517 			break;
518 		case 'd':	/* same as e except no session leaders */
519 			dflg++;
520 			break;
521 		case 'L':	/* show lwps */
522 			Lflg++;
523 			break;
524 		case 'P':	/* show bound processor */
525 			Pflg++;
526 			break;
527 		case 'W':	/* truncate long names */
528 			Wflg++;
529 			break;
530 		case 'y':	/* omit F & ADDR, report RSS & SZ in Kby */
531 			yflg++;
532 			break;
533 		case 'n':	/* no longer needed; retain as no-op */
534 			(void) fprintf(stderr,
535 			    gettext("ps: warning: -n option ignored\n"));
536 			break;
537 		case 't':		/* terminals */
538 #define	TSZ	30
539 			tflg++;
540 			p1 = optarg;
541 			do {
542 				char nambuf[TSZ+6];	/* for "/dev/" + '\0' */
543 				struct stat64 s;
544 				parg = getarg(&p1);
545 				p = Realloc(NULL, TSZ+1);	/* for '\0' */
546 				/* zero the buffer before using it */
547 				p[0] = '\0';
548 				size = TSZ;
549 				if (isdigit(*parg)) {
550 					(void) strcpy(p, "tty");
551 					size -= 3;
552 				}
553 				(void) strncat(p, parg, size);
554 				if (ntty == ttysz) {
555 					if ((ttysz *= 2) == 0)
556 						ttysz = NTTYS;
557 					tty = Realloc(tty,
558 					    (ttysz + 1) * sizeof (struct tty));
559 				}
560 				tty[ntty].tdev = PRNODEV;
561 				(void) strcpy(nambuf, "/dev/");
562 				(void) strcat(nambuf, p);
563 				if (stat64(nambuf, &s) == 0)
564 					tty[ntty].tdev = s.st_rdev;
565 				tty[ntty++].tname = p;
566 			} while (*p1);
567 			break;
568 		case 'p':		/* proc ids */
569 			pflg++;
570 			p1 = optarg;
571 			do {
572 				pid_t id;
573 
574 				parg = getarg(&p1);
575 				if ((ret = str2id(parg, &id, 0, maxpid)) != 0) {
576 					pgerrflg++;
577 					(void) fprintf(stderr,
578 					    gettext("ps: %s "), parg);
579 					if (ret == EINVAL)
580 						(void) fprintf(stderr,
581 						    gettext("is an invalid "
582 						    "non-numeric argument"));
583 					else
584 						(void) fprintf(stderr,
585 						    gettext("exceeds valid "
586 						    "range"));
587 					(void) fprintf(stderr,
588 					    gettext(" for -p option\n"));
589 					continue;
590 				}
591 
592 				if (npid == pidsz) {
593 					if ((pidsz *= 2) == 0)
594 						pidsz = SIZ;
595 					pid = Realloc(pid,
596 					    pidsz * sizeof (pid_t));
597 				}
598 				pid[npid++] = id;
599 			} while (*p1);
600 			break;
601 		case 's':		/* session */
602 			sflg++;
603 			p1 = optarg;
604 			do {
605 				pid_t id;
606 
607 				parg = getarg(&p1);
608 				if ((ret = str2id(parg, &id, 0, maxpid)) != 0) {
609 					pgerrflg++;
610 					(void) fprintf(stderr,
611 					    gettext("ps: %s "), parg);
612 					if (ret == EINVAL)
613 						(void) fprintf(stderr,
614 						    gettext("is an invalid "
615 						    "non-numeric argument"));
616 					else
617 						(void) fprintf(stderr,
618 						    gettext("exceeds valid "
619 						    "range"));
620 					(void) fprintf(stderr,
621 					    gettext(" for -s option\n"));
622 					continue;
623 				}
624 
625 				if (nsessid == sessidsz) {
626 					if ((sessidsz *= 2) == 0)
627 						sessidsz = SIZ;
628 					sessid = Realloc(sessid,
629 					    sessidsz * sizeof (pid_t));
630 				}
631 				sessid[nsessid++] = id;
632 			} while (*p1);
633 			break;
634 		case 'g':		/* proc group */
635 			gflg++;
636 			p1 = optarg;
637 			do {
638 				pid_t id;
639 
640 				parg = getarg(&p1);
641 				if ((ret = str2id(parg, &id, 0, maxpid)) != 0) {
642 					pgerrflg++;
643 					(void) fprintf(stderr,
644 					    gettext("ps: %s "), parg);
645 					if (ret == EINVAL)
646 						(void) fprintf(stderr,
647 						    gettext("is an invalid "
648 						    "non-numeric argument"));
649 					else
650 						(void) fprintf(stderr,
651 						    gettext("exceeds valid "
652 						    "range"));
653 					(void) fprintf(stderr,
654 					    gettext(" for -g option\n"));
655 					continue;
656 				}
657 
658 				if (ngrpid == grpidsz) {
659 					if ((grpidsz *= 2) == 0)
660 						grpidsz = SIZ;
661 					grpid = Realloc(grpid,
662 					    grpidsz * sizeof (pid_t));
663 				}
664 				grpid[ngrpid++] = id;
665 			} while (*p1);
666 			break;
667 		case 'u':		/* effective user name or number */
668 			uflg++;
669 			p1 = optarg;
670 			do {
671 				parg = getarg(&p1);
672 				add_ugentry(&euid_tbl, parg);
673 			} while (*p1);
674 			break;
675 		case 'U':		/* real user name or number */
676 			Uflg++;
677 			p1 = optarg;
678 			do {
679 				parg = getarg(&p1);
680 				add_ugentry(&ruid_tbl, parg);
681 			} while (*p1);
682 			break;
683 		case 'G':		/* real group name or number */
684 			Gflg++;
685 			p1 = optarg;
686 			do {
687 				parg = getarg(&p1);
688 				add_ugentry(&rgid_tbl, parg);
689 			} while (*p1);
690 			break;
691 		case 'o':		/* output format */
692 			p = optarg;
693 			while ((p = parse_format(p)) != NULL)
694 				;
695 			break;
696 		case 'z':		/* zone name or number */
697 			zflg++;
698 			p1 = optarg;
699 			do {
700 				zoneid_t id;
701 
702 				parg = getarg(&p1);
703 				if (zone_get_id(parg, &id) != 0) {
704 					pgerrflg++;
705 					(void) fprintf(stderr,
706 					    gettext("ps: unknown zone %s\n"),
707 					    parg);
708 					continue;
709 				}
710 
711 				if (nzoneid == zoneidsz) {
712 					if ((zoneidsz *= 2) == 0)
713 						zoneidsz = SIZ;
714 					zoneid = Realloc(zoneid,
715 					    zoneidsz * sizeof (zoneid_t));
716 				}
717 				zoneid[nzoneid++] = id;
718 			} while (*p1);
719 			break;
720 		case 'Z':		/* show zone name */
721 			Zflg++;
722 			break;
723 		default:			/* error on ? */
724 			errflg++;
725 			break;
726 		}
727 
728 	if (errflg || optind < argc || pgerrflg)
729 		usage();
730 
731 	if (tflg)
732 		tty[ntty].tname = NULL;
733 	/*
734 	 * If an appropriate option has not been specified, use the
735 	 * current terminal and effective uid as the default.
736 	 */
737 	if (!(aflg|Aflg|dflg|Gflg|hflg|Uflg|uflg|tflg|pflg|gflg|sflg|zflg)) {
738 		psinfo_t info;
739 		int procfd;
740 		char *name;
741 		char pname[100];
742 
743 		/* get our own controlling tty name using /proc */
744 		(void) snprintf(pname, sizeof (pname),
745 		    "%s/self/psinfo", procdir);
746 		if ((procfd = open(pname, O_RDONLY)) < 0 ||
747 		    read(procfd, (char *)&info, sizeof (info)) < 0 ||
748 		    info.pr_ttydev == PRNODEV) {
749 			(void) fprintf(stderr,
750 			    gettext("ps: no controlling terminal\n"));
751 			exit(1);
752 		}
753 		(void) close(procfd);
754 
755 		i = 0;
756 		name = gettty(&info);
757 		if (*name == '?') {
758 			(void) fprintf(stderr,
759 			    gettext("ps: can't find controlling terminal\n"));
760 			exit(1);
761 		}
762 		if (ntty == ttysz) {
763 			if ((ttysz *= 2) == 0)
764 				ttysz = NTTYS;
765 			tty = Realloc(tty, (ttysz + 1) * sizeof (struct tty));
766 		}
767 		tty[ntty].tdev = info.pr_ttydev;
768 		tty[ntty++].tname = name;
769 		tty[ntty].tname = NULL;
770 		tflg++;
771 		tuid = getuid();
772 	}
773 	if (Aflg) {
774 		Gflg = Uflg = uflg = pflg = sflg = gflg = aflg = dflg = 0;
775 		zflg = hflg = 0;
776 	}
777 	if (Aflg | aflg | dflg)
778 		tflg = 0;
779 
780 	i = 0;		/* prepare to exit on name lookup errors */
781 	i += uconv(&euid_tbl);
782 	i += uconv(&ruid_tbl);
783 	i += gconv(&egid_tbl);
784 	i += gconv(&rgid_tbl);
785 	if (i)
786 		exit(1);
787 
788 	/* allocate a buffer for lwpsinfo structures */
789 	lpbufsize = 4096;
790 	if (Lflg && (lpsinfobuf = malloc(lpbufsize)) == NULL) {
791 		(void) fprintf(stderr,
792 		    gettext("ps: no memory\n"));
793 		exit(1);
794 	}
795 
796 	if (fields) {	/* print user-specified header */
797 		if (do_header) {
798 			struct field *f;
799 
800 			for (f = fields; f != NULL; f = f->next) {
801 				if (f != fields)
802 					(void) printf(" ");
803 				switch (f->fname) {
804 				case F_TTY:
805 					(void) printf("%-*s",
806 					    f->width, f->header);
807 					break;
808 				case F_LWPNAME:
809 				case F_FNAME:
810 				case F_COMM:
811 				case F_ARGS:
812 					/*
813 					 * Print these headers full width
814 					 * unless they appear at the end.
815 					 */
816 					if (f->next != NULL) {
817 						(void) printf("%-*s",
818 						    f->width, f->header);
819 					} else {
820 						(void) printf("%s",
821 						    f->header);
822 					}
823 					break;
824 				default:
825 					(void) printf("%*s",
826 					    f->width, f->header);
827 					break;
828 				}
829 			}
830 			(void) printf("\n");
831 		}
832 	} else {	/* print standard header */
833 		/*
834 		 * All fields before 'PID' are printed with a trailing space
835 		 * as a separator and that is how we print the headers too.
836 		 */
837 		if (lflg) {
838 			if (yflg)
839 				(void) printf("S ");
840 			else
841 				(void) printf(" F S ");
842 		}
843 		if (Zflg)
844 			(void) printf("    ZONE ");
845 		if (fflg) {
846 			(void) printf("     UID ");
847 		} else if (lflg)
848 			(void) printf("   UID ");
849 
850 		(void) printf("%*s", pidwidth,  "PID");
851 		if (lflg || fflg)
852 			(void) printf(" %*s", pidwidth, "PPID");
853 		if (jflg)
854 			(void) printf(" %*s %*s", pidwidth, "PGID",
855 			    pidwidth, "SID");
856 		if (Lflg)
857 			(void) printf("   LWP");
858 		if (Pflg)
859 			(void) printf(" PSR");
860 		if (Lflg && fflg)
861 			(void) printf("  NLWP");
862 		if (cflg)
863 			(void) printf("  CLS PRI");
864 		else if (lflg || fflg) {
865 			(void) printf("   C");
866 			if (lflg)
867 				(void) printf(" PRI NI");
868 		}
869 		if (lflg) {
870 			if (yflg)
871 				(void) printf("   RSS     SZ    WCHAN");
872 			else
873 				(void) printf("     ADDR     SZ    WCHAN");
874 		}
875 		if (fflg)
876 			(void) printf(" %s", loc_stime_str);
877 		if (Hflg)
878 			(void) printf(" LGRP");
879 		if (Lflg)
880 			(void) printf(" TTY        LTIME CMD\n");
881 		else
882 			(void) printf(" TTY         TIME CMD\n");
883 	}
884 
885 
886 	if (pflg && !(aflg|Aflg|dflg|Gflg|Uflg|uflg|hflg|tflg|gflg|sflg|zflg) &&
887 	    npid <= PTHRESHOLD) {
888 		/*
889 		 * If we are looking at specific processes go straight
890 		 * to their /proc entries and don't scan /proc.
891 		 */
892 		int i;
893 
894 		(void) qsort(pid, npid, sizeof (pid_t), pidcmp);
895 		for (i = 0; i < npid; i++) {
896 			char pname[12];
897 
898 			if (i >= 1 && pid[i] == pid[i - 1])
899 				continue;
900 			(void) sprintf(pname, "%d", (int)pid[i]);
901 			if (print_proc(pname) == 0)
902 				retcode = 0;
903 		}
904 	} else {
905 		/*
906 		 * Determine which processes to print info about by searching
907 		 * the /proc directory and looking at each process.
908 		 */
909 		if ((dirp = opendir(procdir)) == NULL) {
910 			(void) fprintf(stderr,
911 			    gettext("ps: cannot open PROC directory %s\n"),
912 			    procdir);
913 			exit(1);
914 		}
915 
916 		/* for each active process --- */
917 		while ((dentp = readdir(dirp)) != NULL) {
918 			if (dentp->d_name[0] == '.')    /* skip . and .. */
919 				continue;
920 			if (print_proc(dentp->d_name) == 0)
921 				retcode = 0;
922 		}
923 
924 		(void) closedir(dirp);
925 	}
926 	return (retcode);
927 }
928 
929 
930 int
931 print_proc(char *pid_name)
932 {
933 	char	pname[PATH_MAX];
934 	int	pdlen;
935 	int	found;
936 	int	procfd; /* filedescriptor for /proc/nnnnn/psinfo */
937 	char	*tp;    /* ptr to ttyname,  if any */
938 	psinfo_t info;  /* process information from /proc */
939 	lwpsinfo_t *lwpsinfo;   /* array of lwpsinfo structs */
940 
941 	pdlen = snprintf(pname, sizeof (pname), "%s/%s/", procdir, pid_name);
942 	if (pdlen >= sizeof (pname) - 10)
943 		return (1);
944 retry:
945 	(void) strcpy(&pname[pdlen], "psinfo");
946 	if ((procfd = open(pname, O_RDONLY)) == -1) {
947 		/* Process may have exited meanwhile. */
948 		return (1);
949 	}
950 	/*
951 	 * Get the info structure for the process and close quickly.
952 	 */
953 	if (read(procfd, (char *)&info, sizeof (info)) < 0) {
954 		int	saverr = errno;
955 
956 		(void) close(procfd);
957 		if (saverr == EAGAIN)
958 			goto retry;
959 		if (saverr != ENOENT)
960 			(void) fprintf(stderr,
961 			    gettext("ps: read() on %s: %s\n"),
962 			    pname, err_string(saverr));
963 		return (1);
964 	}
965 	(void) close(procfd);
966 
967 	found = 0;
968 	if (info.pr_lwp.pr_state == 0)	/* can't happen? */
969 		return (1);
970 
971 	/*
972 	 * Omit session group leaders for 'a' and 'd' options.
973 	 */
974 	if ((info.pr_pid == info.pr_sid) && (dflg || aflg))
975 		return (1);
976 	if (Aflg || dflg)
977 		found++;
978 	else if (pflg && search(pid, npid, info.pr_pid))
979 		found++;	/* ppid in p option arg list */
980 	else if (uflg && ugfind((id_t)info.pr_euid, &euid_tbl))
981 		found++;	/* puid in u option arg list */
982 	else if (Uflg && ugfind((id_t)info.pr_uid, &ruid_tbl))
983 		found++;	/* puid in U option arg list */
984 #ifdef NOT_YET
985 	else if (gflg && ugfind((id_t)info.pr_egid, &egid_tbl))
986 		found++;	/* pgid in g option arg list */
987 #endif	/* NOT_YET */
988 	else if (Gflg && ugfind((id_t)info.pr_gid, &rgid_tbl))
989 		found++;	/* pgid in G option arg list */
990 	else if (gflg && search(grpid, ngrpid, info.pr_pgid))
991 		found++;	/* grpid in g option arg list */
992 	else if (sflg && search(sessid, nsessid, info.pr_sid))
993 		found++;	/* sessid in s option arg list */
994 	else if (zflg && search(zoneid, nzoneid, info.pr_zoneid))
995 		found++;	/* zoneid in z option arg list */
996 	else if (hflg && search((pid_t *)lgrps, nlgrps, info.pr_lwp.pr_lgrp))
997 		found++;	/* home lgroup in h option arg list */
998 	if (!found && !tflg && !aflg)
999 		return (1);
1000 	if (!prfind(found, &info, &tp))
1001 		return (1);
1002 	if (Lflg && (info.pr_nlwp + info.pr_nzomb) > 1) {
1003 		ssize_t prsz;
1004 
1005 		(void) strcpy(&pname[pdlen], "lpsinfo");
1006 		if ((procfd = open(pname, O_RDONLY)) == -1)
1007 			return (1);
1008 		/*
1009 		 * Get the info structures for the lwps.
1010 		 */
1011 		prsz = read(procfd, lpsinfobuf, lpbufsize);
1012 		if (prsz == -1) {
1013 			int	saverr = errno;
1014 
1015 			(void) close(procfd);
1016 			if (saverr == EAGAIN)
1017 				goto retry;
1018 			if (saverr != ENOENT)
1019 				(void) fprintf(stderr,
1020 				    gettext("ps: read() on %s: %s\n"),
1021 				    pname, err_string(saverr));
1022 			return (1);
1023 		}
1024 		(void) close(procfd);
1025 		if (prsz == lpbufsize) {
1026 			/*
1027 			 * buffer overflow. Realloc new buffer.
1028 			 * Error handling is done in Realloc().
1029 			 */
1030 			lpbufsize *= 2;
1031 			lpsinfobuf = Realloc(lpsinfobuf, lpbufsize);
1032 			goto retry;
1033 		}
1034 		if (lpsinfobuf->pr_nent != (info.pr_nlwp + info.pr_nzomb))
1035 			goto retry;
1036 		lwpsinfo = (lwpsinfo_t *)(lpsinfobuf + 1);
1037 	}
1038 	if (!Lflg || (info.pr_nlwp + info.pr_nzomb) <= 1) {
1039 		prcom(&info, tp);
1040 	} else {
1041 		int nlwp = 0;
1042 
1043 		do {
1044 			info.pr_lwp = *lwpsinfo;
1045 			prcom(&info, tp);
1046 			/* LINTED improper alignment */
1047 			lwpsinfo = (lwpsinfo_t *)((char *)lwpsinfo +
1048 			    lpsinfobuf->pr_entsize);
1049 		} while (++nlwp < lpsinfobuf->pr_nent);
1050 	}
1051 	return (0);
1052 }
1053 
1054 static int
1055 field_cmp(const void *l, const void *r)
1056 {
1057 	struct def_field *lhs = *((struct def_field **)l);
1058 	struct def_field *rhs = *((struct def_field **)r);
1059 
1060 	return (strcmp(lhs->fname, rhs->fname));
1061 }
1062 
1063 static void
1064 usage(void)		/* print usage message and quit */
1065 {
1066 	struct def_field *df, *sorted[NFIELDS];
1067 	int pos = 80, i = 0;
1068 
1069 	static char usage1[] =
1070 	    "ps [ -aAdefHlcjLPWyZ ] [ -o format ] [ -t termlist ]";
1071 	static char usage2[] =
1072 	    "\t[ -u userlist ] [ -U userlist ] [ -G grouplist ]";
1073 	static char usage3[] =
1074 	    "\t[ -p proclist ] [ -g pgrplist ] [ -s sidlist ]";
1075 	static char usage4[] =
1076 	    "\t[ -z zonelist ] [-h lgrplist]";
1077 	static char usage5[] =
1078 	    "  'format' is one or more of:";
1079 
1080 	(void) fprintf(stderr,
1081 	    gettext("usage: %s\n%s\n%s\n%s\n%s"),
1082 	    gettext(usage1), gettext(usage2), gettext(usage3),
1083 	    gettext(usage4), gettext(usage5));
1084 
1085 	/*
1086 	 * Now print out the possible output formats such that they neatly fit
1087 	 * into eighty columns.  Note that the fact that we are determining
1088 	 * this output programmatically means that a gettext() is impossible --
1089 	 * but it would be a mistake to localize the output formats anyway as
1090 	 * they are tokens for input, not output themselves.
1091 	 */
1092 	for (df = &fname[0]; df < &fname[NFIELDS]; df++)
1093 		sorted[i++] = df;
1094 
1095 	(void) qsort(sorted, NFIELDS, sizeof (void *), field_cmp);
1096 
1097 	for (i = 0; i < NFIELDS; i++) {
1098 		if (pos + strlen((df = sorted[i])->fname) + 1 >= 80) {
1099 			(void) fprintf(stderr, "\n\t");
1100 			pos = 8;
1101 		}
1102 
1103 		(void) fprintf(stderr, "%s%s", pos > 8 ? " " : "", df->fname);
1104 		pos += strlen(df->fname) + 1;
1105 	}
1106 
1107 	(void) fprintf(stderr, "\n");
1108 
1109 	exit(1);
1110 }
1111 
1112 /*
1113  * getarg() finds the next argument in list and copies arg into argbuf.
1114  * p1 first pts to arg passed back from getopt routine.  p1 is then
1115  * bumped to next character that is not a comma or blank -- p1 NULL
1116  * indicates end of list.
1117  */
1118 static char *
1119 getarg(char **pp1)
1120 {
1121 	static char argbuf[ARGSIZ];
1122 	char *p1 = *pp1;
1123 	char *parga = argbuf;
1124 	int c;
1125 
1126 	while ((c = *p1) != '\0' && (c == ',' || isspace(c)))
1127 		p1++;
1128 
1129 	while ((c = *p1) != '\0' && c != ',' && !isspace(c)) {
1130 		if (parga < argbuf + ARGSIZ - 1)
1131 			*parga++ = c;
1132 		p1++;
1133 	}
1134 	*parga = '\0';
1135 
1136 	while ((c = *p1) != '\0' && (c == ',' || isspace(c)))
1137 		p1++;
1138 
1139 	*pp1 = p1;
1140 
1141 	return (argbuf);
1142 }
1143 
1144 /*
1145  * parse_format() takes the argument to the -o option,
1146  * sets up the next output field structure, and returns
1147  * a pointer to any further output field specifier(s).
1148  * As a side-effect, it increments errflg if encounters a format error.
1149  */
1150 static char *
1151 parse_format(char *arg)
1152 {
1153 	int c;
1154 	char *name;
1155 	char *header = NULL;
1156 	int width = 0;
1157 	struct def_field *df;
1158 	struct field *f;
1159 
1160 	while ((c = *arg) != '\0' && (c == ',' || isspace(c)))
1161 		arg++;
1162 	if (c == '\0')
1163 		return (NULL);
1164 	name = arg;
1165 	arg = strpbrk(arg, " \t\r\v\f\n,=");
1166 	if (arg != NULL) {
1167 		c = *arg;
1168 		*arg++ = '\0';
1169 		if (c == '=') {
1170 			char *s;
1171 
1172 			header = arg;
1173 			arg = NULL;
1174 			width = strlen(header);
1175 			s = header + width;
1176 			while (s > header && isspace(*--s))
1177 				*s = '\0';
1178 			while (isspace(*header))
1179 				header++;
1180 		}
1181 	}
1182 	for (df = &fname[0]; df < &fname[NFIELDS]; df++)
1183 		if (strcmp(name, df->fname) == 0) {
1184 			if (strcmp(name, "lwp") == 0 ||
1185 			    strcmp(name, "lwpname") == 0)
1186 				Lflg++;
1187 			break;
1188 		}
1189 	if (df >= &fname[NFIELDS]) {
1190 		(void) fprintf(stderr,
1191 		    gettext("ps: unknown output format: -o %s\n"),
1192 		    name);
1193 		errflg++;
1194 		return (arg);
1195 	}
1196 	if ((f = malloc(sizeof (*f))) == NULL) {
1197 		(void) fprintf(stderr,
1198 		    gettext("ps: malloc() for output format failed, %s\n"),
1199 		    err_string(errno));
1200 		exit(1);
1201 	}
1202 	f->next = NULL;
1203 	f->fname = df - &fname[0];
1204 	f->header = header? header : df->header;
1205 	if (width == 0)
1206 		width = df->width;
1207 	if (*f->header != '\0')
1208 		do_header = 1;
1209 	f->width = max(width, df->minwidth);
1210 
1211 	if (fields == NULL)
1212 		fields = last_field = f;
1213 	else {
1214 		last_field->next = f;
1215 		last_field = f;
1216 	}
1217 
1218 	return (arg);
1219 }
1220 
1221 static char *
1222 devlookup(dev_t ddev)
1223 {
1224 	struct devl *dp;
1225 	int i;
1226 
1227 	for (dp = devl, i = 0; i < ndev; dp++, i++) {
1228 		if (dp->ddev == ddev)
1229 			return (dp->dname);
1230 	}
1231 	return (NULL);
1232 }
1233 
1234 static char *
1235 devadd(char *name, dev_t ddev)
1236 {
1237 	struct devl *dp;
1238 	int leng, start, i;
1239 
1240 	if (ndev == maxdev) {
1241 		maxdev += DNINCR;
1242 		devl = Realloc(devl, maxdev * sizeof (struct devl));
1243 	}
1244 	dp = &devl[ndev++];
1245 
1246 	dp->ddev = ddev;
1247 	if (name == NULL) {
1248 		(void) strcpy(dp->dname, "??");
1249 		return (dp->dname);
1250 	}
1251 
1252 	leng = strlen(name);
1253 	/* Strip off /dev/ */
1254 	if (leng < DNSIZE + 4)
1255 		(void) strcpy(dp->dname, &name[5]);
1256 	else {
1257 		start = leng - DNSIZE - 1;
1258 
1259 		for (i = start; i < leng && name[i] != '/'; i++)
1260 				;
1261 		if (i == leng)
1262 			(void) strncpy(dp->dname, &name[start], DNSIZE);
1263 		else
1264 			(void) strncpy(dp->dname, &name[i+1], DNSIZE);
1265 	}
1266 	return (dp->dname);
1267 }
1268 
1269 /*
1270  * gettty returns the user's tty number or ? if none.
1271  */
1272 static char *
1273 gettty(psinfo_t *psinfo)
1274 {
1275 	extern char *_ttyname_dev(dev_t, char *, size_t);
1276 	static zoneid_t zid = -1;
1277 	char devname[TTYNAME_MAX];
1278 	char *retval;
1279 
1280 	if (zid == -1)
1281 		zid = getzoneid();
1282 
1283 	if (psinfo->pr_ttydev == PRNODEV || psinfo->pr_zoneid != zid)
1284 		return ("?");
1285 
1286 	if ((retval = devlookup(psinfo->pr_ttydev)) != NULL)
1287 		return (retval);
1288 
1289 	retval = _ttyname_dev(psinfo->pr_ttydev, devname, sizeof (devname));
1290 
1291 	return (devadd(retval, psinfo->pr_ttydev));
1292 }
1293 
1294 /*
1295  * Find the process's tty and return 1 if process is to be printed.
1296  */
1297 static int
1298 prfind(int found, psinfo_t *psinfo, char **tpp)
1299 {
1300 	char	*tp;
1301 	struct tty *ttyp;
1302 
1303 	if (psinfo->pr_nlwp == 0) {
1304 		/* process is a zombie */
1305 		*tpp = "?";
1306 		if (tflg && !found)
1307 			return (0);
1308 		return (1);
1309 	}
1310 
1311 	/*
1312 	 * Get current terminal.  If none ("?") and 'a' is set, don't print
1313 	 * info.  If 't' is set, check if term is in list of desired terminals
1314 	 * and print it if it is.
1315 	 */
1316 	tp = gettty(psinfo);
1317 	if (aflg && *tp == '?') {
1318 		*tpp = tp;
1319 		return (0);
1320 	}
1321 	if (tflg && !found) {
1322 		int match = 0;
1323 		char *other = NULL;
1324 		for (ttyp = tty; ttyp->tname != NULL; ttyp++) {
1325 			/*
1326 			 * Look for a name match
1327 			 */
1328 			if (strcmp(tp, ttyp->tname) == 0) {
1329 				match = 1;
1330 				break;
1331 			}
1332 			/*
1333 			 * Look for same device under different names.
1334 			 */
1335 			if ((other == NULL) &&
1336 			    (ttyp->tdev != PRNODEV) &&
1337 			    (psinfo->pr_ttydev == ttyp->tdev))
1338 				other = ttyp->tname;
1339 		}
1340 		if (!match && (other != NULL)) {
1341 			/*
1342 			 * found under a different name
1343 			 */
1344 			match = 1;
1345 			tp = other;
1346 		}
1347 		if (!match || (tuid != (uid_t)-1 && tuid != psinfo->pr_euid)) {
1348 			/*
1349 			 * not found OR not matching euid
1350 			 */
1351 			*tpp = tp;
1352 			return (0);
1353 		}
1354 	}
1355 	*tpp = tp;
1356 	return (1);
1357 }
1358 
1359 /*
1360  * Print info about the process.
1361  */
1362 static void
1363 prcom(psinfo_t *psinfo, char *ttyp)
1364 {
1365 	char	*cp;
1366 	long	tm;
1367 	int	bytesleft;
1368 	int	wcnt, length;
1369 	wchar_t	wchar;
1370 	struct passwd *pwd;
1371 	int	zombie_lwp;
1372 	char	zonename[ZONENAME_MAX];
1373 
1374 	/*
1375 	 * If process is zombie, call zombie print routine and return.
1376 	 */
1377 	if (psinfo->pr_nlwp == 0) {
1378 		if (fields != NULL)
1379 			pr_fields(psinfo, ttyp, print_zombie_field);
1380 		else
1381 			przom(psinfo);
1382 		return;
1383 	}
1384 
1385 	zombie_lwp = (Lflg && psinfo->pr_lwp.pr_sname == 'Z');
1386 
1387 	/*
1388 	 * If user specified '-o format', print requested fields and return.
1389 	 */
1390 	if (fields != NULL) {
1391 		pr_fields(psinfo, ttyp, print_field);
1392 		return;
1393 	}
1394 
1395 	/*
1396 	 * All fields before 'PID' are printed with a trailing space as a
1397 	 * separator, rather than keeping track of which column is first.  All
1398 	 * other fields are printed with a leading space.
1399 	 */
1400 	if (lflg) {
1401 		if (!yflg)
1402 			(void) printf("%2x ", psinfo->pr_flag & 0377); /* F */
1403 		(void) printf("%c ", psinfo->pr_lwp.pr_sname);	/* S */
1404 	}
1405 
1406 	if (Zflg) {						/* ZONE */
1407 		if (getzonenamebyid(psinfo->pr_zoneid, zonename,
1408 		    sizeof (zonename)) < 0) {
1409 			if (snprintf(NULL, 0, "%d",
1410 			    ((int)psinfo->pr_zoneid)) > 7)
1411 				(void) printf(" %6.6d%c ",
1412 				    ((int)psinfo->pr_zoneid), '*');
1413 			else
1414 				(void) printf(" %7.7d ",
1415 				    ((int)psinfo->pr_zoneid));
1416 		} else {
1417 			size_t nw;
1418 
1419 			nw = mbstowcs(NULL, zonename, 0);
1420 			if (nw == (size_t)-1)
1421 				(void) printf("%8.8s ", "ERROR");
1422 			else if (nw > 8)
1423 				(void) wprintf(L"%7.7s%c ", zonename, '*');
1424 			else
1425 				(void) wprintf(L"%8.8s ", zonename);
1426 		}
1427 	}
1428 
1429 	if (fflg) {						/* UID */
1430 		if ((pwd = getpwuid(psinfo->pr_euid)) != NULL) {
1431 			size_t nw;
1432 
1433 			nw = mbstowcs(NULL, pwd->pw_name, 0);
1434 			if (nw == (size_t)-1)
1435 				(void) printf("%8.8s ", "ERROR");
1436 			else if (nw > 8)
1437 				(void) wprintf(L"%7.7s%c ", pwd->pw_name, '*');
1438 			else
1439 				(void) wprintf(L"%8.8s ", pwd->pw_name);
1440 		} else {
1441 			if (snprintf(NULL, 0, "%u",
1442 			    (psinfo->pr_euid)) > 7)
1443 				(void) printf(" %6.6u%c ", psinfo->pr_euid,
1444 				    '*');
1445 			else
1446 				(void) printf(" %7.7u ", psinfo->pr_euid);
1447 		}
1448 	} else if (lflg) {
1449 		if (snprintf(NULL, 0, "%u", (psinfo->pr_euid)) > 6)
1450 			(void) printf("%5.5u%c ", psinfo->pr_euid, '*');
1451 		else
1452 			(void) printf("%6u ", psinfo->pr_euid);
1453 	}
1454 	(void) printf("%*d", pidwidth, (int)psinfo->pr_pid); /* PID */
1455 	if (lflg || fflg)
1456 		(void) printf(" %*d", pidwidth,
1457 		    (int)psinfo->pr_ppid); /* PPID */
1458 	if (jflg) {
1459 		(void) printf(" %*d", pidwidth,
1460 		    (int)psinfo->pr_pgid);	/* PGID */
1461 		(void) printf(" %*d", pidwidth,
1462 		    (int)psinfo->pr_sid);	/* SID  */
1463 	}
1464 	if (Lflg)
1465 		(void) printf(" %5d", (int)psinfo->pr_lwp.pr_lwpid); /* LWP */
1466 	if (Pflg) {
1467 		if (psinfo->pr_lwp.pr_bindpro == PBIND_NONE)	/* PSR */
1468 			(void) printf("   -");
1469 		else
1470 			(void) printf(" %3d", psinfo->pr_lwp.pr_bindpro);
1471 	}
1472 	if (Lflg && fflg)					/* NLWP */
1473 		(void) printf(" %5d", psinfo->pr_nlwp + psinfo->pr_nzomb);
1474 	if (cflg) {
1475 		if (zombie_lwp)					/* CLS */
1476 			(void) printf("     ");
1477 		else
1478 			(void) printf(" %4s", psinfo->pr_lwp.pr_clname);
1479 		(void) printf(" %3d", psinfo->pr_lwp.pr_pri);	/* PRI */
1480 	} else if (lflg || fflg) {
1481 		(void) printf(" %3d", psinfo->pr_lwp.pr_cpu & 0377); /* C   */
1482 		if (lflg) {					    /* PRI NI */
1483 			/*
1484 			 * Print priorities the old way (lower numbers
1485 			 * mean higher priority) and print nice value
1486 			 * for time sharing procs.
1487 			 */
1488 			(void) printf(" %3d", psinfo->pr_lwp.pr_oldpri);
1489 			if (psinfo->pr_lwp.pr_oldpri != 0)
1490 				(void) printf(" %2d", psinfo->pr_lwp.pr_nice);
1491 			else
1492 				(void) printf(" %2.2s",
1493 				    psinfo->pr_lwp.pr_clname);
1494 		}
1495 	}
1496 	if (lflg) {
1497 		if (yflg) {
1498 			if (psinfo->pr_flag & SSYS)		/* RSS */
1499 				(void) printf("     0");
1500 			else if (psinfo->pr_rssize)
1501 				(void) printf(" %5lu",
1502 				    (ulong_t)psinfo->pr_rssize);
1503 			else
1504 				(void) printf("     ?");
1505 			if (psinfo->pr_flag & SSYS)		/* SZ */
1506 				(void) printf("      0");
1507 			else if (psinfo->pr_size)
1508 				(void) printf(" %6lu",
1509 				    (ulong_t)psinfo->pr_size);
1510 			else
1511 				(void) printf("      ?");
1512 		} else {
1513 #ifndef _LP64
1514 			if (psinfo->pr_addr)			/* ADDR */
1515 				(void) printf(" %8lx",
1516 				    (ulong_t)psinfo->pr_addr);
1517 			else
1518 #endif
1519 				(void) printf("        ?");
1520 			if (psinfo->pr_flag & SSYS)		/* SZ */
1521 				(void) printf("      0");
1522 			else if (psinfo->pr_size)
1523 				(void) printf(" %6lu",
1524 				    (ulong_t)psinfo->pr_size / kbytes_per_page);
1525 			else
1526 				(void) printf("      ?");
1527 		}
1528 		if (psinfo->pr_lwp.pr_sname != 'S')		/* WCHAN */
1529 			(void) printf("         ");
1530 #ifndef _LP64
1531 		else if (psinfo->pr_lwp.pr_wchan)
1532 			(void) printf(" %8lx",
1533 			    (ulong_t)psinfo->pr_lwp.pr_wchan);
1534 #endif
1535 		else
1536 			(void) printf("        ?");
1537 	}
1538 	if (fflg) {						/* STIME */
1539 		int width = fname[F_STIME].width;
1540 		if (Lflg)
1541 			prtime(psinfo->pr_lwp.pr_start, width + 1, 1);
1542 		else
1543 			prtime(psinfo->pr_start, width + 1, 1);
1544 	}
1545 
1546 	if (Hflg) {
1547 		/* Display home lgroup */
1548 		(void) printf(" %4d", (int)psinfo->pr_lwp.pr_lgrp);
1549 	}
1550 
1551 	(void) printf(" %-8.14s", ttyp);			/* TTY */
1552 	if (Lflg) {
1553 		tm = psinfo->pr_lwp.pr_time.tv_sec;
1554 		if (psinfo->pr_lwp.pr_time.tv_nsec > 500000000)
1555 			tm++;
1556 	} else {
1557 		tm = psinfo->pr_time.tv_sec;
1558 		if (psinfo->pr_time.tv_nsec > 500000000)
1559 			tm++;
1560 	}
1561 	(void) printf(" %4ld:%.2ld", tm / 60, tm % 60);		/* [L]TIME */
1562 
1563 	if (zombie_lwp) {
1564 		(void) printf(" <defunct>\n");
1565 		return;
1566 	}
1567 
1568 	if (!fflg) {						/* CMD */
1569 		wcnt = namencnt(psinfo->pr_fname, 16, 8);
1570 		(void) printf(" %.*s\n", wcnt, psinfo->pr_fname);
1571 		return;
1572 	}
1573 
1574 
1575 	/*
1576 	 * PRARGSZ == length of cmd arg string.
1577 	 */
1578 	psinfo->pr_psargs[PRARGSZ-1] = '\0';
1579 	bytesleft = PRARGSZ;
1580 	for (cp = psinfo->pr_psargs; *cp != '\0'; cp += length) {
1581 		length = mbtowc(&wchar, cp, MB_LEN_MAX);
1582 		if (length == 0)
1583 			break;
1584 		if (length < 0 || !iswprint(wchar)) {
1585 			if (length < 0)
1586 				length = 1;
1587 			if (bytesleft <= length) {
1588 				*cp = '\0';
1589 				break;
1590 			}
1591 			/* omit the unprintable character */
1592 			(void) memmove(cp, cp+length, bytesleft-length);
1593 			length = 0;
1594 		}
1595 		bytesleft -= length;
1596 	}
1597 	wcnt = namencnt(psinfo->pr_psargs, PRARGSZ, lflg ? 35 : PRARGSZ);
1598 	(void) printf(" %.*s\n", wcnt, psinfo->pr_psargs);
1599 }
1600 
1601 /*
1602  * Print percent from 16-bit binary fraction [0 .. 1]
1603  * Round up .01 to .1 to indicate some small percentage (the 0x7000 below).
1604  */
1605 static void
1606 prtpct(ushort_t pct, int width)
1607 {
1608 	uint_t value = pct;	/* need 32 bits to compute with */
1609 
1610 	value = ((value * 1000) + 0x7000) >> 15;	/* [0 .. 1000] */
1611 	if (value >= 1000)
1612 		value = 999;
1613 	if ((width -= 2) < 2)
1614 		width = 2;
1615 	(void) printf("%*u.%u", width, value / 10, value % 10);
1616 }
1617 
1618 static void
1619 print_time(time_t tim, int width)
1620 {
1621 	char buf[30];
1622 	time_t seconds;
1623 	time_t minutes;
1624 	time_t hours;
1625 	time_t days;
1626 
1627 	if (tim < 0) {
1628 		(void) printf("%*s", width, "-");
1629 		return;
1630 	}
1631 
1632 	seconds = tim % 60;
1633 	tim /= 60;
1634 	minutes = tim % 60;
1635 	tim /= 60;
1636 	hours = tim % 24;
1637 	days = tim / 24;
1638 
1639 	if (days > 0) {
1640 		(void) snprintf(buf, sizeof (buf), "%ld-%2.2ld:%2.2ld:%2.2ld",
1641 		    days, hours, minutes, seconds);
1642 	} else if (hours > 0) {
1643 		(void) snprintf(buf, sizeof (buf), "%2.2ld:%2.2ld:%2.2ld",
1644 		    hours, minutes, seconds);
1645 	} else {
1646 		(void) snprintf(buf, sizeof (buf), "%2.2ld:%2.2ld",
1647 		    minutes, seconds);
1648 	}
1649 
1650 	(void) printf("%*s", width, buf);
1651 }
1652 
1653 static void
1654 print_field(psinfo_t *psinfo, struct field *f, const char *ttyp)
1655 {
1656 	int width = f->width;
1657 	struct passwd *pwd;
1658 	struct group *grp;
1659 	time_t cputime;
1660 	int bytesleft;
1661 	int wcnt;
1662 	wchar_t	wchar;
1663 	char *cp;
1664 	int length;
1665 	ulong_t mask;
1666 	char c = '\0', *csave = NULL;
1667 	int zombie_lwp;
1668 
1669 	zombie_lwp = (Lflg && psinfo->pr_lwp.pr_sname == 'Z');
1670 
1671 	switch (f->fname) {
1672 	case F_RUSER:
1673 		if ((pwd = getpwuid(psinfo->pr_uid)) != NULL) {
1674 			size_t nw;
1675 
1676 			nw = mbstowcs(NULL, pwd->pw_name, 0);
1677 			if (nw == (size_t)-1)
1678 				(void) printf("%*s ", width, "ERROR");
1679 			else if (Wflg && nw > width)
1680 				(void) wprintf(L"%.*s%c", width - 1,
1681 				    pwd->pw_name, '*');
1682 			else
1683 				(void) wprintf(L"%*s", width, pwd->pw_name);
1684 		} else {
1685 			if (Wflg && snprintf(NULL, 0, "%u",
1686 			    (psinfo->pr_uid)) > width)
1687 
1688 				(void) printf("%*u%c", width - 1,
1689 				    psinfo->pr_uid, '*');
1690 			else
1691 				(void) printf("%*u", width, psinfo->pr_uid);
1692 		}
1693 		break;
1694 	case F_USER:
1695 		if ((pwd = getpwuid(psinfo->pr_euid)) != NULL) {
1696 			size_t nw;
1697 
1698 			nw = mbstowcs(NULL, pwd->pw_name, 0);
1699 			if (nw == (size_t)-1)
1700 				(void) printf("%*s ", width, "ERROR");
1701 			else if (Wflg && nw > width)
1702 				(void) wprintf(L"%.*s%c", width - 1,
1703 				    pwd->pw_name, '*');
1704 			else
1705 				(void) wprintf(L"%*s", width, pwd->pw_name);
1706 		} else {
1707 			if (Wflg && snprintf(NULL, 0, "%u",
1708 			    (psinfo->pr_euid)) > width)
1709 
1710 				(void) printf("%*u%c", width - 1,
1711 				    psinfo->pr_euid, '*');
1712 			else
1713 				(void) printf("%*u", width, psinfo->pr_euid);
1714 		}
1715 		break;
1716 	case F_RGROUP:
1717 		if ((grp = getgrgid(psinfo->pr_gid)) != NULL)
1718 			(void) printf("%*s", width, grp->gr_name);
1719 		else
1720 			(void) printf("%*u", width, psinfo->pr_gid);
1721 		break;
1722 	case F_GROUP:
1723 		if ((grp = getgrgid(psinfo->pr_egid)) != NULL)
1724 			(void) printf("%*s", width, grp->gr_name);
1725 		else
1726 			(void) printf("%*u", width, psinfo->pr_egid);
1727 		break;
1728 	case F_RUID:
1729 		(void) printf("%*u", width, psinfo->pr_uid);
1730 		break;
1731 	case F_UID:
1732 		(void) printf("%*u", width, psinfo->pr_euid);
1733 		break;
1734 	case F_RGID:
1735 		(void) printf("%*u", width, psinfo->pr_gid);
1736 		break;
1737 	case F_GID:
1738 		(void) printf("%*u", width, psinfo->pr_egid);
1739 		break;
1740 	case F_PID:
1741 		(void) printf("%*d", width, (int)psinfo->pr_pid);
1742 		break;
1743 	case F_PPID:
1744 		(void) printf("%*d", width, (int)psinfo->pr_ppid);
1745 		break;
1746 	case F_PGID:
1747 		(void) printf("%*d", width, (int)psinfo->pr_pgid);
1748 		break;
1749 	case F_SID:
1750 		(void) printf("%*d", width, (int)psinfo->pr_sid);
1751 		break;
1752 	case F_PSR:
1753 		if (zombie_lwp || psinfo->pr_lwp.pr_bindpro == PBIND_NONE)
1754 			(void) printf("%*s", width, "-");
1755 		else
1756 			(void) printf("%*d", width, psinfo->pr_lwp.pr_bindpro);
1757 		break;
1758 	case F_LWP:
1759 		(void) printf("%*d", width, (int)psinfo->pr_lwp.pr_lwpid);
1760 		break;
1761 	case F_LWPNAME: {
1762 		char lwpname[THREAD_NAME_MAX] = "";
1763 		char *path = NULL;
1764 		int fd;
1765 
1766 		if (asprintf(&path, "%s/%d/lwp/%d/lwpname", procdir,
1767 		    (int)psinfo->pr_pid, (int)psinfo->pr_lwp.pr_lwpid) != -1 &&
1768 		    (fd = open(path, O_RDONLY)) != -1) {
1769 			(void) read(fd, lwpname, sizeof (lwpname));
1770 			lwpname[THREAD_NAME_MAX - 1] = '\0';
1771 			(void) close(fd);
1772 		}
1773 
1774 		free(path);
1775 
1776 		if (f->next != NULL)
1777 			(void) printf("%-*s", width, lwpname);
1778 		else
1779 			(void) printf("%s", lwpname);
1780 		break;
1781 	}
1782 	case F_NLWP:
1783 		(void) printf("%*d", width, psinfo->pr_nlwp + psinfo->pr_nzomb);
1784 		break;
1785 	case F_OPRI:
1786 		if (zombie_lwp)
1787 			(void) printf("%*s", width, "-");
1788 		else
1789 			(void) printf("%*d", width, psinfo->pr_lwp.pr_oldpri);
1790 		break;
1791 	case F_PRI:
1792 		if (zombie_lwp)
1793 			(void) printf("%*s", width, "-");
1794 		else
1795 			(void) printf("%*d", width, psinfo->pr_lwp.pr_pri);
1796 		break;
1797 	case F_F:
1798 		mask = 0xffffffffUL;
1799 		if (width < 8)
1800 			mask >>= (8 - width) * 4;
1801 		(void) printf("%*lx", width, psinfo->pr_flag & mask);
1802 		break;
1803 	case F_S:
1804 		(void) printf("%*c", width, psinfo->pr_lwp.pr_sname);
1805 		break;
1806 	case F_C:
1807 		if (zombie_lwp)
1808 			(void) printf("%*s", width, "-");
1809 		else
1810 			(void) printf("%*d", width, psinfo->pr_lwp.pr_cpu);
1811 		break;
1812 	case F_PCPU:
1813 		if (zombie_lwp)
1814 			(void) printf("%*s", width, "-");
1815 		else if (Lflg)
1816 			prtpct(psinfo->pr_lwp.pr_pctcpu, width);
1817 		else
1818 			prtpct(psinfo->pr_pctcpu, width);
1819 		break;
1820 	case F_PMEM:
1821 		prtpct(psinfo->pr_pctmem, width);
1822 		break;
1823 	case F_OSZ:
1824 		(void) printf("%*lu", width,
1825 		    (ulong_t)psinfo->pr_size / kbytes_per_page);
1826 		break;
1827 	case F_VSZ:
1828 		(void) printf("%*lu", width, (ulong_t)psinfo->pr_size);
1829 		break;
1830 	case F_RSS:
1831 		(void) printf("%*lu", width, (ulong_t)psinfo->pr_rssize);
1832 		break;
1833 	case F_NICE:
1834 		/* if pr_oldpri is zero, then this class has no nice */
1835 		if (zombie_lwp)
1836 			(void) printf("%*s", width, "-");
1837 		else if (psinfo->pr_lwp.pr_oldpri != 0)
1838 			(void) printf("%*d", width, psinfo->pr_lwp.pr_nice);
1839 		else
1840 			(void) printf("%*.*s", width, width,
1841 			    psinfo->pr_lwp.pr_clname);
1842 		break;
1843 	case F_CLASS:
1844 		if (zombie_lwp)
1845 			(void) printf("%*s", width, "-");
1846 		else
1847 			(void) printf("%*.*s", width, width,
1848 			    psinfo->pr_lwp.pr_clname);
1849 		break;
1850 	case F_STIME:
1851 		if (Lflg)
1852 			prtime(psinfo->pr_lwp.pr_start, width, 0);
1853 		else
1854 			prtime(psinfo->pr_start, width, 0);
1855 		break;
1856 	case F_ETIME:
1857 		if (Lflg)
1858 			print_time(delta_secs(&psinfo->pr_lwp.pr_start),
1859 			    width);
1860 		else
1861 			print_time(delta_secs(&psinfo->pr_start), width);
1862 		break;
1863 	case F_TIME:
1864 		if (Lflg) {
1865 			cputime = psinfo->pr_lwp.pr_time.tv_sec;
1866 			if (psinfo->pr_lwp.pr_time.tv_nsec > 500000000)
1867 				cputime++;
1868 		} else {
1869 			cputime = psinfo->pr_time.tv_sec;
1870 			if (psinfo->pr_time.tv_nsec > 500000000)
1871 				cputime++;
1872 		}
1873 		print_time(cputime, width);
1874 		break;
1875 	case F_TTY:
1876 		(void) printf("%-*s", width, ttyp);
1877 		break;
1878 	case F_ADDR:
1879 		if (zombie_lwp)
1880 			(void) printf("%*s", width, "-");
1881 		else if (Lflg)
1882 			(void) printf("%*lx", width,
1883 			    (long)psinfo->pr_lwp.pr_addr);
1884 		else
1885 			(void) printf("%*lx", width, (long)psinfo->pr_addr);
1886 		break;
1887 	case F_WCHAN:
1888 		if (!zombie_lwp && psinfo->pr_lwp.pr_wchan)
1889 			(void) printf("%*lx", width,
1890 			    (long)psinfo->pr_lwp.pr_wchan);
1891 		else
1892 			(void) printf("%*.*s", width, width, "-");
1893 		break;
1894 	case F_FNAME:
1895 		/*
1896 		 * Print full width unless this is the last output format.
1897 		 */
1898 		if (zombie_lwp) {
1899 			if (f->next != NULL)
1900 				(void) printf("%-*s", width, "<defunct>");
1901 			else
1902 				(void) printf("%s", "<defunct>");
1903 			break;
1904 		}
1905 		wcnt = namencnt(psinfo->pr_fname, 16, width);
1906 		if (f->next != NULL)
1907 			(void) printf("%-*.*s", width, wcnt, psinfo->pr_fname);
1908 		else
1909 			(void) printf("%-.*s", wcnt, psinfo->pr_fname);
1910 		break;
1911 	case F_COMM:
1912 		if (zombie_lwp) {
1913 			if (f->next != NULL)
1914 				(void) printf("%-*s", width, "<defunct>");
1915 			else
1916 				(void) printf("%s", "<defunct>");
1917 			break;
1918 		}
1919 		csave = strpbrk(psinfo->pr_psargs, " \t\r\v\f\n");
1920 		if (csave) {
1921 			c = *csave;
1922 			*csave = '\0';
1923 		}
1924 		/* FALLTHROUGH */
1925 	case F_ARGS:
1926 		/*
1927 		 * PRARGSZ == length of cmd arg string.
1928 		 */
1929 		if (zombie_lwp) {
1930 			(void) printf("%-*s", width, "<defunct>");
1931 			break;
1932 		}
1933 		psinfo->pr_psargs[PRARGSZ-1] = '\0';
1934 		bytesleft = PRARGSZ;
1935 		for (cp = psinfo->pr_psargs; *cp != '\0'; cp += length) {
1936 			length = mbtowc(&wchar, cp, MB_LEN_MAX);
1937 			if (length == 0)
1938 				break;
1939 			if (length < 0 || !iswprint(wchar)) {
1940 				if (length < 0)
1941 					length = 1;
1942 				if (bytesleft <= length) {
1943 					*cp = '\0';
1944 					break;
1945 				}
1946 				/* omit the unprintable character */
1947 				(void) memmove(cp, cp+length, bytesleft-length);
1948 				length = 0;
1949 			}
1950 			bytesleft -= length;
1951 		}
1952 		wcnt = namencnt(psinfo->pr_psargs, PRARGSZ, width);
1953 		/*
1954 		 * Print full width unless this is the last format.
1955 		 */
1956 		if (f->next != NULL)
1957 			(void) printf("%-*.*s", width, wcnt,
1958 			    psinfo->pr_psargs);
1959 		else
1960 			(void) printf("%-.*s", wcnt,
1961 			    psinfo->pr_psargs);
1962 		if (f->fname == F_COMM && csave)
1963 			*csave = c;
1964 		break;
1965 	case F_TASKID:
1966 		(void) printf("%*d", width, (int)psinfo->pr_taskid);
1967 		break;
1968 	case F_PROJID:
1969 		(void) printf("%*d", width, (int)psinfo->pr_projid);
1970 		break;
1971 	case F_PROJECT:
1972 		{
1973 			struct project cproj;
1974 			char proj_buf[PROJECT_BUFSZ];
1975 
1976 			if ((getprojbyid(psinfo->pr_projid, &cproj,
1977 			    (void *)&proj_buf, PROJECT_BUFSZ)) == NULL) {
1978 				if (Wflg && snprintf(NULL, 0, "%d",
1979 				    ((int)psinfo->pr_projid)) > width)
1980 					(void) printf("%.*d%c", width - 1,
1981 					    ((int)psinfo->pr_projid), '*');
1982 				else
1983 					(void) printf("%*d", width,
1984 					    (int)psinfo->pr_projid);
1985 			} else {
1986 				size_t nw;
1987 
1988 				if (cproj.pj_name != NULL)
1989 					nw = mbstowcs(NULL, cproj.pj_name, 0);
1990 				if (cproj.pj_name == NULL)
1991 					(void) printf("%*s ", width, "---");
1992 				else if (nw == (size_t)-1)
1993 					(void) printf("%*s ", width, "ERROR");
1994 				else if (Wflg && nw > width)
1995 					(void) wprintf(L"%.*s%c", width - 1,
1996 					    cproj.pj_name, '*');
1997 				else
1998 					(void) wprintf(L"%*s", width,
1999 					    cproj.pj_name);
2000 			}
2001 		}
2002 		break;
2003 	case F_PSET:
2004 		if (zombie_lwp || psinfo->pr_lwp.pr_bindpset == PS_NONE)
2005 			(void) printf("%*s", width, "-");
2006 		else
2007 			(void) printf("%*d", width, psinfo->pr_lwp.pr_bindpset);
2008 		break;
2009 	case F_ZONEID:
2010 		(void) printf("%*d", width, (int)psinfo->pr_zoneid);
2011 		break;
2012 	case F_ZONE:
2013 		{
2014 			char zonename[ZONENAME_MAX];
2015 
2016 			if (getzonenamebyid(psinfo->pr_zoneid, zonename,
2017 			    sizeof (zonename)) < 0) {
2018 				if (Wflg && snprintf(NULL, 0, "%d",
2019 				    ((int)psinfo->pr_zoneid)) > width)
2020 					(void) printf("%.*d%c", width - 1,
2021 					    ((int)psinfo->pr_zoneid), '*');
2022 				else
2023 					(void) printf("%*d", width,
2024 					    (int)psinfo->pr_zoneid);
2025 			} else {
2026 				size_t nw;
2027 
2028 				nw = mbstowcs(NULL, zonename, 0);
2029 				if (nw == (size_t)-1)
2030 					(void) printf("%*s ", width, "ERROR");
2031 				else if (Wflg && nw > width)
2032 					(void) wprintf(L"%.*s%c", width - 1,
2033 					    zonename, '*');
2034 				else
2035 					(void) wprintf(L"%*s", width, zonename);
2036 			}
2037 		}
2038 		break;
2039 	case F_CTID:
2040 		if (psinfo->pr_contract == -1)
2041 			(void) printf("%*s", width, "-");
2042 		else
2043 			(void) printf("%*ld", width, (long)psinfo->pr_contract);
2044 		break;
2045 	case F_LGRP:
2046 		/* Display home lgroup */
2047 		(void) printf("%*d", width, (int)psinfo->pr_lwp.pr_lgrp);
2048 		break;
2049 
2050 	case F_DMODEL:
2051 		(void) printf("%*s", width,
2052 		    psinfo->pr_dmodel == PR_MODEL_LP64 ? "_LP64" : "_ILP32");
2053 		break;
2054 	}
2055 }
2056 
2057 static void
2058 print_zombie_field(psinfo_t *psinfo, struct field *f, const char *ttyp)
2059 {
2060 	int wcnt;
2061 	int width = f->width;
2062 
2063 	switch (f->fname) {
2064 	case F_FNAME:
2065 	case F_COMM:
2066 	case F_ARGS:
2067 		/*
2068 		 * Print full width unless this is the last output format.
2069 		 */
2070 		wcnt = min(width, sizeof ("<defunct>"));
2071 		if (f->next != NULL)
2072 			(void) printf("%-*.*s", width, wcnt, "<defunct>");
2073 		else
2074 			(void) printf("%-.*s", wcnt, "<defunct>");
2075 		break;
2076 
2077 	case F_PSR:
2078 	case F_PCPU:
2079 	case F_PMEM:
2080 	case F_NICE:
2081 	case F_CLASS:
2082 	case F_STIME:
2083 	case F_ETIME:
2084 	case F_WCHAN:
2085 	case F_PSET:
2086 		(void) printf("%*s", width, "-");
2087 		break;
2088 
2089 	case F_OPRI:
2090 	case F_PRI:
2091 	case F_OSZ:
2092 	case F_VSZ:
2093 	case F_RSS:
2094 		(void) printf("%*d", width, 0);
2095 		break;
2096 
2097 	default:
2098 		print_field(psinfo, f, ttyp);
2099 		break;
2100 	}
2101 }
2102 
2103 static void
2104 pr_fields(psinfo_t *psinfo, const char *ttyp,
2105     void (*print_fld)(psinfo_t *, struct field *, const char *))
2106 {
2107 	struct field *f;
2108 
2109 	for (f = fields; f != NULL; f = f->next) {
2110 		print_fld(psinfo, f, ttyp);
2111 		if (f->next != NULL)
2112 			(void) printf(" ");
2113 	}
2114 	(void) printf("\n");
2115 }
2116 
2117 /*
2118  * Returns 1 if arg is found in array arr, of length num; 0 otherwise.
2119  */
2120 static int
2121 search(pid_t *arr, int number, pid_t arg)
2122 {
2123 	int i;
2124 
2125 	for (i = 0; i < number; i++)
2126 		if (arg == arr[i])
2127 			return (1);
2128 	return (0);
2129 }
2130 
2131 /*
2132  * Add an entry (user, group) to the specified table.
2133  */
2134 static void
2135 add_ugentry(struct ughead *tbl, char *name)
2136 {
2137 	struct ugdata *entp;
2138 
2139 	if (tbl->size == tbl->nent) {	/* reallocate the table entries */
2140 		if ((tbl->size *= 2) == 0)
2141 			tbl->size = 32;		/* first time */
2142 		tbl->ent = Realloc(tbl->ent, tbl->size*sizeof (struct ugdata));
2143 	}
2144 	entp = &tbl->ent[tbl->nent++];
2145 	entp->id = 0;
2146 	(void) strncpy(entp->name, name, MAXUGNAME);
2147 	entp->name[MAXUGNAME] = '\0';
2148 }
2149 
2150 static int
2151 uconv(struct ughead *uhead)
2152 {
2153 	struct ugdata *utbl = uhead->ent;
2154 	int n = uhead->nent;
2155 	struct passwd *pwd;
2156 	int i;
2157 	int fnd = 0;
2158 	uid_t uid;
2159 
2160 	/*
2161 	 * Ask the name service for names.
2162 	 */
2163 	for (i = 0; i < n; i++) {
2164 		/*
2165 		 * If name is numeric, ask for numeric id
2166 		 */
2167 		if (str2uid(utbl[i].name, &uid, 0, MAXEPHUID) == 0)
2168 			pwd = getpwuid(uid);
2169 		else
2170 			pwd = getpwnam(utbl[i].name);
2171 
2172 		/*
2173 		 * If found, enter found index into tbl array.
2174 		 */
2175 		if (pwd == NULL) {
2176 			(void) fprintf(stderr,
2177 			    gettext("ps: unknown user %s\n"), utbl[i].name);
2178 			continue;
2179 		}
2180 
2181 		utbl[fnd].id = pwd->pw_uid;
2182 		(void) strncpy(utbl[fnd].name, pwd->pw_name, MAXUGNAME);
2183 		fnd++;
2184 	}
2185 
2186 	uhead->nent = fnd;	/* in case it changed */
2187 	return (n - fnd);
2188 }
2189 
2190 static int
2191 gconv(struct ughead *ghead)
2192 {
2193 	struct ugdata *gtbl = ghead->ent;
2194 	int n = ghead->nent;
2195 	struct group *grp;
2196 	gid_t gid;
2197 	int i;
2198 	int fnd = 0;
2199 
2200 	/*
2201 	 * Ask the name service for names.
2202 	 */
2203 	for (i = 0; i < n; i++) {
2204 		/*
2205 		 * If name is numeric, ask for numeric id
2206 		 */
2207 		if (str2uid(gtbl[i].name, (uid_t *)&gid, 0, MAXEPHUID) == 0)
2208 			grp = getgrgid(gid);
2209 		else
2210 			grp = getgrnam(gtbl[i].name);
2211 		/*
2212 		 * If found, enter found index into tbl array.
2213 		 */
2214 		if (grp == NULL) {
2215 			(void) fprintf(stderr,
2216 			    gettext("ps: unknown group %s\n"), gtbl[i].name);
2217 			continue;
2218 		}
2219 
2220 		gtbl[fnd].id = grp->gr_gid;
2221 		(void) strncpy(gtbl[fnd].name, grp->gr_name, MAXUGNAME);
2222 		fnd++;
2223 	}
2224 
2225 	ghead->nent = fnd;	/* in case it changed */
2226 	return (n - fnd);
2227 }
2228 
2229 /*
2230  * Return 1 if puid is in table, otherwise 0.
2231  */
2232 static int
2233 ugfind(id_t id, struct ughead *ughead)
2234 {
2235 	struct ugdata *utbl = ughead->ent;
2236 	int n = ughead->nent;
2237 	int i;
2238 
2239 	for (i = 0; i < n; i++)
2240 		if (utbl[i].id == id)
2241 			return (1);
2242 	return (0);
2243 }
2244 
2245 /*
2246  * Print starting time of process unless process started more than 24 hours
2247  * ago, in which case the date is printed.  The date is printed in the form
2248  * "MMM dd" if old format, else the blank is replaced with an '_' so
2249  * it appears as a single word (for parseability).
2250  */
2251 static void
2252 prtime(timestruc_t st, int width, int old)
2253 {
2254 	char sttim[26];
2255 	time_t starttime;
2256 
2257 	starttime = st.tv_sec;
2258 	if (st.tv_nsec > 500000000)
2259 		starttime++;
2260 	if ((now.tv_sec - starttime) >= 24*60*60) {
2261 		(void) strftime(sttim, sizeof (sttim), old?
2262 		/*
2263 		 * TRANSLATION_NOTE
2264 		 * This time format is used by STIME field when -f option
2265 		 * is specified.  Used for processes that begun more than
2266 		 * 24 hours.
2267 		 */
2268 		    dcgettext(NULL, "%b %d", LC_TIME) :
2269 		/*
2270 		 * TRANSLATION_NOTE
2271 		 * This time format is used by STIME field when -o option
2272 		 * is specified.  Used for processes that begun more than
2273 		 * 24 hours.
2274 		 */
2275 		    dcgettext(NULL, "%b_%d", LC_TIME), localtime(&starttime));
2276 	} else {
2277 		/*
2278 		 * TRANSLATION_NOTE
2279 		 * This time format is used by STIME field when -f or -o option
2280 		 * is specified.  Used for processes that begun less than
2281 		 * 24 hours.
2282 		 */
2283 		(void) strftime(sttim, sizeof (sttim),
2284 		    dcgettext(NULL, "%H:%M:%S", LC_TIME),
2285 		    localtime(&starttime));
2286 	}
2287 	(void) printf("%*.*s", width, width, sttim);
2288 }
2289 
2290 static void
2291 przom(psinfo_t *psinfo)
2292 {
2293 	long	tm;
2294 	struct passwd *pwd;
2295 	char zonename[ZONENAME_MAX];
2296 
2297 	/*
2298 	 * All fields before 'PID' are printed with a trailing space as a
2299 	 * spearator, rather than keeping track of which column is first.  All
2300 	 * other fields are printed with a leading space.
2301 	 */
2302 	if (lflg) {	/* F S */
2303 		if (!yflg)
2304 			(void) printf("%2x ", psinfo->pr_flag & 0377); /* F */
2305 		(void) printf("%c ", psinfo->pr_lwp.pr_sname);	/* S */
2306 	}
2307 	if (Zflg) {
2308 		if (getzonenamebyid(psinfo->pr_zoneid, zonename,
2309 		    sizeof (zonename)) < 0) {
2310 			if (snprintf(NULL, 0, "%d",
2311 			    ((int)psinfo->pr_zoneid)) > 7)
2312 				(void) printf(" %6.6d%c ",
2313 				    ((int)psinfo->pr_zoneid), '*');
2314 			else
2315 				(void) printf(" %7.7d ",
2316 				    ((int)psinfo->pr_zoneid));
2317 		} else {
2318 			size_t nw;
2319 
2320 			nw = mbstowcs(NULL, zonename, 0);
2321 			if (nw == (size_t)-1)
2322 				(void) printf("%8.8s ", "ERROR");
2323 			else if (nw > 8)
2324 				(void) wprintf(L"%7.7s%c ", zonename, '*');
2325 			else
2326 				(void) wprintf(L"%8.8s ", zonename);
2327 		}
2328 	}
2329 	if (Hflg) {
2330 		/* Display home lgroup */
2331 		(void) printf(" %6d", (int)psinfo->pr_lwp.pr_lgrp); /* LGRP */
2332 	}
2333 	if (fflg) {
2334 		if ((pwd = getpwuid(psinfo->pr_euid)) != NULL) {
2335 			size_t nw;
2336 
2337 			nw = mbstowcs(NULL, pwd->pw_name, 0);
2338 			if (nw == (size_t)-1)
2339 				(void) printf("%8.8s ", "ERROR");
2340 			else if (nw > 8)
2341 				(void) wprintf(L"%7.7s%c ", pwd->pw_name, '*');
2342 			else
2343 				(void) wprintf(L"%8.8s ", pwd->pw_name);
2344 		} else {
2345 			if (snprintf(NULL, 0, "%u",
2346 			    (psinfo->pr_euid)) > 7)
2347 				(void) printf(" %6.6u%c ", psinfo->pr_euid,
2348 				    '*');
2349 			else
2350 				(void) printf(" %7.7u ", psinfo->pr_euid);
2351 		}
2352 	} else if (lflg) {
2353 		if (snprintf(NULL, 0, "%u", (psinfo->pr_euid)) > 6)
2354 			(void) printf("%5.5u%c ", psinfo->pr_euid, '*');
2355 		else
2356 			(void) printf("%6u ", psinfo->pr_euid);
2357 	}
2358 
2359 	(void) printf("%*d", pidwidth, (int)psinfo->pr_pid);	/* PID */
2360 	if (lflg || fflg)
2361 		(void) printf(" %*d", pidwidth,
2362 		    (int)psinfo->pr_ppid);			/* PPID */
2363 
2364 	if (jflg) {
2365 		(void) printf(" %*d", pidwidth,
2366 		    (int)psinfo->pr_pgid);			/* PGID */
2367 		(void) printf(" %*d", pidwidth,
2368 		    (int)psinfo->pr_sid);			/* SID  */
2369 	}
2370 
2371 	if (Lflg)
2372 		(void) printf(" %5d", 0);			/* LWP */
2373 	if (Pflg)
2374 		(void) printf("   -");				/* PSR */
2375 	if (Lflg && fflg)
2376 		(void) printf(" %5d", 0);			/* NLWP */
2377 
2378 	if (cflg) {
2379 		(void) printf(" %4s", "-");	/* zombies have no class */
2380 		(void) printf(" %3d", psinfo->pr_lwp.pr_pri);	/* PRI	*/
2381 	} else if (lflg || fflg) {
2382 		(void) printf(" %3d", psinfo->pr_lwp.pr_cpu & 0377); /* C   */
2383 		if (lflg)
2384 			(void) printf(" %3d %2s",
2385 			    psinfo->pr_lwp.pr_oldpri, "-");	/* PRI NI */
2386 	}
2387 	if (lflg) {
2388 		if (yflg)				/* RSS SZ WCHAN */
2389 			(void) printf(" %5d %6d %8s", 0, 0, "-");
2390 		else					/* ADDR SZ WCHAN */
2391 			(void) printf(" %8s %6d %8s", "-", 0, "-");
2392 	}
2393 	if (fflg) {
2394 		int width = fname[F_STIME].width;
2395 		(void) printf(" %*.*s", width, width, "-");	/* STIME */
2396 	}
2397 	(void) printf(" %-8.14s", "?");				/* TTY */
2398 
2399 	tm = psinfo->pr_time.tv_sec;
2400 	if (psinfo->pr_time.tv_nsec > 500000000)
2401 		tm++;
2402 	(void) printf(" %4ld:%.2ld", tm / 60, tm % 60);	/* TIME */
2403 	(void) printf(" <defunct>\n");
2404 }
2405 
2406 /*
2407  * Function to compute the number of printable bytes in a multibyte
2408  * command string ("internationalization").
2409  */
2410 static int
2411 namencnt(char *cmd, int csisize, int scrsize)
2412 {
2413 	int csiwcnt = 0, scrwcnt = 0;
2414 	int ncsisz, nscrsz;
2415 	wchar_t  wchar;
2416 	int	 len;
2417 
2418 	while (*cmd != '\0') {
2419 		if ((len = csisize - csiwcnt) > (int)MB_CUR_MAX)
2420 			len = MB_CUR_MAX;
2421 		if ((ncsisz = mbtowc(&wchar, cmd, len)) < 0)
2422 			return (8); /* default to use for illegal chars */
2423 		if ((nscrsz = wcwidth(wchar)) <= 0)
2424 			return (8);
2425 		if (csiwcnt + ncsisz > csisize || scrwcnt + nscrsz > scrsize)
2426 			break;
2427 		csiwcnt += ncsisz;
2428 		scrwcnt += nscrsz;
2429 		cmd += ncsisz;
2430 	}
2431 	return (csiwcnt);
2432 }
2433 
2434 static char *
2435 err_string(int err)
2436 {
2437 	static char buf[32];
2438 	char *str = strerror(err);
2439 
2440 	if (str == NULL)
2441 		(void) snprintf(str = buf, sizeof (buf), "Errno #%d", err);
2442 
2443 	return (str);
2444 }
2445 
2446 /* If allocation fails, die */
2447 static void *
2448 Realloc(void *ptr, size_t size)
2449 {
2450 	ptr = realloc(ptr, size);
2451 	if (ptr == NULL) {
2452 		(void) fprintf(stderr, gettext("ps: no memory\n"));
2453 		exit(1);
2454 	}
2455 	return (ptr);
2456 }
2457 
2458 static time_t
2459 delta_secs(const timestruc_t *start)
2460 {
2461 	time_t seconds = now.tv_sec - start->tv_sec;
2462 	long nanosecs = now.tv_usec * 1000 - start->tv_nsec;
2463 
2464 	if (nanosecs >= (NANOSEC / 2))
2465 		seconds++;
2466 	else if (nanosecs < -(NANOSEC / 2))
2467 		seconds--;
2468 
2469 	return (seconds);
2470 }
2471 
2472 /*
2473  * Returns the following:
2474  *
2475  *	0	No error
2476  *	EINVAL	Invalid number
2477  *	ERANGE	Value exceeds (min, max) range
2478  */
2479 static int
2480 str2id(const char *p, pid_t *val, long min, long max)
2481 {
2482 	char *q;
2483 	long number;
2484 	int error;
2485 
2486 	errno = 0;
2487 	number = strtol(p, &q, 10);
2488 
2489 	if (errno != 0 || q == p || *q != '\0') {
2490 		if ((error = errno) == 0) {
2491 			/*
2492 			 * strtol() can fail without setting errno, or it can
2493 			 * set it to EINVAL or ERANGE.  In the case errno is
2494 			 * still zero, return EINVAL.
2495 			 */
2496 			error = EINVAL;
2497 		}
2498 	} else if (number < min || number > max) {
2499 		error = ERANGE;
2500 	} else {
2501 		error = 0;
2502 	}
2503 
2504 	*val = number;
2505 
2506 	return (error);
2507 }
2508 
2509 /*
2510  * Returns the following:
2511  *
2512  *	0	No error
2513  *	EINVAL	Invalid number
2514  *	ERANGE	Value exceeds (min, max) range
2515  */
2516 static int
2517 str2uid(const char *p, uid_t *val, unsigned long min, unsigned long max)
2518 {
2519 	char *q;
2520 	unsigned long number;
2521 	int error;
2522 
2523 	errno = 0;
2524 	number = strtoul(p, &q, 10);
2525 
2526 	if (errno != 0 || q == p || *q != '\0') {
2527 		if ((error = errno) == 0) {
2528 			/*
2529 			 * strtoul() can fail without setting errno, or it can
2530 			 * set it to EINVAL or ERANGE.  In the case errno is
2531 			 * still zero, return EINVAL.
2532 			 */
2533 			error = EINVAL;
2534 		}
2535 	} else if (number < min || number > max) {
2536 		error = ERANGE;
2537 	} else {
2538 		error = 0;
2539 	}
2540 
2541 	*val = number;
2542 
2543 	return (error);
2544 }
2545 
2546 static int
2547 pidcmp(const void *p1, const void *p2)
2548 {
2549 	pid_t i = *((pid_t *)p1);
2550 	pid_t j = *((pid_t *)p2);
2551 
2552 	return (i - j);
2553 }
2554