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