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 (c) 2009, 2010, Oracle and/or its affiliates. All rights reserved. 24 */ 25 26 /* 27 * Copyright 2012 Nexenta Systems, Inc. All rights reserved. 28 */ 29 30 /* 31 * System includes 32 */ 33 34 #include <assert.h> 35 #include <stdio.h> 36 #include <strings.h> 37 #include <libzfs.h> 38 #include <locale.h> 39 #include <langinfo.h> 40 #include <stdlib.h> 41 #include <wchar.h> 42 #include <sys/types.h> 43 44 #include "libbe.h" 45 46 #ifndef lint 47 #define _(x) gettext(x) 48 #else 49 #define _(x) (x) 50 #endif 51 52 #ifndef TEXT_DOMAIN 53 #define TEXT_DOMAIN "SYS_TEST" 54 #endif 55 56 #define DT_BUF_LEN (128) 57 #define NUM_COLS (6) 58 59 static int be_do_activate(int argc, char **argv); 60 static int be_do_create(int argc, char **argv); 61 static int be_do_destroy(int argc, char **argv); 62 static int be_do_list(int argc, char **argv); 63 static int be_do_mount(int argc, char **argv); 64 static int be_do_unmount(int argc, char **argv); 65 static int be_do_rename(int argc, char **argv); 66 static int be_do_rollback(int argc, char **argv); 67 static void usage(void); 68 69 /* 70 * single column name/width output format description 71 */ 72 struct col_info { 73 const char *col_name; 74 size_t width; 75 }; 76 77 /* 78 * all columns output format 79 */ 80 struct hdr_info { 81 struct col_info cols[NUM_COLS]; 82 }; 83 84 /* 85 * type of possible output formats 86 */ 87 enum be_fmt { 88 BE_FMT_DEFAULT, 89 BE_FMT_DATASET, 90 BE_FMT_SNAPSHOT, 91 BE_FMT_ALL 92 }; 93 94 /* 95 * command handler description 96 */ 97 typedef struct be_command { 98 const char *name; 99 int (*func)(int argc, char **argv); 100 } be_command_t; 101 102 /* 103 * sorted list of be commands 104 */ 105 static const be_command_t be_command_tbl[] = { 106 { "activate", be_do_activate }, 107 { "create", be_do_create }, 108 { "destroy", be_do_destroy }, 109 { "list", be_do_list }, 110 { "mount", be_do_mount }, 111 { "unmount", be_do_unmount }, 112 { "umount", be_do_unmount }, /* unmount alias */ 113 { "rename", be_do_rename }, 114 { "rollback", be_do_rollback }, 115 { NULL, NULL }, 116 }; 117 118 static void 119 usage(void) 120 { 121 (void) fprintf(stderr, _("usage:\n" 122 "\tbeadm subcommand cmd_options\n" 123 "\n" 124 "\tsubcommands:\n" 125 "\n" 126 "\tbeadm activate [-v] beName\n" 127 "\tbeadm create [-a] [-d BE_desc]\n" 128 "\t\t[-o property=value] ... [-p zpool] \n" 129 "\t\t[-e nonActiveBe | beName@snapshot] [-v] beName\n" 130 "\tbeadm create [-d BE_desc]\n" 131 "\t\t[-o property=value] ... [-p zpool] [-v] beName@snapshot\n" 132 "\tbeadm destroy [-Ffsv] beName \n" 133 "\tbeadm destroy [-Fv] beName@snapshot \n" 134 "\tbeadm list [[-a] | [-d] [-s]] [-H] [-v] [beName]\n" 135 "\tbeadm mount [-s ro|rw] [-v] beName [mountpoint]\n" 136 "\tbeadm unmount [-fv] beName | mountpoint\n" 137 "\tbeadm umount [-fv] beName | mountpoint\n" 138 "\tbeadm rename [-v] origBeName newBeName\n" 139 "\tbeadm rollback [-v] beName snapshot\n" 140 "\tbeadm rollback [-v] beName@snapshot\n")); 141 } 142 143 static int 144 run_be_cmd(const char *cmdname, int argc, char **argv) 145 { 146 const be_command_t *command; 147 148 for (command = &be_command_tbl[0]; command->name != NULL; command++) 149 if (strcmp(command->name, cmdname) == 0) 150 return (command->func(argc, argv)); 151 152 (void) fprintf(stderr, _("Invalid command: %s\n"), cmdname); 153 usage(); 154 return (1); 155 } 156 157 int 158 main(int argc, char **argv) 159 { 160 const char *cmdname; 161 162 (void) setlocale(LC_ALL, ""); 163 (void) textdomain(TEXT_DOMAIN); 164 165 if (argc < 2) { 166 usage(); 167 return (1); 168 } 169 170 cmdname = argv[1]; 171 172 /* Turn error printing off */ 173 libbe_print_errors(B_FALSE); 174 175 return (run_be_cmd(cmdname, --argc, ++argv)); 176 } 177 178 static void 179 print_hdr(struct hdr_info *hdr_info) 180 { 181 boolean_t first = B_TRUE; 182 size_t i; 183 for (i = 0; i < NUM_COLS; i++) { 184 struct col_info *col_info = &hdr_info->cols[i]; 185 const char *name = col_info->col_name; 186 size_t width = col_info->width; 187 if (name == NULL) 188 continue; 189 190 if (first) { 191 (void) printf("%-*s", width, name); 192 first = B_FALSE; 193 } else 194 (void) printf(" %-*s", width, name); 195 } 196 (void) putchar('\n'); 197 } 198 199 static void 200 init_hdr_cols(enum be_fmt be_fmt, struct hdr_info *hdr) 201 { 202 struct col_info *col = hdr->cols; 203 size_t i; 204 205 col[1].col_name = _("Active"); 206 col[2].col_name = _("Mountpoint"); 207 col[3].col_name = _("Space"); 208 col[4].col_name = _("Policy"); 209 col[5].col_name = _("Created"); 210 col[6].col_name = NULL; 211 212 switch (be_fmt) { 213 case BE_FMT_ALL: 214 col[0].col_name = _("BE/Dataset/Snapshot"); 215 break; 216 case BE_FMT_DATASET: 217 col[0].col_name = _("BE/Dataset"); 218 break; 219 case BE_FMT_SNAPSHOT: 220 col[0].col_name = _("BE/Snapshot"); 221 col[1].col_name = NULL; 222 col[2].col_name = NULL; 223 break; 224 case BE_FMT_DEFAULT: 225 default: 226 col[0].col_name = _("BE"); 227 } 228 229 for (i = 0; i < NUM_COLS; i++) { 230 const char *name = col[i].col_name; 231 col[i].width = 0; 232 233 if (name != NULL) { 234 wchar_t wname[128]; 235 size_t sz = mbstowcs(wname, name, sizeof (wname) / 236 sizeof (wchar_t)); 237 if (sz > 0) { 238 int wcsw = wcswidth(wname, sz); 239 if (wcsw > 0) 240 col[i].width = wcsw; 241 else 242 col[i].width = sz; 243 } else { 244 col[i].width = strlen(name); 245 } 246 } 247 } 248 } 249 250 static void 251 nicenum(uint64_t num, char *buf, size_t buflen) 252 { 253 uint64_t n = num; 254 int index = 0; 255 char u; 256 257 while (n >= 1024) { 258 n /= 1024; 259 index++; 260 } 261 262 u = " KMGTPE"[index]; 263 264 if (index == 0) { 265 (void) snprintf(buf, buflen, "%llu", n); 266 } else { 267 int i; 268 for (i = 2; i >= 0; i--) { 269 if (snprintf(buf, buflen, "%.*f%c", i, 270 (double)num / (1ULL << 10 * index), u) <= 5) 271 break; 272 } 273 } 274 } 275 276 static void 277 count_widths(enum be_fmt be_fmt, struct hdr_info *hdr, be_node_list_t *be_nodes) 278 { 279 size_t len[NUM_COLS]; 280 char buf[DT_BUF_LEN]; 281 int i; 282 be_node_list_t *cur_be; 283 284 for (i = 0; i < NUM_COLS; i++) 285 len[i] = hdr->cols[i].width; 286 287 for (cur_be = be_nodes; cur_be != NULL; cur_be = cur_be->be_next_node) { 288 char name[ZFS_MAXNAMELEN+1]; 289 const char *be_name = cur_be->be_node_name; 290 const char *root_ds = cur_be->be_root_ds; 291 char *pos; 292 size_t node_name_len = strlen(cur_be->be_node_name); 293 size_t root_ds_len = strlen(cur_be->be_root_ds); 294 size_t mntpt_len = 0; 295 size_t policy_len = 0; 296 size_t used_len; 297 uint64_t used = cur_be->be_space_used; 298 be_snapshot_list_t *snap = NULL; 299 300 if (cur_be->be_mntpt != NULL) 301 mntpt_len = strlen(cur_be->be_mntpt); 302 if (cur_be->be_policy_type != NULL) 303 policy_len = strlen(cur_be->be_policy_type); 304 305 (void) strlcpy(name, root_ds, sizeof (name)); 306 pos = strstr(name, be_name); 307 308 if (be_fmt == BE_FMT_DEFAULT) { 309 if (node_name_len > len[0]) 310 len[0] = node_name_len; 311 } else { 312 if (root_ds_len + 3 > len[0]) 313 len[0] = root_ds_len + 3; 314 } 315 316 if (mntpt_len > len[2]) 317 len[2] = mntpt_len; 318 if (policy_len > len[4]) 319 len[4] = policy_len; 320 321 for (snap = cur_be->be_node_snapshots; snap != NULL; 322 snap = snap->be_next_snapshot) { 323 uint64_t snap_used = snap->be_snapshot_space_used; 324 const char *snap_name = snap->be_snapshot_name; 325 (void) strcpy(pos, snap_name); 326 327 if (be_fmt == BE_FMT_DEFAULT) 328 used += snap_used; 329 else if (be_fmt & BE_FMT_SNAPSHOT) { 330 int snap_len = strlen(name) + 3; 331 if (be_fmt == BE_FMT_SNAPSHOT) 332 snap_len -= pos - name; 333 if (snap_len > len[0]) 334 len[0] = snap_len; 335 nicenum(snap_used, buf, sizeof (buf)); 336 used_len = strlen(buf); 337 if (used_len > len[3]) 338 len[3] = used_len; 339 } 340 } 341 342 if (be_fmt == BE_FMT_DEFAULT) { 343 int used_len; 344 nicenum(used, buf, sizeof (buf)); 345 used_len = strlen(buf); 346 if (used_len > len[3]) 347 len[3] = used_len; 348 } 349 350 nicenum(used, buf, sizeof (buf)); 351 } 352 353 for (i = 0; i < NUM_COLS; i++) 354 hdr->cols[i].width = len[i]; 355 } 356 357 static void 358 print_be_nodes(const char *be_name, boolean_t parsable, struct hdr_info *hdr, 359 be_node_list_t *nodes) 360 { 361 char buf[64]; 362 char datetime[DT_BUF_LEN]; 363 be_node_list_t *cur_be; 364 365 for (cur_be = nodes; cur_be != NULL; cur_be = cur_be->be_next_node) { 366 char active[3] = "-\0"; 367 int ai = 0; 368 const char *datetime_fmt = "%F %R"; 369 const char *name = cur_be->be_node_name; 370 const char *mntpt = cur_be->be_mntpt; 371 be_snapshot_list_t *snap = NULL; 372 uint64_t used = cur_be->be_space_used; 373 time_t creation = cur_be->be_node_creation; 374 struct tm *tm; 375 376 if (be_name != NULL && strcmp(be_name, name) != 0) 377 continue; 378 379 if (parsable) 380 active[0] = '\0'; 381 382 tm = localtime(&creation); 383 (void) strftime(datetime, DT_BUF_LEN, datetime_fmt, tm); 384 385 for (snap = cur_be->be_node_snapshots; snap != NULL; 386 snap = snap->be_next_snapshot) 387 used += snap->be_snapshot_space_used; 388 389 if (cur_be->be_active) 390 active[ai++] = 'N'; 391 if (cur_be->be_active_on_boot) 392 active[ai] = 'R'; 393 394 nicenum(used, buf, sizeof (buf)); 395 if (parsable) 396 (void) printf("%s;%s;%s;%s;%llu;%s;%ld\n", 397 name, 398 cur_be->be_uuid_str, 399 active, 400 (cur_be->be_mounted ? mntpt: ""), 401 used, 402 cur_be->be_policy_type, 403 creation); 404 else 405 (void) printf("%-*s %-*s %-*s %-*s %-*s %-*s\n", 406 hdr->cols[0].width, name, 407 hdr->cols[1].width, active, 408 hdr->cols[2].width, (cur_be->be_mounted ? mntpt: 409 "-"), 410 hdr->cols[3].width, buf, 411 hdr->cols[4].width, cur_be->be_policy_type, 412 hdr->cols[5].width, datetime); 413 } 414 } 415 416 static void 417 print_be_snapshots(be_node_list_t *be, struct hdr_info *hdr, boolean_t parsable) 418 { 419 char buf[64]; 420 char datetime[DT_BUF_LEN]; 421 be_snapshot_list_t *snap = NULL; 422 423 for (snap = be->be_node_snapshots; snap != NULL; 424 snap = snap->be_next_snapshot) { 425 char name[ZFS_MAXNAMELEN+1]; 426 const char *datetime_fmt = "%F %R"; 427 const char *be_name = be->be_node_name; 428 const char *root_ds = be->be_root_ds; 429 const char *snap_name = snap->be_snapshot_name; 430 char *pos; 431 uint64_t used = snap->be_snapshot_space_used; 432 time_t creation = snap->be_snapshot_creation; 433 struct tm *tm = localtime(&creation); 434 435 (void) strncpy(name, root_ds, sizeof (name)); 436 pos = strstr(name, be_name); 437 (void) strcpy(pos, snap_name); 438 439 (void) strftime(datetime, DT_BUF_LEN, datetime_fmt, tm); 440 nicenum(used, buf, sizeof (buf)); 441 442 if (parsable) 443 if (hdr->cols[1].width != 0) 444 (void) printf("%s;%s;%s;%s;%llu;%s;%ld\n", 445 be_name, 446 snap_name, 447 "", 448 "", 449 used, 450 be->be_policy_type, 451 creation); 452 else 453 (void) printf("%s;%s;%llu;%s;%ld\n", 454 be_name, 455 snap_name, 456 used, 457 be->be_policy_type, 458 creation); 459 else 460 if (hdr->cols[1].width != 0) 461 (void) printf(" %-*s %-*s %-*s %-*s %-*s " 462 "%-*s\n", 463 hdr->cols[0].width-3, name, 464 hdr->cols[1].width, "-", 465 hdr->cols[2].width, "-", 466 hdr->cols[3].width, buf, 467 hdr->cols[4].width, be->be_policy_type, 468 hdr->cols[5].width, datetime); 469 else 470 (void) printf(" %-*s %-*s %-*s %-*s\n", 471 hdr->cols[0].width-3, snap_name, 472 hdr->cols[3].width, buf, 473 hdr->cols[4].width, be->be_policy_type, 474 hdr->cols[5].width, datetime); 475 } 476 } 477 478 static void 479 print_fmt_nodes(const char *be_name, enum be_fmt be_fmt, boolean_t parsable, 480 struct hdr_info *hdr, be_node_list_t *nodes) 481 { 482 char buf[64]; 483 char datetime[DT_BUF_LEN]; 484 be_node_list_t *cur_be; 485 486 for (cur_be = nodes; cur_be != NULL; cur_be = cur_be->be_next_node) { 487 char active[3] = "-\0"; 488 int ai = 0; 489 const char *datetime_fmt = "%F %R"; 490 const char *name = cur_be->be_node_name; 491 const char *mntpt = cur_be->be_mntpt; 492 uint64_t used = cur_be->be_space_used; 493 time_t creation = cur_be->be_node_creation; 494 struct tm *tm; 495 496 if (be_name != NULL && strcmp(be_name, name) != 0) 497 continue; 498 499 if (!parsable) 500 (void) printf("%-s\n", name); 501 else 502 active[0] = '\0'; 503 504 tm = localtime(&creation); 505 (void) strftime(datetime, DT_BUF_LEN, datetime_fmt, tm); 506 507 if (cur_be->be_active) 508 active[ai++] = 'N'; 509 if (cur_be->be_active_on_boot) 510 active[ai] = 'R'; 511 512 nicenum(used, buf, sizeof (buf)); 513 if (be_fmt & BE_FMT_DATASET) 514 if (parsable) 515 (void) printf("%s;%s;%s;%s;%llu;%s;%ld\n", 516 cur_be->be_node_name, 517 cur_be->be_root_ds, 518 active, 519 (cur_be->be_mounted ? mntpt: ""), 520 used, 521 cur_be->be_policy_type, 522 creation); 523 else 524 (void) printf(" %-*s %-*s %-*s %-*s %-*s " 525 "%-*s\n", 526 hdr->cols[0].width-3, cur_be->be_root_ds, 527 hdr->cols[1].width, active, 528 hdr->cols[2].width, (cur_be->be_mounted ? 529 mntpt: "-"), 530 hdr->cols[3].width, buf, 531 hdr->cols[4].width, cur_be->be_policy_type, 532 hdr->cols[5].width, datetime); 533 534 if (be_fmt & BE_FMT_SNAPSHOT) 535 print_be_snapshots(cur_be, hdr, parsable); 536 } 537 } 538 539 static void 540 print_nodes(const char *be_name, boolean_t dsets, boolean_t snaps, 541 boolean_t parsable, be_node_list_t *be_nodes) 542 { 543 struct hdr_info hdr; 544 enum be_fmt be_fmt = BE_FMT_DEFAULT; 545 546 if (dsets) 547 be_fmt |= BE_FMT_DATASET; 548 if (snaps) 549 be_fmt |= BE_FMT_SNAPSHOT; 550 551 if (!parsable) { 552 init_hdr_cols(be_fmt, &hdr); 553 count_widths(be_fmt, &hdr, be_nodes); 554 print_hdr(&hdr); 555 } 556 557 if (be_fmt == BE_FMT_DEFAULT) 558 print_be_nodes(be_name, parsable, &hdr, be_nodes); 559 else 560 print_fmt_nodes(be_name, be_fmt, parsable, &hdr, be_nodes); 561 } 562 563 static boolean_t 564 confirm_destroy(const char *name) 565 { 566 boolean_t res = B_FALSE; 567 const char *yesre = nl_langinfo(YESEXPR); 568 const char *nore = nl_langinfo(NOEXPR); 569 regex_t yes_re; 570 regex_t no_re; 571 char buf[128]; 572 char *answer; 573 int cflags = REG_EXTENDED; 574 575 if (regcomp(&yes_re, yesre, cflags) != 0) { 576 /* should not happen */ 577 (void) fprintf(stderr, _("Failed to compile 'yes' regexp\n")); 578 return (res); 579 } 580 if (regcomp(&no_re, nore, cflags) != 0) { 581 /* should not happen */ 582 (void) fprintf(stderr, _("Failed to compile 'no' regexp\n")); 583 regfree(&yes_re); 584 return (res); 585 } 586 587 (void) printf(_("Are you sure you want to destroy %s?\n" 588 "This action cannot be undone (y/[n]): "), name); 589 590 answer = fgets(buf, sizeof (buf), stdin); 591 if (answer == NULL || *answer == '\0' || *answer == 10) 592 goto out; 593 594 if (regexec(&yes_re, answer, 0, NULL, 0) == 0) { 595 res = B_TRUE; 596 } else if (regexec(&no_re, answer, 0, NULL, 0) != 0) { 597 (void) fprintf(stderr, _("Invalid response. " 598 "Please enter 'y' or 'n'.\n")); 599 } 600 601 out: 602 regfree(&yes_re); 603 regfree(&no_re); 604 return (res); 605 } 606 607 static int 608 be_nvl_alloc(nvlist_t **nvlp) 609 { 610 assert(nvlp != NULL); 611 612 if (nvlist_alloc(nvlp, NV_UNIQUE_NAME, 0) != 0) { 613 (void) perror(_("nvlist_alloc failed.\n")); 614 return (1); 615 } 616 617 return (0); 618 } 619 620 static int 621 be_nvl_add_string(nvlist_t *nvl, const char *name, const char *val) 622 { 623 assert(nvl != NULL); 624 625 if (nvlist_add_string(nvl, name, val) != 0) { 626 (void) fprintf(stderr, _("nvlist_add_string failed for " 627 "%s (%s).\n"), name, val); 628 return (1); 629 } 630 631 return (0); 632 } 633 634 static int 635 be_nvl_add_nvlist(nvlist_t *nvl, const char *name, nvlist_t *val) 636 { 637 assert(nvl != NULL); 638 639 if (nvlist_add_nvlist(nvl, name, val) != 0) { 640 (void) fprintf(stderr, _("nvlist_add_nvlist failed for %s.\n"), 641 name); 642 return (1); 643 } 644 645 return (0); 646 } 647 648 static int 649 be_nvl_add_uint16(nvlist_t *nvl, const char *name, uint16_t val) 650 { 651 assert(nvl != NULL); 652 653 if (nvlist_add_uint16(nvl, name, val) != 0) { 654 (void) fprintf(stderr, _("nvlist_add_uint16 failed for " 655 "%s (%hu).\n"), name, val); 656 return (1); 657 } 658 659 return (0); 660 } 661 662 static int 663 be_do_activate(int argc, char **argv) 664 { 665 nvlist_t *be_attrs; 666 int err = 1; 667 int c; 668 char *obe_name; 669 670 while ((c = getopt(argc, argv, "v")) != -1) { 671 switch (c) { 672 case 'v': 673 libbe_print_errors(B_TRUE); 674 break; 675 default: 676 usage(); 677 return (1); 678 } 679 } 680 681 argc -= optind; 682 argv += optind; 683 684 if (argc != 1) { 685 usage(); 686 return (1); 687 } 688 689 obe_name = argv[0]; 690 691 if (be_nvl_alloc(&be_attrs) != 0) 692 return (1); 693 694 if (be_nvl_add_string(be_attrs, BE_ATTR_ORIG_BE_NAME, obe_name) != 0) 695 goto out; 696 697 err = be_activate(be_attrs); 698 699 switch (err) { 700 case BE_SUCCESS: 701 (void) printf(_("Activated successfully\n")); 702 break; 703 case BE_ERR_BE_NOENT: 704 (void) fprintf(stderr, _("%s does not exist or appear " 705 "to be a valid BE.\nPlease check that the name of " 706 "the BE provided is correct.\n"), obe_name); 707 break; 708 case BE_ERR_PERM: 709 case BE_ERR_ACCESS: 710 (void) fprintf(stderr, _("Unable to activate %s.\n"), obe_name); 711 (void) fprintf(stderr, _("You have insufficient privileges to " 712 "execute this command.\n")); 713 break; 714 case BE_ERR_ACTIVATE_CURR: 715 default: 716 (void) fprintf(stderr, _("Unable to activate %s.\n"), obe_name); 717 (void) fprintf(stderr, "%s\n", be_err_to_str(err)); 718 } 719 720 out: 721 nvlist_free(be_attrs); 722 return (err); 723 } 724 725 static int 726 be_do_create(int argc, char **argv) 727 { 728 nvlist_t *be_attrs; 729 nvlist_t *zfs_props = NULL; 730 boolean_t activate = B_FALSE; 731 boolean_t is_snap = B_FALSE; 732 int c; 733 int err = 1; 734 char *obe_name = NULL; 735 char *snap_name = NULL; 736 char *nbe_zpool = NULL; 737 char *nbe_name = NULL; 738 char *nbe_desc = NULL; 739 char *propname = NULL; 740 char *propval = NULL; 741 char *strval = NULL; 742 743 while ((c = getopt(argc, argv, "ad:e:io:p:v")) != -1) { 744 switch (c) { 745 case 'a': 746 activate = B_TRUE; 747 break; 748 case 'd': 749 nbe_desc = optarg; 750 break; 751 case 'e': 752 obe_name = optarg; 753 break; 754 case 'o': 755 if (zfs_props == NULL && be_nvl_alloc(&zfs_props) != 0) 756 return (1); 757 758 propname = optarg; 759 if ((propval = strchr(propname, '=')) == NULL) { 760 (void) fprintf(stderr, _("missing " 761 "'=' for -o option\n")); 762 goto out2; 763 } 764 *propval = '\0'; 765 propval++; 766 if (nvlist_lookup_string(zfs_props, propname, 767 &strval) == 0) { 768 (void) fprintf(stderr, _("property '%s' " 769 "specified multiple times\n"), propname); 770 goto out2; 771 772 } 773 if (be_nvl_add_string(zfs_props, propname, propval) 774 != 0) 775 goto out2; 776 777 break; 778 case 'p': 779 nbe_zpool = optarg; 780 break; 781 case 'v': 782 libbe_print_errors(B_TRUE); 783 break; 784 default: 785 usage(); 786 goto out2; 787 } 788 } 789 790 argc -= optind; 791 argv += optind; 792 793 if (argc != 1) { 794 usage(); 795 goto out2; 796 } 797 798 nbe_name = argv[0]; 799 800 if ((snap_name = strrchr(nbe_name, '@')) != NULL) { 801 if (snap_name[1] == '\0') { 802 usage(); 803 goto out2; 804 } 805 806 snap_name[0] = '\0'; 807 snap_name++; 808 is_snap = B_TRUE; 809 } 810 811 if (obe_name) { 812 if (is_snap) { 813 usage(); 814 goto out2; 815 } 816 817 /* 818 * Check if obe_name is really a snapshot name. 819 * If so, split it out. 820 */ 821 if ((snap_name = strrchr(obe_name, '@')) != NULL) { 822 if (snap_name[1] == '\0') { 823 usage(); 824 goto out2; 825 } 826 827 snap_name[0] = '\0'; 828 snap_name++; 829 } 830 } else if (is_snap) { 831 obe_name = nbe_name; 832 nbe_name = NULL; 833 } 834 835 if (be_nvl_alloc(&be_attrs) != 0) 836 goto out2; 837 838 839 if (zfs_props != NULL && be_nvl_add_nvlist(be_attrs, 840 BE_ATTR_ORIG_BE_NAME, zfs_props) != 0) 841 goto out; 842 843 if (obe_name != NULL && be_nvl_add_string(be_attrs, 844 BE_ATTR_ORIG_BE_NAME, obe_name) != 0) 845 goto out; 846 847 if (snap_name != NULL && be_nvl_add_string(be_attrs, 848 BE_ATTR_SNAP_NAME, snap_name) != 0) 849 goto out; 850 851 if (nbe_zpool != NULL && be_nvl_add_string(be_attrs, 852 BE_ATTR_NEW_BE_POOL, nbe_zpool) != 0) 853 goto out; 854 855 if (nbe_name != NULL && be_nvl_add_string(be_attrs, 856 BE_ATTR_NEW_BE_NAME, nbe_name) != 0) 857 goto out; 858 859 if (nbe_desc != NULL && be_nvl_add_string(be_attrs, 860 BE_ATTR_NEW_BE_DESC, nbe_desc) != 0) 861 goto out; 862 863 if (is_snap) 864 err = be_create_snapshot(be_attrs); 865 else 866 err = be_copy(be_attrs); 867 868 switch (err) { 869 case BE_SUCCESS: 870 if (!is_snap && !nbe_name) { 871 /* 872 * We requested an auto named BE; find out the 873 * name of the BE that was created for us and 874 * the auto snapshot created from the original BE. 875 */ 876 if (nvlist_lookup_string(be_attrs, BE_ATTR_NEW_BE_NAME, 877 &nbe_name) != 0) { 878 (void) fprintf(stderr, _("failed to get %s " 879 "attribute\n"), BE_ATTR_NEW_BE_NAME); 880 break; 881 } else 882 (void) printf(_("Auto named BE: %s\n"), 883 nbe_name); 884 885 if (nvlist_lookup_string(be_attrs, BE_ATTR_SNAP_NAME, 886 &snap_name) != 0) { 887 (void) fprintf(stderr, _("failed to get %s " 888 "attribute\n"), BE_ATTR_SNAP_NAME); 889 break; 890 } else 891 (void) printf(_("Auto named snapshot: %s\n"), 892 snap_name); 893 } 894 895 if (!is_snap && activate) { 896 char *args[] = { "activate", "", NULL }; 897 args[1] = nbe_name; 898 optind = 1; 899 900 err = be_do_activate(2, args); 901 goto out; 902 } 903 904 (void) printf(_("Created successfully\n")); 905 break; 906 case BE_ERR_BE_EXISTS: 907 (void) fprintf(stderr, _("BE %s already exists\n." 908 "Please choose a different BE name.\n"), nbe_name); 909 break; 910 case BE_ERR_SS_EXISTS: 911 (void) fprintf(stderr, _("BE %s snapshot %s already exists.\n" 912 "Please choose a different snapshot name.\n"), obe_name, 913 snap_name); 914 break; 915 case BE_ERR_PERM: 916 case BE_ERR_ACCESS: 917 if (is_snap) 918 (void) fprintf(stderr, _("Unable to create snapshot " 919 "%s.\n"), snap_name); 920 else 921 (void) fprintf(stderr, _("Unable to create %s.\n"), 922 nbe_name); 923 (void) fprintf(stderr, _("You have insufficient privileges to " 924 "execute this command.\n")); 925 break; 926 default: 927 if (is_snap) 928 (void) fprintf(stderr, _("Unable to create snapshot " 929 "%s.\n"), snap_name); 930 else 931 (void) fprintf(stderr, _("Unable to create %s.\n"), 932 nbe_name); 933 (void) fprintf(stderr, "%s\n", be_err_to_str(err)); 934 } 935 936 out: 937 nvlist_free(be_attrs); 938 out2: 939 if (zfs_props != NULL) 940 nvlist_free(zfs_props); 941 942 return (err); 943 } 944 945 static int 946 be_do_destroy(int argc, char **argv) 947 { 948 nvlist_t *be_attrs; 949 boolean_t is_snap = B_FALSE; 950 boolean_t suppress_prompt = B_FALSE; 951 int err = 1; 952 int c; 953 int destroy_flags = 0; 954 char *snap_name; 955 char *be_name; 956 957 while ((c = getopt(argc, argv, "fFsv")) != -1) { 958 switch (c) { 959 case 'f': 960 destroy_flags |= BE_DESTROY_FLAG_FORCE_UNMOUNT; 961 break; 962 case 's': 963 destroy_flags |= BE_DESTROY_FLAG_SNAPSHOTS; 964 break; 965 case 'v': 966 libbe_print_errors(B_TRUE); 967 break; 968 case 'F': 969 suppress_prompt = B_TRUE; 970 break; 971 default: 972 usage(); 973 return (1); 974 } 975 } 976 977 argc -= optind; 978 argv += optind; 979 980 if (argc != 1) { 981 usage(); 982 return (1); 983 } 984 985 be_name = argv[0]; 986 if (!suppress_prompt && !confirm_destroy(be_name)) { 987 (void) printf(_("%s has not been destroyed.\n"), be_name); 988 return (0); 989 } 990 991 if ((snap_name = strrchr(be_name, '@')) != NULL) { 992 if (snap_name[1] == '\0') { 993 usage(); 994 return (1); 995 } 996 997 is_snap = B_TRUE; 998 *snap_name = '\0'; 999 snap_name++; 1000 } 1001 1002 if (be_nvl_alloc(&be_attrs) != 0) 1003 return (1); 1004 1005 1006 if (be_nvl_add_string(be_attrs, BE_ATTR_ORIG_BE_NAME, be_name) != 0) 1007 goto out; 1008 1009 if (is_snap) { 1010 if (be_nvl_add_string(be_attrs, BE_ATTR_SNAP_NAME, 1011 snap_name) != 0) 1012 goto out; 1013 1014 err = be_destroy_snapshot(be_attrs); 1015 } else { 1016 if (be_nvl_add_uint16(be_attrs, BE_ATTR_DESTROY_FLAGS, 1017 destroy_flags) != 0) 1018 goto out; 1019 1020 err = be_destroy(be_attrs); 1021 } 1022 1023 switch (err) { 1024 case BE_SUCCESS: 1025 (void) printf(_("Destroyed successfully\n")); 1026 break; 1027 case BE_ERR_MOUNTED: 1028 (void) fprintf(stderr, _("Unable to destroy %s.\n"), be_name); 1029 (void) fprintf(stderr, _("It is currently mounted and must be " 1030 "unmounted before it can be destroyed.\n" "Use 'beadm " 1031 "unmount %s' to unmount the BE before destroying\nit or " 1032 "'beadm destroy -f %s'.\n"), be_name, be_name); 1033 break; 1034 case BE_ERR_DESTROY_CURR_BE: 1035 (void) fprintf(stderr, _("%s is the currently active BE and " 1036 "cannot be destroyed.\nYou must boot from another BE in " 1037 "order to destroy %s.\n"), be_name, be_name); 1038 break; 1039 case BE_ERR_ZONES_UNMOUNT: 1040 (void) fprintf(stderr, _("Unable to destroy one of " "%s's " 1041 "zone BE's.\nUse 'beadm destroy -f %s' or " 1042 "'zfs -f destroy <dataset>'.\n"), be_name, be_name); 1043 break; 1044 case BE_ERR_SS_NOENT: 1045 (void) fprintf(stderr, _("%s does not exist or appear " 1046 "to be a valid snapshot.\nPlease check that the name of " 1047 "the snapshot provided is correct.\n"), snap_name); 1048 break; 1049 case BE_ERR_PERM: 1050 case BE_ERR_ACCESS: 1051 (void) fprintf(stderr, _("Unable to destroy %s.\n"), be_name); 1052 (void) fprintf(stderr, _("You have insufficient privileges to " 1053 "execute this command.\n")); 1054 break; 1055 case BE_ERR_SS_EXISTS: 1056 (void) fprintf(stderr, _("Unable to destroy %s: " 1057 "BE has snapshots.\nUse 'beadm destroy -s %s' or " 1058 "'zfs -r destroy <dataset>'.\n"), be_name, be_name); 1059 break; 1060 default: 1061 (void) fprintf(stderr, _("Unable to destroy %s.\n"), be_name); 1062 (void) fprintf(stderr, "%s\n", be_err_to_str(err)); 1063 } 1064 1065 out: 1066 nvlist_free(be_attrs); 1067 return (err); 1068 } 1069 1070 static int 1071 be_do_list(int argc, char **argv) 1072 { 1073 be_node_list_t *be_nodes = NULL; 1074 boolean_t all = B_FALSE; 1075 boolean_t dsets = B_FALSE; 1076 boolean_t snaps = B_FALSE; 1077 boolean_t parsable = B_FALSE; 1078 int err = 1; 1079 int c = 0; 1080 char *be_name = NULL; 1081 1082 while ((c = getopt(argc, argv, "adsvH")) != -1) { 1083 switch (c) { 1084 case 'a': 1085 all = B_TRUE; 1086 break; 1087 case 'd': 1088 dsets = B_TRUE; 1089 break; 1090 case 's': 1091 snaps = B_TRUE; 1092 break; 1093 case 'v': 1094 libbe_print_errors(B_TRUE); 1095 break; 1096 case 'H': 1097 parsable = B_TRUE; 1098 break; 1099 default: 1100 usage(); 1101 return (1); 1102 } 1103 } 1104 1105 if (all) { 1106 if (dsets) { 1107 (void) fprintf(stderr, _("Invalid options: -a and %s " 1108 "are mutually exclusive.\n"), "-d"); 1109 usage(); 1110 return (1); 1111 } 1112 if (snaps) { 1113 (void) fprintf(stderr, _("Invalid options: -a and %s " 1114 "are mutually exclusive.\n"), "-s"); 1115 usage(); 1116 return (1); 1117 } 1118 1119 dsets = B_TRUE; 1120 snaps = B_TRUE; 1121 } 1122 1123 argc -= optind; 1124 argv += optind; 1125 1126 1127 if (argc == 1) 1128 be_name = argv[0]; 1129 1130 err = be_list(be_name, &be_nodes); 1131 1132 switch (err) { 1133 case BE_SUCCESS: 1134 print_nodes(be_name, dsets, snaps, parsable, be_nodes); 1135 break; 1136 case BE_ERR_BE_NOENT: 1137 if (be_name == NULL) 1138 (void) fprintf(stderr, _("No boot environments found " 1139 "on this system.\n")); 1140 else { 1141 (void) fprintf(stderr, _("%s does not exist or appear " 1142 "to be a valid BE.\nPlease check that the name of " 1143 "the BE provided is correct.\n"), be_name); 1144 } 1145 break; 1146 default: 1147 (void) fprintf(stderr, _("Unable to display Boot " 1148 "Environment\n")); 1149 (void) fprintf(stderr, "%s\n", be_err_to_str(err)); 1150 } 1151 1152 if (be_nodes != NULL) 1153 be_free_list(be_nodes); 1154 return (err); 1155 } 1156 1157 static int 1158 be_do_mount(int argc, char **argv) 1159 { 1160 nvlist_t *be_attrs; 1161 boolean_t shared_fs = B_FALSE; 1162 int err = 1; 1163 int c; 1164 int mount_flags = 0; 1165 char *obe_name; 1166 char *mountpoint; 1167 char *tmp_mp = NULL; 1168 1169 while ((c = getopt(argc, argv, "s:v")) != -1) { 1170 switch (c) { 1171 case 's': 1172 shared_fs = B_TRUE; 1173 1174 mount_flags |= BE_MOUNT_FLAG_SHARED_FS; 1175 1176 if (strcmp(optarg, "rw") == 0) { 1177 mount_flags |= BE_MOUNT_FLAG_SHARED_RW; 1178 } else if (strcmp(optarg, "ro") != 0) { 1179 (void) fprintf(stderr, _("The -s flag " 1180 "requires an argument [ rw | ro ]\n")); 1181 usage(); 1182 return (1); 1183 } 1184 1185 break; 1186 case 'v': 1187 libbe_print_errors(B_TRUE); 1188 break; 1189 default: 1190 usage(); 1191 return (1); 1192 } 1193 } 1194 1195 argc -= optind; 1196 argv += optind; 1197 1198 if (argc < 1 || argc > 2) { 1199 usage(); 1200 return (1); 1201 } 1202 1203 obe_name = argv[0]; 1204 1205 if (argc == 2) { 1206 mountpoint = argv[1]; 1207 if (mountpoint[0] != '/') { 1208 (void) fprintf(stderr, _("Invalid mount point %s. " 1209 "Mount point must start with a /.\n"), mountpoint); 1210 return (1); 1211 } 1212 } else { 1213 const char *tmpdir = getenv("TMPDIR"); 1214 const char *tmpname = "tmp.XXXXXX"; 1215 int sz; 1216 1217 if (tmpdir == NULL) 1218 tmpdir = "/tmp"; 1219 1220 sz = asprintf(&tmp_mp, "%s/%s", tmpdir, tmpname); 1221 if (sz < 0) { 1222 (void) fprintf(stderr, _("internal error: " 1223 "out of memory\n")); 1224 return (1); 1225 } 1226 1227 mountpoint = mkdtemp(tmp_mp); 1228 } 1229 1230 if (be_nvl_alloc(&be_attrs) != 0) 1231 return (1); 1232 1233 if (be_nvl_add_string(be_attrs, BE_ATTR_ORIG_BE_NAME, obe_name) != 0) 1234 goto out; 1235 1236 if (be_nvl_add_string(be_attrs, BE_ATTR_MOUNTPOINT, mountpoint) != 0) 1237 goto out; 1238 1239 if (shared_fs && be_nvl_add_uint16(be_attrs, BE_ATTR_MOUNT_FLAGS, 1240 mount_flags) != 0) 1241 goto out; 1242 1243 err = be_mount(be_attrs); 1244 1245 switch (err) { 1246 case BE_SUCCESS: 1247 (void) printf(_("Mounted successfully on: '%s'\n"), mountpoint); 1248 break; 1249 case BE_ERR_BE_NOENT: 1250 (void) fprintf(stderr, _("%s does not exist or appear " 1251 "to be a valid BE.\nPlease check that the name of " 1252 "the BE provided is correct.\n"), obe_name); 1253 break; 1254 case BE_ERR_MOUNTED: 1255 (void) fprintf(stderr, _("%s is already mounted.\n" 1256 "Please unmount the BE before mounting it again.\n"), 1257 obe_name); 1258 break; 1259 case BE_ERR_PERM: 1260 case BE_ERR_ACCESS: 1261 (void) fprintf(stderr, _("Unable to mount %s.\n"), obe_name); 1262 (void) fprintf(stderr, _("You have insufficient privileges to " 1263 "execute this command.\n")); 1264 break; 1265 default: 1266 (void) fprintf(stderr, _("Unable to mount %s.\n"), obe_name); 1267 (void) fprintf(stderr, "%s\n", be_err_to_str(err)); 1268 } 1269 1270 out: 1271 if (tmp_mp != NULL) 1272 free(tmp_mp); 1273 nvlist_free(be_attrs); 1274 return (err); 1275 } 1276 1277 static int 1278 be_do_unmount(int argc, char **argv) 1279 { 1280 nvlist_t *be_attrs; 1281 char *obe_name; 1282 int err = 1; 1283 int c; 1284 int unmount_flags = 0; 1285 1286 while ((c = getopt(argc, argv, "fv")) != -1) { 1287 switch (c) { 1288 case 'f': 1289 unmount_flags |= BE_UNMOUNT_FLAG_FORCE; 1290 break; 1291 case 'v': 1292 libbe_print_errors(B_TRUE); 1293 break; 1294 default: 1295 usage(); 1296 return (1); 1297 } 1298 } 1299 1300 argc -= optind; 1301 argv += optind; 1302 1303 if (argc != 1) { 1304 usage(); 1305 return (1); 1306 } 1307 1308 obe_name = argv[0]; 1309 1310 if (be_nvl_alloc(&be_attrs) != 0) 1311 return (1); 1312 1313 1314 if (be_nvl_add_string(be_attrs, BE_ATTR_ORIG_BE_NAME, obe_name) != 0) 1315 goto out; 1316 1317 if (be_nvl_add_uint16(be_attrs, BE_ATTR_UNMOUNT_FLAGS, 1318 unmount_flags) != 0) 1319 goto out; 1320 1321 err = be_unmount(be_attrs); 1322 1323 switch (err) { 1324 case BE_SUCCESS: 1325 (void) printf(_("Unmounted successfully\n")); 1326 break; 1327 case BE_ERR_BE_NOENT: 1328 (void) fprintf(stderr, _("%s does not exist or appear " 1329 "to be a valid BE.\nPlease check that the name of " 1330 "the BE provided is correct.\n"), obe_name); 1331 break; 1332 case BE_ERR_UMOUNT_CURR_BE: 1333 (void) fprintf(stderr, _("%s is the currently active BE.\n" 1334 "It cannot be unmounted unless another BE is the " 1335 "currently active BE.\n"), obe_name); 1336 break; 1337 case BE_ERR_UMOUNT_SHARED: 1338 (void) fprintf(stderr, _("%s is a shared file system and it " 1339 "cannot be unmounted.\n"), obe_name); 1340 break; 1341 case BE_ERR_PERM: 1342 case BE_ERR_ACCESS: 1343 (void) fprintf(stderr, _("Unable to unmount %s.\n"), obe_name); 1344 (void) fprintf(stderr, _("You have insufficient privileges to " 1345 "execute this command.\n")); 1346 break; 1347 default: 1348 (void) fprintf(stderr, _("Unable to unmount %s.\n"), obe_name); 1349 (void) fprintf(stderr, "%s\n", be_err_to_str(err)); 1350 } 1351 1352 out: 1353 nvlist_free(be_attrs); 1354 return (err); 1355 } 1356 1357 static int 1358 be_do_rename(int argc, char **argv) 1359 { 1360 nvlist_t *be_attrs; 1361 char *obe_name; 1362 char *nbe_name; 1363 int err = 1; 1364 int c; 1365 1366 while ((c = getopt(argc, argv, "v")) != -1) { 1367 switch (c) { 1368 case 'v': 1369 libbe_print_errors(B_TRUE); 1370 break; 1371 default: 1372 usage(); 1373 return (1); 1374 } 1375 } 1376 1377 argc -= optind; 1378 argv += optind; 1379 1380 if (argc != 2) { 1381 usage(); 1382 return (1); 1383 } 1384 1385 obe_name = argv[0]; 1386 nbe_name = argv[1]; 1387 1388 if (be_nvl_alloc(&be_attrs) != 0) 1389 return (1); 1390 1391 if (be_nvl_add_string(be_attrs, BE_ATTR_ORIG_BE_NAME, obe_name) != 0) 1392 goto out; 1393 1394 if (be_nvl_add_string(be_attrs, BE_ATTR_NEW_BE_NAME, nbe_name) != 0) 1395 goto out; 1396 1397 err = be_rename(be_attrs); 1398 1399 switch (err) { 1400 case BE_SUCCESS: 1401 (void) printf(_("Renamed successfully\n")); 1402 break; 1403 case BE_ERR_BE_NOENT: 1404 (void) fprintf(stderr, _("%s does not exist or appear " 1405 "to be a valid BE.\nPlease check that the name of " 1406 "the BE provided is correct.\n"), obe_name); 1407 break; 1408 case BE_ERR_PERM: 1409 case BE_ERR_ACCESS: 1410 (void) fprintf(stderr, _("Rename of BE %s failed.\n"), 1411 obe_name); 1412 (void) fprintf(stderr, _("You have insufficient privileges to " 1413 "execute this command.\n")); 1414 break; 1415 default: 1416 (void) fprintf(stderr, _("Rename of BE %s failed.\n"), 1417 obe_name); 1418 (void) fprintf(stderr, "%s\n", be_err_to_str(err)); 1419 } 1420 1421 out: 1422 nvlist_free(be_attrs); 1423 return (err); 1424 } 1425 1426 static int 1427 be_do_rollback(int argc, char **argv) 1428 { 1429 nvlist_t *be_attrs; 1430 char *obe_name; 1431 char *snap_name; 1432 int err = 1; 1433 int c; 1434 1435 while ((c = getopt(argc, argv, "v")) != -1) { 1436 switch (c) { 1437 case 'v': 1438 libbe_print_errors(B_TRUE); 1439 break; 1440 default: 1441 usage(); 1442 return (1); 1443 } 1444 } 1445 1446 argc -= optind; 1447 argv += optind; 1448 1449 if (argc < 1 || argc > 2) { 1450 usage(); 1451 return (1); 1452 } 1453 1454 obe_name = argv[0]; 1455 if (argc == 2) 1456 snap_name = argv[1]; 1457 else { /* argc == 1 */ 1458 if ((snap_name = strrchr(obe_name, '@')) != NULL) { 1459 if (snap_name[1] == '\0') { 1460 usage(); 1461 return (1); 1462 } 1463 1464 snap_name[0] = '\0'; 1465 snap_name++; 1466 } else { 1467 usage(); 1468 return (1); 1469 } 1470 } 1471 1472 if (be_nvl_alloc(&be_attrs) != 0) 1473 return (1); 1474 1475 if (be_nvl_add_string(be_attrs, BE_ATTR_ORIG_BE_NAME, obe_name) != 0) 1476 goto out; 1477 1478 if (be_nvl_add_string(be_attrs, BE_ATTR_SNAP_NAME, snap_name) != 0) 1479 goto out; 1480 1481 err = be_rollback(be_attrs); 1482 1483 switch (err) { 1484 case BE_SUCCESS: 1485 (void) printf(_("Rolled back successfully\n")); 1486 break; 1487 case BE_ERR_BE_NOENT: 1488 (void) fprintf(stderr, _("%s does not exist or appear " 1489 "to be a valid BE.\nPlease check that the name of " 1490 "the BE provided is correct.\n"), obe_name); 1491 break; 1492 case BE_ERR_SS_NOENT: 1493 (void) fprintf(stderr, _("%s does not exist or appear " 1494 "to be a valid snapshot.\nPlease check that the name of " 1495 "the snapshot provided is correct.\n"), snap_name); 1496 break; 1497 case BE_ERR_PERM: 1498 case BE_ERR_ACCESS: 1499 (void) fprintf(stderr, _("Rollback of BE %s snapshot %s " 1500 "failed.\n"), obe_name, snap_name); 1501 (void) fprintf(stderr, _("You have insufficient privileges to " 1502 "execute this command.\n")); 1503 break; 1504 default: 1505 (void) fprintf(stderr, _("Rollback of BE %s snapshot %s " 1506 "failed.\n"), obe_name, snap_name); 1507 (void) fprintf(stderr, "%s\n", be_err_to_str(err)); 1508 } 1509 1510 out: 1511 nvlist_free(be_attrs); 1512 return (err); 1513 }