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