xref: /illumos-gate/usr/src/cmd/prstat/prstat.c (revision 78a542e2b33ce060556f645cb0084b07f96ce1eb)
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 2009 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  *
26  * Portions Copyright 2009 Chad Mynhier
27  */
28 
29 #include <sys/types.h>
30 #include <sys/resource.h>
31 #include <sys/loadavg.h>
32 #include <sys/time.h>
33 #include <sys/pset.h>
34 #include <sys/vm_usage.h>
35 #include <zone.h>
36 #include <libzonecfg.h>
37 
38 #include <stdio.h>
39 #include <stdlib.h>
40 #include <unistd.h>
41 #include <dirent.h>
42 #include <string.h>
43 #include <errno.h>
44 #include <poll.h>
45 #include <ctype.h>
46 #include <fcntl.h>
47 #include <limits.h>
48 #include <signal.h>
49 #include <time.h>
50 #include <project.h>
51 
52 #include <langinfo.h>
53 #include <libintl.h>
54 #include <locale.h>
55 
56 #include "prstat.h"
57 #include "prutil.h"
58 #include "prtable.h"
59 #include "prsort.h"
60 #include "prfile.h"
61 
62 /*
63  * x86 <sys/regs.h> ERR conflicts with <curses.h> ERR.  For the purposes
64  * of this file, we care about the curses.h ERR so include that last.
65  */
66 
67 #if	defined(ERR)
68 #undef	ERR
69 #endif
70 
71 #ifndef	TEXT_DOMAIN			/* should be defined by cc -D */
72 #define	TEXT_DOMAIN	"SYS_TEST"	/* use this only if it wasn't */
73 #endif
74 
75 #include <curses.h>
76 #include <term.h>
77 
78 #define	PSINFO_HEADER_PROC \
79 "   PID USERNAME  SIZE   RSS STATE  PRI NICE      TIME  CPU PROCESS/NLWP       "
80 #define	PSINFO_HEADER_PROC_LGRP \
81 "   PID USERNAME  SIZE   RSS STATE  PRI NICE      TIME  CPU LGRP PROCESS/NLWP  "
82 #define	PSINFO_HEADER_LWP \
83 "   PID USERNAME  SIZE   RSS STATE  PRI NICE      TIME  CPU PROCESS/LWPID      "
84 #define	PSINFO_HEADER_LWP_LGRP \
85 "   PID USERNAME  SIZE   RSS STATE  PRI NICE      TIME  CPU LGRP PROCESS/LWPID "
86 #define	USAGE_HEADER_PROC \
87 "   PID USERNAME USR SYS TRP TFL DFL LCK SLP LAT VCX ICX SCL SIG PROCESS/NLWP  "
88 #define	USAGE_HEADER_LWP \
89 "   PID USERNAME USR SYS TRP TFL DFL LCK SLP LAT VCX ICX SCL SIG PROCESS/LWPID "
90 #define	USER_HEADER_PROC \
91 " NPROC USERNAME  SWAP   RSS MEMORY      TIME  CPU                             "
92 #define	USER_HEADER_LWP \
93 "  NLWP USERNAME  SWAP   RSS MEMORY      TIME  CPU                             "
94 #define	TASK_HEADER_PROC \
95 "TASKID    NPROC  SWAP   RSS MEMORY      TIME  CPU PROJECT                     "
96 #define	TASK_HEADER_LWP \
97 "TASKID     NLWP  SWAP   RSS MEMORY      TIME  CPU PROJECT                     "
98 #define	PROJECT_HEADER_PROC \
99 "PROJID    NPROC  SWAP   RSS MEMORY      TIME  CPU PROJECT                     "
100 #define	PROJECT_HEADER_LWP \
101 "PROJID     NLWP  SWAP   RSS MEMORY      TIME  CPU PROJECT                     "
102 #define	ZONE_HEADER_PROC \
103 "ZONEID    NPROC  SWAP   RSS MEMORY      TIME  CPU ZONE                        "
104 #define	ZONE_HEADER_LWP \
105 "ZONEID     NLWP  SWAP   RSS MEMORY      TIME  CPU ZONE                        "
106 #define	PSINFO_LINE \
107 "%6d %-8s %5s %5s %-6s %3s  %3s %9s %3.3s%% %-.16s/%d"
108 #define	PSINFO_LINE_LGRP \
109 "%6d %-8s %5s %5s %-6s %3s  %3s %9s %3.3s%% %4d %-.16s/%d"
110 #define	USAGE_LINE \
111 "%6d %-8s %3.3s %3.3s %3.3s %3.3s %3.3s %3.3s %3.3s %3.3s %3.3s %3.3s %3.3s "\
112 "%3.3s %-.12s/%d"
113 #define	USER_LINE \
114 "%6d %-8s %5.5s %5.5s   %3.3s%% %9s %3.3s%%"
115 #define	TASK_LINE \
116 "%6d %8d %5s %5s   %3.3s%% %9s %3.3s%% %28s"
117 #define	PROJECT_LINE \
118 "%6d %8d %5s %5s   %3.3s%% %9s %3.3s%% %28s"
119 #define	ZONE_LINE \
120 "%6d %8d %5s %5s   %3.3s%% %9s %3.3s%% %28s"
121 
122 #define	TOTAL_LINE \
123 "Total: %d processes, %d lwps, load averages: %3.2f, %3.2f, %3.2f"
124 
125 /* global variables */
126 
127 static char	*t_ulon;			/* termcap: start underline */
128 static char	*t_uloff;			/* termcap: end underline */
129 static char	*t_up;				/* termcap: cursor 1 line up */
130 static char	*t_eol;				/* termcap: clear end of line */
131 static char	*t_smcup;			/* termcap: cursor mvcap on */
132 static char	*t_rmcup;			/* termcap: cursor mvcap off */
133 static char	*t_home;			/* termcap: move cursor home */
134 static char	*movecur = NULL;		/* termcap: move up string */
135 static char	*empty_string = "\0";		/* termcap: empty string */
136 static uint_t	print_movecur = FALSE;		/* print movecur or not */
137 static int	is_curses_on = FALSE;		/* current curses state */
138 
139 static table_t	pid_tbl = {0, 0, NULL};		/* selected processes */
140 static table_t	cpu_tbl = {0, 0, NULL};		/* selected processors */
141 static table_t  set_tbl = {0, 0, NULL};		/* selected processor sets */
142 static table_t	prj_tbl = {0, 0, NULL};		/* selected projects */
143 static table_t	tsk_tbl = {0, 0, NULL};		/* selected tasks */
144 static table_t	lgr_tbl = {0, 0, NULL};		/* selected lgroups */
145 static zonetbl_t zone_tbl = {0, 0, NULL};	/* selected zones */
146 static uidtbl_t euid_tbl = {0, 0, NULL}; 	/* selected effective users */
147 static uidtbl_t ruid_tbl = {0, 0, NULL}; 	/* selected real users */
148 
149 static uint_t	total_procs;			/* total number of procs */
150 static uint_t	total_lwps;			/* total number of lwps */
151 static float	total_cpu;			/* total cpu usage */
152 static float	total_mem;			/* total memory usage */
153 
154 static list_t	lwps;				/* list of lwps/processes */
155 static list_t	users;				/* list of users */
156 static list_t	tasks;				/* list of tasks */
157 static list_t	projects;			/* list of projects */
158 static list_t	zones;				/* list of zones */
159 static list_t	lgroups;			/* list of lgroups */
160 
161 static volatile uint_t sigwinch = 0;
162 static volatile uint_t sigtstp = 0;
163 static volatile uint_t sigterm = 0;
164 
165 static long pagesize;
166 
167 /* default settings */
168 
169 static optdesc_t opts = {
170 	5,			/* interval between updates, seconds */
171 	15,			/* number of lines in top part */
172 	5,			/* number of lines in bottom part */
173 	-1,			/* number of iterations; infinitely */
174 	OPT_PSINFO | OPT_FULLSCREEN | OPT_USEHOME | OPT_TERMCAP,
175 	-1			/* sort in decreasing order */
176 };
177 
178 /*
179  * Print timestamp as decimal reprentation of time_t value (-d u was specified)
180  * or the standard date format (-d d was specified).
181  */
182 static void
183 print_timestamp(void)
184 {
185 	time_t t = time(NULL);
186 	static char *fmt = NULL;
187 
188 	/* We only need to retrieve this once per invocation */
189 	if (fmt == NULL)
190 		fmt = nl_langinfo(_DATE_FMT);
191 
192 	if (opts.o_outpmode & OPT_UDATE) {
193 		(void) printf("%ld", t);
194 	} else if (opts.o_outpmode & OPT_DDATE) {
195 		char dstr[64];
196 		int len;
197 
198 		len = strftime(dstr, sizeof (dstr), fmt, localtime(&t));
199 		if (len > 0)
200 			(void) printf("%s", dstr);
201 	}
202 	(void) putp(t_eol);
203 	(void) putchar('\n');
204 }
205 
206 static void
207 psetloadavg(long psetid, void *ptr)
208 {
209 	double psetloadavg[3];
210 	double *loadavg = ptr;
211 
212 	if (pset_getloadavg((psetid_t)psetid, psetloadavg, 3) != -1) {
213 		*loadavg++ += psetloadavg[0];
214 		*loadavg++ += psetloadavg[1];
215 		*loadavg += psetloadavg[2];
216 	}
217 }
218 
219 /*
220  * Queries the memory virtual and rss size for each member of a list.
221  * This will override the values computed by /proc aggregation.
222  */
223 static void
224 list_getsize(list_t *list)
225 {
226 	id_info_t *id;
227 	vmusage_t *results, *next;
228 	vmusage_t *match;
229 	size_t nres = 0;
230 	size_t i;
231 	uint_t flags = 0;
232 	int ret;
233 	size_t physmem = sysconf(_SC_PHYS_PAGES) * pagesize;
234 
235 	/*
236 	 * Determine what swap/rss results to calculate.  getvmusage() will
237 	 * prune results returned to non-global zones automatically, so
238 	 * there is no need to pass different flags when calling from a
239 	 * non-global zone.
240 	 *
241 	 * Currently list_getsize() is only called with a single flag.  This
242 	 * is because -Z, -J, -T, and -a are mutually exclusive.  Regardless
243 	 * of this, we handle multiple flags.
244 	 */
245 	if (opts.o_outpmode & OPT_USERS) {
246 		/*
247 		 * Gather rss for all users in all zones.  Treat the same
248 		 * uid in different zones as the same user.
249 		 */
250 		flags |= VMUSAGE_COL_RUSERS;
251 
252 	} else if (opts.o_outpmode & OPT_TASKS) {
253 		/* Gather rss for all tasks in all zones */
254 		flags |= VMUSAGE_ALL_TASKS;
255 
256 	} else if (opts.o_outpmode & OPT_PROJECTS) {
257 		/*
258 		 * Gather rss for all projects in all zones.  Treat the same
259 		 * projid in diffrent zones as the same project.
260 		 */
261 		flags |= VMUSAGE_COL_PROJECTS;
262 
263 	} else if (opts.o_outpmode & OPT_ZONES) {
264 		/* Gather rss for all zones */
265 		flags |= VMUSAGE_ALL_ZONES;
266 
267 	} else {
268 		Die(gettext(
269 		    "Cannot determine rss flags for output options %x\n"),
270 		    opts.o_outpmode);
271 	}
272 
273 	/*
274 	 * getvmusage() returns an array of result structures.  One for
275 	 * each zone, project, task, or user on the system, depending on
276 	 * flags.
277 	 *
278 	 * If getvmusage() fails, prstat will use the size already gathered
279 	 * from psinfo
280 	 */
281 	if (getvmusage(flags, opts.o_interval, NULL, &nres) != 0)
282 		return;
283 
284 	results = (vmusage_t *)Malloc(sizeof (vmusage_t) * nres);
285 	for (;;) {
286 		ret = getvmusage(flags, opts.o_interval, results, &nres);
287 		if (ret == 0)
288 			break;
289 		if (errno == EOVERFLOW) {
290 			results = (vmusage_t *)Realloc(results,
291 			    sizeof (vmusage_t) * nres);
292 			continue;
293 		}
294 		/*
295 		 * Failure for some other reason.  Prstat will use the size
296 		 * already gathered from psinfo.
297 		 */
298 		free(results);
299 		return;
300 	}
301 	for (id = list->l_head; id != NULL; id = id->id_next) {
302 
303 		match = NULL;
304 		next = results;
305 		for (i = 0; i < nres; i++, next++) {
306 			switch (flags) {
307 			case VMUSAGE_COL_RUSERS:
308 				if (next->vmu_id == id->id_uid)
309 					match = next;
310 				break;
311 			case VMUSAGE_ALL_TASKS:
312 				if (next->vmu_id == id->id_taskid)
313 					match = next;
314 				break;
315 			case VMUSAGE_COL_PROJECTS:
316 				if (next->vmu_id == id->id_projid)
317 					match = next;
318 				break;
319 			case VMUSAGE_ALL_ZONES:
320 				if (next->vmu_id == id->id_zoneid)
321 					match = next;
322 				break;
323 			default:
324 				Die(gettext(
325 				    "Unknown vmusage flags %d\n"), flags);
326 			}
327 		}
328 		if (match != NULL) {
329 			id->id_size = match->vmu_swap_all / 1024;
330 			id->id_rssize = match->vmu_rss_all / 1024;
331 			id->id_pctmem = (100.0 * (float)match->vmu_rss_all) /
332 			    (float)physmem;
333 			/* Output using data from getvmusage() */
334 			id->id_sizematch = B_TRUE;
335 		}
336 		/*
337 		 * If no match is found, prstat will use the size already
338 		 * gathered from psinfo.
339 		 */
340 	}
341 	free(results);
342 }
343 
344 /*
345  * A routine to display the contents of the list on the screen
346  */
347 static void
348 list_print(list_t *list)
349 {
350 	lwp_info_t *lwp;
351 	id_info_t *id;
352 	char usr[4], sys[4], trp[4], tfl[4];
353 	char dfl[4], lck[4], slp[4], lat[4];
354 	char vcx[4], icx[4], scl[4], sig[4];
355 	char psize[6], prssize[6], pmem[6], pcpu[6], ptime[12];
356 	char pstate[7], pnice[4], ppri[4];
357 	char pname[LOGNAME_MAX+1];
358 	char projname[PROJNAME_MAX+1];
359 	char zonename[ZONENAME_MAX+1];
360 	float cpu, mem;
361 	double loadavg[3] = {0, 0, 0};
362 	int i, lwpid;
363 
364 	if (foreach_element(&set_tbl, &loadavg, psetloadavg) == 0) {
365 		/*
366 		 * If processor sets aren't specified, we display system-wide
367 		 * load averages.
368 		 */
369 		(void) getloadavg(loadavg, 3);
370 	}
371 
372 	if (((opts.o_outpmode & OPT_UDATE) || (opts.o_outpmode & OPT_DDATE)) &&
373 	    ((list->l_type == LT_LWPS) || !(opts.o_outpmode & OPT_SPLIT)))
374 		print_timestamp();
375 	if (opts.o_outpmode & OPT_TTY)
376 		(void) putchar('\r');
377 	(void) putp(t_ulon);
378 
379 	switch (list->l_type) {
380 	case LT_PROJECTS:
381 		if (opts.o_outpmode & OPT_LWPS)
382 			(void) printf(PROJECT_HEADER_LWP);
383 		else
384 			(void) printf(PROJECT_HEADER_PROC);
385 		break;
386 	case LT_TASKS:
387 		if (opts.o_outpmode & OPT_LWPS)
388 			(void) printf(TASK_HEADER_LWP);
389 		else
390 			(void) printf(TASK_HEADER_PROC);
391 		break;
392 	case LT_ZONES:
393 		if (opts.o_outpmode & OPT_LWPS)
394 			(void) printf(ZONE_HEADER_LWP);
395 		else
396 			(void) printf(ZONE_HEADER_PROC);
397 		break;
398 	case LT_USERS:
399 		if (opts.o_outpmode & OPT_LWPS)
400 			(void) printf(USER_HEADER_LWP);
401 		else
402 			(void) printf(USER_HEADER_PROC);
403 		break;
404 	case LT_LWPS:
405 		if (opts.o_outpmode & OPT_LWPS) {
406 			if (opts.o_outpmode & OPT_PSINFO) {
407 				if (opts.o_outpmode & OPT_LGRP)
408 					(void) printf(PSINFO_HEADER_LWP_LGRP);
409 				else
410 					(void) printf(PSINFO_HEADER_LWP);
411 			}
412 			if (opts.o_outpmode & OPT_MSACCT)
413 				(void) printf(USAGE_HEADER_LWP);
414 		} else {
415 			if (opts.o_outpmode & OPT_PSINFO) {
416 				if (opts.o_outpmode & OPT_LGRP)
417 					(void) printf(PSINFO_HEADER_PROC_LGRP);
418 				else
419 					(void) printf(PSINFO_HEADER_PROC);
420 			}
421 			if (opts.o_outpmode & OPT_MSACCT)
422 				(void) printf(USAGE_HEADER_PROC);
423 		}
424 		break;
425 	}
426 
427 	(void) putp(t_uloff);
428 	(void) putp(t_eol);
429 	(void) putchar('\n');
430 
431 	for (i = 0; i < list->l_used; i++) {
432 		switch (list->l_type) {
433 		case LT_PROJECTS:
434 		case LT_TASKS:
435 		case LT_USERS:
436 		case LT_ZONES:
437 			id = list->l_ptrs[i];
438 			/*
439 			 * CPU usage and memory usage normalization
440 			 */
441 			if (total_cpu >= 100)
442 				cpu = (100 * id->id_pctcpu) / total_cpu;
443 			else
444 				cpu = id->id_pctcpu;
445 			if (id->id_sizematch == B_FALSE && total_mem >= 100)
446 				mem = (100 * id->id_pctmem) / total_mem;
447 			else
448 				mem = id->id_pctmem;
449 			if (list->l_type == LT_USERS)
450 				pwd_getname(id->id_uid, pname, LOGNAME_MAX + 1,
451 				    opts.o_outpmode & OPT_NORESOLVE);
452 			else if (list->l_type == LT_ZONES)
453 				getzonename(id->id_zoneid, zonename,
454 				    ZONENAME_MAX);
455 			else
456 				getprojname(id->id_projid, projname,
457 				    PROJNAME_MAX,
458 				    opts.o_outpmode & OPT_NORESOLVE);
459 			Format_size(psize, id->id_size, 6);
460 			Format_size(prssize, id->id_rssize, 6);
461 			Format_pct(pmem, mem, 4);
462 			Format_pct(pcpu, cpu, 4);
463 			Format_time(ptime, id->id_time, 10);
464 			if (opts.o_outpmode & OPT_TTY)
465 				(void) putchar('\r');
466 			if (list->l_type == LT_PROJECTS)
467 				(void) printf(PROJECT_LINE, (int)id->id_projid,
468 				    id->id_nproc, psize, prssize, pmem, ptime,
469 				    pcpu, projname);
470 			else if (list->l_type == LT_TASKS)
471 				(void) printf(TASK_LINE, (int)id->id_taskid,
472 				    id->id_nproc, psize, prssize, pmem, ptime,
473 				    pcpu, projname);
474 			else if (list->l_type == LT_ZONES)
475 				(void) printf(ZONE_LINE, (int)id->id_zoneid,
476 				    id->id_nproc, psize, prssize, pmem, ptime,
477 				    pcpu, zonename);
478 			else
479 				(void) printf(USER_LINE, id->id_nproc, pname,
480 				    psize, prssize, pmem, ptime, pcpu);
481 			(void) putp(t_eol);
482 			(void) putchar('\n');
483 			break;
484 		case LT_LWPS:
485 			lwp = list->l_ptrs[i];
486 			if (opts.o_outpmode & OPT_LWPS)
487 				lwpid = lwp->li_info.pr_lwp.pr_lwpid;
488 			else
489 				lwpid = lwp->li_info.pr_nlwp +
490 				    lwp->li_info.pr_nzomb;
491 			pwd_getname(lwp->li_info.pr_uid, pname, LOGNAME_MAX + 1,
492 			    opts.o_outpmode & OPT_NORESOLVE);
493 			if (opts.o_outpmode & OPT_PSINFO) {
494 				Format_size(psize, lwp->li_info.pr_size, 6);
495 				Format_size(prssize, lwp->li_info.pr_rssize, 6);
496 				Format_state(pstate,
497 				    lwp->li_info.pr_lwp.pr_sname,
498 				    lwp->li_info.pr_lwp.pr_onpro, 7);
499 				if (strcmp(lwp->li_info.pr_lwp.pr_clname,
500 				    "RT") == 0 ||
501 				    strcmp(lwp->li_info.pr_lwp.pr_clname,
502 				    "SYS") == 0 ||
503 				    lwp->li_info.pr_lwp.pr_sname == 'Z')
504 					(void) strcpy(pnice, "  -");
505 				else
506 					Format_num(pnice,
507 					    lwp->li_info.pr_lwp.pr_nice - NZERO,
508 					    4);
509 				Format_num(ppri, lwp->li_info.pr_lwp.pr_pri, 4);
510 				Format_pct(pcpu,
511 				    FRC2PCT(lwp->li_info.pr_lwp.pr_pctcpu), 4);
512 				if (opts.o_outpmode & OPT_LWPS)
513 					Format_time(ptime,
514 					    lwp->li_info.pr_lwp.pr_time.tv_sec,
515 					    10);
516 				else
517 					Format_time(ptime,
518 					    lwp->li_info.pr_time.tv_sec, 10);
519 				if (opts.o_outpmode & OPT_TTY)
520 					(void) putchar('\r');
521 				stripfname(lwp->li_info.pr_fname);
522 				if (opts.o_outpmode & OPT_LGRP) {
523 					(void) printf(PSINFO_LINE_LGRP,
524 					    (int)lwp->li_info.pr_pid, pname,
525 					    psize, prssize, pstate, ppri, pnice,
526 					    ptime, pcpu,
527 					    (int)lwp->li_info.pr_lwp.pr_lgrp,
528 					    lwp->li_info.pr_fname, lwpid);
529 				} else {
530 					(void) printf(PSINFO_LINE,
531 					    (int)lwp->li_info.pr_pid, pname,
532 					    psize, prssize, pstate, ppri, pnice,
533 					    ptime, pcpu,
534 					    lwp->li_info.pr_fname, lwpid);
535 				}
536 				(void) putp(t_eol);
537 				(void) putchar('\n');
538 			}
539 			if (opts.o_outpmode & OPT_MSACCT) {
540 				Format_pct(usr, lwp->li_usr, 4);
541 				Format_pct(sys, lwp->li_sys, 4);
542 				Format_pct(slp, lwp->li_slp, 4);
543 				Format_num(vcx, lwp->li_vcx, 4);
544 				Format_num(icx, lwp->li_icx, 4);
545 				Format_num(scl, lwp->li_scl, 4);
546 				Format_num(sig, lwp->li_sig, 4);
547 				Format_pct(trp, lwp->li_trp, 4);
548 				Format_pct(tfl, lwp->li_tfl, 4);
549 				Format_pct(dfl, lwp->li_dfl, 4);
550 				Format_pct(lck, lwp->li_lck, 4);
551 				Format_pct(lat, lwp->li_lat, 4);
552 				if (opts.o_outpmode & OPT_TTY)
553 					(void) putchar('\r');
554 				stripfname(lwp->li_info.pr_fname);
555 				(void) printf(USAGE_LINE,
556 				    (int)lwp->li_info.pr_pid, pname,
557 				    usr, sys, trp, tfl, dfl, lck,
558 				    slp, lat, vcx, icx, scl, sig,
559 				    lwp->li_info.pr_fname, lwpid);
560 				(void) putp(t_eol);
561 				(void) putchar('\n');
562 			}
563 			break;
564 		}
565 	}
566 
567 	if (opts.o_outpmode & OPT_TTY)
568 		(void) putchar('\r');
569 	if (opts.o_outpmode & OPT_TERMCAP) {
570 		switch (list->l_type) {
571 		case LT_PROJECTS:
572 		case LT_USERS:
573 		case LT_TASKS:
574 		case LT_ZONES:
575 			while (i++ < opts.o_nbottom) {
576 				(void) putp(t_eol);
577 				(void) putchar('\n');
578 			}
579 			break;
580 		case LT_LWPS:
581 			while (i++ < opts.o_ntop) {
582 				(void) putp(t_eol);
583 				(void) putchar('\n');
584 			}
585 		}
586 	}
587 
588 	if (opts.o_outpmode & OPT_TTY)
589 		(void) putchar('\r');
590 
591 	if ((opts.o_outpmode & OPT_SPLIT) && list->l_type == LT_LWPS)
592 		return;
593 
594 	(void) printf(TOTAL_LINE, total_procs, total_lwps,
595 	    loadavg[LOADAVG_1MIN], loadavg[LOADAVG_5MIN],
596 	    loadavg[LOADAVG_15MIN]);
597 	(void) putp(t_eol);
598 	(void) putchar('\n');
599 	if (opts.o_outpmode & OPT_TTY)
600 		(void) putchar('\r');
601 	(void) putp(t_eol);
602 	(void) fflush(stdout);
603 }
604 
605 static lwp_info_t *
606 list_add_lwp(list_t *list, pid_t pid, id_t lwpid)
607 {
608 	lwp_info_t *lwp;
609 
610 	if (list->l_head == NULL) {
611 		list->l_head = list->l_tail = lwp = Zalloc(sizeof (lwp_info_t));
612 	} else {
613 		lwp = Zalloc(sizeof (lwp_info_t));
614 		lwp->li_prev = list->l_tail;
615 		((lwp_info_t *)list->l_tail)->li_next = lwp;
616 		list->l_tail = lwp;
617 	}
618 	lwp->li_info.pr_pid = pid;
619 	lwp->li_info.pr_lwp.pr_lwpid = lwpid;
620 	lwpid_add(lwp, pid, lwpid);
621 	list->l_count++;
622 	return (lwp);
623 }
624 
625 static void
626 list_remove_lwp(list_t *list, lwp_info_t *lwp)
627 {
628 	if (lwp->li_prev)
629 		lwp->li_prev->li_next = lwp->li_next;
630 	else
631 		list->l_head = lwp->li_next;	/* removing the head */
632 	if (lwp->li_next)
633 		lwp->li_next->li_prev = lwp->li_prev;
634 	else
635 		list->l_tail = lwp->li_prev;	/* removing the tail */
636 	lwpid_del(lwp->li_info.pr_pid, lwp->li_info.pr_lwp.pr_lwpid);
637 	if (lwpid_pidcheck(lwp->li_info.pr_pid) == 0)
638 		fds_rm(lwp->li_info.pr_pid);
639 	list->l_count--;
640 	free(lwp);
641 }
642 
643 static void
644 list_clear(list_t *list)
645 {
646 	if (list->l_type == LT_LWPS) {
647 		lwp_info_t	*lwp = list->l_tail;
648 		lwp_info_t	*lwp_tmp;
649 
650 		fd_closeall();
651 		while (lwp) {
652 			lwp_tmp = lwp;
653 			lwp = lwp->li_prev;
654 			list_remove_lwp(&lwps, lwp_tmp);
655 		}
656 	} else {
657 		id_info_t *id = list->l_head;
658 		id_info_t *nextid;
659 
660 		while (id) {
661 			nextid = id->id_next;
662 			free(id);
663 			id = nextid;
664 		}
665 		list->l_count = 0;
666 		list->l_head = list->l_tail = NULL;
667 	}
668 }
669 
670 static void
671 list_update(list_t *list, lwp_info_t *lwp)
672 {
673 	id_info_t *id;
674 
675 	if (list->l_head == NULL) {			/* first element */
676 		list->l_head = list->l_tail = id = Zalloc(sizeof (id_info_t));
677 		goto update;
678 	}
679 
680 	for (id = list->l_head; id; id = id->id_next) {
681 		if ((list->l_type == LT_USERS) &&
682 		    (id->id_uid != lwp->li_info.pr_uid))
683 			continue;
684 		if ((list->l_type == LT_TASKS) &&
685 		    (id->id_taskid != lwp->li_info.pr_taskid))
686 			continue;
687 		if ((list->l_type == LT_PROJECTS) &&
688 		    (id->id_projid != lwp->li_info.pr_projid))
689 			continue;
690 		if ((list->l_type == LT_ZONES) &&
691 		    (id->id_zoneid != lwp->li_info.pr_zoneid))
692 			continue;
693 		if ((list->l_type == LT_LGRPS) &&
694 		    (id->id_lgroup != lwp->li_info.pr_lwp.pr_lgrp))
695 			continue;
696 		id->id_nproc++;
697 		id->id_taskid	= lwp->li_info.pr_taskid;
698 		id->id_projid	= lwp->li_info.pr_projid;
699 		id->id_zoneid	= lwp->li_info.pr_zoneid;
700 		id->id_lgroup	= lwp->li_info.pr_lwp.pr_lgrp;
701 
702 		if (lwp->li_flags & LWP_REPRESENT) {
703 			id->id_size	+= lwp->li_info.pr_size;
704 			id->id_rssize	+= lwp->li_info.pr_rssize;
705 		}
706 		id->id_pctcpu	+= FRC2PCT(lwp->li_info.pr_lwp.pr_pctcpu);
707 		if (opts.o_outpmode & OPT_LWPS)
708 			id->id_time += TIME2SEC(lwp->li_info.pr_lwp.pr_time);
709 		else
710 			id->id_time += TIME2SEC(lwp->li_info.pr_time);
711 		id->id_pctmem	+= FRC2PCT(lwp->li_info.pr_pctmem);
712 		id->id_key	+= lwp->li_key;
713 		total_cpu	+= FRC2PCT(lwp->li_info.pr_lwp.pr_pctcpu);
714 		total_mem	+= FRC2PCT(lwp->li_info.pr_pctmem);
715 		return;
716 	}
717 
718 	id = list->l_tail;
719 	id->id_next = Zalloc(sizeof (id_info_t));
720 	id->id_next->id_prev = list->l_tail;
721 	id->id_next->id_next = NULL;
722 	list->l_tail = id->id_next;
723 	id = list->l_tail;
724 update:
725 	id->id_uid	= lwp->li_info.pr_uid;
726 	id->id_projid	= lwp->li_info.pr_projid;
727 	id->id_taskid	= lwp->li_info.pr_taskid;
728 	id->id_zoneid	= lwp->li_info.pr_zoneid;
729 	id->id_lgroup	= lwp->li_info.pr_lwp.pr_lgrp;
730 	id->id_nproc++;
731 	id->id_sizematch = B_FALSE;
732 	if (lwp->li_flags & LWP_REPRESENT) {
733 		id->id_size	= lwp->li_info.pr_size;
734 		id->id_rssize	= lwp->li_info.pr_rssize;
735 	}
736 	id->id_pctcpu	= FRC2PCT(lwp->li_info.pr_lwp.pr_pctcpu);
737 	if (opts.o_outpmode & OPT_LWPS)
738 		id->id_time = TIME2SEC(lwp->li_info.pr_lwp.pr_time);
739 	else
740 		id->id_time = TIME2SEC(lwp->li_info.pr_time);
741 	id->id_pctmem	= FRC2PCT(lwp->li_info.pr_pctmem);
742 	id->id_key	= lwp->li_key;
743 	total_cpu	+= id->id_pctcpu;
744 	total_mem	+= id->id_pctmem;
745 	list->l_count++;
746 }
747 
748 static void
749 lwp_update(lwp_info_t *lwp, pid_t pid, id_t lwpid, struct prusage *usage)
750 {
751 	float period;
752 
753 	if (!lwpid_is_active(pid, lwpid)) {
754 		/*
755 		 * If we are reading cpu times for the first time then
756 		 * calculate average cpu times based on whole process
757 		 * execution time.
758 		 */
759 		(void) memcpy(&lwp->li_usage, usage, sizeof (prusage_t));
760 		period = TIME2NSEC(usage->pr_rtime);
761 		period = period/(float)100;
762 
763 		if (period == 0) { /* zombie */
764 			period = 1;
765 			lwp->li_usr = 0;
766 			lwp->li_sys = 0;
767 			lwp->li_slp = 0;
768 		} else {
769 			lwp->li_usr = TIME2NSEC(usage->pr_utime)/period;
770 			lwp->li_sys = TIME2NSEC(usage->pr_stime)/period;
771 			lwp->li_slp = TIME2NSEC(usage->pr_slptime)/period;
772 		}
773 		lwp->li_trp = TIME2NSEC(usage->pr_ttime)/period;
774 		lwp->li_tfl = TIME2NSEC(usage->pr_tftime)/period;
775 		lwp->li_dfl = TIME2NSEC(usage->pr_dftime)/period;
776 		lwp->li_lck = TIME2NSEC(usage->pr_ltime)/period;
777 		lwp->li_lat = TIME2NSEC(usage->pr_wtime)/period;
778 		period = (period / NANOSEC)*(float)100; /* now in seconds */
779 		lwp->li_vcx = (ulong_t)
780 		    (opts.o_interval * (usage->pr_vctx/period));
781 		lwp->li_icx = (ulong_t)
782 		    (opts.o_interval * (usage->pr_ictx/period));
783 		lwp->li_scl = (ulong_t)
784 		    (opts.o_interval * (usage->pr_sysc/period));
785 		lwp->li_sig = (ulong_t)
786 		    (opts.o_interval * (usage->pr_sigs/period));
787 		(void) lwpid_set_active(pid, lwpid);
788 	} else {
789 		/*
790 		 * If this is not a first time we are reading a process's
791 		 * CPU times then recalculate CPU times based on fresh data
792 		 * obtained from procfs and previous CPU time usage values.
793 		 */
794 		period = TIME2NSEC(usage->pr_rtime)-
795 		    TIME2NSEC(lwp->li_usage.pr_rtime);
796 		period = period/(float)100;
797 
798 		if (period == 0) { /* zombie */
799 			period = 1;
800 			lwp->li_usr = 0;
801 			lwp->li_sys = 0;
802 			lwp->li_slp = 0;
803 		} else {
804 			lwp->li_usr = (TIME2NSEC(usage->pr_utime)-
805 			    TIME2NSEC(lwp->li_usage.pr_utime))/period;
806 			lwp->li_sys = (TIME2NSEC(usage->pr_stime) -
807 			    TIME2NSEC(lwp->li_usage.pr_stime))/period;
808 			lwp->li_slp = (TIME2NSEC(usage->pr_slptime) -
809 			    TIME2NSEC(lwp->li_usage.pr_slptime))/period;
810 		}
811 		lwp->li_trp = (TIME2NSEC(usage->pr_ttime) -
812 		    TIME2NSEC(lwp->li_usage.pr_ttime))/period;
813 		lwp->li_tfl = (TIME2NSEC(usage->pr_tftime) -
814 		    TIME2NSEC(lwp->li_usage.pr_tftime))/period;
815 		lwp->li_dfl = (TIME2NSEC(usage->pr_dftime) -
816 		    TIME2NSEC(lwp->li_usage.pr_dftime))/period;
817 		lwp->li_lck = (TIME2NSEC(usage->pr_ltime) -
818 		    TIME2NSEC(lwp->li_usage.pr_ltime))/period;
819 		lwp->li_lat = (TIME2NSEC(usage->pr_wtime) -
820 		    TIME2NSEC(lwp->li_usage.pr_wtime))/period;
821 		lwp->li_vcx = usage->pr_vctx - lwp->li_usage.pr_vctx;
822 		lwp->li_icx = usage->pr_ictx - lwp->li_usage.pr_ictx;
823 		lwp->li_scl = usage->pr_sysc - lwp->li_usage.pr_sysc;
824 		lwp->li_sig = usage->pr_sigs - lwp->li_usage.pr_sigs;
825 		(void) memcpy(&lwp->li_usage, usage, sizeof (prusage_t));
826 	}
827 }
828 
829 static int
830 read_procfile(fd_t **fd, char *pidstr, char *file, void *buf, size_t bufsize)
831 {
832 	char procfile[MAX_PROCFS_PATH];
833 
834 	(void) snprintf(procfile, MAX_PROCFS_PATH,
835 	    "/proc/%s/%s", pidstr, file);
836 	if ((*fd = fd_open(procfile, O_RDONLY, *fd)) == NULL)
837 		return (1);
838 	if (pread(fd_getfd(*fd), buf, bufsize, 0) != bufsize) {
839 		fd_close(*fd);
840 		return (1);
841 	}
842 	return (0);
843 }
844 
845 static void
846 add_proc(psinfo_t *psinfo)
847 {
848 	lwp_info_t *lwp;
849 	id_t lwpid;
850 	pid_t pid = psinfo->pr_pid;
851 
852 	lwpid = psinfo->pr_lwp.pr_lwpid;
853 	if ((lwp = lwpid_get(pid, lwpid)) == NULL)
854 		lwp = list_add_lwp(&lwps, pid, lwpid);
855 	lwp->li_flags |= LWP_ALIVE | LWP_REPRESENT;
856 	(void) memcpy(&lwp->li_info, psinfo, sizeof (psinfo_t));
857 	lwp->li_info.pr_lwp.pr_pctcpu = lwp->li_info.pr_pctcpu;
858 }
859 
860 static void
861 add_lwp(psinfo_t *psinfo, lwpsinfo_t *lwpsinfo, int flags)
862 {
863 	lwp_info_t *lwp;
864 	pid_t pid = psinfo->pr_pid;
865 	id_t lwpid = lwpsinfo->pr_lwpid;
866 
867 	if ((lwp = lwpid_get(pid, lwpid)) == NULL)
868 		lwp = list_add_lwp(&lwps, pid, lwpid);
869 	lwp->li_flags &= ~LWP_REPRESENT;
870 	lwp->li_flags |= LWP_ALIVE;
871 	lwp->li_flags |= flags;
872 	(void) memcpy(&lwp->li_info, psinfo,
873 	    sizeof (psinfo_t) - sizeof (lwpsinfo_t));
874 	(void) memcpy(&lwp->li_info.pr_lwp, lwpsinfo, sizeof (lwpsinfo_t));
875 }
876 
877 static void
878 prstat_scandir(DIR *procdir)
879 {
880 	char *pidstr;
881 	pid_t pid;
882 	id_t lwpid;
883 	size_t entsz;
884 	long nlwps, nent, i;
885 	char *buf, *ptr;
886 
887 	fds_t *fds;
888 	lwp_info_t *lwp;
889 	dirent_t *direntp;
890 
891 	prheader_t	header;
892 	psinfo_t	psinfo;
893 	prusage_t	usage;
894 	lwpsinfo_t	*lwpsinfo;
895 	prusage_t	*lwpusage;
896 
897 	total_procs = 0;
898 	total_lwps = 0;
899 	total_cpu = 0;
900 	total_mem = 0;
901 
902 	convert_zone(&zone_tbl);
903 	for (rewinddir(procdir); (direntp = readdir(procdir)); ) {
904 		pidstr = direntp->d_name;
905 		if (pidstr[0] == '.')	/* skip "." and ".."  */
906 			continue;
907 		pid = atoi(pidstr);
908 		if (pid == 0 || pid == 2 || pid == 3)
909 			continue;	/* skip sched, pageout and fsflush */
910 		if (has_element(&pid_tbl, pid) == 0)
911 			continue;	/* check if we really want this pid */
912 		fds = fds_get(pid);	/* get ptr to file descriptors */
913 
914 		if (read_procfile(&fds->fds_psinfo, pidstr,
915 		    "psinfo", &psinfo, sizeof (psinfo_t)) != 0)
916 			continue;
917 		if (!has_uid(&ruid_tbl, psinfo.pr_uid) ||
918 		    !has_uid(&euid_tbl, psinfo.pr_euid) ||
919 		    !has_element(&prj_tbl, psinfo.pr_projid) ||
920 		    !has_element(&tsk_tbl, psinfo.pr_taskid) ||
921 		    !has_zone(&zone_tbl, psinfo.pr_zoneid)) {
922 			fd_close(fds->fds_psinfo);
923 			continue;
924 		}
925 		nlwps = psinfo.pr_nlwp + psinfo.pr_nzomb;
926 
927 		if (nlwps > 1 && (opts.o_outpmode & (OPT_LWPS | OPT_PSETS))) {
928 			int rep_lwp = 0;
929 
930 			if (read_procfile(&fds->fds_lpsinfo, pidstr, "lpsinfo",
931 			    &header, sizeof (prheader_t)) != 0) {
932 				fd_close(fds->fds_psinfo);
933 				continue;
934 			}
935 
936 			nent = header.pr_nent;
937 			entsz = header.pr_entsize * nent;
938 			ptr = buf = Malloc(entsz);
939 			if (pread(fd_getfd(fds->fds_lpsinfo), buf,
940 			    entsz, sizeof (struct prheader)) != entsz) {
941 				fd_close(fds->fds_lpsinfo);
942 				fd_close(fds->fds_psinfo);
943 				free(buf);
944 				continue;
945 			}
946 
947 			nlwps = 0;
948 			for (i = 0; i < nent; i++, ptr += header.pr_entsize) {
949 				/*LINTED ALIGNMENT*/
950 				lwpsinfo = (lwpsinfo_t *)ptr;
951 				if (!has_element(&cpu_tbl,
952 				    lwpsinfo->pr_onpro) ||
953 				    !has_element(&set_tbl,
954 				    lwpsinfo->pr_bindpset) ||
955 				    !has_element(&lgr_tbl, lwpsinfo->pr_lgrp))
956 					continue;
957 				nlwps++;
958 				if ((opts.o_outpmode & (OPT_PSETS | OPT_LWPS))
959 				    == OPT_PSETS) {
960 					/*
961 					 * If one of process's LWPs is bound
962 					 * to a given processor set, report the
963 					 * whole process.  We may be doing this
964 					 * a few times but we'll get an accurate
965 					 * lwp count in return.
966 					 */
967 					add_proc(&psinfo);
968 				} else {
969 					if (rep_lwp == 0) {
970 						rep_lwp = 1;
971 						add_lwp(&psinfo, lwpsinfo,
972 						    LWP_REPRESENT);
973 					} else {
974 						add_lwp(&psinfo, lwpsinfo, 0);
975 					}
976 				}
977 			}
978 			free(buf);
979 			if (nlwps == 0) {
980 				fd_close(fds->fds_lpsinfo);
981 				fd_close(fds->fds_psinfo);
982 				continue;
983 			}
984 		} else {
985 			if (!has_element(&cpu_tbl, psinfo.pr_lwp.pr_onpro) ||
986 			    !has_element(&set_tbl, psinfo.pr_lwp.pr_bindpset) ||
987 			    !has_element(&lgr_tbl, psinfo.pr_lwp.pr_lgrp)) {
988 				fd_close(fds->fds_psinfo);
989 				continue;
990 			}
991 			add_proc(&psinfo);
992 		}
993 		if (!(opts.o_outpmode & OPT_MSACCT)) {
994 			total_procs++;
995 			total_lwps += nlwps;
996 			continue;
997 		}
998 		/*
999 		 * Get more information about processes from /proc/pid/usage.
1000 		 * If process has more than one lwp, then we may have to
1001 		 * also look at the /proc/pid/lusage file.
1002 		 */
1003 		if ((opts.o_outpmode & OPT_LWPS) && (nlwps > 1)) {
1004 			if (read_procfile(&fds->fds_lusage, pidstr, "lusage",
1005 			    &header, sizeof (prheader_t)) != 0) {
1006 				fd_close(fds->fds_lpsinfo);
1007 				fd_close(fds->fds_psinfo);
1008 				continue;
1009 			}
1010 			nent = header.pr_nent;
1011 			entsz = header.pr_entsize * nent;
1012 			buf = Malloc(entsz);
1013 			if (pread(fd_getfd(fds->fds_lusage), buf,
1014 			    entsz, sizeof (struct prheader)) != entsz) {
1015 				fd_close(fds->fds_lusage);
1016 				fd_close(fds->fds_lpsinfo);
1017 				fd_close(fds->fds_psinfo);
1018 				free(buf);
1019 				continue;
1020 			}
1021 			for (i = 1, ptr = buf + header.pr_entsize; i < nent;
1022 			    i++, ptr += header.pr_entsize) {
1023 				/*LINTED ALIGNMENT*/
1024 				lwpusage = (prusage_t *)ptr;
1025 				lwpid = lwpusage->pr_lwpid;
1026 				/*
1027 				 * New LWPs created after we read lpsinfo
1028 				 * will be ignored.  Don't want to do
1029 				 * everything all over again.
1030 				 */
1031 				if ((lwp = lwpid_get(pid, lwpid)) == NULL)
1032 					continue;
1033 				lwp_update(lwp, pid, lwpid, lwpusage);
1034 			}
1035 			free(buf);
1036 		} else {
1037 			if (read_procfile(&fds->fds_usage, pidstr, "usage",
1038 			    &usage, sizeof (prusage_t)) != 0) {
1039 				fd_close(fds->fds_lpsinfo);
1040 				fd_close(fds->fds_psinfo);
1041 				continue;
1042 			}
1043 			lwpid = psinfo.pr_lwp.pr_lwpid;
1044 			if ((lwp = lwpid_get(pid, lwpid)) == NULL)
1045 				continue;
1046 			lwp_update(lwp, pid, lwpid, &usage);
1047 		}
1048 		total_procs++;
1049 		total_lwps += nlwps;
1050 	}
1051 	fd_update();
1052 }
1053 
1054 /*
1055  * This procedure removes all dead lwps from the linked list of all lwps.
1056  * It also creates linked list of ids if necessary.
1057  */
1058 static void
1059 list_refresh(list_t *list)
1060 {
1061 	lwp_info_t *lwp, *lwp_next;
1062 
1063 	if (!(list->l_type & LT_LWPS))
1064 		return;
1065 
1066 	for (lwp = list->l_head; lwp != NULL; ) {
1067 		if (lwp->li_flags & LWP_ALIVE) {
1068 			/*
1069 			 * Process all live LWPs.
1070 			 * When we're done, mark them as dead.
1071 			 * They will be marked "alive" on the next
1072 			 * /proc scan if they still exist.
1073 			 */
1074 			lwp->li_key = list_getkeyval(list, lwp);
1075 			if (opts.o_outpmode & OPT_USERS)
1076 				list_update(&users, lwp);
1077 			if (opts.o_outpmode & OPT_TASKS)
1078 				list_update(&tasks, lwp);
1079 			if (opts.o_outpmode & OPT_PROJECTS)
1080 				list_update(&projects, lwp);
1081 			if (opts.o_outpmode & OPT_ZONES)
1082 				list_update(&zones, lwp);
1083 			if (opts.o_outpmode & OPT_LGRP)
1084 				list_update(&lgroups, lwp);
1085 			lwp->li_flags &= ~LWP_ALIVE;
1086 			lwp = lwp->li_next;
1087 
1088 		} else {
1089 			lwp_next = lwp->li_next;
1090 			list_remove_lwp(&lwps, lwp);
1091 			lwp = lwp_next;
1092 		}
1093 	}
1094 }
1095 
1096 static void
1097 curses_on()
1098 {
1099 	if ((opts.o_outpmode & OPT_TERMCAP) && (is_curses_on == FALSE)) {
1100 		(void) initscr();
1101 		(void) nonl();
1102 		(void) putp(t_smcup);
1103 		is_curses_on = TRUE;
1104 	}
1105 }
1106 
1107 static void
1108 curses_off()
1109 {
1110 	if ((is_curses_on == TRUE) && (opts.o_outpmode & OPT_TERMCAP)) {
1111 		(void) putp(t_rmcup);
1112 		(void) endwin();
1113 		is_curses_on = FALSE;
1114 	}
1115 	(void) fflush(stdout);
1116 }
1117 
1118 static int
1119 nlines()
1120 {
1121 	struct winsize ws;
1122 	char *envp;
1123 	int n;
1124 	if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &ws) != -1) {
1125 		if (ws.ws_row > 0)
1126 			return (ws.ws_row);
1127 	}
1128 	if (envp = getenv("LINES")) {
1129 		if ((n = Atoi(envp)) > 0) {
1130 			opts.o_outpmode &= ~OPT_USEHOME;
1131 			return (n);
1132 		}
1133 	}
1134 	return (-1);
1135 }
1136 
1137 static void
1138 setmovecur()
1139 {
1140 	int i, n;
1141 	if ((opts.o_outpmode & OPT_FULLSCREEN) &&
1142 	    (opts.o_outpmode & OPT_USEHOME)) {
1143 		movecur = t_home;
1144 		return;
1145 	}
1146 	if (opts.o_outpmode & OPT_SPLIT) {
1147 		n = opts.o_ntop + opts.o_nbottom + 2;
1148 	} else {
1149 		if (opts.o_outpmode & OPT_USERS)
1150 			n = opts.o_nbottom + 1;
1151 		else
1152 			n = opts.o_ntop + 1;
1153 	}
1154 	if (((opts.o_outpmode & OPT_UDATE) || (opts.o_outpmode & OPT_DDATE)))
1155 		n++;
1156 
1157 	if (movecur != NULL && movecur != empty_string && movecur != t_home)
1158 		free(movecur);
1159 	movecur = Zalloc(strlen(t_up) * (n + 5));
1160 	for (i = 0; i <= n; i++)
1161 		(void) strcat(movecur, t_up);
1162 }
1163 
1164 static int
1165 setsize()
1166 {
1167 	static int oldn = 0;
1168 	int n;
1169 
1170 	if (opts.o_outpmode & OPT_FULLSCREEN) {
1171 		n = nlines();
1172 		if (n == oldn)
1173 			return (0);
1174 		oldn = n;
1175 		if (n == -1) {
1176 			opts.o_outpmode &= ~OPT_USEHOME;
1177 			setmovecur();		/* set default window size */
1178 			return (1);
1179 		}
1180 		n = n - 3;	/* minus header, total and cursor lines */
1181 		if ((opts.o_outpmode & OPT_UDATE) ||
1182 		    (opts.o_outpmode & OPT_DDATE))
1183 			n--;	/* minus timestamp */
1184 		if (n < 1)
1185 			Die(gettext("window is too small (try -n)\n"));
1186 		if (opts.o_outpmode & OPT_SPLIT) {
1187 			if (n < 8) {
1188 				Die(gettext("window is too small (try -n)\n"));
1189 			} else {
1190 				opts.o_ntop = (n / 4) * 3;
1191 				opts.o_nbottom = n - 1 - opts.o_ntop;
1192 			}
1193 		} else {
1194 			if (opts.o_outpmode & OPT_USERS)
1195 				opts.o_nbottom = n;
1196 			else
1197 				opts.o_ntop = n;
1198 		}
1199 	}
1200 	setmovecur();
1201 	return (1);
1202 }
1203 
1204 static void
1205 ldtermcap()
1206 {
1207 	int err;
1208 	if (setupterm(NULL, STDIN_FILENO, &err) == ERR) {
1209 		switch (err) {
1210 		case 0:
1211 			Warn(gettext("failed to load terminal info, "
1212 			    "defaulting to -c option\n"));
1213 			break;
1214 		case -1:
1215 			Warn(gettext("terminfo database not found, "
1216 			    "defaulting to -c option\n"));
1217 			break;
1218 		default:
1219 			Warn(gettext("failed to initialize terminal, "
1220 			    "defaulting to -c option\n"));
1221 		}
1222 		opts.o_outpmode &= ~OPT_TERMCAP;
1223 		t_up = t_eol = t_smcup = t_rmcup = movecur = empty_string;
1224 		t_ulon = t_uloff = empty_string;
1225 		return;
1226 	}
1227 	t_ulon	= tigetstr("smul");
1228 	t_uloff	= tigetstr("rmul");
1229 	t_up	= tigetstr("cuu1");
1230 	t_eol	= tigetstr("el");
1231 	t_smcup	= tigetstr("smcup");
1232 	t_rmcup = tigetstr("rmcup");
1233 	t_home  = tigetstr("home");
1234 	if ((t_up == (char *)-1) || (t_eol == (char *)-1) ||
1235 	    (t_smcup == (char *)-1) || (t_rmcup == (char *)-1)) {
1236 		opts.o_outpmode &= ~OPT_TERMCAP;
1237 		t_up = t_eol = t_smcup = t_rmcup = movecur = empty_string;
1238 		return;
1239 	}
1240 	if (t_up == NULL || t_eol == NULL) {
1241 		opts.o_outpmode &= ~OPT_TERMCAP;
1242 		t_eol = t_up = movecur = empty_string;
1243 		return;
1244 	}
1245 	if (t_ulon == (char *)-1 || t_uloff == (char *)-1 ||
1246 	    t_ulon == NULL || t_uloff == NULL) {
1247 		t_ulon = t_uloff = empty_string;  /* can live without it */
1248 	}
1249 	if (t_smcup == NULL || t_rmcup == NULL)
1250 		t_smcup = t_rmcup = empty_string;
1251 	if (t_home == (char *)-1 || t_home == NULL) {
1252 		opts.o_outpmode &= ~OPT_USEHOME;
1253 		t_home = empty_string;
1254 	}
1255 }
1256 
1257 static void
1258 sig_handler(int sig)
1259 {
1260 	switch (sig) {
1261 	case SIGTSTP:	sigtstp = 1;
1262 			break;
1263 	case SIGWINCH:	sigwinch = 1;
1264 			break;
1265 	case SIGINT:
1266 	case SIGTERM:	sigterm = 1;
1267 			break;
1268 	}
1269 }
1270 
1271 static void
1272 set_signals()
1273 {
1274 	(void) signal(SIGTSTP, sig_handler);
1275 	(void) signal(SIGINT, sig_handler);
1276 	(void) signal(SIGTERM, sig_handler);
1277 	if (opts.o_outpmode & OPT_FULLSCREEN)
1278 		(void) signal(SIGWINCH, sig_handler);
1279 }
1280 
1281 static void
1282 fill_table(table_t *table, char *arg, char option)
1283 {
1284 	char *p = strtok(arg, ", ");
1285 
1286 	if (p == NULL)
1287 		Die(gettext("invalid argument for -%c\n"), option);
1288 
1289 	add_element(table, (long)Atoi(p));
1290 	while (p = strtok(NULL, ", "))
1291 		add_element(table, (long)Atoi(p));
1292 }
1293 
1294 static void
1295 fill_prj_table(char *arg)
1296 {
1297 	projid_t projid;
1298 	char *p = strtok(arg, ", ");
1299 
1300 	if (p == NULL)
1301 		Die(gettext("invalid argument for -j\n"));
1302 
1303 	if ((projid = getprojidbyname(p)) == -1)
1304 		projid = Atoi(p);
1305 	add_element(&prj_tbl, (long)projid);
1306 
1307 	while (p = strtok(NULL, ", ")) {
1308 		if ((projid = getprojidbyname(p)) == -1)
1309 			projid = Atoi(p);
1310 		add_element(&prj_tbl, (long)projid);
1311 	}
1312 }
1313 
1314 static void
1315 fill_set_table(char *arg)
1316 {
1317 	char *p = strtok(arg, ", ");
1318 	psetid_t id;
1319 
1320 	if (p == NULL)
1321 		Die(gettext("invalid argument for -C\n"));
1322 
1323 	if ((id = Atoi(p)) == 0)
1324 		id = PS_NONE;
1325 	add_element(&set_tbl, id);
1326 	while (p = strtok(NULL, ", ")) {
1327 		if ((id = Atoi(p)) == 0)
1328 			id = PS_NONE;
1329 		if (!has_element(&set_tbl, id))
1330 			add_element(&set_tbl, id);
1331 	}
1332 }
1333 
1334 static void
1335 Exit()
1336 {
1337 	curses_off();
1338 	list_clear(&lwps);
1339 	list_clear(&users);
1340 	list_clear(&tasks);
1341 	list_clear(&projects);
1342 	list_clear(&zones);
1343 	fd_exit();
1344 }
1345 
1346 
1347 int
1348 main(int argc, char **argv)
1349 {
1350 	DIR *procdir;
1351 	char *p;
1352 	char *sortk = "cpu";	/* default sort key */
1353 	int opt;
1354 	int timeout;
1355 	struct pollfd pollset;
1356 	char key;
1357 
1358 	(void) setlocale(LC_ALL, "");
1359 	(void) textdomain(TEXT_DOMAIN);
1360 	Progname(argv[0]);
1361 	lwpid_init();
1362 	fd_init(Setrlimit());
1363 
1364 	pagesize = sysconf(_SC_PAGESIZE);
1365 
1366 	while ((opt = getopt(argc, argv,
1367 	    "vcd:HmarRLtu:U:n:p:C:P:h:s:S:j:k:TJz:Z")) != (int)EOF) {
1368 		switch (opt) {
1369 		case 'r':
1370 			opts.o_outpmode |= OPT_NORESOLVE;
1371 			break;
1372 		case 'R':
1373 			opts.o_outpmode |= OPT_REALTIME;
1374 			break;
1375 		case 'c':
1376 			opts.o_outpmode &= ~OPT_TERMCAP;
1377 			opts.o_outpmode &= ~OPT_FULLSCREEN;
1378 			break;
1379 		case 'd':
1380 			if (optarg) {
1381 				if (*optarg == 'u')
1382 					opts.o_outpmode |= OPT_UDATE;
1383 				else if (*optarg == 'd')
1384 					opts.o_outpmode |= OPT_DDATE;
1385 				else
1386 					Usage();
1387 			} else {
1388 				Usage();
1389 			}
1390 			break;
1391 		case 'h':
1392 			fill_table(&lgr_tbl, optarg, 'h');
1393 			break;
1394 		case 'H':
1395 			opts.o_outpmode |= OPT_LGRP;
1396 			break;
1397 		case 'm':
1398 		case 'v':
1399 			opts.o_outpmode &= ~OPT_PSINFO;
1400 			opts.o_outpmode |=  OPT_MSACCT;
1401 			break;
1402 		case 't':
1403 			opts.o_outpmode &= ~OPT_PSINFO;
1404 			opts.o_outpmode |= OPT_USERS;
1405 			break;
1406 		case 'a':
1407 			opts.o_outpmode |= OPT_SPLIT | OPT_USERS;
1408 			break;
1409 		case 'T':
1410 			opts.o_outpmode |= OPT_SPLIT | OPT_TASKS;
1411 			break;
1412 		case 'J':
1413 			opts.o_outpmode |= OPT_SPLIT | OPT_PROJECTS;
1414 			break;
1415 		case 'n':
1416 			if ((p = strtok(optarg, ",")) == NULL)
1417 				Die(gettext("invalid argument for -n\n"));
1418 			opts.o_ntop = Atoi(p);
1419 			if (p = strtok(NULL, ","))
1420 				opts.o_nbottom = Atoi(p);
1421 			opts.o_outpmode &= ~OPT_FULLSCREEN;
1422 			break;
1423 		case 's':
1424 			opts.o_sortorder = -1;
1425 			sortk = optarg;
1426 			break;
1427 		case 'S':
1428 			opts.o_sortorder = 1;
1429 			sortk = optarg;
1430 			break;
1431 		case 'u':
1432 			if ((p = strtok(optarg, ", ")) == NULL)
1433 				Die(gettext("invalid argument for -u\n"));
1434 			add_uid(&euid_tbl, p);
1435 			while (p = strtok(NULL, ", "))
1436 				add_uid(&euid_tbl, p);
1437 			break;
1438 		case 'U':
1439 			if ((p = strtok(optarg, ", ")) == NULL)
1440 				Die(gettext("invalid argument for -U\n"));
1441 			add_uid(&ruid_tbl, p);
1442 			while (p = strtok(NULL, ", "))
1443 				add_uid(&ruid_tbl, p);
1444 			break;
1445 		case 'p':
1446 			fill_table(&pid_tbl, optarg, 'p');
1447 			break;
1448 		case 'C':
1449 			fill_set_table(optarg);
1450 			opts.o_outpmode |= OPT_PSETS;
1451 			break;
1452 		case 'P':
1453 			fill_table(&cpu_tbl, optarg, 'P');
1454 			break;
1455 		case 'k':
1456 			fill_table(&tsk_tbl, optarg, 'k');
1457 			break;
1458 		case 'j':
1459 			fill_prj_table(optarg);
1460 			break;
1461 		case 'L':
1462 			opts.o_outpmode |= OPT_LWPS;
1463 			break;
1464 		case 'z':
1465 			if ((p = strtok(optarg, ", ")) == NULL)
1466 				Die(gettext("invalid argument for -z\n"));
1467 			add_zone(&zone_tbl, p);
1468 			while (p = strtok(NULL, ", "))
1469 				add_zone(&zone_tbl, p);
1470 			break;
1471 		case 'Z':
1472 			opts.o_outpmode |= OPT_SPLIT | OPT_ZONES;
1473 			break;
1474 		default:
1475 			Usage();
1476 		}
1477 	}
1478 
1479 	(void) atexit(Exit);
1480 	if ((opts.o_outpmode & OPT_USERS) &&
1481 	    !(opts.o_outpmode & OPT_SPLIT))
1482 		opts.o_nbottom = opts.o_ntop;
1483 	if (opts.o_ntop == 0 || opts.o_nbottom == 0)
1484 		Die(gettext("invalid argument for -n\n"));
1485 	if (!(opts.o_outpmode & OPT_SPLIT) && (opts.o_outpmode & OPT_USERS) &&
1486 	    ((opts.o_outpmode & (OPT_PSINFO | OPT_MSACCT))))
1487 		Die(gettext("-t option cannot be used with -v or -m\n"));
1488 
1489 	if ((opts.o_outpmode & OPT_SPLIT) && (opts.o_outpmode && OPT_USERS) &&
1490 	    !((opts.o_outpmode & (OPT_PSINFO | OPT_MSACCT))))
1491 		Die(gettext("-t option cannot be used with "
1492 		    "-a, -J, -T or -Z\n"));
1493 
1494 	if ((opts.o_outpmode & OPT_USERS) &&
1495 	    (opts.o_outpmode & (OPT_TASKS | OPT_PROJECTS | OPT_ZONES)))
1496 		Die(gettext("-a option cannot be used with "
1497 		    "-t, -J, -T or -Z\n"));
1498 
1499 	if (((opts.o_outpmode & OPT_TASKS) &&
1500 	    (opts.o_outpmode & (OPT_PROJECTS|OPT_ZONES))) ||
1501 	    ((opts.o_outpmode & OPT_PROJECTS) &&
1502 	    (opts.o_outpmode & (OPT_TASKS|OPT_ZONES)))) {
1503 		Die(gettext(
1504 		    "-J, -T and -Z options are mutually exclusive\n"));
1505 	}
1506 
1507 	/*
1508 	 * There is not enough space to combine microstate information and
1509 	 * lgroup information and still fit in 80-column output.
1510 	 */
1511 	if ((opts.o_outpmode & OPT_LGRP) && (opts.o_outpmode & OPT_MSACCT)) {
1512 		Die(gettext("-H and -m options are mutually exclusive\n"));
1513 	}
1514 
1515 	if (argc > optind)
1516 		opts.o_interval = Atoi(argv[optind++]);
1517 	if (argc > optind)
1518 		opts.o_count = Atoi(argv[optind++]);
1519 	if (opts.o_count == 0)
1520 		Die(gettext("invalid counter value\n"));
1521 	if (argc > optind)
1522 		Usage();
1523 	if (opts.o_outpmode & OPT_REALTIME)
1524 		Priocntl("RT");
1525 	if (isatty(STDOUT_FILENO) == 1 && isatty(STDIN_FILENO))
1526 		opts.o_outpmode |= OPT_TTY;	/* interactive */
1527 	if (!(opts.o_outpmode & OPT_TTY)) {
1528 		opts.o_outpmode &= ~OPT_TERMCAP; /* no termcap for pipes */
1529 		opts.o_outpmode &= ~OPT_FULLSCREEN;
1530 	}
1531 	if (opts.o_outpmode & OPT_TERMCAP)
1532 		ldtermcap();		/* can turn OPT_TERMCAP off */
1533 	if (opts.o_outpmode & OPT_TERMCAP)
1534 		(void) setsize();
1535 	list_alloc(&lwps, opts.o_ntop);
1536 	list_alloc(&users, opts.o_nbottom);
1537 	list_alloc(&tasks, opts.o_nbottom);
1538 	list_alloc(&projects, opts.o_nbottom);
1539 	list_alloc(&zones, opts.o_nbottom);
1540 	list_alloc(&lgroups, opts.o_nbottom);
1541 	list_setkeyfunc(sortk, &opts, &lwps, LT_LWPS);
1542 	list_setkeyfunc(NULL, &opts, &users, LT_USERS);
1543 	list_setkeyfunc(NULL, &opts, &tasks, LT_TASKS);
1544 	list_setkeyfunc(NULL, &opts, &projects, LT_PROJECTS);
1545 	list_setkeyfunc(NULL, &opts, &zones, LT_ZONES);
1546 	list_setkeyfunc(NULL, &opts, &lgroups, LT_LGRPS);
1547 	if (opts.o_outpmode & OPT_TERMCAP)
1548 		curses_on();
1549 	if ((procdir = opendir("/proc")) == NULL)
1550 		Die(gettext("cannot open /proc directory\n"));
1551 	if (opts.o_outpmode & OPT_TTY) {
1552 		(void) printf(gettext("Please wait...\r"));
1553 		if (!(opts.o_outpmode & OPT_TERMCAP))
1554 			(void) putchar('\n');
1555 		(void) fflush(stdout);
1556 	}
1557 	set_signals();
1558 	pollset.fd = STDIN_FILENO;
1559 	pollset.events = POLLIN;
1560 	timeout = opts.o_interval * MILLISEC;
1561 
1562 	/*
1563 	 * main program loop
1564 	 */
1565 	do {
1566 		if (sigterm == 1)
1567 			break;
1568 		if (sigtstp == 1) {
1569 			curses_off();
1570 			(void) signal(SIGTSTP, SIG_DFL);
1571 			(void) kill(0, SIGTSTP);
1572 			/*
1573 			 * prstat stops here until it receives SIGCONT signal.
1574 			 */
1575 			sigtstp = 0;
1576 			(void) signal(SIGTSTP, sig_handler);
1577 			curses_on();
1578 			print_movecur = FALSE;
1579 			if (opts.o_outpmode & OPT_FULLSCREEN)
1580 				sigwinch = 1;
1581 		}
1582 		if (sigwinch == 1) {
1583 			if (setsize() == 1) {
1584 				list_free(&lwps);
1585 				list_free(&users);
1586 				list_free(&tasks);
1587 				list_free(&projects);
1588 				list_free(&zones);
1589 				list_alloc(&lwps, opts.o_ntop);
1590 				list_alloc(&users, opts.o_nbottom);
1591 				list_alloc(&tasks, opts.o_nbottom);
1592 				list_alloc(&projects, opts.o_nbottom);
1593 				list_alloc(&zones, opts.o_nbottom);
1594 			}
1595 			sigwinch = 0;
1596 			(void) signal(SIGWINCH, sig_handler);
1597 		}
1598 		prstat_scandir(procdir);
1599 		list_refresh(&lwps);
1600 		if (print_movecur)
1601 			(void) putp(movecur);
1602 		print_movecur = TRUE;
1603 		if ((opts.o_outpmode & OPT_PSINFO) ||
1604 		    (opts.o_outpmode & OPT_MSACCT)) {
1605 			list_sort(&lwps);
1606 			list_print(&lwps);
1607 		}
1608 		if (opts.o_outpmode & OPT_USERS) {
1609 			list_getsize(&users);
1610 			list_sort(&users);
1611 			list_print(&users);
1612 			list_clear(&users);
1613 		}
1614 		if (opts.o_outpmode & OPT_TASKS) {
1615 			list_getsize(&tasks);
1616 			list_sort(&tasks);
1617 			list_print(&tasks);
1618 			list_clear(&tasks);
1619 		}
1620 		if (opts.o_outpmode & OPT_PROJECTS) {
1621 			list_getsize(&projects);
1622 			list_sort(&projects);
1623 			list_print(&projects);
1624 			list_clear(&projects);
1625 		}
1626 		if (opts.o_outpmode & OPT_ZONES) {
1627 			list_getsize(&zones);
1628 			list_sort(&zones);
1629 			list_print(&zones);
1630 			list_clear(&zones);
1631 		}
1632 		if (opts.o_count == 1)
1633 			break;
1634 		/*
1635 		 * If poll() returns -1 and sets errno to EINTR here because
1636 		 * the process received a signal, it is Ok to abort this
1637 		 * timeout and loop around because we check the signals at the
1638 		 * top of the loop.
1639 		 */
1640 		if (opts.o_outpmode & OPT_TTY) {
1641 			if (poll(&pollset, (nfds_t)1, timeout) > 0) {
1642 				if (read(STDIN_FILENO, &key, 1) == 1) {
1643 					if (tolower(key) == 'q')
1644 						break;
1645 				}
1646 			}
1647 		} else {
1648 			(void) sleep(opts.o_interval);
1649 		}
1650 	} while (opts.o_count == (-1) || --opts.o_count);
1651 
1652 	if (opts.o_outpmode & OPT_TTY)
1653 		(void) putchar('\r');
1654 	return (0);
1655 }
1656