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