xref: /titanic_50/usr/src/lib/libbe/common/be_utils.c (revision 53089ab7c84db6fb76c16ca50076c147cda11757)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /*
23  * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
24  */
25 
26 /*
27  * Copyright 2011 Nexenta Systems, Inc. All rights reserved.
28  */
29 
30 
31 /*
32  * System includes
33  */
34 #include <assert.h>
35 #include <errno.h>
36 #include <libgen.h>
37 #include <libintl.h>
38 #include <libnvpair.h>
39 #include <libzfs.h>
40 #include <libgen.h>
41 #include <stdio.h>
42 #include <stdlib.h>
43 #include <string.h>
44 #include <sys/stat.h>
45 #include <sys/types.h>
46 #include <sys/vfstab.h>
47 #include <sys/param.h>
48 #include <sys/systeminfo.h>
49 #include <ctype.h>
50 #include <time.h>
51 #include <unistd.h>
52 #include <fcntl.h>
53 #include <deflt.h>
54 #include <wait.h>
55 #include <libdevinfo.h>
56 
57 #include <libbe.h>
58 #include <libbe_priv.h>
59 
60 /* Private function prototypes */
61 static int update_dataset(char *, int, char *, char *, char *);
62 static int _update_vfstab(char *, char *, char *, char *, be_fs_list_data_t *);
63 static int be_open_menu(char *, char *, FILE **, char *, boolean_t);
64 static int be_create_menu(char *, char *, FILE **, char *);
65 static char *be_get_auto_name(char *, char *, boolean_t);
66 
67 /*
68  * Global error printing
69  */
70 boolean_t do_print = B_FALSE;
71 
72 /*
73  * Private datatypes
74  */
75 typedef struct zone_be_name_cb_data {
76 	char *base_be_name;
77 	int num;
78 } zone_be_name_cb_data_t;
79 
80 /* ********************************************************************	*/
81 /*			Public Functions				*/
82 /* ******************************************************************** */
83 
84 /*
85  * Function:	be_max_avail
86  * Description:	Returns the available size for the zfs dataset passed in.
87  * Parameters:
88  *		dataset - The dataset we want to get the available space for.
89  *		ret - The available size will be returned in this.
90  * Returns:
91  *		The error returned by the zfs get property function.
92  * Scope:
93  *		Public
94  */
95 int
96 be_max_avail(char *dataset, uint64_t *ret)
97 {
98 	zfs_handle_t *zhp;
99 	int err = 0;
100 
101 	/* Initialize libzfs handle */
102 	if (!be_zfs_init())
103 		return (BE_ERR_INIT);
104 
105 	zhp = zfs_open(g_zfs, dataset, ZFS_TYPE_DATASET);
106 	if (zhp == NULL) {
107 		/*
108 		 * The zfs_open failed return an error
109 		 */
110 		err = zfs_err_to_be_err(g_zfs);
111 	} else {
112 		err = be_maxsize_avail(zhp, ret);
113 	}
114 	ZFS_CLOSE(zhp);
115 	be_zfs_fini();
116 	return (err);
117 }
118 
119 /*
120  * Function:	libbe_print_errors
121  * Description:	Turns on/off error output for the library.
122  * Parameter:
123  *		set_do_print - Boolean that turns library error
124  *			       printing on or off.
125  * Returns:
126  *		None
127  * Scope:
128  *		Public;
129  */
130 void
131 libbe_print_errors(boolean_t set_do_print)
132 {
133 	do_print = set_do_print;
134 }
135 
136 /* ********************************************************************	*/
137 /*			Semi-Private Functions				*/
138 /* ******************************************************************** */
139 
140 /*
141  * Function:	be_zfs_init
142  * Description:	Initializes the libary global libzfs handle.
143  * Parameters:
144  *		None
145  * Returns:
146  *		B_TRUE - Success
147  *		B_FALSE - Failure
148  * Scope:
149  *		Semi-private (library wide use only)
150  */
151 boolean_t
152 be_zfs_init(void)
153 {
154 	be_zfs_fini();
155 
156 	if ((g_zfs = libzfs_init()) == NULL) {
157 		be_print_err(gettext("be_zfs_init: failed to initialize ZFS "
158 		    "library\n"));
159 		return (B_FALSE);
160 	}
161 
162 	return (B_TRUE);
163 }
164 
165 /*
166  * Function:	be_zfs_fini
167  * Description:	Closes the library global libzfs handle if it currently open.
168  * Parameter:
169  *		None
170  * Returns:
171  *		None
172  * Scope:
173  *		Semi-private (library wide use only)
174  */
175 void
176 be_zfs_fini(void)
177 {
178 	if (g_zfs)
179 		libzfs_fini(g_zfs);
180 
181 	g_zfs = NULL;
182 }
183 
184 /*
185  * Function:	be_get_defaults
186  * Description:	Open defaults and gets be default paramets
187  * Parameters:
188  *		defaults - be defaults struct
189  * Returns:
190  *		None
191  * Scope:
192  *		Semi-private (library wide use only)
193  */
194 void
195 be_get_defaults(struct be_defaults *defaults)
196 {
197 	void	*defp;
198 
199 	defaults->be_deflt_rpool_container = B_FALSE;
200 	defaults->be_deflt_bename_starts_with[0] = '\0';
201 
202 	if ((defp = defopen_r(BE_DEFAULTS)) != NULL) {
203 		const char *res = defread_r(BE_DFLT_BENAME_STARTS, defp);
204 		if (res != NULL && res[0] != NULL) {
205 			(void) strlcpy(defaults->be_deflt_bename_starts_with,
206 			    res, ZFS_MAXNAMELEN);
207 			defaults->be_deflt_rpool_container = B_TRUE;
208 		}
209 		defclose_r(defp);
210 	}
211 }
212 
213 /*
214  * Function:	be_make_root_ds
215  * Description:	Generate string for BE's root dataset given the pool
216  *		it lives in and the BE name.
217  * Parameters:
218  *		zpool - pointer zpool name.
219  *		be_name - pointer to BE name.
220  *		be_root_ds - pointer to buffer to return BE root dataset in.
221  *		be_root_ds_size - size of be_root_ds
222  * Returns:
223  *		None
224  * Scope:
225  *		Semi-private (library wide use only)
226  */
227 void
228 be_make_root_ds(const char *zpool, const char *be_name, char *be_root_ds,
229     int be_root_ds_size)
230 {
231 	struct be_defaults be_defaults;
232 	be_get_defaults(&be_defaults);
233 
234 	if (be_defaults.be_deflt_rpool_container)
235 		(void) snprintf(be_root_ds, be_root_ds_size, "%s/%s", zpool,
236 		    be_name);
237 	else
238 		(void) snprintf(be_root_ds, be_root_ds_size, "%s/%s/%s", zpool,
239 		    BE_CONTAINER_DS_NAME, be_name);
240 }
241 
242 /*
243  * Function:	be_make_container_ds
244  * Description:	Generate string for the BE container dataset given a pool name.
245  * Parameters:
246  *		zpool - pointer zpool name.
247  *		container_ds - pointer to buffer to return BE container
248  *			dataset in.
249  *		container_ds_size - size of container_ds
250  * Returns:
251  *		None
252  * Scope:
253  *		Semi-private (library wide use only)
254  */
255 void
256 be_make_container_ds(const char *zpool,  char *container_ds,
257     int container_ds_size)
258 {
259 	struct be_defaults be_defaults;
260 	be_get_defaults(&be_defaults);
261 
262 	if (be_defaults.be_deflt_rpool_container)
263 		(void) snprintf(container_ds, container_ds_size, "%s", zpool);
264 	else
265 		(void) snprintf(container_ds, container_ds_size, "%s/%s", zpool,
266 		    BE_CONTAINER_DS_NAME);
267 }
268 
269 /*
270  * Function:	be_make_name_from_ds
271  * Description:	This function takes a dataset name and strips off the
272  *		BE container dataset portion from the beginning.  The
273  *		returned name is allocated in heap storage, so the caller
274  *		is responsible for freeing it.
275  * Parameters:
276  *		dataset - dataset to get name from.
277  *		rc_loc - dataset underwhich the root container dataset lives.
278  * Returns:
279  *		name of dataset relative to BE container dataset.
280  *		NULL if dataset is not under a BE root dataset.
281  * Scope:
282  *		Semi-primate (library wide use only)
283  */
284 char *
285 be_make_name_from_ds(const char *dataset, char *rc_loc)
286 {
287 	char	ds[ZFS_MAXNAMELEN];
288 	char	*tok = NULL;
289 	char	*name = NULL;
290 	struct be_defaults be_defaults;
291 	int	rlen = strlen(rc_loc);
292 
293 	be_get_defaults(&be_defaults);
294 
295 	/*
296 	 * First token is the location of where the root container dataset
297 	 * lives; it must match rc_loc.
298 	 */
299 	if (strncmp(dataset, rc_loc, rlen) == 0 && dataset[rlen] == '/')
300 		(void) strlcpy(ds, dataset + rlen + 1, sizeof (ds));
301 	else
302 		return (NULL);
303 
304 	if (be_defaults.be_deflt_rpool_container) {
305 		if ((name = strdup(ds)) == NULL) {
306 			be_print_err(gettext("be_make_name_from_ds: "
307 			    "memory allocation failed\n"));
308 			return (NULL);
309 		}
310 	} else {
311 		/* Second token must be BE container dataset name */
312 		if ((tok = strtok(ds, "/")) == NULL ||
313 		    strcmp(tok, BE_CONTAINER_DS_NAME) != 0)
314 			return (NULL);
315 
316 		/* Return the remaining token if one exists */
317 		if ((tok = strtok(NULL, "")) == NULL)
318 			return (NULL);
319 
320 		if ((name = strdup(tok)) == NULL) {
321 			be_print_err(gettext("be_make_name_from_ds: "
322 			    "memory allocation failed\n"));
323 			return (NULL);
324 		}
325 	}
326 
327 	return (name);
328 }
329 
330 /*
331  * Function:	be_maxsize_avail
332  * Description:	Returns the available size for the zfs handle passed in.
333  * Parameters:
334  *		zhp - A pointer to the open zfs handle.
335  *		ret - The available size will be returned in this.
336  * Returns:
337  *		The error returned by the zfs get property function.
338  * Scope:
339  *		Semi-private (library wide use only)
340  */
341 int
342 be_maxsize_avail(zfs_handle_t *zhp, uint64_t *ret)
343 {
344 	return ((*ret = zfs_prop_get_int(zhp, ZFS_PROP_AVAILABLE)));
345 }
346 
347 /*
348  * Function:	be_append_menu
349  * Description:	Appends an entry for a BE into the menu.lst.
350  * Parameters:
351  *		be_name - pointer to name of BE to add boot menu entry for.
352  *		be_root_pool - pointer to name of pool BE lives in.
353  *		boot_pool - Used if the pool containing the grub menu is
354  *			    different than the one contaiing the BE. This
355  *			    will normally be NULL.
356  *		be_orig_root_ds - The root dataset for the BE. This is
357  *			used to check to see if an entry already exists
358  *			for this BE.
359  *		description - pointer to description of BE to be added in
360  *			the title line for this BEs entry.
361  * Returns:
362  *		BE_SUCCESS - Success
363  *		be_errno_t - Failure
364  * Scope:
365  *		Semi-private (library wide use only)
366  */
367 int
368 be_append_menu(char *be_name, char *be_root_pool, char *boot_pool,
369     char *be_orig_root_ds, char *description)
370 {
371 	zfs_handle_t *zhp = NULL;
372 	char menu_file[MAXPATHLEN];
373 	char be_root_ds[MAXPATHLEN];
374 	char line[BUFSIZ];
375 	char temp_line[BUFSIZ];
376 	char title[MAXPATHLEN];
377 	char *entries[BUFSIZ];
378 	char *tmp_entries[BUFSIZ];
379 	char *pool_mntpnt = NULL;
380 	char *ptmp_mntpnt = NULL;
381 	char *orig_mntpnt = NULL;
382 	boolean_t found_be = B_FALSE;
383 	boolean_t found_orig_be = B_FALSE;
384 	boolean_t found_title = B_FALSE;
385 	boolean_t pool_mounted = B_FALSE;
386 	boolean_t collect_lines = B_FALSE;
387 	FILE *menu_fp = NULL;
388 	int err = 0, ret = BE_SUCCESS;
389 	int i, num_tmp_lines = 0, num_lines = 0;
390 
391 	if (be_name == NULL || be_root_pool == NULL)
392 		return (BE_ERR_INVAL);
393 
394 	if (boot_pool == NULL)
395 		boot_pool = be_root_pool;
396 
397 	if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) {
398 		be_print_err(gettext("be_append_menu: failed to open "
399 		    "pool dataset for %s: %s\n"), be_root_pool,
400 		    libzfs_error_description(g_zfs));
401 		return (zfs_err_to_be_err(g_zfs));
402 	}
403 
404 	/*
405 	 * Check to see if the pool's dataset is mounted. If it isn't we'll
406 	 * attempt to mount it.
407 	 */
408 	if ((ret = be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt,
409 	    &pool_mounted)) != BE_SUCCESS) {
410 		be_print_err(gettext("be_append_menu: pool dataset "
411 		    "(%s) could not be mounted\n"), be_root_pool);
412 		ZFS_CLOSE(zhp);
413 		return (ret);
414 	}
415 
416 	/*
417 	 * Get the mountpoint for the root pool dataset.
418 	 */
419 	if (!zfs_is_mounted(zhp, &pool_mntpnt)) {
420 		be_print_err(gettext("be_append_menu: pool "
421 		    "dataset (%s) is not mounted. Can't set "
422 		    "the default BE in the grub menu.\n"), be_root_pool);
423 		ret = BE_ERR_NO_MENU;
424 		goto cleanup;
425 	}
426 
427 	/*
428 	 * Check to see if this system supports grub
429 	 */
430 	if (be_has_grub()) {
431 		(void) snprintf(menu_file, sizeof (menu_file),
432 		    "%s%s", pool_mntpnt, BE_GRUB_MENU);
433 	} else {
434 		(void) snprintf(menu_file, sizeof (menu_file),
435 		    "%s%s", pool_mntpnt, BE_SPARC_MENU);
436 	}
437 
438 	be_make_root_ds(be_root_pool, be_name, be_root_ds, sizeof (be_root_ds));
439 
440 	/*
441 	 * Iterate through menu first to make sure the BE doesn't already
442 	 * have an entry in the menu.
443 	 *
444 	 * Additionally while iterating through the menu, if we have an
445 	 * original root dataset for a BE we're cloning from, we need to keep
446 	 * track of that BE's menu entry. We will then use the lines from
447 	 * that entry to create the entry for the new BE.
448 	 */
449 	if ((ret = be_open_menu(be_root_pool, menu_file,
450 	    &menu_fp, "r", B_TRUE)) != BE_SUCCESS) {
451 		goto cleanup;
452 	} else if (menu_fp == NULL) {
453 		ret = BE_ERR_NO_MENU;
454 		goto cleanup;
455 	}
456 
457 	free(pool_mntpnt);
458 	pool_mntpnt = NULL;
459 
460 	while (fgets(line, BUFSIZ, menu_fp)) {
461 		char *tok = NULL;
462 
463 		(void) strlcpy(temp_line, line, BUFSIZ);
464 		tok = strtok(line, BE_WHITE_SPACE);
465 
466 		if (tok == NULL || tok[0] == '#') {
467 			continue;
468 		} else if (strcmp(tok, "title") == 0) {
469 			collect_lines = B_FALSE;
470 			if ((tok = strtok(NULL, "\n")) == NULL)
471 				(void) strlcpy(title, "", sizeof (title));
472 			else
473 				(void) strlcpy(title, tok, sizeof (title));
474 			found_title = B_TRUE;
475 
476 			if (num_tmp_lines != 0) {
477 				for (i = 0; i < num_tmp_lines; i++) {
478 					free(tmp_entries[i]);
479 					tmp_entries[i] = NULL;
480 				}
481 				num_tmp_lines = 0;
482 			}
483 		} else if (strcmp(tok, "bootfs") == 0) {
484 			char *bootfs = strtok(NULL, BE_WHITE_SPACE);
485 			found_title = B_FALSE;
486 			if (bootfs == NULL)
487 				continue;
488 
489 			if (strcmp(bootfs, be_root_ds) == 0) {
490 				found_be = B_TRUE;
491 				break;
492 			}
493 
494 			if (be_orig_root_ds != NULL &&
495 			    strcmp(bootfs, be_orig_root_ds) == 0 &&
496 			    !found_orig_be) {
497 				char str[BUFSIZ];
498 				found_orig_be = B_TRUE;
499 				num_lines = 0;
500 				/*
501 				 * Store the new title line
502 				 */
503 				(void) snprintf(str, BUFSIZ, "title %s\n",
504 				    description ? description : be_name);
505 				entries[num_lines] = strdup(str);
506 				num_lines++;
507 				/*
508 				 * If there are any lines between the title
509 				 * and the bootfs line store these. Also
510 				 * free the temporary lines.
511 				 */
512 				for (i = 0; i < num_tmp_lines; i++) {
513 					entries[num_lines] = tmp_entries[i];
514 					tmp_entries[i] = NULL;
515 					num_lines++;
516 				}
517 				num_tmp_lines = 0;
518 				/*
519 				 * Store the new bootfs line.
520 				 */
521 				(void) snprintf(str, BUFSIZ, "bootfs %s\n",
522 				    be_root_ds);
523 				entries[num_lines] = strdup(str);
524 				num_lines++;
525 				collect_lines = B_TRUE;
526 			}
527 		} else if (found_orig_be && collect_lines) {
528 			/*
529 			 * get the rest of the lines for the original BE and
530 			 * store them.
531 			 */
532 			if (strstr(line, BE_GRUB_COMMENT) != NULL ||
533 			    strstr(line, "BOOTADM") != NULL)
534 				continue;
535 			if (strcmp(tok, "splashimage") == 0) {
536 				entries[num_lines] =
537 				    strdup("splashimage "
538 				    "/boot/splashimage.xpm\n");
539 			} else {
540 				entries[num_lines] = strdup(temp_line);
541 			}
542 			num_lines++;
543 		} else if (found_title && !found_orig_be) {
544 			tmp_entries[num_tmp_lines] = strdup(temp_line);
545 			num_tmp_lines++;
546 		}
547 	}
548 
549 	(void) fclose(menu_fp);
550 
551 	if (found_be) {
552 		/*
553 		 * If an entry for this BE was already in the menu, then if
554 		 * that entry's title matches what we would have put in
555 		 * return success.  Otherwise return failure.
556 		 */
557 		char *new_title = description ? description : be_name;
558 
559 		if (strcmp(title, new_title) == 0) {
560 			ret = BE_SUCCESS;
561 			goto cleanup;
562 		} else {
563 			if (be_remove_menu(be_name, be_root_pool,
564 			    boot_pool) != BE_SUCCESS) {
565 				be_print_err(gettext("be_append_menu: "
566 				    "Failed to remove existing unusable "
567 				    "entry '%s' in boot menu.\n"), be_name);
568 				ret = BE_ERR_BE_EXISTS;
569 				goto cleanup;
570 			}
571 		}
572 	}
573 
574 	/* Append BE entry to the end of the file */
575 	menu_fp = fopen(menu_file, "a+");
576 	err = errno;
577 	if (menu_fp == NULL) {
578 		be_print_err(gettext("be_append_menu: failed "
579 		    "to open menu.lst file %s\n"), menu_file);
580 		ret = errno_to_be_err(err);
581 		goto cleanup;
582 	}
583 
584 	if (found_orig_be) {
585 		/*
586 		 * write out all the stored lines
587 		 */
588 		for (i = 0; i < num_lines; i++) {
589 			(void) fprintf(menu_fp, "%s", entries[i]);
590 			free(entries[i]);
591 		}
592 		num_lines = 0;
593 
594 		/*
595 		 * Check to see if this system supports grub
596 		 */
597 		if (be_has_grub())
598 			(void) fprintf(menu_fp, "%s\n", BE_GRUB_COMMENT);
599 		ret = BE_SUCCESS;
600 	} else {
601 		(void) fprintf(menu_fp, "title %s\n",
602 		    description ? description : be_name);
603 		(void) fprintf(menu_fp, "bootfs %s\n", be_root_ds);
604 
605 		/*
606 		 * Check to see if this system supports grub
607 		 */
608 		if (be_has_grub()) {
609 			(void) fprintf(menu_fp, "kernel$ "
610 			    "/platform/i86pc/kernel/$ISADIR/unix -B "
611 			    "$ZFS-BOOTFS\n");
612 			(void) fprintf(menu_fp, "module$ "
613 			    "/platform/i86pc/$ISADIR/boot_archive\n");
614 			(void) fprintf(menu_fp, "%s\n", BE_GRUB_COMMENT);
615 		}
616 		ret = BE_SUCCESS;
617 	}
618 	(void) fclose(menu_fp);
619 cleanup:
620 	if (pool_mounted) {
621 		int err = BE_SUCCESS;
622 		err = be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt);
623 		if (ret == BE_SUCCESS)
624 			ret = err;
625 		free(orig_mntpnt);
626 		free(ptmp_mntpnt);
627 	}
628 	ZFS_CLOSE(zhp);
629 	if (num_tmp_lines > 0) {
630 		for (i = 0; i < num_tmp_lines; i++) {
631 			free(tmp_entries[i]);
632 			tmp_entries[i] = NULL;
633 		}
634 	}
635 	if (num_lines > 0) {
636 		for (i = 0; i < num_lines; i++) {
637 			free(entries[i]);
638 			entries[i] = NULL;
639 		}
640 	}
641 	return (ret);
642 }
643 
644 /*
645  * Function:	be_remove_menu
646  * Description:	Removes a BE's entry from a menu.lst file.
647  * Parameters:
648  *		be_name - the name of BE whose entry is to be removed from
649  *			the menu.lst file.
650  *		be_root_pool - the pool that be_name lives in.
651  *		boot_pool - the pool where the BE is, if different than
652  *			the pool containing the boot menu.  If this is
653  *			NULL it will be set to be_root_pool.
654  * Returns:
655  *		BE_SUCCESS - Success
656  *		be_errno_t - Failure
657  * Scope:
658  *		Semi-private (library wide use only)
659  */
660 int
661 be_remove_menu(char *be_name, char *be_root_pool, char *boot_pool)
662 {
663 	zfs_handle_t	*zhp = NULL;
664 	char		be_root_ds[MAXPATHLEN];
665 	char		**buffer = NULL;
666 	char		menu_buf[BUFSIZ];
667 	char		menu[MAXPATHLEN];
668 	char		*pool_mntpnt = NULL;
669 	char		*ptmp_mntpnt = NULL;
670 	char		*orig_mntpnt = NULL;
671 	char		*tmp_menu = NULL;
672 	FILE		*menu_fp = NULL;
673 	FILE		*tmp_menu_fp = NULL;
674 	struct stat	sb;
675 	int		ret = BE_SUCCESS;
676 	int		i;
677 	int		fd;
678 	int		err = 0;
679 	int		nlines = 0;
680 	int		default_entry = 0;
681 	int		entry_cnt = 0;
682 	int		entry_del = 0;
683 	int		num_entry_del = 0;
684 	int		tmp_menu_len = 0;
685 	boolean_t	write = B_TRUE;
686 	boolean_t	do_buffer = B_FALSE;
687 	boolean_t	pool_mounted = B_FALSE;
688 
689 	if (boot_pool == NULL)
690 		boot_pool = be_root_pool;
691 
692 	/* Get name of BE's root dataset */
693 	be_make_root_ds(be_root_pool, be_name, be_root_ds, sizeof (be_root_ds));
694 
695 	/* Get handle to pool dataset */
696 	if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) {
697 		be_print_err(gettext("be_remove_menu: "
698 		    "failed to open pool dataset for %s: %s"),
699 		    be_root_pool, libzfs_error_description(g_zfs));
700 		return (zfs_err_to_be_err(g_zfs));
701 	}
702 
703 	/*
704 	 * Check to see if the pool's dataset is mounted. If it isn't we'll
705 	 * attempt to mount it.
706 	 */
707 	if ((ret = be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt,
708 	    &pool_mounted)) != BE_SUCCESS) {
709 		be_print_err(gettext("be_remove_menu: pool dataset "
710 		    "(%s) could not be mounted\n"), be_root_pool);
711 		ZFS_CLOSE(zhp);
712 		return (ret);
713 	}
714 
715 	/*
716 	 * Get the mountpoint for the root pool dataset.
717 	 */
718 	if (!zfs_is_mounted(zhp, &pool_mntpnt)) {
719 		be_print_err(gettext("be_remove_menu: pool "
720 		    "dataset (%s) is not mounted. Can't set "
721 		    "the default BE in the grub menu.\n"), be_root_pool);
722 		ret = BE_ERR_NO_MENU;
723 		goto cleanup;
724 	}
725 
726 	/* Get path to boot menu */
727 	(void) strlcpy(menu, pool_mntpnt, sizeof (menu));
728 
729 	/*
730 	 * Check to see if this system supports grub
731 	 */
732 	if (be_has_grub())
733 		(void) strlcat(menu, BE_GRUB_MENU, sizeof (menu));
734 	else
735 		(void) strlcat(menu, BE_SPARC_MENU, sizeof (menu));
736 
737 	/* Get handle to boot menu file */
738 	if ((ret = be_open_menu(be_root_pool, menu, &menu_fp, "r",
739 	    B_TRUE)) != BE_SUCCESS) {
740 		goto cleanup;
741 	} else if (menu_fp == NULL) {
742 		ret = BE_ERR_NO_MENU;
743 		goto cleanup;
744 	}
745 
746 	free(pool_mntpnt);
747 	pool_mntpnt = NULL;
748 
749 	/* Grab the stats of the original menu file */
750 	if (stat(menu, &sb) != 0) {
751 		err = errno;
752 		be_print_err(gettext("be_remove_menu: "
753 		    "failed to stat file %s: %s\n"), menu, strerror(err));
754 		ret = errno_to_be_err(err);
755 		goto cleanup;
756 	}
757 
758 	/* Create a tmp file for the modified menu.lst */
759 	tmp_menu_len = strlen(menu) + 7;
760 	if ((tmp_menu = (char *)malloc(tmp_menu_len)) == NULL) {
761 		be_print_err(gettext("be_remove_menu: malloc failed\n"));
762 		ret = BE_ERR_NOMEM;
763 		goto cleanup;
764 	}
765 	(void) memset(tmp_menu, 0, tmp_menu_len);
766 	(void) strlcpy(tmp_menu, menu, tmp_menu_len);
767 	(void) strlcat(tmp_menu, "XXXXXX", tmp_menu_len);
768 	if ((fd = mkstemp(tmp_menu)) == -1) {
769 		err = errno;
770 		be_print_err(gettext("be_remove_menu: mkstemp failed\n"));
771 		ret = errno_to_be_err(err);
772 		free(tmp_menu);
773 		tmp_menu = NULL;
774 		goto cleanup;
775 	}
776 	if ((tmp_menu_fp = fdopen(fd, "w")) == NULL) {
777 		err = errno;
778 		be_print_err(gettext("be_remove_menu: "
779 		    "could not open tmp file for write: %s\n"), strerror(err));
780 		(void) close(fd);
781 		ret = errno_to_be_err(err);
782 		goto cleanup;
783 	}
784 
785 	while (fgets(menu_buf, BUFSIZ, menu_fp)) {
786 		char tline [BUFSIZ];
787 		char *tok = NULL;
788 
789 		(void) strlcpy(tline, menu_buf, sizeof (tline));
790 
791 		/* Tokenize line */
792 		tok = strtok(tline, BE_WHITE_SPACE);
793 
794 		if (tok == NULL || tok[0] == '#') {
795 			/* Found empty line or comment line */
796 			if (do_buffer) {
797 				/* Buffer this line */
798 				if ((buffer = (char **)realloc(buffer,
799 				    sizeof (char *)*(nlines + 1))) == NULL) {
800 					ret = BE_ERR_NOMEM;
801 					goto cleanup;
802 				}
803 				if ((buffer[nlines++] = strdup(menu_buf))
804 				    == NULL) {
805 					ret = BE_ERR_NOMEM;
806 					goto cleanup;
807 				}
808 
809 			} else if (write || strncmp(menu_buf, BE_GRUB_COMMENT,
810 			    strlen(BE_GRUB_COMMENT)) != 0) {
811 				/* Write this line out */
812 				(void) fputs(menu_buf, tmp_menu_fp);
813 			}
814 		} else if (strcmp(tok, "default") == 0) {
815 			/*
816 			 * Record what 'default' is set to because we might
817 			 * need to adjust this upon deleting an entry.
818 			 */
819 			tok = strtok(NULL, BE_WHITE_SPACE);
820 
821 			if (tok != NULL) {
822 				default_entry = atoi(tok);
823 			}
824 
825 			(void) fputs(menu_buf, tmp_menu_fp);
826 		} else if (strcmp(tok, "title") == 0) {
827 			/*
828 			 * If we've reached a 'title' line and do_buffer is
829 			 * is true, that means we've just buffered an entire
830 			 * entry without finding a 'bootfs' directive.  We
831 			 * need to write that entry out and keep searching.
832 			 */
833 			if (do_buffer) {
834 				for (i = 0; i < nlines; i++) {
835 					(void) fputs(buffer[i], tmp_menu_fp);
836 					free(buffer[i]);
837 				}
838 				free(buffer);
839 				buffer = NULL;
840 				nlines = 0;
841 			}
842 
843 			/*
844 			 * Turn writing off and buffering on, and increment
845 			 * our entry counter.
846 			 */
847 			write = B_FALSE;
848 			do_buffer = B_TRUE;
849 			entry_cnt++;
850 
851 			/* Buffer this 'title' line */
852 			if ((buffer = (char **)realloc(buffer,
853 			    sizeof (char *)*(nlines + 1))) == NULL) {
854 				ret = BE_ERR_NOMEM;
855 				goto cleanup;
856 			}
857 			if ((buffer[nlines++] = strdup(menu_buf)) == NULL) {
858 				ret = BE_ERR_NOMEM;
859 				goto cleanup;
860 			}
861 
862 		} else if (strcmp(tok, "bootfs") == 0) {
863 			char *bootfs = NULL;
864 
865 			/*
866 			 * Found a 'bootfs' line.  See if it matches the
867 			 * BE we're looking for.
868 			 */
869 			if ((bootfs = strtok(NULL, BE_WHITE_SPACE)) == NULL ||
870 			    strcmp(bootfs, be_root_ds) != 0) {
871 				/*
872 				 * Either there's nothing after the 'bootfs'
873 				 * or this is not the BE we're looking for,
874 				 * write out the line(s) we've buffered since
875 				 * finding the title.
876 				 */
877 				for (i = 0; i < nlines; i++) {
878 					(void) fputs(buffer[i], tmp_menu_fp);
879 					free(buffer[i]);
880 				}
881 				free(buffer);
882 				buffer = NULL;
883 				nlines = 0;
884 
885 				/*
886 				 * Turn writing back on, and turn off buffering
887 				 * since this isn't the entry we're looking
888 				 * for.
889 				 */
890 				write = B_TRUE;
891 				do_buffer = B_FALSE;
892 
893 				/* Write this 'bootfs' line out. */
894 				(void) fputs(menu_buf, tmp_menu_fp);
895 			} else {
896 				/*
897 				 * Found the entry we're looking for.
898 				 * Record its entry number, increment the
899 				 * number of entries we've deleted, and turn
900 				 * writing off.  Also, throw away the lines
901 				 * we've buffered for this entry so far, we
902 				 * don't need them.
903 				 */
904 				entry_del = entry_cnt - 1;
905 				num_entry_del++;
906 				write = B_FALSE;
907 				do_buffer = B_FALSE;
908 
909 				for (i = 0; i < nlines; i++) {
910 					free(buffer[i]);
911 				}
912 				free(buffer);
913 				buffer = NULL;
914 				nlines = 0;
915 			}
916 		} else {
917 			if (do_buffer) {
918 				/* Buffer this line */
919 				if ((buffer = (char **)realloc(buffer,
920 				    sizeof (char *)*(nlines + 1))) == NULL) {
921 					ret = BE_ERR_NOMEM;
922 					goto cleanup;
923 				}
924 				if ((buffer[nlines++] = strdup(menu_buf))
925 				    == NULL) {
926 					ret = BE_ERR_NOMEM;
927 					goto cleanup;
928 				}
929 			} else if (write) {
930 				/* Write this line out */
931 				(void) fputs(menu_buf, tmp_menu_fp);
932 			}
933 		}
934 	}
935 
936 	(void) fclose(menu_fp);
937 	menu_fp = NULL;
938 	(void) fclose(tmp_menu_fp);
939 	tmp_menu_fp = NULL;
940 
941 	/* Copy the modified menu.lst into place */
942 	if (rename(tmp_menu, menu) != 0) {
943 		err = errno;
944 		be_print_err(gettext("be_remove_menu: "
945 		    "failed to rename file %s to %s: %s\n"),
946 		    tmp_menu, menu, strerror(err));
947 		ret = errno_to_be_err(err);
948 		goto cleanup;
949 	}
950 	free(tmp_menu);
951 	tmp_menu = NULL;
952 
953 	/*
954 	 * If we've removed an entry, see if we need to
955 	 * adjust the default value in the menu.lst.  If the
956 	 * entry we've deleted comes before the default entry
957 	 * we need to adjust the default value accordingly.
958 	 *
959 	 * be_has_grub is used here to check to see if this system
960 	 * supports grub.
961 	 */
962 	if (be_has_grub() && num_entry_del > 0) {
963 		if (entry_del <= default_entry) {
964 			default_entry = default_entry - num_entry_del;
965 			if (default_entry < 0)
966 				default_entry = 0;
967 
968 			/*
969 			 * Adjust the default value by rewriting the
970 			 * menu.lst file.  This may be overkill, but to
971 			 * preserve the location of the 'default' entry
972 			 * in the file, we need to do this.
973 			 */
974 
975 			/* Get handle to boot menu file */
976 			if ((menu_fp = fopen(menu, "r")) == NULL) {
977 				err = errno;
978 				be_print_err(gettext("be_remove_menu: "
979 				    "failed to open menu.lst (%s): %s\n"),
980 				    menu, strerror(err));
981 				ret = errno_to_be_err(err);
982 				goto cleanup;
983 			}
984 
985 			/* Create a tmp file for the modified menu.lst */
986 			tmp_menu_len = strlen(menu) + 7;
987 			if ((tmp_menu = (char *)malloc(tmp_menu_len))
988 			    == NULL) {
989 				be_print_err(gettext("be_remove_menu: "
990 				    "malloc failed\n"));
991 				ret = BE_ERR_NOMEM;
992 				goto cleanup;
993 			}
994 			(void) memset(tmp_menu, 0, tmp_menu_len);
995 			(void) strlcpy(tmp_menu, menu, tmp_menu_len);
996 			(void) strlcat(tmp_menu, "XXXXXX", tmp_menu_len);
997 			if ((fd = mkstemp(tmp_menu)) == -1) {
998 				err = errno;
999 				be_print_err(gettext("be_remove_menu: "
1000 				    "mkstemp failed: %s\n"), strerror(err));
1001 				ret = errno_to_be_err(err);
1002 				free(tmp_menu);
1003 				tmp_menu = NULL;
1004 				goto cleanup;
1005 			}
1006 			if ((tmp_menu_fp = fdopen(fd, "w")) == NULL) {
1007 				err = errno;
1008 				be_print_err(gettext("be_remove_menu: "
1009 				    "could not open tmp file for write: %s\n"),
1010 				    strerror(err));
1011 				(void) close(fd);
1012 				ret = errno_to_be_err(err);
1013 				goto cleanup;
1014 			}
1015 
1016 			while (fgets(menu_buf, BUFSIZ, menu_fp)) {
1017 				char tline [BUFSIZ];
1018 				char *tok = NULL;
1019 
1020 				(void) strlcpy(tline, menu_buf, sizeof (tline));
1021 
1022 				/* Tokenize line */
1023 				tok = strtok(tline, BE_WHITE_SPACE);
1024 
1025 				if (tok == NULL) {
1026 					/* Found empty line, write it out */
1027 					(void) fputs(menu_buf, tmp_menu_fp);
1028 				} else if (strcmp(tok, "default") == 0) {
1029 					/* Found the default line, adjust it */
1030 					(void) snprintf(tline, sizeof (tline),
1031 					    "default %d\n", default_entry);
1032 
1033 					(void) fputs(tline, tmp_menu_fp);
1034 				} else {
1035 					/* Pass through all other lines */
1036 					(void) fputs(menu_buf, tmp_menu_fp);
1037 				}
1038 			}
1039 
1040 			(void) fclose(menu_fp);
1041 			menu_fp = NULL;
1042 			(void) fclose(tmp_menu_fp);
1043 			tmp_menu_fp = NULL;
1044 
1045 			/* Copy the modified menu.lst into place */
1046 			if (rename(tmp_menu, menu) != 0) {
1047 				err = errno;
1048 				be_print_err(gettext("be_remove_menu: "
1049 				    "failed to rename file %s to %s: %s\n"),
1050 				    tmp_menu, menu, strerror(err));
1051 				ret = errno_to_be_err(err);
1052 				goto cleanup;
1053 			}
1054 
1055 			free(tmp_menu);
1056 			tmp_menu = NULL;
1057 		}
1058 	}
1059 
1060 	/* Set the perms and ownership of the updated file */
1061 	if (chmod(menu, sb.st_mode) != 0) {
1062 		err = errno;
1063 		be_print_err(gettext("be_remove_menu: "
1064 		    "failed to chmod %s: %s\n"), menu, strerror(err));
1065 		ret = errno_to_be_err(err);
1066 		goto cleanup;
1067 	}
1068 	if (chown(menu, sb.st_uid, sb.st_gid) != 0) {
1069 		err = errno;
1070 		be_print_err(gettext("be_remove_menu: "
1071 		    "failed to chown %s: %s\n"), menu, strerror(err));
1072 		ret = errno_to_be_err(err);
1073 		goto cleanup;
1074 	}
1075 
1076 cleanup:
1077 	if (pool_mounted) {
1078 		int err = BE_SUCCESS;
1079 		err = be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt);
1080 		if (ret == BE_SUCCESS)
1081 			ret = err;
1082 		free(orig_mntpnt);
1083 		free(ptmp_mntpnt);
1084 	}
1085 	ZFS_CLOSE(zhp);
1086 
1087 	free(buffer);
1088 	if (menu_fp != NULL)
1089 		(void) fclose(menu_fp);
1090 	if (tmp_menu_fp != NULL)
1091 		(void) fclose(tmp_menu_fp);
1092 	if (tmp_menu != NULL) {
1093 		(void) unlink(tmp_menu);
1094 		free(tmp_menu);
1095 	}
1096 
1097 	return (ret);
1098 }
1099 
1100 /*
1101  * Function:	be_default_grub_bootfs
1102  * Description:	This function returns the dataset in the default entry of
1103  *		the grub menu. If no default entry is found with a valid bootfs
1104  *		entry NULL is returned.
1105  * Parameters:
1106  *		be_root_pool - This is the name of the root pool where the
1107  *			       grub menu can be found.
1108  *              def_bootfs - This is used to pass back the bootfs string. On
1109  *				error NULL is returned here.
1110  * Returns:
1111  *		Success - BE_SUCCESS is returned.
1112  *		Failure - a be_errno_t is returned.
1113  * Scope:
1114  *		Semi-private (library wide use only)
1115  */
1116 int
1117 be_default_grub_bootfs(const char *be_root_pool, char **def_bootfs)
1118 {
1119 	zfs_handle_t	*zhp = NULL;
1120 	char		grub_file[MAXPATHLEN];
1121 	FILE		*menu_fp;
1122 	char		line[BUFSIZ];
1123 	char		*pool_mntpnt = NULL;
1124 	char		*ptmp_mntpnt = NULL;
1125 	char		*orig_mntpnt = NULL;
1126 	int		default_entry = 0, entries = 0;
1127 	int		found_default = 0;
1128 	int		ret = BE_SUCCESS;
1129 	boolean_t	pool_mounted = B_FALSE;
1130 
1131 	errno = 0;
1132 
1133 	/*
1134 	 * Check to see if this system supports grub
1135 	 */
1136 	if (!be_has_grub()) {
1137 		be_print_err(gettext("be_default_grub_bootfs: operation "
1138 		    "not supported on this architecture\n"));
1139 		return (BE_ERR_NOTSUP);
1140 	}
1141 
1142 	*def_bootfs = NULL;
1143 
1144 	/* Get handle to pool dataset */
1145 	if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) {
1146 		be_print_err(gettext("be_default_grub_bootfs: "
1147 		    "failed to open pool dataset for %s: %s"),
1148 		    be_root_pool, libzfs_error_description(g_zfs));
1149 		return (zfs_err_to_be_err(g_zfs));
1150 	}
1151 
1152 	/*
1153 	 * Check to see if the pool's dataset is mounted. If it isn't we'll
1154 	 * attempt to mount it.
1155 	 */
1156 	if ((ret = be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt,
1157 	    &pool_mounted)) != BE_SUCCESS) {
1158 		be_print_err(gettext("be_default_grub_bootfs: pool dataset "
1159 		    "(%s) could not be mounted\n"), be_root_pool);
1160 		ZFS_CLOSE(zhp);
1161 		return (ret);
1162 	}
1163 
1164 	/*
1165 	 * Get the mountpoint for the root pool dataset.
1166 	 */
1167 	if (!zfs_is_mounted(zhp, &pool_mntpnt)) {
1168 		be_print_err(gettext("be_default_grub_bootfs: failed "
1169 		    "to get mount point for the root pool. Can't set "
1170 		    "the default BE in the grub menu.\n"));
1171 		ret = BE_ERR_NO_MENU;
1172 		goto cleanup;
1173 	}
1174 
1175 	(void) snprintf(grub_file, MAXPATHLEN, "%s%s",
1176 	    pool_mntpnt, BE_GRUB_MENU);
1177 
1178 	if ((ret = be_open_menu((char *)be_root_pool, grub_file,
1179 	    &menu_fp, "r", B_FALSE)) != BE_SUCCESS) {
1180 		goto cleanup;
1181 	} else if (menu_fp == NULL) {
1182 		ret = BE_ERR_NO_MENU;
1183 		goto cleanup;
1184 	}
1185 
1186 	free(pool_mntpnt);
1187 	pool_mntpnt = NULL;
1188 
1189 	while (fgets(line, BUFSIZ, menu_fp)) {
1190 		char *tok = strtok(line, BE_WHITE_SPACE);
1191 
1192 		if (tok != NULL && tok[0] != '#') {
1193 			if (!found_default) {
1194 				if (strcmp(tok, "default") == 0) {
1195 					tok = strtok(NULL, BE_WHITE_SPACE);
1196 					if (tok != NULL) {
1197 						default_entry = atoi(tok);
1198 						rewind(menu_fp);
1199 						found_default = 1;
1200 					}
1201 				}
1202 				continue;
1203 			}
1204 			if (strcmp(tok, "title") == 0) {
1205 				entries++;
1206 			} else if (default_entry == entries - 1) {
1207 				if (strcmp(tok, "bootfs") == 0) {
1208 					tok = strtok(NULL, BE_WHITE_SPACE);
1209 					(void) fclose(menu_fp);
1210 
1211 					if (tok == NULL) {
1212 						ret = BE_SUCCESS;
1213 						goto cleanup;
1214 					}
1215 
1216 					if ((*def_bootfs = strdup(tok)) !=
1217 					    NULL) {
1218 						ret = BE_SUCCESS;
1219 						goto cleanup;
1220 					}
1221 					be_print_err(gettext(
1222 					    "be_default_grub_bootfs: "
1223 					    "memory allocation failed\n"));
1224 					ret = BE_ERR_NOMEM;
1225 					goto cleanup;
1226 				}
1227 			} else if (default_entry < entries - 1) {
1228 				/*
1229 				 * no bootfs entry for the default entry.
1230 				 */
1231 				break;
1232 			}
1233 		}
1234 	}
1235 	(void) fclose(menu_fp);
1236 
1237 cleanup:
1238 	if (pool_mounted) {
1239 		int err = BE_SUCCESS;
1240 		err = be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt);
1241 		if (ret == BE_SUCCESS)
1242 			ret = err;
1243 		free(orig_mntpnt);
1244 		free(ptmp_mntpnt);
1245 	}
1246 	ZFS_CLOSE(zhp);
1247 	return (ret);
1248 }
1249 
1250 /*
1251  * Function:	be_change_grub_default
1252  * Description:	This function takes two parameters. These are the name of
1253  *		the BE we want to have as the default booted in the grub
1254  *		menu and the root pool where the path to the grub menu exists.
1255  *		The code takes this and finds the BE's entry in the grub menu
1256  *		and changes the default entry to point to that entry in the
1257  *		list.
1258  * Parameters:
1259  *		be_name - This is the name of the BE wanted as the default
1260  *			for the next boot.
1261  *		be_root_pool - This is the name of the root pool where the
1262  *			grub menu can be found.
1263  * Returns:
1264  *		BE_SUCCESS - Success
1265  *		be_errno_t - Failure
1266  * Scope:
1267  *		Semi-private (library wide use only)
1268  */
1269 int
1270 be_change_grub_default(char *be_name, char *be_root_pool)
1271 {
1272 	zfs_handle_t	*zhp = NULL;
1273 	char	grub_file[MAXPATHLEN];
1274 	char	*temp_grub;
1275 	char	*pool_mntpnt = NULL;
1276 	char	*ptmp_mntpnt = NULL;
1277 	char	*orig_mntpnt = NULL;
1278 	char	line[BUFSIZ];
1279 	char	temp_line[BUFSIZ];
1280 	char	be_root_ds[MAXPATHLEN];
1281 	FILE	*grub_fp = NULL;
1282 	FILE	*temp_fp = NULL;
1283 	struct stat	sb;
1284 	int	temp_grub_len = 0;
1285 	int	fd, entries = 0;
1286 	int	err = 0;
1287 	int	ret = BE_SUCCESS;
1288 	boolean_t	found_default = B_FALSE;
1289 	boolean_t	pool_mounted = B_FALSE;
1290 
1291 	errno = 0;
1292 
1293 	/*
1294 	 * Check to see if this system supports grub
1295 	 */
1296 	if (!be_has_grub()) {
1297 		be_print_err(gettext("be_change_grub_default: operation "
1298 		    "not supported on this architecture\n"));
1299 		return (BE_ERR_NOTSUP);
1300 	}
1301 
1302 	/* Generate string for BE's root dataset */
1303 	be_make_root_ds(be_root_pool, be_name, be_root_ds, sizeof (be_root_ds));
1304 
1305 	/* Get handle to pool dataset */
1306 	if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) {
1307 		be_print_err(gettext("be_change_grub_default: "
1308 		    "failed to open pool dataset for %s: %s"),
1309 		    be_root_pool, libzfs_error_description(g_zfs));
1310 		return (zfs_err_to_be_err(g_zfs));
1311 	}
1312 
1313 	/*
1314 	 * Check to see if the pool's dataset is mounted. If it isn't we'll
1315 	 * attempt to mount it.
1316 	 */
1317 	if ((ret = be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt,
1318 	    &pool_mounted)) != BE_SUCCESS) {
1319 		be_print_err(gettext("be_change_grub_default: pool dataset "
1320 		    "(%s) could not be mounted\n"), be_root_pool);
1321 		ZFS_CLOSE(zhp);
1322 		return (ret);
1323 	}
1324 
1325 	/*
1326 	 * Get the mountpoint for the root pool dataset.
1327 	 */
1328 	if (!zfs_is_mounted(zhp, &pool_mntpnt)) {
1329 		be_print_err(gettext("be_change_grub_default: pool "
1330 		    "dataset (%s) is not mounted. Can't set "
1331 		    "the default BE in the grub menu.\n"), be_root_pool);
1332 		ret = BE_ERR_NO_MENU;
1333 		goto cleanup;
1334 	}
1335 
1336 	(void) snprintf(grub_file, MAXPATHLEN, "%s%s",
1337 	    pool_mntpnt, BE_GRUB_MENU);
1338 
1339 	if ((ret = be_open_menu(be_root_pool, grub_file,
1340 	    &grub_fp, "r+", B_TRUE)) != BE_SUCCESS) {
1341 		goto cleanup;
1342 	} else if (grub_fp == NULL) {
1343 		ret = BE_ERR_NO_MENU;
1344 		goto cleanup;
1345 	}
1346 
1347 	free(pool_mntpnt);
1348 	pool_mntpnt = NULL;
1349 
1350 	/* Grab the stats of the original menu file */
1351 	if (stat(grub_file, &sb) != 0) {
1352 		err = errno;
1353 		be_print_err(gettext("be_change_grub_default: "
1354 		    "failed to stat file %s: %s\n"), grub_file, strerror(err));
1355 		ret = errno_to_be_err(err);
1356 		goto cleanup;
1357 	}
1358 
1359 	/* Create a tmp file for the modified menu.lst */
1360 	temp_grub_len = strlen(grub_file) + 7;
1361 	if ((temp_grub = (char *)malloc(temp_grub_len)) == NULL) {
1362 		be_print_err(gettext("be_change_grub_default: "
1363 		    "malloc failed\n"));
1364 		ret = BE_ERR_NOMEM;
1365 		goto cleanup;
1366 	}
1367 	(void) memset(temp_grub, 0, temp_grub_len);
1368 	(void) strlcpy(temp_grub, grub_file, temp_grub_len);
1369 	(void) strlcat(temp_grub, "XXXXXX", temp_grub_len);
1370 	if ((fd = mkstemp(temp_grub)) == -1) {
1371 		err = errno;
1372 		be_print_err(gettext("be_change_grub_default: "
1373 		    "mkstemp failed: %s\n"), strerror(err));
1374 		ret = errno_to_be_err(err);
1375 		free(temp_grub);
1376 		temp_grub = NULL;
1377 		goto cleanup;
1378 	}
1379 	if ((temp_fp = fdopen(fd, "w")) == NULL) {
1380 		err = errno;
1381 		be_print_err(gettext("be_change_grub_default: "
1382 		    "failed to open %s file: %s\n"),
1383 		    temp_grub, strerror(err));
1384 		(void) close(fd);
1385 		ret = errno_to_be_err(err);
1386 		goto cleanup;
1387 	}
1388 
1389 	while (fgets(line, BUFSIZ, grub_fp)) {
1390 		char *tok = strtok(line, BE_WHITE_SPACE);
1391 
1392 		if (tok == NULL || tok[0] == '#') {
1393 			continue;
1394 		} else if (strcmp(tok, "title") == 0) {
1395 			entries++;
1396 			continue;
1397 		} else if (strcmp(tok, "bootfs") == 0) {
1398 			char *bootfs = strtok(NULL, BE_WHITE_SPACE);
1399 			if (bootfs == NULL)
1400 				continue;
1401 
1402 			if (strcmp(bootfs, be_root_ds) == 0) {
1403 				found_default = B_TRUE;
1404 				break;
1405 			}
1406 		}
1407 	}
1408 
1409 	if (!found_default) {
1410 		be_print_err(gettext("be_change_grub_default: failed "
1411 		    "to find entry for %s in the grub menu\n"),
1412 		    be_name);
1413 		ret = BE_ERR_BE_NOENT;
1414 		goto cleanup;
1415 	}
1416 
1417 	rewind(grub_fp);
1418 
1419 	while (fgets(line, BUFSIZ, grub_fp)) {
1420 		char *tok = NULL;
1421 
1422 		(void) strncpy(temp_line, line, BUFSIZ);
1423 
1424 		if ((tok = strtok(temp_line, BE_WHITE_SPACE)) != NULL &&
1425 		    strcmp(tok, "default") == 0) {
1426 			(void) snprintf(temp_line, BUFSIZ, "default %d\n",
1427 			    entries - 1 >= 0 ? entries - 1 : 0);
1428 			(void) fputs(temp_line, temp_fp);
1429 		} else {
1430 			(void) fputs(line, temp_fp);
1431 		}
1432 	}
1433 
1434 	(void) fclose(grub_fp);
1435 	grub_fp = NULL;
1436 	(void) fclose(temp_fp);
1437 	temp_fp = NULL;
1438 
1439 	if (rename(temp_grub, grub_file) != 0) {
1440 		err = errno;
1441 		be_print_err(gettext("be_change_grub_default: "
1442 		    "failed to rename file %s to %s: %s\n"),
1443 		    temp_grub, grub_file, strerror(err));
1444 		ret = errno_to_be_err(err);
1445 		goto cleanup;
1446 	}
1447 	free(temp_grub);
1448 	temp_grub = NULL;
1449 
1450 	/* Set the perms and ownership of the updated file */
1451 	if (chmod(grub_file, sb.st_mode) != 0) {
1452 		err = errno;
1453 		be_print_err(gettext("be_change_grub_default: "
1454 		    "failed to chmod %s: %s\n"), grub_file, strerror(err));
1455 		ret = errno_to_be_err(err);
1456 		goto cleanup;
1457 	}
1458 	if (chown(grub_file, sb.st_uid, sb.st_gid) != 0) {
1459 		err = errno;
1460 		be_print_err(gettext("be_change_grub_default: "
1461 		    "failed to chown %s: %s\n"), grub_file, strerror(err));
1462 		ret = errno_to_be_err(err);
1463 	}
1464 
1465 cleanup:
1466 	if (pool_mounted) {
1467 		int err = BE_SUCCESS;
1468 		err = be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt);
1469 		if (ret == BE_SUCCESS)
1470 			ret = err;
1471 		free(orig_mntpnt);
1472 		free(ptmp_mntpnt);
1473 	}
1474 	ZFS_CLOSE(zhp);
1475 	if (grub_fp != NULL)
1476 		(void) fclose(grub_fp);
1477 	if (temp_fp != NULL)
1478 		(void) fclose(temp_fp);
1479 	if (temp_grub != NULL) {
1480 		(void) unlink(temp_grub);
1481 		free(temp_grub);
1482 	}
1483 
1484 	return (ret);
1485 }
1486 
1487 /*
1488  * Function:	be_update_menu
1489  * Description:	This function is used by be_rename to change the BE name in
1490  *		an existing entry in the grub menu to the new name of the BE.
1491  * Parameters:
1492  *		be_orig_name - the original name of the BE
1493  *		be_new_name - the new name the BE is being renameed to.
1494  *		be_root_pool - The pool which contains the grub menu
1495  *		boot_pool - the pool where the BE is, if different than
1496  *			the pool containing the boot menu.  If this is
1497  *			NULL it will be set to be_root_pool.
1498  * Returns:
1499  *		BE_SUCCESS - Success
1500  *		be_errno_t - Failure
1501  * Scope:
1502  *		Semi-private (library wide use only)
1503  */
1504 int
1505 be_update_menu(char *be_orig_name, char *be_new_name, char *be_root_pool,
1506     char *boot_pool)
1507 {
1508 	zfs_handle_t *zhp = NULL;
1509 	char menu_file[MAXPATHLEN];
1510 	char be_root_ds[MAXPATHLEN];
1511 	char be_new_root_ds[MAXPATHLEN];
1512 	char line[BUFSIZ];
1513 	char *pool_mntpnt = NULL;
1514 	char *ptmp_mntpnt = NULL;
1515 	char *orig_mntpnt = NULL;
1516 	char *temp_menu = NULL;
1517 	FILE *menu_fp = NULL;
1518 	FILE *new_fp = NULL;
1519 	struct stat sb;
1520 	int temp_menu_len = 0;
1521 	int tmp_fd;
1522 	int ret = BE_SUCCESS;
1523 	int err = 0;
1524 	boolean_t pool_mounted = B_FALSE;
1525 
1526 	errno = 0;
1527 
1528 	if (boot_pool == NULL)
1529 		boot_pool = be_root_pool;
1530 
1531 	if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) {
1532 		be_print_err(gettext("be_update_menu: failed to open "
1533 		    "pool dataset for %s: %s\n"), be_root_pool,
1534 		    libzfs_error_description(g_zfs));
1535 		return (zfs_err_to_be_err(g_zfs));
1536 	}
1537 
1538 	/*
1539 	 * Check to see if the pool's dataset is mounted. If it isn't we'll
1540 	 * attempt to mount it.
1541 	 */
1542 	if ((ret = be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt,
1543 	    &pool_mounted)) != BE_SUCCESS) {
1544 		be_print_err(gettext("be_update_menu: pool dataset "
1545 		    "(%s) could not be mounted\n"), be_root_pool);
1546 		ZFS_CLOSE(zhp);
1547 		return (ret);
1548 	}
1549 
1550 	/*
1551 	 * Get the mountpoint for the root pool dataset.
1552 	 */
1553 	if (!zfs_is_mounted(zhp, &pool_mntpnt)) {
1554 		be_print_err(gettext("be_update_menu: failed "
1555 		    "to get mount point for the root pool. Can't set "
1556 		    "the default BE in the grub menu.\n"));
1557 		ret = BE_ERR_NO_MENU;
1558 		goto cleanup;
1559 	}
1560 
1561 	/*
1562 	 * Check to see if this system supports grub
1563 	 */
1564 	if (be_has_grub()) {
1565 		(void) snprintf(menu_file, sizeof (menu_file),
1566 		    "%s%s", pool_mntpnt, BE_GRUB_MENU);
1567 	} else {
1568 		(void) snprintf(menu_file, sizeof (menu_file),
1569 		    "%s%s", pool_mntpnt, BE_SPARC_MENU);
1570 	}
1571 
1572 	be_make_root_ds(be_root_pool, be_orig_name, be_root_ds,
1573 	    sizeof (be_root_ds));
1574 	be_make_root_ds(be_root_pool, be_new_name, be_new_root_ds,
1575 	    sizeof (be_new_root_ds));
1576 
1577 	if ((ret = be_open_menu(be_root_pool, menu_file,
1578 	    &menu_fp, "r", B_TRUE)) != BE_SUCCESS) {
1579 		goto cleanup;
1580 	} else if (menu_fp == NULL) {
1581 		ret = BE_ERR_NO_MENU;
1582 		goto cleanup;
1583 	}
1584 
1585 	free(pool_mntpnt);
1586 	pool_mntpnt = NULL;
1587 
1588 	/* Grab the stat of the original menu file */
1589 	if (stat(menu_file, &sb) != 0) {
1590 		err = errno;
1591 		be_print_err(gettext("be_update_menu: "
1592 		    "failed to stat file %s: %s\n"), menu_file, strerror(err));
1593 		(void) fclose(menu_fp);
1594 		ret = errno_to_be_err(err);
1595 		goto cleanup;
1596 	}
1597 
1598 	/* Create tmp file for modified menu.lst */
1599 	temp_menu_len = strlen(menu_file) + 7;
1600 	if ((temp_menu = (char *)malloc(temp_menu_len))
1601 	    == NULL) {
1602 		be_print_err(gettext("be_update_menu: "
1603 		    "malloc failed\n"));
1604 		(void) fclose(menu_fp);
1605 		ret = BE_ERR_NOMEM;
1606 		goto cleanup;
1607 	}
1608 	(void) memset(temp_menu, 0, temp_menu_len);
1609 	(void) strlcpy(temp_menu, menu_file, temp_menu_len);
1610 	(void) strlcat(temp_menu, "XXXXXX", temp_menu_len);
1611 	if ((tmp_fd = mkstemp(temp_menu)) == -1) {
1612 		err = errno;
1613 		be_print_err(gettext("be_update_menu: "
1614 		    "mkstemp failed: %s\n"), strerror(err));
1615 		(void) fclose(menu_fp);
1616 		free(temp_menu);
1617 		ret = errno_to_be_err(err);
1618 		goto cleanup;
1619 	}
1620 	if ((new_fp = fdopen(tmp_fd, "w")) == NULL) {
1621 		err = errno;
1622 		be_print_err(gettext("be_update_menu: "
1623 		    "fdopen failed: %s\n"), strerror(err));
1624 		(void) close(tmp_fd);
1625 		(void) fclose(menu_fp);
1626 		free(temp_menu);
1627 		ret = errno_to_be_err(err);
1628 		goto cleanup;
1629 	}
1630 
1631 	while (fgets(line, BUFSIZ, menu_fp)) {
1632 		char tline[BUFSIZ];
1633 		char new_line[BUFSIZ];
1634 		char *c = NULL;
1635 
1636 		(void) strlcpy(tline, line, sizeof (tline));
1637 
1638 		/* Tokenize line */
1639 		c = strtok(tline, BE_WHITE_SPACE);
1640 
1641 		if (c == NULL) {
1642 			/* Found empty line, write it out. */
1643 			(void) fputs(line, new_fp);
1644 		} else if (c[0] == '#') {
1645 			/* Found a comment line, write it out. */
1646 			(void) fputs(line, new_fp);
1647 		} else if (strcmp(c, "title") == 0) {
1648 			char *name = NULL;
1649 			char *desc = NULL;
1650 
1651 			/*
1652 			 * Found a 'title' line, parse out BE name or
1653 			 * the description.
1654 			 */
1655 			name = strtok(NULL, BE_WHITE_SPACE);
1656 
1657 			if (name == NULL) {
1658 				/*
1659 				 * Nothing after 'title', just push
1660 				 * this line through
1661 				 */
1662 				(void) fputs(line, new_fp);
1663 			} else {
1664 				/*
1665 				 * Grab the remainder of the title which
1666 				 * could be a multi worded description
1667 				 */
1668 				desc = strtok(NULL, "\n");
1669 
1670 				if (strcmp(name, be_orig_name) == 0) {
1671 					/*
1672 					 * The first token of the title is
1673 					 * the old BE name, replace it with
1674 					 * the new one, and write it out
1675 					 * along with the remainder of
1676 					 * description if there is one.
1677 					 */
1678 					if (desc) {
1679 						(void) snprintf(new_line,
1680 						    sizeof (new_line),
1681 						    "title %s %s\n",
1682 						    be_new_name, desc);
1683 					} else {
1684 						(void) snprintf(new_line,
1685 						    sizeof (new_line),
1686 						    "title %s\n", be_new_name);
1687 					}
1688 
1689 					(void) fputs(new_line, new_fp);
1690 				} else {
1691 					(void) fputs(line, new_fp);
1692 				}
1693 			}
1694 		} else if (strcmp(c, "bootfs") == 0) {
1695 			/*
1696 			 * Found a 'bootfs' line, parse out the BE root
1697 			 * dataset value.
1698 			 */
1699 			char *root_ds = strtok(NULL, BE_WHITE_SPACE);
1700 
1701 			if (root_ds == NULL) {
1702 				/*
1703 				 * Nothing after 'bootfs', just push
1704 				 * this line through
1705 				 */
1706 				(void) fputs(line, new_fp);
1707 			} else {
1708 				/*
1709 				 * If this bootfs is the one we're renaming,
1710 				 * write out the new root dataset value
1711 				 */
1712 				if (strcmp(root_ds, be_root_ds) == 0) {
1713 					(void) snprintf(new_line,
1714 					    sizeof (new_line), "bootfs %s\n",
1715 					    be_new_root_ds);
1716 
1717 					(void) fputs(new_line, new_fp);
1718 				} else {
1719 					(void) fputs(line, new_fp);
1720 				}
1721 			}
1722 		} else {
1723 			/*
1724 			 * Found some other line we don't care
1725 			 * about, write it out.
1726 			 */
1727 			(void) fputs(line, new_fp);
1728 		}
1729 	}
1730 
1731 	(void) fclose(menu_fp);
1732 	(void) fclose(new_fp);
1733 	(void) close(tmp_fd);
1734 
1735 	if (rename(temp_menu, menu_file) != 0) {
1736 		err = errno;
1737 		be_print_err(gettext("be_update_menu: "
1738 		    "failed to rename file %s to %s: %s\n"),
1739 		    temp_menu, menu_file, strerror(err));
1740 		ret = errno_to_be_err(err);
1741 	}
1742 	free(temp_menu);
1743 
1744 	/* Set the perms and ownership of the updated file */
1745 	if (chmod(menu_file, sb.st_mode) != 0) {
1746 		err = errno;
1747 		be_print_err(gettext("be_update_menu: "
1748 		    "failed to chmod %s: %s\n"), menu_file, strerror(err));
1749 		ret = errno_to_be_err(err);
1750 		goto cleanup;
1751 	}
1752 	if (chown(menu_file, sb.st_uid, sb.st_gid) != 0) {
1753 		err = errno;
1754 		be_print_err(gettext("be_update_menu: "
1755 		    "failed to chown %s: %s\n"), menu_file, strerror(err));
1756 		ret = errno_to_be_err(err);
1757 	}
1758 
1759 cleanup:
1760 	if (pool_mounted) {
1761 		int err = BE_SUCCESS;
1762 		err = be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt);
1763 		if (ret == BE_SUCCESS)
1764 			ret = err;
1765 		free(orig_mntpnt);
1766 		free(ptmp_mntpnt);
1767 	}
1768 	ZFS_CLOSE(zhp);
1769 	return (ret);
1770 }
1771 
1772 /*
1773  * Function:	be_has_menu_entry
1774  * Description:	Checks to see if the BEs root dataset has an entry in the grub
1775  *		menu.
1776  * Parameters:
1777  *		be_dataset - The root dataset of the BE
1778  *		be_root_pool - The pool which contains the boot menu
1779  *		entry - A pointer the the entry number of the BE if found.
1780  * Returns:
1781  *		B_TRUE - Success
1782  *		B_FALSE - Failure
1783  * Scope:
1784  *		Semi-private (library wide use only)
1785  */
1786 boolean_t
1787 be_has_menu_entry(char *be_dataset, char *be_root_pool, int *entry)
1788 {
1789 	zfs_handle_t *zhp = NULL;
1790 	char		menu_file[MAXPATHLEN];
1791 	FILE		*menu_fp;
1792 	char		line[BUFSIZ];
1793 	char		*last;
1794 	char		*rpool_mntpnt = NULL;
1795 	char		*ptmp_mntpnt = NULL;
1796 	char		*orig_mntpnt = NULL;
1797 	int		ent_num = 0;
1798 	boolean_t	ret = 0;
1799 	boolean_t	pool_mounted = B_FALSE;
1800 
1801 
1802 	/*
1803 	 * Check to see if this system supports grub
1804 	 */
1805 	if ((zhp = zfs_open(g_zfs, be_root_pool, ZFS_TYPE_DATASET)) == NULL) {
1806 		be_print_err(gettext("be_has_menu_entry: failed to open "
1807 		    "pool dataset for %s: %s\n"), be_root_pool,
1808 		    libzfs_error_description(g_zfs));
1809 		return (B_FALSE);
1810 	}
1811 
1812 	/*
1813 	 * Check to see if the pool's dataset is mounted. If it isn't we'll
1814 	 * attempt to mount it.
1815 	 */
1816 	if (be_mount_pool(zhp, &ptmp_mntpnt, &orig_mntpnt,
1817 	    &pool_mounted) != 0) {
1818 		be_print_err(gettext("be_has_menu_entry: pool dataset "
1819 		    "(%s) could not be mounted\n"), be_root_pool);
1820 		ZFS_CLOSE(zhp);
1821 		return (B_FALSE);
1822 	}
1823 
1824 	/*
1825 	 * Get the mountpoint for the root pool dataset.
1826 	 */
1827 	if (!zfs_is_mounted(zhp, &rpool_mntpnt)) {
1828 		be_print_err(gettext("be_has_menu_entry: pool "
1829 		    "dataset (%s) is not mounted. Can't set "
1830 		    "the default BE in the grub menu.\n"), be_root_pool);
1831 		ret = B_FALSE;
1832 		goto cleanup;
1833 	}
1834 
1835 	if (be_has_grub()) {
1836 		(void) snprintf(menu_file, MAXPATHLEN, "/%s%s",
1837 		    rpool_mntpnt, BE_GRUB_MENU);
1838 	} else {
1839 		(void) snprintf(menu_file, MAXPATHLEN, "/%s%s",
1840 		    rpool_mntpnt, BE_SPARC_MENU);
1841 	}
1842 
1843 	if (be_open_menu(be_root_pool, menu_file, &menu_fp, "r",
1844 	    B_FALSE) != 0) {
1845 		ret = B_FALSE;
1846 		goto cleanup;
1847 	} else if (menu_fp == NULL) {
1848 		ret = B_FALSE;
1849 		goto cleanup;
1850 	}
1851 
1852 	free(rpool_mntpnt);
1853 	rpool_mntpnt = NULL;
1854 
1855 	while (fgets(line, BUFSIZ, menu_fp)) {
1856 		char *tok = strtok_r(line, BE_WHITE_SPACE, &last);
1857 
1858 		if (tok != NULL && tok[0] != '#') {
1859 			if (strcmp(tok, "bootfs") == 0) {
1860 				tok = strtok_r(last, BE_WHITE_SPACE, &last);
1861 				if (tok != NULL && strcmp(tok,
1862 				    be_dataset) == 0) {
1863 					(void) fclose(menu_fp);
1864 					/*
1865 					 * The entry number needs to be
1866 					 * decremented here because the title
1867 					 * will always be the first line for
1868 					 * an entry. Because of this we'll
1869 					 * always be off by one entry when we
1870 					 * check for bootfs.
1871 					 */
1872 					*entry = ent_num - 1;
1873 					ret = B_TRUE;
1874 					goto cleanup;
1875 				}
1876 			} else if (strcmp(tok, "title") == 0)
1877 				ent_num++;
1878 		}
1879 	}
1880 
1881 cleanup:
1882 	if (pool_mounted) {
1883 		(void) be_unmount_pool(zhp, ptmp_mntpnt, orig_mntpnt);
1884 		free(orig_mntpnt);
1885 		free(ptmp_mntpnt);
1886 	}
1887 	ZFS_CLOSE(zhp);
1888 	(void) fclose(menu_fp);
1889 	return (ret);
1890 }
1891 
1892 /*
1893  * Function:	be_update_vfstab
1894  * Description:	This function digs into a BE's vfstab and updates all
1895  *		entries with file systems listed in be_fs_list_data_t.
1896  *		The entry's root container dataset and be_name will be
1897  *		updated with the parameters passed in.
1898  * Parameters:
1899  *		be_name - name of BE to update
1900  *		old_rc_loc - dataset under which the root container dataset
1901  *			of the old BE resides in.
1902  *		new_rc_loc - dataset under which the root container dataset
1903  *			of the new BE resides in.
1904  *		fld - be_fs_list_data_t pointer providing the list of
1905  *			file systems to look for in vfstab.
1906  *		mountpoint - directory of where BE is currently mounted.
1907  *			If NULL, then BE is not currently mounted.
1908  * Returns:
1909  *		BE_SUCCESS - Success
1910  *		be_errno_t - Failure
1911  * Scope:
1912  *		Semi-private (library wide use only)
1913  */
1914 int
1915 be_update_vfstab(char *be_name, char *old_rc_loc, char *new_rc_loc,
1916     be_fs_list_data_t *fld, char *mountpoint)
1917 {
1918 	char		*tmp_mountpoint = NULL;
1919 	char		alt_vfstab[MAXPATHLEN];
1920 	int		ret = BE_SUCCESS, err = BE_SUCCESS;
1921 
1922 	if (fld == NULL || fld->fs_list == NULL || fld->fs_num == 0)
1923 		return (BE_SUCCESS);
1924 
1925 	/* If BE not already mounted, mount the BE */
1926 	if (mountpoint == NULL) {
1927 		if ((ret = _be_mount(be_name, &tmp_mountpoint,
1928 		    BE_MOUNT_FLAG_NO_ZONES)) != BE_SUCCESS) {
1929 			be_print_err(gettext("be_update_vfstab: "
1930 			    "failed to mount BE (%s)\n"), be_name);
1931 			return (ret);
1932 		}
1933 	} else {
1934 		tmp_mountpoint = mountpoint;
1935 	}
1936 
1937 	/* Get string for vfstab in the mounted BE. */
1938 	(void) snprintf(alt_vfstab, sizeof (alt_vfstab), "%s/etc/vfstab",
1939 	    tmp_mountpoint);
1940 
1941 	/* Update the vfstab */
1942 	ret = _update_vfstab(alt_vfstab, be_name, old_rc_loc, new_rc_loc,
1943 	    fld);
1944 
1945 	/* Unmount BE if we mounted it */
1946 	if (mountpoint == NULL) {
1947 		if ((err = _be_unmount(be_name, 0)) == BE_SUCCESS) {
1948 			/* Remove temporary mountpoint */
1949 			(void) rmdir(tmp_mountpoint);
1950 		} else {
1951 			be_print_err(gettext("be_update_vfstab: "
1952 			    "failed to unmount BE %s mounted at %s\n"),
1953 			    be_name, tmp_mountpoint);
1954 			if (ret == BE_SUCCESS)
1955 				ret = err;
1956 		}
1957 
1958 		free(tmp_mountpoint);
1959 	}
1960 
1961 	return (ret);
1962 }
1963 
1964 /*
1965  * Function:	be_update_zone_vfstab
1966  * Description:	This function digs into a zone BE's vfstab and updates all
1967  *		entries with file systems listed in be_fs_list_data_t.
1968  *		The entry's root container dataset and be_name will be
1969  *		updated with the parameters passed in.
1970  * Parameters:
1971  *		zhp - zfs_handle_t pointer to zone root dataset.
1972  *		be_name - name of zone BE to update
1973  *		old_rc_loc - dataset under which the root container dataset
1974  *			of the old zone BE resides in.
1975  *		new_rc_loc - dataset under which the root container dataset
1976  *			of the new zone BE resides in.
1977  *		fld - be_fs_list_data_t pointer providing the list of
1978  *			file systems to look for in vfstab.
1979  * Returns:
1980  *		BE_SUCCESS - Success
1981  *		be_errno_t - Failure
1982  * Scope:
1983  *		Semi-private (library wide use only)
1984  */
1985 int
1986 be_update_zone_vfstab(zfs_handle_t *zhp, char *be_name, char *old_rc_loc,
1987     char *new_rc_loc, be_fs_list_data_t *fld)
1988 {
1989 	be_mount_data_t		md = { 0 };
1990 	be_unmount_data_t	ud = { 0 };
1991 	char			alt_vfstab[MAXPATHLEN];
1992 	boolean_t		mounted_here = B_FALSE;
1993 	int			ret = BE_SUCCESS;
1994 
1995 	/*
1996 	 * If zone root not already mounted, mount it at a
1997 	 * temporary location.
1998 	 */
1999 	if (!zfs_is_mounted(zhp, &md.altroot)) {
2000 		/* Generate temporary mountpoint to mount zone root */
2001 		if ((ret = be_make_tmp_mountpoint(&md.altroot)) != BE_SUCCESS) {
2002 			be_print_err(gettext("be_update_zone_vfstab: "
2003 			    "failed to make temporary mountpoint to "
2004 			    "mount zone root\n"));
2005 			return (ret);
2006 		}
2007 
2008 		if (be_mount_zone_root(zhp, &md) != BE_SUCCESS) {
2009 			be_print_err(gettext("be_update_zone_vfstab: "
2010 			    "failed to mount zone root %s\n"),
2011 			    zfs_get_name(zhp));
2012 			free(md.altroot);
2013 			return (BE_ERR_MOUNT_ZONEROOT);
2014 		}
2015 		mounted_here = B_TRUE;
2016 	}
2017 
2018 	/* Get string from vfstab in the mounted zone BE */
2019 	(void) snprintf(alt_vfstab, sizeof (alt_vfstab), "%s/etc/vfstab",
2020 	    md.altroot);
2021 
2022 	/* Update the vfstab */
2023 	ret = _update_vfstab(alt_vfstab, be_name, old_rc_loc, new_rc_loc,
2024 	    fld);
2025 
2026 	/* Unmount zone root if we mounted it */
2027 	if (mounted_here) {
2028 		ud.force = B_TRUE;
2029 
2030 		if (be_unmount_zone_root(zhp, &ud) == BE_SUCCESS) {
2031 			/* Remove the temporary mountpoint */
2032 			(void) rmdir(md.altroot);
2033 		} else {
2034 			be_print_err(gettext("be_update_zone_vfstab: "
2035 			    "failed to unmount zone root %s from %s\n"),
2036 			    zfs_get_name(zhp), md.altroot);
2037 			if (ret == 0)
2038 				ret = BE_ERR_UMOUNT_ZONEROOT;
2039 		}
2040 	}
2041 
2042 	free(md.altroot);
2043 	return (ret);
2044 }
2045 
2046 /*
2047  * Function:	be_auto_snap_name
2048  * Description:	Generate an auto snapshot name constructed based on the
2049  *		current date and time.  The auto snapshot name is of the form:
2050  *
2051  *			<date>-<time>
2052  *
2053  *		where <date> is in ISO standard format, so the resultant name
2054  *		is of the form:
2055  *
2056  *			%Y-%m-%d-%H:%M:%S
2057  *
2058  * Parameters:
2059  *		None
2060  * Returns:
2061  *		Success - pointer to auto generated snapshot name.  The name
2062  *			is allocated in heap storage so the caller is
2063  *			responsible for free'ing the name.
2064  *		Failure - NULL
2065  * Scope:
2066  *		Semi-private (library wide use only)
2067  */
2068 char *
2069 be_auto_snap_name(void)
2070 {
2071 	time_t		utc_tm = NULL;
2072 	struct tm	*gmt_tm = NULL;
2073 	char		gmt_time_str[64];
2074 	char		*auto_snap_name = NULL;
2075 
2076 	if (time(&utc_tm) == -1) {
2077 		be_print_err(gettext("be_auto_snap_name: time() failed\n"));
2078 		return (NULL);
2079 	}
2080 
2081 	if ((gmt_tm = gmtime(&utc_tm)) == NULL) {
2082 		be_print_err(gettext("be_auto_snap_name: gmtime() failed\n"));
2083 		return (NULL);
2084 	}
2085 
2086 	(void) strftime(gmt_time_str, sizeof (gmt_time_str), "%F-%T", gmt_tm);
2087 
2088 	if ((auto_snap_name = strdup(gmt_time_str)) == NULL) {
2089 		be_print_err(gettext("be_auto_snap_name: "
2090 		    "memory allocation failed\n"));
2091 		return (NULL);
2092 	}
2093 
2094 	return (auto_snap_name);
2095 }
2096 
2097 /*
2098  * Function:	be_auto_be_name
2099  * Description:	Generate an auto BE name constructed based on the BE name
2100  *		of the original BE being cloned.
2101  * Parameters:
2102  *		obe_name - name of the original BE being cloned.
2103  * Returns:
2104  *		Success - pointer to auto generated BE name.  The name
2105  *			is allocated in heap storage so the caller is
2106  *			responsible for free'ing the name.
2107  *		Failure - NULL
2108  * Scope:
2109  *		Semi-private (library wide use only)
2110  */
2111 char *
2112 be_auto_be_name(char *obe_name)
2113 {
2114 	return (be_get_auto_name(obe_name, NULL, B_FALSE));
2115 }
2116 
2117 /*
2118  * Function:	be_auto_zone_be_name
2119  * Description:	Generate an auto BE name for a zone constructed based on
2120  *              the BE name of the original zone BE being cloned.
2121  * Parameters:
2122  *              container_ds - container dataset for the zone.
2123  *		zbe_name - name of the original zone BE being cloned.
2124  * Returns:
2125  *		Success - pointer to auto generated BE name.  The name
2126  *			is allocated in heap storage so the caller is
2127  *			responsible for free'ing the name.
2128  *		Failure - NULL
2129  * Scope:
2130  *		Semi-private (library wide use only)
2131  */
2132 char *
2133 be_auto_zone_be_name(char *container_ds, char *zbe_name)
2134 {
2135 	return (be_get_auto_name(zbe_name, container_ds, B_TRUE));
2136 }
2137 
2138 /*
2139  * Function:	be_valid_be_name
2140  * Description:	Validates a BE name.
2141  * Parameters:
2142  *		be_name - name of BE to validate
2143  * Returns:
2144  *		B_TRUE - be_name is valid
2145  *		B_FALSE - be_name is invalid
2146  * Scope:
2147  *		Semi-private (library wide use only)
2148  */
2149 
2150 boolean_t
2151 be_valid_be_name(const char *be_name)
2152 {
2153 	const char	*c = NULL;
2154 	struct be_defaults be_defaults;
2155 
2156 	if (be_name == NULL)
2157 		return (B_FALSE);
2158 
2159 	be_get_defaults(&be_defaults);
2160 
2161 	/*
2162 	 * A BE name must not be a multi-level dataset name.  We also check
2163 	 * that it does not contain the ' ' and '%' characters.  The ' ' is
2164 	 * a valid character for datasets, however we don't allow that in a
2165 	 * BE name.  The '%' is invalid, but zfs_name_valid() allows it for
2166 	 * internal reasons, so we explicitly check for it here.
2167 	 */
2168 	c = be_name;
2169 	while (*c != '\0' && *c != '/' && *c != ' ' && *c != '%')
2170 		c++;
2171 
2172 	if (*c != '\0')
2173 		return (B_FALSE);
2174 
2175 	/*
2176 	 * The BE name must comply with a zfs dataset filesystem. We also
2177 	 * verify its length to be < BE_NAME_MAX_LEN.
2178 	 */
2179 	if (!zfs_name_valid(be_name, ZFS_TYPE_FILESYSTEM) ||
2180 	    strlen(be_name) > BE_NAME_MAX_LEN)
2181 		return (B_FALSE);
2182 
2183 	if (be_defaults.be_deflt_bename_starts_with[0] != '\0' &&
2184 	    strstr(be_name, be_defaults.be_deflt_bename_starts_with) == NULL) {
2185 		return (B_FALSE);
2186 	}
2187 
2188 	return (B_TRUE);
2189 }
2190 
2191 /*
2192  * Function:	be_valid_auto_snap_name
2193  * Description:	This function checks that a snapshot name is a valid auto
2194  *		generated snapshot name.  A valid auto generated snapshot
2195  *		name is of the form:
2196  *
2197  *			%Y-%m-%d-%H:%M:%S
2198  *
2199  *		An older form of the auto generated snapshot name also
2200  *		included the snapshot's BE cleanup policy and a reserved
2201  *		field.  Those names will also be verified by this function.
2202  *
2203  *		Examples of valid auto snapshot names are:
2204  *
2205  *			2008-03-31-18:41:30
2206  *			2008-03-31-22:17:24
2207  *			<policy>:-:2008:04-05-09:12:55
2208  *			<policy>:-:2008:04-06-15:34:12
2209  *
2210  * Parameters:
2211  *		name - name of the snapshot to be validated.
2212  * Returns:
2213  *		B_TRUE - the name is a valid auto snapshot name.
2214  *		B_FALSE - the name is not a valid auto snapshot name.
2215  * Scope:
2216  *		Semi-private (library wide use only)
2217  */
2218 boolean_t
2219 be_valid_auto_snap_name(char *name)
2220 {
2221 	struct tm gmt_tm;
2222 
2223 	char *policy = NULL;
2224 	char *reserved = NULL;
2225 	char *date = NULL;
2226 	char *c = NULL;
2227 
2228 	/* Validate the snapshot name by converting it into utc time */
2229 	if (strptime(name, "%Y-%m-%d-%T", &gmt_tm) != NULL &&
2230 	    (mktime(&gmt_tm) != -1)) {
2231 		return (B_TRUE);
2232 	}
2233 
2234 	/*
2235 	 * Validate the snapshot name against the older form of an
2236 	 * auto generated snapshot name.
2237 	 */
2238 	policy = strdup(name);
2239 
2240 	/*
2241 	 * Get the first field from the snapshot name,
2242 	 * which is the BE policy
2243 	 */
2244 	c = strchr(policy, ':');
2245 	if (c == NULL) {
2246 		free(policy);
2247 		return (B_FALSE);
2248 	}
2249 	c[0] = '\0';
2250 
2251 	/* Validate the policy name */
2252 	if (!valid_be_policy(policy)) {
2253 		free(policy);
2254 		return (B_FALSE);
2255 	}
2256 
2257 	/* Get the next field, which is the reserved field. */
2258 	if (c[1] == NULL || c[1] == '\0') {
2259 		free(policy);
2260 		return (B_FALSE);
2261 	}
2262 	reserved = c+1;
2263 	c = strchr(reserved, ':');
2264 	if (c == NULL) {
2265 		free(policy);
2266 		return (B_FALSE);
2267 	}
2268 	c[0] = '\0';
2269 
2270 	/* Validate the reserved field */
2271 	if (strcmp(reserved, "-") != 0) {
2272 		free(policy);
2273 		return (B_FALSE);
2274 	}
2275 
2276 	/* The remaining string should be the date field */
2277 	if (c[1] == NULL || c[1] == '\0') {
2278 		free(policy);
2279 		return (B_FALSE);
2280 	}
2281 	date = c+1;
2282 
2283 	/* Validate the date string by converting it into utc time */
2284 	if (strptime(date, "%Y-%m-%d-%T", &gmt_tm) == NULL ||
2285 	    (mktime(&gmt_tm) == -1)) {
2286 		be_print_err(gettext("be_valid_auto_snap_name: "
2287 		    "invalid auto snapshot name\n"));
2288 		free(policy);
2289 		return (B_FALSE);
2290 	}
2291 
2292 	free(policy);
2293 	return (B_TRUE);
2294 }
2295 
2296 /*
2297  * Function:	be_default_policy
2298  * Description:	Temporary hardcoded policy support.  This function returns
2299  *		the default policy type to be used to create a BE or a BE
2300  *		snapshot.
2301  * Parameters:
2302  *		None
2303  * Returns:
2304  *		Name of default BE policy.
2305  * Scope:
2306  *		Semi-private (library wide use only)
2307  */
2308 char *
2309 be_default_policy(void)
2310 {
2311 	return (BE_PLCY_STATIC);
2312 }
2313 
2314 /*
2315  * Function:	valid_be_policy
2316  * Description:	Temporary hardcoded policy support.  This function valids
2317  *		whether a policy is a valid known policy or not.
2318  * Paramters:
2319  *		policy - name of policy to validate.
2320  * Returns:
2321  *		B_TRUE - policy is a valid.
2322  *		B_FALSE - policy is invalid.
2323  * Scope:
2324  *		Semi-private (library wide use only)
2325  */
2326 boolean_t
2327 valid_be_policy(char *policy)
2328 {
2329 	if (policy == NULL)
2330 		return (B_FALSE);
2331 
2332 	if (strcmp(policy, BE_PLCY_STATIC) == 0 ||
2333 	    strcmp(policy, BE_PLCY_VOLATILE) == 0) {
2334 		return (B_TRUE);
2335 	}
2336 
2337 	return (B_FALSE);
2338 }
2339 
2340 /*
2341  * Function:	be_print_err
2342  * Description:	This function prints out error messages if do_print is
2343  *		set to B_TRUE or if the BE_PRINT_ERR environment variable
2344  *		is set to true.
2345  * Paramters:
2346  *		prnt_str - the string we wish to print and any arguments
2347  *		for the format of that string.
2348  * Returns:
2349  *		void
2350  * Scope:
2351  *		Semi-private (library wide use only)
2352  */
2353 void
2354 be_print_err(char *prnt_str, ...)
2355 {
2356 	va_list ap;
2357 	char buf[BUFSIZ];
2358 	char *env_buf;
2359 	static boolean_t env_checked = B_FALSE;
2360 
2361 	if (!env_checked) {
2362 		if ((env_buf = getenv("BE_PRINT_ERR")) != NULL) {
2363 			if (strcasecmp(env_buf, "true") == 0) {
2364 				do_print = B_TRUE;
2365 			}
2366 		}
2367 		env_checked = B_TRUE;
2368 	}
2369 
2370 	if (do_print) {
2371 		va_start(ap, prnt_str);
2372 		/* LINTED variable format specifier */
2373 		(void) vsnprintf(buf, BUFSIZ, prnt_str, ap);
2374 		(void) fputs(buf, stderr);
2375 		va_end(ap);
2376 	}
2377 }
2378 
2379 /*
2380  * Function:	be_find_current_be
2381  * Description:	Find the currently "active" BE. Fill in the
2382  * 		passed in be_transaction_data_t reference with the
2383  *		active BE's data.
2384  * Paramters:
2385  *		none
2386  * Returns:
2387  *		BE_SUCCESS - Success
2388  *		be_errnot_t - Failure
2389  * Scope:
2390  *		Semi-private (library wide use only)
2391  * Notes:
2392  *		The caller is responsible for initializing the libzfs handle
2393  *		and freeing the memory used by the active be_name.
2394  */
2395 int
2396 be_find_current_be(be_transaction_data_t *bt)
2397 {
2398 	int	zret;
2399 
2400 	if ((zret = zpool_iter(g_zfs, be_zpool_find_current_be_callback,
2401 	    bt)) == 0) {
2402 		be_print_err(gettext("be_find_current_be: failed to "
2403 		    "find current BE name\n"));
2404 		return (BE_ERR_BE_NOENT);
2405 	} else if (zret < 0) {
2406 		be_print_err(gettext("be_find_current_be: "
2407 		    "zpool_iter failed: %s\n"),
2408 		    libzfs_error_description(g_zfs));
2409 		return (zfs_err_to_be_err(g_zfs));
2410 	}
2411 
2412 	return (BE_SUCCESS);
2413 }
2414 
2415 /*
2416  * Function:	be_zpool_find_current_be_callback
2417  * Description: Callback function used to iterate through all existing pools
2418  *		to find the BE that is the currently booted BE.
2419  * Parameters:
2420  *		zlp - zpool_handle_t pointer to the current pool being
2421  *			looked at.
2422  *		data - be_transaction_data_t pointer.
2423  *			Upon successfully finding the current BE, the
2424  *			obe_zpool member of this parameter is set to the
2425  *			pool it is found in.
2426  * Return:
2427  *		1 - Found current BE in this pool.
2428  *		0 - Did not find current BE in this pool.
2429  * Scope:
2430  *		Semi-private (library wide use only)
2431  */
2432 int
2433 be_zpool_find_current_be_callback(zpool_handle_t *zlp, void *data)
2434 {
2435 	be_transaction_data_t	*bt = data;
2436 	zfs_handle_t		*zhp = NULL;
2437 	const char		*zpool =  zpool_get_name(zlp);
2438 	char			be_container_ds[MAXPATHLEN];
2439 
2440 	/*
2441 	 * Generate string for BE container dataset
2442 	 */
2443 	be_make_container_ds(zpool, be_container_ds, sizeof (be_container_ds));
2444 
2445 	/*
2446 	 * Check if a BE container dataset exists in this pool.
2447 	 */
2448 	if (!zfs_dataset_exists(g_zfs, be_container_ds, ZFS_TYPE_FILESYSTEM)) {
2449 		zpool_close(zlp);
2450 		return (0);
2451 	}
2452 
2453 	/*
2454 	 * Get handle to this zpool's BE container dataset.
2455 	 */
2456 	if ((zhp = zfs_open(g_zfs, be_container_ds, ZFS_TYPE_FILESYSTEM)) ==
2457 	    NULL) {
2458 		be_print_err(gettext("be_zpool_find_current_be_callback: "
2459 		    "failed to open BE container dataset (%s)\n"),
2460 		    be_container_ds);
2461 		zpool_close(zlp);
2462 		return (0);
2463 	}
2464 
2465 	/*
2466 	 * Iterate through all potential BEs in this zpool
2467 	 */
2468 	if (zfs_iter_filesystems(zhp, be_zfs_find_current_be_callback, bt)) {
2469 		/*
2470 		 * Found current BE dataset; set obe_zpool
2471 		 */
2472 		if ((bt->obe_zpool = strdup(zpool)) == NULL) {
2473 			be_print_err(gettext(
2474 			    "be_zpool_find_current_be_callback: "
2475 			    "memory allocation failed\n"));
2476 			ZFS_CLOSE(zhp);
2477 			zpool_close(zlp);
2478 			return (0);
2479 		}
2480 
2481 		ZFS_CLOSE(zhp);
2482 		zpool_close(zlp);
2483 		return (1);
2484 	}
2485 
2486 	ZFS_CLOSE(zhp);
2487 	zpool_close(zlp);
2488 
2489 	return (0);
2490 }
2491 
2492 /*
2493  * Function:	be_zfs_find_current_be_callback
2494  * Description:	Callback function used to iterate through all BEs in a
2495  *		pool to find the BE that is the currently booted BE.
2496  * Parameters:
2497  *		zhp - zfs_handle_t pointer to current filesystem being checked.
2498  *		data - be_transaction-data_t pointer
2499  *			Upon successfully finding the current BE, the
2500  *			obe_name and obe_root_ds members of this parameter
2501  *			are set to the BE name and BE's root dataset
2502  *			respectively.
2503  * Return:
2504  *		1 - Found current BE.
2505  *		0 - Did not find current BE.
2506  * Scope:
2507  *		Semi-private (library wide use only)
2508  */
2509 int
2510 be_zfs_find_current_be_callback(zfs_handle_t *zhp, void *data)
2511 {
2512 	be_transaction_data_t	*bt = data;
2513 	char			*mp = NULL;
2514 
2515 	/*
2516 	 * Check if dataset is mounted, and if so where.
2517 	 */
2518 	if (zfs_is_mounted(zhp, &mp)) {
2519 		/*
2520 		 * If mounted at root, set obe_root_ds and obe_name
2521 		 */
2522 		if (mp != NULL && strcmp(mp, "/") == 0) {
2523 			free(mp);
2524 
2525 			if ((bt->obe_root_ds = strdup(zfs_get_name(zhp)))
2526 			    == NULL) {
2527 				be_print_err(gettext(
2528 				    "be_zfs_find_current_be_callback: "
2529 				    "memory allocation failed\n"));
2530 				ZFS_CLOSE(zhp);
2531 				return (0);
2532 			}
2533 
2534 			if ((bt->obe_name = strdup(basename(bt->obe_root_ds)))
2535 			    == NULL) {
2536 				be_print_err(gettext(
2537 				    "be_zfs_find_current_be_callback: "
2538 				    "memory allocation failed\n"));
2539 				ZFS_CLOSE(zhp);
2540 				return (0);
2541 			}
2542 
2543 			ZFS_CLOSE(zhp);
2544 			return (1);
2545 		}
2546 
2547 		free(mp);
2548 	}
2549 	ZFS_CLOSE(zhp);
2550 
2551 	return (0);
2552 }
2553 
2554 /*
2555  * Function:	be_check_be_roots_callback
2556  * Description:	This function checks whether or not the dataset name passed
2557  *		is hierachically located under the BE root container dataset
2558  *		for this pool.
2559  * Parameters:
2560  *		zlp - zpool_handle_t pointer to current pool being processed.
2561  *		data - name of dataset to check
2562  * Returns:
2563  *		0 - dataset is not in this pool's BE root container dataset
2564  *		1 - dataset is in this pool's BE root container dataset
2565  * Scope:
2566  *		Semi-private (library wide use only)
2567  */
2568 int
2569 be_check_be_roots_callback(zpool_handle_t *zlp, void *data)
2570 {
2571 	const char	*zpool = zpool_get_name(zlp);
2572 	char		*ds = data;
2573 	char		be_container_ds[MAXPATHLEN];
2574 
2575 	/* Generate string for this pool's BE root container dataset */
2576 	be_make_container_ds(zpool, be_container_ds, sizeof (be_container_ds));
2577 
2578 	/*
2579 	 * If dataset lives under the BE root container dataset
2580 	 * of this pool, return failure.
2581 	 */
2582 	if (strncmp(be_container_ds, ds, strlen(be_container_ds)) == 0 &&
2583 	    ds[strlen(be_container_ds)] == '/') {
2584 		zpool_close(zlp);
2585 		return (1);
2586 	}
2587 
2588 	zpool_close(zlp);
2589 	return (0);
2590 }
2591 
2592 /*
2593  * Function:	zfs_err_to_be_err
2594  * Description:	This function takes the error stored in the libzfs handle
2595  *		and maps it to an be_errno_t. If there are no matching
2596  *		be_errno_t's then BE_ERR_ZFS is returned.
2597  * Paramters:
2598  *		zfsh - The libzfs handle containing the error we're looking up.
2599  * Returns:
2600  *		be_errno_t
2601  * Scope:
2602  *		Semi-private (library wide use only)
2603  */
2604 int
2605 zfs_err_to_be_err(libzfs_handle_t *zfsh)
2606 {
2607 	int err = libzfs_errno(zfsh);
2608 
2609 	switch (err) {
2610 	case 0:
2611 		return (BE_SUCCESS);
2612 	case EZFS_PERM:
2613 		return (BE_ERR_PERM);
2614 	case EZFS_INTR:
2615 		return (BE_ERR_INTR);
2616 	case EZFS_NOENT:
2617 		return (BE_ERR_NOENT);
2618 	case EZFS_NOSPC:
2619 		return (BE_ERR_NOSPC);
2620 	case EZFS_MOUNTFAILED:
2621 		return (BE_ERR_MOUNT);
2622 	case EZFS_UMOUNTFAILED:
2623 		return (BE_ERR_UMOUNT);
2624 	case EZFS_EXISTS:
2625 		return (BE_ERR_BE_EXISTS);
2626 	case EZFS_BUSY:
2627 		return (BE_ERR_DEV_BUSY);
2628 	case EZFS_POOLREADONLY:
2629 		return (BE_ERR_ROFS);
2630 	case EZFS_NAMETOOLONG:
2631 		return (BE_ERR_NAMETOOLONG);
2632 	case EZFS_NODEVICE:
2633 		return (BE_ERR_NODEV);
2634 	case EZFS_POOL_INVALARG:
2635 		return (BE_ERR_INVAL);
2636 	case EZFS_PROPTYPE:
2637 		return (BE_ERR_INVALPROP);
2638 	case EZFS_BADTYPE:
2639 		return (BE_ERR_DSTYPE);
2640 	case EZFS_PROPNONINHERIT:
2641 		return (BE_ERR_NONINHERIT);
2642 	case EZFS_PROPREADONLY:
2643 		return (BE_ERR_READONLYPROP);
2644 	case EZFS_RESILVERING:
2645 	case EZFS_POOLUNAVAIL:
2646 		return (BE_ERR_UNAVAIL);
2647 	case EZFS_DSREADONLY:
2648 		return (BE_ERR_READONLYDS);
2649 	default:
2650 		return (BE_ERR_ZFS);
2651 	}
2652 }
2653 
2654 /*
2655  * Function:	errno_to_be_err
2656  * Description:	This function takes an errno and maps it to an be_errno_t.
2657  *		If there are no matching be_errno_t's then BE_ERR_UNKNOWN is
2658  *		returned.
2659  * Paramters:
2660  *		err - The errno we're compairing against.
2661  * Returns:
2662  *		be_errno_t
2663  * Scope:
2664  *		Semi-private (library wide use only)
2665  */
2666 int
2667 errno_to_be_err(int err)
2668 {
2669 	switch (err) {
2670 	case EPERM:
2671 		return (BE_ERR_PERM);
2672 	case EACCES:
2673 		return (BE_ERR_ACCESS);
2674 	case ECANCELED:
2675 		return (BE_ERR_CANCELED);
2676 	case EINTR:
2677 		return (BE_ERR_INTR);
2678 	case ENOENT:
2679 		return (BE_ERR_NOENT);
2680 	case ENOSPC:
2681 	case EDQUOT:
2682 		return (BE_ERR_NOSPC);
2683 	case EEXIST:
2684 		return (BE_ERR_BE_EXISTS);
2685 	case EBUSY:
2686 		return (BE_ERR_BUSY);
2687 	case EROFS:
2688 		return (BE_ERR_ROFS);
2689 	case ENAMETOOLONG:
2690 		return (BE_ERR_NAMETOOLONG);
2691 	case ENXIO:
2692 		return (BE_ERR_NXIO);
2693 	case EINVAL:
2694 		return (BE_ERR_INVAL);
2695 	case EFAULT:
2696 		return (BE_ERR_FAULT);
2697 	default:
2698 		return (BE_ERR_UNKNOWN);
2699 	}
2700 }
2701 
2702 /*
2703  * Function:	be_err_to_str
2704  * Description:	This function takes a be_errno_t and maps it to a message.
2705  *		If there are no matching be_errno_t's then NULL is returned.
2706  * Paramters:
2707  *		be_errno_t - The be_errno_t we're mapping.
2708  * Returns:
2709  *		string or NULL if the error code is not known.
2710  * Scope:
2711  *		Semi-private (library wide use only)
2712  */
2713 char *
2714 be_err_to_str(int err)
2715 {
2716 	switch (err) {
2717 	case BE_ERR_ACCESS:
2718 		return (gettext("Permission denied."));
2719 	case BE_ERR_ACTIVATE_CURR:
2720 		return (gettext("Activation of current BE failed."));
2721 	case BE_ERR_AUTONAME:
2722 		return (gettext("Auto naming failed."));
2723 	case BE_ERR_BE_NOENT:
2724 		return (gettext("No such BE."));
2725 	case BE_ERR_BUSY:
2726 		return (gettext("Mount busy."));
2727 	case BE_ERR_DEV_BUSY:
2728 		return (gettext("Device busy."));
2729 	case BE_ERR_CANCELED:
2730 		return (gettext("Operation canceled."));
2731 	case BE_ERR_CLONE:
2732 		return (gettext("BE clone failed."));
2733 	case BE_ERR_COPY:
2734 		return (gettext("BE copy failed."));
2735 	case BE_ERR_CREATDS:
2736 		return (gettext("Dataset creation failed."));
2737 	case BE_ERR_CURR_BE_NOT_FOUND:
2738 		return (gettext("Can't find current BE."));
2739 	case BE_ERR_DESTROY:
2740 		return (gettext("Failed to destroy BE or snapshot."));
2741 	case BE_ERR_DESTROY_CURR_BE:
2742 		return (gettext("Cannot destroy current BE."));
2743 	case BE_ERR_DEMOTE:
2744 		return (gettext("BE demotion failed."));
2745 	case BE_ERR_DSTYPE:
2746 		return (gettext("Invalid dataset type."));
2747 	case BE_ERR_BE_EXISTS:
2748 		return (gettext("BE exists."));
2749 	case BE_ERR_INIT:
2750 		return (gettext("be_zfs_init failed."));
2751 	case BE_ERR_INTR:
2752 		return (gettext("Interupted system call."));
2753 	case BE_ERR_INVAL:
2754 		return (gettext("Invalid argument."));
2755 	case BE_ERR_INVALPROP:
2756 		return (gettext("Invalid property for dataset."));
2757 	case BE_ERR_INVALMOUNTPOINT:
2758 		return (gettext("Unexpected mountpoint."));
2759 	case BE_ERR_MOUNT:
2760 		return (gettext("Mount failed."));
2761 	case BE_ERR_MOUNTED:
2762 		return (gettext("Already mounted."));
2763 	case BE_ERR_NAMETOOLONG:
2764 		return (gettext("name > BUFSIZ."));
2765 	case BE_ERR_NOENT:
2766 		return (gettext("Doesn't exist."));
2767 	case BE_ERR_POOL_NOENT:
2768 		return (gettext("No such pool."));
2769 	case BE_ERR_NODEV:
2770 		return (gettext("No such device."));
2771 	case BE_ERR_NOTMOUNTED:
2772 		return (gettext("File system not mounted."));
2773 	case BE_ERR_NOMEM:
2774 		return (gettext("Not enough memory."));
2775 	case BE_ERR_NONINHERIT:
2776 		return (gettext(
2777 		    "Property is not inheritable for the BE dataset."));
2778 	case BE_ERR_NXIO:
2779 		return (gettext("No such device or address."));
2780 	case BE_ERR_NOSPC:
2781 		return (gettext("No space on device."));
2782 	case BE_ERR_NOTSUP:
2783 		return (gettext("Operation not supported."));
2784 	case BE_ERR_OPEN:
2785 		return (gettext("Open failed."));
2786 	case BE_ERR_PERM:
2787 		return (gettext("Not owner."));
2788 	case BE_ERR_UNAVAIL:
2789 		return (gettext("The BE is currently unavailable."));
2790 	case BE_ERR_PROMOTE:
2791 		return (gettext("BE promotion failed."));
2792 	case BE_ERR_ROFS:
2793 		return (gettext("Read only file system."));
2794 	case BE_ERR_READONLYDS:
2795 		return (gettext("Read only dataset."));
2796 	case BE_ERR_READONLYPROP:
2797 		return (gettext("Read only property."));
2798 	case BE_ERR_RENAME_ACTIVE:
2799 		return (gettext("Renaming the active BE is not supported."));
2800 	case BE_ERR_SS_EXISTS:
2801 		return (gettext("Snapshot exists."));
2802 	case BE_ERR_SS_NOENT:
2803 		return (gettext("No such snapshot."));
2804 	case BE_ERR_UMOUNT:
2805 		return (gettext("Unmount failed."));
2806 	case BE_ERR_UMOUNT_CURR_BE:
2807 		return (gettext("Can't unmount the current BE."));
2808 	case BE_ERR_UMOUNT_SHARED:
2809 		return (gettext("Unmount of a shared File System failed."));
2810 	case BE_ERR_FAULT:
2811 		return (gettext("Bad address."));
2812 	case BE_ERR_UNKNOWN:
2813 		return (gettext("Unknown error."));
2814 	case BE_ERR_ZFS:
2815 		return (gettext("ZFS returned an error."));
2816 	case BE_ERR_GEN_UUID:
2817 		return (gettext("Failed to generate uuid."));
2818 	case BE_ERR_PARSE_UUID:
2819 		return (gettext("Failed to parse uuid."));
2820 	case BE_ERR_NO_UUID:
2821 		return (gettext("No uuid"));
2822 	case BE_ERR_ZONE_NO_PARENTBE:
2823 		return (gettext("No parent uuid"));
2824 	case BE_ERR_ZONE_MULTIPLE_ACTIVE:
2825 		return (gettext("Multiple active zone roots"));
2826 	case BE_ERR_ZONE_NO_ACTIVE_ROOT:
2827 		return (gettext("No active zone root"));
2828 	case BE_ERR_ZONE_ROOT_NOT_LEGACY:
2829 		return (gettext("Zone root not legacy"));
2830 	case BE_ERR_MOUNT_ZONEROOT:
2831 		return (gettext("Failed to mount a zone root."));
2832 	case BE_ERR_UMOUNT_ZONEROOT:
2833 		return (gettext("Failed to unmount a zone root."));
2834 	case BE_ERR_NO_MOUNTED_ZONE:
2835 		return (gettext("Zone is not mounted"));
2836 	case BE_ERR_ZONES_UNMOUNT:
2837 		return (gettext("Unable to unmount a zone BE."));
2838 	case BE_ERR_NO_MENU:
2839 		return (gettext("Missing boot menu file."));
2840 	case BE_ERR_BAD_MENU_PATH:
2841 		return (gettext("Invalid path for menu.lst file"));
2842 	case BE_ERR_ZONE_SS_EXISTS:
2843 		return (gettext("Zone snapshot exists."));
2844 	case BE_ERR_BOOTFILE_INST:
2845 		return (gettext("Error installing boot files."));
2846 	case BE_ERR_EXTCMD:
2847 		return (gettext("Error running an external command."));
2848 	default:
2849 		return (NULL);
2850 	}
2851 }
2852 
2853 /*
2854  * Function:    be_has_grub
2855  * Description: Boolean function indicating whether the current system
2856  *		uses grub.
2857  * Return:      B_FALSE - the system does not have grub
2858  *              B_TRUE - the system does have grub.
2859  * Scope:
2860  *		Semi-private (library wide use only)
2861  */
2862 boolean_t
2863 be_has_grub(void)
2864 {
2865 	/*
2866 	 * TODO: This will need to be expanded to check for the existence of
2867 	 * grub if and when there is grub support for SPARC.
2868 	 */
2869 	return (be_is_isa("i386"));
2870 }
2871 
2872 /*
2873  * Function:    be_is_isa
2874  * Description: Boolean function indicating whether the instruction set
2875  *              architecture of the executing system matches the name provided.
2876  *              The string must match a system defined architecture (e.g.
2877  *              "i386", "sparc") and is case sensitive.
2878  * Parameters:  name - string representing the name of instruction set
2879  *			architecture being tested
2880  * Returns:     B_FALSE - the system instruction set architecture is different
2881  *			from the one specified
2882  *              B_TRUE - the system instruction set architecture is the same
2883  *			as the one specified
2884  * Scope:
2885  *		Semi-private (library wide use only)
2886  */
2887 boolean_t
2888 be_is_isa(char *name)
2889 {
2890 	return ((strcmp((char *)be_get_default_isa(), name) == 0));
2891 }
2892 
2893 /*
2894  * Function: be_get_default_isa
2895  * Description:
2896  *      Returns the default instruction set architecture of the
2897  *      machine it is executed on. (eg. sparc, i386, ...)
2898  *      NOTE:   SYS_INST environment variable may override default
2899  *              return value
2900  * Parameters:
2901  *		none
2902  * Returns:
2903  *		NULL - the architecture returned by sysinfo() was too
2904  *			long for local variables
2905  *		char * - pointer to a string containing the default
2906  *			implementation
2907  * Scope:
2908  *		Semi-private (library wide use only)
2909  */
2910 char *
2911 be_get_default_isa(void)
2912 {
2913 	int	i;
2914 	char	*envp;
2915 	static char	default_inst[ARCH_LENGTH] = "";
2916 
2917 	if (default_inst[0] == '\0') {
2918 		if ((envp = getenv("SYS_INST")) != NULL) {
2919 			if ((int)strlen(envp) >= ARCH_LENGTH)
2920 				return (NULL);
2921 			else
2922 				(void) strcpy(default_inst, envp);
2923 		} else  {
2924 			i = sysinfo(SI_ARCHITECTURE, default_inst, ARCH_LENGTH);
2925 			if (i < 0 || i > ARCH_LENGTH)
2926 				return (NULL);
2927 		}
2928 	}
2929 	return (default_inst);
2930 }
2931 
2932 /*
2933  * Function: be_run_cmd
2934  * Description:
2935  *	Runs a command in a separate subprocess.  Splits out stdout from stderr
2936  *	and sends each to its own buffer.  Buffers must be pre-allocated and
2937  *	passed in as arguments.  Buffer sizes are also passed in as arguments.
2938  *
2939  *	Notes / caveats:
2940  *	- Command being run is assumed to not have any stdout or stderr
2941  *		redirection.
2942  *	- Commands which emit total stderr output of greater than PIPE_BUF
2943  *		bytes can hang.  For such commands, a different implementation
2944  *		which uses poll(2) must be used.
2945  *	- stdout_buf can be NULL.  In this case, stdout_bufsize is ignored, and
2946  *		the stream which would have gone to it is sent to the bit
2947  *		bucket.
2948  *	- stderr_buf cannot be NULL.
2949  *	- Only subprocess errors are appended to the stderr_buf.  Errors
2950  *		running the command are reported through be_print_err().
2951  *	- Data which would overflow its respective buffer is sent to the bit
2952  *		bucket.
2953  *
2954  * Parameters:
2955  *		command: command to run.  Assumed not to have embedded stdout
2956  *			or stderr redirection.  May have stdin redirection,
2957  *			however.
2958  *		stderr_buf: buffer returning subprocess stderr data.  Errors
2959  *			reported by this function are reported through
2960  *			be_print_err().
2961  *		stderr_bufsize: size of stderr_buf
2962  *		stdout_buf: buffer returning subprocess stdout data.
2963  *		stdout_bufsize: size of stdout_buf
2964  * Returns:
2965  *		BE_SUCCESS - The command ran successfully without returning
2966  *			errors.
2967  *		BE_ERR_EXTCMD
2968  *			- The command could not be run.
2969  *			- The command terminated with error status.
2970  *			- There were errors extracting or returning subprocess
2971  *				data.
2972  *		BE_ERR_NOMEM - The command exceeds the command buffer size.
2973  *		BE_ERR_INVAL - An invalid argument was specified.
2974  * Scope:
2975  *		Semi-private (library wide use only)
2976  */
2977 int
2978 be_run_cmd(char *command, char *stderr_buf, int stderr_bufsize,
2979     char *stdout_buf, int stdout_bufsize)
2980 {
2981 	char *temp_filename = strdup(tmpnam(NULL));
2982 	FILE *stdout_str = NULL;
2983 	FILE *stderr_str = NULL;
2984 	char cmdline[BUFSIZ];
2985 	char oneline[BUFSIZ];
2986 	int exit_status;
2987 	int rval = BE_SUCCESS;
2988 
2989 	if ((command == NULL) || (stderr_buf == NULL) ||
2990 	    (stderr_bufsize <= 0) || (stdout_bufsize <  0) ||
2991 	    ((stdout_buf != NULL) ^ (stdout_bufsize != 0))) {
2992 		return (BE_ERR_INVAL);
2993 }
2994 
2995 	/* Set up command so popen returns stderr, not stdout */
2996 	if (snprintf(cmdline, BUFSIZ, "%s 2> %s", command,
2997 	    temp_filename) >= BUFSIZ) {
2998 		rval = BE_ERR_NOMEM;
2999 		goto cleanup;
3000 	}
3001 
3002 	/* Set up the fifo that will make stderr available. */
3003 	if (mkfifo(temp_filename, 0600) != 0) {
3004 		(void) be_print_err(gettext("be_run_cmd: mkfifo: %s\n"),
3005 		    strerror(errno));
3006 		rval = BE_ERR_EXTCMD;
3007 		goto cleanup;
3008 	}
3009 
3010 	if ((stdout_str = popen(cmdline, "r")) == NULL) {
3011 		(void) be_print_err(gettext("be_run_cmd: popen: %s\n"),
3012 		    strerror(errno));
3013 		rval = BE_ERR_EXTCMD;
3014 		goto cleanup;
3015 	}
3016 
3017 	if ((stderr_str = fopen(temp_filename, "r")) == NULL) {
3018 		(void) be_print_err(gettext("be_run_cmd: fopen: %s\n"),
3019 		    strerror(errno));
3020 		(void) pclose(stdout_str);
3021 		rval = BE_ERR_EXTCMD;
3022 		goto cleanup;
3023 	}
3024 
3025 	/* Read stdout first, as it usually outputs more than stderr. */
3026 	oneline[BUFSIZ-1] = '\0';
3027 	while (fgets(oneline, BUFSIZ-1, stdout_str) != NULL) {
3028 		if (stdout_str != NULL) {
3029 			(void) strlcat(stdout_buf, oneline, stdout_bufsize);
3030 		}
3031 	}
3032 
3033 	while (fgets(oneline, BUFSIZ-1, stderr_str) != NULL) {
3034 		(void) strlcat(stderr_buf, oneline, stderr_bufsize);
3035 	}
3036 
3037 	/* Close pipe, get exit status. */
3038 	if ((exit_status = pclose(stdout_str)) == -1) {
3039 		(void) be_print_err(gettext("be_run_cmd: pclose: %s\n"),
3040 		    strerror(errno));
3041 		rval = BE_ERR_EXTCMD;
3042 	} else if (WIFEXITED(exit_status)) {
3043 		exit_status = (int)((char)WEXITSTATUS(exit_status));
3044 		if (exit_status != 0) {
3045 			(void) snprintf(oneline, BUFSIZ, gettext("be_run_cmd: "
3046 			    "command terminated with error status: %d\n"),
3047 			    exit_status);
3048 			(void) strlcat(stderr_buf, oneline, stderr_bufsize);
3049 			rval = BE_ERR_EXTCMD;
3050 		}
3051 	} else {
3052 		(void) snprintf(oneline, BUFSIZ, gettext("be_run_cmd: command "
3053 		    "terminated on signal: %s\n"),
3054 		    strsignal(WTERMSIG(exit_status)));
3055 		(void) strlcat(stderr_buf, oneline, stderr_bufsize);
3056 		rval = BE_ERR_EXTCMD;
3057 	}
3058 
3059 cleanup:
3060 	(void) unlink(temp_filename);
3061 	(void) free(temp_filename);
3062 
3063 	return (rval);
3064 }
3065 
3066 /* ********************************************************************	*/
3067 /*			Private Functions				*/
3068 /* ******************************************************************** */
3069 
3070 /*
3071  * Function:	update_dataset
3072  * Description:	This function takes a dataset name and replaces the zpool
3073  *		and be_name components of the dataset with the new be_name
3074  *		zpool passed in.
3075  * Parameters:
3076  *		dataset - name of dataset
3077  *		dataset_len - lenth of buffer in which dataset is passed in.
3078  *		be_name - name of new BE name to update to.
3079  *		old_rc_loc - dataset under which the root container dataset
3080  *			for the old BE lives.
3081  *		new_rc_loc - dataset under which the root container dataset
3082  *			for the new BE lives.
3083  * Returns:
3084  *		BE_SUCCESS - Success
3085  *		be_errno_t - Failure
3086  * Scope:
3087  *		Private
3088  */
3089 static int
3090 update_dataset(char *dataset, int dataset_len, char *be_name,
3091     char *old_rc_loc, char *new_rc_loc)
3092 {
3093 	char	*ds = NULL;
3094 	char	*sub_ds = NULL;
3095 
3096 	/* Tear off the BE container dataset */
3097 	if ((ds = be_make_name_from_ds(dataset, old_rc_loc)) == NULL) {
3098 		return (BE_ERR_INVAL);
3099 	}
3100 
3101 	/* Get dataset name relative to BE root, if there is one */
3102 	sub_ds = strchr(ds, '/');
3103 
3104 	/* Generate the BE root dataset name */
3105 	be_make_root_ds(new_rc_loc, be_name, dataset, dataset_len);
3106 
3107 	/* If a subordinate dataset name was found, append it */
3108 	if (sub_ds != NULL)
3109 		(void) strlcat(dataset, sub_ds, dataset_len);
3110 
3111 	free(ds);
3112 	return (BE_SUCCESS);
3113 }
3114 
3115 /*
3116  * Function:	_update_vfstab
3117  * Description:	This function updates a vfstab file to reflect the new
3118  *		root container dataset location and be_name for all
3119  *		entries listed in the be_fs_list_data_t structure passed in.
3120  * Parameters:
3121  *		vfstab - vfstab file to modify
3122  *		be_name - name of BE to update.
3123  *		old_rc_loc - dataset under which the root container dataset
3124  *			of the old BE resides in.
3125  *		new_rc_loc - dataset under which the root container dataset
3126  *			of the new BE resides in.
3127  *		fld - be_fs_list_data_t pointer providing the list of
3128  *			file systems to look for in vfstab.
3129  * Returns:
3130  *		BE_SUCCESS - Success
3131  *		be_errno_t - Failure
3132  * Scope:
3133  *		Private
3134  */
3135 static int
3136 _update_vfstab(char *vfstab, char *be_name, char *old_rc_loc,
3137     char *new_rc_loc, be_fs_list_data_t *fld)
3138 {
3139 	struct vfstab	vp;
3140 	char		*tmp_vfstab = NULL;
3141 	char		comments_buf[BUFSIZ];
3142 	FILE		*comments = NULL;
3143 	FILE		*vfs_ents = NULL;
3144 	FILE		*tfile = NULL;
3145 	struct stat	sb;
3146 	char		dev[MAXPATHLEN];
3147 	char		*c;
3148 	int		fd;
3149 	int		ret = BE_SUCCESS, err = 0;
3150 	int		i;
3151 	int		tmp_vfstab_len = 0;
3152 
3153 	errno = 0;
3154 
3155 	/*
3156 	 * Open vfstab for reading twice.  First is for comments,
3157 	 * second is for actual entries.
3158 	 */
3159 	if ((comments = fopen(vfstab, "r")) == NULL ||
3160 	    (vfs_ents = fopen(vfstab, "r")) == NULL) {
3161 		err = errno;
3162 		be_print_err(gettext("_update_vfstab: "
3163 		    "failed to open vfstab (%s): %s\n"), vfstab,
3164 		    strerror(err));
3165 		ret = errno_to_be_err(err);
3166 		goto cleanup;
3167 	}
3168 
3169 	/* Grab the stats of the original vfstab file */
3170 	if (stat(vfstab, &sb) != 0) {
3171 		err = errno;
3172 		be_print_err(gettext("_update_vfstab: "
3173 		    "failed to stat file %s: %s\n"), vfstab,
3174 		    strerror(err));
3175 		ret = errno_to_be_err(err);
3176 		goto cleanup;
3177 	}
3178 
3179 	/* Create tmp file for modified vfstab */
3180 	if ((tmp_vfstab = (char *)malloc(strlen(vfstab) + 7))
3181 	    == NULL) {
3182 		be_print_err(gettext("_update_vfstab: "
3183 		    "malloc failed\n"));
3184 		ret = BE_ERR_NOMEM;
3185 		goto cleanup;
3186 	}
3187 	tmp_vfstab_len = strlen(vfstab) + 7;
3188 	(void) memset(tmp_vfstab, 0, tmp_vfstab_len);
3189 	(void) strlcpy(tmp_vfstab, vfstab, tmp_vfstab_len);
3190 	(void) strlcat(tmp_vfstab, "XXXXXX", tmp_vfstab_len);
3191 	if ((fd = mkstemp(tmp_vfstab)) == -1) {
3192 		err = errno;
3193 		be_print_err(gettext("_update_vfstab: "
3194 		    "mkstemp failed: %s\n"), strerror(err));
3195 		ret = errno_to_be_err(err);
3196 		goto cleanup;
3197 	}
3198 	if ((tfile = fdopen(fd, "w")) == NULL) {
3199 		err = errno;
3200 		be_print_err(gettext("_update_vfstab: "
3201 		    "could not open file for write\n"));
3202 		(void) close(fd);
3203 		ret = errno_to_be_err(err);
3204 		goto cleanup;
3205 	}
3206 
3207 	while (fgets(comments_buf, BUFSIZ, comments)) {
3208 		for (c = comments_buf; *c != '\0' && isspace(*c); c++)
3209 			;
3210 		if (*c == '\0') {
3211 			continue;
3212 		} else if (*c == '#') {
3213 			/*
3214 			 * If line is a comment line, just put
3215 			 * it through to the tmp vfstab.
3216 			 */
3217 			(void) fputs(comments_buf, tfile);
3218 		} else {
3219 			/*
3220 			 * Else line is a vfstab entry, grab it
3221 			 * into a vfstab struct.
3222 			 */
3223 			if (getvfsent(vfs_ents, &vp) != 0) {
3224 				err = errno;
3225 				be_print_err(gettext("_update_vfstab: "
3226 				    "getvfsent failed: %s\n"), strerror(err));
3227 				ret = errno_to_be_err(err);
3228 				goto cleanup;
3229 			}
3230 
3231 			if (vp.vfs_special == NULL || vp.vfs_mountp == NULL) {
3232 				(void) putvfsent(tfile, &vp);
3233 				continue;
3234 			}
3235 
3236 			/*
3237 			 * If the entry is one of the entries in the list
3238 			 * of file systems to update, modify it's device
3239 			 * field to be correct for this BE.
3240 			 */
3241 			for (i = 0; i < fld->fs_num; i++) {
3242 				if (strcmp(vp.vfs_special, fld->fs_list[i])
3243 				    == 0) {
3244 					/*
3245 					 * Found entry that needs an update.
3246 					 * Replace the root container dataset
3247 					 * location and be_name in the
3248 					 * entry's device.
3249 					 */
3250 					(void) strlcpy(dev, vp.vfs_special,
3251 					    sizeof (dev));
3252 
3253 					if ((ret = update_dataset(dev,
3254 					    sizeof (dev), be_name, old_rc_loc,
3255 					    new_rc_loc)) != 0) {
3256 						be_print_err(
3257 						    gettext("_update_vfstab: "
3258 						    "Failed to update device "
3259 						    "field for vfstab entry "
3260 						    "%s\n"), fld->fs_list[i]);
3261 						goto cleanup;
3262 					}
3263 
3264 					vp.vfs_special = dev;
3265 					break;
3266 				}
3267 			}
3268 
3269 			/* Put entry through to tmp vfstab */
3270 			(void) putvfsent(tfile, &vp);
3271 		}
3272 	}
3273 
3274 	(void) fclose(comments);
3275 	comments = NULL;
3276 	(void) fclose(vfs_ents);
3277 	vfs_ents = NULL;
3278 	(void) fclose(tfile);
3279 	tfile = NULL;
3280 
3281 	/* Copy tmp vfstab into place */
3282 	if (rename(tmp_vfstab, vfstab) != 0) {
3283 		err = errno;
3284 		be_print_err(gettext("_update_vfstab: "
3285 		    "failed to rename file %s to %s: %s\n"), tmp_vfstab,
3286 		    vfstab, strerror(err));
3287 		ret = errno_to_be_err(err);
3288 		goto cleanup;
3289 	}
3290 
3291 	/* Set the perms and ownership of the updated file */
3292 	if (chmod(vfstab, sb.st_mode) != 0) {
3293 		err = errno;
3294 		be_print_err(gettext("_update_vfstab: "
3295 		    "failed to chmod %s: %s\n"), vfstab, strerror(err));
3296 		ret = errno_to_be_err(err);
3297 		goto cleanup;
3298 	}
3299 	if (chown(vfstab, sb.st_uid, sb.st_gid) != 0) {
3300 		err = errno;
3301 		be_print_err(gettext("_update_vfstab: "
3302 		    "failed to chown %s: %s\n"), vfstab, strerror(err));
3303 		ret = errno_to_be_err(err);
3304 		goto cleanup;
3305 	}
3306 
3307 cleanup:
3308 	if (comments != NULL)
3309 		(void) fclose(comments);
3310 	if (vfs_ents != NULL)
3311 		(void) fclose(vfs_ents);
3312 	(void) unlink(tmp_vfstab);
3313 	(void) free(tmp_vfstab);
3314 	if (tfile != NULL)
3315 		(void) fclose(tfile);
3316 
3317 	return (ret);
3318 }
3319 
3320 
3321 /*
3322  * Function:	be_get_auto_name
3323  * Description:	Generate an auto name constructed based on the BE name
3324  *		of the original BE or zone BE being cloned.
3325  * Parameters:
3326  *		obe_name - name of the original BE or zone BE being cloned.
3327  *              container_ds - container dataset for the zone.
3328  *                             Note: if zone_be is false this should be
3329  *                                  NULL.
3330  *		zone_be - flag that indicates if we are operating on a zone BE.
3331  * Returns:
3332  *		Success - pointer to auto generated BE name.  The name
3333  *			is allocated in heap storage so the caller is
3334  *			responsible for free'ing the name.
3335  *		Failure - NULL
3336  * Scope:
3337  *		Private
3338  */
3339 static char *
3340 be_get_auto_name(char *obe_name, char *be_container_ds, boolean_t zone_be)
3341 {
3342 	be_node_list_t	*be_nodes = NULL;
3343 	be_node_list_t	*cur_be = NULL;
3344 	char		auto_be_name[MAXPATHLEN];
3345 	char		base_be_name[MAXPATHLEN];
3346 	char		cur_be_name[MAXPATHLEN];
3347 	char		*num_str = NULL;
3348 	char		*c = NULL;
3349 	int		num = 0;
3350 	int		cur_num = 0;
3351 
3352 	errno = 0;
3353 
3354 	/*
3355 	 * Check if obe_name is already in an auto BE name format.
3356 	 * If it is, then strip off the increment number to get the
3357 	 * base name.
3358 	 */
3359 	(void) strlcpy(base_be_name, obe_name, sizeof (base_be_name));
3360 
3361 	if ((num_str = strrchr(base_be_name, BE_AUTO_NAME_DELIM))
3362 	    != NULL) {
3363 		/* Make sure remaining string is all digits */
3364 		c = num_str + 1;
3365 		while (c[0] != '\0' && isdigit(c[0]))
3366 			c++;
3367 		/*
3368 		 * If we're now at the end of the string strip off the
3369 		 * increment number.
3370 		 */
3371 		if (c[0] == '\0')
3372 			num_str[0] = '\0';
3373 	}
3374 
3375 	if (zone_be) {
3376 		if (be_container_ds == NULL)
3377 			return (NULL);
3378 		if (be_get_zone_be_list(obe_name, be_container_ds,
3379 		    &be_nodes) != BE_SUCCESS) {
3380 			be_print_err(gettext("be_get_auto_name: "
3381 			    "be_get_zone_be_list failed\n"));
3382 			return (NULL);
3383 		}
3384 	} else if (_be_list(NULL, &be_nodes) != BE_SUCCESS) {
3385 		be_print_err(gettext("be_get_auto_name: be_list failed\n"));
3386 		return (NULL);
3387 	}
3388 
3389 	for (cur_be = be_nodes; cur_be != NULL; cur_be = cur_be->be_next_node) {
3390 		(void) strlcpy(cur_be_name, cur_be->be_node_name,
3391 		    sizeof (cur_be_name));
3392 
3393 		/* If cur_be_name doesn't match at least base be name, skip. */
3394 		if (strncmp(cur_be_name, base_be_name, strlen(base_be_name))
3395 		    != 0)
3396 			continue;
3397 
3398 		/* Get the string following the base be name */
3399 		num_str = cur_be_name + strlen(base_be_name);
3400 
3401 		/*
3402 		 * If nothing follows the base be name, this cur_be_name
3403 		 * is the BE named with the base be name, skip.
3404 		 */
3405 		if (num_str == NULL || num_str[0] == '\0')
3406 			continue;
3407 
3408 		/*
3409 		 * Remove the name delimiter.  If its not there,
3410 		 * cur_be_name isn't part of this BE name stream, skip.
3411 		 */
3412 		if (num_str[0] == BE_AUTO_NAME_DELIM)
3413 			num_str++;
3414 		else
3415 			continue;
3416 
3417 		/* Make sure remaining string is all digits */
3418 		c = num_str;
3419 		while (c[0] != '\0' && isdigit(c[0]))
3420 			c++;
3421 		if (c[0] != '\0')
3422 			continue;
3423 
3424 		/* Convert the number string to an int */
3425 		cur_num = atoi(num_str);
3426 
3427 		/*
3428 		 * If failed to convert the string, skip it.  If its too
3429 		 * long to be converted to an int, we wouldn't auto generate
3430 		 * this number anyway so there couldn't be a conflict.
3431 		 * We treat it as a manually created BE name.
3432 		 */
3433 		if (cur_num == 0 && errno == EINVAL)
3434 			continue;
3435 
3436 		/*
3437 		 * Compare current number to current max number,
3438 		 * take higher of the two.
3439 		 */
3440 		if (cur_num > num)
3441 			num = cur_num;
3442 	}
3443 
3444 	/*
3445 	 * Store off a copy of 'num' incase we need it later.  If incrementing
3446 	 * 'num' causes it to roll over, this means 'num' is the largest
3447 	 * positive int possible; we'll need it later in the loop to determine
3448 	 * if we've exhausted all possible increment numbers.  We store it in
3449 	 * 'cur_num'.
3450 	 */
3451 	cur_num = num;
3452 
3453 	/* Increment 'num' to get new auto BE name number */
3454 	if (++num <= 0) {
3455 		int ret = 0;
3456 
3457 		/*
3458 		 * Since incrementing 'num' caused it to rollover, start
3459 		 * over at 0 and find the first available number.
3460 		 */
3461 		for (num = 0; num < cur_num; num++) {
3462 
3463 			(void) snprintf(cur_be_name, sizeof (cur_be_name),
3464 			    "%s%c%d", base_be_name, BE_AUTO_NAME_DELIM, num);
3465 
3466 			ret = zpool_iter(g_zfs, be_exists_callback,
3467 			    cur_be_name);
3468 
3469 			if (ret == 0) {
3470 				/*
3471 				 * BE name doesn't exist, break out
3472 				 * to use 'num'.
3473 				 */
3474 				break;
3475 			} else if (ret == 1) {
3476 				/* BE name exists, continue looking */
3477 				continue;
3478 			} else {
3479 				be_print_err(gettext("be_get_auto_name: "
3480 				    "zpool_iter failed: %s\n"),
3481 				    libzfs_error_description(g_zfs));
3482 				be_free_list(be_nodes);
3483 				return (NULL);
3484 			}
3485 		}
3486 
3487 		/*
3488 		 * If 'num' equals 'cur_num', we've exhausted all possible
3489 		 * auto BE names for this base BE name.
3490 		 */
3491 		if (num == cur_num) {
3492 			be_print_err(gettext("be_get_auto_name: "
3493 			    "No more available auto BE names for base "
3494 			    "BE name %s\n"), base_be_name);
3495 			be_free_list(be_nodes);
3496 			return (NULL);
3497 		}
3498 	}
3499 
3500 	be_free_list(be_nodes);
3501 
3502 	/*
3503 	 * Generate string for auto BE name.
3504 	 */
3505 	(void) snprintf(auto_be_name, sizeof (auto_be_name), "%s%c%d",
3506 	    base_be_name, BE_AUTO_NAME_DELIM, num);
3507 
3508 	if ((c = strdup(auto_be_name)) == NULL) {
3509 		be_print_err(gettext("be_get_auto_name: "
3510 		    "memory allocation failed\n"));
3511 		return (NULL);
3512 	}
3513 
3514 	return (c);
3515 }
3516 
3517 /*
3518  * Function:	be_get_console_prop
3519  * Description:	Determine console device.
3520  * Returns:
3521  *		Success - pointer to console setting.
3522  *		Failure - NULL
3523  * Scope:
3524  *		Private
3525  */
3526 static char *
3527 be_get_console_prop(void)
3528 {
3529 	di_node_t	dn;
3530 	char *console = NULL;
3531 
3532 	if ((dn = di_init("/", DINFOPROP)) == DI_NODE_NIL) {
3533 		be_print_err(gettext("be_get_console_prop: "
3534 		    "di_init() failed\n"));
3535 		return (NULL);
3536 	}
3537 
3538 	if (di_prop_lookup_strings(DDI_DEV_T_ANY, dn,
3539 	    "console", &console) != -1) {
3540 		di_fini(dn);
3541 		return (console);
3542 	}
3543 
3544 	if (console == NULL) {
3545 		if (di_prop_lookup_strings(DDI_DEV_T_ANY, dn,
3546 		    "output-device", &console) != -1) {
3547 			di_fini(dn);
3548 			if (strncmp(console, "screen", strlen("screen")) == 0)
3549 				console = BE_DEFAULT_CONSOLE;
3550 		}
3551 	}
3552 
3553 	/*
3554 	 * Default console to text
3555 	 */
3556 	if (console == NULL) {
3557 		console = BE_DEFAULT_CONSOLE;
3558 	}
3559 
3560 	return (console);
3561 }
3562 
3563 /*
3564  * Function:	be_create_menu
3565  * Description:
3566  *		This function is used if no menu.lst file exists. In
3567  *		this case a new file is created and if needed default
3568  *		lines are added to the file.
3569  * Parameters:
3570  *		pool - The name of the pool the menu.lst file is on
3571  *		menu_file - The name of the file we're creating.
3572  *		menu_fp - A pointer to the file pointer of the file we
3573  *			  created. This is also used to pass back the file
3574  *			  pointer to the newly created file.
3575  *		mode - the original mode used for the failed attempt to
3576  *		       non-existent file.
3577  * Returns:
3578  *		BE_SUCCESS - Success
3579  *		be_errno_t - Failure
3580  * Scope:
3581  *		Private
3582  */
3583 static int
3584 be_create_menu(
3585 	char *pool,
3586 	char *menu_file,
3587 	FILE **menu_fp,
3588 	char *mode)
3589 {
3590 	be_node_list_t	*be_nodes = NULL;
3591 	char *menu_path = NULL;
3592 	char *be_rpool = NULL;
3593 	char *be_name = NULL;
3594 	char *console = NULL;
3595 	errno = 0;
3596 
3597 	if (menu_file == NULL || menu_fp == NULL || mode == NULL)
3598 		return (BE_ERR_INVAL);
3599 
3600 	menu_path = strdup(menu_file);
3601 	if (menu_path == NULL)
3602 		return (BE_ERR_NOMEM);
3603 
3604 	(void) dirname(menu_path);
3605 	if (*menu_path == '.') {
3606 		free(menu_path);
3607 		return (BE_ERR_BAD_MENU_PATH);
3608 	}
3609 	if (mkdirp(menu_path,
3610 	    S_IRWXU | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH) == -1 &&
3611 	    errno != EEXIST) {
3612 		free(menu_path);
3613 		be_print_err(gettext("be_create_menu: Failed to create the %s "
3614 		    "directory: %s\n"), menu_path, strerror(errno));
3615 		return (errno_to_be_err(errno));
3616 	}
3617 	free(menu_path);
3618 
3619 	/*
3620 	 * Check to see if this system supports grub
3621 	 */
3622 	if (be_has_grub()) {
3623 		/*
3624 		 * The grub menu is missing so we need to create it
3625 		 * and fill in the first few lines.
3626 		 */
3627 		FILE *temp_fp = fopen(menu_file, "a+");
3628 		if (temp_fp == NULL) {
3629 			*menu_fp = NULL;
3630 			return (errno_to_be_err(errno));
3631 		}
3632 
3633 		if ((console = be_get_console_prop()) != NULL) {
3634 
3635 			/*
3636 			 * If console is redirected to serial line,
3637 			 * GRUB splash screen will not be enabled.
3638 			 */
3639 			if (strncmp(console, "text", strlen("text")) == 0 ||
3640 			    strncmp(console, "graphics",
3641 			    strlen("graphics")) == 0) {
3642 
3643 				(void) fprintf(temp_fp, "%s\n", BE_GRUB_SPLASH);
3644 				(void) fprintf(temp_fp, "%s\n",
3645 				    BE_GRUB_FOREGROUND);
3646 				(void) fprintf(temp_fp, "%s\n",
3647 				    BE_GRUB_BACKGROUND);
3648 				(void) fprintf(temp_fp, "%s\n",
3649 				    BE_GRUB_DEFAULT);
3650 			} else {
3651 				be_print_err(gettext("be_create_menu: "
3652 				    "console on serial line, "
3653 				    "GRUB splash image will be disabled\n"));
3654 			}
3655 		}
3656 
3657 		(void) fprintf(temp_fp,	"timeout 30\n");
3658 		(void) fclose(temp_fp);
3659 
3660 	} else {
3661 		/*
3662 		 * The menu file doesn't exist so we need to create a
3663 		 * blank file.
3664 		 */
3665 		FILE *temp_fp = fopen(menu_file, "w+");
3666 		if (temp_fp == NULL) {
3667 			*menu_fp = NULL;
3668 			return (errno_to_be_err(errno));
3669 		}
3670 		(void) fclose(temp_fp);
3671 	}
3672 
3673 	/*
3674 	 * Now we need to add all the BE's back into the the file.
3675 	 */
3676 	if (_be_list(NULL, &be_nodes) == BE_SUCCESS) {
3677 		while (be_nodes != NULL) {
3678 			if (strcmp(pool, be_nodes->be_rpool) == 0) {
3679 				(void) be_append_menu(be_nodes->be_node_name,
3680 				    be_nodes->be_rpool, NULL, NULL, NULL);
3681 			}
3682 			if (be_nodes->be_active_on_boot) {
3683 				be_rpool = strdup(be_nodes->be_rpool);
3684 				be_name = strdup(be_nodes->be_node_name);
3685 			}
3686 
3687 			be_nodes = be_nodes->be_next_node;
3688 		}
3689 	}
3690 	be_free_list(be_nodes);
3691 
3692 	/*
3693 	 * Check to see if this system supports grub
3694 	 */
3695 	if (be_has_grub()) {
3696 		int err = be_change_grub_default(be_name, be_rpool);
3697 		if (err != BE_SUCCESS)
3698 			return (err);
3699 	}
3700 	*menu_fp = fopen(menu_file, mode);
3701 	if (*menu_fp == NULL)
3702 		return (errno_to_be_err(errno));
3703 
3704 	return (BE_SUCCESS);
3705 }
3706 
3707 /*
3708  * Function:	be_open_menu
3709  * Description:
3710  *		This function is used it open the menu.lst file. If this
3711  *              file does not exist be_create_menu is called to create it
3712  *              and the open file pointer is returned. If the file does
3713  *              exist it is simply opened using the mode passed in.
3714  * Parameters:
3715  *		pool - The name of the pool the menu.lst file is on
3716  *		menu_file - The name of the file we're opening.
3717  *		menu_fp - A pointer to the file pointer of the file we're
3718  *			  opening. This is also used to pass back the file
3719  *			  pointer.
3720  *		mode - the original mode to be used for opening the menu.lst
3721  *                     file.
3722  *              create_menu - If this is true and the menu.lst file does not
3723  *                            exist we will attempt to re-create it. However
3724  *                            if it's false the error returned from the fopen
3725  *                            will be returned.
3726  * Returns:
3727  *		BE_SUCCESS - Success
3728  *		be_errno_t - Failure
3729  * Scope:
3730  *		Private
3731  */
3732 static int
3733 be_open_menu(
3734 	char *pool,
3735 	char *menu_file,
3736 	FILE **menu_fp,
3737 	char *mode,
3738 	boolean_t create_menu)
3739 {
3740 	int	err = 0;
3741 	boolean_t	set_print = B_FALSE;
3742 
3743 	*menu_fp = fopen(menu_file, mode);
3744 	err = errno;
3745 	if (*menu_fp == NULL) {
3746 		if (err == ENOENT && create_menu) {
3747 			be_print_err(gettext("be_open_menu: menu.lst "
3748 			    "file %s does not exist,\n"), menu_file);
3749 			if (!do_print) {
3750 				set_print = B_TRUE;
3751 				do_print = B_TRUE;
3752 			}
3753 			be_print_err(gettext("WARNING: menu.lst "
3754 			    "file %s does not exist,\n         generating "
3755 			    "a new menu.lst file\n"), menu_file);
3756 			if (set_print)
3757 				do_print = B_FALSE;
3758 			err = 0;
3759 			if ((err = be_create_menu(pool, menu_file,
3760 			    menu_fp, mode)) == ENOENT)
3761 				return (BE_ERR_NO_MENU);
3762 			else if (err != BE_SUCCESS)
3763 				return (err);
3764 			else if (*menu_fp == NULL)
3765 				return (BE_ERR_NO_MENU);
3766 		} else {
3767 			be_print_err(gettext("be_open_menu: failed "
3768 			    "to open menu.lst file %s\n"), menu_file);
3769 			if (err == ENOENT)
3770 				return (BE_ERR_NO_MENU);
3771 			else
3772 				return (errno_to_be_err(err));
3773 		}
3774 	}
3775 	return (BE_SUCCESS);
3776 }
3777