add UNLEASHED_OBJ to unleashed.mk
[unleashed/tickless.git] / usr / src / cmd / stat / vmstat / vmstat.c
bloba901a635467aed510ff469b6468bb960dafa1233
1 /*
2 * Copyright (c) 1991, 2010, Oracle and/or its affiliates. All rights reserved.
3 */
5 /*
6 * Copyright (c) 1980 Regents of the University of California.
7 * All rights reserved. The Berkeley software License Agreement
8 * specifies the terms and conditions for redistribution.
9 */
11 /* from UCB 5.4 5/17/86 */
12 /* from SunOS 4.1, SID 1.31 */
14 #include <stdio.h>
15 #include <stdlib.h>
16 #include <stdarg.h>
17 #include <ctype.h>
18 #include <unistd.h>
19 #include <memory.h>
20 #include <string.h>
21 #include <fcntl.h>
22 #include <errno.h>
23 #include <signal.h>
24 #include <values.h>
25 #include <poll.h>
26 #include <locale.h>
28 #include "statcommon.h"
30 char *cmdname = "vmstat";
31 int caught_cont = 0;
33 static uint_t timestamp_fmt = NODATE;
35 static int hz;
36 static int pagesize;
37 static double etime;
38 static int lines = 1;
39 static int swflag = 0, pflag = 0;
40 static int suppress_state;
41 static long iter = 0;
42 static hrtime_t period_n = 0;
43 static struct snapshot *ss;
45 struct iodev_filter df;
47 #define pgtok(a) ((a) * (pagesize >> 10))
48 #define denom(x) ((x) ? (x) : 1)
49 #define REPRINT 19
51 static void dovmstats(struct snapshot *old, struct snapshot *new);
52 static void printhdr(int);
53 static void dosum(struct sys_snapshot *ss);
54 static void dointr(struct snapshot *ss);
55 static void usage(void);
57 int
58 main(int argc, char **argv)
60 struct snapshot *old = NULL;
61 enum snapshot_types types = SNAP_SYSTEM;
62 int summary = 0;
63 int intr = 0;
64 kstat_ctl_t *kc;
65 int forever = 0;
66 hrtime_t start_n;
67 int c;
69 (void) setlocale(LC_ALL, "");
70 #if !defined(TEXT_DOMAIN) /* Should be defined by cc -D */
71 #define TEXT_DOMAIN "SYS_TEST" /* Use this only if it weren't */
72 #endif
73 (void) textdomain(TEXT_DOMAIN);
75 pagesize = sysconf(_SC_PAGESIZE);
76 hz = sysconf(_SC_CLK_TCK);
78 while ((c = getopt(argc, argv, "ipqsST:")) != EOF)
79 switch (c) {
80 case 'S':
81 swflag = !swflag;
82 break;
83 case 's':
84 summary = 1;
85 break;
86 case 'i':
87 intr = 1;
88 break;
89 case 'q':
90 suppress_state = 1;
91 break;
92 case 'p':
93 pflag++; /* detailed paging info */
94 break;
95 case 'T':
96 if (optarg) {
97 if (*optarg == 'u')
98 timestamp_fmt = UDATE;
99 else if (*optarg == 'd')
100 timestamp_fmt = DDATE;
101 else
102 usage();
103 } else {
104 usage();
106 break;
107 default:
108 usage();
111 argc -= optind;
112 argv += optind;
114 /* consistency with iostat */
115 types |= SNAP_CPUS;
117 if (intr)
118 types |= SNAP_INTERRUPTS;
119 if (!intr)
120 types |= SNAP_IODEVS;
122 /* max to fit in less than 80 characters */
123 df.if_max_iodevs = 4;
124 df.if_allowed_types = IODEV_DISK;
125 df.if_nr_names = 0;
126 df.if_names = safe_alloc(df.if_max_iodevs * sizeof (char *));
127 (void) memset(df.if_names, 0, df.if_max_iodevs * sizeof (char *));
129 while (argc > 0 && !isdigit(argv[0][0]) &&
130 df.if_nr_names < df.if_max_iodevs) {
131 df.if_names[df.if_nr_names] = *argv;
132 df.if_nr_names++;
133 argc--, argv++;
136 kc = open_kstat();
138 start_n = gethrtime();
140 ss = acquire_snapshot(kc, types, &df);
142 /* time, in seconds, since boot */
143 etime = ss->s_sys.ss_ticks / hz;
145 if (intr) {
146 dointr(ss);
147 free_snapshot(ss);
148 exit(0);
150 if (summary) {
151 dosum(&ss->s_sys);
152 free_snapshot(ss);
153 exit(0);
156 if (argc > 0) {
157 long interval;
158 char *endptr;
160 errno = 0;
161 interval = strtol(argv[0], &endptr, 10);
163 if (errno > 0 || *endptr != '\0' || interval <= 0 ||
164 interval > MAXINT)
165 usage();
166 period_n = (hrtime_t)interval * NANOSEC;
167 if (period_n <= 0)
168 usage();
169 iter = MAXLONG;
170 if (argc > 1) {
171 iter = strtol(argv[1], NULL, 10);
172 if (errno > 0 || *endptr != '\0' || iter <= 0)
173 usage();
174 } else
175 forever = 1;
176 if (argc > 2)
177 usage();
180 (void) sigset(SIGCONT, printhdr);
182 dovmstats(old, ss);
183 while (forever || --iter > 0) {
184 /* (void) poll(NULL, 0, poll_interval); */
186 /* Have a kip */
187 sleep_until(&start_n, period_n, forever, &caught_cont);
189 free_snapshot(old);
190 old = ss;
191 ss = acquire_snapshot(kc, types, &df);
193 if (!suppress_state)
194 snapshot_report_changes(old, ss);
196 /* if config changed, show stats from boot */
197 if (snapshot_has_changed(old, ss)) {
198 free_snapshot(old);
199 old = NULL;
202 dovmstats(old, ss);
205 free_snapshot(old);
206 free_snapshot(ss);
207 free(df.if_names);
208 (void) kstat_close(kc);
209 return (0);
212 #define DELTA(v) (new->v - (old ? old->v : 0))
213 #define ADJ(n) ((adj <= 0) ? n : (adj >= n) ? 1 : n - adj)
214 #define adjprintf(fmt, n, val) adj -= (n + 1) - printf(fmt, ADJ(n), val)
216 static int adj; /* number of excess columns */
218 /*ARGSUSED*/
219 static void
220 show_disk(void *v1, void *v2, void *d)
222 struct iodev_snapshot *old = (struct iodev_snapshot *)v1;
223 struct iodev_snapshot *new = (struct iodev_snapshot *)v2;
224 hrtime_t oldtime = new->is_crtime;
225 double hr_etime;
226 double reads, writes;
228 if (new == NULL)
229 return;
231 if (old)
232 oldtime = old->is_stats.wlastupdate;
233 hr_etime = new->is_stats.wlastupdate - oldtime;
234 if (hr_etime == 0.0)
235 hr_etime = NANOSEC;
236 reads = new->is_stats.reads - (old ? old->is_stats.reads : 0);
237 writes = new->is_stats.writes - (old ? old->is_stats.writes : 0);
238 adjprintf(" %*.0f", 2, (reads + writes) / hr_etime * NANOSEC);
241 static void
242 dovmstats(struct snapshot *old, struct snapshot *new)
244 kstat_t *oldsys = NULL;
245 kstat_t *newsys = &new->s_sys.ss_agg_sys;
246 kstat_t *oldvm = NULL;
247 kstat_t *newvm = &new->s_sys.ss_agg_vm;
248 double percent_factor;
249 ulong_t sys_updates, vm_updates;
250 int count;
252 adj = 0;
254 if (old) {
255 oldsys = &old->s_sys.ss_agg_sys;
256 oldvm = &old->s_sys.ss_agg_vm;
259 etime = cpu_ticks_delta(oldsys, newsys);
261 percent_factor = 100.0 / denom(etime);
263 * If any time has passed, convert etime to seconds per CPU
265 etime = etime >= 1.0 ? (etime / nr_active_cpus(new)) / hz : 1.0;
266 sys_updates = denom(DELTA(s_sys.ss_sysinfo.updates));
267 vm_updates = denom(DELTA(s_sys.ss_vminfo.updates));
269 if (timestamp_fmt != NODATE) {
270 print_timestamp(timestamp_fmt);
271 lines--;
274 if (--lines <= 0)
275 printhdr(0);
277 adj = 0;
279 if (pflag) {
280 adjprintf(" %*u", 6,
281 pgtok((int)(DELTA(s_sys.ss_vminfo.swap_avail)
282 / vm_updates)));
283 adjprintf(" %*u", 5,
284 pgtok((int)(DELTA(s_sys.ss_vminfo.freemem) / vm_updates)));
285 adjprintf(" %*.0f", 3, kstat_delta(oldvm, newvm, "pgrec")
286 / etime);
287 adjprintf(" %*.0f", 3, (kstat_delta(oldvm, newvm, "hat_fault") +
288 kstat_delta(oldvm, newvm, "as_fault")) / etime);
289 adjprintf(" %*.0f", 3, pgtok(kstat_delta(oldvm, newvm, "dfree"))
290 / etime);
291 adjprintf(" %*ld", 3, pgtok(new->s_sys.ss_deficit));
292 adjprintf(" %*.0f", 3, kstat_delta(oldvm, newvm, "scan")
293 / etime);
294 adjprintf(" %*.0f", 4,
295 pgtok(kstat_delta(oldvm, newvm, "execpgin")) / etime);
296 adjprintf(" %*.0f", 4,
297 pgtok(kstat_delta(oldvm, newvm, "execpgout")) / etime);
298 adjprintf(" %*.0f", 4,
299 pgtok(kstat_delta(oldvm, newvm, "execfree")) / etime);
300 adjprintf(" %*.0f", 4,
301 pgtok(kstat_delta(oldvm, newvm, "anonpgin")) / etime);
302 adjprintf(" %*.0f", 4,
303 pgtok(kstat_delta(oldvm, newvm, "anonpgout")) / etime);
304 adjprintf(" %*.0f", 4,
305 pgtok(kstat_delta(oldvm, newvm, "anonfree")) / etime);
306 adjprintf(" %*.0f", 4,
307 pgtok(kstat_delta(oldvm, newvm, "fspgin")) / etime);
308 adjprintf(" %*.0f", 4,
309 pgtok(kstat_delta(oldvm, newvm, "fspgout")) / etime);
310 adjprintf(" %*.0f\n", 4,
311 pgtok(kstat_delta(oldvm, newvm, "fsfree")) / etime);
312 (void) fflush(stdout);
313 return;
316 adjprintf(" %*lu", 1, DELTA(s_sys.ss_sysinfo.runque) / sys_updates);
317 adjprintf(" %*lu", 1, DELTA(s_sys.ss_sysinfo.waiting) / sys_updates);
318 adjprintf(" %*lu", 1, DELTA(s_sys.ss_sysinfo.swpque) / sys_updates);
319 adjprintf(" %*u", 6, pgtok((int)(DELTA(s_sys.ss_vminfo.swap_avail)
320 / vm_updates)));
321 adjprintf(" %*u", 5, pgtok((int)(DELTA(s_sys.ss_vminfo.freemem)
322 / vm_updates)));
323 adjprintf(" %*.0f", 3, swflag? 0 :
324 kstat_delta(oldvm, newvm, "pgrec") / etime);
325 adjprintf(" %*.0f", 3, swflag? 0 :
326 (kstat_delta(oldvm, newvm, "hat_fault")
327 + kstat_delta(oldvm, newvm, "as_fault"))
328 / etime);
329 adjprintf(" %*.0f", 2, pgtok(kstat_delta(oldvm, newvm, "pgpgin"))
330 / etime);
331 adjprintf(" %*.0f", 2, pgtok(kstat_delta(oldvm, newvm, "pgpgout"))
332 / etime);
333 adjprintf(" %*.0f", 2, pgtok(kstat_delta(oldvm, newvm, "dfree"))
334 / etime);
335 adjprintf(" %*ld", 2, pgtok(new->s_sys.ss_deficit));
336 adjprintf(" %*.0f", 2, kstat_delta(oldvm, newvm, "scan") / etime);
338 (void) snapshot_walk(SNAP_IODEVS, old, new, show_disk, NULL);
340 count = df.if_max_iodevs - new->s_nr_iodevs;
341 while (count-- > 0)
342 adjprintf(" %*d", 2, 0);
344 adjprintf(" %*.0f", 4, kstat_delta(oldsys, newsys, "intr") / etime);
345 adjprintf(" %*.0f", 4, kstat_delta(oldsys, newsys, "syscall") / etime);
346 adjprintf(" %*.0f", 4, kstat_delta(oldsys, newsys, "pswitch") / etime);
347 adjprintf(" %*.0f", 2,
348 kstat_delta(oldsys, newsys, "cpu_ticks_user") * percent_factor);
349 adjprintf(" %*.0f", 2, kstat_delta(oldsys, newsys, "cpu_ticks_kernel")
350 * percent_factor);
351 adjprintf(" %*.0f\n", 2, (kstat_delta(oldsys, newsys, "cpu_ticks_idle")
352 + kstat_delta(oldsys, newsys, "cpu_ticks_wait"))
353 * percent_factor);
354 (void) fflush(stdout);
357 /*ARGSUSED*/
358 static void
359 print_disk(void *v, void *v2, void *d)
361 struct iodev_snapshot *iodev = (struct iodev_snapshot *)v2;
363 if (iodev == NULL)
364 return;
366 (void) printf("%c%c ", iodev->is_name[0], iodev->is_name[2]);
369 /* ARGSUSED */
370 static void
371 printhdr(int sig)
373 int i = df.if_max_iodevs - ss->s_nr_iodevs;
375 if (sig == SIGCONT)
376 caught_cont = 1;
378 if (pflag) {
379 (void) printf(" memory page ");
380 (void) printf("executable anonymous filesystem \n");
381 (void) printf(" swap free re mf fr de sr ");
382 (void) printf("epi epo epf api apo apf fpi fpo fpf\n");
383 lines = REPRINT;
384 return;
387 (void) printf(" kthr memory page ");
388 (void) printf("disk faults cpu\n");
390 if (swflag)
391 (void) printf(" r b w swap free si so pi po fr de sr ");
392 else
393 (void) printf(" r b w swap free re mf pi po fr de sr ");
395 (void) snapshot_walk(SNAP_IODEVS, NULL, ss, print_disk, NULL);
397 while (i-- > 0)
398 (void) printf("-- ");
400 (void) printf(" in sy cs us sy id\n");
401 lines = REPRINT;
404 static void
405 sum_out(char const *pretty, kstat_t *ks, char *name)
407 kstat_named_t *ksn = kstat_data_lookup(ks, name);
408 if (ksn == NULL) {
409 fail(0, "kstat_data_lookup('%s', '%s') failed",
410 ks->ks_name, name);
413 (void) printf("%9llu %s\n", ksn->value.ui64, pretty);
416 static void
417 zero_out(char const *pretty)
419 (void) printf("%9u %s\n", 0, pretty);
422 static void
423 dosum(struct sys_snapshot *ss)
425 uint64_t total_faults;
426 kstat_named_t *ksn;
427 long double nchtotal;
428 uint64_t nchhits;
430 zero_out("swap ins");
431 zero_out("swap outs");
432 zero_out("pages swapped in");
433 zero_out("pages swapped out");
435 ksn = kstat_data_lookup(&ss->ss_agg_vm, "hat_fault");
436 if (ksn == NULL) {
437 fail(0, "kstat_data_lookup('%s', 'hat_fault') failed",
438 ss->ss_agg_vm.ks_name);
440 total_faults = ksn->value.ui64;
441 ksn = kstat_data_lookup(&ss->ss_agg_vm, "as_fault");
442 if (ksn == NULL) {
443 fail(0, "kstat_data_lookup('%s', 'as_fault') failed",
444 ss->ss_agg_vm.ks_name);
446 total_faults += ksn->value.ui64;
448 (void) printf("%9llu total address trans. faults taken\n",
449 total_faults);
451 sum_out("page ins", &ss->ss_agg_vm, "pgin");
452 sum_out("page outs", &ss->ss_agg_vm, "pgout");
453 sum_out("pages paged in", &ss->ss_agg_vm, "pgpgin");
454 sum_out("pages paged out", &ss->ss_agg_vm, "pgpgout");
455 sum_out("total reclaims", &ss->ss_agg_vm, "pgrec");
456 sum_out("reclaims from free list", &ss->ss_agg_vm, "pgfrec");
457 sum_out("micro (hat) faults", &ss->ss_agg_vm, "hat_fault");
458 sum_out("minor (as) faults", &ss->ss_agg_vm, "as_fault");
459 sum_out("major faults", &ss->ss_agg_vm, "maj_fault");
460 sum_out("copy-on-write faults", &ss->ss_agg_vm, "cow_fault");
461 sum_out("zero fill page faults", &ss->ss_agg_vm, "zfod");
462 sum_out("pages examined by the clock daemon", &ss->ss_agg_vm, "scan");
463 sum_out("revolutions of the clock hand", &ss->ss_agg_vm, "rev");
464 sum_out("pages freed by the clock daemon", &ss->ss_agg_vm, "dfree");
465 sum_out("forks", &ss->ss_agg_sys, "sysfork");
466 sum_out("vforks", &ss->ss_agg_sys, "sysvfork");
467 sum_out("execs", &ss->ss_agg_sys, "sysexec");
468 sum_out("cpu context switches", &ss->ss_agg_sys, "pswitch");
469 sum_out("device interrupts", &ss->ss_agg_sys, "intr");
470 sum_out("traps", &ss->ss_agg_sys, "trap");
471 sum_out("system calls", &ss->ss_agg_sys, "syscall");
473 nchtotal = (long double) ss->ss_nc.ncs_hits.value.ui64 +
474 (long double) ss->ss_nc.ncs_misses.value.ui64;
475 nchhits = ss->ss_nc.ncs_hits.value.ui64;
476 (void) printf("%9.0Lf total name lookups (cache hits %.0Lf%%)\n",
477 nchtotal, nchhits / denom(nchtotal) * 100);
479 sum_out("user cpu", &ss->ss_agg_sys, "cpu_ticks_user");
480 sum_out("system cpu", &ss->ss_agg_sys, "cpu_ticks_kernel");
481 sum_out("idle cpu", &ss->ss_agg_sys, "cpu_ticks_idle");
482 sum_out("wait cpu", &ss->ss_agg_sys, "cpu_ticks_wait");
485 static void
486 dointr(struct snapshot *ss)
488 size_t i;
489 ulong_t total = 0;
491 (void) printf("interrupt total rate\n");
492 (void) printf("--------------------------------\n");
494 for (i = 0; i < ss->s_nr_intrs; i++) {
495 (void) printf("%-12.8s %10lu %8.0f\n",
496 ss->s_intrs[i].is_name, ss->s_intrs[i].is_total,
497 ss->s_intrs[i].is_total / etime);
498 total += ss->s_intrs[i].is_total;
501 (void) printf("--------------------------------\n");
502 (void) printf("Total %10lu %8.0f\n", total, total / etime);
505 static void
506 usage(void)
508 (void) fprintf(stderr,
509 "Usage: vmstat [-ipqsS] [-T d|u] [disk ...] "
510 "[interval [count]]\n");
511 exit(1);