xref: /illumos-gate/usr/src/cmd/zpool/zpool_main.c (revision 5cce9d40d191f7d11762f0803b81ddffaabafd3e)
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, Version 1.0 only
6  * (the "License").  You may not use this file except in compliance
7  * with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or http://www.opensolaris.org/os/licensing.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  */
22 /*
23  * Copyright 2006 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 #pragma ident	"%Z%%M%	%I%	%E% SMI"
28 
29 #include <assert.h>
30 #include <ctype.h>
31 #include <dirent.h>
32 #include <errno.h>
33 #include <fcntl.h>
34 #include <libgen.h>
35 #include <libintl.h>
36 #include <libuutil.h>
37 #include <locale.h>
38 #include <stdio.h>
39 #include <stdlib.h>
40 #include <string.h>
41 #include <strings.h>
42 #include <unistd.h>
43 #include <priv.h>
44 
45 #include <sys/stat.h>
46 
47 #include <libzfs.h>
48 
49 #include "zpool_util.h"
50 
51 static int zpool_do_create(int, char **);
52 static int zpool_do_destroy(int, char **);
53 
54 static int zpool_do_add(int, char **);
55 
56 static int zpool_do_list(int, char **);
57 static int zpool_do_iostat(int, char **);
58 static int zpool_do_status(int, char **);
59 
60 static int zpool_do_online(int, char **);
61 static int zpool_do_offline(int, char **);
62 
63 static int zpool_do_attach(int, char **);
64 static int zpool_do_detach(int, char **);
65 static int zpool_do_replace(int, char **);
66 
67 static int zpool_do_scrub(int, char **);
68 
69 static int zpool_do_import(int, char **);
70 static int zpool_do_export(int, char **);
71 
72 /*
73  * These libumem hooks provide a reasonable set of defaults for the allocator's
74  * debugging facilities.
75  */
76 const char *
77 _umem_debug_init()
78 {
79 	return ("default,verbose"); /* $UMEM_DEBUG setting */
80 }
81 
82 const char *
83 _umem_logging_init(void)
84 {
85 	return ("fail,contents"); /* $UMEM_LOGGING setting */
86 }
87 
88 typedef struct zpool_command {
89 	const char	*name;
90 	int		(*func)(int, char **);
91 	const char	*usage;
92 } zpool_command_t;
93 
94 /*
95  * Master command table.  Each ZFS command has a name, associated function, and
96  * usage message.  These commands are organized according to how they are
97  * displayed in the usage message.  An empty command (one with a NULL name)
98  * indicates an empty line in the generic usage message.
99  */
100 static zpool_command_t command_table[] = {
101 	{ "create",	zpool_do_create,
102 	    "\tcreate  [-fn] [-R root] [-m mountpoint] <pool> <vdev> ...\n" },
103 	{ "destroy",	zpool_do_destroy,
104 	    "\tdestroy [-f] <pool>\n"					},
105 
106 
107 	{ NULL },
108 
109 	{ "add",	zpool_do_add,
110 	    "\tadd [-fn] <pool> <vdev> ...\n"				},
111 
112 	{ NULL },
113 
114 	{ "list",	zpool_do_list,
115 	    "\tlist [-H] [-o field[,field]*] [pool] ...\n"		},
116 	{ "iostat",	zpool_do_iostat,
117 	    "\tiostat [-v] [pool] ... [interval [count]]\n"		},
118 	{ "status",	zpool_do_status,
119 	    "\tstatus [-vx] [pool] ...\n"				},
120 
121 	{ NULL },
122 
123 	{ "online",	zpool_do_online,
124 	    "\tonline <pool> <device>\n"				},
125 	{ "offline",	zpool_do_offline,
126 	    "\toffline <pool> <device>\n"				},
127 
128 	{ NULL },
129 
130 	{ "attach",	zpool_do_attach,
131 	    "\tattach [-f] <pool> <device> <new_device>\n"		},
132 	{ "detach",	zpool_do_detach,
133 	    "\tdetach <pool> <device>\n"				},
134 	{ "replace",	zpool_do_replace,
135 	    "\treplace [-f] <pool> <device> [new_device]\n"		},
136 
137 	{ NULL },
138 
139 	{ "scrub",	zpool_do_scrub,
140 	    "\tscrub [-s] <pool> ...\n"					},
141 
142 	{ NULL },
143 
144 	{ "import",	zpool_do_import,
145 	    "\timport [-d dir]\n"
146 	    "\timport [-d dir] [-f] [-o opts] [-R root] -a\n"
147 	    "\timport [-d dir] [-f] [-o opts] [-R root ]<pool | id> "
148 	    "[newpool]\n"						},
149 	{ "export",	zpool_do_export,
150 	    "\texport [-f] <pool> ...\n"				},
151 
152 	{ NULL }
153 };
154 
155 #define	NCOMMAND	(sizeof (command_table) / sizeof (command_table[0]))
156 
157 zpool_command_t *current_command;
158 
159 /*
160  * Fields available for 'zpool list'.
161  */
162 typedef enum {
163 	ZPOOL_FIELD_NAME,
164 	ZPOOL_FIELD_SIZE,
165 	ZPOOL_FIELD_USED,
166 	ZPOOL_FIELD_AVAILABLE,
167 	ZPOOL_FIELD_CAPACITY,
168 	ZPOOL_FIELD_HEALTH,
169 	ZPOOL_FIELD_ROOT
170 } zpool_field_t;
171 
172 #define	MAX_FIELDS	10
173 
174 typedef struct column_def {
175 	const char	*cd_title;
176 	size_t		cd_width;
177 	enum {
178 		left_justify,
179 		right_justify
180 	}		cd_justify;
181 } column_def_t;
182 
183 static column_def_t column_table[] = {
184 	{ "NAME",	20,	left_justify	},
185 	{ "SIZE",	6,	right_justify	},
186 	{ "USED",	6,	right_justify	},
187 	{ "AVAIL",	6,	right_justify	},
188 	{ "CAP",	5,	right_justify	},
189 	{ "HEALTH",	9,	left_justify	},
190 	{ "ALTROOT",	15,	left_justify	}
191 };
192 
193 static char *column_subopts[] = {
194 	"name",
195 	"size",
196 	"used",
197 	"available",
198 	"capacity",
199 	"health",
200 	"root",
201 	NULL
202 };
203 
204 /*
205  * Display usage message.  If we're inside a command, display only the usage for
206  * that command.  Otherwise, iterate over the entire command table and display
207  * a complete usage message.
208  */
209 void
210 usage(int requested)
211 {
212 	int i;
213 	FILE *fp = requested ? stdout : stderr;
214 
215 	if (current_command == NULL) {
216 		int i;
217 
218 		(void) fprintf(fp, gettext("usage: zpool command args ...\n"));
219 		(void) fprintf(fp,
220 		    gettext("where 'command' is one of the following:\n\n"));
221 
222 		for (i = 0; i < NCOMMAND; i++) {
223 			if (command_table[i].name == NULL)
224 				(void) fprintf(fp, "\n");
225 			else
226 				(void) fprintf(fp, "%s",
227 				    command_table[i].usage);
228 		}
229 	} else {
230 		(void) fprintf(fp, gettext("usage:\n"));
231 		(void) fprintf(fp, current_command->usage);
232 
233 		if (strcmp(current_command->name, "list") == 0) {
234 			(void) fprintf(fp, gettext("\nwhere 'field' is one "
235 			    "of the following:\n\n"));
236 
237 			for (i = 0; column_subopts[i] != NULL; i++)
238 				(void) fprintf(fp, "\t%s\n", column_subopts[i]);
239 		}
240 	}
241 
242 	exit(requested ? 0 : 2);
243 }
244 
245 const char *
246 state_to_name(int state)
247 {
248 	switch (state) {
249 	case VDEV_STATE_CLOSED:
250 	case VDEV_STATE_CANT_OPEN:
251 		return (gettext("FAULTED"));
252 	case VDEV_STATE_OFFLINE:
253 		return (gettext("OFFLINE"));
254 	case VDEV_STATE_DEGRADED:
255 		return (gettext("DEGRADED"));
256 	case VDEV_STATE_HEALTHY:
257 		return (gettext("ONLINE"));
258 	}
259 
260 	return (gettext("UNKNOWN"));
261 }
262 
263 void
264 print_vdev_tree(zpool_handle_t *zhp, const char *name, nvlist_t *nv, int indent)
265 {
266 	nvlist_t **child;
267 	uint_t c, children;
268 	char *vname;
269 
270 	if (name != NULL)
271 		(void) printf("\t%*s%s\n", indent, "", name);
272 
273 	if (nvlist_lookup_nvlist_array(nv, ZPOOL_CONFIG_CHILDREN,
274 	    &child, &children) != 0)
275 		return;
276 
277 	for (c = 0; c < children; c++) {
278 		vname = zpool_vdev_name(zhp, child[c]);
279 		print_vdev_tree(zhp, vname, child[c], indent + 2);
280 		free(vname);
281 	}
282 }
283 
284 /*
285  * zpool add [-fn] <pool> <vdev> ...
286  *
287  *	-f	Force addition of devices, even if they appear in use
288  *	-n	Do not add the devices, but display the resulting layout if
289  *		they were to be added.
290  *
291  * Adds the given vdevs to 'pool'.  As with create, the bulk of this work is
292  * handled by get_vdev_spec(), which constructs the nvlist needed to pass to
293  * libzfs.
294  */
295 int
296 zpool_do_add(int argc, char **argv)
297 {
298 	int force = FALSE;
299 	int dryrun = FALSE;
300 	int c;
301 	nvlist_t *nvroot;
302 	char *poolname;
303 	int ret;
304 	zpool_handle_t *zhp;
305 	nvlist_t *config;
306 
307 	/* check options */
308 	while ((c = getopt(argc, argv, "fn")) != -1) {
309 		switch (c) {
310 		case 'f':
311 			force = TRUE;
312 			break;
313 		case 'n':
314 			dryrun = TRUE;
315 			break;
316 		case '?':
317 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
318 			    optopt);
319 			usage(FALSE);
320 		}
321 	}
322 
323 	argc -= optind;
324 	argv += optind;
325 
326 	/* get pool name and check number of arguments */
327 	if (argc < 1) {
328 		(void) fprintf(stderr, gettext("missing pool name argument\n"));
329 		usage(FALSE);
330 	}
331 	if (argc < 2) {
332 		(void) fprintf(stderr, gettext("missing vdev specification\n"));
333 		usage(FALSE);
334 	}
335 
336 	poolname = argv[0];
337 
338 	argc--;
339 	argv++;
340 
341 	if ((zhp = zpool_open(poolname)) == NULL)
342 		return (1);
343 
344 	if ((config = zpool_get_config(zhp, NULL)) == NULL) {
345 		(void) fprintf(stderr, gettext("pool '%s' is unavailable\n"),
346 		    poolname);
347 		zpool_close(zhp);
348 		return (1);
349 	}
350 
351 	/* pass off to get_vdev_spec for processing */
352 	nvroot = make_root_vdev(config, force, !force, argc, argv);
353 	if (nvroot == NULL) {
354 		zpool_close(zhp);
355 		return (1);
356 	}
357 
358 	if (dryrun) {
359 		nvlist_t *poolnvroot;
360 
361 		verify(nvlist_lookup_nvlist(config, ZPOOL_CONFIG_VDEV_TREE,
362 		    &poolnvroot) == 0);
363 
364 		(void) printf(gettext("would update '%s' to the following "
365 		    "configuration:\n"), zpool_get_name(zhp));
366 
367 		print_vdev_tree(zhp, poolname, poolnvroot, 0);
368 		print_vdev_tree(zhp, NULL, nvroot, 0);
369 
370 		ret = 0;
371 	} else {
372 		ret = (zpool_add(zhp, nvroot) != 0);
373 	}
374 
375 	return (ret);
376 }
377 
378 /*
379  * zpool create [-fn] [-R root] [-m mountpoint] <pool> <dev> ...
380  *
381  *	-f	Force creation, even if devices appear in use
382  *	-n	Do not create the pool, but display the resulting layout if it
383  *		were to be created.
384  *      -R	Create a pool under an alternate root
385  *      -m	Set default mountpoint for the root dataset.  By default it's
386  *      	'/<pool>'
387  *
388  * Creates the the named pool according to the given vdev specification.  The
389  * bulk of the vdev processing is done in get_vdev_spec() in zpool_vdev.c.  Once
390  * we get the nvlist back from get_vdev_spec(), we either print out the contents
391  * (if '-n' was specified), or pass it to libzfs to do the creation.
392  */
393 int
394 zpool_do_create(int argc, char **argv)
395 {
396 	int force = FALSE;
397 	int dryrun = FALSE;
398 	int c;
399 	nvlist_t *nvroot;
400 	char *poolname;
401 	int ret;
402 	char *altroot = NULL;
403 	char *mountpoint = NULL;
404 
405 	/* check options */
406 	while ((c = getopt(argc, argv, ":fnR:m:")) != -1) {
407 		switch (c) {
408 		case 'f':
409 			force = TRUE;
410 			break;
411 		case 'n':
412 			dryrun = TRUE;
413 			break;
414 		case 'R':
415 			altroot = optarg;
416 			break;
417 		case 'm':
418 			mountpoint = optarg;
419 			break;
420 		case ':':
421 			(void) fprintf(stderr, gettext("missing argument for "
422 			    "'%c' option\n"), optopt);
423 			usage(FALSE);
424 			break;
425 		case '?':
426 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
427 			    optopt);
428 			usage(FALSE);
429 		}
430 	}
431 
432 	argc -= optind;
433 	argv += optind;
434 
435 	/* get pool name and check number of arguments */
436 	if (argc < 1) {
437 		(void) fprintf(stderr, gettext("missing pool name argument\n"));
438 		usage(FALSE);
439 	}
440 	if (argc < 2) {
441 		(void) fprintf(stderr, gettext("missing vdev specification\n"));
442 		usage(FALSE);
443 	}
444 
445 	poolname = argv[0];
446 
447 	/*
448 	 * As a special case, check for use of '/' in the name, and direct the
449 	 * user to use 'zfs create' instead.
450 	 */
451 	if (strchr(poolname, '/') != NULL) {
452 		(void) fprintf(stderr, gettext("cannot create '%s': invalid "
453 		    "character '/' in pool name\n"), poolname);
454 		(void) fprintf(stderr, gettext("use 'zfs create' to "
455 		    "create a dataset\n"));
456 		return (1);
457 	}
458 
459 	/* pass off to get_vdev_spec for bulk processing */
460 	nvroot = make_root_vdev(NULL, force, !force, argc - 1, argv + 1);
461 	if (nvroot == NULL)
462 		return (1);
463 
464 	if (altroot != NULL && altroot[0] != '/') {
465 		(void) fprintf(stderr, gettext("invalid alternate root '%s': "
466 		    "must be an absolute path\n"));
467 		return (1);
468 	}
469 
470 	/*
471 	 * Check the validity of the mountpoint and direct the user to use the
472 	 * '-m' mountpoint option if it looks like its in use.
473 	 */
474 	if (mountpoint == NULL ||
475 	    (strcmp(mountpoint, ZFS_MOUNTPOINT_LEGACY) != 0 &&
476 	    strcmp(mountpoint, ZFS_MOUNTPOINT_NONE) != 0)) {
477 		char buf[MAXPATHLEN];
478 		struct stat64 statbuf;
479 
480 		if (mountpoint && mountpoint[0] != '/') {
481 			(void) fprintf(stderr, gettext("invalid mountpoint "
482 			    "'%s': must be an absolute path, 'legacy', or "
483 			    "'none'\n"), mountpoint);
484 			return (1);
485 		}
486 
487 		if (mountpoint == NULL) {
488 			if (altroot != NULL)
489 				(void) snprintf(buf, sizeof (buf), "%s/%s",
490 				    altroot, poolname);
491 			else
492 				(void) snprintf(buf, sizeof (buf), "/%s",
493 				    poolname);
494 		} else {
495 			if (altroot != NULL)
496 				(void) snprintf(buf, sizeof (buf), "%s%s",
497 				    altroot, mountpoint);
498 			else
499 				(void) snprintf(buf, sizeof (buf), "%s",
500 				    mountpoint);
501 		}
502 
503 		if (stat64(buf, &statbuf) == 0 &&
504 		    statbuf.st_nlink != 2) {
505 			if (mountpoint == NULL)
506 				(void) fprintf(stderr, gettext("default "
507 				    "mountpoint '%s' exists and is not "
508 				    "empty\n"), buf);
509 			else
510 				(void) fprintf(stderr, gettext("mountpoint "
511 				    "'%s' exists and is not empty\n"), buf);
512 			(void) fprintf(stderr, gettext("use '-m' "
513 			    "option to provide a different default\n"));
514 			return (1);
515 		}
516 	}
517 
518 
519 	if (dryrun) {
520 		/*
521 		 * For a dry run invocation, print out a basic message and run
522 		 * through all the vdevs in the list and print out in an
523 		 * appropriate hierarchy.
524 		 *
525 		 * XXZFS find out of we can create the pool?
526 		 */
527 		(void) printf(gettext("would create '%s' with the "
528 		    "following layout:\n\n"), poolname);
529 
530 		print_vdev_tree(NULL, poolname, nvroot, 0);
531 
532 		ret = 0;
533 	} else {
534 		ret = 1;
535 		/*
536 		 * Hand off to libzfs.
537 		 */
538 		if (zpool_create(poolname, nvroot, altroot) == 0) {
539 			zfs_handle_t *pool = zfs_open(poolname,
540 			    ZFS_TYPE_FILESYSTEM);
541 			if (pool != NULL) {
542 				if (mountpoint != NULL)
543 					verify(zfs_prop_set(pool,
544 					    ZFS_PROP_MOUNTPOINT,
545 					    mountpoint) == 0);
546 				if (zfs_mount(pool, NULL, 0) == 0)
547 					ret = zfs_share(pool);
548 				zfs_close(pool);
549 			}
550 		}
551 
552 	}
553 
554 	nvlist_free(nvroot);
555 
556 	return (ret);
557 }
558 
559 /*
560  * zpool destroy <pool>
561  *
562  * 	-f	Forcefully unmount any datasets
563  *
564  * Destroy the given pool.  Automatically unmounts any datasets in the pool.
565  */
566 int
567 zpool_do_destroy(int argc, char **argv)
568 {
569 	int force = FALSE;
570 	int c;
571 	char *pool;
572 	zpool_handle_t *zhp;
573 	int ret;
574 
575 	/* check options */
576 	while ((c = getopt(argc, argv, "f")) != -1) {
577 		switch (c) {
578 		case 'f':
579 			force = TRUE;
580 			break;
581 		case '?':
582 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
583 			    optopt);
584 			usage(FALSE);
585 		}
586 	}
587 
588 	argc -= optind;
589 	argv += optind;
590 
591 	/* check arguments */
592 	if (argc < 1) {
593 		(void) fprintf(stderr, gettext("missing pool argument\n"));
594 		usage(FALSE);
595 	}
596 	if (argc > 1) {
597 		(void) fprintf(stderr, gettext("too many arguments\n"));
598 		usage(FALSE);
599 	}
600 
601 	pool = argv[0];
602 
603 	if ((zhp = zpool_open_canfail(pool)) == NULL) {
604 		/*
605 		 * As a special case, check for use of '/' in the name, and
606 		 * direct the user to use 'zfs destroy' instead.
607 		 */
608 		if (strchr(pool, '/') != NULL)
609 			(void) fprintf(stderr, gettext("use 'zfs destroy' to "
610 			    "destroy a dataset\n"));
611 		return (1);
612 	}
613 
614 	if (unmount_datasets(zhp, force) != 0) {
615 		(void) fprintf(stderr, gettext("could not destroy '%s': "
616 		    "could not unmount datasets\n"), zpool_get_name(zhp));
617 		return (1);
618 	}
619 
620 	ret = (zpool_destroy(zhp) != 0);
621 
622 	zpool_close(zhp);
623 
624 	return (ret);
625 }
626 
627 /*
628  * zpool export [-f] <pool> ...
629  *
630  *	-f	Forcefully unmount datasets
631  *
632  * Export the the given pools.  By default, the command will attempt to cleanly
633  * unmount any active datasets within the pool.  If the '-f' flag is specified,
634  * then the datasets will be forcefully unmounted.
635  */
636 int
637 zpool_do_export(int argc, char **argv)
638 {
639 	int force = FALSE;
640 	int c;
641 	zpool_handle_t *zhp;
642 	int ret;
643 	int i;
644 
645 	/* check options */
646 	while ((c = getopt(argc, argv, "f")) != -1) {
647 		switch (c) {
648 		case 'f':
649 			force = TRUE;
650 			break;
651 		case '?':
652 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
653 			    optopt);
654 			usage(FALSE);
655 		}
656 	}
657 
658 	argc -= optind;
659 	argv += optind;
660 
661 	/* check arguments */
662 	if (argc < 1) {
663 		(void) fprintf(stderr, gettext("missing pool argument\n"));
664 		usage(FALSE);
665 	}
666 
667 	ret = 0;
668 	for (i = 0; i < argc; i++) {
669 		if ((zhp = zpool_open_canfail(argv[i])) == NULL) {
670 			ret = 1;
671 			continue;
672 		}
673 
674 		if (unmount_datasets(zhp, force) != 0) {
675 			ret = 1;
676 			zpool_close(zhp);
677 			continue;
678 		}
679 
680 		if (zpool_export(zhp) != 0)
681 			ret = 1;
682 
683 		zpool_close(zhp);
684 	}
685 
686 	return (ret);
687 }
688 
689 /*
690  * Given a vdev configuration, determine the maximum width needed for the device
691  * name column.
692  */
693 static int
694 max_width(zpool_handle_t *zhp, nvlist_t *nv, int depth, int max)
695 {
696 	char *name = zpool_vdev_name(zhp, nv);
697 	nvlist_t **child;
698 	uint_t c, children;
699 	int ret;
700 
701 	if (strlen(name) + depth > max)
702 		max = strlen(name) + depth;
703 
704 	free(name);
705 
706 	if (nvlist_lookup_nvlist_array(nv, ZPOOL_CONFIG_CHILDREN,
707 	    &child, &children) != 0)
708 		return (max);
709 
710 	for (c = 0; c < children; c++)
711 		if ((ret = max_width(zhp, child[c], depth + 2, max)) > max)
712 			max = ret;
713 
714 	return (max);
715 }
716 
717 
718 /*
719  * Print the configuration of an exported pool.  Iterate over all vdevs in the
720  * pool, printing out the name and status for each one.
721  */
722 void
723 print_import_config(const char *name, nvlist_t *nv, int namewidth, int depth)
724 {
725 	nvlist_t **child;
726 	uint_t c, children;
727 	vdev_stat_t *vs;
728 	char *type, *vname;
729 
730 	verify(nvlist_lookup_string(nv, ZPOOL_CONFIG_TYPE, &type) == 0);
731 	if (strcmp(type, VDEV_TYPE_MISSING) == 0)
732 		return;
733 
734 	verify(nvlist_lookup_uint64_array(nv, ZPOOL_CONFIG_STATS,
735 	    (uint64_t **)&vs, &c) == 0);
736 
737 	(void) printf("\t%*s%-*s", depth, "", namewidth - depth, name);
738 
739 	if (vs->vs_aux != 0) {
740 		(void) printf("  %-8s  ", state_to_name(vs->vs_state));
741 
742 		switch (vs->vs_aux) {
743 		case VDEV_AUX_OPEN_FAILED:
744 			(void) printf(gettext("cannot open"));
745 			break;
746 
747 		case VDEV_AUX_BAD_GUID_SUM:
748 			(void) printf(gettext("missing device"));
749 			break;
750 
751 		case VDEV_AUX_NO_REPLICAS:
752 			(void) printf(gettext("insufficient replicas"));
753 			break;
754 
755 		default:
756 			(void) printf(gettext("corrupted data"));
757 			break;
758 		}
759 	} else {
760 		(void) printf("  %s", state_to_name(vs->vs_state));
761 	}
762 	(void) printf("\n");
763 
764 	if (nvlist_lookup_nvlist_array(nv, ZPOOL_CONFIG_CHILDREN,
765 	    &child, &children) != 0)
766 		return;
767 
768 	for (c = 0; c < children; c++) {
769 		vname = zpool_vdev_name(NULL, child[c]);
770 		print_import_config(vname, child[c],
771 		    namewidth, depth + 2);
772 		free(vname);
773 	}
774 }
775 
776 /*
777  * Display the status for the given pool.
778  */
779 static void
780 show_import(nvlist_t *config)
781 {
782 	uint64_t pool_state;
783 	vdev_stat_t *vs;
784 	char *name;
785 	uint64_t guid;
786 	char *msgid;
787 	nvlist_t *nvroot;
788 	int reason;
789 	char *health;
790 	uint_t vsc;
791 	int namewidth;
792 
793 	verify(nvlist_lookup_string(config, ZPOOL_CONFIG_POOL_NAME,
794 	    &name) == 0);
795 	verify(nvlist_lookup_uint64(config, ZPOOL_CONFIG_POOL_GUID,
796 	    &guid) == 0);
797 	verify(nvlist_lookup_uint64(config, ZPOOL_CONFIG_POOL_STATE,
798 	    &pool_state) == 0);
799 	verify(nvlist_lookup_string(config, ZPOOL_CONFIG_POOL_HEALTH,
800 	    &health) == 0);
801 	verify(nvlist_lookup_nvlist(config, ZPOOL_CONFIG_VDEV_TREE,
802 	    &nvroot) == 0);
803 
804 	verify(nvlist_lookup_uint64_array(nvroot, ZPOOL_CONFIG_STATS,
805 	    (uint64_t **)&vs, &vsc) == 0);
806 
807 	reason = zpool_import_status(config, &msgid);
808 
809 	(void) printf("  pool: %s\n", name);
810 	(void) printf("    id: %llu\n", guid);
811 	(void) printf(" state: %s\n", health);
812 
813 	switch (reason) {
814 	case ZPOOL_STATUS_MISSING_DEV_R:
815 	case ZPOOL_STATUS_MISSING_DEV_NR:
816 	case ZPOOL_STATUS_BAD_GUID_SUM:
817 		(void) printf(gettext("status: One or more devices are missing "
818 		    "from the system.\n"));
819 		break;
820 
821 	case ZPOOL_STATUS_CORRUPT_LABEL_R:
822 	case ZPOOL_STATUS_CORRUPT_LABEL_NR:
823 		(void) printf(gettext("status: One or more devices contains "
824 		    "corrupted data.\n"));
825 		break;
826 
827 	case ZPOOL_STATUS_CORRUPT_DATA:
828 		(void) printf(gettext("status: The pool data is corrupted.\n"));
829 		break;
830 
831 	default:
832 		/*
833 		 * No other status can be seen when importing pools.
834 		 */
835 		assert(reason == ZPOOL_STATUS_OK);
836 	}
837 
838 	/*
839 	 * Print out an action according to the overall state of the pool.
840 	 */
841 	if (strcmp(health, gettext("ONLINE")) == 0) {
842 		(void) printf(gettext("action: The pool can be imported"
843 		    " using its name or numeric identifier."));
844 		if (pool_state != POOL_STATE_EXPORTED)
845 			(void) printf(gettext("  The\n\tpool may be active on "
846 			    "on another system, but can be imported using\n\t"
847 			    "the '-f' flag.\n"));
848 		else
849 			(void) printf("\n");
850 	} else if (strcmp(health, gettext("DEGRADED")) == 0) {
851 		(void) printf(gettext("action: The pool can be imported "
852 		    "despite missing or damaged devices.  The\n\tfault "
853 		    "tolerance of the pool may be compromised if imported."));
854 		if (pool_state != POOL_STATE_EXPORTED)
855 			(void) printf(gettext("  The\n\tpool may be active on "
856 			    "on another system, but can be imported using\n\t"
857 			    "the '-f' flag.\n"));
858 		else
859 			(void) printf("\n");
860 	} else {
861 		if (reason == ZPOOL_STATUS_MISSING_DEV_R ||
862 		    reason == ZPOOL_STATUS_MISSING_DEV_NR ||
863 		    reason == ZPOOL_STATUS_BAD_GUID_SUM)
864 			(void) printf(gettext("action: The pool cannot be "
865 			    "imported. Attach the missing\n\tdevices and try "
866 			    "again.\n"));
867 		else
868 			(void) printf(gettext("action: The pool cannot be "
869 			    "imported due to damaged devices or data.\n"));
870 	}
871 
872 	if (msgid != NULL)
873 		(void) printf(gettext("   see: http://www.sun.com/msg/%s\n"),
874 		    msgid);
875 
876 	(void) printf(gettext("config:\n\n"));
877 
878 	namewidth = max_width(NULL, nvroot, 0, 0);
879 	if (namewidth < 10)
880 		namewidth = 10;
881 	print_import_config(name, nvroot, namewidth, 0);
882 
883 	if (reason == ZPOOL_STATUS_BAD_GUID_SUM) {
884 		(void) printf("\n\tAdditional devices are known to "
885 		    "be part of this pool, though their\n\texact "
886 		    "configuration cannot be determined.\n");
887 	}
888 }
889 
890 /*
891  * Perform the import for the given configuration.  This passes the heavy
892  * lifting off to zpool_import(), and then mounts the datasets contained within
893  * the pool.
894  */
895 static int
896 do_import(nvlist_t *config, const char *newname, const char *mntopts,
897     const char *altroot, int force)
898 {
899 	zpool_handle_t *zhp;
900 	char *name;
901 	uint64_t state;
902 
903 	verify(nvlist_lookup_string(config, ZPOOL_CONFIG_POOL_NAME,
904 	    &name) == 0);
905 
906 	verify(nvlist_lookup_uint64(config,
907 	    ZPOOL_CONFIG_POOL_STATE, &state) == 0);
908 	if (state != POOL_STATE_EXPORTED && !force) {
909 		(void) fprintf(stderr, gettext("cannot import '%s': pool "
910 		    "may be in use from other system\n"), name);
911 		(void) fprintf(stderr, gettext("use '-f' to import anyway\n"));
912 		return (1);
913 	}
914 
915 	if (zpool_import(config, newname, altroot) != 0)
916 		return (1);
917 
918 	if (newname != NULL)
919 		name = (char *)newname;
920 
921 	verify((zhp = zpool_open(name)) != NULL);
922 
923 	if (mount_share_datasets(zhp, mntopts) != 0) {
924 		zpool_close(zhp);
925 		return (1);
926 	}
927 
928 	zpool_close(zhp);
929 	return (0);
930 }
931 
932 /*
933  * zpool import [-d dir]
934  *       import [-R root] [-d dir] [-f] -a
935  *       import [-R root] [-d dir] [-f] <pool | id> [newpool]
936  *
937  *       -d	Scan in a specific directory, other than /dev/dsk.  More than
938  *		one directory can be specified using multiple '-d' options.
939  *
940  *       -R	Temporarily import the pool, with all mountpoints relative to
941  *		the given root.  The pool will remain exported when the machine
942  *		is rebooted.
943  *
944  *       -f	Force import, even if it appears that the pool is active.
945  *
946  *       -a	Import all pools found.
947  *
948  * The import command scans for pools to import, and import pools based on pool
949  * name and GUID.  The pool can also be renamed as part of the import process.
950  */
951 int
952 zpool_do_import(int argc, char **argv)
953 {
954 	char **searchdirs = NULL;
955 	int nsearch = 0;
956 	int c;
957 	int err;
958 	nvlist_t *pools;
959 	int do_all = FALSE;
960 	char *altroot = NULL;
961 	char *mntopts = NULL;
962 	int do_force = FALSE;
963 	nvpair_t *elem;
964 	nvlist_t *config;
965 	uint64_t searchguid;
966 	char *searchname;
967 	nvlist_t *found_config;
968 	int first;
969 
970 	/* check options */
971 	while ((c = getopt(argc, argv, ":fd:R:ao:")) != -1) {
972 		switch (c) {
973 		case 'a':
974 			do_all = TRUE;
975 			break;
976 		case 'd':
977 			if (searchdirs == NULL) {
978 				searchdirs = safe_malloc(sizeof (char *));
979 			} else {
980 				char **tmp = safe_malloc((nsearch + 1) *
981 				    sizeof (char *));
982 				bcopy(searchdirs, tmp, nsearch *
983 				    sizeof (char *));
984 				free(searchdirs);
985 				searchdirs = tmp;
986 			}
987 			searchdirs[nsearch++] = optarg;
988 			break;
989 		case 'f':
990 			do_force = TRUE;
991 			break;
992 		case 'o':
993 			mntopts = optarg;
994 			break;
995 		case 'R':
996 			altroot = optarg;
997 			break;
998 		case ':':
999 			(void) fprintf(stderr, gettext("missing argument for "
1000 			    "'%c' option\n"), optopt);
1001 			usage(FALSE);
1002 			break;
1003 		case '?':
1004 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
1005 			    optopt);
1006 			usage(FALSE);
1007 		}
1008 	}
1009 
1010 	argc -= optind;
1011 	argv += optind;
1012 
1013 	if (searchdirs == NULL) {
1014 		searchdirs = safe_malloc(sizeof (char *));
1015 		searchdirs[0] = "/dev/dsk";
1016 		nsearch = 1;
1017 	}
1018 
1019 	/* check argument count */
1020 	if (do_all) {
1021 		if (argc != 0) {
1022 			(void) fprintf(stderr, gettext("too many arguments\n"));
1023 			usage(FALSE);
1024 		}
1025 	} else {
1026 		if (argc > 2) {
1027 			(void) fprintf(stderr, gettext("too many arguments\n"));
1028 			usage(FALSE);
1029 		}
1030 
1031 		/*
1032 		 * Check for the SYS_CONFIG privilege.  We do this explicitly
1033 		 * here because otherwise any attempt to discover pools will
1034 		 * silently fail.
1035 		 */
1036 		if (argc == 0 && !priv_ineffect(PRIV_SYS_CONFIG)) {
1037 			(void) fprintf(stderr, gettext("cannot "
1038 			    "discover pools: permission denied\n"));
1039 			return (1);
1040 		}
1041 	}
1042 
1043 	if ((pools = zpool_find_import(nsearch, searchdirs)) == NULL)
1044 		return (1);
1045 
1046 	/*
1047 	 * We now have a list of all available pools in the given directories.
1048 	 * Depending on the arguments given, we do one of the following:
1049 	 *
1050 	 *	<none>	Iterate through all pools and display information about
1051 	 *		each one.
1052 	 *
1053 	 *	-a	Iterate through all pools and try to import each one.
1054 	 *
1055 	 *	<id>	Find the pool that corresponds to the given GUID/pool
1056 	 *		name and import that one.
1057 	 */
1058 	if (argc != 0) {
1059 		char *endptr;
1060 
1061 		errno = 0;
1062 		searchguid = strtoull(argv[0], &endptr, 10);
1063 		if (errno != 0 || *endptr != '\0')
1064 			searchname = argv[0];
1065 		else
1066 			searchname = NULL;
1067 		found_config = NULL;
1068 	}
1069 
1070 	err = 0;
1071 	elem = NULL;
1072 	first = TRUE;
1073 	while ((elem = nvlist_next_nvpair(pools, elem)) != NULL) {
1074 
1075 		verify(nvpair_value_nvlist(elem, &config) == 0);
1076 
1077 		if (argc == 0) {
1078 			if (first)
1079 				first = FALSE;
1080 			else
1081 				(void) printf("\n");
1082 
1083 			if (do_all)
1084 				err |= do_import(config, NULL, mntopts,
1085 				    altroot, do_force);
1086 			else
1087 				show_import(config);
1088 		} else if (searchname != NULL) {
1089 			char *name;
1090 
1091 			/*
1092 			 * We are searching for a pool based on name.
1093 			 */
1094 			verify(nvlist_lookup_string(config,
1095 			    ZPOOL_CONFIG_POOL_NAME, &name) == 0);
1096 
1097 			if (strcmp(name, searchname) == 0) {
1098 				if (found_config != NULL) {
1099 					(void) fprintf(stderr, gettext(
1100 					    "cannot import '%s': more than "
1101 					    "one matching pool\n"), searchname);
1102 					(void) fprintf(stderr, gettext(
1103 					    "import by numeric ID instead\n"));
1104 					err = TRUE;
1105 				}
1106 				found_config = config;
1107 			}
1108 		} else {
1109 			uint64_t guid;
1110 
1111 			/*
1112 			 * Search for a pool by guid.
1113 			 */
1114 			verify(nvlist_lookup_uint64(config,
1115 			    ZPOOL_CONFIG_POOL_GUID, &guid) == 0);
1116 
1117 			if (guid == searchguid)
1118 				found_config = config;
1119 		}
1120 	}
1121 
1122 	/*
1123 	 * If we were searching for a specific pool, verify that we found a
1124 	 * pool, and then do the import.
1125 	 */
1126 	if (argc != 0 && err == 0) {
1127 		if (found_config == NULL) {
1128 			(void) fprintf(stderr, gettext("cannot import '%s': "
1129 			    "no such pool available\n"), argv[0]);
1130 			err = TRUE;
1131 		} else {
1132 			err |= do_import(found_config, argc == 1 ? NULL :
1133 			    argv[1], mntopts, altroot, do_force);
1134 		}
1135 	}
1136 
1137 	/*
1138 	 * If we were just looking for pools, report an error if none were
1139 	 * found.
1140 	 */
1141 	if (argc == 0 && first)
1142 		(void) fprintf(stderr,
1143 		    gettext("no pools available to import\n"));
1144 
1145 	nvlist_free(pools);
1146 
1147 	return (err ? 1 : 0);
1148 }
1149 
1150 typedef struct iostat_cbdata {
1151 	zpool_list_t *cb_list;
1152 	int cb_verbose;
1153 	int cb_iteration;
1154 	int cb_namewidth;
1155 } iostat_cbdata_t;
1156 
1157 static void
1158 print_iostat_separator(iostat_cbdata_t *cb)
1159 {
1160 	int i = 0;
1161 
1162 	for (i = 0; i < cb->cb_namewidth; i++)
1163 		(void) printf("-");
1164 	(void) printf("  -----  -----  -----  -----  -----  -----\n");
1165 }
1166 
1167 static void
1168 print_iostat_header(iostat_cbdata_t *cb)
1169 {
1170 	(void) printf("%*s     capacity     operations    bandwidth\n",
1171 	    cb->cb_namewidth, "");
1172 	(void) printf("%-*s   used  avail   read  write   read  write\n",
1173 	    cb->cb_namewidth, "pool");
1174 	print_iostat_separator(cb);
1175 }
1176 
1177 /*
1178  * Display a single statistic.
1179  */
1180 void
1181 print_one_stat(uint64_t value)
1182 {
1183 	char buf[64];
1184 
1185 	zfs_nicenum(value, buf, sizeof (buf));
1186 	(void) printf("  %5s", buf);
1187 }
1188 
1189 /*
1190  * Print out all the statistics for the given vdev.  This can either be the
1191  * toplevel configuration, or called recursively.  If 'name' is NULL, then this
1192  * is a verbose output, and we don't want to display the toplevel pool stats.
1193  */
1194 void
1195 print_vdev_stats(zpool_handle_t *zhp, const char *name, nvlist_t *oldnv,
1196     nvlist_t *newnv, iostat_cbdata_t *cb, int depth)
1197 {
1198 	nvlist_t **oldchild, **newchild;
1199 	uint_t c, children;
1200 	vdev_stat_t *oldvs, *newvs;
1201 	vdev_stat_t zerovs = { 0 };
1202 	uint64_t tdelta;
1203 	double scale;
1204 	char *vname;
1205 
1206 	if (oldnv != NULL) {
1207 		verify(nvlist_lookup_uint64_array(oldnv, ZPOOL_CONFIG_STATS,
1208 		    (uint64_t **)&oldvs, &c) == 0);
1209 	} else {
1210 		oldvs = &zerovs;
1211 	}
1212 
1213 	verify(nvlist_lookup_uint64_array(newnv, ZPOOL_CONFIG_STATS,
1214 	    (uint64_t **)&newvs, &c) == 0);
1215 
1216 	if (strlen(name) + depth > cb->cb_namewidth)
1217 		(void) printf("%*s%s", depth, "", name);
1218 	else
1219 		(void) printf("%*s%s%*s", depth, "", name,
1220 		    (int)(cb->cb_namewidth - strlen(name) - depth), "");
1221 
1222 	tdelta = newvs->vs_timestamp - oldvs->vs_timestamp;
1223 
1224 	if (tdelta == 0)
1225 		scale = 1.0;
1226 	else
1227 		scale = (double)NANOSEC / tdelta;
1228 
1229 	/* only toplevel vdevs have capacity stats */
1230 	if (newvs->vs_space == 0) {
1231 		(void) printf("      -      -");
1232 	} else {
1233 		print_one_stat(newvs->vs_alloc);
1234 		print_one_stat(newvs->vs_space - newvs->vs_alloc);
1235 	}
1236 
1237 	print_one_stat((uint64_t)(scale * (newvs->vs_ops[ZIO_TYPE_READ] -
1238 	    oldvs->vs_ops[ZIO_TYPE_READ])));
1239 
1240 	print_one_stat((uint64_t)(scale * (newvs->vs_ops[ZIO_TYPE_WRITE] -
1241 	    oldvs->vs_ops[ZIO_TYPE_WRITE])));
1242 
1243 	print_one_stat((uint64_t)(scale * (newvs->vs_bytes[ZIO_TYPE_READ] -
1244 	    oldvs->vs_bytes[ZIO_TYPE_READ])));
1245 
1246 	print_one_stat((uint64_t)(scale * (newvs->vs_bytes[ZIO_TYPE_WRITE] -
1247 	    oldvs->vs_bytes[ZIO_TYPE_WRITE])));
1248 
1249 	(void) printf("\n");
1250 
1251 	if (!cb->cb_verbose)
1252 		return;
1253 
1254 	if (nvlist_lookup_nvlist_array(newnv, ZPOOL_CONFIG_CHILDREN,
1255 	    &newchild, &children) != 0)
1256 		return;
1257 
1258 	if (oldnv && nvlist_lookup_nvlist_array(oldnv, ZPOOL_CONFIG_CHILDREN,
1259 	    &oldchild, &c) != 0)
1260 		return;
1261 
1262 	for (c = 0; c < children; c++) {
1263 		vname = zpool_vdev_name(zhp, newchild[c]);
1264 		print_vdev_stats(zhp, vname, oldnv ? oldchild[c] : NULL,
1265 		    newchild[c], cb, depth + 2);
1266 		free(vname);
1267 	}
1268 }
1269 
1270 static int
1271 refresh_iostat(zpool_handle_t *zhp, void *data)
1272 {
1273 	iostat_cbdata_t *cb = data;
1274 
1275 	/*
1276 	 * If the pool has disappeared, remove it from the list and continue.
1277 	 */
1278 	if (zpool_refresh_stats(zhp) != 0)
1279 		pool_list_remove(cb->cb_list, zhp);
1280 
1281 	return (0);
1282 }
1283 
1284 /*
1285  * Callback to print out the iostats for the given pool.
1286  */
1287 int
1288 print_iostat(zpool_handle_t *zhp, void *data)
1289 {
1290 	iostat_cbdata_t *cb = data;
1291 	nvlist_t *oldconfig, *newconfig;
1292 	nvlist_t *oldnvroot, *newnvroot;
1293 
1294 	newconfig = zpool_get_config(zhp, &oldconfig);
1295 
1296 	if (cb->cb_iteration == 1)
1297 		oldconfig = NULL;
1298 
1299 	verify(nvlist_lookup_nvlist(newconfig, ZPOOL_CONFIG_VDEV_TREE,
1300 	    &newnvroot) == 0);
1301 
1302 	if (oldconfig == NULL)
1303 		oldnvroot = NULL;
1304 	else
1305 		verify(nvlist_lookup_nvlist(oldconfig, ZPOOL_CONFIG_VDEV_TREE,
1306 		    &oldnvroot) == 0);
1307 
1308 	/*
1309 	 * Print out the statistics for the pool.
1310 	 */
1311 	print_vdev_stats(zhp, zpool_get_name(zhp), oldnvroot, newnvroot, cb, 0);
1312 
1313 	if (cb->cb_verbose)
1314 		print_iostat_separator(cb);
1315 
1316 	return (0);
1317 }
1318 
1319 int
1320 get_namewidth(zpool_handle_t *zhp, void *data)
1321 {
1322 	iostat_cbdata_t *cb = data;
1323 	nvlist_t *config, *nvroot;
1324 
1325 	if ((config = zpool_get_config(zhp, NULL)) != NULL) {
1326 		verify(nvlist_lookup_nvlist(config, ZPOOL_CONFIG_VDEV_TREE,
1327 		    &nvroot) == 0);
1328 		if (!cb->cb_verbose)
1329 			cb->cb_namewidth = strlen(zpool_get_name(zhp));
1330 		else
1331 			cb->cb_namewidth = max_width(zhp, nvroot, 0, 0);
1332 	}
1333 
1334 	/*
1335 	 * The width must fall into the range [10,38].  The upper limit is the
1336 	 * maximum we can have and still fit in 80 columns.
1337 	 */
1338 	if (cb->cb_namewidth < 10)
1339 		cb->cb_namewidth = 10;
1340 	if (cb->cb_namewidth > 38)
1341 		cb->cb_namewidth = 38;
1342 
1343 	return (0);
1344 }
1345 
1346 /*
1347  * zpool iostat [-v] [pool] ... [interval [count]]
1348  *
1349  *	-v	Display statistics for individual vdevs
1350  *
1351  * This command can be tricky because we want to be able to deal with pool
1352  * creation/destruction as well as vdev configuration changes.  The bulk of this
1353  * processing is handled by the pool_list_* routines in zpool_iter.c.  We rely
1354  * on pool_list_update() to detect the addition of new pools.  Configuration
1355  * changes are all handled within libzfs.
1356  */
1357 int
1358 zpool_do_iostat(int argc, char **argv)
1359 {
1360 	int c;
1361 	int ret;
1362 	int npools;
1363 	unsigned long interval = 0, count = 0;
1364 	zpool_list_t *list;
1365 	int verbose = FALSE;
1366 	iostat_cbdata_t cb;
1367 
1368 	/* check options */
1369 	while ((c = getopt(argc, argv, "v")) != -1) {
1370 		switch (c) {
1371 		case 'v':
1372 			verbose = TRUE;
1373 			break;
1374 		case '?':
1375 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
1376 			    optopt);
1377 			usage(FALSE);
1378 		}
1379 	}
1380 
1381 	argc -= optind;
1382 	argv += optind;
1383 
1384 	/*
1385 	 * Determine if the last argument is an integer or a pool name
1386 	 */
1387 	if (argc > 0 && isdigit(argv[argc - 1][0])) {
1388 		char *end;
1389 
1390 		errno = 0;
1391 		interval = strtoul(argv[argc - 1], &end, 10);
1392 
1393 		if (*end == '\0' && errno == 0) {
1394 			if (interval == 0) {
1395 				(void) fprintf(stderr, gettext("interval "
1396 				    "cannot be zero\n"));
1397 				usage(FALSE);
1398 			}
1399 
1400 			/*
1401 			 * Ignore the last parameter
1402 			 */
1403 			argc--;
1404 		} else {
1405 			/*
1406 			 * If this is not a valid number, just plow on.  The
1407 			 * user will get a more informative error message later
1408 			 * on.
1409 			 */
1410 			interval = 0;
1411 		}
1412 	}
1413 
1414 	/*
1415 	 * If the last argument is also an integer, then we have both a count
1416 	 * and an integer.
1417 	 */
1418 	if (argc > 0 && isdigit(argv[argc - 1][0])) {
1419 		char *end;
1420 
1421 		errno = 0;
1422 		count = interval;
1423 		interval = strtoul(argv[argc - 1], &end, 10);
1424 
1425 		if (*end == '\0' && errno == 0) {
1426 			if (interval == 0) {
1427 				(void) fprintf(stderr, gettext("interval "
1428 				    "cannot be zero\n"));
1429 				usage(FALSE);
1430 			}
1431 
1432 			/*
1433 			 * Ignore the last parameter
1434 			 */
1435 			argc--;
1436 		} else {
1437 			interval = 0;
1438 		}
1439 	}
1440 
1441 	/*
1442 	 * Construct the list of all interesting pools.
1443 	 */
1444 	ret = 0;
1445 	if ((list = pool_list_get(argc, argv, &ret)) == NULL)
1446 		return (1);
1447 
1448 	if (pool_list_count(list) == 0 && argc != 0)
1449 		return (1);
1450 
1451 	if (pool_list_count(list) == 0 && interval == 0) {
1452 		(void) fprintf(stderr, gettext("no pools available\n"));
1453 		return (1);
1454 	}
1455 
1456 	/*
1457 	 * Enter the main iostat loop.
1458 	 */
1459 	cb.cb_list = list;
1460 	cb.cb_verbose = verbose;
1461 	cb.cb_iteration = 0;
1462 	cb.cb_namewidth = 0;
1463 
1464 	for (;;) {
1465 		pool_list_update(list);
1466 
1467 		if ((npools = pool_list_count(list)) == 0)
1468 			break;
1469 
1470 		/*
1471 		 * Refresh all statistics.  This is done as an explicit step
1472 		 * before calculating the maximum name width, so that any
1473 		 * configuration changes are properly accounted for.
1474 		 */
1475 		(void) pool_list_iter(list, FALSE, refresh_iostat, &cb);
1476 
1477 		/*
1478 		 * Iterate over all pools to determine the maximum width
1479 		 * for the pool / device name column across all pools.
1480 		 */
1481 		cb.cb_namewidth = 0;
1482 		(void) pool_list_iter(list, FALSE, get_namewidth, &cb);
1483 
1484 		/*
1485 		 * If it's the first time, or verbose mode, print the header.
1486 		 */
1487 		if (++cb.cb_iteration == 1 || verbose)
1488 			print_iostat_header(&cb);
1489 
1490 		(void) pool_list_iter(list, FALSE, print_iostat, &cb);
1491 
1492 		/*
1493 		 * If there's more than one pool, and we're not in verbose mode
1494 		 * (which prints a separator for us), then print a separator.
1495 		 */
1496 		if (npools > 1 && !verbose)
1497 			print_iostat_separator(&cb);
1498 
1499 		if (verbose)
1500 			(void) printf("\n");
1501 
1502 		if (interval == 0)
1503 			break;
1504 
1505 		if (count != 0 && --count == 0)
1506 			break;
1507 
1508 		(void) sleep(interval);
1509 	}
1510 
1511 	pool_list_free(list);
1512 
1513 	return (ret);
1514 }
1515 
1516 typedef struct list_cbdata {
1517 	int	cb_scripted;
1518 	int	cb_first;
1519 	int	cb_fields[MAX_FIELDS];
1520 	int	cb_fieldcount;
1521 } list_cbdata_t;
1522 
1523 /*
1524  * Given a list of columns to display, output appropriate headers for each one.
1525  */
1526 void
1527 print_header(int *fields, size_t count)
1528 {
1529 	int i;
1530 	column_def_t *col;
1531 	const char *fmt;
1532 
1533 	for (i = 0; i < count; i++) {
1534 		col = &column_table[fields[i]];
1535 		if (i != 0)
1536 			(void) printf("  ");
1537 		if (col->cd_justify == left_justify)
1538 			fmt = "%-*s";
1539 		else
1540 			fmt = "%*s";
1541 
1542 		(void) printf(fmt, i == count - 1 ? strlen(col->cd_title) :
1543 		    col->cd_width, col->cd_title);
1544 	}
1545 
1546 	(void) printf("\n");
1547 }
1548 
1549 int
1550 list_callback(zpool_handle_t *zhp, void *data)
1551 {
1552 	list_cbdata_t *cbp = data;
1553 	nvlist_t *config;
1554 	int i;
1555 	char buf[ZPOOL_MAXNAMELEN];
1556 	uint64_t total;
1557 	uint64_t used;
1558 	const char *fmt;
1559 	column_def_t *col;
1560 
1561 	if (cbp->cb_first) {
1562 		if (!cbp->cb_scripted)
1563 			print_header(cbp->cb_fields, cbp->cb_fieldcount);
1564 		cbp->cb_first = FALSE;
1565 	}
1566 
1567 	if (zpool_get_state(zhp) == POOL_STATE_UNAVAIL) {
1568 		config = NULL;
1569 	} else {
1570 		config = zpool_get_config(zhp, NULL);
1571 		total = zpool_get_space_total(zhp);
1572 		used = zpool_get_space_used(zhp);
1573 	}
1574 
1575 	for (i = 0; i < cbp->cb_fieldcount; i++) {
1576 		if (i != 0) {
1577 			if (cbp->cb_scripted)
1578 				(void) printf("\t");
1579 			else
1580 				(void) printf("  ");
1581 		}
1582 
1583 		col = &column_table[cbp->cb_fields[i]];
1584 
1585 		switch (cbp->cb_fields[i]) {
1586 		case ZPOOL_FIELD_NAME:
1587 			(void) strlcpy(buf, zpool_get_name(zhp), sizeof (buf));
1588 			break;
1589 
1590 		case ZPOOL_FIELD_SIZE:
1591 			if (config == NULL)
1592 				(void) strlcpy(buf, "-", sizeof (buf));
1593 			else
1594 				zfs_nicenum(total, buf, sizeof (buf));
1595 			break;
1596 
1597 		case ZPOOL_FIELD_USED:
1598 			if (config == NULL)
1599 				(void) strlcpy(buf, "-", sizeof (buf));
1600 			else
1601 				zfs_nicenum(used, buf, sizeof (buf));
1602 			break;
1603 
1604 		case ZPOOL_FIELD_AVAILABLE:
1605 			if (config == NULL)
1606 				(void) strlcpy(buf, "-", sizeof (buf));
1607 			else
1608 				zfs_nicenum(total - used, buf, sizeof (buf));
1609 			break;
1610 
1611 		case ZPOOL_FIELD_CAPACITY:
1612 			if (config == NULL) {
1613 				(void) strlcpy(buf, "-", sizeof (buf));
1614 			} else {
1615 				uint64_t capacity = (total == 0 ? 0 :
1616 				    (used * 100 / total));
1617 				(void) snprintf(buf, sizeof (buf), "%llu%%",
1618 				    capacity);
1619 			}
1620 			break;
1621 
1622 		case ZPOOL_FIELD_HEALTH:
1623 			if (config == NULL) {
1624 				(void) strlcpy(buf, "FAULTED", sizeof (buf));
1625 			} else {
1626 				nvlist_t *nvroot;
1627 				vdev_stat_t *vs;
1628 				uint_t vsc;
1629 
1630 				verify(nvlist_lookup_nvlist(config,
1631 				    ZPOOL_CONFIG_VDEV_TREE, &nvroot) == 0);
1632 				verify(nvlist_lookup_uint64_array(nvroot,
1633 				    ZPOOL_CONFIG_STATS, (uint64_t **)&vs,
1634 				    &vsc) == 0);
1635 				(void) strlcpy(buf, state_to_name(vs->vs_state),
1636 				    sizeof (buf));
1637 			}
1638 			break;
1639 
1640 		case ZPOOL_FIELD_ROOT:
1641 			if (config == NULL)
1642 				(void) strlcpy(buf, "-", sizeof (buf));
1643 			else if (zpool_get_root(zhp, buf, sizeof (buf)) != 0)
1644 				(void) strlcpy(buf, "-", sizeof (buf));
1645 			break;
1646 		}
1647 
1648 		if (cbp->cb_scripted)
1649 			(void) printf("%s", buf);
1650 		else {
1651 			if (col->cd_justify == left_justify)
1652 				fmt = "%-*s";
1653 			else
1654 				fmt = "%*s";
1655 
1656 			(void) printf(fmt, i == cbp->cb_fieldcount - 1 ?
1657 			    strlen(buf) : col->cd_width, buf);
1658 		}
1659 	}
1660 
1661 	(void) printf("\n");
1662 
1663 	return (0);
1664 }
1665 
1666 /*
1667  * zpool list [-H] [-o field[,field]*] [pool] ...
1668  *
1669  *	-H	Scripted mode.  Don't display headers, and separate fields by
1670  *		a single tab.
1671  *	-o	List of fields to display.  Defaults to all fields, or
1672  *		"name,size,used,available,capacity,health,root"
1673  *
1674  * List all pools in the system, whether or not they're healthy.  Output space
1675  * statistics for each one, as well as health status summary.
1676  */
1677 int
1678 zpool_do_list(int argc, char **argv)
1679 {
1680 	int c;
1681 	int ret;
1682 	list_cbdata_t cb = { 0 };
1683 	static char default_fields[] =
1684 	    "name,size,used,available,capacity,health,root";
1685 	char *fields = default_fields;
1686 	char *value;
1687 
1688 	/* check options */
1689 	while ((c = getopt(argc, argv, ":Ho:")) != -1) {
1690 		switch (c) {
1691 		case 'H':
1692 			cb.cb_scripted = TRUE;
1693 			break;
1694 		case 'o':
1695 			fields = optarg;
1696 			break;
1697 		case ':':
1698 			(void) fprintf(stderr, gettext("missing argument for "
1699 			    "'%c' option\n"), optopt);
1700 			usage(FALSE);
1701 			break;
1702 		case '?':
1703 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
1704 			    optopt);
1705 			usage(FALSE);
1706 		}
1707 	}
1708 
1709 	argc -= optind;
1710 	argv += optind;
1711 
1712 	while (*fields != '\0') {
1713 		if (cb.cb_fieldcount == MAX_FIELDS) {
1714 			(void) fprintf(stderr, gettext("too many "
1715 			    "properties given to -o option\n"));
1716 			usage(FALSE);
1717 		}
1718 
1719 		if ((cb.cb_fields[cb.cb_fieldcount] = getsubopt(&fields,
1720 		    column_subopts, &value)) == -1) {
1721 			(void) fprintf(stderr, gettext("invalid property "
1722 			    "'%s'\n"), value);
1723 			usage(FALSE);
1724 		}
1725 
1726 		cb.cb_fieldcount++;
1727 	}
1728 
1729 
1730 	cb.cb_first = TRUE;
1731 
1732 	ret = for_each_pool(argc, argv, TRUE, list_callback, &cb);
1733 
1734 	if (argc == 0 && cb.cb_first) {
1735 		(void) printf(gettext("no pools available\n"));
1736 		return (0);
1737 	}
1738 
1739 	return (ret);
1740 }
1741 
1742 static nvlist_t *
1743 zpool_get_vdev_by_name(nvlist_t *nv, char *name)
1744 {
1745 	nvlist_t **child;
1746 	uint_t c, children;
1747 	nvlist_t *match;
1748 	char *path;
1749 
1750 	if (nvlist_lookup_nvlist_array(nv, ZPOOL_CONFIG_CHILDREN,
1751 	    &child, &children) != 0) {
1752 		verify(nvlist_lookup_string(nv, ZPOOL_CONFIG_PATH, &path) == 0);
1753 		if (strncmp(name, "/dev/dsk/", 9) == 0)
1754 			name += 9;
1755 		if (strncmp(path, "/dev/dsk/", 9) == 0)
1756 			path += 9;
1757 		if (strcmp(name, path) == 0)
1758 			return (nv);
1759 		return (NULL);
1760 	}
1761 
1762 	for (c = 0; c < children; c++)
1763 		if ((match = zpool_get_vdev_by_name(child[c], name)) != NULL)
1764 			return (match);
1765 
1766 	return (NULL);
1767 }
1768 
1769 static int
1770 zpool_do_attach_or_replace(int argc, char **argv, int replacing)
1771 {
1772 	int force = FALSE;
1773 	int c;
1774 	nvlist_t *nvroot;
1775 	char *poolname, *old_disk, *new_disk;
1776 	zpool_handle_t *zhp;
1777 	nvlist_t *config;
1778 
1779 	/* check options */
1780 	while ((c = getopt(argc, argv, "f")) != -1) {
1781 		switch (c) {
1782 		case 'f':
1783 			force = TRUE;
1784 			break;
1785 		case '?':
1786 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
1787 			    optopt);
1788 			usage(FALSE);
1789 		}
1790 	}
1791 
1792 	argc -= optind;
1793 	argv += optind;
1794 
1795 	/* get pool name and check number of arguments */
1796 	if (argc < 1) {
1797 		(void) fprintf(stderr, gettext("missing pool name argument\n"));
1798 		usage(FALSE);
1799 	}
1800 
1801 	poolname = argv[0];
1802 
1803 	if (argc < 2) {
1804 		(void) fprintf(stderr,
1805 		    gettext("missing <device> specification\n"));
1806 		usage(FALSE);
1807 	}
1808 
1809 	old_disk = argv[1];
1810 
1811 	if (argc < 3) {
1812 		if (!replacing) {
1813 			(void) fprintf(stderr,
1814 			    gettext("missing <new_device> specification\n"));
1815 			usage(FALSE);
1816 		}
1817 		new_disk = old_disk;
1818 		argc -= 1;
1819 		argv += 1;
1820 	} else {
1821 		new_disk = argv[2];
1822 		argc -= 2;
1823 		argv += 2;
1824 	}
1825 
1826 	if (argc > 1) {
1827 		(void) fprintf(stderr, gettext("too many arguments\n"));
1828 		usage(FALSE);
1829 	}
1830 
1831 	if ((zhp = zpool_open(poolname)) == NULL)
1832 		return (1);
1833 
1834 	if ((config = zpool_get_config(zhp, NULL)) == NULL) {
1835 		(void) fprintf(stderr, gettext("pool '%s' is unavailable\n"),
1836 		    poolname);
1837 		zpool_close(zhp);
1838 		return (1);
1839 	}
1840 
1841 	nvroot = make_root_vdev(config, force, B_FALSE, argc, argv);
1842 	if (nvroot == NULL) {
1843 		zpool_close(zhp);
1844 		return (1);
1845 	}
1846 
1847 	return (zpool_vdev_attach(zhp, old_disk, new_disk, nvroot, replacing));
1848 }
1849 
1850 /*
1851  * zpool replace [-f] <pool> <device> <new_device>
1852  *
1853  *	-f	Force attach, even if <new_device> appears to be in use.
1854  *
1855  * Replace <device> with <new_device>.
1856  */
1857 /* ARGSUSED */
1858 int
1859 zpool_do_replace(int argc, char **argv)
1860 {
1861 	return (zpool_do_attach_or_replace(argc, argv, B_TRUE));
1862 }
1863 
1864 /*
1865  * zpool attach [-f] <pool> <device> <new_device>
1866  *
1867  *	-f	Force attach, even if <new_device> appears to be in use.
1868  *
1869  * Attach <new_device> to the mirror containing <device>.  If <device> is not
1870  * part of a mirror, then <device> will be transformed into a mirror of
1871  * <device> and <new_device>.  In either case, <new_device> will begin life
1872  * with a DTL of [0, now], and will immediately begin to resilver itself.
1873  */
1874 int
1875 zpool_do_attach(int argc, char **argv)
1876 {
1877 	return (zpool_do_attach_or_replace(argc, argv, B_FALSE));
1878 }
1879 
1880 /*
1881  * zpool detach [-f] <pool> <device>
1882  *
1883  *	-f	Force detach of <device>, even if DTLs argue against it
1884  *		(not supported yet)
1885  *
1886  * Detach a device from a mirror.  The operation will be refused if <device>
1887  * is the last device in the mirror, or if the DTLs indicate that this device
1888  * has the only valid copy of some data.
1889  */
1890 /* ARGSUSED */
1891 int
1892 zpool_do_detach(int argc, char **argv)
1893 {
1894 	int c;
1895 	char *poolname, *path;
1896 	zpool_handle_t *zhp;
1897 
1898 	/* check options */
1899 	while ((c = getopt(argc, argv, "f")) != -1) {
1900 		switch (c) {
1901 		case 'f':
1902 		case '?':
1903 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
1904 			    optopt);
1905 			usage(FALSE);
1906 		}
1907 	}
1908 
1909 	argc -= optind;
1910 	argv += optind;
1911 
1912 	/* get pool name and check number of arguments */
1913 	if (argc < 1) {
1914 		(void) fprintf(stderr, gettext("missing pool name argument\n"));
1915 		usage(FALSE);
1916 	}
1917 
1918 	if (argc < 2) {
1919 		(void) fprintf(stderr,
1920 		    gettext("missing <device> specification\n"));
1921 		usage(FALSE);
1922 	}
1923 
1924 	poolname = argv[0];
1925 	path = argv[1];
1926 
1927 	if ((zhp = zpool_open(poolname)) == NULL)
1928 		return (1);
1929 
1930 	return (zpool_vdev_detach(zhp, path));
1931 }
1932 
1933 /*
1934  * zpool online [-t] <pool> <device>
1935  *
1936  *	-t	Only bring the device on-line temporarily.  The online
1937  *		state will not be persistent across reboots.
1938  */
1939 /* ARGSUSED */
1940 int
1941 zpool_do_online(int argc, char **argv)
1942 {
1943 	int c, i;
1944 	char *poolname;
1945 	zpool_handle_t *zhp;
1946 	int ret = 0;
1947 
1948 	/* check options */
1949 	while ((c = getopt(argc, argv, "t")) != -1) {
1950 		switch (c) {
1951 		case 't':
1952 		case '?':
1953 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
1954 			    optopt);
1955 			usage(FALSE);
1956 		}
1957 	}
1958 
1959 	argc -= optind;
1960 	argv += optind;
1961 
1962 	/* get pool name and check number of arguments */
1963 	if (argc < 1) {
1964 		(void) fprintf(stderr, gettext("missing pool name\n"));
1965 		usage(FALSE);
1966 	}
1967 	if (argc < 2) {
1968 		(void) fprintf(stderr, gettext("missing device name\n"));
1969 		usage(FALSE);
1970 	}
1971 
1972 	poolname = argv[0];
1973 
1974 	if ((zhp = zpool_open(poolname)) == NULL)
1975 		return (1);
1976 
1977 	for (i = 1; i < argc; i++)
1978 		if (zpool_vdev_online(zhp, argv[i]) == 0)
1979 			(void) printf(gettext("Bringing device %s online\n"),
1980 			    argv[i]);
1981 		else
1982 			ret = 1;
1983 
1984 	return (ret);
1985 }
1986 
1987 /*
1988  * zpool offline [-ft] <pool> <device>
1989  *
1990  *	-f	Force the device into the offline state, even if doing
1991  *		so would appear to compromise pool availability.
1992  *		(not supported yet)
1993  *
1994  *	-t	Only take the device off-line temporarily.  The offline
1995  *		state will not be persistent across reboots.
1996  *		(not supported yet)
1997  */
1998 /* ARGSUSED */
1999 int
2000 zpool_do_offline(int argc, char **argv)
2001 {
2002 	int c, i;
2003 	char *poolname;
2004 	zpool_handle_t *zhp;
2005 	int ret = 0;
2006 
2007 	/* check options */
2008 	while ((c = getopt(argc, argv, "ft")) != -1) {
2009 		switch (c) {
2010 		case 'f':
2011 		case 't':
2012 		case '?':
2013 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
2014 			    optopt);
2015 			usage(FALSE);
2016 		}
2017 	}
2018 
2019 	argc -= optind;
2020 	argv += optind;
2021 
2022 	/* get pool name and check number of arguments */
2023 	if (argc < 1) {
2024 		(void) fprintf(stderr, gettext("missing pool name\n"));
2025 		usage(FALSE);
2026 	}
2027 	if (argc < 2) {
2028 		(void) fprintf(stderr, gettext("missing device name\n"));
2029 		usage(FALSE);
2030 	}
2031 
2032 	poolname = argv[0];
2033 
2034 	if ((zhp = zpool_open(poolname)) == NULL)
2035 		return (1);
2036 
2037 	for (i = 1; i < argc; i++)
2038 		if (zpool_vdev_offline(zhp, argv[i]) == 0)
2039 			(void) printf(gettext("Bringing device %s offline\n"),
2040 			    argv[i]);
2041 		else
2042 			ret = 1;
2043 
2044 	return (ret);
2045 }
2046 
2047 typedef struct scrub_cbdata {
2048 	int	cb_type;
2049 } scrub_cbdata_t;
2050 
2051 int
2052 scrub_callback(zpool_handle_t *zhp, void *data)
2053 {
2054 	scrub_cbdata_t *cb = data;
2055 
2056 	return (zpool_scrub(zhp, cb->cb_type) != 0);
2057 }
2058 
2059 /*
2060  * zpool scrub [-s] <pool> ...
2061  *
2062  *	-s	Stop.  Stops any in-progress scrub.
2063  */
2064 int
2065 zpool_do_scrub(int argc, char **argv)
2066 {
2067 	int c;
2068 	scrub_cbdata_t cb;
2069 
2070 	cb.cb_type = POOL_SCRUB_EVERYTHING;
2071 
2072 	/* check options */
2073 	while ((c = getopt(argc, argv, "s")) != -1) {
2074 		switch (c) {
2075 		case 's':
2076 			cb.cb_type = POOL_SCRUB_NONE;
2077 			break;
2078 		case '?':
2079 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
2080 			    optopt);
2081 			usage(FALSE);
2082 		}
2083 	}
2084 
2085 	argc -= optind;
2086 	argv += optind;
2087 
2088 	if (argc < 1) {
2089 		(void) fprintf(stderr, gettext("missing pool name argument\n"));
2090 		usage(FALSE);
2091 	}
2092 
2093 	return (for_each_pool(argc, argv, TRUE, scrub_callback, &cb));
2094 }
2095 
2096 typedef struct status_cbdata {
2097 	int	cb_verbose;
2098 	int	cb_explain;
2099 	int	cb_count;
2100 	int	cb_first;
2101 } status_cbdata_t;
2102 
2103 /*
2104  * Print out detailed scrub status.
2105  */
2106 void
2107 print_scrub_status(nvlist_t *nvroot)
2108 {
2109 	vdev_stat_t *vs;
2110 	uint_t vsc;
2111 	time_t start, end, now;
2112 	double fraction_done;
2113 	uint64_t examined, total, minutes_left;
2114 	char *scrub_type;
2115 
2116 	verify(nvlist_lookup_uint64_array(nvroot, ZPOOL_CONFIG_STATS,
2117 	    (uint64_t **)&vs, &vsc) == 0);
2118 
2119 	/*
2120 	 * If there's never been a scrub, there's not much to say.
2121 	 */
2122 	if (vs->vs_scrub_end == 0 && vs->vs_scrub_type == POOL_SCRUB_NONE) {
2123 		(void) printf(gettext("none requested\n"));
2124 		return;
2125 	}
2126 
2127 	scrub_type = (vs->vs_scrub_type == POOL_SCRUB_RESILVER) ?
2128 	    "resilver" : "scrub";
2129 
2130 	start = vs->vs_scrub_start;
2131 	end = vs->vs_scrub_end;
2132 	now = time(NULL);
2133 	examined = vs->vs_scrub_examined;
2134 	total = vs->vs_alloc;
2135 
2136 	if (end != 0) {
2137 		(void) printf(gettext("%s %s with %llu errors on %s"),
2138 		    scrub_type, vs->vs_scrub_complete ? "completed" : "stopped",
2139 		    (u_longlong_t)vs->vs_scrub_errors, ctime(&end));
2140 		return;
2141 	}
2142 
2143 	if (examined == 0)
2144 		examined = 1;
2145 	if (examined > total)
2146 		total = examined;
2147 
2148 	fraction_done = (double)examined / total;
2149 	minutes_left = (uint64_t)((now - start) *
2150 	    (1 - fraction_done) / fraction_done / 60);
2151 
2152 	(void) printf(gettext("%s in progress, %.2f%% done, %lluh%um to go\n"),
2153 	    scrub_type, 100 * fraction_done,
2154 	    (u_longlong_t)(minutes_left / 60), (uint_t)(minutes_left % 60));
2155 }
2156 
2157 /*
2158  * Print out configuration state as requested by status_callback.
2159  */
2160 void
2161 print_status_config(zpool_handle_t *zhp, const char *name, nvlist_t *nv,
2162     int namewidth, int depth)
2163 {
2164 	nvlist_t **child;
2165 	uint_t c, children;
2166 	vdev_stat_t *vs;
2167 	char rbuf[6], wbuf[6], cbuf[6], repaired[6];
2168 	char *vname;
2169 
2170 	verify(nvlist_lookup_uint64_array(nv, ZPOOL_CONFIG_STATS,
2171 	    (uint64_t **)&vs, &c) == 0);
2172 
2173 	if (nvlist_lookup_nvlist_array(nv, ZPOOL_CONFIG_CHILDREN,
2174 	    &child, &children) != 0)
2175 		children = 0;
2176 
2177 	(void) printf("\t%*s%-*s  %-8s", depth, "", namewidth - depth,
2178 	    name, state_to_name(vs->vs_state));
2179 
2180 	zfs_nicenum(vs->vs_read_errors, rbuf, sizeof (rbuf));
2181 	zfs_nicenum(vs->vs_write_errors, wbuf, sizeof (wbuf));
2182 	zfs_nicenum(vs->vs_checksum_errors, cbuf, sizeof (cbuf));
2183 	(void) printf(" %5s %5s %5s", rbuf, wbuf, cbuf);
2184 
2185 	if (vs->vs_aux != 0) {
2186 		(void) printf("  ");
2187 
2188 		switch (vs->vs_aux) {
2189 		case VDEV_AUX_OPEN_FAILED:
2190 			(void) printf(gettext("cannot open"));
2191 			break;
2192 
2193 		case VDEV_AUX_BAD_GUID_SUM:
2194 			(void) printf(gettext("missing device"));
2195 			break;
2196 
2197 		case VDEV_AUX_NO_REPLICAS:
2198 			(void) printf(gettext("insufficient replicas"));
2199 			break;
2200 
2201 		default:
2202 			(void) printf(gettext("corrupted data"));
2203 			break;
2204 		}
2205 	} else if (vs->vs_scrub_repaired != 0 && children == 0) {
2206 		/*
2207 		 * Report bytes resilvered/repaired on leaf devices.
2208 		 */
2209 		zfs_nicenum(vs->vs_scrub_repaired, repaired, sizeof (repaired));
2210 		(void) printf(gettext("  %s %s"), repaired,
2211 		    (vs->vs_scrub_type == POOL_SCRUB_RESILVER) ?
2212 		    "resilvered" : "repaired");
2213 	}
2214 
2215 	(void) printf("\n");
2216 
2217 	for (c = 0; c < children; c++) {
2218 		vname = zpool_vdev_name(zhp, child[c]);
2219 		print_status_config(zhp, vname, child[c],
2220 		    namewidth, depth + 2);
2221 		free(vname);
2222 	}
2223 }
2224 
2225 /*
2226  * Display a summary of pool status.  Displays a summary such as:
2227  *
2228  *        pool: tank
2229  *	status: DEGRADED
2230  *	reason: One or more devices ...
2231  *         see: http://www.sun.com/msg/ZFS-xxxx-01
2232  *	config:
2233  *		mirror		DEGRADED
2234  *                c1t0d0	OK
2235  *                c2t0d0	FAULTED
2236  *
2237  * When given the '-v' option, we print out the complete config.  If the '-e'
2238  * option is specified, then we print out error rate information as well.
2239  */
2240 int
2241 status_callback(zpool_handle_t *zhp, void *data)
2242 {
2243 	status_cbdata_t *cbp = data;
2244 	nvlist_t *config, *nvroot;
2245 	char *msgid;
2246 	int reason;
2247 	char *health;
2248 
2249 	config = zpool_get_config(zhp, NULL);
2250 	reason = zpool_get_status(zhp, &msgid);
2251 
2252 	cbp->cb_count++;
2253 
2254 	/*
2255 	 * If we were given 'zpool status -x', only report those pools with
2256 	 * problems.
2257 	 */
2258 	if (reason == ZPOOL_STATUS_OK && cbp->cb_explain)
2259 		return (0);
2260 
2261 	if (cbp->cb_first)
2262 		cbp->cb_first = FALSE;
2263 	else
2264 		(void) printf("\n");
2265 
2266 	verify(nvlist_lookup_string(config, ZPOOL_CONFIG_POOL_HEALTH,
2267 	    &health) == 0);
2268 
2269 	(void) printf(gettext("  pool: %s\n"), zpool_get_name(zhp));
2270 	(void) printf(gettext(" state: %s\n"), health);
2271 
2272 	switch (reason) {
2273 	case ZPOOL_STATUS_MISSING_DEV_R:
2274 		(void) printf(gettext("status: One or more devices could not "
2275 		    "be opened.  Sufficient replicas exist for\n\tthe pool to "
2276 		    "continue functioning in a degraded state.\n"));
2277 		(void) printf(gettext("action: Attach the missing device and "
2278 		    "online it using 'zpool online'.\n"));
2279 		break;
2280 
2281 	case ZPOOL_STATUS_MISSING_DEV_NR:
2282 		(void) printf(gettext("status: One or more devices could not "
2283 		    "be opened.  There are insufficient\n\treplicas for the "
2284 		    "pool to continue functioning.\n"));
2285 		(void) printf(gettext("action: Attach the missing device and "
2286 		    "online it using 'zpool online'.\n"));
2287 		break;
2288 
2289 	case ZPOOL_STATUS_CORRUPT_LABEL_R:
2290 		(void) printf(gettext("status: One or more devices could not "
2291 		    "be used because the label is missing or\n\tinvalid.  "
2292 		    "Sufficient replicas exist for the pool to continue\n\t"
2293 		    "functioning in a degraded state.\n"));
2294 		(void) printf(gettext("action: Replace the device using "
2295 		    "'zpool replace'.\n"));
2296 		break;
2297 
2298 	case ZPOOL_STATUS_CORRUPT_LABEL_NR:
2299 		(void) printf(gettext("status: One or more devices could not "
2300 		    "be used because the the label is missing \n\tor invalid.  "
2301 		    "There are insufficient replicas for the pool to "
2302 		    "continue\n\tfunctioning.\n"));
2303 		(void) printf(gettext("action: Destroy and re-create the pool "
2304 		    "from a backup source.\n"));
2305 		break;
2306 
2307 	case ZPOOL_STATUS_FAILING_DEV:
2308 		(void) printf(gettext("status: One or more devices has "
2309 		    "experienced an unrecoverable error.  An\n\tattempt was "
2310 		    "made to correct the error.  Applications are "
2311 		    "unaffected.\n"));
2312 		(void) printf(gettext("action: Determine if the device needs "
2313 		    "to be replaced, and clear the errors\n\tusing "
2314 		    "'zpool online' or replace the device with 'zpool "
2315 		    "replace'.\n"));
2316 		break;
2317 
2318 	case ZPOOL_STATUS_OFFLINE_DEV:
2319 		(void) printf(gettext("status: One or more devices has "
2320 		    "been taken offline by the adminstrator.\n\tSufficient "
2321 		    "replicas exist for the pool to continue functioning in "
2322 		    "a\n\tdegraded state.\n"));
2323 		(void) printf(gettext("action: Online the device using "
2324 		    "'zpool online' or replace the device with\n\t'zpool "
2325 		    "replace'.\n"));
2326 		break;
2327 
2328 	case ZPOOL_STATUS_RESILVERING:
2329 		(void) printf(gettext("status: One or more devices is "
2330 		    "currently being resilvered.  The pool will\n\tcontinue "
2331 		    "to function, possibly in a degraded state.\n"));
2332 		(void) printf(gettext("action: Wait for the resilver to "
2333 		    "complete.\n"));
2334 		break;
2335 
2336 	default:
2337 		/*
2338 		 * The remaining errors can't actually be generated, yet.
2339 		 */
2340 		assert(reason == ZPOOL_STATUS_OK);
2341 	}
2342 
2343 	if (msgid != NULL)
2344 		(void) printf(gettext("   see: http://www.sun.com/msg/%s\n"),
2345 		    msgid);
2346 
2347 	if (config != NULL) {
2348 		int namewidth;
2349 
2350 		verify(nvlist_lookup_nvlist(config, ZPOOL_CONFIG_VDEV_TREE,
2351 		    &nvroot) == 0);
2352 
2353 		(void) printf(gettext(" scrub: "));
2354 		print_scrub_status(nvroot);
2355 
2356 		namewidth = max_width(zhp, nvroot, 0, 0);
2357 		if (namewidth < 10)
2358 			namewidth = 10;
2359 
2360 		(void) printf(gettext("config:\n\n"));
2361 		(void) printf(gettext("\t%-*s  %-8s %5s %5s %5s\n"), namewidth,
2362 		    "NAME", "STATE", "READ", "WRITE", "CKSUM");
2363 		print_status_config(zhp, zpool_get_name(zhp), nvroot,
2364 		    namewidth, 0);
2365 	} else {
2366 		(void) printf(gettext("config: The configuration cannot be "
2367 		    "determined.\n"));
2368 	}
2369 
2370 	return (0);
2371 }
2372 
2373 /*
2374  * zpool status [-vx] [pool] ...
2375  *
2376  *	-v	Display complete error logs
2377  *	-x	Display only pools with potential problems
2378  *
2379  * Describes the health status of all pools or some subset.
2380  */
2381 int
2382 zpool_do_status(int argc, char **argv)
2383 {
2384 	int c;
2385 	int ret;
2386 	status_cbdata_t cb = { 0 };
2387 
2388 	/* check options */
2389 	while ((c = getopt(argc, argv, "vx")) != -1) {
2390 		switch (c) {
2391 		case 'v':
2392 			cb.cb_verbose = TRUE;
2393 			break;
2394 		case 'x':
2395 			cb.cb_explain = TRUE;
2396 			break;
2397 		case '?':
2398 			(void) fprintf(stderr, gettext("invalid option '%c'\n"),
2399 			    optopt);
2400 			usage(FALSE);
2401 		}
2402 	}
2403 
2404 	argc -= optind;
2405 	argv += optind;
2406 
2407 	cb.cb_first = TRUE;
2408 
2409 	ret = for_each_pool(argc, argv, TRUE, status_callback, &cb);
2410 
2411 	if (argc == 0 && cb.cb_count == 0)
2412 		(void) printf(gettext("no pools available\n"));
2413 	else if (cb.cb_explain && cb.cb_first) {
2414 		if (argc == 0) {
2415 			(void) printf(gettext("all pools are healthy\n"));
2416 		} else {
2417 			int i;
2418 			for (i = 0; i < argc; i++)
2419 				(void) printf(gettext("pool '%s' is healthy\n"),
2420 				    argv[i]);
2421 		}
2422 	}
2423 
2424 	return (ret);
2425 }
2426 
2427 int
2428 main(int argc, char **argv)
2429 {
2430 	int ret;
2431 	int i;
2432 	char *cmdname;
2433 
2434 	(void) setlocale(LC_ALL, "");
2435 	(void) textdomain(TEXT_DOMAIN);
2436 
2437 	opterr = 0;
2438 
2439 	/*
2440 	 * Make sure the user has specified some command.
2441 	 */
2442 	if (argc < 2) {
2443 		(void) fprintf(stderr, gettext("missing command\n"));
2444 		usage(FALSE);
2445 	}
2446 
2447 	cmdname = argv[1];
2448 
2449 	/*
2450 	 * Special case '-?'
2451 	 */
2452 	if (strcmp(cmdname, "-?") == 0)
2453 		usage(TRUE);
2454 
2455 	/*
2456 	 * Run the appropriate command.
2457 	 */
2458 	for (i = 0; i < NCOMMAND; i++) {
2459 		if (command_table[i].name == NULL)
2460 			continue;
2461 
2462 		if (strcmp(cmdname, command_table[i].name) == 0) {
2463 			current_command = &command_table[i];
2464 			ret = command_table[i].func(argc - 1, argv + 1);
2465 			break;
2466 		}
2467 	}
2468 
2469 	/*
2470 	 * 'freeze' is a vile debugging abomination, so we treat it as such.
2471 	 */
2472 	if (strcmp(cmdname, "freeze") == 0 && argc == 3) {
2473 		char buf[8192];
2474 		int fd = open("/dev/zpoolctl", O_RDWR);
2475 		(void) strcpy((void *)buf, argv[2]);
2476 		return (!!ioctl(fd, ZFS_IOC_POOL_FREEZE, buf));
2477 	}
2478 
2479 	if (i == NCOMMAND) {
2480 		(void) fprintf(stderr, gettext("unrecognized "
2481 		    "command '%s'\n"), cmdname);
2482 		usage(FALSE);
2483 	}
2484 
2485 	/*
2486 	 * The 'ZFS_ABORT' environment variable causes us to dump core on exit
2487 	 * for the purposes of running ::findleaks.
2488 	 */
2489 	if (getenv("ZFS_ABORT") != NULL) {
2490 		(void) printf("dumping core by request\n");
2491 		abort();
2492 	}
2493 
2494 	return (ret);
2495 }
2496