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