xref: /illumos-gate/usr/src/cmd/boot/bootadm/bootadm.c (revision 34349410540191e0eed7567bd5f2f365c7f705f2)
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  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 /*
27  * bootadm(1M) is a new utility for managing bootability of
28  * Solaris *Newboot* environments. It has two primary tasks:
29  * 	- Allow end users to manage bootability of Newboot Solaris instances
30  *	- Provide services to other subsystems in Solaris (primarily Install)
31  */
32 
33 /* Headers */
34 #include <stdio.h>
35 #include <errno.h>
36 #include <stdlib.h>
37 #include <string.h>
38 #include <unistd.h>
39 #include <sys/types.h>
40 #include <sys/stat.h>
41 #include <stdarg.h>
42 #include <limits.h>
43 #include <signal.h>
44 #include <sys/wait.h>
45 #include <sys/mnttab.h>
46 #include <sys/mntent.h>
47 #include <sys/statvfs.h>
48 #include <libnvpair.h>
49 #include <ftw.h>
50 #include <fcntl.h>
51 #include <strings.h>
52 #include <utime.h>
53 #include <sys/systeminfo.h>
54 #include <sys/dktp/fdisk.h>
55 #include <sys/param.h>
56 #include <dirent.h>
57 #include <ctype.h>
58 #include <libgen.h>
59 #include <sys/sysmacros.h>
60 #include <sys/elf.h>
61 #include <libscf.h>
62 #include <zlib.h>
63 #include <sys/lockfs.h>
64 #include <sys/filio.h>
65 
66 #if !defined(_OPB)
67 #include <sys/ucode.h>
68 #endif
69 
70 #include <pwd.h>
71 #include <grp.h>
72 #include <device_info.h>
73 #include <sys/vtoc.h>
74 #include <sys/efi_partition.h>
75 
76 #include <locale.h>
77 
78 #include "message.h"
79 #include "bootadm.h"
80 
81 #ifndef TEXT_DOMAIN
82 #define	TEXT_DOMAIN	"SUNW_OST_OSCMD"
83 #endif	/* TEXT_DOMAIN */
84 
85 /* Type definitions */
86 
87 /* Primary subcmds */
88 typedef enum {
89 	BAM_MENU = 3,
90 	BAM_ARCHIVE
91 } subcmd_t;
92 
93 typedef enum {
94     OPT_ABSENT = 0,	/* No option */
95     OPT_REQ,		/* option required */
96     OPT_OPTIONAL	/* option may or may not be present */
97 } option_t;
98 
99 typedef struct {
100 	char	*subcmd;
101 	option_t option;
102 	error_t (*handler)();
103 	int	unpriv;			/* is this an unprivileged command */
104 } subcmd_defn_t;
105 
106 #define	LINE_INIT	0	/* lineNum initial value */
107 #define	ENTRY_INIT	-1	/* entryNum initial value */
108 #define	ALL_ENTRIES	-2	/* selects all boot entries */
109 
110 #define	GRUB_DIR		"/boot/grub"
111 #define	GRUB_STAGE2		GRUB_DIR "/stage2"
112 #define	GRUB_MENU		"/boot/grub/menu.lst"
113 #define	MENU_TMP		"/boot/grub/menu.lst.tmp"
114 #define	GRUB_BACKUP_MENU	"/etc/lu/GRUB_backup_menu"
115 #define	RAMDISK_SPECIAL		"/ramdisk"
116 #define	STUBBOOT		"/stubboot"
117 #define	MULTIBOOT		"/platform/i86pc/multiboot"
118 #define	GRUBSIGN_DIR		"/boot/grub/bootsign"
119 #define	GRUBSIGN_BACKUP		"/etc/bootsign"
120 #define	GRUBSIGN_UFS_PREFIX	"rootfs"
121 #define	GRUBSIGN_ZFS_PREFIX	"pool_"
122 #define	GRUBSIGN_LU_PREFIX	"BE_"
123 #define	UFS_SIGNATURE_LIST	"/var/run/grub_ufs_signatures"
124 #define	ZFS_LEGACY_MNTPT	"/tmp/bootadm_mnt_zfs_legacy"
125 
126 #define	BOOTADM_RDONLY_TEST	"BOOTADM_RDONLY_TEST"
127 
128 /* lock related */
129 #define	BAM_LOCK_FILE		"/var/run/bootadm.lock"
130 #define	LOCK_FILE_PERMS		(S_IRUSR|S_IWUSR|S_IRGRP|S_IROTH)
131 
132 #define	CREATE_RAMDISK		"boot/solaris/bin/create_ramdisk"
133 #define	CREATE_DISKMAP		"boot/solaris/bin/create_diskmap"
134 #define	EXTRACT_BOOT_FILELIST	"boot/solaris/bin/extract_boot_filelist"
135 #define	GRUBDISK_MAP		"/var/run/solaris_grubdisk.map"
136 
137 #define	GRUB_slice		"/etc/lu/GRUB_slice"
138 #define	GRUB_root		"/etc/lu/GRUB_root"
139 #define	GRUB_fdisk		"/etc/lu/GRUB_fdisk"
140 #define	GRUB_fdisk_target	"/etc/lu/GRUB_fdisk_target"
141 #define	FINDROOT_INSTALLGRUB	"/etc/lu/installgrub.findroot"
142 #define	LULIB			"/usr/lib/lu/lulib"
143 #define	LULIB_PROPAGATE_FILE	"lulib_propagate_file"
144 #define	CKSUM			"/usr/bin/cksum"
145 #define	LU_MENU_CKSUM		"/etc/lu/menu.cksum"
146 #define	BOOTADM			"/sbin/bootadm"
147 
148 #define	INSTALLGRUB		"/sbin/installgrub"
149 #define	STAGE1			"/boot/grub/stage1"
150 #define	STAGE2			"/boot/grub/stage2"
151 
152 typedef enum zfs_mnted {
153 	ZFS_MNT_ERROR = -1,
154 	LEGACY_MOUNTED = 1,
155 	LEGACY_ALREADY,
156 	ZFS_MOUNTED,
157 	ZFS_ALREADY
158 } zfs_mnted_t;
159 
160 /*
161  * The following two defines are used to detect and create the correct
162  * boot archive  when safemode patching is underway.  LOFS_PATCH_FILE is a
163  * contracted private interface between bootadm and the install
164  * consolidation.  It is set by pdo.c when a patch with SUNW_PATCH_SAFEMODE
165  * is applied.
166  */
167 #define	LOFS_PATCH_FILE		"/var/run/.patch_loopback_mode"
168 #define	LOFS_PATCH_MNT		"/var/run/.patch_root_loopbackmnt"
169 
170 /*
171  * Default file attributes
172  */
173 #define	DEFAULT_DEV_MODE	0644	/* default permissions */
174 #define	DEFAULT_DEV_UID		0	/* user root */
175 #define	DEFAULT_DEV_GID		3	/* group sys */
176 
177 /*
178  * Menu related
179  * menu_cmd_t and menu_cmds must be kept in sync
180  */
181 char *menu_cmds[] = {
182 	"default",	/* DEFAULT_CMD */
183 	"timeout",	/* TIMEOUT_CMD */
184 	"title",	/* TITLE_CMD */
185 	"root",		/* ROOT_CMD */
186 	"kernel",	/* KERNEL_CMD */
187 	"kernel$",	/* KERNEL_DOLLAR_CMD */
188 	"module",	/* MODULE_CMD */
189 	"module$",	/* MODULE_DOLLAR_CMD */
190 	" ",		/* SEP_CMD */
191 	"#",		/* COMMENT_CMD */
192 	"chainloader",	/* CHAINLOADER_CMD */
193 	"args",		/* ARGS_CMD */
194 	"findroot",	/* FINDROOT_CMD */
195 	NULL
196 };
197 
198 #define	OPT_ENTRY_NUM	"entry"
199 
200 /*
201  * exec_cmd related
202  */
203 typedef struct {
204 	line_t *head;
205 	line_t *tail;
206 } filelist_t;
207 
208 #define	BOOT_FILE_LIST	"boot/solaris/filelist.ramdisk"
209 #define	ETC_FILE_LIST	"etc/boot/solaris/filelist.ramdisk"
210 
211 #define	FILE_STAT	"boot/solaris/filestat.ramdisk"
212 #define	FILE_STAT_TMP	"boot/solaris/filestat.ramdisk.tmp"
213 #define	DIR_PERMS	(S_IRWXU|S_IRGRP|S_IXGRP|S_IROTH|S_IXOTH)
214 #define	FILE_STAT_MODE	(S_IRUSR|S_IWUSR|S_IRGRP|S_IROTH)
215 
216 /* Globals */
217 int bam_verbose;
218 int bam_force;
219 int bam_debug;
220 static char *prog;
221 static subcmd_t bam_cmd;
222 static char *bam_root;
223 static int bam_rootlen;
224 static int bam_root_readonly;
225 static int bam_alt_root;
226 static int bam_extend = 0;
227 static int bam_purge = 0;
228 static char *bam_subcmd;
229 static char *bam_opt;
230 static char **bam_argv;
231 static int bam_argc;
232 static int bam_check;
233 static int bam_smf_check;
234 static int bam_lock_fd = -1;
235 static int bam_zfs;
236 static char rootbuf[PATH_MAX] = "/";
237 static int bam_update_all;
238 static int bam_alt_platform;
239 static char *bam_platform;
240 
241 /* function prototypes */
242 static void parse_args_internal(int, char *[]);
243 static void parse_args(int, char *argv[]);
244 static error_t bam_menu(char *, char *, int, char *[]);
245 static error_t bam_archive(char *, char *);
246 
247 static void bam_exit(int);
248 static void bam_lock(void);
249 static void bam_unlock(void);
250 
251 static int exec_cmd(char *, filelist_t *);
252 static error_t read_globals(menu_t *, char *, char *, int);
253 static int menu_on_bootdisk(char *os_root, char *menu_root);
254 static menu_t *menu_read(char *);
255 static error_t menu_write(char *, menu_t *);
256 static void linelist_free(line_t *);
257 static void menu_free(menu_t *);
258 static void filelist_free(filelist_t *);
259 static error_t list2file(char *, char *, char *, line_t *);
260 static error_t list_entry(menu_t *, char *, char *);
261 static error_t delete_all_entries(menu_t *, char *, char *);
262 static error_t update_entry(menu_t *mp, char *menu_root, char *opt);
263 static error_t update_temp(menu_t *mp, char *dummy, char *opt);
264 
265 static error_t update_archive(char *, char *);
266 static error_t list_archive(char *, char *);
267 static error_t update_all(char *, char *);
268 static error_t read_list(char *, filelist_t *);
269 static error_t set_global(menu_t *, char *, int);
270 static error_t set_option(menu_t *, char *, char *);
271 static error_t set_kernel(menu_t *, menu_cmd_t, char *, char *, size_t);
272 static error_t get_kernel(menu_t *, menu_cmd_t, char *, size_t);
273 static char *expand_path(const char *);
274 
275 static long s_strtol(char *);
276 static int s_fputs(char *, FILE *);
277 
278 static int is_zfs(char *root);
279 static int is_ufs(char *root);
280 static int is_pcfs(char *root);
281 static int is_amd64(void);
282 static char *get_machine(void);
283 static void append_to_flist(filelist_t *, char *);
284 static char *mount_top_dataset(char *pool, zfs_mnted_t *mnted);
285 static int umount_top_dataset(char *pool, zfs_mnted_t mnted, char *mntpt);
286 static int ufs_add_to_sign_list(char *sign);
287 static error_t synchronize_BE_menu(void);
288 
289 #if !defined(_OPB)
290 static void ucode_install();
291 #endif
292 
293 /* Menu related sub commands */
294 static subcmd_defn_t menu_subcmds[] = {
295 	"set_option",		OPT_ABSENT,	set_option, 0,	/* PUB */
296 	"list_entry",		OPT_OPTIONAL,	list_entry, 1,	/* PUB */
297 	"delete_all_entries",	OPT_ABSENT,	delete_all_entries, 0, /* PVT */
298 	"update_entry",		OPT_REQ,	update_entry, 0, /* menu */
299 	"update_temp",		OPT_OPTIONAL,	update_temp, 0,	/* reboot */
300 	"upgrade",		OPT_ABSENT,	upgrade_menu, 0, /* menu */
301 	NULL,			0,		NULL, 0	/* must be last */
302 };
303 
304 /* Archive related sub commands */
305 static subcmd_defn_t arch_subcmds[] = {
306 	"update",		OPT_ABSENT,	update_archive, 0, /* PUB */
307 	"update_all",		OPT_ABSENT,	update_all, 0,	/* PVT */
308 	"list",			OPT_OPTIONAL,	list_archive, 1, /* PUB */
309 	NULL,			0,		NULL, 0	/* must be last */
310 };
311 
312 enum dircache_copy_opt {
313 	FILE32 = 0,
314 	FILE64,
315 	CACHEDIR_NUM
316 };
317 
318 /*
319  * Directory specific flags:
320  * NEED_UPDATE : the specified archive needs to be updated
321  * NO_MULTI : don't extend the specified archive, but recreate it
322  */
323 #define	NEED_UPDATE		0x00000001
324 #define	NO_MULTI		0x00000002
325 
326 #define	set_dir_flag(id, f)	(walk_arg.dirinfo[id].flags |= f)
327 #define	unset_dir_flag(id, f)	(walk_arg.dirinfo[id].flags &= ~f)
328 #define	is_dir_flag_on(id, f)	(walk_arg.dirinfo[id].flags & f ? 1 : 0)
329 
330 #define	get_cachedir(id)	(walk_arg.dirinfo[id].cdir_path)
331 #define	get_updatedir(id)	(walk_arg.dirinfo[id].update_path)
332 #define	get_count(id)		(walk_arg.dirinfo[id].count)
333 #define	has_cachedir(id)	(walk_arg.dirinfo[id].has_dir)
334 #define	set_dir_present(id)	(walk_arg.dirinfo[id].has_dir = 1)
335 
336 /*
337  * dirinfo_t (specific cache directory information):
338  * cdir_path:   path to the archive cache directory
339  * update_path: path to the update directory (contains the files that will be
340  *              used to extend the archive)
341  * has_dir:	the specified cache directory is active
342  * count:	the number of files to update
343  * flags:	directory specific flags
344  */
345 typedef struct _dirinfo {
346 	char	cdir_path[PATH_MAX];
347 	char	update_path[PATH_MAX];
348 	int	has_dir;
349 	int	count;
350 	int	flags;
351 } dirinfo_t;
352 
353 /*
354  * Update flags:
355  * NEED_CACHE_DIR : cache directory is missing and needs to be created
356  * IS_SPARC_TARGET : the target mountpoint is a SPARC environment
357  * UPDATE_ERROR : an error occourred while traversing the list of files
358  * RDONLY_FSCHK : the target filesystem is read-only
359  * RAMDSK_FSCHK : the target filesystem is on a ramdisk
360  */
361 #define	NEED_CACHE_DIR		0x00000001
362 #define	IS_SPARC_TARGET		0x00000002
363 #define	UPDATE_ERROR		0x00000004
364 #define	RDONLY_FSCHK		0x00000008
365 #define	RAMDSK_FSCHK		0x00000010
366 
367 #define	is_flag_on(flag)	(walk_arg.update_flags & flag ? 1 : 0)
368 #define	set_flag(flag)		(walk_arg.update_flags |= flag)
369 #define	unset_flag(flag)	(walk_arg.update_flags &= ~flag)
370 
371 /*
372  * struct walk_arg :
373  * update_flags: flags related to the current updating process
374  * new_nvlp/old_nvlp: new and old list of archive-files / attributes pairs
375  * sparcfile: list of file paths for mkisofs -path-list (SPARC only)
376  */
377 static struct {
378 	int 		update_flags;
379 	nvlist_t 	*new_nvlp;
380 	nvlist_t 	*old_nvlp;
381 	FILE 		*sparcfile;
382 	dirinfo_t	dirinfo[CACHEDIR_NUM];
383 } walk_arg;
384 
385 struct safefile {
386 	char *name;
387 	struct safefile *next;
388 };
389 
390 static struct safefile *safefiles = NULL;
391 #define	NEED_UPDATE_FILE "/etc/svc/volatile/boot_archive_needs_update"
392 
393 /* Thanks growisofs */
394 #define	CD_BLOCK	((off64_t)2048)
395 #define	VOLDESC_OFF	16
396 #define	DVD_BLOCK	(32*1024)
397 #define	MAX_IVDs	16
398 
399 struct iso_pdesc {
400     unsigned char type	[1];
401     unsigned char id	[5];
402     unsigned char void1	[80-5-1];
403     unsigned char volume_space_size [8];
404     unsigned char void2	[2048-80-8];
405 };
406 
407 /*
408  * COUNT_MAX:	maximum number of changed files to justify a multisession update
409  * BA_SIZE_MAX:	maximum size of the boot_archive to justify a multisession
410  * 		update
411  */
412 #define	COUNT_MAX		50
413 #define	BA_SIZE_MAX		(50 * 1024 * 1024)
414 
415 #define	bam_nowrite()		(bam_check || bam_smf_check)
416 
417 static int sync_menu = 1;	/* whether we need to sync the BE menus */
418 
419 static void
420 usage(void)
421 {
422 	(void) fprintf(stderr, "USAGE:\n");
423 
424 	/* archive usage */
425 	(void) fprintf(stderr,
426 	    "\t%s update-archive [-vn] [-R altroot [-p platform>]]\n", prog);
427 	(void) fprintf(stderr,
428 	    "\t%s list-archive [-R altroot [-p platform>]]\n", prog);
429 #if !defined(_OPB)
430 	/* x86 only */
431 	(void) fprintf(stderr, "\t%s set-menu [-R altroot] key=value\n", prog);
432 	(void) fprintf(stderr, "\t%s list-menu [-R altroot]\n", prog);
433 #endif
434 }
435 
436 int
437 main(int argc, char *argv[])
438 {
439 	error_t ret;
440 
441 	(void) setlocale(LC_ALL, "");
442 	(void) textdomain(TEXT_DOMAIN);
443 
444 	if ((prog = strrchr(argv[0], '/')) == NULL) {
445 		prog = argv[0];
446 	} else {
447 		prog++;
448 	}
449 
450 	INJECT_ERROR1("ASSERT_ON", assert(0))
451 
452 	/*
453 	 * Don't depend on caller's umask
454 	 */
455 	(void) umask(0022);
456 
457 	parse_args(argc, argv);
458 
459 	switch (bam_cmd) {
460 		case BAM_MENU:
461 			ret = bam_menu(bam_subcmd, bam_opt, bam_argc, bam_argv);
462 			break;
463 		case BAM_ARCHIVE:
464 			ret = bam_archive(bam_subcmd, bam_opt);
465 			break;
466 		default:
467 			usage();
468 			bam_exit(1);
469 	}
470 
471 	if (ret != BAM_SUCCESS)
472 		bam_exit(1);
473 
474 	bam_unlock();
475 	return (0);
476 }
477 
478 /*
479  * Equivalence of public and internal commands:
480  *	update-archive  -- -a update
481  *	list-archive	-- -a list
482  *	set-menu	-- -m set_option
483  *	list-menu	-- -m list_entry
484  *	update-menu	-- -m update_entry
485  */
486 static struct cmd_map {
487 	char *bam_cmdname;
488 	int bam_cmd;
489 	char *bam_subcmd;
490 } cmd_map[] = {
491 	{ "update-archive",	BAM_ARCHIVE,	"update"},
492 	{ "list-archive",	BAM_ARCHIVE,	"list"},
493 	{ "set-menu",		BAM_MENU,	"set_option"},
494 	{ "list-menu",		BAM_MENU,	"list_entry"},
495 	{ "update-menu",	BAM_MENU,	"update_entry"},
496 	{ NULL,			0,		NULL}
497 };
498 
499 /*
500  * Commands syntax published in bootadm(1M) are parsed here
501  */
502 static void
503 parse_args(int argc, char *argv[])
504 {
505 	struct cmd_map *cmp = cmd_map;
506 
507 	/* command conforming to the final spec */
508 	if (argc > 1 && argv[1][0] != '-') {
509 		/*
510 		 * Map commands to internal table.
511 		 */
512 		while (cmp->bam_cmdname) {
513 			if (strcmp(argv[1], cmp->bam_cmdname) == 0) {
514 				bam_cmd = cmp->bam_cmd;
515 				bam_subcmd = cmp->bam_subcmd;
516 				break;
517 			}
518 			cmp++;
519 		}
520 		if (cmp->bam_cmdname == NULL) {
521 			usage();
522 			bam_exit(1);
523 		}
524 		argc--;
525 		argv++;
526 	}
527 
528 	parse_args_internal(argc, argv);
529 }
530 
531 /*
532  * A combination of public and private commands are parsed here.
533  * The internal syntax and the corresponding functionality are:
534  *	-a update	-- update-archive
535  *	-a list		-- list-archive
536  *	-a update-all	-- (reboot to sync all mounted OS archive)
537  *	-m update_entry	-- update-menu
538  *	-m list_entry	-- list-menu
539  *	-m update_temp	-- (reboot -- [boot-args])
540  *	-m delete_all_entries -- (called from install)
541  * A set of private flags is there too:
542  *	-F		-- purge the cache directories and rebuild them
543  *	-e		-- use the (faster) archive update approach (used by
544  *			   reboot)
545  */
546 static void
547 parse_args_internal(int argc, char *argv[])
548 {
549 	int c, error;
550 	extern char *optarg;
551 	extern int optind, opterr;
552 
553 	/* Suppress error message from getopt */
554 	opterr = 0;
555 
556 	error = 0;
557 	while ((c = getopt(argc, argv, "a:d:fm:no:veFCR:p:Z")) != -1) {
558 		switch (c) {
559 		case 'a':
560 			if (bam_cmd) {
561 				error = 1;
562 				bam_error(MULT_CMDS, c);
563 			}
564 			bam_cmd = BAM_ARCHIVE;
565 			bam_subcmd = optarg;
566 			break;
567 		case 'd':
568 			if (bam_debug) {
569 				error = 1;
570 				bam_error(DUP_OPT, c);
571 			}
572 			bam_debug = s_strtol(optarg);
573 			break;
574 		case 'f':
575 			bam_force = 1;
576 			break;
577 		case 'F':
578 			bam_purge = 1;
579 			break;
580 		case 'm':
581 			if (bam_cmd) {
582 				error = 1;
583 				bam_error(MULT_CMDS, c);
584 			}
585 			bam_cmd = BAM_MENU;
586 			bam_subcmd = optarg;
587 			break;
588 		case 'n':
589 			bam_check = 1;
590 			break;
591 		case 'o':
592 			if (bam_opt) {
593 				error = 1;
594 				bam_error(DUP_OPT, c);
595 			}
596 			bam_opt = optarg;
597 			break;
598 		case 'v':
599 			bam_verbose = 1;
600 			break;
601 		case 'C':
602 			bam_smf_check = 1;
603 			break;
604 		case 'R':
605 			if (bam_root) {
606 				error = 1;
607 				bam_error(DUP_OPT, c);
608 				break;
609 			} else if (realpath(optarg, rootbuf) == NULL) {
610 				error = 1;
611 				bam_error(CANT_RESOLVE, optarg,
612 				    strerror(errno));
613 				break;
614 			}
615 			bam_alt_root = 1;
616 			bam_root = rootbuf;
617 			bam_rootlen = strlen(rootbuf);
618 			break;
619 		case 'p':
620 			bam_alt_platform = 1;
621 			bam_platform = optarg;
622 			if ((strcmp(bam_platform, "i86pc") != 0) &&
623 			    (strcmp(bam_platform, "sun4u") != 0) &&
624 			    (strcmp(bam_platform, "sun4v") != 0)) {
625 				error = 1;
626 				bam_error(INVALID_PLAT, bam_platform);
627 			}
628 			break;
629 		case 'Z':
630 			bam_zfs = 1;
631 			break;
632 		case 'e':
633 			bam_extend = 1;
634 			break;
635 		case '?':
636 			error = 1;
637 			bam_error(BAD_OPT, optopt);
638 			break;
639 		default :
640 			error = 1;
641 			bam_error(BAD_OPT, c);
642 			break;
643 		}
644 	}
645 
646 	/*
647 	 * An alternate platform requires an alternate root
648 	 */
649 	if (bam_alt_platform && bam_alt_root == 0) {
650 		usage();
651 		bam_exit(0);
652 	}
653 
654 	/*
655 	 * A command option must be specfied
656 	 */
657 	if (!bam_cmd) {
658 		if (bam_opt && strcmp(bam_opt, "all") == 0) {
659 			usage();
660 			bam_exit(0);
661 		}
662 		bam_error(NEED_CMD);
663 		error = 1;
664 	}
665 
666 	if (error) {
667 		usage();
668 		bam_exit(1);
669 	}
670 
671 	if (optind > argc) {
672 		bam_error(INT_ERROR, "parse_args");
673 		bam_exit(1);
674 	} else if (optind < argc) {
675 		bam_argv = &argv[optind];
676 		bam_argc = argc - optind;
677 	}
678 
679 	/*
680 	 * -n implies verbose mode
681 	 */
682 	if (bam_check)
683 		bam_verbose = 1;
684 }
685 
686 static error_t
687 check_subcmd_and_options(
688 	char *subcmd,
689 	char *opt,
690 	subcmd_defn_t *table,
691 	error_t (**fp)())
692 {
693 	int i;
694 
695 	if (subcmd == NULL) {
696 		bam_error(NEED_SUBCMD);
697 		return (BAM_ERROR);
698 	}
699 
700 	if (strcmp(subcmd, "set_option") == 0) {
701 		if (bam_argc == 0 || bam_argv == NULL || bam_argv[0] == NULL) {
702 			bam_error(MISSING_ARG);
703 			usage();
704 			return (BAM_ERROR);
705 		} else if (bam_argc > 1 || bam_argv[1] != NULL) {
706 			bam_error(TRAILING_ARGS);
707 			usage();
708 			return (BAM_ERROR);
709 		}
710 	} else if (strcmp(subcmd, "update_all") == 0) {
711 		/*
712 		 * The only option we accept for the "update_all"
713 		 * subcmd is "fastboot".
714 		 */
715 		if (bam_argc > 1 || (bam_argc == 1 &&
716 		    strcmp(bam_argv[0], "fastboot") != 0)) {
717 			bam_error(TRAILING_ARGS);
718 			usage();
719 			return (BAM_ERROR);
720 		}
721 		if (bam_argc == 1)
722 			sync_menu = 0;
723 	} else if (bam_argc || bam_argv) {
724 		bam_error(TRAILING_ARGS);
725 		usage();
726 		return (BAM_ERROR);
727 	}
728 
729 	if (bam_root == NULL) {
730 		bam_root = rootbuf;
731 		bam_rootlen = 1;
732 	}
733 
734 	/* verify that subcmd is valid */
735 	for (i = 0; table[i].subcmd != NULL; i++) {
736 		if (strcmp(table[i].subcmd, subcmd) == 0)
737 			break;
738 	}
739 
740 	if (table[i].subcmd == NULL) {
741 		bam_error(INVALID_SUBCMD, subcmd);
742 		return (BAM_ERROR);
743 	}
744 
745 	if (table[i].unpriv == 0 && geteuid() != 0) {
746 		bam_error(MUST_BE_ROOT);
747 		return (BAM_ERROR);
748 	}
749 
750 	/*
751 	 * Currently only privileged commands need a lock
752 	 */
753 	if (table[i].unpriv == 0)
754 		bam_lock();
755 
756 	/* subcmd verifies that opt is appropriate */
757 	if (table[i].option != OPT_OPTIONAL) {
758 		if ((table[i].option == OPT_REQ) ^ (opt != NULL)) {
759 			if (opt)
760 				bam_error(NO_OPT_REQ, subcmd);
761 			else
762 				bam_error(MISS_OPT, subcmd);
763 			return (BAM_ERROR);
764 		}
765 	}
766 
767 	*fp = table[i].handler;
768 
769 	return (BAM_SUCCESS);
770 }
771 
772 /*
773  * NOTE: A single "/" is also considered a trailing slash and will
774  * be deleted.
775  */
776 static void
777 elide_trailing_slash(const char *src, char *dst, size_t dstsize)
778 {
779 	size_t dstlen;
780 
781 	assert(src);
782 	assert(dst);
783 
784 	(void) strlcpy(dst, src, dstsize);
785 
786 	dstlen = strlen(dst);
787 	if (dst[dstlen - 1] == '/') {
788 		dst[dstlen - 1] = '\0';
789 	}
790 }
791 
792 static error_t
793 bam_menu(char *subcmd, char *opt, int largc, char *largv[])
794 {
795 	error_t			ret;
796 	char			menu_path[PATH_MAX];
797 	char			clean_menu_root[PATH_MAX];
798 	char			path[PATH_MAX];
799 	menu_t			*menu;
800 	char			menu_root[PATH_MAX];
801 	struct stat		sb;
802 	error_t (*f)(menu_t *mp, char *menu_path, char *opt);
803 	char			*special;
804 	char			*pool = NULL;
805 	zfs_mnted_t		zmnted;
806 	char			*zmntpt;
807 	char			*osdev;
808 	char			*osroot;
809 	const char		*fcn = "bam_menu()";
810 
811 	/*
812 	 * Menu sub-command only applies to GRUB (i.e. x86)
813 	 */
814 	if (!is_grub(bam_alt_root ? bam_root : "/")) {
815 		bam_error(NOT_GRUB_BOOT);
816 		return (BAM_ERROR);
817 	}
818 
819 	/*
820 	 * Check arguments
821 	 */
822 	ret = check_subcmd_and_options(subcmd, opt, menu_subcmds, &f);
823 	if (ret == BAM_ERROR) {
824 		return (BAM_ERROR);
825 	}
826 
827 	assert(bam_root);
828 
829 	(void) strlcpy(menu_root, bam_root, sizeof (menu_root));
830 	osdev = osroot = NULL;
831 
832 	if (strcmp(subcmd, "update_entry") == 0) {
833 		assert(opt);
834 
835 		osdev = strtok(opt, ",");
836 		assert(osdev);
837 		osroot = strtok(NULL, ",");
838 		if (osroot) {
839 			/* fixup bam_root so that it points at osroot */
840 			if (realpath(osroot, rootbuf) == NULL) {
841 				bam_error(CANT_RESOLVE, osroot,
842 				    strerror(errno));
843 				return (BAM_ERROR);
844 			}
845 			bam_alt_root = 1;
846 			bam_root  = rootbuf;
847 			bam_rootlen = strlen(rootbuf);
848 		}
849 	}
850 
851 	/*
852 	 * We support menu on PCFS (under certain conditions), but
853 	 * not the OS root
854 	 */
855 	if (is_pcfs(bam_root)) {
856 		bam_error(PCFS_ROOT_NOTSUP, bam_root);
857 		return (BAM_ERROR);
858 	}
859 
860 	if (stat(menu_root, &sb) == -1) {
861 		bam_error(CANNOT_LOCATE_GRUB_MENU);
862 		return (BAM_ERROR);
863 	}
864 
865 	BAM_DPRINTF((D_MENU_ROOT, fcn, menu_root));
866 
867 	/*
868 	 * We no longer use the GRUB slice file. If it exists, then
869 	 * the user is doing something that is unsupported (such as
870 	 * standard upgrading an old Live Upgrade BE). If that
871 	 * happens, mimic existing behavior i.e. pretend that it is
872 	 * not a BE. Emit a warning though.
873 	 */
874 	if (bam_alt_root) {
875 		(void) snprintf(path, sizeof (path), "%s%s", bam_root,
876 		    GRUB_slice);
877 	} else {
878 		(void) snprintf(path, sizeof (path), "%s", GRUB_slice);
879 	}
880 
881 	if (bam_verbose && stat(path, &sb) == 0)
882 		bam_error(GRUB_SLICE_FILE_EXISTS, path);
883 
884 	if (is_zfs(menu_root)) {
885 		assert(strcmp(menu_root, bam_root) == 0);
886 		special = get_special(menu_root);
887 		INJECT_ERROR1("Z_MENU_GET_SPECIAL", special = NULL);
888 		if (special == NULL) {
889 			bam_error(CANT_FIND_SPECIAL, menu_root);
890 			return (BAM_ERROR);
891 		}
892 		pool = strtok(special, "/");
893 		INJECT_ERROR1("Z_MENU_GET_POOL", pool = NULL);
894 		if (pool == NULL) {
895 			free(special);
896 			bam_error(CANT_FIND_POOL, menu_root);
897 			return (BAM_ERROR);
898 		}
899 		BAM_DPRINTF((D_Z_MENU_GET_POOL_FROM_SPECIAL, fcn, pool));
900 
901 		zmntpt = mount_top_dataset(pool, &zmnted);
902 		INJECT_ERROR1("Z_MENU_MOUNT_TOP_DATASET", zmntpt = NULL);
903 		if (zmntpt == NULL) {
904 			bam_error(CANT_MOUNT_POOL_DATASET, pool);
905 			free(special);
906 			return (BAM_ERROR);
907 		}
908 		BAM_DPRINTF((D_Z_GET_MENU_MOUNT_TOP_DATASET, fcn, zmntpt));
909 
910 		(void) strlcpy(menu_root, zmntpt, sizeof (menu_root));
911 		BAM_DPRINTF((D_Z_GET_MENU_MENU_ROOT, fcn, menu_root));
912 	}
913 
914 	elide_trailing_slash(menu_root, clean_menu_root,
915 	    sizeof (clean_menu_root));
916 
917 	BAM_DPRINTF((D_CLEAN_MENU_ROOT, fcn, clean_menu_root));
918 
919 	(void) strlcpy(menu_path, clean_menu_root, sizeof (menu_path));
920 	(void) strlcat(menu_path, GRUB_MENU, sizeof (menu_path));
921 
922 	BAM_DPRINTF((D_MENU_PATH, fcn, menu_path));
923 
924 	/*
925 	 * If listing the menu, display the menu location
926 	 */
927 	if (strcmp(subcmd, "list_entry") == 0) {
928 		bam_print(GRUB_MENU_PATH, menu_path);
929 	}
930 
931 
932 	menu = menu_read(menu_path);
933 	assert(menu);
934 
935 	/*
936 	 * We already checked the following case in
937 	 * check_subcmd_and_suboptions() above. Complete the
938 	 * final step now.
939 	 */
940 	if (strcmp(subcmd, "set_option") == 0) {
941 		assert(largc == 1 && largv[0] && largv[1] == NULL);
942 		opt = largv[0];
943 	} else {
944 		assert(largc == 0 && largv == NULL);
945 	}
946 
947 	ret = get_boot_cap(bam_root);
948 	if (ret != BAM_SUCCESS) {
949 		BAM_DPRINTF((D_BOOT_GET_CAP_FAILED, fcn));
950 		goto out;
951 	}
952 
953 	/*
954 	 * Once the sub-cmd handler has run
955 	 * only the line field is guaranteed to have valid values
956 	 */
957 	if (strcmp(subcmd, "update_entry") == 0)
958 		ret = f(menu, menu_root, osdev);
959 	else if (strcmp(subcmd, "upgrade") == 0)
960 		ret = f(menu, bam_root, menu_root);
961 	else if (strcmp(subcmd, "list_entry") == 0)
962 		ret = f(menu, menu_path, opt);
963 	else
964 		ret = f(menu, NULL, opt);
965 
966 	if (ret == BAM_WRITE) {
967 		BAM_DPRINTF((D_WRITING_MENU_ROOT, fcn, clean_menu_root));
968 		ret = menu_write(clean_menu_root, menu);
969 	}
970 
971 out:
972 	INJECT_ERROR1("POOL_SET", pool = "/pooldata");
973 	assert((is_zfs(menu_root)) ^ (pool == NULL));
974 	if (pool) {
975 		(void) umount_top_dataset(pool, zmnted, zmntpt);
976 		free(special);
977 	}
978 	menu_free(menu);
979 	return (ret);
980 }
981 
982 
983 static error_t
984 bam_archive(
985 	char *subcmd,
986 	char *opt)
987 {
988 	error_t			ret;
989 	error_t			(*f)(char *root, char *opt);
990 	const char		*fcn = "bam_archive()";
991 
992 	/*
993 	 * Add trailing / for archive subcommands
994 	 */
995 	if (rootbuf[strlen(rootbuf) - 1] != '/')
996 		(void) strcat(rootbuf, "/");
997 	bam_rootlen = strlen(rootbuf);
998 
999 	/*
1000 	 * Check arguments
1001 	 */
1002 	ret = check_subcmd_and_options(subcmd, opt, arch_subcmds, &f);
1003 	if (ret != BAM_SUCCESS) {
1004 		return (BAM_ERROR);
1005 	}
1006 
1007 	ret = get_boot_cap(rootbuf);
1008 	if (ret != BAM_SUCCESS) {
1009 		BAM_DPRINTF((D_BOOT_GET_CAP_FAILED, fcn));
1010 		return (ret);
1011 	}
1012 
1013 	/*
1014 	 * Check archive not supported with update_all
1015 	 * since it is awkward to display out-of-sync
1016 	 * information for each BE.
1017 	 */
1018 	if (bam_check && strcmp(subcmd, "update_all") == 0) {
1019 		bam_error(CHECK_NOT_SUPPORTED, subcmd);
1020 		return (BAM_ERROR);
1021 	}
1022 
1023 	if (strcmp(subcmd, "update_all") == 0)
1024 		bam_update_all = 1;
1025 
1026 #if !defined(_OPB)
1027 	ucode_install(bam_root);
1028 #endif
1029 
1030 	ret = f(bam_root, opt);
1031 
1032 	bam_update_all = 0;
1033 
1034 	return (ret);
1035 }
1036 
1037 /*PRINTFLIKE1*/
1038 void
1039 bam_error(char *format, ...)
1040 {
1041 	va_list ap;
1042 
1043 	va_start(ap, format);
1044 	(void) fprintf(stderr, "%s: ", prog);
1045 	(void) vfprintf(stderr, format, ap);
1046 	va_end(ap);
1047 }
1048 
1049 /*PRINTFLIKE1*/
1050 void
1051 bam_derror(char *format, ...)
1052 {
1053 	va_list ap;
1054 
1055 	assert(bam_debug);
1056 
1057 	va_start(ap, format);
1058 	(void) fprintf(stderr, "DEBUG: ");
1059 	(void) vfprintf(stderr, format, ap);
1060 	va_end(ap);
1061 }
1062 
1063 /*PRINTFLIKE1*/
1064 void
1065 bam_print(char *format, ...)
1066 {
1067 	va_list ap;
1068 
1069 	va_start(ap, format);
1070 	(void) vfprintf(stdout, format, ap);
1071 	va_end(ap);
1072 }
1073 
1074 /*PRINTFLIKE1*/
1075 void
1076 bam_print_stderr(char *format, ...)
1077 {
1078 	va_list ap;
1079 
1080 	va_start(ap, format);
1081 	(void) vfprintf(stderr, format, ap);
1082 	va_end(ap);
1083 }
1084 
1085 static void
1086 bam_exit(int excode)
1087 {
1088 	bam_unlock();
1089 	exit(excode);
1090 }
1091 
1092 static void
1093 bam_lock(void)
1094 {
1095 	struct flock lock;
1096 	pid_t pid;
1097 
1098 	bam_lock_fd = open(BAM_LOCK_FILE, O_CREAT|O_RDWR, LOCK_FILE_PERMS);
1099 	if (bam_lock_fd < 0) {
1100 		/*
1101 		 * We may be invoked early in boot for archive verification.
1102 		 * In this case, root is readonly and /var/run may not exist.
1103 		 * Proceed without the lock
1104 		 */
1105 		if (errno == EROFS || errno == ENOENT) {
1106 			bam_root_readonly = 1;
1107 			return;
1108 		}
1109 
1110 		bam_error(OPEN_FAIL, BAM_LOCK_FILE, strerror(errno));
1111 		bam_exit(1);
1112 	}
1113 
1114 	lock.l_type = F_WRLCK;
1115 	lock.l_whence = SEEK_SET;
1116 	lock.l_start = 0;
1117 	lock.l_len = 0;
1118 
1119 	if (fcntl(bam_lock_fd, F_SETLK, &lock) == -1) {
1120 		if (errno != EACCES && errno != EAGAIN) {
1121 			bam_error(LOCK_FAIL, BAM_LOCK_FILE, strerror(errno));
1122 			(void) close(bam_lock_fd);
1123 			bam_lock_fd = -1;
1124 			bam_exit(1);
1125 		}
1126 		pid = 0;
1127 		(void) pread(bam_lock_fd, &pid, sizeof (pid_t), 0);
1128 		bam_print(FILE_LOCKED, pid);
1129 
1130 		lock.l_type = F_WRLCK;
1131 		lock.l_whence = SEEK_SET;
1132 		lock.l_start = 0;
1133 		lock.l_len = 0;
1134 		if (fcntl(bam_lock_fd, F_SETLKW, &lock) == -1) {
1135 			bam_error(LOCK_FAIL, BAM_LOCK_FILE, strerror(errno));
1136 			(void) close(bam_lock_fd);
1137 			bam_lock_fd = -1;
1138 			bam_exit(1);
1139 		}
1140 	}
1141 
1142 	/* We own the lock now */
1143 	pid = getpid();
1144 	(void) write(bam_lock_fd, &pid, sizeof (pid));
1145 }
1146 
1147 static void
1148 bam_unlock(void)
1149 {
1150 	struct flock unlock;
1151 
1152 	/*
1153 	 * NOP if we don't hold the lock
1154 	 */
1155 	if (bam_lock_fd < 0) {
1156 		return;
1157 	}
1158 
1159 	unlock.l_type = F_UNLCK;
1160 	unlock.l_whence = SEEK_SET;
1161 	unlock.l_start = 0;
1162 	unlock.l_len = 0;
1163 
1164 	if (fcntl(bam_lock_fd, F_SETLK, &unlock) == -1) {
1165 		bam_error(UNLOCK_FAIL, BAM_LOCK_FILE, strerror(errno));
1166 	}
1167 
1168 	if (close(bam_lock_fd) == -1) {
1169 		bam_error(CLOSE_FAIL, BAM_LOCK_FILE, strerror(errno));
1170 	}
1171 	bam_lock_fd = -1;
1172 }
1173 
1174 static error_t
1175 list_archive(char *root, char *opt)
1176 {
1177 	filelist_t flist;
1178 	filelist_t *flistp = &flist;
1179 	line_t *lp;
1180 
1181 	assert(root);
1182 	assert(opt == NULL);
1183 
1184 	flistp->head = flistp->tail = NULL;
1185 	if (read_list(root, flistp) != BAM_SUCCESS) {
1186 		return (BAM_ERROR);
1187 	}
1188 	assert(flistp->head && flistp->tail);
1189 
1190 	for (lp = flistp->head; lp; lp = lp->next) {
1191 		bam_print(PRINT, lp->line);
1192 	}
1193 
1194 	filelist_free(flistp);
1195 
1196 	return (BAM_SUCCESS);
1197 }
1198 
1199 /*
1200  * This routine writes a list of lines to a file.
1201  * The list is *not* freed
1202  */
1203 static error_t
1204 list2file(char *root, char *tmp, char *final, line_t *start)
1205 {
1206 	char		tmpfile[PATH_MAX];
1207 	char		path[PATH_MAX];
1208 	FILE		*fp;
1209 	int		ret;
1210 	struct stat	sb;
1211 	mode_t		mode;
1212 	uid_t		root_uid;
1213 	gid_t		sys_gid;
1214 	struct passwd	*pw;
1215 	struct group	*gp;
1216 	const char	*fcn = "list2file()";
1217 
1218 	(void) snprintf(path, sizeof (path), "%s%s", root, final);
1219 
1220 	if (start == NULL) {
1221 		/* Empty GRUB menu */
1222 		if (stat(path, &sb) != -1) {
1223 			bam_print(UNLINK_EMPTY, path);
1224 			if (unlink(path) != 0) {
1225 				bam_error(UNLINK_FAIL, path, strerror(errno));
1226 				return (BAM_ERROR);
1227 			} else {
1228 				return (BAM_SUCCESS);
1229 			}
1230 		}
1231 		return (BAM_SUCCESS);
1232 	}
1233 
1234 	/*
1235 	 * Preserve attributes of existing file if possible,
1236 	 * otherwise ask the system for uid/gid of root/sys.
1237 	 * If all fails, fall back on hard-coded defaults.
1238 	 */
1239 	if (stat(path, &sb) != -1) {
1240 		mode = sb.st_mode;
1241 		root_uid = sb.st_uid;
1242 		sys_gid = sb.st_gid;
1243 	} else {
1244 		mode = DEFAULT_DEV_MODE;
1245 		if ((pw = getpwnam(DEFAULT_DEV_USER)) != NULL) {
1246 			root_uid = pw->pw_uid;
1247 		} else {
1248 			bam_error(CANT_FIND_USER,
1249 			    DEFAULT_DEV_USER, DEFAULT_DEV_UID);
1250 			root_uid = (uid_t)DEFAULT_DEV_UID;
1251 		}
1252 		if ((gp = getgrnam(DEFAULT_DEV_GROUP)) != NULL) {
1253 			sys_gid = gp->gr_gid;
1254 		} else {
1255 			bam_error(CANT_FIND_GROUP,
1256 			    DEFAULT_DEV_GROUP, DEFAULT_DEV_GID);
1257 			sys_gid = (gid_t)DEFAULT_DEV_GID;
1258 		}
1259 	}
1260 
1261 	(void) snprintf(tmpfile, sizeof (tmpfile), "%s%s", root, tmp);
1262 
1263 	/* Truncate tmpfile first */
1264 	fp = fopen(tmpfile, "w");
1265 	if (fp == NULL) {
1266 		bam_error(OPEN_FAIL, tmpfile, strerror(errno));
1267 		return (BAM_ERROR);
1268 	}
1269 	ret = fclose(fp);
1270 	INJECT_ERROR1("LIST2FILE_TRUNC_FCLOSE", ret = EOF);
1271 	if (ret == EOF) {
1272 		bam_error(CLOSE_FAIL, tmpfile, strerror(errno));
1273 		return (BAM_ERROR);
1274 	}
1275 
1276 	/* Now open it in append mode */
1277 	fp = fopen(tmpfile, "a");
1278 	if (fp == NULL) {
1279 		bam_error(OPEN_FAIL, tmpfile, strerror(errno));
1280 		return (BAM_ERROR);
1281 	}
1282 
1283 	for (; start; start = start->next) {
1284 		ret = s_fputs(start->line, fp);
1285 		INJECT_ERROR1("LIST2FILE_FPUTS", ret = EOF);
1286 		if (ret == EOF) {
1287 			bam_error(WRITE_FAIL, tmpfile, strerror(errno));
1288 			(void) fclose(fp);
1289 			return (BAM_ERROR);
1290 		}
1291 	}
1292 
1293 	ret = fclose(fp);
1294 	INJECT_ERROR1("LIST2FILE_APPEND_FCLOSE", ret = EOF);
1295 	if (ret == EOF) {
1296 		bam_error(CLOSE_FAIL, tmpfile, strerror(errno));
1297 		return (BAM_ERROR);
1298 	}
1299 
1300 	/*
1301 	 * Set up desired attributes.  Ignore failures on filesystems
1302 	 * not supporting these operations - pcfs reports unsupported
1303 	 * operations as EINVAL.
1304 	 */
1305 	ret = chmod(tmpfile, mode);
1306 	if (ret == -1 &&
1307 	    errno != EINVAL && errno != ENOTSUP) {
1308 		bam_error(CHMOD_FAIL, tmpfile, strerror(errno));
1309 		return (BAM_ERROR);
1310 	}
1311 
1312 	ret = chown(tmpfile, root_uid, sys_gid);
1313 	if (ret == -1 &&
1314 	    errno != EINVAL && errno != ENOTSUP) {
1315 		bam_error(CHOWN_FAIL, tmpfile, strerror(errno));
1316 		return (BAM_ERROR);
1317 	}
1318 
1319 
1320 	/*
1321 	 * Do an atomic rename
1322 	 */
1323 	ret = rename(tmpfile, path);
1324 	INJECT_ERROR1("LIST2FILE_RENAME", ret = -1);
1325 	if (ret != 0) {
1326 		bam_error(RENAME_FAIL, path, strerror(errno));
1327 		return (BAM_ERROR);
1328 	}
1329 
1330 	BAM_DPRINTF((D_WROTE_FILE, fcn, path));
1331 	return (BAM_SUCCESS);
1332 }
1333 
1334 /*
1335  * Checks if the path specified (without the file name at the end) exists
1336  * and creates it if not. If the path exists and is not a directory, an attempt
1337  * to unlink is made.
1338  */
1339 static int
1340 setup_path(char *path)
1341 {
1342 	char 		*p;
1343 	int		ret;
1344 	struct stat	sb;
1345 
1346 	p = strrchr(path, '/');
1347 	if (p != NULL) {
1348 		*p = '\0';
1349 		if (stat(path, &sb) != 0 || !(S_ISDIR(sb.st_mode))) {
1350 			/* best effort attempt, mkdirp will catch the error */
1351 			(void) unlink(path);
1352 			if (bam_verbose)
1353 				bam_print(NEED_DIRPATH, path);
1354 			ret = mkdirp(path, DIR_PERMS);
1355 			if (ret == -1) {
1356 				bam_error(MKDIR_FAILED, path, strerror(errno));
1357 				*p = '/';
1358 				return (BAM_ERROR);
1359 			}
1360 		}
1361 		*p = '/';
1362 		return (BAM_SUCCESS);
1363 	}
1364 	return (BAM_SUCCESS);
1365 }
1366 
1367 typedef union {
1368 	gzFile	gzfile;
1369 	int	fdfile;
1370 } outfile;
1371 
1372 typedef struct {
1373 	char		path[PATH_MAX];
1374 	outfile		out;
1375 } cachefile;
1376 
1377 static int
1378 setup_file(char *base, const char *path, cachefile *cf)
1379 {
1380 	int	ret;
1381 	char	*strip;
1382 
1383 	/* init gzfile or fdfile in case we fail before opening */
1384 	if (bam_direct == BAM_DIRECT_DBOOT)
1385 		cf->out.gzfile = NULL;
1386 	else
1387 		cf->out.fdfile = -1;
1388 
1389 	/* strip the trailing altroot path */
1390 	strip = (char *)path + strlen(rootbuf);
1391 
1392 	ret = snprintf(cf->path, sizeof (cf->path), "%s/%s", base, strip);
1393 	if (ret >= sizeof (cf->path)) {
1394 		bam_error(PATH_TOO_LONG, rootbuf);
1395 		return (BAM_ERROR);
1396 	}
1397 
1398 	/* Check if path is present in the archive cache directory */
1399 	if (setup_path(cf->path) == BAM_ERROR)
1400 		return (BAM_ERROR);
1401 
1402 	if (bam_direct == BAM_DIRECT_DBOOT) {
1403 		if ((cf->out.gzfile = gzopen(cf->path, "wb")) == NULL) {
1404 			bam_error(GZ_OPEN_FAIL, gzerror(cf->out.gzfile, NULL));
1405 			return (BAM_ERROR);
1406 		}
1407 		(void) gzsetparams(cf->out.gzfile, Z_BEST_SPEED,
1408 		    Z_DEFAULT_STRATEGY);
1409 	} else {
1410 		if ((cf->out.fdfile = open(cf->path, O_WRONLY | O_CREAT, 0644))
1411 		    == -1) {
1412 			bam_error(OPEN_FAIL, cf->path, strerror(errno));
1413 			return (BAM_ERROR);
1414 		}
1415 	}
1416 
1417 	return (BAM_SUCCESS);
1418 }
1419 
1420 static int
1421 cache_write(cachefile cf, char *buf, int size)
1422 {
1423 	int	err;
1424 
1425 	if (bam_direct == BAM_DIRECT_DBOOT) {
1426 		if (gzwrite(cf.out.gzfile, buf, size) < 1) {
1427 			bam_error(GZ_WRITE_FAIL, gzerror(cf.out.gzfile, &err));
1428 			if (err == Z_ERRNO)
1429 				bam_error(WRITE_FAIL, cf.path, strerror(errno));
1430 			return (BAM_ERROR);
1431 		}
1432 	} else {
1433 		if (write(cf.out.fdfile, buf, size) < 1) {
1434 			bam_error(WRITE_FAIL, cf.path, strerror(errno));
1435 			return (BAM_ERROR);
1436 		}
1437 	}
1438 	return (BAM_SUCCESS);
1439 }
1440 
1441 static int
1442 cache_close(cachefile cf)
1443 {
1444 	int	ret;
1445 
1446 	if (bam_direct == BAM_DIRECT_DBOOT) {
1447 		if (cf.out.gzfile) {
1448 			ret = gzclose(cf.out.gzfile);
1449 			if (ret == Z_OK)
1450 				return (BAM_SUCCESS);
1451 			else if (ret == Z_ERRNO)
1452 				bam_error(CLOSE_FAIL, cf.path, strerror(errno));
1453 			else
1454 				bam_error(GZCLOSE_FAIL, gzerror(cf.out.gzfile,
1455 				    NULL));
1456 			return (BAM_ERROR);
1457 		}
1458 	} else {
1459 		if (cf.out.fdfile != -1) {
1460 			ret = close(cf.out.fdfile);
1461 			if (ret != 0) {
1462 				bam_error(CLOSE_FAIL, cf.path, strerror(errno));
1463 				return (BAM_ERROR);
1464 			}
1465 		}
1466 	}
1467 
1468 	return (BAM_SUCCESS);
1469 }
1470 
1471 static int
1472 dircache_updatefile(const char *path, int what)
1473 {
1474 	int 		ret, exitcode;
1475 	char 		buf[4096 * 4];
1476 	FILE 		*infile;
1477 	cachefile 	outfile, outupdt;
1478 
1479 	if (bam_nowrite())
1480 		return (BAM_SUCCESS);
1481 
1482 	if ((infile = fopen(path, "rb")) == NULL) {
1483 		bam_error(OPEN_FAIL, path, strerror(errno));
1484 		return (BAM_ERROR);
1485 	}
1486 
1487 	ret = setup_file(get_cachedir(what), path, &outfile);
1488 	if (ret == BAM_ERROR) {
1489 		exitcode = BAM_ERROR;
1490 		goto out;
1491 	}
1492 	if (!is_dir_flag_on(what, NO_MULTI)) {
1493 		ret = setup_file(get_updatedir(what), path, &outupdt);
1494 		if (ret == BAM_ERROR)
1495 			set_dir_flag(what, NO_MULTI);
1496 	}
1497 
1498 	while ((ret = fread(buf, 1, sizeof (buf), infile)) > 0) {
1499 		if (cache_write(outfile, buf, ret) == BAM_ERROR) {
1500 			exitcode = BAM_ERROR;
1501 			goto out;
1502 		}
1503 		if (!is_dir_flag_on(what, NO_MULTI))
1504 			if (cache_write(outupdt, buf, ret) == BAM_ERROR)
1505 				set_dir_flag(what, NO_MULTI);
1506 	}
1507 
1508 	set_dir_flag(what, NEED_UPDATE);
1509 	get_count(what)++;
1510 	exitcode = BAM_SUCCESS;
1511 out:
1512 	(void) fclose(infile);
1513 	if (cache_close(outfile) == BAM_ERROR)
1514 		exitcode = BAM_ERROR;
1515 	if (!is_dir_flag_on(what, NO_MULTI) &&
1516 	    cache_close(outupdt) == BAM_ERROR)
1517 		exitcode = BAM_ERROR;
1518 	if (exitcode == BAM_ERROR)
1519 		set_flag(UPDATE_ERROR);
1520 	return (exitcode);
1521 }
1522 
1523 static int
1524 dircache_updatedir(const char *path, int what, int updt)
1525 {
1526 	int		ret;
1527 	char		dpath[PATH_MAX];
1528 	char		*strip;
1529 	struct stat	sb;
1530 
1531 	strip = (char *)path + strlen(rootbuf);
1532 
1533 	ret = snprintf(dpath, sizeof (dpath), "%s/%s", updt ?
1534 	    get_updatedir(what) : get_cachedir(what), strip);
1535 
1536 	if (ret >= sizeof (dpath)) {
1537 		bam_error(PATH_TOO_LONG, rootbuf);
1538 		set_flag(UPDATE_ERROR);
1539 		return (BAM_ERROR);
1540 	}
1541 
1542 	if (stat(dpath, &sb) == 0 && S_ISDIR(sb.st_mode))
1543 		return (BAM_SUCCESS);
1544 
1545 	if (updt) {
1546 		if (!is_dir_flag_on(what, NO_MULTI))
1547 			if (!bam_nowrite() && mkdirp(dpath, DIR_PERMS) == -1)
1548 				set_dir_flag(what, NO_MULTI);
1549 	} else {
1550 		if (!bam_nowrite() && mkdirp(dpath, DIR_PERMS) == -1) {
1551 			set_flag(UPDATE_ERROR);
1552 			return (BAM_ERROR);
1553 		}
1554 	}
1555 
1556 	set_dir_flag(what, NEED_UPDATE);
1557 	return (BAM_SUCCESS);
1558 }
1559 
1560 #define	DO_CACHE_DIR	0
1561 #define	DO_UPDATE_DIR	1
1562 
1563 #if defined(_LP64) || defined(_LONGLONG_TYPE)
1564 typedef		Elf64_Ehdr	_elfhdr;
1565 #else
1566 typedef		Elf32_Ehdr	_elfhdr;
1567 #endif
1568 
1569 /*
1570  * This routine updates the contents of the cache directory
1571  */
1572 static int
1573 update_dircache(const char *path, int flags)
1574 {
1575 	int rc = BAM_SUCCESS;
1576 
1577 	switch (flags) {
1578 	case FTW_F:
1579 		{
1580 		int	fd;
1581 		_elfhdr	elf;
1582 
1583 		if ((fd = open(path, O_RDONLY)) < 0) {
1584 			bam_error(OPEN_FAIL, path, strerror(errno));
1585 			set_flag(UPDATE_ERROR);
1586 			rc = BAM_ERROR;
1587 			break;
1588 		}
1589 
1590 		/*
1591 		 * libelf and gelf would be a cleaner and easier way to handle
1592 		 * this, but libelf fails compilation if _ILP32 is defined &&
1593 		 * _FILE_OFFSET_BITS is != 32 ...
1594 		 */
1595 		if (read(fd, (void *)&elf, sizeof (_elfhdr)) < 0) {
1596 			bam_error(READ_FAIL, path, strerror(errno));
1597 			set_flag(UPDATE_ERROR);
1598 			(void) close(fd);
1599 			rc = BAM_ERROR;
1600 			break;
1601 		}
1602 		(void) close(fd);
1603 
1604 		/*
1605 		 * If the file is not an executable and is not inside an amd64
1606 		 * directory, we copy it in both the cache directories,
1607 		 * otherwise, we only copy it inside the 64-bit one.
1608 		 */
1609 		if (memcmp(elf.e_ident, ELFMAG, 4) != 0) {
1610 			if (strstr(path, "/amd64")) {
1611 				if (has_cachedir(FILE64))
1612 					rc = dircache_updatefile(path, FILE64);
1613 			} else if (has_cachedir(FILE64)) {
1614 				rc = dircache_updatefile(path, FILE32);
1615 				if (rc == BAM_SUCCESS)
1616 					rc = dircache_updatefile(path, FILE64);
1617 			} else {
1618 				rc = dircache_updatefile(path, FILE32);
1619 			}
1620 		} else {
1621 			/*
1622 			 * Based on the ELF class we copy the file in the 32-bit
1623 			 * or the 64-bit cache directory.
1624 			 */
1625 			if (elf.e_ident[EI_CLASS] == ELFCLASS32)
1626 				rc = dircache_updatefile(path, FILE32);
1627 			else if (elf.e_ident[EI_CLASS] == ELFCLASS64)
1628 				if (has_cachedir(FILE64))
1629 					rc = dircache_updatefile(path, FILE64);
1630 			else {
1631 				bam_print(NO3264ELF, path);
1632 				/* paranoid */
1633 				rc  = dircache_updatefile(path, FILE32);
1634 				if (rc == BAM_SUCCESS)
1635 					rc = dircache_updatefile(path, FILE64);
1636 			}
1637 		}
1638 		break;
1639 		}
1640 	case FTW_D:
1641 		if (strstr(path, "/amd64") == NULL) {
1642 			rc = dircache_updatedir(path, FILE32, DO_UPDATE_DIR);
1643 			if (rc == BAM_SUCCESS)
1644 				rc = dircache_updatedir(path, FILE32,
1645 				    DO_CACHE_DIR);
1646 		} else {
1647 			if (has_cachedir(FILE64)) {
1648 				rc = dircache_updatedir(path, FILE64,
1649 				    DO_UPDATE_DIR);
1650 				if (rc == BAM_SUCCESS)
1651 					rc = dircache_updatedir(path, FILE64,
1652 					    DO_CACHE_DIR);
1653 			}
1654 		}
1655 		break;
1656 	default:
1657 		rc = BAM_ERROR;
1658 		break;
1659 	}
1660 
1661 	return (rc);
1662 }
1663 
1664 /*ARGSUSED*/
1665 static int
1666 cmpstat(
1667 	const char *file,
1668 	const struct stat *st,
1669 	int flags,
1670 	struct FTW *ftw)
1671 {
1672 	uint_t 		sz;
1673 	uint64_t 	*value;
1674 	uint64_t 	filestat[2];
1675 	int 		error, ret;
1676 
1677 	struct safefile *safefilep;
1678 	FILE 		*fp;
1679 	struct stat	sb;
1680 
1681 	/*
1682 	 * On SPARC we create/update links too.
1683 	 */
1684 	if (flags != FTW_F && flags != FTW_D && (flags == FTW_SL &&
1685 	    !is_flag_on(IS_SPARC_TARGET)))
1686 		return (0);
1687 
1688 	/*
1689 	 * Ignore broken links
1690 	 */
1691 	if (flags == FTW_SL && stat(file, &sb) < 0)
1692 		return (0);
1693 
1694 	/*
1695 	 * new_nvlp may be NULL if there were errors earlier
1696 	 * but this is not fatal to update determination.
1697 	 */
1698 	if (walk_arg.new_nvlp) {
1699 		filestat[0] = st->st_size;
1700 		filestat[1] = st->st_mtime;
1701 		error = nvlist_add_uint64_array(walk_arg.new_nvlp,
1702 		    file + bam_rootlen, filestat, 2);
1703 		if (error)
1704 			bam_error(NVADD_FAIL, file, strerror(error));
1705 	}
1706 
1707 	/*
1708 	 * If we are invoked as part of system/filesystem/boot-archive, then
1709 	 * there are a number of things we should not worry about
1710 	 */
1711 	if (bam_smf_check) {
1712 		/* ignore amd64 modules unless we are booted amd64. */
1713 		if (!is_amd64() && strstr(file, "/amd64/") != 0)
1714 			return (0);
1715 
1716 		/* read in list of safe files */
1717 		if (safefiles == NULL)
1718 			if (fp = fopen("/boot/solaris/filelist.safe", "r")) {
1719 				safefiles = s_calloc(1,
1720 				    sizeof (struct safefile));
1721 				safefilep = safefiles;
1722 				safefilep->name = s_calloc(1, MAXPATHLEN +
1723 				    MAXNAMELEN);
1724 				safefilep->next = NULL;
1725 				while (s_fgets(safefilep->name, MAXPATHLEN +
1726 				    MAXNAMELEN, fp) != NULL) {
1727 					safefilep->next = s_calloc(1,
1728 					    sizeof (struct safefile));
1729 					safefilep = safefilep->next;
1730 					safefilep->name = s_calloc(1,
1731 					    MAXPATHLEN + MAXNAMELEN);
1732 					safefilep->next = NULL;
1733 				}
1734 				(void) fclose(fp);
1735 			}
1736 	}
1737 
1738 	/*
1739 	 * On SPARC we create a -path-list file for mkisofs
1740 	 */
1741 	if (is_flag_on(IS_SPARC_TARGET) && !bam_nowrite()) {
1742 		if (flags != FTW_D) {
1743 			char	*strip;
1744 
1745 			strip = (char *)file + strlen(rootbuf);
1746 			(void) fprintf(walk_arg.sparcfile, "/%s=%s\n", strip,
1747 			    file);
1748 		}
1749 	}
1750 
1751 	/*
1752 	 * We are transitioning from the old model to the dircache or the cache
1753 	 * directory was removed: create the entry without further checkings.
1754 	 */
1755 	if (is_flag_on(NEED_CACHE_DIR)) {
1756 		if (bam_verbose)
1757 			bam_print(PARSEABLE_NEW_FILE, file);
1758 
1759 		if (is_flag_on(IS_SPARC_TARGET)) {
1760 			set_dir_flag(FILE64, NEED_UPDATE);
1761 			return (0);
1762 		}
1763 
1764 		ret = update_dircache(file, flags);
1765 		if (ret == BAM_ERROR) {
1766 			bam_error(UPDT_CACHE_FAIL, file);
1767 			return (-1);
1768 		}
1769 
1770 		return (0);
1771 	}
1772 
1773 	/*
1774 	 * We need an update if file doesn't exist in old archive
1775 	 */
1776 	if (walk_arg.old_nvlp == NULL ||
1777 	    nvlist_lookup_uint64_array(walk_arg.old_nvlp,
1778 	    file + bam_rootlen, &value, &sz) != 0) {
1779 		if (bam_smf_check)	/* ignore new during smf check */
1780 			return (0);
1781 
1782 		if (is_flag_on(IS_SPARC_TARGET)) {
1783 			set_dir_flag(FILE64, NEED_UPDATE);
1784 		} else {
1785 			ret = update_dircache(file, flags);
1786 			if (ret == BAM_ERROR) {
1787 				bam_error(UPDT_CACHE_FAIL, file);
1788 				return (-1);
1789 			}
1790 		}
1791 
1792 		if (bam_verbose)
1793 			bam_print(PARSEABLE_NEW_FILE, file);
1794 		return (0);
1795 	}
1796 
1797 	/*
1798 	 * If we got there, the file is already listed as to be included in the
1799 	 * iso image. We just need to know if we are going to rebuild it or not
1800 	 */
1801 
1802 	if (is_flag_on(IS_SPARC_TARGET) &&
1803 	    is_dir_flag_on(FILE64, NEED_UPDATE) && !bam_smf_check)
1804 		return (0);
1805 
1806 	/*
1807 	 * File exists in old archive. Check if file has changed
1808 	 */
1809 	assert(sz == 2);
1810 	bcopy(value, filestat, sizeof (filestat));
1811 
1812 	if (flags != FTW_D && (filestat[0] != st->st_size ||
1813 	    filestat[1] != st->st_mtime)) {
1814 		if (bam_smf_check) {
1815 			safefilep = safefiles;
1816 			while (safefilep != NULL) {
1817 				if (strcmp(file + bam_rootlen,
1818 				    safefilep->name) == 0) {
1819 					(void) creat(NEED_UPDATE_FILE, 0644);
1820 					return (0);
1821 				}
1822 				safefilep = safefilep->next;
1823 			}
1824 		}
1825 
1826 		if (is_flag_on(IS_SPARC_TARGET)) {
1827 			set_dir_flag(FILE64, NEED_UPDATE);
1828 		} else {
1829 			ret = update_dircache(file, flags);
1830 			if (ret == BAM_ERROR) {
1831 				bam_error(UPDT_CACHE_FAIL, file);
1832 				return (-1);
1833 			}
1834 		}
1835 
1836 		if (bam_verbose)
1837 			if (bam_smf_check)
1838 				bam_print("    %s\n", file);
1839 			else
1840 				bam_print(PARSEABLE_OUT_DATE, file);
1841 	}
1842 
1843 	return (0);
1844 }
1845 
1846 /*
1847  * Remove a directory path recursively
1848  */
1849 static int
1850 rmdir_r(char *path)
1851 {
1852 	struct dirent 	*d = NULL;
1853 	DIR 		*dir = NULL;
1854 	char 		tpath[PATH_MAX];
1855 	struct stat 	sb;
1856 
1857 	if ((dir = opendir(path)) == NULL)
1858 		return (-1);
1859 
1860 	while (d = readdir(dir)) {
1861 		if ((strcmp(d->d_name, ".") != 0) &&
1862 		    (strcmp(d->d_name, "..") != 0)) {
1863 			(void) snprintf(tpath, sizeof (tpath), "%s/%s",
1864 			    path, d->d_name);
1865 			if (stat(tpath, &sb) == 0) {
1866 				if (sb.st_mode & S_IFDIR)
1867 					(void) rmdir_r(tpath);
1868 				else
1869 					(void) remove(tpath);
1870 			}
1871 		}
1872 	}
1873 	return (remove(path));
1874 }
1875 
1876 /*
1877  * Check if cache directory exists and, if not, create it and update flags
1878  * accordingly. If the path exists, but it's not a directory, a best effort
1879  * attempt to remove and recreate it is made.
1880  * If the user requested a 'purge', always recreate the directory from scratch.
1881  */
1882 static int
1883 set_cache_dir(char *root, int what)
1884 {
1885 	struct stat	sb;
1886 	int		ret = 0;
1887 
1888 	ret = snprintf(get_cachedir(what), sizeof (get_cachedir(what)),
1889 	    "%s%s%s%s%s", root, ARCHIVE_PREFIX, get_machine(), what == FILE64 ?
1890 	    "/amd64" : "", CACHEDIR_SUFFIX);
1891 
1892 	if (ret >= sizeof (get_cachedir(what))) {
1893 		bam_error(PATH_TOO_LONG, rootbuf);
1894 		return (BAM_ERROR);
1895 	}
1896 
1897 	if (bam_purge)
1898 		(void) rmdir_r(get_cachedir(what));
1899 
1900 	if (stat(get_cachedir(what), &sb) != 0 || !(S_ISDIR(sb.st_mode))) {
1901 		/* best effort unlink attempt, mkdir will catch errors */
1902 		(void) unlink(get_cachedir(what));
1903 
1904 		if (bam_verbose)
1905 			bam_print(UPDATE_CDIR_MISS, get_cachedir(what));
1906 		ret = mkdir(get_cachedir(what), DIR_PERMS);
1907 		if (ret < 0) {
1908 			bam_error(MKDIR_FAILED, get_cachedir(what),
1909 			    strerror(errno));
1910 			get_cachedir(what)[0] = '\0';
1911 			return (ret);
1912 		}
1913 		set_flag(NEED_CACHE_DIR);
1914 		set_dir_flag(what, NO_MULTI);
1915 	}
1916 
1917 	return (BAM_SUCCESS);
1918 }
1919 
1920 static int
1921 set_update_dir(char *root, int what)
1922 {
1923 	struct stat	sb;
1924 	int		ret;
1925 
1926 	if (is_dir_flag_on(what, NO_MULTI))
1927 		return (BAM_SUCCESS);
1928 
1929 	if (!bam_extend) {
1930 		set_dir_flag(what, NO_MULTI);
1931 		return (BAM_SUCCESS);
1932 	}
1933 
1934 	if (what == FILE64 && !is_flag_on(IS_SPARC_TARGET))
1935 		ret = snprintf(get_updatedir(what),
1936 		    sizeof (get_updatedir(what)), "%s%s%s/amd64%s", root,
1937 		    ARCHIVE_PREFIX, get_machine(), UPDATEDIR_SUFFIX);
1938 	else
1939 		ret = snprintf(get_updatedir(what),
1940 		    sizeof (get_updatedir(what)), "%s%s%s%s", root,
1941 		    ARCHIVE_PREFIX, get_machine(), UPDATEDIR_SUFFIX);
1942 
1943 	if (ret >= sizeof (get_updatedir(what))) {
1944 		bam_error(PATH_TOO_LONG, rootbuf);
1945 		return (BAM_ERROR);
1946 	}
1947 
1948 	if (stat(get_updatedir(what), &sb) == 0) {
1949 		if (S_ISDIR(sb.st_mode))
1950 			ret = rmdir_r(get_updatedir(what));
1951 		else
1952 			ret = unlink(get_updatedir(what));
1953 
1954 		if (ret != 0)
1955 			set_dir_flag(what, NO_MULTI);
1956 	}
1957 
1958 	if (mkdir(get_updatedir(what), DIR_PERMS) < 0)
1959 		set_dir_flag(what, NO_MULTI);
1960 
1961 	return (BAM_SUCCESS);
1962 }
1963 
1964 static int
1965 is_valid_archive(char *root, int what)
1966 {
1967 	char 		archive_path[PATH_MAX];
1968 	struct stat 	sb;
1969 	int 		ret;
1970 
1971 	if (what == FILE64 && !is_flag_on(IS_SPARC_TARGET))
1972 		ret = snprintf(archive_path, sizeof (archive_path),
1973 		    "%s%s%s/amd64%s", root, ARCHIVE_PREFIX, get_machine(),
1974 		    ARCHIVE_SUFFIX);
1975 	else
1976 		ret = snprintf(archive_path, sizeof (archive_path), "%s%s%s%s",
1977 		    root, ARCHIVE_PREFIX, get_machine(), ARCHIVE_SUFFIX);
1978 
1979 	if (ret >= sizeof (archive_path)) {
1980 		bam_error(PATH_TOO_LONG, rootbuf);
1981 		return (BAM_ERROR);
1982 	}
1983 
1984 	if (stat(archive_path, &sb) != 0) {
1985 		if (bam_verbose && !bam_check)
1986 			bam_print(UPDATE_ARCH_MISS, archive_path);
1987 		set_dir_flag(what, NEED_UPDATE);
1988 		set_dir_flag(what, NO_MULTI);
1989 		return (BAM_SUCCESS);
1990 	}
1991 
1992 	if (is_flag_on(IS_SPARC_TARGET))
1993 		return (BAM_SUCCESS);
1994 
1995 	if (bam_extend && sb.st_size > BA_SIZE_MAX) {
1996 		if (bam_verbose && !bam_check)
1997 			bam_print(MULTI_SIZE, archive_path, BA_SIZE_MAX);
1998 		set_dir_flag(what, NO_MULTI);
1999 	}
2000 
2001 	return (BAM_SUCCESS);
2002 }
2003 
2004 /*
2005  * Check flags and presence of required files and directories.
2006  * The force flag and/or absence of files should
2007  * trigger an update.
2008  * Suppress stdout output if check (-n) option is set
2009  * (as -n should only produce parseable output.)
2010  */
2011 static int
2012 check_flags_and_files(char *root)
2013 {
2014 
2015 	struct stat 	sb;
2016 	int 		ret;
2017 
2018 	/*
2019 	 * If archive is missing, create archive
2020 	 */
2021 	if (is_flag_on(IS_SPARC_TARGET)) {
2022 		ret = is_valid_archive(root, FILE64);
2023 	} else {
2024 		int	what = FILE32;
2025 		do {
2026 			ret = is_valid_archive(root, what);
2027 			if (ret == BAM_ERROR)
2028 				return (BAM_ERROR);
2029 			what++;
2030 		} while (bam_direct == BAM_DIRECT_DBOOT && what < CACHEDIR_NUM);
2031 	}
2032 
2033 	if (bam_nowrite())
2034 		return (BAM_SUCCESS);
2035 
2036 
2037 	/*
2038 	 * check if cache directories exist on x86.
2039 	 * check (and always open) the cache file on SPARC.
2040 	 */
2041 	if (is_sparc()) {
2042 		ret = snprintf(get_cachedir(FILE64),
2043 		    sizeof (get_cachedir(FILE64)), "%s%s%s/%s", root,
2044 		    ARCHIVE_PREFIX, get_machine(), CACHEDIR_SUFFIX);
2045 
2046 		if (ret >= sizeof (get_cachedir(FILE64))) {
2047 			bam_error(PATH_TOO_LONG, rootbuf);
2048 			return (BAM_ERROR);
2049 		}
2050 
2051 		if (stat(get_cachedir(FILE64), &sb) != 0) {
2052 			set_flag(NEED_CACHE_DIR);
2053 			set_dir_flag(FILE64, NEED_UPDATE);
2054 		}
2055 
2056 		walk_arg.sparcfile = fopen(get_cachedir(FILE64), "w");
2057 		if (walk_arg.sparcfile == NULL) {
2058 			bam_error(OPEN_FAIL, get_cachedir(FILE64),
2059 			    strerror(errno));
2060 			return (BAM_ERROR);
2061 		}
2062 
2063 		set_dir_present(FILE64);
2064 	} else {
2065 		int	what = FILE32;
2066 
2067 		do {
2068 			if (set_cache_dir(root, what) != 0)
2069 				return (BAM_ERROR);
2070 
2071 			set_dir_present(what);
2072 
2073 			if (set_update_dir(root, what) != 0)
2074 				return (BAM_ERROR);
2075 			what++;
2076 		} while (bam_direct == BAM_DIRECT_DBOOT && what < CACHEDIR_NUM);
2077 	}
2078 
2079 	/*
2080 	 * if force, create archive unconditionally
2081 	 */
2082 	if (bam_force) {
2083 		if (!is_sparc())
2084 			set_dir_flag(FILE32, NEED_UPDATE);
2085 		set_dir_flag(FILE64, NEED_UPDATE);
2086 		if (bam_verbose)
2087 			bam_print(UPDATE_FORCE);
2088 		return (BAM_SUCCESS);
2089 	}
2090 
2091 	return (BAM_SUCCESS);
2092 }
2093 
2094 static error_t
2095 read_one_list(char *root, filelist_t  *flistp, char *filelist)
2096 {
2097 	char 		path[PATH_MAX];
2098 	FILE 		*fp;
2099 	char 		buf[BAM_MAXLINE];
2100 	const char 	*fcn = "read_one_list()";
2101 
2102 	(void) snprintf(path, sizeof (path), "%s%s", root, filelist);
2103 
2104 	fp = fopen(path, "r");
2105 	if (fp == NULL) {
2106 		BAM_DPRINTF((D_FLIST_FAIL, fcn, path, strerror(errno)));
2107 		return (BAM_ERROR);
2108 	}
2109 	while (s_fgets(buf, sizeof (buf), fp) != NULL) {
2110 		/* skip blank lines */
2111 		if (strspn(buf, " \t") == strlen(buf))
2112 			continue;
2113 		append_to_flist(flistp, buf);
2114 	}
2115 	if (fclose(fp) != 0) {
2116 		bam_error(CLOSE_FAIL, path, strerror(errno));
2117 		return (BAM_ERROR);
2118 	}
2119 	return (BAM_SUCCESS);
2120 }
2121 
2122 static error_t
2123 read_list(char *root, filelist_t  *flistp)
2124 {
2125 	char 		path[PATH_MAX];
2126 	char 		cmd[PATH_MAX];
2127 	struct stat 	sb;
2128 	int 		n, rval;
2129 	const char 	*fcn = "read_list()";
2130 
2131 	flistp->head = flistp->tail = NULL;
2132 
2133 	/*
2134 	 * build and check path to extract_boot_filelist.ksh
2135 	 */
2136 	n = snprintf(path, sizeof (path), "%s%s", root, EXTRACT_BOOT_FILELIST);
2137 	if (n >= sizeof (path)) {
2138 		bam_error(NO_FLIST);
2139 		return (BAM_ERROR);
2140 	}
2141 
2142 	/*
2143 	 * If extract_boot_filelist is present, exec it, otherwise read
2144 	 * the filelists directly, for compatibility with older images.
2145 	 */
2146 	if (stat(path, &sb) == 0) {
2147 		/*
2148 		 * build arguments to exec extract_boot_filelist.ksh
2149 		 */
2150 		char *rootarg, *platarg;
2151 		int platarglen = 1, rootarglen = 1;
2152 		if (strlen(root) > 1)
2153 			rootarglen += strlen(root) + strlen("-R ");
2154 		if (bam_alt_platform)
2155 			platarglen += strlen(bam_platform) + strlen("-p ");
2156 		platarg = s_calloc(1, platarglen);
2157 		rootarg = s_calloc(1, rootarglen);
2158 		*platarg = 0;
2159 		*rootarg = 0;
2160 
2161 		if (strlen(root) > 1) {
2162 			(void) snprintf(rootarg, rootarglen,
2163 			    "-R %s", root);
2164 		}
2165 		if (bam_alt_platform) {
2166 			(void) snprintf(platarg, platarglen,
2167 			    "-p %s", bam_platform);
2168 		}
2169 		n = snprintf(cmd, sizeof (cmd), "%s %s %s /%s /%s",
2170 		    path, rootarg, platarg, BOOT_FILE_LIST, ETC_FILE_LIST);
2171 		free(platarg);
2172 		free(rootarg);
2173 		if (n >= sizeof (cmd)) {
2174 			bam_error(NO_FLIST);
2175 			return (BAM_ERROR);
2176 		}
2177 		if (exec_cmd(cmd, flistp) != 0) {
2178 			BAM_DPRINTF((D_FLIST_FAIL, fcn, path, strerror(errno)));
2179 			return (BAM_ERROR);
2180 		}
2181 	} else {
2182 		/*
2183 		 * Read current lists of files - only the first is mandatory
2184 		 */
2185 		rval = read_one_list(root, flistp, BOOT_FILE_LIST);
2186 		if (rval != BAM_SUCCESS)
2187 			return (rval);
2188 		(void) read_one_list(root, flistp, ETC_FILE_LIST);
2189 	}
2190 
2191 	if (flistp->head == NULL) {
2192 		bam_error(NO_FLIST);
2193 		return (BAM_ERROR);
2194 	}
2195 
2196 	return (BAM_SUCCESS);
2197 }
2198 
2199 static void
2200 getoldstat(char *root)
2201 {
2202 	char 		path[PATH_MAX];
2203 	int 		fd, error;
2204 	struct stat 	sb;
2205 	char 		*ostat;
2206 
2207 	(void) snprintf(path, sizeof (path), "%s%s", root, FILE_STAT);
2208 	fd = open(path, O_RDONLY);
2209 	if (fd == -1) {
2210 		if (bam_verbose)
2211 			bam_print(OPEN_FAIL, path, strerror(errno));
2212 		goto out_err;
2213 	}
2214 
2215 	if (fstat(fd, &sb) != 0) {
2216 		bam_error(STAT_FAIL, path, strerror(errno));
2217 		goto out_err;
2218 	}
2219 
2220 	ostat = s_calloc(1, sb.st_size);
2221 
2222 	if (read(fd, ostat, sb.st_size) != sb.st_size) {
2223 		bam_error(READ_FAIL, path, strerror(errno));
2224 		free(ostat);
2225 		goto out_err;
2226 	}
2227 
2228 	(void) close(fd);
2229 	fd = -1;
2230 
2231 	walk_arg.old_nvlp = NULL;
2232 	error = nvlist_unpack(ostat, sb.st_size, &walk_arg.old_nvlp, 0);
2233 
2234 	free(ostat);
2235 
2236 	if (error) {
2237 		bam_error(UNPACK_FAIL, path, strerror(error));
2238 		walk_arg.old_nvlp = NULL;
2239 		goto out_err;
2240 	} else {
2241 		return;
2242 	}
2243 
2244 out_err:
2245 	if (fd != -1)
2246 		(void) close(fd);
2247 	set_dir_flag(FILE32, NEED_UPDATE);
2248 	set_dir_flag(FILE64, NEED_UPDATE);
2249 }
2250 
2251 /* Best effort stale entry removal */
2252 static void
2253 delete_stale(char *file, int what)
2254 {
2255 	char		path[PATH_MAX];
2256 	struct stat	sb;
2257 
2258 	(void) snprintf(path, sizeof (path), "%s/%s", get_cachedir(what), file);
2259 	if (!bam_check && stat(path, &sb) == 0) {
2260 		if (sb.st_mode & S_IFDIR)
2261 			(void) rmdir_r(path);
2262 		else
2263 			(void) unlink(path);
2264 
2265 		set_dir_flag(what, (NEED_UPDATE | NO_MULTI));
2266 	}
2267 	if (bam_verbose)
2268 		bam_print(PARSEABLE_STALE_FILE, path);
2269 }
2270 
2271 /*
2272  * Checks if a file in the current (old) archive has
2273  * been deleted from the root filesystem. This is needed for
2274  * software like Trusted Extensions (TX) that switch early
2275  * in boot based on presence/absence of a kernel module.
2276  */
2277 static void
2278 check4stale(char *root)
2279 {
2280 	nvpair_t	*nvp;
2281 	nvlist_t	*nvlp;
2282 	char 		*file;
2283 	char		path[PATH_MAX];
2284 
2285 	/*
2286 	 * Skip stale file check during smf check
2287 	 */
2288 	if (bam_smf_check)
2289 		return;
2290 
2291 	/*
2292 	 * If we need to (re)create the cache, there's no need to check for
2293 	 * stale files
2294 	 */
2295 	if (is_flag_on(NEED_CACHE_DIR))
2296 		return;
2297 
2298 	/* Nothing to do if no old stats */
2299 	if ((nvlp = walk_arg.old_nvlp) == NULL)
2300 		return;
2301 
2302 	for (nvp = nvlist_next_nvpair(nvlp, NULL); nvp;
2303 	    nvp = nvlist_next_nvpair(nvlp, nvp)) {
2304 		file = nvpair_name(nvp);
2305 		if (file == NULL)
2306 			continue;
2307 		(void) snprintf(path, sizeof (path), "%s/%s",
2308 		    root, file);
2309 		if (access(path, F_OK) < 0) {
2310 			int	what;
2311 
2312 			if (is_flag_on(IS_SPARC_TARGET))
2313 				set_dir_flag(FILE64, NEED_UPDATE);
2314 
2315 			for (what = FILE32; what < CACHEDIR_NUM; what++)
2316 				if (has_cachedir(what))
2317 					delete_stale(file, what);
2318 		}
2319 	}
2320 }
2321 
2322 static void
2323 create_newstat(void)
2324 {
2325 	int error;
2326 
2327 	error = nvlist_alloc(&walk_arg.new_nvlp, NV_UNIQUE_NAME, 0);
2328 	if (error) {
2329 		/*
2330 		 * Not fatal - we can still create archive
2331 		 */
2332 		walk_arg.new_nvlp = NULL;
2333 		bam_error(NVALLOC_FAIL, strerror(error));
2334 	}
2335 }
2336 
2337 static int
2338 walk_list(char *root, filelist_t *flistp)
2339 {
2340 	char path[PATH_MAX];
2341 	line_t *lp;
2342 
2343 	for (lp = flistp->head; lp; lp = lp->next) {
2344 		/*
2345 		 * Don't follow symlinks.  A symlink must refer to
2346 		 * a file that would appear in the archive through
2347 		 * a direct reference.  This matches the archive
2348 		 * construction behavior.
2349 		 */
2350 		(void) snprintf(path, sizeof (path), "%s%s", root, lp->line);
2351 		if (nftw(path, cmpstat, 20, FTW_PHYS) == -1) {
2352 			if (is_flag_on(UPDATE_ERROR))
2353 				return (BAM_ERROR);
2354 			/*
2355 			 * Some files may not exist.
2356 			 * For example: etc/rtc_config on a x86 diskless system
2357 			 * Emit verbose message only
2358 			 */
2359 			if (bam_verbose)
2360 				bam_print(NFTW_FAIL, path, strerror(errno));
2361 		}
2362 	}
2363 
2364 	return (BAM_SUCCESS);
2365 }
2366 
2367 static void
2368 savenew(char *root)
2369 {
2370 	char 	path[PATH_MAX];
2371 	char 	path2[PATH_MAX];
2372 	size_t 	sz;
2373 	char 	*nstat;
2374 	int 	fd, wrote, error;
2375 
2376 	nstat = NULL;
2377 	sz = 0;
2378 	error = nvlist_pack(walk_arg.new_nvlp, &nstat, &sz,
2379 	    NV_ENCODE_XDR, 0);
2380 	if (error) {
2381 		bam_error(PACK_FAIL, strerror(error));
2382 		return;
2383 	}
2384 
2385 	(void) snprintf(path, sizeof (path), "%s%s", root, FILE_STAT_TMP);
2386 	fd = open(path, O_RDWR|O_CREAT|O_TRUNC, FILE_STAT_MODE);
2387 	if (fd == -1) {
2388 		bam_error(OPEN_FAIL, path, strerror(errno));
2389 		free(nstat);
2390 		return;
2391 	}
2392 	wrote = write(fd, nstat, sz);
2393 	if (wrote != sz) {
2394 		bam_error(WRITE_FAIL, path, strerror(errno));
2395 		(void) close(fd);
2396 		free(nstat);
2397 		return;
2398 	}
2399 	(void) close(fd);
2400 	free(nstat);
2401 
2402 	(void) snprintf(path2, sizeof (path2), "%s%s", root, FILE_STAT);
2403 	if (rename(path, path2) != 0) {
2404 		bam_error(RENAME_FAIL, path2, strerror(errno));
2405 	}
2406 }
2407 
2408 #define	init_walk_args()	bzero(&walk_arg, sizeof (walk_arg))
2409 
2410 static void
2411 clear_walk_args(void)
2412 {
2413 	if (walk_arg.old_nvlp)
2414 		nvlist_free(walk_arg.old_nvlp);
2415 	if (walk_arg.new_nvlp)
2416 		nvlist_free(walk_arg.new_nvlp);
2417 	if (walk_arg.sparcfile)
2418 		(void) fclose(walk_arg.sparcfile);
2419 	walk_arg.old_nvlp = NULL;
2420 	walk_arg.new_nvlp = NULL;
2421 	walk_arg.sparcfile = NULL;
2422 }
2423 
2424 /*
2425  * Returns:
2426  *	0 - no update necessary
2427  *	1 - update required.
2428  *	BAM_ERROR (-1) - An error occurred
2429  *
2430  * Special handling for check (-n):
2431  * ================================
2432  * The check (-n) option produces parseable output.
2433  * To do this, we suppress all stdout messages unrelated
2434  * to out of sync files.
2435  * All stderr messages are still printed though.
2436  *
2437  */
2438 static int
2439 update_required(char *root)
2440 {
2441 	struct stat 	sb;
2442 	char 		path[PATH_MAX];
2443 	filelist_t 	flist;
2444 	filelist_t 	*flistp = &flist;
2445 	int 		ret;
2446 
2447 	flistp->head = flistp->tail = NULL;
2448 
2449 	if (is_sparc())
2450 		set_flag(IS_SPARC_TARGET);
2451 
2452 	/*
2453 	 * Check if cache directories and archives are present
2454 	 */
2455 
2456 	ret = check_flags_and_files(root);
2457 	if (ret < 0)
2458 		return (BAM_ERROR);
2459 
2460 	/*
2461 	 * In certain deployment scenarios, filestat may not
2462 	 * exist. Ignore it during boot-archive SMF check.
2463 	 */
2464 	if (bam_smf_check) {
2465 		(void) snprintf(path, sizeof (path), "%s%s", root, FILE_STAT);
2466 		if (stat(path, &sb) != 0)
2467 			return (0);
2468 	}
2469 
2470 	getoldstat(root);
2471 
2472 	/*
2473 	 * Check if the archive contains files that are no longer
2474 	 * present on the root filesystem.
2475 	 */
2476 	check4stale(root);
2477 
2478 	/*
2479 	 * read list of files
2480 	 */
2481 	if (read_list(root, flistp) != BAM_SUCCESS) {
2482 		clear_walk_args();
2483 		return (BAM_ERROR);
2484 	}
2485 
2486 	assert(flistp->head && flistp->tail);
2487 
2488 	/*
2489 	 * At this point either the update is required
2490 	 * or the decision is pending. In either case
2491 	 * we need to create new stat nvlist
2492 	 */
2493 	create_newstat();
2494 	/*
2495 	 * This walk does 2 things:
2496 	 *  	- gets new stat data for every file
2497 	 *	- (optional) compare old and new stat data
2498 	 */
2499 	ret = walk_list(root, &flist);
2500 
2501 	/* done with the file list */
2502 	filelist_free(flistp);
2503 
2504 	/* something went wrong */
2505 
2506 	if (ret == BAM_ERROR) {
2507 		bam_error(CACHE_FAIL);
2508 		return (BAM_ERROR);
2509 	}
2510 
2511 	if (walk_arg.new_nvlp == NULL) {
2512 		(void) fclose(walk_arg.sparcfile);
2513 		bam_error(NO_NEW_STAT);
2514 	}
2515 
2516 	/* If nothing was updated, discard newstat. */
2517 
2518 	if (!is_dir_flag_on(FILE32, NEED_UPDATE) &&
2519 	    !is_dir_flag_on(FILE64, NEED_UPDATE)) {
2520 		clear_walk_args();
2521 		return (0);
2522 	}
2523 
2524 	(void) fclose(walk_arg.sparcfile);
2525 
2526 	return (1);
2527 }
2528 
2529 /*
2530  * Returns 1 if we're dealing with a bfu archive update, 0 otherwise
2531  */
2532 static int
2533 is_bfu()
2534 {
2535 	char *path;
2536 
2537 	path = getenv("PATH");
2538 	if (path != NULL &&
2539 	    strncmp(path, "/tmp/bfubin", strlen("/tmp/bfubin")) == 0) {
2540 		struct stat	sb;
2541 		if (stat("/tmp/bfubin", &sb) == 0) {
2542 			if (!(sb.st_mode & S_IFDIR))
2543 				return (0);
2544 			if (sb.st_uid != getuid())
2545 				return (0);
2546 			return (1);
2547 		}
2548 	}
2549 	return (0);
2550 }
2551 
2552 #define	LOCKFS_PATH	(is_bfu() ? LOCKFS_BFU : LOCKFS_BIN)
2553 
2554 static int
2555 flushfs(char *root)
2556 {
2557 	char	cmd[PATH_MAX + 30];
2558 
2559 	(void) snprintf(cmd, sizeof (cmd), "%s -f \"%s\" 2>/dev/null",
2560 	    LOCKFS_PATH, root);
2561 
2562 	return (exec_cmd(cmd, NULL));
2563 }
2564 
2565 static int
2566 do_archive_copy(char *source, char *dest)
2567 {
2568 
2569 	(void) sync();
2570 
2571 	/* the equivalent of mv archive-new-$pid boot_archive */
2572 	if (rename(source, dest) != 0)
2573 		return (BAM_ERROR);
2574 
2575 	if (flushfs(bam_root) != 0)
2576 		(void) sync();
2577 
2578 	return (BAM_SUCCESS);
2579 }
2580 
2581 static int
2582 check_cmdline(filelist_t flist)
2583 {
2584 	line_t	*lp;
2585 
2586 	for (lp = flist.head; lp; lp = lp->next) {
2587 		if (strstr(lp->line, "Error:") != NULL ||
2588 		    strstr(lp->line, "Inode number overflow") != NULL) {
2589 			(void) fprintf(stderr, "%s", lp->line);
2590 			return (BAM_ERROR);
2591 		}
2592 	}
2593 
2594 	return (BAM_SUCCESS);
2595 }
2596 
2597 static int
2598 check_archive(char *dest)
2599 {
2600 	struct stat	sb;
2601 
2602 	if (stat(dest, &sb) != 0 || !S_ISREG(sb.st_mode) ||
2603 	    sb.st_size < 10000) {
2604 		bam_error(ARCHIVE_BAD, dest);
2605 		(void) unlink(dest);
2606 		return (BAM_ERROR);
2607 	}
2608 
2609 	return (BAM_SUCCESS);
2610 }
2611 
2612 /*
2613  * Returns 1 if mkiso is in the expected PATH, 0 otherwise
2614  */
2615 static int
2616 is_mkisofs()
2617 {
2618 	if (is_bfu()) {
2619 		if (access(MKISOFS_BFUBIN, X_OK) == 0)
2620 			return (1);
2621 	} else {
2622 		if (access(MKISOFS_BIN, X_OK) == 0)
2623 			return (1);
2624 	}
2625 	return (0);
2626 }
2627 
2628 #define	MKISO_PARAMS	" -quiet -graft-points -dlrDJN -relaxed-filenames "
2629 #define	MKISO_PATH	(is_bfu() ? MKISOFS_BFUBIN : MKISOFS_BIN)
2630 #define	DD_PATH		(is_bfu() ? DD_PATH_BFU : DD_PATH_USR)
2631 
2632 static int
2633 create_sparc_archive(char *archive, char *tempname, char *bootblk, char *list)
2634 {
2635 	int		ret;
2636 	char		cmdline[3 * PATH_MAX + 64];
2637 	filelist_t	flist = {0};
2638 	const char	*func = "create_sparc_archive()";
2639 
2640 	if (access(bootblk, R_OK) == 1) {
2641 		bam_error(BOOTBLK_FAIL, bootblk);
2642 		return (BAM_ERROR);
2643 	}
2644 
2645 	/*
2646 	 * Prepare mkisofs command line and execute it
2647 	 */
2648 	(void) snprintf(cmdline, sizeof (cmdline), "%s %s -G %s -o \"%s\" "
2649 	    "-path-list \"%s\" 2>&1", MKISO_PATH, MKISO_PARAMS, bootblk,
2650 	    tempname, list);
2651 
2652 	BAM_DPRINTF((D_CMDLINE, func, cmdline));
2653 
2654 	ret = exec_cmd(cmdline, &flist);
2655 	if (ret != 0 || check_cmdline(flist) == BAM_ERROR)
2656 		goto out_err;
2657 
2658 	filelist_free(&flist);
2659 
2660 	/*
2661 	 * Prepare dd command line to copy the bootblk on the new archive and
2662 	 * execute it
2663 	 */
2664 	(void) snprintf(cmdline, sizeof (cmdline), "%s if=\"%s\" of=\"%s\""
2665 	    " bs=1b oseek=1 count=15 conv=notrunc conv=sync 2>&1", DD_PATH,
2666 	    bootblk, tempname);
2667 
2668 	BAM_DPRINTF((D_CMDLINE, func, cmdline));
2669 
2670 	ret = exec_cmd(cmdline, &flist);
2671 	if (ret != 0 || check_cmdline(flist) == BAM_ERROR)
2672 		goto out_err;
2673 
2674 	filelist_free(&flist);
2675 
2676 	/* Did we get a valid archive ? */
2677 	if (check_archive(tempname) == BAM_ERROR)
2678 		return (BAM_ERROR);
2679 
2680 	return (do_archive_copy(tempname, archive));
2681 
2682 out_err:
2683 	filelist_free(&flist);
2684 	bam_error(ARCHIVE_FAIL, cmdline);
2685 	(void) unlink(tempname);
2686 	return (BAM_ERROR);
2687 }
2688 
2689 static unsigned int
2690 from_733(unsigned char *s)
2691 {
2692 	int		i;
2693 	unsigned int	ret = 0;
2694 
2695 	for (i = 0; i < 4; i++)
2696 		ret |= s[i] << (8 * i);
2697 
2698 	return (ret);
2699 }
2700 
2701 static void
2702 to_733(unsigned char *s, unsigned int val)
2703 {
2704 	int	i;
2705 
2706 	for (i = 0; i < 4; i++)
2707 		s[i] = s[7-i] = (val >> (8 * i)) & 0xFF;
2708 }
2709 
2710 /*
2711  * Extends the current boot archive without recreating it from scratch
2712  */
2713 static int
2714 extend_iso_archive(char *archive, char *tempname, char *update_dir)
2715 {
2716 	int			fd = -1, newfd = -1, ret, i;
2717 	int			next_session = 0, new_size = 0;
2718 	char			cmdline[3 * PATH_MAX + 64];
2719 	const char		*func = "extend_iso_archive()";
2720 	filelist_t		flist = {0};
2721 	struct iso_pdesc	saved_desc[MAX_IVDs];
2722 
2723 	fd = open(archive, O_RDWR);
2724 	if (fd == -1) {
2725 		bam_error(OPEN_FAIL, archive, strerror(errno));
2726 		goto out_err;
2727 	}
2728 
2729 	/*
2730 	 * A partial read is likely due to a corrupted file
2731 	 */
2732 	ret = pread64(fd, saved_desc, sizeof (saved_desc),
2733 	    VOLDESC_OFF * CD_BLOCK);
2734 	if (ret != sizeof (saved_desc)) {
2735 		bam_error(READ_FAIL, archive, strerror(errno));
2736 		goto out_err;
2737 	}
2738 
2739 	if (memcmp(saved_desc[0].type, "\1CD001", 6)) {
2740 		bam_error(SIGN_FAIL, archive);
2741 		goto out_err;
2742 	}
2743 
2744 	/*
2745 	 * Read primary descriptor and locate next_session offset (it should
2746 	 * point to the end of the archive)
2747 	 */
2748 	next_session = P2ROUNDUP(from_733(saved_desc[0].volume_space_size), 16);
2749 
2750 	(void) snprintf(cmdline, sizeof (cmdline), "%s -C 16,%d -M %s %s -o \""
2751 	    "%s\" \"%s\" 2>&1", MKISO_PATH, next_session, archive, MKISO_PARAMS,
2752 	    tempname, update_dir);
2753 
2754 	BAM_DPRINTF((D_CMDLINE, func, cmdline));
2755 
2756 	ret = exec_cmd(cmdline, &flist);
2757 	if (ret != 0 || check_cmdline(flist) == BAM_ERROR) {
2758 		bam_error(MULTI_FAIL, cmdline);
2759 		goto out_flist_err;
2760 	}
2761 	filelist_free(&flist);
2762 
2763 	newfd = open(tempname, O_RDONLY);
2764 	if (newfd == -1) {
2765 		bam_error(OPEN_FAIL, archive, strerror(errno));
2766 		goto out_err;
2767 	}
2768 
2769 	ret = pread64(newfd, saved_desc, sizeof (saved_desc),
2770 	    VOLDESC_OFF * CD_BLOCK);
2771 	if (ret != sizeof (saved_desc)) {
2772 		bam_error(READ_FAIL, archive, strerror(errno));
2773 		goto out_err;
2774 	}
2775 
2776 	if (memcmp(saved_desc[0].type, "\1CD001", 6)) {
2777 		bam_error(SIGN_FAIL, archive);
2778 		goto out_err;
2779 	}
2780 
2781 	new_size = from_733(saved_desc[0].volume_space_size) + next_session;
2782 	to_733(saved_desc[0].volume_space_size, new_size);
2783 
2784 	for (i = 1; i < MAX_IVDs; i++) {
2785 		if (saved_desc[i].type[0] == (unsigned char)255)
2786 			break;
2787 		if (memcmp(saved_desc[i].id, "CD001", 5))
2788 			break;
2789 
2790 		if (bam_verbose)
2791 			bam_print("%s: Updating descriptor entry [%d]\n", func,
2792 			    i);
2793 
2794 		to_733(saved_desc[i].volume_space_size, new_size);
2795 	}
2796 
2797 	ret = pwrite64(fd, saved_desc, DVD_BLOCK, VOLDESC_OFF*CD_BLOCK);
2798 	if (ret != DVD_BLOCK) {
2799 		bam_error(WRITE_FAIL, archive, strerror(errno));
2800 		goto out_err;
2801 	}
2802 	(void) close(newfd);
2803 	newfd = -1;
2804 
2805 	ret = close(fd);
2806 	if (ret != 0) {
2807 		bam_error(CLOSE_FAIL, archive, strerror(errno));
2808 		return (BAM_ERROR);
2809 	}
2810 	fd = -1;
2811 
2812 	(void) snprintf(cmdline, sizeof (cmdline), "%s if=%s of=%s bs=32k "
2813 	    "seek=%d conv=sync 2>&1", DD_PATH, tempname, archive,
2814 	    (next_session/16));
2815 
2816 	BAM_DPRINTF((D_CMDLINE, func, cmdline));
2817 
2818 	ret = exec_cmd(cmdline, &flist);
2819 	if (ret != 0 || check_cmdline(flist) == BAM_ERROR) {
2820 		bam_error(MULTI_FAIL, cmdline);
2821 		goto out_flist_err;
2822 	}
2823 	filelist_free(&flist);
2824 
2825 	(void) unlink(tempname);
2826 
2827 	if (bam_verbose)
2828 		bam_print("boot archive updated successfully\n");
2829 
2830 	return (BAM_SUCCESS);
2831 
2832 out_flist_err:
2833 	filelist_free(&flist);
2834 out_err:
2835 	if (fd != -1)
2836 		(void) close(fd);
2837 	if (newfd != -1)
2838 		(void) close(newfd);
2839 	return (BAM_ERROR);
2840 }
2841 
2842 static int
2843 create_x86_archive(char *archive, char *tempname, char *update_dir)
2844 {
2845 	int		ret;
2846 	char		cmdline[3 * PATH_MAX + 64];
2847 	filelist_t	flist = {0};
2848 	const char	*func = "create_x86_archive()";
2849 
2850 	(void) snprintf(cmdline, sizeof (cmdline), "%s %s -o \"%s\" \"%s\" "
2851 	    "2>&1", MKISO_PATH, MKISO_PARAMS, tempname, update_dir);
2852 
2853 	BAM_DPRINTF((D_CMDLINE, func, cmdline));
2854 
2855 	ret = exec_cmd(cmdline, &flist);
2856 	if (ret != 0 || check_cmdline(flist) == BAM_ERROR) {
2857 		bam_error(ARCHIVE_FAIL, cmdline);
2858 		filelist_free(&flist);
2859 		(void) unlink(tempname);
2860 		return (BAM_ERROR);
2861 	}
2862 
2863 	filelist_free(&flist);
2864 
2865 	if (check_archive(tempname) == BAM_ERROR)
2866 		return (BAM_ERROR);
2867 
2868 	return (do_archive_copy(tempname, archive));
2869 }
2870 
2871 static int
2872 mkisofs_archive(char *root, int what)
2873 {
2874 	int		ret;
2875 	char		temp[PATH_MAX];
2876 	char 		bootblk[PATH_MAX];
2877 	char		boot_archive[PATH_MAX];
2878 
2879 	if (what == FILE64 && !is_flag_on(IS_SPARC_TARGET))
2880 		ret = snprintf(temp, sizeof (temp),
2881 		    "%s%s%s/amd64/archive-new-%d", root, ARCHIVE_PREFIX,
2882 		    get_machine(), getpid());
2883 	else
2884 		ret = snprintf(temp, sizeof (temp), "%s%s%s/archive-new-%d",
2885 		    root, ARCHIVE_PREFIX, get_machine(), getpid());
2886 
2887 	if (ret >= sizeof (temp))
2888 		goto out_path_err;
2889 
2890 	if (what == FILE64 && !is_flag_on(IS_SPARC_TARGET))
2891 		ret = snprintf(boot_archive, sizeof (boot_archive),
2892 		    "%s%s%s/amd64%s", root, ARCHIVE_PREFIX, get_machine(),
2893 		    ARCHIVE_SUFFIX);
2894 	else
2895 		ret = snprintf(boot_archive, sizeof (boot_archive),
2896 		    "%s%s%s%s", root, ARCHIVE_PREFIX, get_machine(),
2897 		    ARCHIVE_SUFFIX);
2898 
2899 	if (ret >= sizeof (boot_archive))
2900 		goto out_path_err;
2901 
2902 	bam_print("updating %s\n", boot_archive);
2903 
2904 	if (is_flag_on(IS_SPARC_TARGET)) {
2905 		ret = snprintf(bootblk, sizeof (bootblk),
2906 		    "%s/platform/%s/lib/fs/hsfs/bootblk", root, get_machine());
2907 		if (ret >= sizeof (bootblk))
2908 			goto out_path_err;
2909 
2910 		ret = create_sparc_archive(boot_archive, temp, bootblk,
2911 		    get_cachedir(what));
2912 	} else {
2913 		if (!is_dir_flag_on(what, NO_MULTI)) {
2914 			if (bam_verbose)
2915 				bam_print("Attempting to extend x86 archive: "
2916 				    "%s\n", boot_archive);
2917 
2918 			ret = extend_iso_archive(boot_archive, temp,
2919 			    get_updatedir(what));
2920 			if (ret == BAM_SUCCESS) {
2921 				if (bam_verbose)
2922 					bam_print("Successfully extended %s\n",
2923 					    boot_archive);
2924 
2925 				(void) rmdir_r(get_updatedir(what));
2926 				return (BAM_SUCCESS);
2927 			}
2928 		}
2929 		/*
2930 		 * The boot archive will be recreated from scratch. We get here
2931 		 * if at least one of these conditions is true:
2932 		 * - bootadm was called without the -e switch
2933 		 * - the archive (or the archive cache) doesn't exist
2934 		 * - archive size is bigger than BA_SIZE_MAX
2935 		 * - more than COUNT_MAX files need to be updated
2936 		 * - an error occourred either populating the /updates directory
2937 		 *   or extend_iso_archive() failed
2938 		 */
2939 		if (bam_verbose)
2940 			bam_print("Unable to extend %s... rebuilding archive\n",
2941 			    boot_archive);
2942 
2943 		if (get_updatedir(what)[0] != '\0')
2944 			(void) rmdir_r(get_updatedir(what));
2945 
2946 
2947 		ret = create_x86_archive(boot_archive, temp,
2948 		    get_cachedir(what));
2949 	}
2950 
2951 	if (ret == BAM_SUCCESS && bam_verbose)
2952 		bam_print("Successfully created %s\n", boot_archive);
2953 
2954 	return (ret);
2955 
2956 out_path_err:
2957 	bam_error(PATH_TOO_LONG, root);
2958 	return (BAM_ERROR);
2959 }
2960 
2961 static error_t
2962 create_ramdisk(char *root)
2963 {
2964 	char *cmdline, path[PATH_MAX];
2965 	size_t len;
2966 	struct stat sb;
2967 	int ret, what;
2968 
2969 	/* If there is mkisofs, use it to create the required archives */
2970 	if (is_mkisofs()) {
2971 		for (what = FILE32; what < CACHEDIR_NUM; what++) {
2972 			if (is_dir_flag_on(what, NEED_UPDATE)) {
2973 				ret = mkisofs_archive(root, what);
2974 				if (ret != 0)
2975 					return (BAM_ERROR);
2976 			}
2977 		}
2978 		return (BAM_SUCCESS);
2979 	}
2980 
2981 	/*
2982 	 * Else setup command args for create_ramdisk.ksh for the UFS archives
2983 	 */
2984 	if (bam_verbose)
2985 		bam_print("mkisofs not found, creating UFS archive\n");
2986 
2987 	(void) snprintf(path, sizeof (path), "%s/%s", root, CREATE_RAMDISK);
2988 	if (stat(path, &sb) != 0) {
2989 		bam_error(ARCH_EXEC_MISS, path, strerror(errno));
2990 		return (BAM_ERROR);
2991 	}
2992 
2993 	len = strlen(path) + strlen(root) + 10;	/* room for space + -R */
2994 	if (bam_alt_platform)
2995 		len += strlen(bam_platform) + strlen("-p ");
2996 	cmdline = s_calloc(1, len);
2997 
2998 	if (bam_alt_platform) {
2999 		assert(strlen(root) > 1);
3000 		(void) snprintf(cmdline, len, "%s -p %s -R %s",
3001 		    path, bam_platform, root);
3002 		/* chop off / at the end */
3003 		cmdline[strlen(cmdline) - 1] = '\0';
3004 	} else if (strlen(root) > 1) {
3005 		(void) snprintf(cmdline, len, "%s -R %s", path, root);
3006 		/* chop off / at the end */
3007 		cmdline[strlen(cmdline) - 1] = '\0';
3008 	} else
3009 		(void) snprintf(cmdline, len, "%s", path);
3010 
3011 	if (exec_cmd(cmdline, NULL) != 0) {
3012 		bam_error(ARCHIVE_FAIL, cmdline);
3013 		free(cmdline);
3014 		return (BAM_ERROR);
3015 	}
3016 	free(cmdline);
3017 	/*
3018 	 * The existence of the expected archives used to be
3019 	 * verified here. This check is done in create_ramdisk as
3020 	 * it needs to be in sync with the altroot operated upon.
3021 	 */
3022 	return (BAM_SUCCESS);
3023 }
3024 
3025 /*
3026  * Checks if target filesystem is on a ramdisk
3027  * 1 - is miniroot
3028  * 0 - is not
3029  * When in doubt assume it is not a ramdisk.
3030  */
3031 static int
3032 is_ramdisk(char *root)
3033 {
3034 	struct extmnttab mnt;
3035 	FILE *fp;
3036 	int found;
3037 	char mntpt[PATH_MAX];
3038 	char *cp;
3039 
3040 	/*
3041 	 * There are 3 situations where creating archive is
3042 	 * of dubious value:
3043 	 *	- create boot_archive on a lofi-mounted boot_archive
3044 	 *	- create it on a ramdisk which is the root filesystem
3045 	 *	- create it on a ramdisk mounted somewhere else
3046 	 * The first is not easy to detect and checking for it is not
3047 	 * worth it.
3048 	 * The other two conditions are handled here
3049 	 */
3050 	fp = fopen(MNTTAB, "r");
3051 	if (fp == NULL) {
3052 		bam_error(OPEN_FAIL, MNTTAB, strerror(errno));
3053 		return (0);
3054 	}
3055 
3056 	resetmnttab(fp);
3057 
3058 	/*
3059 	 * Remove any trailing / from the mount point
3060 	 */
3061 	(void) strlcpy(mntpt, root, sizeof (mntpt));
3062 	if (strcmp(root, "/") != 0) {
3063 		cp = mntpt + strlen(mntpt) - 1;
3064 		if (*cp == '/')
3065 			*cp = '\0';
3066 	}
3067 	found = 0;
3068 	while (getextmntent(fp, &mnt, sizeof (mnt)) == 0) {
3069 		if (strcmp(mnt.mnt_mountp, mntpt) == 0) {
3070 			found = 1;
3071 			break;
3072 		}
3073 	}
3074 
3075 	if (!found) {
3076 		if (bam_verbose)
3077 			bam_error(NOT_IN_MNTTAB, mntpt);
3078 		(void) fclose(fp);
3079 		return (0);
3080 	}
3081 
3082 	if (strstr(mnt.mnt_special, RAMDISK_SPECIAL) != NULL) {
3083 		if (bam_verbose)
3084 			bam_error(IS_RAMDISK, bam_root);
3085 		(void) fclose(fp);
3086 		return (1);
3087 	}
3088 
3089 	(void) fclose(fp);
3090 
3091 	return (0);
3092 }
3093 
3094 static int
3095 is_boot_archive(char *root)
3096 {
3097 	char		path[PATH_MAX];
3098 	struct stat	sb;
3099 	int		error;
3100 	const char	*fcn = "is_boot_archive()";
3101 
3102 	/*
3103 	 * We can't create an archive without the create_ramdisk script
3104 	 */
3105 	(void) snprintf(path, sizeof (path), "%s/%s", root, CREATE_RAMDISK);
3106 	error = stat(path, &sb);
3107 	INJECT_ERROR1("NOT_ARCHIVE_BASED", error = -1);
3108 	if (error == -1) {
3109 		if (bam_verbose)
3110 			bam_print(FILE_MISS, path);
3111 		BAM_DPRINTF((D_NOT_ARCHIVE_BOOT, fcn, root));
3112 		return (0);
3113 	}
3114 
3115 	BAM_DPRINTF((D_IS_ARCHIVE_BOOT, fcn, root));
3116 	return (1);
3117 }
3118 
3119 /*
3120  * Need to call this for anything that operates on the GRUB menu
3121  * In the x86 live upgrade case the directory /boot/grub may be present
3122  * even on pre-newboot BEs. The authoritative way to check for a GRUB target
3123  * is to check for the presence of the stage2 binary which is present
3124  * only on GRUB targets (even on x86 boot partitions). Checking for the
3125  * presence of the multiboot binary is not correct as it is not present
3126  * on x86 boot partitions.
3127  */
3128 int
3129 is_grub(const char *root)
3130 {
3131 	char path[PATH_MAX];
3132 	struct stat sb;
3133 	const char *fcn = "is_grub()";
3134 
3135 	(void) snprintf(path, sizeof (path), "%s%s", root, GRUB_STAGE2);
3136 	if (stat(path, &sb) == -1) {
3137 		BAM_DPRINTF((D_NO_GRUB_DIR, fcn, path));
3138 		return (0);
3139 	}
3140 
3141 	return (1);
3142 }
3143 
3144 static int
3145 is_zfs(char *root)
3146 {
3147 	struct statvfs		vfs;
3148 	int			ret;
3149 	const char		*fcn = "is_zfs()";
3150 
3151 	ret = statvfs(root, &vfs);
3152 	INJECT_ERROR1("STATVFS_ZFS", ret = 1);
3153 	if (ret != 0) {
3154 		bam_error(STATVFS_FAIL, root, strerror(errno));
3155 		return (0);
3156 	}
3157 
3158 	if (strncmp(vfs.f_basetype, "zfs", strlen("zfs")) == 0) {
3159 		BAM_DPRINTF((D_IS_ZFS, fcn, root));
3160 		return (1);
3161 	} else {
3162 		BAM_DPRINTF((D_IS_NOT_ZFS, fcn, root));
3163 		return (0);
3164 	}
3165 }
3166 
3167 static int
3168 is_ufs(char *root)
3169 {
3170 	struct statvfs		vfs;
3171 	int			ret;
3172 	const char		*fcn = "is_ufs()";
3173 
3174 	ret = statvfs(root, &vfs);
3175 	INJECT_ERROR1("STATVFS_UFS", ret = 1);
3176 	if (ret != 0) {
3177 		bam_error(STATVFS_FAIL, root, strerror(errno));
3178 		return (0);
3179 	}
3180 
3181 	if (strncmp(vfs.f_basetype, "ufs", strlen("ufs")) == 0) {
3182 		BAM_DPRINTF((D_IS_UFS, fcn, root));
3183 		return (1);
3184 	} else {
3185 		BAM_DPRINTF((D_IS_NOT_UFS, fcn, root));
3186 		return (0);
3187 	}
3188 }
3189 
3190 static int
3191 is_pcfs(char *root)
3192 {
3193 	struct statvfs		vfs;
3194 	int			ret;
3195 	const char		*fcn = "is_pcfs()";
3196 
3197 	ret = statvfs(root, &vfs);
3198 	INJECT_ERROR1("STATVFS_PCFS", ret = 1);
3199 	if (ret != 0) {
3200 		bam_error(STATVFS_FAIL, root, strerror(errno));
3201 		return (0);
3202 	}
3203 
3204 	if (strncmp(vfs.f_basetype, "pcfs", strlen("pcfs")) == 0) {
3205 		BAM_DPRINTF((D_IS_PCFS, fcn, root));
3206 		return (1);
3207 	} else {
3208 		BAM_DPRINTF((D_IS_NOT_PCFS, fcn, root));
3209 		return (0);
3210 	}
3211 }
3212 
3213 static int
3214 is_readonly(char *root)
3215 {
3216 	int		fd;
3217 	int		error;
3218 	char		testfile[PATH_MAX];
3219 	const char	*fcn = "is_readonly()";
3220 
3221 	/*
3222 	 * Using statvfs() to check for a read-only filesystem is not
3223 	 * reliable. The only way to reliably test is to attempt to
3224 	 * create a file
3225 	 */
3226 	(void) snprintf(testfile, sizeof (testfile), "%s/%s.%d",
3227 	    root, BOOTADM_RDONLY_TEST, getpid());
3228 
3229 	(void) unlink(testfile);
3230 
3231 	errno = 0;
3232 	fd = open(testfile, O_RDWR|O_CREAT|O_EXCL, 0644);
3233 	error = errno;
3234 	INJECT_ERROR2("RDONLY_TEST_ERROR", fd = -1, error = EACCES);
3235 	if (fd == -1 && error == EROFS) {
3236 		BAM_DPRINTF((D_RDONLY_FS, fcn, root));
3237 		return (1);
3238 	} else if (fd == -1) {
3239 		bam_error(RDONLY_TEST_ERROR, root, strerror(error));
3240 	}
3241 
3242 	(void) close(fd);
3243 	(void) unlink(testfile);
3244 
3245 	BAM_DPRINTF((D_RDWR_FS, fcn, root));
3246 	return (0);
3247 }
3248 
3249 static error_t
3250 update_archive(char *root, char *opt)
3251 {
3252 	error_t ret;
3253 
3254 	assert(root);
3255 	assert(opt == NULL);
3256 
3257 	init_walk_args();
3258 	(void) umask(022);
3259 
3260 	/*
3261 	 * root must belong to a boot archive based OS,
3262 	 */
3263 	if (!is_boot_archive(root)) {
3264 		/*
3265 		 * Emit message only if not in context of update_all.
3266 		 * If in update_all, emit only if verbose flag is set.
3267 		 */
3268 		if (!bam_update_all || bam_verbose)
3269 			bam_print(NOT_ARCHIVE_BOOT, root);
3270 		return (BAM_ERROR);
3271 	}
3272 
3273 	/*
3274 	 * If smf check is requested when / is writable (can happen
3275 	 * on first reboot following an upgrade because service
3276 	 * dependency is messed up), skip the check.
3277 	 */
3278 	if (bam_smf_check && !bam_root_readonly && !is_zfs(root))
3279 		return (BAM_SUCCESS);
3280 
3281 	/*
3282 	 * root must be writable. This check applies to alternate
3283 	 * root (-R option); bam_root_readonly applies to '/' only.
3284 	 * The behaviour translates into being the one of a 'check'.
3285 	 */
3286 	if (!bam_smf_check && !bam_check && is_readonly(root)) {
3287 		set_flag(RDONLY_FSCHK);
3288 		bam_check = 1;
3289 	}
3290 
3291 	/*
3292 	 * Don't generate archive on ramdisk, but still check if the
3293 	 * archive is up-to-date
3294 	 */
3295 	if (is_ramdisk(root)) {
3296 		set_flag(RAMDSK_FSCHK);
3297 		bam_check = 1;
3298 	}
3299 
3300 	/*
3301 	 * Now check if updated is really needed
3302 	 */
3303 	ret = update_required(root);
3304 
3305 	/*
3306 	 * The check command (-n) is *not* a dry run
3307 	 * It only checks if the archive is in sync.
3308 	 * A readonly filesystem or being on a ramdisk have to be considered
3309 	 * errors only if an update is required.
3310 	 */
3311 	if (bam_check) {
3312 		if (is_flag_on(RDONLY_FSCHK)) {
3313 			if (ret > 0)
3314 				bam_error(RDONLY_FS, root);
3315 			if (bam_update_all)
3316 				return ((ret != 0) ? BAM_ERROR : BAM_SUCCESS);
3317 		}
3318 
3319 		if (is_flag_on(RAMDSK_FSCHK)) {
3320 			if (ret > 0)
3321 				bam_error(SKIP_RAMDISK, root);
3322 			if (bam_update_all)
3323 				return ((ret != 0) ? BAM_ERROR : BAM_SUCCESS);
3324 		}
3325 
3326 		bam_exit((ret != 0) ? 1 : 0);
3327 	}
3328 
3329 	if (ret == 1) {
3330 		/* create the ramdisk */
3331 		ret = create_ramdisk(root);
3332 	}
3333 
3334 	/* if the archive is updated, save the new stat data */
3335 	if (ret == 0 && walk_arg.new_nvlp != NULL) {
3336 		savenew(root);
3337 	}
3338 
3339 	clear_walk_args();
3340 
3341 	return (ret);
3342 }
3343 
3344 static error_t
3345 synchronize_BE_menu(void)
3346 {
3347 	struct stat	sb;
3348 	char		cmdline[PATH_MAX];
3349 	char		cksum_line[PATH_MAX];
3350 	filelist_t	flist = {0};
3351 	char		*old_cksum_str;
3352 	char		*old_size_str;
3353 	char		*old_file;
3354 	char		*curr_cksum_str;
3355 	char		*curr_size_str;
3356 	char		*curr_file;
3357 	FILE		*cfp;
3358 	int		found;
3359 	int		ret;
3360 	const char	*fcn = "synchronize_BE_menu()";
3361 
3362 	BAM_DPRINTF((D_FUNC_ENTRY0, fcn));
3363 
3364 	/* Check if findroot enabled LU BE */
3365 	if (stat(FINDROOT_INSTALLGRUB, &sb) != 0) {
3366 		BAM_DPRINTF((D_NOT_LU_BE, fcn));
3367 		return (BAM_SUCCESS);
3368 	}
3369 
3370 	if (stat(LU_MENU_CKSUM, &sb) != 0) {
3371 		BAM_DPRINTF((D_NO_CKSUM_FILE, fcn, LU_MENU_CKSUM));
3372 		goto menu_sync;
3373 	}
3374 
3375 	cfp = fopen(LU_MENU_CKSUM, "r");
3376 	INJECT_ERROR1("CKSUM_FILE_MISSING", cfp = NULL);
3377 	if (cfp == NULL) {
3378 		bam_error(CANNOT_READ_LU_CKSUM, LU_MENU_CKSUM);
3379 		goto menu_sync;
3380 	}
3381 	BAM_DPRINTF((D_CKSUM_FILE_OPENED, fcn, LU_MENU_CKSUM));
3382 
3383 	found = 0;
3384 	while (s_fgets(cksum_line, sizeof (cksum_line), cfp) != NULL) {
3385 		INJECT_ERROR1("MULTIPLE_CKSUM", found = 1);
3386 		if (found) {
3387 			bam_error(MULTIPLE_LU_CKSUM, LU_MENU_CKSUM);
3388 			(void) fclose(cfp);
3389 			goto menu_sync;
3390 		}
3391 		found = 1;
3392 	}
3393 	BAM_DPRINTF((D_CKSUM_FILE_READ, fcn, LU_MENU_CKSUM));
3394 
3395 
3396 	old_cksum_str = strtok(cksum_line, " \t");
3397 	old_size_str = strtok(NULL, " \t");
3398 	old_file = strtok(NULL, " \t");
3399 
3400 	INJECT_ERROR1("OLD_CKSUM_NULL", old_cksum_str = NULL);
3401 	INJECT_ERROR1("OLD_SIZE_NULL", old_size_str = NULL);
3402 	INJECT_ERROR1("OLD_FILE_NULL", old_file = NULL);
3403 	if (old_cksum_str == NULL || old_size_str == NULL || old_file == NULL) {
3404 		bam_error(CANNOT_PARSE_LU_CKSUM, LU_MENU_CKSUM);
3405 		goto menu_sync;
3406 	}
3407 	BAM_DPRINTF((D_CKSUM_FILE_PARSED, fcn, LU_MENU_CKSUM));
3408 
3409 	/* Get checksum of current menu */
3410 	(void) snprintf(cmdline, sizeof (cmdline), "%s %s",
3411 	    CKSUM, GRUB_MENU);
3412 	ret = exec_cmd(cmdline, &flist);
3413 	INJECT_ERROR1("GET_CURR_CKSUM", ret = 1);
3414 	if (ret != 0) {
3415 		bam_error(MENU_CKSUM_FAIL);
3416 		return (BAM_ERROR);
3417 	}
3418 	BAM_DPRINTF((D_CKSUM_GEN_SUCCESS, fcn));
3419 
3420 	INJECT_ERROR1("GET_CURR_CKSUM_OUTPUT", flist.head = NULL);
3421 	if ((flist.head == NULL) || (flist.head != flist.tail)) {
3422 		bam_error(BAD_CKSUM);
3423 		filelist_free(&flist);
3424 		return (BAM_ERROR);
3425 	}
3426 	BAM_DPRINTF((D_CKSUM_GEN_OUTPUT_VALID, fcn));
3427 
3428 	curr_cksum_str = strtok(flist.head->line, " \t");
3429 	curr_size_str = strtok(NULL, " \t");
3430 	curr_file = strtok(NULL, " \t");
3431 
3432 	INJECT_ERROR1("CURR_CKSUM_NULL", curr_cksum_str = NULL);
3433 	INJECT_ERROR1("CURR_SIZE_NULL", curr_size_str = NULL);
3434 	INJECT_ERROR1("CURR_FILE_NULL", curr_file = NULL);
3435 	if (curr_cksum_str == NULL || curr_size_str == NULL ||
3436 	    curr_file == NULL) {
3437 		bam_error(BAD_CKSUM_PARSE);
3438 		filelist_free(&flist);
3439 		return (BAM_ERROR);
3440 	}
3441 	BAM_DPRINTF((D_CKSUM_GEN_PARSED, fcn));
3442 
3443 	if (strcmp(old_cksum_str, curr_cksum_str) == 0 &&
3444 	    strcmp(old_size_str, curr_size_str) == 0 &&
3445 	    strcmp(old_file, curr_file) == 0) {
3446 		filelist_free(&flist);
3447 		BAM_DPRINTF((D_CKSUM_NO_CHANGE, fcn));
3448 		return (BAM_SUCCESS);
3449 	}
3450 
3451 	filelist_free(&flist);
3452 
3453 	/* cksum doesn't match - the menu has changed */
3454 	BAM_DPRINTF((D_CKSUM_HAS_CHANGED, fcn));
3455 
3456 menu_sync:
3457 	bam_print(PROP_GRUB_MENU);
3458 
3459 	(void) snprintf(cmdline, sizeof (cmdline),
3460 	    "/bin/sh -c '. %s > /dev/null; %s %s yes > /dev/null'",
3461 	    LULIB, LULIB_PROPAGATE_FILE, GRUB_MENU);
3462 	ret = exec_cmd(cmdline, NULL);
3463 	INJECT_ERROR1("PROPAGATE_MENU", ret = 1);
3464 	if (ret != 0) {
3465 		bam_error(MENU_PROP_FAIL);
3466 		return (BAM_ERROR);
3467 	}
3468 	BAM_DPRINTF((D_PROPAGATED_MENU, fcn));
3469 
3470 	(void) snprintf(cmdline, sizeof (cmdline), "/bin/cp %s %s > /dev/null",
3471 	    GRUB_MENU, GRUB_BACKUP_MENU);
3472 	ret = exec_cmd(cmdline, NULL);
3473 	INJECT_ERROR1("CREATE_BACKUP", ret = 1);
3474 	if (ret != 0) {
3475 		bam_error(MENU_BACKUP_FAIL, GRUB_BACKUP_MENU);
3476 		return (BAM_ERROR);
3477 	}
3478 	BAM_DPRINTF((D_CREATED_BACKUP, fcn, GRUB_BACKUP_MENU));
3479 
3480 	(void) snprintf(cmdline, sizeof (cmdline),
3481 	    "/bin/sh -c '. %s > /dev/null; %s %s no > /dev/null'",
3482 	    LULIB, LULIB_PROPAGATE_FILE, GRUB_BACKUP_MENU);
3483 	ret = exec_cmd(cmdline, NULL);
3484 	INJECT_ERROR1("PROPAGATE_BACKUP", ret = 1);
3485 	if (ret != 0) {
3486 		bam_error(BACKUP_PROP_FAIL, GRUB_BACKUP_MENU);
3487 		return (BAM_ERROR);
3488 	}
3489 	BAM_DPRINTF((D_PROPAGATED_BACKUP, fcn, GRUB_BACKUP_MENU));
3490 
3491 	(void) snprintf(cmdline, sizeof (cmdline), "%s %s > %s",
3492 	    CKSUM, GRUB_MENU, LU_MENU_CKSUM);
3493 	ret = exec_cmd(cmdline, NULL);
3494 	INJECT_ERROR1("CREATE_CKSUM_FILE", ret = 1);
3495 	if (ret != 0) {
3496 		bam_error(MENU_CKSUM_WRITE_FAIL, LU_MENU_CKSUM);
3497 		return (BAM_ERROR);
3498 	}
3499 	BAM_DPRINTF((D_CREATED_CKSUM_FILE, fcn, LU_MENU_CKSUM));
3500 
3501 	(void) snprintf(cmdline, sizeof (cmdline),
3502 	    "/bin/sh -c '. %s > /dev/null; %s %s no > /dev/null'",
3503 	    LULIB, LULIB_PROPAGATE_FILE, LU_MENU_CKSUM);
3504 	ret = exec_cmd(cmdline, NULL);
3505 	INJECT_ERROR1("PROPAGATE_MENU_CKSUM_FILE", ret = 1);
3506 	if (ret != 0) {
3507 		bam_error(MENU_CKSUM_PROP_FAIL, LU_MENU_CKSUM);
3508 		return (BAM_ERROR);
3509 	}
3510 	BAM_DPRINTF((D_PROPAGATED_CKSUM_FILE, fcn, LU_MENU_CKSUM));
3511 
3512 	(void) snprintf(cmdline, sizeof (cmdline),
3513 	    "/bin/sh -c '. %s > /dev/null; %s %s no > /dev/null'",
3514 	    LULIB, LULIB_PROPAGATE_FILE, BOOTADM);
3515 	ret = exec_cmd(cmdline, NULL);
3516 	INJECT_ERROR1("PROPAGATE_BOOTADM_FILE", ret = 1);
3517 	if (ret != 0) {
3518 		bam_error(BOOTADM_PROP_FAIL, BOOTADM);
3519 		return (BAM_ERROR);
3520 	}
3521 	BAM_DPRINTF((D_PROPAGATED_BOOTADM, fcn, BOOTADM));
3522 
3523 	return (BAM_SUCCESS);
3524 }
3525 
3526 static error_t
3527 update_all(char *root, char *opt)
3528 {
3529 	struct extmnttab mnt;
3530 	struct stat sb;
3531 	FILE *fp;
3532 	char multibt[PATH_MAX];
3533 	char creatram[PATH_MAX];
3534 	error_t ret = BAM_SUCCESS;
3535 
3536 	assert(root);
3537 	assert(opt == NULL);
3538 
3539 	if (bam_rootlen != 1 || *root != '/') {
3540 		elide_trailing_slash(root, multibt, sizeof (multibt));
3541 		bam_error(ALT_ROOT_INVALID, multibt);
3542 		return (BAM_ERROR);
3543 	}
3544 
3545 	/*
3546 	 * Check to see if we are in the midst of safemode patching
3547 	 * If so skip building the archive for /. Instead build it
3548 	 * against the latest bits obtained by creating a fresh lofs
3549 	 * mount of root.
3550 	 */
3551 	if (stat(LOFS_PATCH_FILE, &sb) == 0)  {
3552 		if (mkdir(LOFS_PATCH_MNT, DIR_PERMS) == -1 &&
3553 		    errno != EEXIST) {
3554 			bam_error(MKDIR_FAILED, "%s", LOFS_PATCH_MNT,
3555 			    strerror(errno));
3556 			ret = BAM_ERROR;
3557 			goto out;
3558 		}
3559 		(void) snprintf(multibt, sizeof (multibt),
3560 		    "/sbin/mount -F lofs -o nosub /  %s", LOFS_PATCH_MNT);
3561 		if (exec_cmd(multibt, NULL) != 0) {
3562 			bam_error(MOUNT_FAILED, LOFS_PATCH_MNT, "lofs");
3563 			ret = BAM_ERROR;
3564 		}
3565 		if (ret != BAM_ERROR) {
3566 			(void) snprintf(rootbuf, sizeof (rootbuf), "%s/",
3567 			    LOFS_PATCH_MNT);
3568 			bam_rootlen = strlen(rootbuf);
3569 			if (update_archive(rootbuf, opt) != BAM_SUCCESS)
3570 				ret = BAM_ERROR;
3571 			/*
3572 			 * unmount the lofs mount since there could be
3573 			 * multiple invocations of bootadm -a update_all
3574 			 */
3575 			(void) snprintf(multibt, sizeof (multibt),
3576 			    "/sbin/umount %s", LOFS_PATCH_MNT);
3577 			if (exec_cmd(multibt, NULL) != 0) {
3578 				bam_error(UMOUNT_FAILED, LOFS_PATCH_MNT);
3579 				ret = BAM_ERROR;
3580 			}
3581 		}
3582 	} else {
3583 		/*
3584 		 * First update archive for current root
3585 		 */
3586 		if (update_archive(root, opt) != BAM_SUCCESS)
3587 			ret = BAM_ERROR;
3588 	}
3589 
3590 	if (ret == BAM_ERROR)
3591 		goto out;
3592 
3593 	/*
3594 	 * Now walk the mount table, performing archive update
3595 	 * for all mounted Newboot root filesystems
3596 	 */
3597 	fp = fopen(MNTTAB, "r");
3598 	if (fp == NULL) {
3599 		bam_error(OPEN_FAIL, MNTTAB, strerror(errno));
3600 		ret = BAM_ERROR;
3601 		goto out;
3602 	}
3603 
3604 	resetmnttab(fp);
3605 
3606 	while (getextmntent(fp, &mnt, sizeof (mnt)) == 0) {
3607 		if (mnt.mnt_special == NULL)
3608 			continue;
3609 		if ((strcmp(mnt.mnt_fstype, MNTTYPE_ZFS) != 0) &&
3610 		    (strncmp(mnt.mnt_special, "/dev/", strlen("/dev/")) != 0))
3611 			continue;
3612 		if (strcmp(mnt.mnt_mountp, "/") == 0)
3613 			continue;
3614 
3615 		(void) snprintf(creatram, sizeof (creatram), "%s/%s",
3616 		    mnt.mnt_mountp, CREATE_RAMDISK);
3617 
3618 		if (stat(creatram, &sb) == -1)
3619 			continue;
3620 
3621 		/*
3622 		 * We put a trailing slash to be consistent with root = "/"
3623 		 * case, such that we don't have to print // in some cases.
3624 		 */
3625 		(void) snprintf(rootbuf, sizeof (rootbuf), "%s/",
3626 		    mnt.mnt_mountp);
3627 		bam_rootlen = strlen(rootbuf);
3628 
3629 		/*
3630 		 * It's possible that other mounts may be an alternate boot
3631 		 * architecture, so check it again.
3632 		 */
3633 		if ((get_boot_cap(rootbuf) != BAM_SUCCESS) ||
3634 		    (update_archive(rootbuf, opt) != BAM_SUCCESS))
3635 			ret = BAM_ERROR;
3636 	}
3637 
3638 	(void) fclose(fp);
3639 
3640 out:
3641 	/*
3642 	 * We no longer use biosdev for Live Upgrade. Hence
3643 	 * there is no need to defer (to shutdown time) any fdisk
3644 	 * updates
3645 	 */
3646 	if (stat(GRUB_fdisk, &sb) == 0 || stat(GRUB_fdisk_target, &sb) == 0) {
3647 		bam_error(FDISK_FILES_FOUND, GRUB_fdisk, GRUB_fdisk_target);
3648 	}
3649 
3650 	/*
3651 	 * If user has updated menu in current BE, propagate the
3652 	 * updates to all BEs.
3653 	 */
3654 	if (sync_menu && synchronize_BE_menu() != BAM_SUCCESS)
3655 		ret = BAM_ERROR;
3656 
3657 	return (ret);
3658 }
3659 
3660 static void
3661 append_line(menu_t *mp, line_t *lp)
3662 {
3663 	if (mp->start == NULL) {
3664 		mp->start = lp;
3665 	} else {
3666 		mp->end->next = lp;
3667 		lp->prev = mp->end;
3668 	}
3669 	mp->end = lp;
3670 }
3671 
3672 void
3673 unlink_line(menu_t *mp, line_t *lp)
3674 {
3675 	/* unlink from list */
3676 	if (lp->prev)
3677 		lp->prev->next = lp->next;
3678 	else
3679 		mp->start = lp->next;
3680 	if (lp->next)
3681 		lp->next->prev = lp->prev;
3682 	else
3683 		mp->end = lp->prev;
3684 }
3685 
3686 static entry_t *
3687 boot_entry_new(menu_t *mp, line_t *start, line_t *end)
3688 {
3689 	entry_t *ent, *prev;
3690 	const char *fcn = "boot_entry_new()";
3691 
3692 	assert(mp);
3693 	assert(start);
3694 	assert(end);
3695 
3696 	ent = s_calloc(1, sizeof (entry_t));
3697 	BAM_DPRINTF((D_ENTRY_NEW, fcn));
3698 	ent->start = start;
3699 	ent->end = end;
3700 
3701 	if (mp->entries == NULL) {
3702 		mp->entries = ent;
3703 		BAM_DPRINTF((D_ENTRY_NEW_FIRST, fcn));
3704 		return (ent);
3705 	}
3706 
3707 	prev = mp->entries;
3708 	while (prev->next)
3709 		prev = prev->next;
3710 	prev->next = ent;
3711 	ent->prev = prev;
3712 	BAM_DPRINTF((D_ENTRY_NEW_LINKED, fcn));
3713 	return (ent);
3714 }
3715 
3716 static void
3717 boot_entry_addline(entry_t *ent, line_t *lp)
3718 {
3719 	if (ent)
3720 		ent->end = lp;
3721 }
3722 
3723 /*
3724  * Check whether cmd matches the one indexed by which, and whether arg matches
3725  * str.  which must be either KERNEL_CMD or MODULE_CMD, and a match to the
3726  * respective *_DOLLAR_CMD is also acceptable.  The arg is searched using
3727  * strstr(), so it can be a partial match.
3728  */
3729 static int
3730 check_cmd(const char *cmd, const int which, const char *arg, const char *str)
3731 {
3732 	int			ret;
3733 	const char		*fcn = "check_cmd()";
3734 
3735 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, arg, str));
3736 
3737 	if ((strcmp(cmd, menu_cmds[which]) != 0) &&
3738 	    (strcmp(cmd, menu_cmds[which + 1]) != 0)) {
3739 		BAM_DPRINTF((D_CHECK_CMD_CMD_NOMATCH,
3740 		    fcn, cmd, menu_cmds[which]));
3741 		return (0);
3742 	}
3743 	ret = (strstr(arg, str) != NULL);
3744 
3745 	if (ret) {
3746 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
3747 	} else {
3748 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
3749 	}
3750 
3751 	return (ret);
3752 }
3753 
3754 static error_t
3755 kernel_parser(entry_t *entry, char *cmd, char *arg, int linenum)
3756 {
3757 	const char		*fcn  = "kernel_parser()";
3758 
3759 	assert(entry);
3760 	assert(cmd);
3761 	assert(arg);
3762 
3763 	if (strcmp(cmd, menu_cmds[KERNEL_CMD]) != 0 &&
3764 	    strcmp(cmd, menu_cmds[KERNEL_DOLLAR_CMD]) != 0) {
3765 		BAM_DPRINTF((D_NOT_KERNEL_CMD, fcn, cmd));
3766 		return (BAM_ERROR);
3767 	}
3768 
3769 	if (strncmp(arg, DIRECT_BOOT_32, sizeof (DIRECT_BOOT_32) - 1) == 0) {
3770 		BAM_DPRINTF((D_SET_DBOOT_32, fcn, arg));
3771 		entry->flags |= BAM_ENTRY_DBOOT | BAM_ENTRY_32BIT;
3772 	} else if (strncmp(arg, DIRECT_BOOT_KERNEL,
3773 	    sizeof (DIRECT_BOOT_KERNEL) - 1) == 0) {
3774 		BAM_DPRINTF((D_SET_DBOOT, fcn, arg));
3775 		entry->flags |= BAM_ENTRY_DBOOT;
3776 	} else if (strncmp(arg, DIRECT_BOOT_64,
3777 	    sizeof (DIRECT_BOOT_64) - 1) == 0) {
3778 		BAM_DPRINTF((D_SET_DBOOT_64, fcn, arg));
3779 		entry->flags |= BAM_ENTRY_DBOOT | BAM_ENTRY_64BIT;
3780 	} else if (strncmp(arg, DIRECT_BOOT_FAILSAFE_KERNEL,
3781 	    sizeof (DIRECT_BOOT_FAILSAFE_KERNEL) - 1) == 0) {
3782 		BAM_DPRINTF((D_SET_DBOOT_FAILSAFE, fcn, arg));
3783 		entry->flags |= BAM_ENTRY_DBOOT | BAM_ENTRY_FAILSAFE;
3784 	} else if (strncmp(arg, DIRECT_BOOT_FAILSAFE_32,
3785 	    sizeof (DIRECT_BOOT_FAILSAFE_32) - 1) == 0) {
3786 		BAM_DPRINTF((D_SET_DBOOT_FAILSAFE_32, fcn, arg));
3787 		entry->flags |= BAM_ENTRY_DBOOT | BAM_ENTRY_FAILSAFE
3788 		    | BAM_ENTRY_32BIT;
3789 	} else if (strncmp(arg, DIRECT_BOOT_FAILSAFE_64,
3790 	    sizeof (DIRECT_BOOT_FAILSAFE_64) - 1) == 0) {
3791 		BAM_DPRINTF((D_SET_DBOOT_FAILSAFE_64, fcn, arg));
3792 		entry->flags |= BAM_ENTRY_DBOOT | BAM_ENTRY_FAILSAFE
3793 		    | BAM_ENTRY_64BIT;
3794 	} else if (strncmp(arg, MULTI_BOOT, sizeof (MULTI_BOOT) - 1) == 0) {
3795 		BAM_DPRINTF((D_SET_MULTIBOOT, fcn, arg));
3796 		entry->flags |= BAM_ENTRY_MULTIBOOT;
3797 	} else if (strncmp(arg, MULTI_BOOT_FAILSAFE,
3798 	    sizeof (MULTI_BOOT_FAILSAFE) - 1) == 0) {
3799 		BAM_DPRINTF((D_SET_MULTIBOOT_FAILSAFE, fcn, arg));
3800 		entry->flags |= BAM_ENTRY_MULTIBOOT | BAM_ENTRY_FAILSAFE;
3801 	} else if (strstr(arg, XEN_KERNEL_SUBSTR)) {
3802 		BAM_DPRINTF((D_SET_HV, fcn, arg));
3803 		entry->flags |= BAM_ENTRY_HV;
3804 	} else if (!(entry->flags & (BAM_ENTRY_BOOTADM|BAM_ENTRY_LU))) {
3805 		BAM_DPRINTF((D_SET_HAND_KERNEL, fcn, arg));
3806 		return (BAM_ERROR);
3807 	} else if (strncmp(arg, KERNEL_PREFIX, strlen(KERNEL_PREFIX)) == 0 &&
3808 	    strstr(arg, UNIX_SPACE)) {
3809 		entry->flags |= BAM_ENTRY_DBOOT | BAM_ENTRY_32BIT;
3810 	} else if (strncmp(arg, KERNEL_PREFIX, strlen(KERNEL_PREFIX)) == 0 &&
3811 	    strstr(arg, AMD_UNIX_SPACE)) {
3812 		entry->flags |= BAM_ENTRY_DBOOT | BAM_ENTRY_64BIT;
3813 	} else {
3814 		BAM_DPRINTF((D_IS_UNKNOWN_KERNEL, fcn, arg));
3815 		bam_error(UNKNOWN_KERNEL_LINE, linenum);
3816 		return (BAM_ERROR);
3817 	}
3818 
3819 	return (BAM_SUCCESS);
3820 }
3821 
3822 static error_t
3823 module_parser(entry_t *entry, char *cmd, char *arg, int linenum)
3824 {
3825 	const char		*fcn = "module_parser()";
3826 
3827 	assert(entry);
3828 	assert(cmd);
3829 	assert(arg);
3830 
3831 	if (strcmp(cmd, menu_cmds[MODULE_CMD]) != 0 &&
3832 	    strcmp(cmd, menu_cmds[MODULE_DOLLAR_CMD]) != 0) {
3833 		BAM_DPRINTF((D_NOT_MODULE_CMD, fcn, cmd));
3834 		return (BAM_ERROR);
3835 	}
3836 
3837 	if (strcmp(arg, DIRECT_BOOT_ARCHIVE) == 0 ||
3838 	    strcmp(arg, DIRECT_BOOT_ARCHIVE_32) == 0 ||
3839 	    strcmp(arg, DIRECT_BOOT_ARCHIVE_64) == 0 ||
3840 	    strcmp(arg, MULTIBOOT_ARCHIVE) == 0 ||
3841 	    strcmp(arg, FAILSAFE_ARCHIVE) == 0 ||
3842 	    strcmp(arg, FAILSAFE_ARCHIVE_32) == 0 ||
3843 	    strcmp(arg, FAILSAFE_ARCHIVE_64) == 0 ||
3844 	    strcmp(arg, XEN_KERNEL_MODULE_LINE) == 0 ||
3845 	    strcmp(arg, XEN_KERNEL_MODULE_LINE_ZFS) == 0) {
3846 		BAM_DPRINTF((D_BOOTADM_LU_MODULE, fcn, arg));
3847 		return (BAM_SUCCESS);
3848 	} else if (!(entry->flags & BAM_ENTRY_BOOTADM) &&
3849 	    !(entry->flags & BAM_ENTRY_LU)) {
3850 		/* don't emit warning for hand entries */
3851 		BAM_DPRINTF((D_IS_HAND_MODULE, fcn, arg));
3852 		return (BAM_ERROR);
3853 	} else {
3854 		BAM_DPRINTF((D_IS_UNKNOWN_MODULE, fcn, arg));
3855 		bam_error(UNKNOWN_MODULE_LINE, linenum);
3856 		return (BAM_ERROR);
3857 	}
3858 }
3859 
3860 /*
3861  * A line in menu.lst looks like
3862  * [ ]*<cmd>[ \t=]*<arg>*
3863  */
3864 static void
3865 line_parser(menu_t *mp, char *str, int *lineNum, int *entryNum)
3866 {
3867 	/*
3868 	 * save state across calls. This is so that
3869 	 * header gets the right entry# after title has
3870 	 * been processed
3871 	 */
3872 	static line_t *prev = NULL;
3873 	static entry_t *curr_ent = NULL;
3874 	static int in_liveupgrade = 0;
3875 
3876 	line_t	*lp;
3877 	char *cmd, *sep, *arg;
3878 	char save, *cp, *line;
3879 	menu_flag_t flag = BAM_INVALID;
3880 	const char *fcn = "line_parser()";
3881 
3882 	if (str == NULL) {
3883 		return;
3884 	}
3885 
3886 	/*
3887 	 * First save a copy of the entire line.
3888 	 * We use this later to set the line field.
3889 	 */
3890 	line = s_strdup(str);
3891 
3892 	/* Eat up leading whitespace */
3893 	while (*str == ' ' || *str == '\t')
3894 		str++;
3895 
3896 	if (*str == '#') {		/* comment */
3897 		cmd = s_strdup("#");
3898 		sep = NULL;
3899 		arg = s_strdup(str + 1);
3900 		flag = BAM_COMMENT;
3901 		if (strstr(arg, BAM_LU_HDR) != NULL) {
3902 			in_liveupgrade = 1;
3903 		} else if (strstr(arg, BAM_LU_FTR) != NULL) {
3904 			in_liveupgrade = 0;
3905 		}
3906 	} else if (*str == '\0') {	/* blank line */
3907 		cmd = sep = arg = NULL;
3908 		flag = BAM_EMPTY;
3909 	} else {
3910 		/*
3911 		 * '=' is not a documented separator in grub syntax.
3912 		 * However various development bits use '=' as a
3913 		 * separator. In addition, external users also
3914 		 * use = as a separator. So we will allow that usage.
3915 		 */
3916 		cp = str;
3917 		while (*str != ' ' && *str != '\t' && *str != '=') {
3918 			if (*str == '\0') {
3919 				cmd = s_strdup(cp);
3920 				sep = arg = NULL;
3921 				break;
3922 			}
3923 			str++;
3924 		}
3925 
3926 		if (*str != '\0') {
3927 			save = *str;
3928 			*str = '\0';
3929 			cmd = s_strdup(cp);
3930 			*str = save;
3931 
3932 			str++;
3933 			save = *str;
3934 			*str = '\0';
3935 			sep = s_strdup(str - 1);
3936 			*str = save;
3937 
3938 			while (*str == ' ' || *str == '\t')
3939 				str++;
3940 			if (*str == '\0')
3941 				arg = NULL;
3942 			else
3943 				arg = s_strdup(str);
3944 		}
3945 	}
3946 
3947 	lp = s_calloc(1, sizeof (line_t));
3948 
3949 	lp->cmd = cmd;
3950 	lp->sep = sep;
3951 	lp->arg = arg;
3952 	lp->line = line;
3953 	lp->lineNum = ++(*lineNum);
3954 	if (cmd && strcmp(cmd, menu_cmds[TITLE_CMD]) == 0) {
3955 		lp->entryNum = ++(*entryNum);
3956 		lp->flags = BAM_TITLE;
3957 		if (prev && prev->flags == BAM_COMMENT &&
3958 		    prev->arg && strcmp(prev->arg, BAM_BOOTADM_HDR) == 0) {
3959 			prev->entryNum = lp->entryNum;
3960 			curr_ent = boot_entry_new(mp, prev, lp);
3961 			curr_ent->flags |= BAM_ENTRY_BOOTADM;
3962 			BAM_DPRINTF((D_IS_BOOTADM_ENTRY, fcn, arg));
3963 		} else {
3964 			curr_ent = boot_entry_new(mp, lp, lp);
3965 			if (in_liveupgrade) {
3966 				curr_ent->flags |= BAM_ENTRY_LU;
3967 				BAM_DPRINTF((D_IS_LU_ENTRY, fcn, arg));
3968 			}
3969 		}
3970 		curr_ent->entryNum = *entryNum;
3971 	} else if (flag != BAM_INVALID) {
3972 		/*
3973 		 * For header comments, the entry# is "fixed up"
3974 		 * by the subsequent title
3975 		 */
3976 		lp->entryNum = *entryNum;
3977 		lp->flags = flag;
3978 	} else {
3979 		lp->entryNum = *entryNum;
3980 
3981 		if (*entryNum == ENTRY_INIT) {
3982 			lp->flags = BAM_GLOBAL;
3983 		} else {
3984 			lp->flags = BAM_ENTRY;
3985 
3986 			if (cmd && arg) {
3987 				if (strcmp(cmd, menu_cmds[ROOT_CMD]) == 0) {
3988 					BAM_DPRINTF((D_IS_ROOT_CMD, fcn, arg));
3989 					curr_ent->flags |= BAM_ENTRY_ROOT;
3990 				} else if (strcmp(cmd, menu_cmds[FINDROOT_CMD])
3991 				    == 0) {
3992 					BAM_DPRINTF((D_IS_FINDROOT_CMD, fcn,
3993 					    arg));
3994 					curr_ent->flags |= BAM_ENTRY_FINDROOT;
3995 				} else if (strcmp(cmd,
3996 				    menu_cmds[CHAINLOADER_CMD]) == 0) {
3997 					BAM_DPRINTF((D_IS_CHAINLOADER_CMD, fcn,
3998 					    arg));
3999 					curr_ent->flags |=
4000 					    BAM_ENTRY_CHAINLOADER;
4001 				} else if (kernel_parser(curr_ent, cmd, arg,
4002 				    lp->lineNum) != BAM_SUCCESS) {
4003 					(void) module_parser(curr_ent, cmd,
4004 					    arg, lp->lineNum);
4005 				}
4006 			}
4007 		}
4008 	}
4009 
4010 	/* record default, old default, and entry line ranges */
4011 	if (lp->flags == BAM_GLOBAL &&
4012 	    strcmp(lp->cmd, menu_cmds[DEFAULT_CMD]) == 0) {
4013 		mp->curdefault = lp;
4014 	} else if (lp->flags == BAM_COMMENT &&
4015 	    strncmp(lp->arg, BAM_OLDDEF, strlen(BAM_OLDDEF)) == 0) {
4016 		mp->olddefault = lp;
4017 	} else if (lp->flags == BAM_COMMENT &&
4018 	    strncmp(lp->arg, BAM_OLD_RC_DEF, strlen(BAM_OLD_RC_DEF)) == 0) {
4019 		mp->old_rc_default = lp;
4020 	} else if (lp->flags == BAM_ENTRY ||
4021 	    (lp->flags == BAM_COMMENT &&
4022 	    strcmp(lp->arg, BAM_BOOTADM_FTR) == 0)) {
4023 		boot_entry_addline(curr_ent, lp);
4024 	}
4025 	append_line(mp, lp);
4026 
4027 	prev = lp;
4028 }
4029 
4030 void
4031 update_numbering(menu_t *mp)
4032 {
4033 	int lineNum;
4034 	int entryNum;
4035 	int old_default_value;
4036 	line_t *lp, *prev, *default_lp, *default_entry;
4037 	char buf[PATH_MAX];
4038 
4039 	if (mp->start == NULL) {
4040 		return;
4041 	}
4042 
4043 	lineNum = LINE_INIT;
4044 	entryNum = ENTRY_INIT;
4045 	old_default_value = ENTRY_INIT;
4046 	lp = default_lp = default_entry = NULL;
4047 
4048 	prev = NULL;
4049 	for (lp = mp->start; lp; prev = lp, lp = lp->next) {
4050 		lp->lineNum = ++lineNum;
4051 
4052 		/*
4053 		 * Get the value of the default command
4054 		 */
4055 		if (lp->entryNum == ENTRY_INIT && lp->cmd &&
4056 		    strcmp(lp->cmd, menu_cmds[DEFAULT_CMD]) == 0 &&
4057 		    lp->arg) {
4058 			old_default_value = atoi(lp->arg);
4059 			default_lp = lp;
4060 		}
4061 
4062 		/*
4063 		 * If not a booting entry, nothing else to fix for this
4064 		 * entry
4065 		 */
4066 		if (lp->entryNum == ENTRY_INIT)
4067 			continue;
4068 
4069 		/*
4070 		 * Record the position of the default entry.
4071 		 * The following works because global
4072 		 * commands like default and timeout should precede
4073 		 * actual boot entries, so old_default_value
4074 		 * is already known (or default cmd is missing).
4075 		 */
4076 		if (default_entry == NULL &&
4077 		    old_default_value != ENTRY_INIT &&
4078 		    lp->entryNum == old_default_value) {
4079 			default_entry = lp;
4080 		}
4081 
4082 		/*
4083 		 * Now fixup the entry number
4084 		 */
4085 		if (lp->cmd && strcmp(lp->cmd, menu_cmds[TITLE_CMD]) == 0) {
4086 			lp->entryNum = ++entryNum;
4087 			/* fixup the bootadm header */
4088 			if (prev && prev->flags == BAM_COMMENT &&
4089 			    prev->arg &&
4090 			    strcmp(prev->arg, BAM_BOOTADM_HDR) == 0) {
4091 				prev->entryNum = lp->entryNum;
4092 			}
4093 		} else {
4094 			lp->entryNum = entryNum;
4095 		}
4096 	}
4097 
4098 	/*
4099 	 * No default command in menu, simply return
4100 	 */
4101 	if (default_lp == NULL) {
4102 		return;
4103 	}
4104 
4105 	free(default_lp->arg);
4106 	free(default_lp->line);
4107 
4108 	if (default_entry == NULL) {
4109 		default_lp->arg = s_strdup("0");
4110 	} else {
4111 		(void) snprintf(buf, sizeof (buf), "%d",
4112 		    default_entry->entryNum);
4113 		default_lp->arg = s_strdup(buf);
4114 	}
4115 
4116 	/*
4117 	 * The following is required since only the line field gets
4118 	 * written back to menu.lst
4119 	 */
4120 	(void) snprintf(buf, sizeof (buf), "%s%s%s",
4121 	    menu_cmds[DEFAULT_CMD], menu_cmds[SEP_CMD], default_lp->arg);
4122 	default_lp->line = s_strdup(buf);
4123 }
4124 
4125 
4126 static menu_t *
4127 menu_read(char *menu_path)
4128 {
4129 	FILE *fp;
4130 	char buf[BAM_MAXLINE], *cp;
4131 	menu_t *mp;
4132 	int line, entry, len, n;
4133 
4134 	mp = s_calloc(1, sizeof (menu_t));
4135 
4136 	fp = fopen(menu_path, "r");
4137 	if (fp == NULL) { /* Let the caller handle this error */
4138 		return (mp);
4139 	}
4140 
4141 
4142 	/* Note: GRUB boot entry number starts with 0 */
4143 	line = LINE_INIT;
4144 	entry = ENTRY_INIT;
4145 	cp = buf;
4146 	len = sizeof (buf);
4147 	while (s_fgets(cp, len, fp) != NULL) {
4148 		n = strlen(cp);
4149 		if (cp[n - 1] == '\\') {
4150 			len -= n - 1;
4151 			assert(len >= 2);
4152 			cp += n - 1;
4153 			continue;
4154 		}
4155 		line_parser(mp, buf, &line, &entry);
4156 		cp = buf;
4157 		len = sizeof (buf);
4158 	}
4159 
4160 	if (fclose(fp) == EOF) {
4161 		bam_error(CLOSE_FAIL, menu_path, strerror(errno));
4162 	}
4163 
4164 	return (mp);
4165 }
4166 
4167 static error_t
4168 selector(menu_t *mp, char *opt, int *entry, char **title)
4169 {
4170 	char *eq;
4171 	char *opt_dup;
4172 	int entryNum;
4173 
4174 	assert(mp);
4175 	assert(mp->start);
4176 	assert(opt);
4177 
4178 	opt_dup = s_strdup(opt);
4179 
4180 	if (entry)
4181 		*entry = ENTRY_INIT;
4182 	if (title)
4183 		*title = NULL;
4184 
4185 	eq = strchr(opt_dup, '=');
4186 	if (eq == NULL) {
4187 		bam_error(INVALID_OPT, opt);
4188 		free(opt_dup);
4189 		return (BAM_ERROR);
4190 	}
4191 
4192 	*eq = '\0';
4193 	if (entry && strcmp(opt_dup, OPT_ENTRY_NUM) == 0) {
4194 		assert(mp->end);
4195 		entryNum = s_strtol(eq + 1);
4196 		if (entryNum < 0 || entryNum > mp->end->entryNum) {
4197 			bam_error(INVALID_ENTRY, eq + 1);
4198 			free(opt_dup);
4199 			return (BAM_ERROR);
4200 		}
4201 		*entry = entryNum;
4202 	} else if (title && strcmp(opt_dup, menu_cmds[TITLE_CMD]) == 0) {
4203 		*title = opt + (eq - opt_dup) + 1;
4204 	} else {
4205 		bam_error(INVALID_OPT, opt);
4206 		free(opt_dup);
4207 		return (BAM_ERROR);
4208 	}
4209 
4210 	free(opt_dup);
4211 	return (BAM_SUCCESS);
4212 }
4213 
4214 /*
4215  * If invoked with no titles/entries (opt == NULL)
4216  * only title lines in file are printed.
4217  *
4218  * If invoked with a title or entry #, all
4219  * lines in *every* matching entry are listed
4220  */
4221 static error_t
4222 list_entry(menu_t *mp, char *menu_path, char *opt)
4223 {
4224 	line_t *lp;
4225 	int entry = ENTRY_INIT;
4226 	int found;
4227 	char *title = NULL;
4228 
4229 	assert(mp);
4230 	assert(menu_path);
4231 
4232 	/* opt is optional */
4233 	BAM_DPRINTF((D_FUNC_ENTRY2, "list_entry", menu_path,
4234 	    opt ? opt : "<NULL>"));
4235 
4236 	if (mp->start == NULL) {
4237 		bam_error(NO_MENU, menu_path);
4238 		return (BAM_ERROR);
4239 	}
4240 
4241 	if (opt != NULL) {
4242 		if (selector(mp, opt, &entry, &title) != BAM_SUCCESS) {
4243 			return (BAM_ERROR);
4244 		}
4245 		assert((entry != ENTRY_INIT) ^ (title != NULL));
4246 	} else {
4247 		(void) read_globals(mp, menu_path, menu_cmds[DEFAULT_CMD], 0);
4248 		(void) read_globals(mp, menu_path, menu_cmds[TIMEOUT_CMD], 0);
4249 	}
4250 
4251 	found = 0;
4252 	for (lp = mp->start; lp; lp = lp->next) {
4253 		if (lp->flags == BAM_COMMENT || lp->flags == BAM_EMPTY)
4254 			continue;
4255 		if (opt == NULL && lp->flags == BAM_TITLE) {
4256 			bam_print(PRINT_TITLE, lp->entryNum,
4257 			    lp->arg);
4258 			found = 1;
4259 			continue;
4260 		}
4261 		if (entry != ENTRY_INIT && lp->entryNum == entry) {
4262 			bam_print(PRINT, lp->line);
4263 			found = 1;
4264 			continue;
4265 		}
4266 
4267 		/*
4268 		 * We set the entry value here so that all lines
4269 		 * in entry get printed. If we subsequently match
4270 		 * title in other entries, all lines in those
4271 		 * entries get printed as well.
4272 		 */
4273 		if (title && lp->flags == BAM_TITLE && lp->arg &&
4274 		    strncmp(title, lp->arg, strlen(title)) == 0) {
4275 			bam_print(PRINT, lp->line);
4276 			entry = lp->entryNum;
4277 			found = 1;
4278 			continue;
4279 		}
4280 	}
4281 
4282 	if (!found) {
4283 		bam_error(NO_MATCH_ENTRY);
4284 		return (BAM_ERROR);
4285 	}
4286 
4287 	return (BAM_SUCCESS);
4288 }
4289 
4290 int
4291 add_boot_entry(menu_t *mp,
4292 	char *title,
4293 	char *findroot,
4294 	char *kernel,
4295 	char *mod_kernel,
4296 	char *module)
4297 {
4298 	int		lineNum;
4299 	int		entryNum;
4300 	char		linebuf[BAM_MAXLINE];
4301 	menu_cmd_t	k_cmd;
4302 	menu_cmd_t	m_cmd;
4303 	const char	*fcn = "add_boot_entry()";
4304 
4305 	assert(mp);
4306 
4307 	INJECT_ERROR1("ADD_BOOT_ENTRY_FINDROOT_NULL", findroot = NULL);
4308 	if (findroot == NULL) {
4309 		bam_error(NULL_FINDROOT);
4310 		return (BAM_ERROR);
4311 	}
4312 
4313 	if (title == NULL) {
4314 		title = "Solaris";	/* default to Solaris */
4315 	}
4316 	if (kernel == NULL) {
4317 		bam_error(SUBOPT_MISS, menu_cmds[KERNEL_CMD]);
4318 		return (BAM_ERROR);
4319 	}
4320 	if (module == NULL) {
4321 		if (bam_direct != BAM_DIRECT_DBOOT) {
4322 			bam_error(SUBOPT_MISS, menu_cmds[MODULE_CMD]);
4323 			return (BAM_ERROR);
4324 		}
4325 
4326 		/* Figure the commands out from the kernel line */
4327 		if (strstr(kernel, "$ISADIR") != NULL) {
4328 			module = DIRECT_BOOT_ARCHIVE;
4329 			k_cmd = KERNEL_DOLLAR_CMD;
4330 			m_cmd = MODULE_DOLLAR_CMD;
4331 		} else if (strstr(kernel, "amd64") != NULL) {
4332 			module = DIRECT_BOOT_ARCHIVE_64;
4333 			k_cmd = KERNEL_CMD;
4334 			m_cmd = MODULE_CMD;
4335 		} else {
4336 			module = DIRECT_BOOT_ARCHIVE_32;
4337 			k_cmd = KERNEL_CMD;
4338 			m_cmd = MODULE_CMD;
4339 		}
4340 	} else if ((bam_direct == BAM_DIRECT_DBOOT) &&
4341 	    (strstr(kernel, "$ISADIR") != NULL)) {
4342 		/*
4343 		 * If it's a non-failsafe dboot kernel, use the "kernel$"
4344 		 * command.  Otherwise, use "kernel".
4345 		 */
4346 		k_cmd = KERNEL_DOLLAR_CMD;
4347 		m_cmd = MODULE_DOLLAR_CMD;
4348 	} else {
4349 		k_cmd = KERNEL_CMD;
4350 		m_cmd = MODULE_CMD;
4351 	}
4352 
4353 	if (mp->start) {
4354 		lineNum = mp->end->lineNum;
4355 		entryNum = mp->end->entryNum;
4356 	} else {
4357 		lineNum = LINE_INIT;
4358 		entryNum = ENTRY_INIT;
4359 	}
4360 
4361 	/*
4362 	 * No separator for comment (HDR/FTR) commands
4363 	 * The syntax for comments is #<comment>
4364 	 */
4365 	(void) snprintf(linebuf, sizeof (linebuf), "%s%s",
4366 	    menu_cmds[COMMENT_CMD], BAM_BOOTADM_HDR);
4367 	line_parser(mp, linebuf, &lineNum, &entryNum);
4368 
4369 	(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
4370 	    menu_cmds[TITLE_CMD], menu_cmds[SEP_CMD], title);
4371 	line_parser(mp, linebuf, &lineNum, &entryNum);
4372 
4373 	(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
4374 	    menu_cmds[FINDROOT_CMD], menu_cmds[SEP_CMD], findroot);
4375 	line_parser(mp, linebuf, &lineNum, &entryNum);
4376 	BAM_DPRINTF((D_ADD_FINDROOT_NUM, fcn, lineNum, entryNum));
4377 
4378 	(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
4379 	    menu_cmds[k_cmd], menu_cmds[SEP_CMD], kernel);
4380 	line_parser(mp, linebuf, &lineNum, &entryNum);
4381 
4382 	if (mod_kernel != NULL) {
4383 		(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
4384 		    menu_cmds[m_cmd], menu_cmds[SEP_CMD], mod_kernel);
4385 		line_parser(mp, linebuf, &lineNum, &entryNum);
4386 	}
4387 
4388 	(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
4389 	    menu_cmds[m_cmd], menu_cmds[SEP_CMD], module);
4390 	line_parser(mp, linebuf, &lineNum, &entryNum);
4391 
4392 	(void) snprintf(linebuf, sizeof (linebuf), "%s%s",
4393 	    menu_cmds[COMMENT_CMD], BAM_BOOTADM_FTR);
4394 	line_parser(mp, linebuf, &lineNum, &entryNum);
4395 
4396 	return (entryNum);
4397 }
4398 
4399 static error_t
4400 do_delete(menu_t *mp, int entryNum)
4401 {
4402 	line_t		*lp;
4403 	line_t		*freed;
4404 	entry_t		*ent;
4405 	entry_t		*tmp;
4406 	int		deleted;
4407 	const char	*fcn = "do_delete()";
4408 
4409 	assert(entryNum != ENTRY_INIT);
4410 
4411 	tmp = NULL;
4412 
4413 	ent = mp->entries;
4414 	while (ent) {
4415 		lp = ent->start;
4416 		/* check entry number and make sure it's a bootadm entry */
4417 		if (lp->flags != BAM_COMMENT ||
4418 		    strcmp(lp->arg, BAM_BOOTADM_HDR) != 0 ||
4419 		    (entryNum != ALL_ENTRIES && lp->entryNum != entryNum)) {
4420 			ent = ent->next;
4421 			continue;
4422 		}
4423 
4424 		/* free the entry content */
4425 		do {
4426 			freed = lp;
4427 			lp = lp->next;	/* prev stays the same */
4428 			BAM_DPRINTF((D_FREEING_LINE, fcn, freed->lineNum));
4429 			unlink_line(mp, freed);
4430 			line_free(freed);
4431 		} while (freed != ent->end);
4432 
4433 		/* free the entry_t structure */
4434 		assert(tmp == NULL);
4435 		tmp = ent;
4436 		ent = ent->next;
4437 		if (tmp->prev)
4438 			tmp->prev->next = ent;
4439 		else
4440 			mp->entries = ent;
4441 		if (ent)
4442 			ent->prev = tmp->prev;
4443 		BAM_DPRINTF((D_FREEING_ENTRY, fcn, tmp->entryNum));
4444 		free(tmp);
4445 		tmp = NULL;
4446 		deleted = 1;
4447 	}
4448 
4449 	assert(tmp == NULL);
4450 
4451 	if (!deleted && entryNum != ALL_ENTRIES) {
4452 		bam_error(NO_BOOTADM_MATCH);
4453 		return (BAM_ERROR);
4454 	}
4455 
4456 	/*
4457 	 * Now that we have deleted an entry, update
4458 	 * the entry numbering and the default cmd.
4459 	 */
4460 	update_numbering(mp);
4461 
4462 	return (BAM_SUCCESS);
4463 }
4464 
4465 static error_t
4466 delete_all_entries(menu_t *mp, char *dummy, char *opt)
4467 {
4468 	assert(mp);
4469 	assert(dummy == NULL);
4470 	assert(opt == NULL);
4471 
4472 	BAM_DPRINTF((D_FUNC_ENTRY0, "delete_all_entries"));
4473 
4474 	if (mp->start == NULL) {
4475 		bam_print(EMPTY_MENU);
4476 		return (BAM_SUCCESS);
4477 	}
4478 
4479 	if (do_delete(mp, ALL_ENTRIES) != BAM_SUCCESS) {
4480 		return (BAM_ERROR);
4481 	}
4482 
4483 	return (BAM_WRITE);
4484 }
4485 
4486 static FILE *
4487 create_diskmap(char *osroot)
4488 {
4489 	FILE *fp;
4490 	char cmd[PATH_MAX];
4491 	const char *fcn = "create_diskmap()";
4492 
4493 	/* make sure we have a map file */
4494 	fp = fopen(GRUBDISK_MAP, "r");
4495 	if (fp == NULL) {
4496 		(void) snprintf(cmd, sizeof (cmd),
4497 		    "%s/%s > /dev/null", osroot, CREATE_DISKMAP);
4498 		if (exec_cmd(cmd, NULL) != 0)
4499 			return (NULL);
4500 		fp = fopen(GRUBDISK_MAP, "r");
4501 		INJECT_ERROR1("DISKMAP_CREATE_FAIL", fp = NULL);
4502 		if (fp) {
4503 			BAM_DPRINTF((D_CREATED_DISKMAP, fcn, GRUBDISK_MAP));
4504 		} else {
4505 			BAM_DPRINTF((D_CREATE_DISKMAP_FAIL, fcn, GRUBDISK_MAP));
4506 		}
4507 	}
4508 	return (fp);
4509 }
4510 
4511 #define	SECTOR_SIZE	512
4512 
4513 static int
4514 get_partition(char *device)
4515 {
4516 	int i, fd, is_pcfs, partno = -1;
4517 	struct mboot *mboot;
4518 	char boot_sect[SECTOR_SIZE];
4519 	char *wholedisk, *slice;
4520 
4521 	/* form whole disk (p0) */
4522 	slice = device + strlen(device) - 2;
4523 	is_pcfs = (*slice != 's');
4524 	if (!is_pcfs)
4525 		*slice = '\0';
4526 	wholedisk = s_calloc(1, strlen(device) + 3);
4527 	(void) snprintf(wholedisk, strlen(device) + 3, "%sp0", device);
4528 	if (!is_pcfs)
4529 		*slice = 's';
4530 
4531 	/* read boot sector */
4532 	fd = open(wholedisk, O_RDONLY);
4533 	free(wholedisk);
4534 	if (fd == -1 || read(fd, boot_sect, SECTOR_SIZE) != SECTOR_SIZE) {
4535 		return (partno);
4536 	}
4537 	(void) close(fd);
4538 
4539 	/* parse fdisk table */
4540 	mboot = (struct mboot *)((void *)boot_sect);
4541 	for (i = 0; i < FD_NUMPART; i++) {
4542 		struct ipart *part =
4543 		    (struct ipart *)(uintptr_t)mboot->parts + i;
4544 		if (is_pcfs) {	/* looking for solaris boot part */
4545 			if (part->systid == 0xbe) {
4546 				partno = i;
4547 				break;
4548 			}
4549 		} else {	/* look for solaris partition, old and new */
4550 			if (part->systid == SUNIXOS ||
4551 			    part->systid == SUNIXOS2) {
4552 				partno = i;
4553 				break;
4554 			}
4555 		}
4556 	}
4557 	return (partno);
4558 }
4559 
4560 char *
4561 get_grubroot(char *osroot, char *osdev, char *menu_root)
4562 {
4563 	char		*grubroot;	/* (hd#,#,#) */
4564 	char		*slice;
4565 	char		*grubhd;
4566 	int		fdiskpart;
4567 	int		found = 0;
4568 	char		*devname;
4569 	char		*ctdname = strstr(osdev, "dsk/");
4570 	char		linebuf[PATH_MAX];
4571 	FILE		*fp;
4572 
4573 	INJECT_ERROR1("GRUBROOT_INVALID_OSDEV", ctdname = NULL);
4574 	if (ctdname == NULL) {
4575 		bam_error(INVALID_DEV_DSK, osdev);
4576 		return (NULL);
4577 	}
4578 
4579 	if (menu_root && !menu_on_bootdisk(osroot, menu_root)) {
4580 		/* menu bears no resemblance to our reality */
4581 		bam_error(CANNOT_GRUBROOT_BOOTDISK, osdev);
4582 		return (NULL);
4583 	}
4584 
4585 	ctdname += strlen("dsk/");
4586 	slice = strrchr(ctdname, 's');
4587 	if (slice)
4588 		*slice = '\0';
4589 
4590 	fp = create_diskmap(osroot);
4591 	if (fp == NULL) {
4592 		bam_error(DISKMAP_FAIL, osroot);
4593 		return (NULL);
4594 	}
4595 
4596 	rewind(fp);
4597 	while (s_fgets(linebuf, sizeof (linebuf), fp) != NULL) {
4598 		grubhd = strtok(linebuf, " \t\n");
4599 		if (grubhd)
4600 			devname = strtok(NULL, " \t\n");
4601 		else
4602 			devname = NULL;
4603 		if (devname && strcmp(devname, ctdname) == 0) {
4604 			found = 1;
4605 			break;
4606 		}
4607 	}
4608 
4609 	if (slice)
4610 		*slice = 's';
4611 
4612 	(void) fclose(fp);
4613 	fp = NULL;
4614 
4615 	INJECT_ERROR1("GRUBROOT_BIOSDEV_FAIL", found = 0);
4616 	if (found == 0) {
4617 		bam_error(BIOSDEV_SKIP, osdev);
4618 		return (NULL);
4619 	}
4620 
4621 	fdiskpart = get_partition(osdev);
4622 	INJECT_ERROR1("GRUBROOT_FDISK_FAIL", fdiskpart = -1);
4623 	if (fdiskpart == -1) {
4624 		bam_error(FDISKPART_FAIL, osdev);
4625 		return (NULL);
4626 	}
4627 
4628 	grubroot = s_calloc(1, 10);
4629 	if (slice) {
4630 		(void) snprintf(grubroot, 10, "(hd%s,%d,%c)",
4631 		    grubhd, fdiskpart, slice[1] + 'a' - '0');
4632 	} else
4633 		(void) snprintf(grubroot, 10, "(hd%s,%d)",
4634 		    grubhd, fdiskpart);
4635 
4636 	assert(fp == NULL);
4637 	assert(strncmp(grubroot, "(hd", strlen("(hd")) == 0);
4638 	return (grubroot);
4639 }
4640 
4641 static char *
4642 find_primary_common(char *mntpt, char *fstype)
4643 {
4644 	char		signdir[PATH_MAX];
4645 	char		tmpsign[MAXNAMELEN + 1];
4646 	char		*lu;
4647 	char		*ufs;
4648 	char		*zfs;
4649 	DIR		*dirp = NULL;
4650 	struct dirent	*entp;
4651 	struct stat	sb;
4652 	const char	*fcn = "find_primary_common()";
4653 
4654 	(void) snprintf(signdir, sizeof (signdir), "%s/%s",
4655 	    mntpt, GRUBSIGN_DIR);
4656 
4657 	if (stat(signdir, &sb) == -1) {
4658 		BAM_DPRINTF((D_NO_SIGNDIR, fcn, signdir));
4659 		return (NULL);
4660 	}
4661 
4662 	dirp = opendir(signdir);
4663 	INJECT_ERROR1("SIGNDIR_OPENDIR_FAIL", dirp = NULL);
4664 	if (dirp == NULL) {
4665 		bam_error(OPENDIR_FAILED, signdir, strerror(errno));
4666 		return (NULL);
4667 	}
4668 
4669 	ufs = zfs = lu = NULL;
4670 
4671 	while (entp = readdir(dirp)) {
4672 		if (strcmp(entp->d_name, ".") == 0 ||
4673 		    strcmp(entp->d_name, "..") == 0)
4674 			continue;
4675 
4676 		(void) snprintf(tmpsign, sizeof (tmpsign), "%s", entp->d_name);
4677 
4678 		if (lu == NULL &&
4679 		    strncmp(tmpsign, GRUBSIGN_LU_PREFIX,
4680 		    strlen(GRUBSIGN_LU_PREFIX)) == 0) {
4681 			lu = s_strdup(tmpsign);
4682 		}
4683 
4684 		if (ufs == NULL &&
4685 		    strncmp(tmpsign, GRUBSIGN_UFS_PREFIX,
4686 		    strlen(GRUBSIGN_UFS_PREFIX)) == 0) {
4687 			ufs = s_strdup(tmpsign);
4688 		}
4689 
4690 		if (zfs == NULL &&
4691 		    strncmp(tmpsign, GRUBSIGN_ZFS_PREFIX,
4692 		    strlen(GRUBSIGN_ZFS_PREFIX)) == 0) {
4693 			zfs = s_strdup(tmpsign);
4694 		}
4695 	}
4696 
4697 	BAM_DPRINTF((D_EXIST_PRIMARY_SIGNS, fcn,
4698 	    zfs ? zfs : "NULL",
4699 	    ufs ? ufs : "NULL",
4700 	    lu ? lu : "NULL"));
4701 
4702 	if (dirp) {
4703 		(void) closedir(dirp);
4704 		dirp = NULL;
4705 	}
4706 
4707 	if (strcmp(fstype, "ufs") == 0 && zfs) {
4708 		bam_error(SIGN_FSTYPE_MISMATCH, zfs, "ufs");
4709 		free(zfs);
4710 		zfs = NULL;
4711 	} else if (strcmp(fstype, "zfs") == 0 && ufs) {
4712 		bam_error(SIGN_FSTYPE_MISMATCH, ufs, "zfs");
4713 		free(ufs);
4714 		ufs = NULL;
4715 	}
4716 
4717 	assert(dirp == NULL);
4718 
4719 	/* For now, we let Live Upgrade take care of its signature itself */
4720 	if (lu) {
4721 		BAM_DPRINTF((D_FREEING_LU_SIGNS, fcn, lu));
4722 		free(lu);
4723 		lu = NULL;
4724 	}
4725 
4726 	return (zfs ? zfs : ufs);
4727 }
4728 
4729 static char *
4730 find_backup_common(char *mntpt, char *fstype)
4731 {
4732 	FILE		*bfp = NULL;
4733 	char		tmpsign[MAXNAMELEN + 1];
4734 	char		backup[PATH_MAX];
4735 	char		*ufs;
4736 	char		*zfs;
4737 	char		*lu;
4738 	int		error;
4739 	const char	*fcn = "find_backup_common()";
4740 
4741 	/*
4742 	 * We didn't find it in the primary directory.
4743 	 * Look at the backup
4744 	 */
4745 	(void) snprintf(backup, sizeof (backup), "%s%s",
4746 	    mntpt, GRUBSIGN_BACKUP);
4747 
4748 	bfp = fopen(backup, "r");
4749 	if (bfp == NULL) {
4750 		error = errno;
4751 		if (bam_verbose) {
4752 			bam_error(OPEN_FAIL, backup, strerror(error));
4753 		}
4754 		BAM_DPRINTF((D_OPEN_FAIL, fcn, backup, strerror(error)));
4755 		return (NULL);
4756 	}
4757 
4758 	ufs = zfs = lu = NULL;
4759 
4760 	while (s_fgets(tmpsign, sizeof (tmpsign), bfp) != NULL) {
4761 
4762 		if (lu == NULL &&
4763 		    strncmp(tmpsign, GRUBSIGN_LU_PREFIX,
4764 		    strlen(GRUBSIGN_LU_PREFIX)) == 0) {
4765 			lu = s_strdup(tmpsign);
4766 		}
4767 
4768 		if (ufs == NULL &&
4769 		    strncmp(tmpsign, GRUBSIGN_UFS_PREFIX,
4770 		    strlen(GRUBSIGN_UFS_PREFIX)) == 0) {
4771 			ufs = s_strdup(tmpsign);
4772 		}
4773 
4774 		if (zfs == NULL &&
4775 		    strncmp(tmpsign, GRUBSIGN_ZFS_PREFIX,
4776 		    strlen(GRUBSIGN_ZFS_PREFIX)) == 0) {
4777 			zfs = s_strdup(tmpsign);
4778 		}
4779 	}
4780 
4781 	BAM_DPRINTF((D_EXIST_BACKUP_SIGNS, fcn,
4782 	    zfs ? zfs : "NULL",
4783 	    ufs ? ufs : "NULL",
4784 	    lu ? lu : "NULL"));
4785 
4786 	if (bfp) {
4787 		(void) fclose(bfp);
4788 		bfp = NULL;
4789 	}
4790 
4791 	if (strcmp(fstype, "ufs") == 0 && zfs) {
4792 		bam_error(SIGN_FSTYPE_MISMATCH, zfs, "ufs");
4793 		free(zfs);
4794 		zfs = NULL;
4795 	} else if (strcmp(fstype, "zfs") == 0 && ufs) {
4796 		bam_error(SIGN_FSTYPE_MISMATCH, ufs, "zfs");
4797 		free(ufs);
4798 		ufs = NULL;
4799 	}
4800 
4801 	assert(bfp == NULL);
4802 
4803 	/* For now, we let Live Upgrade take care of its signature itself */
4804 	if (lu) {
4805 		BAM_DPRINTF((D_FREEING_LU_SIGNS, fcn, lu));
4806 		free(lu);
4807 		lu = NULL;
4808 	}
4809 
4810 	return (zfs ? zfs : ufs);
4811 }
4812 
4813 static char *
4814 find_ufs_existing(char *osroot)
4815 {
4816 	char		*sign;
4817 	const char	*fcn = "find_ufs_existing()";
4818 
4819 	sign = find_primary_common(osroot, "ufs");
4820 	if (sign == NULL) {
4821 		sign = find_backup_common(osroot, "ufs");
4822 		BAM_DPRINTF((D_EXIST_BACKUP_SIGN, fcn, sign ? sign : "NULL"));
4823 	} else {
4824 		BAM_DPRINTF((D_EXIST_PRIMARY_SIGN, fcn, sign));
4825 	}
4826 
4827 	return (sign);
4828 }
4829 
4830 char *
4831 get_mountpoint(char *special, char *fstype)
4832 {
4833 	FILE		*mntfp;
4834 	struct mnttab	mp = {0};
4835 	struct mnttab	mpref = {0};
4836 	int		error;
4837 	int		ret;
4838 	const char	*fcn = "get_mountpoint()";
4839 
4840 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, special, fstype));
4841 
4842 	mntfp = fopen(MNTTAB, "r");
4843 	error = errno;
4844 	INJECT_ERROR1("MNTTAB_ERR_GET_MNTPT", mntfp = NULL);
4845 	if (mntfp == NULL) {
4846 		bam_error(OPEN_FAIL, MNTTAB, strerror(error));
4847 		return (NULL);
4848 	}
4849 
4850 	mpref.mnt_special = special;
4851 	mpref.mnt_fstype = fstype;
4852 
4853 	ret = getmntany(mntfp, &mp, &mpref);
4854 	INJECT_ERROR1("GET_MOUNTPOINT_MNTANY", ret = 1);
4855 	if (ret != 0) {
4856 		(void) fclose(mntfp);
4857 		BAM_DPRINTF((D_NO_MNTPT, fcn, special, fstype));
4858 		return (NULL);
4859 	}
4860 	(void) fclose(mntfp);
4861 
4862 	assert(mp.mnt_mountp);
4863 
4864 	BAM_DPRINTF((D_GET_MOUNTPOINT_RET, fcn, special, mp.mnt_mountp));
4865 
4866 	return (s_strdup(mp.mnt_mountp));
4867 }
4868 
4869 /*
4870  * Mounts a "legacy" top dataset (if needed)
4871  * Returns:	The mountpoint of the legacy top dataset or NULL on error
4872  * 		mnted returns one of the above values defined for zfs_mnted_t
4873  */
4874 static char *
4875 mount_legacy_dataset(char *pool, zfs_mnted_t *mnted)
4876 {
4877 	char		cmd[PATH_MAX];
4878 	char		tmpmnt[PATH_MAX];
4879 	filelist_t	flist = {0};
4880 	char		*is_mounted;
4881 	struct stat	sb;
4882 	int		ret;
4883 	const char	*fcn = "mount_legacy_dataset()";
4884 
4885 	BAM_DPRINTF((D_FUNC_ENTRY1, fcn, pool));
4886 
4887 	*mnted = ZFS_MNT_ERROR;
4888 
4889 	(void) snprintf(cmd, sizeof (cmd),
4890 	    "/sbin/zfs get -Ho value mounted %s",
4891 	    pool);
4892 
4893 	ret = exec_cmd(cmd, &flist);
4894 	INJECT_ERROR1("Z_MOUNT_LEG_GET_MOUNTED_CMD", ret = 1);
4895 	if (ret != 0) {
4896 		bam_error(ZFS_MNTED_FAILED, pool);
4897 		return (NULL);
4898 	}
4899 
4900 	INJECT_ERROR1("Z_MOUNT_LEG_GET_MOUNTED_OUT", flist.head = NULL);
4901 	if ((flist.head == NULL) || (flist.head != flist.tail)) {
4902 		bam_error(BAD_ZFS_MNTED, pool);
4903 		filelist_free(&flist);
4904 		return (NULL);
4905 	}
4906 
4907 	is_mounted = strtok(flist.head->line, " \t\n");
4908 	INJECT_ERROR1("Z_MOUNT_LEG_GET_MOUNTED_STRTOK_YES", is_mounted = "yes");
4909 	INJECT_ERROR1("Z_MOUNT_LEG_GET_MOUNTED_STRTOK_NO", is_mounted = "no");
4910 	if (strcmp(is_mounted, "no") != 0) {
4911 		filelist_free(&flist);
4912 		*mnted = LEGACY_ALREADY;
4913 		/* get_mountpoint returns a strdup'ed string */
4914 		BAM_DPRINTF((D_Z_MOUNT_TOP_LEG_ALREADY, fcn, pool));
4915 		return (get_mountpoint(pool, "zfs"));
4916 	}
4917 
4918 	filelist_free(&flist);
4919 
4920 	/*
4921 	 * legacy top dataset is not mounted. Mount it now
4922 	 * First create a mountpoint.
4923 	 */
4924 	(void) snprintf(tmpmnt, sizeof (tmpmnt), "%s.%d",
4925 	    ZFS_LEGACY_MNTPT, getpid());
4926 
4927 	ret = stat(tmpmnt, &sb);
4928 	if (ret == -1) {
4929 		BAM_DPRINTF((D_Z_MOUNT_TOP_LEG_MNTPT_ABS, fcn, pool, tmpmnt));
4930 		ret = mkdirp(tmpmnt, DIR_PERMS);
4931 		INJECT_ERROR1("Z_MOUNT_TOP_LEG_MNTPT_MKDIRP", ret = -1);
4932 		if (ret == -1) {
4933 			bam_error(MKDIR_FAILED, tmpmnt, strerror(errno));
4934 			return (NULL);
4935 		}
4936 	} else {
4937 		BAM_DPRINTF((D_Z_MOUNT_TOP_LEG_MNTPT_PRES, fcn, pool, tmpmnt));
4938 	}
4939 
4940 	(void) snprintf(cmd, sizeof (cmd),
4941 	    "/sbin/mount -F zfs %s %s",
4942 	    pool, tmpmnt);
4943 
4944 	ret = exec_cmd(cmd, NULL);
4945 	INJECT_ERROR1("Z_MOUNT_TOP_LEG_MOUNT_CMD", ret = 1);
4946 	if (ret != 0) {
4947 		bam_error(ZFS_MOUNT_FAILED, pool);
4948 		(void) rmdir(tmpmnt);
4949 		return (NULL);
4950 	}
4951 
4952 	*mnted = LEGACY_MOUNTED;
4953 	BAM_DPRINTF((D_Z_MOUNT_TOP_LEG_MOUNTED, fcn, pool, tmpmnt));
4954 	return (s_strdup(tmpmnt));
4955 }
4956 
4957 /*
4958  * Mounts the top dataset (if needed)
4959  * Returns:	The mountpoint of the top dataset or NULL on error
4960  * 		mnted returns one of the above values defined for zfs_mnted_t
4961  */
4962 static char *
4963 mount_top_dataset(char *pool, zfs_mnted_t *mnted)
4964 {
4965 	char		cmd[PATH_MAX];
4966 	filelist_t	flist = {0};
4967 	char		*is_mounted;
4968 	char		*mntpt;
4969 	char		*zmntpt;
4970 	int		ret;
4971 	const char	*fcn = "mount_top_dataset()";
4972 
4973 	*mnted = ZFS_MNT_ERROR;
4974 
4975 	BAM_DPRINTF((D_FUNC_ENTRY1, fcn, pool));
4976 
4977 	/*
4978 	 * First check if the top dataset is a "legacy" dataset
4979 	 */
4980 	(void) snprintf(cmd, sizeof (cmd),
4981 	    "/sbin/zfs get -Ho value mountpoint %s",
4982 	    pool);
4983 	ret = exec_cmd(cmd, &flist);
4984 	INJECT_ERROR1("Z_MOUNT_TOP_GET_MNTPT", ret = 1);
4985 	if (ret != 0) {
4986 		bam_error(ZFS_MNTPT_FAILED, pool);
4987 		return (NULL);
4988 	}
4989 
4990 	if (flist.head && (flist.head == flist.tail)) {
4991 		char *legacy = strtok(flist.head->line, " \t\n");
4992 		if (legacy && strcmp(legacy, "legacy") == 0) {
4993 			filelist_free(&flist);
4994 			BAM_DPRINTF((D_Z_IS_LEGACY, fcn, pool));
4995 			return (mount_legacy_dataset(pool, mnted));
4996 		}
4997 	}
4998 
4999 	filelist_free(&flist);
5000 
5001 	BAM_DPRINTF((D_Z_IS_NOT_LEGACY, fcn, pool));
5002 
5003 	(void) snprintf(cmd, sizeof (cmd),
5004 	    "/sbin/zfs get -Ho value mounted %s",
5005 	    pool);
5006 
5007 	ret = exec_cmd(cmd, &flist);
5008 	INJECT_ERROR1("Z_MOUNT_TOP_NONLEG_GET_MOUNTED", ret = 1);
5009 	if (ret != 0) {
5010 		bam_error(ZFS_MNTED_FAILED, pool);
5011 		return (NULL);
5012 	}
5013 
5014 	INJECT_ERROR1("Z_MOUNT_TOP_NONLEG_GET_MOUNTED_VAL", flist.head = NULL);
5015 	if ((flist.head == NULL) || (flist.head != flist.tail)) {
5016 		bam_error(BAD_ZFS_MNTED, pool);
5017 		filelist_free(&flist);
5018 		return (NULL);
5019 	}
5020 
5021 	is_mounted = strtok(flist.head->line, " \t\n");
5022 	INJECT_ERROR1("Z_MOUNT_TOP_NONLEG_GET_MOUNTED_YES", is_mounted = "yes");
5023 	INJECT_ERROR1("Z_MOUNT_TOP_NONLEG_GET_MOUNTED_NO", is_mounted = "no");
5024 	if (strcmp(is_mounted, "no") != 0) {
5025 		filelist_free(&flist);
5026 		*mnted = ZFS_ALREADY;
5027 		BAM_DPRINTF((D_Z_MOUNT_TOP_NONLEG_MOUNTED_ALREADY, fcn, pool));
5028 		goto mounted;
5029 	}
5030 
5031 	filelist_free(&flist);
5032 	BAM_DPRINTF((D_Z_MOUNT_TOP_NONLEG_MOUNTED_NOT_ALREADY, fcn, pool));
5033 
5034 	/* top dataset is not mounted. Mount it now */
5035 	(void) snprintf(cmd, sizeof (cmd),
5036 	    "/sbin/zfs mount %s", pool);
5037 	ret = exec_cmd(cmd, NULL);
5038 	INJECT_ERROR1("Z_MOUNT_TOP_NONLEG_MOUNT_CMD", ret = 1);
5039 	if (ret != 0) {
5040 		bam_error(ZFS_MOUNT_FAILED, pool);
5041 		return (NULL);
5042 	}
5043 	*mnted = ZFS_MOUNTED;
5044 	BAM_DPRINTF((D_Z_MOUNT_TOP_NONLEG_MOUNTED_NOW, fcn, pool));
5045 	/*FALLTHRU*/
5046 mounted:
5047 	/*
5048 	 * Now get the mountpoint
5049 	 */
5050 	(void) snprintf(cmd, sizeof (cmd),
5051 	    "/sbin/zfs get -Ho value mountpoint %s",
5052 	    pool);
5053 
5054 	ret = exec_cmd(cmd, &flist);
5055 	INJECT_ERROR1("Z_MOUNT_TOP_NONLEG_GET_MNTPT_CMD", ret = 1);
5056 	if (ret != 0) {
5057 		bam_error(ZFS_MNTPT_FAILED, pool);
5058 		goto error;
5059 	}
5060 
5061 	INJECT_ERROR1("Z_MOUNT_TOP_NONLEG_GET_MNTPT_OUT", flist.head = NULL);
5062 	if ((flist.head == NULL) || (flist.head != flist.tail)) {
5063 		bam_error(NULL_ZFS_MNTPT, pool);
5064 		goto error;
5065 	}
5066 
5067 	mntpt = strtok(flist.head->line, " \t\n");
5068 	INJECT_ERROR1("Z_MOUNT_TOP_NONLEG_GET_MNTPT_STRTOK", mntpt = "foo");
5069 	if (*mntpt != '/') {
5070 		bam_error(BAD_ZFS_MNTPT, pool, mntpt);
5071 		goto error;
5072 	}
5073 	zmntpt = s_strdup(mntpt);
5074 
5075 	filelist_free(&flist);
5076 
5077 	BAM_DPRINTF((D_Z_MOUNT_TOP_NONLEG_MNTPT, fcn, pool, zmntpt));
5078 
5079 	return (zmntpt);
5080 
5081 error:
5082 	filelist_free(&flist);
5083 	(void) umount_top_dataset(pool, *mnted, NULL);
5084 	BAM_DPRINTF((D_RETURN_FAILURE, fcn));
5085 	return (NULL);
5086 }
5087 
5088 static int
5089 umount_top_dataset(char *pool, zfs_mnted_t mnted, char *mntpt)
5090 {
5091 	char		cmd[PATH_MAX];
5092 	int		ret;
5093 	const char	*fcn = "umount_top_dataset()";
5094 
5095 	INJECT_ERROR1("Z_UMOUNT_TOP_INVALID_STATE", mnted = ZFS_MNT_ERROR);
5096 	switch (mnted) {
5097 	case LEGACY_ALREADY:
5098 	case ZFS_ALREADY:
5099 		/* nothing to do */
5100 		BAM_DPRINTF((D_Z_UMOUNT_TOP_ALREADY_NOP, fcn, pool,
5101 		    mntpt ? mntpt : "NULL"));
5102 		free(mntpt);
5103 		return (BAM_SUCCESS);
5104 	case LEGACY_MOUNTED:
5105 		(void) snprintf(cmd, sizeof (cmd),
5106 		    "/sbin/umount %s", pool);
5107 		ret = exec_cmd(cmd, NULL);
5108 		INJECT_ERROR1("Z_UMOUNT_TOP_LEGACY_UMOUNT_FAIL", ret = 1);
5109 		if (ret != 0) {
5110 			bam_error(UMOUNT_FAILED, pool);
5111 			free(mntpt);
5112 			return (BAM_ERROR);
5113 		}
5114 		if (mntpt)
5115 			(void) rmdir(mntpt);
5116 		free(mntpt);
5117 		BAM_DPRINTF((D_Z_UMOUNT_TOP_LEGACY, fcn, pool));
5118 		return (BAM_SUCCESS);
5119 	case ZFS_MOUNTED:
5120 		free(mntpt);
5121 		(void) snprintf(cmd, sizeof (cmd),
5122 		    "/sbin/zfs unmount %s", pool);
5123 		ret = exec_cmd(cmd, NULL);
5124 		INJECT_ERROR1("Z_UMOUNT_TOP_NONLEG_UMOUNT_FAIL", ret = 1);
5125 		if (ret != 0) {
5126 			bam_error(UMOUNT_FAILED, pool);
5127 			return (BAM_ERROR);
5128 		}
5129 		BAM_DPRINTF((D_Z_UMOUNT_TOP_NONLEG, fcn, pool));
5130 		return (BAM_SUCCESS);
5131 	default:
5132 		bam_error(INT_BAD_MNTSTATE, pool);
5133 		return (BAM_ERROR);
5134 	}
5135 	/*NOTREACHED*/
5136 }
5137 
5138 /*
5139  * For ZFS, osdev can be one of two forms
5140  * It can be a "special" file as seen in mnttab: rpool/ROOT/szboot_0402
5141  * It can be a /dev/[r]dsk special file. We handle both instances
5142  */
5143 static char *
5144 get_pool(char *osdev)
5145 {
5146 	char		cmd[PATH_MAX];
5147 	char		buf[PATH_MAX];
5148 	filelist_t	flist = {0};
5149 	char		*pool;
5150 	char		*cp;
5151 	char		*slash;
5152 	int		ret;
5153 	const char	*fcn = "get_pool()";
5154 
5155 	INJECT_ERROR1("GET_POOL_OSDEV", osdev = NULL);
5156 	if (osdev == NULL) {
5157 		bam_error(GET_POOL_OSDEV_NULL);
5158 		return (NULL);
5159 	}
5160 
5161 	BAM_DPRINTF((D_GET_POOL_OSDEV, fcn, osdev));
5162 
5163 	if (osdev[0] != '/') {
5164 		(void) strlcpy(buf, osdev, sizeof (buf));
5165 		slash = strchr(buf, '/');
5166 		if (slash)
5167 			*slash = '\0';
5168 		pool = s_strdup(buf);
5169 		BAM_DPRINTF((D_GET_POOL_RET, fcn, pool));
5170 		return (pool);
5171 	} else if (strncmp(osdev, "/dev/dsk/", strlen("/dev/dsk/")) != 0 &&
5172 	    strncmp(osdev, "/dev/rdsk/", strlen("/dev/rdsk/")) != 0) {
5173 		bam_error(GET_POOL_BAD_OSDEV, osdev);
5174 		return (NULL);
5175 	}
5176 
5177 	(void) snprintf(cmd, sizeof (cmd),
5178 	    "/usr/sbin/fstyp -a %s 2>/dev/null | /bin/grep '^name:'",
5179 	    osdev);
5180 
5181 	ret = exec_cmd(cmd, &flist);
5182 	INJECT_ERROR1("GET_POOL_FSTYP", ret = 1);
5183 	if (ret != 0) {
5184 		bam_error(FSTYP_A_FAILED, osdev);
5185 		return (NULL);
5186 	}
5187 
5188 	INJECT_ERROR1("GET_POOL_FSTYP_OUT", flist.head = NULL);
5189 	if ((flist.head == NULL) || (flist.head != flist.tail)) {
5190 		bam_error(NULL_FSTYP_A, osdev);
5191 		filelist_free(&flist);
5192 		return (NULL);
5193 	}
5194 
5195 	(void) strtok(flist.head->line, "'");
5196 	cp = strtok(NULL, "'");
5197 	INJECT_ERROR1("GET_POOL_FSTYP_STRTOK", cp = NULL);
5198 	if (cp == NULL) {
5199 		bam_error(BAD_FSTYP_A, osdev);
5200 		filelist_free(&flist);
5201 		return (NULL);
5202 	}
5203 
5204 	pool = s_strdup(cp);
5205 
5206 	filelist_free(&flist);
5207 
5208 	BAM_DPRINTF((D_GET_POOL_RET, fcn, pool));
5209 
5210 	return (pool);
5211 }
5212 
5213 static char *
5214 find_zfs_existing(char *osdev)
5215 {
5216 	char		*pool;
5217 	zfs_mnted_t	mnted;
5218 	char		*mntpt;
5219 	char		*sign;
5220 	const char	*fcn = "find_zfs_existing()";
5221 
5222 	pool = get_pool(osdev);
5223 	INJECT_ERROR1("ZFS_FIND_EXIST_POOL", pool = NULL);
5224 	if (pool == NULL) {
5225 		bam_error(ZFS_GET_POOL_FAILED, osdev);
5226 		return (NULL);
5227 	}
5228 
5229 	mntpt = mount_top_dataset(pool, &mnted);
5230 	INJECT_ERROR1("ZFS_FIND_EXIST_MOUNT_TOP", mntpt = NULL);
5231 	if (mntpt == NULL) {
5232 		bam_error(ZFS_MOUNT_TOP_DATASET_FAILED, pool);
5233 		free(pool);
5234 		return (NULL);
5235 	}
5236 
5237 	sign = find_primary_common(mntpt, "zfs");
5238 	if (sign == NULL) {
5239 		sign = find_backup_common(mntpt, "zfs");
5240 		BAM_DPRINTF((D_EXIST_BACKUP_SIGN, fcn, sign ? sign : "NULL"));
5241 	} else {
5242 		BAM_DPRINTF((D_EXIST_PRIMARY_SIGN, fcn, sign));
5243 	}
5244 
5245 	(void) umount_top_dataset(pool, mnted, mntpt);
5246 
5247 	free(pool);
5248 
5249 	return (sign);
5250 }
5251 
5252 static char *
5253 find_existing_sign(char *osroot, char *osdev, char *fstype)
5254 {
5255 	const char		*fcn = "find_existing_sign()";
5256 
5257 	INJECT_ERROR1("FIND_EXIST_NOTSUP_FS", fstype = "foofs");
5258 	if (strcmp(fstype, "ufs") == 0) {
5259 		BAM_DPRINTF((D_CHECK_UFS_EXIST_SIGN, fcn));
5260 		return (find_ufs_existing(osroot));
5261 	} else if (strcmp(fstype, "zfs") == 0) {
5262 		BAM_DPRINTF((D_CHECK_ZFS_EXIST_SIGN, fcn));
5263 		return (find_zfs_existing(osdev));
5264 	} else {
5265 		bam_error(GRUBSIGN_NOTSUP, fstype);
5266 		return (NULL);
5267 	}
5268 }
5269 
5270 #define	MH_HASH_SZ	16
5271 
5272 typedef enum {
5273 	MH_ERROR = -1,
5274 	MH_NOMATCH,
5275 	MH_MATCH
5276 } mh_search_t;
5277 
5278 typedef struct mcache {
5279 	char	*mc_special;
5280 	char	*mc_mntpt;
5281 	char	*mc_fstype;
5282 	struct mcache *mc_next;
5283 } mcache_t;
5284 
5285 typedef struct mhash {
5286 	mcache_t *mh_hash[MH_HASH_SZ];
5287 } mhash_t;
5288 
5289 static int
5290 mhash_fcn(char *key)
5291 {
5292 	int		i;
5293 	uint64_t	sum = 0;
5294 
5295 	for (i = 0; key[i] != '\0'; i++) {
5296 		sum += (uchar_t)key[i];
5297 	}
5298 
5299 	sum %= MH_HASH_SZ;
5300 
5301 	assert(sum < MH_HASH_SZ);
5302 
5303 	return (sum);
5304 }
5305 
5306 static mhash_t *
5307 cache_mnttab(void)
5308 {
5309 	FILE		*mfp;
5310 	struct extmnttab mnt;
5311 	mcache_t	*mcp;
5312 	mhash_t		*mhp;
5313 	char		*ctds;
5314 	int		idx;
5315 	int		error;
5316 	char		*special_dup;
5317 	const char	*fcn = "cache_mnttab()";
5318 
5319 	mfp = fopen(MNTTAB, "r");
5320 	error = errno;
5321 	INJECT_ERROR1("CACHE_MNTTAB_MNTTAB_ERR", mfp = NULL);
5322 	if (mfp == NULL) {
5323 		bam_error(OPEN_FAIL, MNTTAB, strerror(error));
5324 		return (NULL);
5325 	}
5326 
5327 	mhp = s_calloc(1, sizeof (mhash_t));
5328 
5329 	resetmnttab(mfp);
5330 
5331 	while (getextmntent(mfp, &mnt, sizeof (mnt)) == 0) {
5332 		/* only cache ufs */
5333 		if (strcmp(mnt.mnt_fstype, "ufs") != 0)
5334 			continue;
5335 
5336 		/* basename() modifies its arg, so dup it */
5337 		special_dup = s_strdup(mnt.mnt_special);
5338 		ctds = basename(special_dup);
5339 
5340 		mcp = s_calloc(1, sizeof (mcache_t));
5341 		mcp->mc_special = s_strdup(ctds);
5342 		mcp->mc_mntpt = s_strdup(mnt.mnt_mountp);
5343 		mcp->mc_fstype = s_strdup(mnt.mnt_fstype);
5344 		BAM_DPRINTF((D_CACHE_MNTS, fcn, ctds,
5345 		    mnt.mnt_mountp, mnt.mnt_fstype));
5346 		idx = mhash_fcn(ctds);
5347 		mcp->mc_next = mhp->mh_hash[idx];
5348 		mhp->mh_hash[idx] = mcp;
5349 		free(special_dup);
5350 	}
5351 
5352 	(void) fclose(mfp);
5353 
5354 	return (mhp);
5355 }
5356 
5357 static void
5358 free_mnttab(mhash_t *mhp)
5359 {
5360 	mcache_t	*mcp;
5361 	int		i;
5362 
5363 	for (i = 0; i < MH_HASH_SZ; i++) {
5364 		/*LINTED*/
5365 		while (mcp = mhp->mh_hash[i]) {
5366 			mhp->mh_hash[i] = mcp->mc_next;
5367 			free(mcp->mc_special);
5368 			free(mcp->mc_mntpt);
5369 			free(mcp->mc_fstype);
5370 			free(mcp);
5371 		}
5372 	}
5373 
5374 	for (i = 0; i < MH_HASH_SZ; i++) {
5375 		assert(mhp->mh_hash[i] == NULL);
5376 	}
5377 	free(mhp);
5378 }
5379 
5380 static mh_search_t
5381 search_hash(mhash_t *mhp, char *special, char **mntpt)
5382 {
5383 	int		idx;
5384 	mcache_t	*mcp;
5385 	const char 	*fcn = "search_hash()";
5386 
5387 	assert(mntpt);
5388 
5389 	*mntpt = NULL;
5390 
5391 	INJECT_ERROR1("SEARCH_HASH_FULL_PATH", special = "/foo");
5392 	if (strchr(special, '/')) {
5393 		bam_error(INVALID_MHASH_KEY, special);
5394 		return (MH_ERROR);
5395 	}
5396 
5397 	idx = mhash_fcn(special);
5398 
5399 	for (mcp = mhp->mh_hash[idx]; mcp; mcp = mcp->mc_next) {
5400 		if (strcmp(mcp->mc_special, special) == 0)
5401 			break;
5402 	}
5403 
5404 	if (mcp == NULL) {
5405 		BAM_DPRINTF((D_MNTTAB_HASH_NOMATCH, fcn, special));
5406 		return (MH_NOMATCH);
5407 	}
5408 
5409 	assert(strcmp(mcp->mc_fstype, "ufs") == 0);
5410 	*mntpt = mcp->mc_mntpt;
5411 	BAM_DPRINTF((D_MNTTAB_HASH_MATCH, fcn, special));
5412 	return (MH_MATCH);
5413 }
5414 
5415 static int
5416 check_add_ufs_sign_to_list(FILE *tfp, char *mntpt)
5417 {
5418 	char		*sign;
5419 	char		*signline;
5420 	char		signbuf[MAXNAMELEN];
5421 	int		len;
5422 	int		error;
5423 	const char	*fcn = "check_add_ufs_sign_to_list()";
5424 
5425 	/* safe to specify NULL as "osdev" arg for UFS */
5426 	sign = find_existing_sign(mntpt, NULL, "ufs");
5427 	if (sign == NULL) {
5428 		/* No existing signature, nothing to add to list */
5429 		BAM_DPRINTF((D_NO_SIGN_TO_LIST, fcn, mntpt));
5430 		return (0);
5431 	}
5432 
5433 	(void) snprintf(signbuf, sizeof (signbuf), "%s\n", sign);
5434 	signline = signbuf;
5435 
5436 	INJECT_ERROR1("UFS_MNTPT_SIGN_NOTUFS", signline = "pool_rpool10\n");
5437 	if (strncmp(signline, GRUBSIGN_UFS_PREFIX,
5438 	    strlen(GRUBSIGN_UFS_PREFIX))) {
5439 		bam_error(INVALID_UFS_SIGNATURE, sign);
5440 		free(sign);
5441 		/* ignore invalid signatures */
5442 		return (0);
5443 	}
5444 
5445 	len = fputs(signline, tfp);
5446 	error = errno;
5447 	INJECT_ERROR1("SIGN_LIST_PUTS_ERROR", len = 0);
5448 	if (len != strlen(signline)) {
5449 		bam_error(SIGN_LIST_FPUTS_ERR, sign, strerror(error));
5450 		free(sign);
5451 		return (-1);
5452 	}
5453 
5454 	free(sign);
5455 
5456 	BAM_DPRINTF((D_SIGN_LIST_PUTS_DONE, fcn, mntpt));
5457 	return (0);
5458 }
5459 
5460 /*
5461  * slice is a basename not a full pathname
5462  */
5463 static int
5464 process_slice_common(char *slice, FILE *tfp, mhash_t *mhp, char *tmpmnt)
5465 {
5466 	int		ret;
5467 	char		cmd[PATH_MAX];
5468 	char		path[PATH_MAX];
5469 	struct stat	sbuf;
5470 	char		*mntpt;
5471 	filelist_t	flist = {0};
5472 	char		*fstype;
5473 	char		blkslice[PATH_MAX];
5474 	const char	*fcn = "process_slice_common()";
5475 
5476 
5477 	ret = search_hash(mhp, slice, &mntpt);
5478 	switch (ret) {
5479 		case MH_MATCH:
5480 			if (check_add_ufs_sign_to_list(tfp, mntpt) == -1)
5481 				return (-1);
5482 			else
5483 				return (0);
5484 		case MH_NOMATCH:
5485 			break;
5486 		case MH_ERROR:
5487 		default:
5488 			return (-1);
5489 	}
5490 
5491 	(void) snprintf(path, sizeof (path), "/dev/rdsk/%s", slice);
5492 	if (stat(path, &sbuf) == -1) {
5493 		BAM_DPRINTF((D_SLICE_ENOENT, fcn, path));
5494 		return (0);
5495 	}
5496 
5497 	/* Check if ufs */
5498 	(void) snprintf(cmd, sizeof (cmd),
5499 	    "/usr/sbin/fstyp /dev/rdsk/%s 2>/dev/null",
5500 	    slice);
5501 
5502 	if (exec_cmd(cmd, &flist) != 0) {
5503 		if (bam_verbose)
5504 			bam_print(FSTYP_FAILED, slice);
5505 		return (0);
5506 	}
5507 
5508 	if ((flist.head == NULL) || (flist.head != flist.tail)) {
5509 		if (bam_verbose)
5510 			bam_print(FSTYP_BAD, slice);
5511 		filelist_free(&flist);
5512 		return (0);
5513 	}
5514 
5515 	fstype = strtok(flist.head->line, " \t\n");
5516 	if (fstype == NULL || strcmp(fstype, "ufs") != 0) {
5517 		if (bam_verbose)
5518 			bam_print(NOT_UFS_SLICE, slice, fstype);
5519 		filelist_free(&flist);
5520 		return (0);
5521 	}
5522 
5523 	filelist_free(&flist);
5524 
5525 	/*
5526 	 * Since we are mounting the filesystem read-only, the
5527 	 * the last mount field of the superblock is unchanged
5528 	 * and does not need to be fixed up post-mount;
5529 	 */
5530 
5531 	(void) snprintf(blkslice, sizeof (blkslice), "/dev/dsk/%s",
5532 	    slice);
5533 
5534 	(void) snprintf(cmd, sizeof (cmd),
5535 	    "/usr/sbin/mount -F ufs -o ro %s %s "
5536 	    "> /dev/null 2>&1", blkslice, tmpmnt);
5537 
5538 	if (exec_cmd(cmd, NULL) != 0) {
5539 		if (bam_verbose)
5540 			bam_print(MOUNT_FAILED, blkslice, "ufs");
5541 		return (0);
5542 	}
5543 
5544 	ret = check_add_ufs_sign_to_list(tfp, tmpmnt);
5545 
5546 	(void) snprintf(cmd, sizeof (cmd),
5547 	    "/usr/sbin/umount -f %s > /dev/null 2>&1",
5548 	    tmpmnt);
5549 
5550 	if (exec_cmd(cmd, NULL) != 0) {
5551 		bam_print(UMOUNT_FAILED, slice);
5552 		return (0);
5553 	}
5554 
5555 	return (ret);
5556 }
5557 
5558 static int
5559 process_vtoc_slices(
5560 	char *s0,
5561 	struct vtoc *vtoc,
5562 	FILE *tfp,
5563 	mhash_t *mhp,
5564 	char *tmpmnt)
5565 {
5566 	int		idx;
5567 	char		slice[PATH_MAX];
5568 	size_t		len;
5569 	char		*cp;
5570 	const char	*fcn = "process_vtoc_slices()";
5571 
5572 	len = strlen(s0);
5573 
5574 	assert(s0[len - 2] == 's' && s0[len - 1] == '0');
5575 
5576 	s0[len - 1] = '\0';
5577 
5578 	(void) strlcpy(slice, s0, sizeof (slice));
5579 
5580 	s0[len - 1] = '0';
5581 
5582 	cp = slice + len - 1;
5583 
5584 	for (idx = 0; idx < vtoc->v_nparts; idx++) {
5585 
5586 		(void) snprintf(cp, sizeof (slice) - (len - 1), "%u", idx);
5587 
5588 		if (vtoc->v_part[idx].p_size == 0) {
5589 			BAM_DPRINTF((D_VTOC_SIZE_ZERO, fcn, slice));
5590 			continue;
5591 		}
5592 
5593 		/* Skip "SWAP", "USR", "BACKUP", "VAR", "HOME", "ALTSCTR" */
5594 		switch (vtoc->v_part[idx].p_tag) {
5595 		case V_SWAP:
5596 		case V_USR:
5597 		case V_BACKUP:
5598 		case V_VAR:
5599 		case V_HOME:
5600 		case V_ALTSCTR:
5601 			BAM_DPRINTF((D_VTOC_NOT_ROOT_TAG, fcn, slice));
5602 			continue;
5603 		default:
5604 			BAM_DPRINTF((D_VTOC_ROOT_TAG, fcn, slice));
5605 			break;
5606 		}
5607 
5608 		/* skip unmountable and readonly slices */
5609 		switch (vtoc->v_part[idx].p_flag) {
5610 		case V_UNMNT:
5611 		case V_RONLY:
5612 			BAM_DPRINTF((D_VTOC_NOT_RDWR_FLAG, fcn, slice));
5613 			continue;
5614 		default:
5615 			BAM_DPRINTF((D_VTOC_RDWR_FLAG, fcn, slice));
5616 			break;
5617 		}
5618 
5619 		if (process_slice_common(slice, tfp, mhp, tmpmnt) == -1) {
5620 			return (-1);
5621 		}
5622 	}
5623 
5624 	return (0);
5625 }
5626 
5627 static int
5628 process_efi_slices(
5629 	char *s0,
5630 	struct dk_gpt *efi,
5631 	FILE *tfp,
5632 	mhash_t *mhp,
5633 	char *tmpmnt)
5634 {
5635 	int		idx;
5636 	char		slice[PATH_MAX];
5637 	size_t		len;
5638 	char		*cp;
5639 	const char	*fcn = "process_efi_slices()";
5640 
5641 	len = strlen(s0);
5642 
5643 	assert(s0[len - 2] == 's' && s0[len - 1] == '0');
5644 
5645 	s0[len - 1] = '\0';
5646 
5647 	(void) strlcpy(slice, s0, sizeof (slice));
5648 
5649 	s0[len - 1] = '0';
5650 
5651 	cp = slice + len - 1;
5652 
5653 	for (idx = 0; idx < efi->efi_nparts; idx++) {
5654 
5655 		(void) snprintf(cp, sizeof (slice) - (len - 1), "%u", idx);
5656 
5657 		if (efi->efi_parts[idx].p_size == 0) {
5658 			BAM_DPRINTF((D_EFI_SIZE_ZERO, fcn, slice));
5659 			continue;
5660 		}
5661 
5662 		/* Skip "SWAP", "USR", "BACKUP", "VAR", "HOME", "ALTSCTR" */
5663 		switch (efi->efi_parts[idx].p_tag) {
5664 		case V_SWAP:
5665 		case V_USR:
5666 		case V_BACKUP:
5667 		case V_VAR:
5668 		case V_HOME:
5669 		case V_ALTSCTR:
5670 			BAM_DPRINTF((D_EFI_NOT_ROOT_TAG, fcn, slice));
5671 			continue;
5672 		default:
5673 			BAM_DPRINTF((D_EFI_ROOT_TAG, fcn, slice));
5674 			break;
5675 		}
5676 
5677 		/* skip unmountable and readonly slices */
5678 		switch (efi->efi_parts[idx].p_flag) {
5679 		case V_UNMNT:
5680 		case V_RONLY:
5681 			BAM_DPRINTF((D_EFI_NOT_RDWR_FLAG, fcn, slice));
5682 			continue;
5683 		default:
5684 			BAM_DPRINTF((D_EFI_RDWR_FLAG, fcn, slice));
5685 			break;
5686 		}
5687 
5688 		if (process_slice_common(slice, tfp, mhp, tmpmnt) == -1) {
5689 			return (-1);
5690 		}
5691 	}
5692 
5693 	return (0);
5694 }
5695 
5696 /*
5697  * s0 is a basename not a full path
5698  */
5699 static int
5700 process_slice0(char *s0, FILE *tfp, mhash_t *mhp, char *tmpmnt)
5701 {
5702 	struct vtoc		vtoc;
5703 	struct dk_gpt		*efi;
5704 	char			s0path[PATH_MAX];
5705 	struct stat		sbuf;
5706 	int			e_flag;
5707 	int			v_flag;
5708 	int			retval;
5709 	int			err;
5710 	int			fd;
5711 	const char		*fcn = "process_slice0()";
5712 
5713 	(void) snprintf(s0path, sizeof (s0path), "/dev/rdsk/%s", s0);
5714 
5715 	if (stat(s0path, &sbuf) == -1) {
5716 		BAM_DPRINTF((D_SLICE0_ENOENT, fcn, s0path));
5717 		return (0);
5718 	}
5719 
5720 	fd = open(s0path, O_NONBLOCK|O_RDONLY);
5721 	if (fd == -1) {
5722 		bam_error(OPEN_FAIL, s0path, strerror(errno));
5723 		return (0);
5724 	}
5725 
5726 	e_flag = v_flag = 0;
5727 	retval = ((err = read_vtoc(fd, &vtoc)) >= 0) ? 0 : err;
5728 	switch (retval) {
5729 		case VT_EIO:
5730 			BAM_DPRINTF((D_VTOC_READ_FAIL, fcn, s0path));
5731 			break;
5732 		case VT_EINVAL:
5733 			BAM_DPRINTF((D_VTOC_INVALID, fcn, s0path));
5734 			break;
5735 		case VT_ERROR:
5736 			BAM_DPRINTF((D_VTOC_UNKNOWN_ERR, fcn, s0path));
5737 			break;
5738 		case VT_ENOTSUP:
5739 			e_flag = 1;
5740 			BAM_DPRINTF((D_VTOC_NOTSUP, fcn, s0path));
5741 			break;
5742 		case 0:
5743 			v_flag = 1;
5744 			BAM_DPRINTF((D_VTOC_READ_SUCCESS, fcn, s0path));
5745 			break;
5746 		default:
5747 			BAM_DPRINTF((D_VTOC_UNKNOWN_RETCODE, fcn, s0path));
5748 			break;
5749 	}
5750 
5751 
5752 	if (e_flag) {
5753 		e_flag = 0;
5754 		retval = ((err = efi_alloc_and_read(fd, &efi)) >= 0) ? 0 : err;
5755 		switch (retval) {
5756 		case VT_EIO:
5757 			BAM_DPRINTF((D_EFI_READ_FAIL, fcn, s0path));
5758 			break;
5759 		case VT_EINVAL:
5760 			BAM_DPRINTF((D_EFI_INVALID, fcn, s0path));
5761 			break;
5762 		case VT_ERROR:
5763 			BAM_DPRINTF((D_EFI_UNKNOWN_ERR, fcn, s0path));
5764 			break;
5765 		case VT_ENOTSUP:
5766 			BAM_DPRINTF((D_EFI_NOTSUP, fcn, s0path));
5767 			break;
5768 		case 0:
5769 			e_flag = 1;
5770 			BAM_DPRINTF((D_EFI_READ_SUCCESS, fcn, s0path));
5771 			break;
5772 		default:
5773 			BAM_DPRINTF((D_EFI_UNKNOWN_RETCODE, fcn, s0path));
5774 			break;
5775 		}
5776 	}
5777 
5778 	(void) close(fd);
5779 
5780 	if (v_flag) {
5781 		retval = process_vtoc_slices(s0,
5782 		    &vtoc, tfp, mhp, tmpmnt);
5783 	} else if (e_flag) {
5784 		retval = process_efi_slices(s0,
5785 		    efi, tfp, mhp, tmpmnt);
5786 	} else {
5787 		BAM_DPRINTF((D_NOT_VTOC_OR_EFI, fcn, s0path));
5788 		return (0);
5789 	}
5790 
5791 	return (retval);
5792 }
5793 
5794 /*
5795  * Find and create a list of all existing UFS boot signatures
5796  */
5797 static int
5798 FindAllUfsSignatures(void)
5799 {
5800 	mhash_t		*mnttab_hash;
5801 	DIR		*dirp = NULL;
5802 	struct dirent	*dp;
5803 	char		tmpmnt[PATH_MAX];
5804 	char		cmd[PATH_MAX];
5805 	struct stat	sb;
5806 	int		fd;
5807 	FILE		*tfp;
5808 	size_t		len;
5809 	int		ret;
5810 	int		error;
5811 	const char	*fcn = "FindAllUfsSignatures()";
5812 
5813 	if (stat(UFS_SIGNATURE_LIST, &sb) != -1)  {
5814 		bam_print(SIGNATURE_LIST_EXISTS, UFS_SIGNATURE_LIST);
5815 		return (0);
5816 	}
5817 
5818 	fd = open(UFS_SIGNATURE_LIST".tmp",
5819 	    O_RDWR|O_CREAT|O_TRUNC, 0644);
5820 	error = errno;
5821 	INJECT_ERROR1("SIGN_LIST_TMP_TRUNC", fd = -1);
5822 	if (fd == -1) {
5823 		bam_error(OPEN_FAIL, UFS_SIGNATURE_LIST".tmp", strerror(error));
5824 		return (-1);
5825 	}
5826 
5827 	ret = close(fd);
5828 	error = errno;
5829 	INJECT_ERROR1("SIGN_LIST_TMP_CLOSE", ret = -1);
5830 	if (ret == -1) {
5831 		bam_error(CLOSE_FAIL, UFS_SIGNATURE_LIST".tmp",
5832 		    strerror(error));
5833 		(void) unlink(UFS_SIGNATURE_LIST".tmp");
5834 		return (-1);
5835 	}
5836 
5837 	tfp = fopen(UFS_SIGNATURE_LIST".tmp", "a");
5838 	error = errno;
5839 	INJECT_ERROR1("SIGN_LIST_APPEND_FOPEN", tfp = NULL);
5840 	if (tfp == NULL) {
5841 		bam_error(OPEN_FAIL, UFS_SIGNATURE_LIST".tmp", strerror(error));
5842 		(void) unlink(UFS_SIGNATURE_LIST".tmp");
5843 		return (-1);
5844 	}
5845 
5846 	mnttab_hash = cache_mnttab();
5847 	INJECT_ERROR1("CACHE_MNTTAB_ERROR", mnttab_hash = NULL);
5848 	if (mnttab_hash == NULL) {
5849 		(void) fclose(tfp);
5850 		(void) unlink(UFS_SIGNATURE_LIST".tmp");
5851 		bam_error(CACHE_MNTTAB_FAIL, fcn);
5852 		return (-1);
5853 	}
5854 
5855 	(void) snprintf(tmpmnt, sizeof (tmpmnt),
5856 	    "/tmp/bootadm_ufs_sign_mnt.%d", getpid());
5857 	(void) unlink(tmpmnt);
5858 
5859 	ret = mkdirp(tmpmnt, DIR_PERMS);
5860 	error = errno;
5861 	INJECT_ERROR1("MKDIRP_SIGN_MNT", ret = -1);
5862 	if (ret == -1) {
5863 		bam_error(MKDIR_FAILED, tmpmnt, strerror(error));
5864 		free_mnttab(mnttab_hash);
5865 		(void) fclose(tfp);
5866 		(void) unlink(UFS_SIGNATURE_LIST".tmp");
5867 		return (-1);
5868 	}
5869 
5870 	dirp = opendir("/dev/rdsk");
5871 	error = errno;
5872 	INJECT_ERROR1("OPENDIR_DEV_RDSK", dirp = NULL);
5873 	if (dirp == NULL) {
5874 		bam_error(OPENDIR_FAILED, "/dev/rdsk", strerror(error));
5875 		goto fail;
5876 	}
5877 
5878 	while (dp = readdir(dirp)) {
5879 		if (strcmp(dp->d_name, ".") == 0 ||
5880 		    strcmp(dp->d_name, "..") == 0)
5881 			continue;
5882 
5883 		/*
5884 		 * we only look for the s0 slice. This is guranteed to
5885 		 * have 's' at len - 2.
5886 		 */
5887 		len = strlen(dp->d_name);
5888 		if (dp->d_name[len - 2 ] != 's' || dp->d_name[len - 1] != '0') {
5889 			BAM_DPRINTF((D_SKIP_SLICE_NOTZERO, fcn, dp->d_name));
5890 			continue;
5891 		}
5892 
5893 		ret = process_slice0(dp->d_name, tfp, mnttab_hash, tmpmnt);
5894 		INJECT_ERROR1("PROCESS_S0_FAIL", ret = -1);
5895 		if (ret == -1)
5896 			goto fail;
5897 	}
5898 
5899 	(void) closedir(dirp);
5900 	free_mnttab(mnttab_hash);
5901 	(void) rmdir(tmpmnt);
5902 
5903 	ret = fclose(tfp);
5904 	error = errno;
5905 	INJECT_ERROR1("FCLOSE_SIGNLIST_TMP", ret = EOF);
5906 	if (ret == EOF) {
5907 		bam_error(CLOSE_FAIL, UFS_SIGNATURE_LIST".tmp",
5908 		    strerror(error));
5909 		(void) unlink(UFS_SIGNATURE_LIST".tmp");
5910 		return (-1);
5911 	}
5912 
5913 	/* We have a list of existing GRUB signatures. Sort it first */
5914 	(void) snprintf(cmd, sizeof (cmd),
5915 	    "/usr/bin/sort -u %s.tmp > %s.sorted",
5916 	    UFS_SIGNATURE_LIST, UFS_SIGNATURE_LIST);
5917 
5918 	ret = exec_cmd(cmd, NULL);
5919 	INJECT_ERROR1("SORT_SIGN_LIST", ret = 1);
5920 	if (ret != 0) {
5921 		bam_error(GRUBSIGN_SORT_FAILED);
5922 		(void) unlink(UFS_SIGNATURE_LIST".sorted");
5923 		(void) unlink(UFS_SIGNATURE_LIST".tmp");
5924 		return (-1);
5925 	}
5926 
5927 	(void) unlink(UFS_SIGNATURE_LIST".tmp");
5928 
5929 	ret = rename(UFS_SIGNATURE_LIST".sorted", UFS_SIGNATURE_LIST);
5930 	error = errno;
5931 	INJECT_ERROR1("RENAME_TMP_SIGNLIST", ret = -1);
5932 	if (ret == -1) {
5933 		bam_error(RENAME_FAIL, UFS_SIGNATURE_LIST, strerror(error));
5934 		(void) unlink(UFS_SIGNATURE_LIST".sorted");
5935 		return (-1);
5936 	}
5937 
5938 	if (stat(UFS_SIGNATURE_LIST, &sb) == 0 && sb.st_size == 0) {
5939 		BAM_DPRINTF((D_ZERO_LEN_SIGNLIST, fcn, UFS_SIGNATURE_LIST));
5940 	}
5941 
5942 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
5943 	return (0);
5944 
5945 fail:
5946 	if (dirp)
5947 		(void) closedir(dirp);
5948 	free_mnttab(mnttab_hash);
5949 	(void) rmdir(tmpmnt);
5950 	(void) fclose(tfp);
5951 	(void) unlink(UFS_SIGNATURE_LIST".tmp");
5952 	BAM_DPRINTF((D_RETURN_FAILURE, fcn));
5953 	return (-1);
5954 }
5955 
5956 static char *
5957 create_ufs_sign(void)
5958 {
5959 	struct stat	sb;
5960 	int		signnum = -1;
5961 	char		tmpsign[MAXNAMELEN + 1];
5962 	char		*numstr;
5963 	int		i;
5964 	FILE		*tfp;
5965 	int		ret;
5966 	int		error;
5967 	const char	*fcn = "create_ufs_sign()";
5968 
5969 	bam_print(SEARCHING_UFS_SIGN);
5970 
5971 	ret = FindAllUfsSignatures();
5972 	INJECT_ERROR1("FIND_ALL_UFS", ret = -1);
5973 	if (ret == -1) {
5974 		bam_error(ERR_FIND_UFS_SIGN);
5975 		return (NULL);
5976 	}
5977 
5978 	/* Make sure the list exists and is owned by root */
5979 	INJECT_ERROR1("SIGNLIST_NOT_CREATED",
5980 	    (void) unlink(UFS_SIGNATURE_LIST));
5981 	if (stat(UFS_SIGNATURE_LIST, &sb) == -1 || sb.st_uid != 0) {
5982 		(void) unlink(UFS_SIGNATURE_LIST);
5983 		bam_error(UFS_SIGNATURE_LIST_MISS, UFS_SIGNATURE_LIST);
5984 		return (NULL);
5985 	}
5986 
5987 	if (sb.st_size == 0) {
5988 		bam_print(GRUBSIGN_UFS_NONE);
5989 		i = 0;
5990 		goto found;
5991 	}
5992 
5993 	/* The signature list was sorted when it was created */
5994 	tfp = fopen(UFS_SIGNATURE_LIST, "r");
5995 	error = errno;
5996 	INJECT_ERROR1("FOPEN_SIGN_LIST", tfp = NULL);
5997 	if (tfp == NULL) {
5998 		bam_error(UFS_SIGNATURE_LIST_OPENERR,
5999 		    UFS_SIGNATURE_LIST, strerror(error));
6000 		(void) unlink(UFS_SIGNATURE_LIST);
6001 		return (NULL);
6002 	}
6003 
6004 	for (i = 0; s_fgets(tmpsign, sizeof (tmpsign), tfp); i++) {
6005 
6006 		if (strncmp(tmpsign, GRUBSIGN_UFS_PREFIX,
6007 		    strlen(GRUBSIGN_UFS_PREFIX)) != 0) {
6008 			(void) fclose(tfp);
6009 			(void) unlink(UFS_SIGNATURE_LIST);
6010 			bam_error(UFS_BADSIGN, tmpsign);
6011 			return (NULL);
6012 		}
6013 		numstr = tmpsign + strlen(GRUBSIGN_UFS_PREFIX);
6014 
6015 		if (numstr[0] == '\0' || !isdigit(numstr[0])) {
6016 			(void) fclose(tfp);
6017 			(void) unlink(UFS_SIGNATURE_LIST);
6018 			bam_error(UFS_BADSIGN, tmpsign);
6019 			return (NULL);
6020 		}
6021 
6022 		signnum = atoi(numstr);
6023 		INJECT_ERROR1("NEGATIVE_SIGN", signnum = -1);
6024 		if (signnum < 0) {
6025 			(void) fclose(tfp);
6026 			(void) unlink(UFS_SIGNATURE_LIST);
6027 			bam_error(UFS_BADSIGN, tmpsign);
6028 			return (NULL);
6029 		}
6030 
6031 		if (i != signnum) {
6032 			BAM_DPRINTF((D_FOUND_HOLE_SIGNLIST, fcn, i));
6033 			break;
6034 		}
6035 	}
6036 
6037 	(void) fclose(tfp);
6038 
6039 found:
6040 	(void) snprintf(tmpsign, sizeof (tmpsign), "rootfs%d", i);
6041 
6042 	/* add the ufs signature to the /var/run list of signatures */
6043 	ret = ufs_add_to_sign_list(tmpsign);
6044 	INJECT_ERROR1("UFS_ADD_TO_SIGN_LIST", ret = -1);
6045 	if (ret == -1) {
6046 		(void) unlink(UFS_SIGNATURE_LIST);
6047 		bam_error(FAILED_ADD_SIGNLIST, tmpsign);
6048 		return (NULL);
6049 	}
6050 
6051 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6052 
6053 	return (s_strdup(tmpsign));
6054 }
6055 
6056 static char *
6057 get_fstype(char *osroot)
6058 {
6059 	FILE		*mntfp;
6060 	struct mnttab	mp = {0};
6061 	struct mnttab	mpref = {0};
6062 	int		error;
6063 	int		ret;
6064 	const char	*fcn = "get_fstype()";
6065 
6066 	INJECT_ERROR1("GET_FSTYPE_OSROOT", osroot = NULL);
6067 	if (osroot == NULL) {
6068 		bam_error(GET_FSTYPE_ARGS);
6069 		return (NULL);
6070 	}
6071 
6072 	mntfp = fopen(MNTTAB, "r");
6073 	error = errno;
6074 	INJECT_ERROR1("GET_FSTYPE_FOPEN", mntfp = NULL);
6075 	if (mntfp == NULL) {
6076 		bam_error(OPEN_FAIL, MNTTAB, strerror(error));
6077 		return (NULL);
6078 	}
6079 
6080 	if (*osroot == '\0')
6081 		mpref.mnt_mountp = "/";
6082 	else
6083 		mpref.mnt_mountp = osroot;
6084 
6085 	ret = getmntany(mntfp, &mp, &mpref);
6086 	INJECT_ERROR1("GET_FSTYPE_GETMNTANY", ret = 1);
6087 	if (ret != 0) {
6088 		bam_error(MNTTAB_MNTPT_NOT_FOUND, osroot, MNTTAB);
6089 		(void) fclose(mntfp);
6090 		return (NULL);
6091 	}
6092 	(void) fclose(mntfp);
6093 
6094 	INJECT_ERROR1("GET_FSTYPE_NULL", mp.mnt_fstype = NULL);
6095 	if (mp.mnt_fstype == NULL) {
6096 		bam_error(MNTTAB_FSTYPE_NULL, osroot);
6097 		return (NULL);
6098 	}
6099 
6100 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6101 
6102 	return (s_strdup(mp.mnt_fstype));
6103 }
6104 
6105 static char *
6106 create_zfs_sign(char *osdev)
6107 {
6108 	char		tmpsign[PATH_MAX];
6109 	char		*pool;
6110 	const char	*fcn = "create_zfs_sign()";
6111 
6112 	BAM_DPRINTF((D_FUNC_ENTRY1, fcn, osdev));
6113 
6114 	/*
6115 	 * First find the pool name
6116 	 */
6117 	pool = get_pool(osdev);
6118 	INJECT_ERROR1("CREATE_ZFS_SIGN_GET_POOL", pool = NULL);
6119 	if (pool == NULL) {
6120 		bam_error(GET_POOL_FAILED, osdev);
6121 		return (NULL);
6122 	}
6123 
6124 	(void) snprintf(tmpsign, sizeof (tmpsign), "pool_%s", pool);
6125 
6126 	BAM_DPRINTF((D_CREATED_ZFS_SIGN, fcn, tmpsign));
6127 
6128 	free(pool);
6129 
6130 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6131 
6132 	return (s_strdup(tmpsign));
6133 }
6134 
6135 static char *
6136 create_new_sign(char *osdev, char *fstype)
6137 {
6138 	char		*sign;
6139 	const char	*fcn = "create_new_sign()";
6140 
6141 	INJECT_ERROR1("NEW_SIGN_FSTYPE", fstype = "foofs");
6142 
6143 	if (strcmp(fstype, "zfs") == 0) {
6144 		BAM_DPRINTF((D_CREATE_NEW_ZFS, fcn));
6145 		sign = create_zfs_sign(osdev);
6146 	} else if (strcmp(fstype, "ufs") == 0) {
6147 		BAM_DPRINTF((D_CREATE_NEW_UFS, fcn));
6148 		sign = create_ufs_sign();
6149 	} else {
6150 		bam_error(GRUBSIGN_NOTSUP, fstype);
6151 		sign = NULL;
6152 	}
6153 
6154 	BAM_DPRINTF((D_CREATED_NEW_SIGN, fcn, sign ? sign : "<NULL>"));
6155 	return (sign);
6156 }
6157 
6158 static int
6159 set_backup_common(char *mntpt, char *sign)
6160 {
6161 	FILE		*bfp;
6162 	char		backup[PATH_MAX];
6163 	char		tmpsign[PATH_MAX];
6164 	int		error;
6165 	char		*bdir;
6166 	char		*backup_dup;
6167 	struct stat	sb;
6168 	int		ret;
6169 	const char	*fcn = "set_backup_common()";
6170 
6171 	(void) snprintf(backup, sizeof (backup), "%s%s",
6172 	    mntpt, GRUBSIGN_BACKUP);
6173 
6174 	/* First read the backup */
6175 	bfp = fopen(backup, "r");
6176 	if (bfp != NULL) {
6177 		while (s_fgets(tmpsign, sizeof (tmpsign), bfp)) {
6178 			if (strcmp(tmpsign, sign) == 0) {
6179 				BAM_DPRINTF((D_FOUND_IN_BACKUP, fcn, sign));
6180 				(void) fclose(bfp);
6181 				return (0);
6182 			}
6183 		}
6184 		(void) fclose(bfp);
6185 		BAM_DPRINTF((D_NOT_FOUND_IN_EXIST_BACKUP, fcn, sign));
6186 	} else {
6187 		BAM_DPRINTF((D_BACKUP_NOT_EXIST, fcn, backup));
6188 	}
6189 
6190 	/*
6191 	 * Didn't find the correct signature. First create
6192 	 * the directory if necessary.
6193 	 */
6194 
6195 	/* dirname() modifies its argument so dup it */
6196 	backup_dup = s_strdup(backup);
6197 	bdir = dirname(backup_dup);
6198 	assert(bdir);
6199 
6200 	ret = stat(bdir, &sb);
6201 	INJECT_ERROR1("SET_BACKUP_STAT", ret = -1);
6202 	if (ret == -1) {
6203 		BAM_DPRINTF((D_BACKUP_DIR_NOEXIST, fcn, bdir));
6204 		ret = mkdirp(bdir, DIR_PERMS);
6205 		error = errno;
6206 		INJECT_ERROR1("SET_BACKUP_MKDIRP", ret = -1);
6207 		if (ret == -1) {
6208 			bam_error(GRUBSIGN_BACKUP_MKDIRERR,
6209 			    GRUBSIGN_BACKUP, strerror(error));
6210 			free(backup_dup);
6211 			return (-1);
6212 		}
6213 	}
6214 	free(backup_dup);
6215 
6216 	/*
6217 	 * Open the backup in append mode to add the correct
6218 	 * signature;
6219 	 */
6220 	bfp = fopen(backup, "a");
6221 	error = errno;
6222 	INJECT_ERROR1("SET_BACKUP_FOPEN_A", bfp = NULL);
6223 	if (bfp == NULL) {
6224 		bam_error(GRUBSIGN_BACKUP_OPENERR,
6225 		    GRUBSIGN_BACKUP, strerror(error));
6226 		return (-1);
6227 	}
6228 
6229 	(void) snprintf(tmpsign, sizeof (tmpsign), "%s\n", sign);
6230 
6231 	ret = fputs(tmpsign, bfp);
6232 	error = errno;
6233 	INJECT_ERROR1("SET_BACKUP_FPUTS", ret = 0);
6234 	if (ret != strlen(tmpsign)) {
6235 		bam_error(GRUBSIGN_BACKUP_WRITEERR,
6236 		    GRUBSIGN_BACKUP, strerror(error));
6237 		(void) fclose(bfp);
6238 		return (-1);
6239 	}
6240 
6241 	(void) fclose(bfp);
6242 
6243 	if (bam_verbose)
6244 		bam_print(GRUBSIGN_BACKUP_UPDATED, GRUBSIGN_BACKUP);
6245 
6246 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6247 
6248 	return (0);
6249 }
6250 
6251 static int
6252 set_backup_ufs(char *osroot, char *sign)
6253 {
6254 	const char	*fcn = "set_backup_ufs()";
6255 
6256 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, osroot, sign));
6257 	return (set_backup_common(osroot, sign));
6258 }
6259 
6260 static int
6261 set_backup_zfs(char *osdev, char *sign)
6262 {
6263 	char		*pool;
6264 	char		*mntpt;
6265 	zfs_mnted_t	mnted;
6266 	int		ret;
6267 	const char	*fcn = "set_backup_zfs()";
6268 
6269 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, osdev, sign));
6270 
6271 	pool = get_pool(osdev);
6272 	INJECT_ERROR1("SET_BACKUP_GET_POOL", pool = NULL);
6273 	if (pool == NULL) {
6274 		bam_error(GET_POOL_FAILED, osdev);
6275 		return (-1);
6276 	}
6277 
6278 	mntpt = mount_top_dataset(pool, &mnted);
6279 	INJECT_ERROR1("SET_BACKUP_MOUNT_DATASET", mntpt = NULL);
6280 	if (mntpt == NULL) {
6281 		bam_error(FAIL_MNT_TOP_DATASET, pool);
6282 		free(pool);
6283 		return (-1);
6284 	}
6285 
6286 	ret = set_backup_common(mntpt, sign);
6287 
6288 	(void) umount_top_dataset(pool, mnted, mntpt);
6289 
6290 	free(pool);
6291 
6292 	INJECT_ERROR1("SET_BACKUP_ZFS_FAIL", ret = 1);
6293 	if (ret == 0) {
6294 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6295 	} else {
6296 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
6297 	}
6298 
6299 	return (ret);
6300 }
6301 
6302 static int
6303 set_backup(char *osroot, char *osdev, char *sign, char *fstype)
6304 {
6305 	const char	*fcn = "set_backup()";
6306 	int		ret;
6307 
6308 	INJECT_ERROR1("SET_BACKUP_FSTYPE", fstype = "foofs");
6309 
6310 	if (strcmp(fstype, "ufs") == 0) {
6311 		BAM_DPRINTF((D_SET_BACKUP_UFS, fcn));
6312 		ret = set_backup_ufs(osroot, sign);
6313 	} else if (strcmp(fstype, "zfs") == 0) {
6314 		BAM_DPRINTF((D_SET_BACKUP_ZFS, fcn));
6315 		ret = set_backup_zfs(osdev, sign);
6316 	} else {
6317 		bam_error(GRUBSIGN_NOTSUP, fstype);
6318 		ret = -1;
6319 	}
6320 
6321 	if (ret == 0) {
6322 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6323 	} else {
6324 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
6325 	}
6326 
6327 	return (ret);
6328 }
6329 
6330 static int
6331 set_primary_common(char *mntpt, char *sign)
6332 {
6333 	char		signfile[PATH_MAX];
6334 	char		signdir[PATH_MAX];
6335 	struct stat	sb;
6336 	int		fd;
6337 	int		error;
6338 	int		ret;
6339 	const char	*fcn = "set_primary_common()";
6340 
6341 	(void) snprintf(signfile, sizeof (signfile), "%s/%s/%s",
6342 	    mntpt, GRUBSIGN_DIR, sign);
6343 
6344 	if (stat(signfile, &sb) != -1) {
6345 		if (bam_verbose)
6346 			bam_print(PRIMARY_SIGN_EXISTS, sign);
6347 		return (0);
6348 	} else {
6349 		BAM_DPRINTF((D_PRIMARY_NOT_EXIST, fcn, signfile));
6350 	}
6351 
6352 	(void) snprintf(signdir, sizeof (signdir), "%s/%s",
6353 	    mntpt, GRUBSIGN_DIR);
6354 
6355 	if (stat(signdir, &sb) == -1) {
6356 		BAM_DPRINTF((D_PRIMARY_DIR_NOEXIST, fcn, signdir));
6357 		ret = mkdirp(signdir, DIR_PERMS);
6358 		error = errno;
6359 		INJECT_ERROR1("SET_PRIMARY_MKDIRP", ret = -1);
6360 		if (ret == -1) {
6361 			bam_error(GRUBSIGN_MKDIR_ERR, signdir, strerror(errno));
6362 			return (-1);
6363 		}
6364 	}
6365 
6366 	fd = open(signfile, O_RDWR|O_CREAT|O_TRUNC, 0444);
6367 	error = errno;
6368 	INJECT_ERROR1("PRIMARY_SIGN_CREAT", fd = -1);
6369 	if (fd == -1) {
6370 		bam_error(GRUBSIGN_PRIMARY_CREATERR, signfile, strerror(error));
6371 		return (-1);
6372 	}
6373 
6374 	ret = fsync(fd);
6375 	error = errno;
6376 	INJECT_ERROR1("PRIMARY_FSYNC", ret = -1);
6377 	if (ret != 0) {
6378 		bam_error(GRUBSIGN_PRIMARY_SYNCERR, signfile, strerror(error));
6379 	}
6380 
6381 	(void) close(fd);
6382 
6383 	if (bam_verbose)
6384 		bam_print(GRUBSIGN_CREATED_PRIMARY, signfile);
6385 
6386 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6387 
6388 	return (0);
6389 }
6390 
6391 static int
6392 set_primary_ufs(char *osroot, char *sign)
6393 {
6394 	const char	*fcn = "set_primary_ufs()";
6395 
6396 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, osroot, sign));
6397 	return (set_primary_common(osroot, sign));
6398 }
6399 
6400 static int
6401 set_primary_zfs(char *osdev, char *sign)
6402 {
6403 	char		*pool;
6404 	char		*mntpt;
6405 	zfs_mnted_t	mnted;
6406 	int		ret;
6407 	const char	*fcn = "set_primary_zfs()";
6408 
6409 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, osdev, sign));
6410 
6411 	pool = get_pool(osdev);
6412 	INJECT_ERROR1("SET_PRIMARY_ZFS_GET_POOL", pool = NULL);
6413 	if (pool == NULL) {
6414 		bam_error(GET_POOL_FAILED, osdev);
6415 		return (-1);
6416 	}
6417 
6418 	/* Pool name must exist in the sign */
6419 	ret = (strstr(sign, pool) != NULL);
6420 	INJECT_ERROR1("SET_PRIMARY_ZFS_POOL_SIGN_INCOMPAT", ret = 0);
6421 	if (ret == 0) {
6422 		bam_error(POOL_SIGN_INCOMPAT, pool, sign);
6423 		free(pool);
6424 		return (-1);
6425 	}
6426 
6427 	mntpt = mount_top_dataset(pool, &mnted);
6428 	INJECT_ERROR1("SET_PRIMARY_ZFS_MOUNT_DATASET", mntpt = NULL);
6429 	if (mntpt == NULL) {
6430 		bam_error(FAIL_MNT_TOP_DATASET, pool);
6431 		free(pool);
6432 		return (-1);
6433 	}
6434 
6435 	ret = set_primary_common(mntpt, sign);
6436 
6437 	(void) umount_top_dataset(pool, mnted, mntpt);
6438 
6439 	free(pool);
6440 
6441 	INJECT_ERROR1("SET_PRIMARY_ZFS_FAIL", ret = 1);
6442 	if (ret == 0) {
6443 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6444 	} else {
6445 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
6446 	}
6447 
6448 	return (ret);
6449 }
6450 
6451 static int
6452 set_primary(char *osroot, char *osdev, char *sign, char *fstype)
6453 {
6454 	const char	*fcn = "set_primary()";
6455 	int		ret;
6456 
6457 	INJECT_ERROR1("SET_PRIMARY_FSTYPE", fstype = "foofs");
6458 	if (strcmp(fstype, "ufs") == 0) {
6459 		BAM_DPRINTF((D_SET_PRIMARY_UFS, fcn));
6460 		ret = set_primary_ufs(osroot, sign);
6461 	} else if (strcmp(fstype, "zfs") == 0) {
6462 		BAM_DPRINTF((D_SET_PRIMARY_ZFS, fcn));
6463 		ret = set_primary_zfs(osdev, sign);
6464 	} else {
6465 		bam_error(GRUBSIGN_NOTSUP, fstype);
6466 		ret = -1;
6467 	}
6468 
6469 	if (ret == 0) {
6470 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6471 	} else {
6472 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
6473 	}
6474 
6475 	return (ret);
6476 }
6477 
6478 static int
6479 ufs_add_to_sign_list(char *sign)
6480 {
6481 	FILE		*tfp;
6482 	char		signline[MAXNAMELEN];
6483 	char		cmd[PATH_MAX];
6484 	int		ret;
6485 	int		error;
6486 	const char	*fcn = "ufs_add_to_sign_list()";
6487 
6488 	INJECT_ERROR1("ADD_TO_SIGN_LIST_NOT_UFS", sign = "pool_rpool5");
6489 	if (strncmp(sign, GRUBSIGN_UFS_PREFIX,
6490 	    strlen(GRUBSIGN_UFS_PREFIX)) != 0) {
6491 		bam_error(INVALID_UFS_SIGN, sign);
6492 		(void) unlink(UFS_SIGNATURE_LIST);
6493 		return (-1);
6494 	}
6495 
6496 	/*
6497 	 * most failures in this routine are not a fatal error
6498 	 * We simply unlink the /var/run file and continue
6499 	 */
6500 
6501 	ret = rename(UFS_SIGNATURE_LIST, UFS_SIGNATURE_LIST".tmp");
6502 	error = errno;
6503 	INJECT_ERROR1("ADD_TO_SIGN_LIST_RENAME", ret = -1);
6504 	if (ret == -1) {
6505 		bam_error(RENAME_FAIL, UFS_SIGNATURE_LIST".tmp",
6506 		    strerror(error));
6507 		(void) unlink(UFS_SIGNATURE_LIST);
6508 		return (0);
6509 	}
6510 
6511 	tfp = fopen(UFS_SIGNATURE_LIST".tmp", "a");
6512 	error = errno;
6513 	INJECT_ERROR1("ADD_TO_SIGN_LIST_FOPEN", tfp = NULL);
6514 	if (tfp == NULL) {
6515 		bam_error(OPEN_FAIL, UFS_SIGNATURE_LIST".tmp", strerror(error));
6516 		(void) unlink(UFS_SIGNATURE_LIST".tmp");
6517 		return (0);
6518 	}
6519 
6520 	(void) snprintf(signline, sizeof (signline), "%s\n", sign);
6521 
6522 	ret = fputs(signline, tfp);
6523 	error = errno;
6524 	INJECT_ERROR1("ADD_TO_SIGN_LIST_FPUTS", ret = 0);
6525 	if (ret != strlen(signline)) {
6526 		bam_error(SIGN_LIST_FPUTS_ERR, sign, strerror(error));
6527 		(void) fclose(tfp);
6528 		(void) unlink(UFS_SIGNATURE_LIST".tmp");
6529 		return (0);
6530 	}
6531 
6532 	ret = fclose(tfp);
6533 	error = errno;
6534 	INJECT_ERROR1("ADD_TO_SIGN_LIST_FCLOSE", ret = EOF);
6535 	if (ret == EOF) {
6536 		bam_error(CLOSE_FAIL, UFS_SIGNATURE_LIST".tmp",
6537 		    strerror(error));
6538 		(void) unlink(UFS_SIGNATURE_LIST".tmp");
6539 		return (0);
6540 	}
6541 
6542 	/* Sort the list again */
6543 	(void) snprintf(cmd, sizeof (cmd),
6544 	    "/usr/bin/sort -u %s.tmp > %s.sorted",
6545 	    UFS_SIGNATURE_LIST, UFS_SIGNATURE_LIST);
6546 
6547 	ret = exec_cmd(cmd, NULL);
6548 	INJECT_ERROR1("ADD_TO_SIGN_LIST_SORT", ret = 1);
6549 	if (ret != 0) {
6550 		bam_error(GRUBSIGN_SORT_FAILED);
6551 		(void) unlink(UFS_SIGNATURE_LIST".sorted");
6552 		(void) unlink(UFS_SIGNATURE_LIST".tmp");
6553 		return (0);
6554 	}
6555 
6556 	(void) unlink(UFS_SIGNATURE_LIST".tmp");
6557 
6558 	ret = rename(UFS_SIGNATURE_LIST".sorted", UFS_SIGNATURE_LIST);
6559 	error = errno;
6560 	INJECT_ERROR1("ADD_TO_SIGN_LIST_RENAME2", ret = -1);
6561 	if (ret == -1) {
6562 		bam_error(RENAME_FAIL, UFS_SIGNATURE_LIST, strerror(error));
6563 		(void) unlink(UFS_SIGNATURE_LIST".sorted");
6564 		return (0);
6565 	}
6566 
6567 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6568 
6569 	return (0);
6570 }
6571 
6572 static int
6573 set_signature(char *osroot, char *osdev, char *sign, char *fstype)
6574 {
6575 	int		ret;
6576 	const char	*fcn = "set_signature()";
6577 
6578 	BAM_DPRINTF((D_FUNC_ENTRY4, fcn, osroot, osdev, sign, fstype));
6579 
6580 	ret = set_backup(osroot, osdev, sign, fstype);
6581 	INJECT_ERROR1("SET_SIGNATURE_BACKUP", ret = -1);
6582 	if (ret == -1) {
6583 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
6584 		bam_error(SET_BACKUP_FAILED, sign, osroot, osdev);
6585 		return (-1);
6586 	}
6587 
6588 	ret = set_primary(osroot, osdev, sign, fstype);
6589 	INJECT_ERROR1("SET_SIGNATURE_PRIMARY", ret = -1);
6590 
6591 	if (ret == 0) {
6592 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6593 	} else {
6594 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
6595 		bam_error(SET_PRIMARY_FAILED, sign, osroot, osdev);
6596 
6597 	}
6598 	return (ret);
6599 }
6600 
6601 char *
6602 get_grubsign(char *osroot, char *osdev)
6603 {
6604 	char		*grubsign;	/* (<sign>,#,#) */
6605 	char		*slice;
6606 	int		fdiskpart;
6607 	char		*sign;
6608 	char		*fstype;
6609 	int		ret;
6610 	const char	*fcn = "get_grubsign()";
6611 
6612 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, osroot, osdev));
6613 
6614 	fstype = get_fstype(osroot);
6615 	INJECT_ERROR1("GET_GRUBSIGN_FSTYPE", fstype = NULL);
6616 	if (fstype == NULL) {
6617 		bam_error(GET_FSTYPE_FAILED, osroot);
6618 		return (NULL);
6619 	}
6620 
6621 	sign = find_existing_sign(osroot, osdev, fstype);
6622 	INJECT_ERROR1("FIND_EXISTING_SIGN", sign = NULL);
6623 	if (sign == NULL) {
6624 		BAM_DPRINTF((D_GET_GRUBSIGN_NO_EXISTING, fcn, osroot, osdev));
6625 		sign = create_new_sign(osdev, fstype);
6626 		INJECT_ERROR1("CREATE_NEW_SIGN", sign = NULL);
6627 		if (sign == NULL) {
6628 			bam_error(GRUBSIGN_CREATE_FAIL, osdev);
6629 			free(fstype);
6630 			return (NULL);
6631 		}
6632 	}
6633 
6634 	ret = set_signature(osroot, osdev, sign, fstype);
6635 	INJECT_ERROR1("SET_SIGNATURE_FAIL", ret = -1);
6636 	if (ret == -1) {
6637 		bam_error(GRUBSIGN_WRITE_FAIL, osdev);
6638 		free(sign);
6639 		free(fstype);
6640 		(void) unlink(UFS_SIGNATURE_LIST);
6641 		return (NULL);
6642 	}
6643 
6644 	free(fstype);
6645 
6646 	if (bam_verbose)
6647 		bam_print(GRUBSIGN_FOUND_OR_CREATED, sign, osdev);
6648 
6649 	fdiskpart = get_partition(osdev);
6650 	INJECT_ERROR1("GET_GRUBSIGN_FDISK", fdiskpart = -1);
6651 	if (fdiskpart == -1) {
6652 		bam_error(FDISKPART_FAIL, osdev);
6653 		free(sign);
6654 		return (NULL);
6655 	}
6656 
6657 	slice = strrchr(osdev, 's');
6658 
6659 	grubsign = s_calloc(1, MAXNAMELEN + 10);
6660 	if (slice) {
6661 		(void) snprintf(grubsign, MAXNAMELEN + 10, "(%s,%d,%c)",
6662 		    sign, fdiskpart, slice[1] + 'a' - '0');
6663 	} else
6664 		(void) snprintf(grubsign, MAXNAMELEN + 10, "(%s,%d)",
6665 		    sign, fdiskpart);
6666 
6667 	free(sign);
6668 
6669 	BAM_DPRINTF((D_GET_GRUBSIGN_SUCCESS, fcn, grubsign));
6670 
6671 	return (grubsign);
6672 }
6673 
6674 static char *
6675 get_title(char *rootdir)
6676 {
6677 	static char	title[80];
6678 	char		*cp = NULL;
6679 	char		release[PATH_MAX];
6680 	FILE		*fp;
6681 	const char	*fcn = "get_title()";
6682 
6683 	/* open the /etc/release file */
6684 	(void) snprintf(release, sizeof (release), "%s/etc/release", rootdir);
6685 
6686 	fp = fopen(release, "r");
6687 	if (fp == NULL) {
6688 		bam_error(OPEN_FAIL, release, strerror(errno));
6689 		cp = NULL;
6690 		goto out;
6691 	}
6692 
6693 	while (s_fgets(title, sizeof (title), fp) != NULL) {
6694 		cp = strstr(title, "Solaris");
6695 		if (cp)
6696 			break;
6697 	}
6698 	(void) fclose(fp);
6699 
6700 out:
6701 	cp = cp ? cp : "Solaris";
6702 
6703 	BAM_DPRINTF((D_GET_TITLE, fcn, cp));
6704 
6705 	return (cp);
6706 }
6707 
6708 char *
6709 get_special(char *mountp)
6710 {
6711 	FILE		*mntfp;
6712 	struct mnttab	mp = {0};
6713 	struct mnttab	mpref = {0};
6714 	int		error;
6715 	int		ret;
6716 	const char 	*fcn = "get_special()";
6717 
6718 	INJECT_ERROR1("GET_SPECIAL_MNTPT", mountp = NULL);
6719 	if (mountp == NULL) {
6720 		bam_error(GET_SPECIAL_NULL_MNTPT);
6721 		return (NULL);
6722 	}
6723 
6724 	mntfp = fopen(MNTTAB, "r");
6725 	error = errno;
6726 	INJECT_ERROR1("GET_SPECIAL_MNTTAB_OPEN", mntfp = NULL);
6727 	if (mntfp == NULL) {
6728 		bam_error(OPEN_FAIL, MNTTAB, strerror(error));
6729 		return (NULL);
6730 	}
6731 
6732 	if (*mountp == '\0')
6733 		mpref.mnt_mountp = "/";
6734 	else
6735 		mpref.mnt_mountp = mountp;
6736 
6737 	ret = getmntany(mntfp, &mp, &mpref);
6738 	INJECT_ERROR1("GET_SPECIAL_MNTTAB_SEARCH", ret = 1);
6739 	if (ret != 0) {
6740 		(void) fclose(mntfp);
6741 		BAM_DPRINTF((D_GET_SPECIAL_NOT_IN_MNTTAB, fcn, mountp));
6742 		return (NULL);
6743 	}
6744 	(void) fclose(mntfp);
6745 
6746 	BAM_DPRINTF((D_GET_SPECIAL, fcn, mp.mnt_special));
6747 
6748 	return (s_strdup(mp.mnt_special));
6749 }
6750 
6751 static void
6752 free_physarray(char **physarray, int n)
6753 {
6754 	int			i;
6755 	const char		*fcn = "free_physarray()";
6756 
6757 	assert(physarray);
6758 	assert(n);
6759 
6760 	BAM_DPRINTF((D_FUNC_ENTRY_N1, fcn, n));
6761 
6762 	for (i = 0; i < n; i++) {
6763 		free(physarray[i]);
6764 	}
6765 	free(physarray);
6766 
6767 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6768 }
6769 
6770 static int
6771 zfs_get_physical(char *special, char ***physarray, int *n)
6772 {
6773 	char			sdup[PATH_MAX];
6774 	char			cmd[PATH_MAX];
6775 	char			dsk[PATH_MAX];
6776 	char			*pool;
6777 	filelist_t		flist = {0};
6778 	line_t			*lp;
6779 	line_t			*startlp;
6780 	char			*comp1;
6781 	int			i;
6782 	int			ret;
6783 	const char		*fcn = "zfs_get_physical()";
6784 
6785 	assert(special);
6786 
6787 	BAM_DPRINTF((D_FUNC_ENTRY1, fcn, special));
6788 
6789 	INJECT_ERROR1("INVALID_ZFS_SPECIAL", special = "/foo");
6790 	if (special[0] == '/') {
6791 		bam_error(INVALID_ZFS_SPECIAL, special);
6792 		return (-1);
6793 	}
6794 
6795 	(void) strlcpy(sdup, special, sizeof (sdup));
6796 
6797 	pool = strtok(sdup, "/");
6798 	INJECT_ERROR1("ZFS_GET_PHYS_POOL", pool = NULL);
6799 	if (pool == NULL) {
6800 		bam_error(CANT_FIND_POOL_FROM_SPECIAL, special);
6801 		return (-1);
6802 	}
6803 
6804 	(void) snprintf(cmd, sizeof (cmd), "/sbin/zpool status %s", pool);
6805 
6806 	ret = exec_cmd(cmd, &flist);
6807 	INJECT_ERROR1("ZFS_GET_PHYS_STATUS", ret = 1);
6808 	if (ret != 0) {
6809 		bam_error(ZFS_GET_POOL_STATUS, pool);
6810 		return (-1);
6811 	}
6812 
6813 	INJECT_ERROR1("ZFS_GET_PHYS_STATUS_OUT", flist.head = NULL);
6814 	if (flist.head == NULL) {
6815 		bam_error(BAD_ZPOOL_STATUS, pool);
6816 		filelist_free(&flist);
6817 		return (-1);
6818 	}
6819 
6820 	for (lp = flist.head; lp; lp = lp->next) {
6821 		BAM_DPRINTF((D_STRTOK_ZPOOL_STATUS, fcn, lp->line));
6822 		comp1 = strtok(lp->line, " \t");
6823 		if (comp1 == NULL) {
6824 			free(lp->line);
6825 			lp->line = NULL;
6826 		} else {
6827 			comp1 = s_strdup(comp1);
6828 			free(lp->line);
6829 			lp->line = comp1;
6830 		}
6831 	}
6832 
6833 	for (lp = flist.head; lp; lp = lp->next) {
6834 		if (lp->line == NULL)
6835 			continue;
6836 		if (strcmp(lp->line, pool) == 0) {
6837 			BAM_DPRINTF((D_FOUND_POOL_IN_ZPOOL_STATUS, fcn, pool));
6838 			break;
6839 		}
6840 	}
6841 
6842 	if (lp == NULL) {
6843 		bam_error(NO_POOL_IN_ZPOOL_STATUS, pool);
6844 		filelist_free(&flist);
6845 		return (-1);
6846 	}
6847 
6848 	startlp = lp->next;
6849 	for (i = 0, lp = startlp; lp; lp = lp->next) {
6850 		if (lp->line == NULL)
6851 			continue;
6852 		if (strcmp(lp->line, "mirror") == 0)
6853 			continue;
6854 		if (lp->line[0] == '\0' || strcmp(lp->line, "errors:") == 0)
6855 			break;
6856 		i++;
6857 		BAM_DPRINTF((D_COUNTING_ZFS_PHYS, fcn, i));
6858 	}
6859 
6860 	if (i == 0) {
6861 		bam_error(NO_PHYS_IN_ZPOOL_STATUS, pool);
6862 		filelist_free(&flist);
6863 		return (-1);
6864 	}
6865 
6866 	*n = i;
6867 	*physarray = s_calloc(*n, sizeof (char *));
6868 	for (i = 0, lp = startlp; lp; lp = lp->next) {
6869 		if (lp->line == NULL)
6870 			continue;
6871 		if (strcmp(lp->line, "mirror") == 0)
6872 			continue;
6873 		if (strcmp(lp->line, "errors:") == 0)
6874 			break;
6875 		if (strncmp(lp->line, "/dev/dsk/", strlen("/dev/dsk/")) != 0 &&
6876 		    strncmp(lp->line, "/dev/rdsk/",
6877 		    strlen("/dev/rdsk/")) != 0)  {
6878 			(void) snprintf(dsk, sizeof (dsk), "/dev/dsk/%s",
6879 			    lp->line);
6880 		} else {
6881 			(void) strlcpy(dsk, lp->line, sizeof (dsk));
6882 		}
6883 		BAM_DPRINTF((D_ADDING_ZFS_PHYS, fcn, dsk, pool));
6884 		(*physarray)[i++] = s_strdup(dsk);
6885 	}
6886 
6887 	assert(i == *n);
6888 
6889 	filelist_free(&flist);
6890 
6891 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
6892 	return (0);
6893 }
6894 
6895 /*
6896  * Certain services needed to run metastat successfully may not
6897  * be enabled. Enable them now.
6898  */
6899 /*
6900  * Checks if the specified service is online
6901  * Returns: 	1 if the service is online
6902  *		0 if the service is not online
6903  *		-1 on error
6904  */
6905 static int
6906 is_svc_online(char *svc)
6907 {
6908 	char			*state;
6909 	const char		*fcn = "is_svc_online()";
6910 
6911 	BAM_DPRINTF((D_FUNC_ENTRY1, fcn, svc));
6912 
6913 	state = smf_get_state(svc);
6914 	INJECT_ERROR2("GET_SVC_STATE", free(state), state = NULL);
6915 	if (state == NULL) {
6916 		bam_error(GET_SVC_STATE_ERR, svc);
6917 		return (-1);
6918 	}
6919 	BAM_DPRINTF((D_GOT_SVC_STATUS, fcn, svc));
6920 
6921 	if (strcmp(state, SCF_STATE_STRING_ONLINE) == 0) {
6922 		BAM_DPRINTF((D_SVC_ONLINE, fcn, svc));
6923 		free(state);
6924 		return (1);
6925 	}
6926 
6927 	BAM_DPRINTF((D_SVC_NOT_ONLINE, fcn, state, svc));
6928 
6929 	free(state);
6930 
6931 	return (0);
6932 }
6933 
6934 static int
6935 enable_svc(char *svc)
6936 {
6937 	int			ret;
6938 	int			sleeptime;
6939 	const char		*fcn = "enable_svc()";
6940 
6941 	ret = is_svc_online(svc);
6942 	if (ret == -1) {
6943 		bam_error(SVC_IS_ONLINE_FAILED, svc);
6944 		return (-1);
6945 	} else if (ret == 1) {
6946 		BAM_DPRINTF((D_SVC_ALREADY_ONLINE, fcn, svc));
6947 		return (0);
6948 	}
6949 
6950 	/* Service is not enabled. Enable it now. */
6951 	ret = smf_enable_instance(svc, 0);
6952 	INJECT_ERROR1("ENABLE_SVC_FAILED", ret = -1);
6953 	if (ret != 0) {
6954 		bam_error(ENABLE_SVC_FAILED, svc);
6955 		return (-1);
6956 	}
6957 
6958 	BAM_DPRINTF((D_SVC_ONLINE_INITIATED, fcn, svc));
6959 
6960 	sleeptime = 0;
6961 	do {
6962 		ret = is_svc_online(svc);
6963 		INJECT_ERROR1("SVC_ONLINE_SUCCESS", ret = 1);
6964 		INJECT_ERROR1("SVC_ONLINE_FAILURE", ret = -1);
6965 		INJECT_ERROR1("SVC_ONLINE_NOTYET", ret = 0);
6966 		if (ret == -1) {
6967 			bam_error(ERR_SVC_GET_ONLINE, svc);
6968 			return (-1);
6969 		} else if (ret == 1) {
6970 			BAM_DPRINTF((D_SVC_NOW_ONLINE, fcn, svc));
6971 			return (1);
6972 		}
6973 		(void) sleep(1);
6974 	} while (sleeptime < 60);
6975 
6976 	bam_error(TIMEOUT_ENABLE_SVC, svc);
6977 
6978 	return (-1);
6979 }
6980 
6981 static int
6982 ufs_get_physical(char *special, char ***physarray, int *n)
6983 {
6984 	char			cmd[PATH_MAX];
6985 	char			*shortname;
6986 	filelist_t		flist = {0};
6987 	char			*meta;
6988 	char			*type;
6989 	char			*comp1;
6990 	char			*comp2;
6991 	char			*comp3;
6992 	char			*comp4;
6993 	int			i;
6994 	line_t			*lp;
6995 	int			ret;
6996 	char			*svc;
6997 	const char		*fcn = "ufs_get_physical()";
6998 
6999 	assert(special);
7000 
7001 	BAM_DPRINTF((D_FUNC_ENTRY1, fcn, special));
7002 
7003 	if (strncmp(special, "/dev/md/", strlen("/dev/md/")) != 0) {
7004 		bam_error(UFS_GET_PHYS_NOT_SVM, special);
7005 		return (-1);
7006 	}
7007 
7008 	if (strncmp(special, "/dev/md/dsk/", strlen("/dev/md/dsk/")) == 0) {
7009 		shortname = special + strlen("/dev/md/dsk/");
7010 	} else if (strncmp(special, "/dev/md/rdsk/",
7011 	    strlen("/dev/md/rdsk/")) == 0) {
7012 		shortname = special + strlen("/dev/md/rdsk");
7013 	} else {
7014 		bam_error(UFS_GET_PHYS_INVALID_SVM, special);
7015 		return (-1);
7016 	}
7017 
7018 	BAM_DPRINTF((D_UFS_SVM_SHORT, fcn, special, shortname));
7019 
7020 	svc = "network/rpc/meta:default";
7021 	if (enable_svc(svc) == -1) {
7022 		bam_error(UFS_SVM_METASTAT_SVC_ERR, svc);
7023 	}
7024 
7025 	(void) snprintf(cmd, sizeof (cmd), "/sbin/metastat -p %s", shortname);
7026 
7027 	ret = exec_cmd(cmd, &flist);
7028 	INJECT_ERROR1("UFS_SVM_METASTAT", ret = 1);
7029 	if (ret != 0) {
7030 		bam_error(UFS_SVM_METASTAT_ERR, shortname);
7031 		return (-1);
7032 	}
7033 
7034 	INJECT_ERROR1("UFS_SVM_METASTAT_OUT", flist.head = NULL);
7035 	if (flist.head == NULL) {
7036 		bam_error(BAD_UFS_SVM_METASTAT, shortname);
7037 		filelist_free(&flist);
7038 		return (-1);
7039 	}
7040 
7041 	/*
7042 	 * Check if not a mirror. We only parse a single metadevice
7043 	 * if not a mirror
7044 	 */
7045 	meta = strtok(flist.head->line, " \t");
7046 	type = strtok(NULL, " \t");
7047 	if (meta == NULL || type == NULL) {
7048 		bam_error(ERROR_PARSE_UFS_SVM_METASTAT, shortname);
7049 		filelist_free(&flist);
7050 		return (-1);
7051 	}
7052 	if (strcmp(type, "-m") != 0) {
7053 		comp1 = strtok(NULL, " \t");
7054 		comp2 = strtok(NULL, " \t");
7055 		if (comp1 == NULL || comp2 != NULL) {
7056 			bam_error(INVALID_UFS_SVM_METASTAT, shortname);
7057 			filelist_free(&flist);
7058 			return (-1);
7059 		}
7060 		BAM_DPRINTF((D_UFS_SVM_ONE_COMP, fcn, comp1, shortname));
7061 		*physarray = s_calloc(1, sizeof (char *));
7062 		(*physarray)[0] = s_strdup(comp1);
7063 		*n = 1;
7064 		filelist_free(&flist);
7065 		return (0);
7066 	}
7067 
7068 	/*
7069 	 * Okay we have a mirror. Everything after the first line
7070 	 * is a submirror
7071 	 */
7072 	for (i = 0, lp = flist.head->next; lp; lp = lp->next) {
7073 		if (strstr(lp->line, "/dev/dsk/") == NULL &&
7074 		    strstr(lp->line, "/dev/rdsk/") == NULL) {
7075 			bam_error(CANNOT_PARSE_UFS_SVM_METASTAT, shortname);
7076 			filelist_free(&flist);
7077 			return (-1);
7078 		}
7079 		i++;
7080 	}
7081 
7082 	*physarray = s_calloc(i, sizeof (char *));
7083 	*n = i;
7084 
7085 	for (i = 0, lp = flist.head->next; lp; lp = lp->next) {
7086 		comp1 = strtok(lp->line, " \t");
7087 		comp2 = strtok(NULL, " \t");
7088 		comp3 = strtok(NULL, " \t");
7089 		comp4 = strtok(NULL, " \t");
7090 
7091 		if (comp3 == NULL || comp4 == NULL ||
7092 		    (strncmp(comp4, "/dev/dsk/", strlen("/dev/dsk/")) != 0 &&
7093 		    strncmp(comp4, "/dev/rdsk/", strlen("/dev/rdsk/")) != 0)) {
7094 			bam_error(CANNOT_PARSE_UFS_SVM_SUBMIRROR, shortname);
7095 			filelist_free(&flist);
7096 			free_physarray(*physarray, *n);
7097 			return (-1);
7098 		}
7099 
7100 		(*physarray)[i++] = s_strdup(comp4);
7101 	}
7102 
7103 	assert(i == *n);
7104 
7105 	filelist_free(&flist);
7106 
7107 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
7108 	return (0);
7109 }
7110 
7111 static int
7112 get_physical(char *menu_root, char ***physarray, int *n)
7113 {
7114 	char			*special;
7115 	int			ret;
7116 	const char		*fcn = "get_physical()";
7117 
7118 	assert(menu_root);
7119 	assert(physarray);
7120 	assert(n);
7121 
7122 	*physarray = NULL;
7123 	*n = 0;
7124 
7125 	BAM_DPRINTF((D_FUNC_ENTRY1, fcn, menu_root));
7126 
7127 	/* First get the device special file from /etc/mnttab */
7128 	special = get_special(menu_root);
7129 	INJECT_ERROR1("GET_PHYSICAL_SPECIAL", special = NULL);
7130 	if (special == NULL) {
7131 		bam_error(GET_SPECIAL_NULL, menu_root);
7132 		return (-1);
7133 	}
7134 
7135 	/* If already a physical device nothing to do */
7136 	if (strncmp(special, "/dev/dsk/", strlen("/dev/dsk/")) == 0 ||
7137 	    strncmp(special, "/dev/rdsk/", strlen("/dev/rdsk/")) == 0) {
7138 		BAM_DPRINTF((D_GET_PHYSICAL_ALREADY, fcn, menu_root, special));
7139 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
7140 		*physarray = s_calloc(1, sizeof (char *));
7141 		(*physarray)[0] = special;
7142 		*n = 1;
7143 		return (0);
7144 	}
7145 
7146 	if (is_zfs(menu_root)) {
7147 		ret = zfs_get_physical(special, physarray, n);
7148 	} else if (is_ufs(menu_root)) {
7149 		ret = ufs_get_physical(special, physarray, n);
7150 	} else {
7151 		bam_error(GET_PHYSICAL_NOTSUP_FSTYPE, menu_root, special);
7152 		ret = -1;
7153 	}
7154 
7155 	free(special);
7156 
7157 	INJECT_ERROR1("GET_PHYSICAL_RET", ret = -1);
7158 	if (ret == -1) {
7159 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
7160 	} else {
7161 		int	i;
7162 		assert (*n > 0);
7163 		for (i = 0; i < *n; i++) {
7164 			BAM_DPRINTF((D_GET_PHYSICAL_RET, fcn, (*physarray)[i]));
7165 		}
7166 	}
7167 
7168 	return (ret);
7169 }
7170 
7171 static int
7172 is_bootdisk(char *osroot, char *physical)
7173 {
7174 	int			ret;
7175 	char			*grubroot;
7176 	char			*bootp;
7177 	const char		*fcn = "is_bootdisk()";
7178 
7179 	assert(osroot);
7180 	assert(physical);
7181 
7182 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, osroot, physical));
7183 
7184 	bootp = strstr(physical, "p0:boot");
7185 	if (bootp)
7186 		*bootp = '\0';
7187 	/*
7188 	 * We just want the BIOS mapping for menu disk.
7189 	 * Don't pass menu_root to get_grubroot() as the
7190 	 * check that it is used for is not relevant here.
7191 	 * The osroot is immaterial as well - it is only used to
7192 	 * to find create_diskmap script. Everything hinges on
7193 	 * "physical"
7194 	 */
7195 	grubroot = get_grubroot(osroot, physical, NULL);
7196 
7197 	INJECT_ERROR1("IS_BOOTDISK_GRUBROOT", grubroot = NULL);
7198 	if (grubroot == NULL) {
7199 		if (bam_verbose)
7200 			bam_error(NO_GRUBROOT_FOR_DISK, physical);
7201 		return (0);
7202 	}
7203 	ret = grubroot[3] == '0';
7204 	free(grubroot);
7205 
7206 	BAM_DPRINTF((D_RETURN_RET, fcn, ret));
7207 
7208 	return (ret);
7209 }
7210 
7211 /*
7212  * Check if menu is on the boot device
7213  * Return 0 (false) on error
7214  */
7215 static int
7216 menu_on_bootdisk(char *osroot, char *menu_root)
7217 {
7218 	char		**physarray;
7219 	int		ret;
7220 	int		n;
7221 	int		i;
7222 	int		on_bootdisk;
7223 	const char	*fcn = "menu_on_bootdisk()";
7224 
7225 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, osroot, menu_root));
7226 
7227 	ret = get_physical(menu_root, &physarray, &n);
7228 	INJECT_ERROR1("MENU_ON_BOOTDISK_PHYSICAL", ret = -1);
7229 	if (ret != 0) {
7230 		bam_error(GET_PHYSICAL_MENU_NULL, menu_root);
7231 		return (0);
7232 	}
7233 
7234 	assert(physarray);
7235 	assert(n > 0);
7236 
7237 	on_bootdisk = 0;
7238 	for (i = 0; i < n; i++) {
7239 		assert(strncmp(physarray[i], "/dev/dsk/",
7240 		    strlen("/dev/dsk/")) == 0 ||
7241 		    strncmp(physarray[i], "/dev/rdsk/",
7242 		    strlen("/dev/rdsk/")) == 0);
7243 
7244 		BAM_DPRINTF((D_CHECK_ON_BOOTDISK, fcn, physarray[i]));
7245 		if (is_bootdisk(osroot, physarray[i])) {
7246 			on_bootdisk = 1;
7247 			BAM_DPRINTF((D_IS_ON_BOOTDISK, fcn, physarray[i]));
7248 		}
7249 	}
7250 
7251 	free_physarray(physarray, n);
7252 
7253 	INJECT_ERROR1("ON_BOOTDISK_YES", on_bootdisk = 1);
7254 	INJECT_ERROR1("ON_BOOTDISK_NO", on_bootdisk = 0);
7255 	if (on_bootdisk) {
7256 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
7257 	} else {
7258 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
7259 	}
7260 
7261 	return (on_bootdisk);
7262 }
7263 
7264 void
7265 bam_add_line(menu_t *mp, entry_t *entry, line_t *prev, line_t *lp)
7266 {
7267 	const char	*fcn = "bam_add_line()";
7268 
7269 	assert(mp);
7270 	assert(entry);
7271 	assert(prev);
7272 	assert(lp);
7273 
7274 	lp->next = prev->next;
7275 	if (prev->next) {
7276 		BAM_DPRINTF((D_ADD_LINE_PREV_NEXT, fcn));
7277 		prev->next->prev = lp;
7278 	} else {
7279 		BAM_DPRINTF((D_ADD_LINE_NOT_PREV_NEXT, fcn));
7280 	}
7281 	prev->next = lp;
7282 	lp->prev = prev;
7283 
7284 	if (entry->end == prev) {
7285 		BAM_DPRINTF((D_ADD_LINE_LAST_LINE_IN_ENTRY, fcn));
7286 		entry->end = lp;
7287 	}
7288 	if (mp->end == prev) {
7289 		assert(lp->next == NULL);
7290 		mp->end = lp;
7291 		BAM_DPRINTF((D_ADD_LINE_LAST_LINE_IN_MENU, fcn));
7292 	}
7293 }
7294 
7295 /*
7296  * look for matching bootadm entry with specified parameters
7297  * Here are the rules (based on existing usage):
7298  * - If title is specified, match on title only
7299  * - Else, match on root/findroot, kernel, and module.
7300  *   Note that, if root_opt is non-zero, the absence of
7301  *   root line is considered a match.
7302  */
7303 static entry_t *
7304 find_boot_entry(
7305 	menu_t *mp,
7306 	char *title,
7307 	char *kernel,
7308 	char *findroot,
7309 	char *root,
7310 	char *module,
7311 	int root_opt,
7312 	int *entry_num)
7313 {
7314 	int		i;
7315 	line_t		*lp;
7316 	entry_t		*ent;
7317 	const char	*fcn = "find_boot_entry()";
7318 
7319 	if (entry_num)
7320 		*entry_num = BAM_ERROR;
7321 
7322 	/* find matching entry */
7323 	for (i = 0, ent = mp->entries; ent; i++, ent = ent->next) {
7324 		lp = ent->start;
7325 
7326 		/* first line of entry must be bootadm comment */
7327 		lp = ent->start;
7328 		if (lp->flags != BAM_COMMENT ||
7329 		    strcmp(lp->arg, BAM_BOOTADM_HDR) != 0) {
7330 			continue;
7331 		}
7332 
7333 		/* advance to title line */
7334 		lp = lp->next;
7335 		if (title) {
7336 			if (lp->flags == BAM_TITLE && lp->arg &&
7337 			    strcmp(lp->arg, title) == 0) {
7338 				BAM_DPRINTF((D_MATCHED_TITLE, fcn, title));
7339 				break;
7340 			}
7341 			BAM_DPRINTF((D_NOMATCH_TITLE, fcn, title, lp->arg));
7342 			continue;	/* check title only */
7343 		}
7344 
7345 		lp = lp->next;	/* advance to root line */
7346 		if (lp == NULL) {
7347 			continue;
7348 		} else if (strcmp(lp->cmd, menu_cmds[FINDROOT_CMD]) == 0) {
7349 			INJECT_ERROR1("FIND_BOOT_ENTRY_NULL_FINDROOT",
7350 			    findroot = NULL);
7351 			if (findroot == NULL) {
7352 				BAM_DPRINTF((D_NOMATCH_FINDROOT_NULL,
7353 				    fcn, lp->arg));
7354 				continue;
7355 			}
7356 			/* findroot command found, try match  */
7357 			if (strcmp(lp->arg, findroot) != 0) {
7358 				BAM_DPRINTF((D_NOMATCH_FINDROOT,
7359 				    fcn, findroot, lp->arg));
7360 				continue;
7361 			}
7362 			BAM_DPRINTF((D_MATCHED_FINDROOT, fcn, findroot));
7363 			lp = lp->next;	/* advance to kernel line */
7364 		} else if (strcmp(lp->cmd, menu_cmds[ROOT_CMD]) == 0) {
7365 			INJECT_ERROR1("FIND_BOOT_ENTRY_NULL_ROOT", root = NULL);
7366 			if (root == NULL) {
7367 				BAM_DPRINTF((D_NOMATCH_ROOT_NULL,
7368 				    fcn, lp->arg));
7369 				continue;
7370 			}
7371 			/* root cmd found, try match */
7372 			if (strcmp(lp->arg, root) != 0) {
7373 				BAM_DPRINTF((D_NOMATCH_ROOT,
7374 				    fcn, root, lp->arg));
7375 				continue;
7376 			}
7377 			BAM_DPRINTF((D_MATCHED_ROOT, fcn, root));
7378 			lp = lp->next;	/* advance to kernel line */
7379 		} else {
7380 			INJECT_ERROR1("FIND_BOOT_ENTRY_ROOT_OPT_NO",
7381 			    root_opt = 0);
7382 			INJECT_ERROR1("FIND_BOOT_ENTRY_ROOT_OPT_YES",
7383 			    root_opt = 1);
7384 			/* no root command, see if root is optional */
7385 			if (root_opt == 0) {
7386 				BAM_DPRINTF((D_NO_ROOT_OPT, fcn));
7387 				continue;
7388 			}
7389 			BAM_DPRINTF((D_ROOT_OPT, fcn));
7390 		}
7391 
7392 		if (lp == NULL || lp->next == NULL) {
7393 			continue;
7394 		}
7395 
7396 		if (kernel &&
7397 		    (!check_cmd(lp->cmd, KERNEL_CMD, lp->arg, kernel))) {
7398 			if (!(ent->flags & BAM_ENTRY_FAILSAFE) ||
7399 			    !(ent->flags & BAM_ENTRY_DBOOT) ||
7400 			    strcmp(kernel, DIRECT_BOOT_FAILSAFE_LINE) != 0)
7401 				continue;
7402 
7403 			ent->flags |= BAM_ENTRY_UPGFSKERNEL;
7404 
7405 		}
7406 		BAM_DPRINTF((D_KERNEL_MATCH, fcn, kernel, lp->arg));
7407 
7408 		/*
7409 		 * Check for matching module entry (failsafe or normal).
7410 		 * If it fails to match, we go around the loop again.
7411 		 * For xpv entries, there are two module lines, so we
7412 		 * do the check twice.
7413 		 */
7414 		lp = lp->next;	/* advance to module line */
7415 		if (check_cmd(lp->cmd, MODULE_CMD, lp->arg, module) ||
7416 		    (((lp = lp->next) != NULL) &&
7417 		    check_cmd(lp->cmd, MODULE_CMD, lp->arg, module))) {
7418 			/* match found */
7419 			BAM_DPRINTF((D_MODULE_MATCH, fcn, module, lp->arg));
7420 			break;
7421 		}
7422 
7423 		if (strcmp(module, FAILSAFE_ARCHIVE) == 0 &&
7424 		    (strcmp(lp->prev->arg, FAILSAFE_ARCHIVE_32) == 0 ||
7425 		    strcmp(lp->prev->arg, FAILSAFE_ARCHIVE_64) == 0)) {
7426 			ent->flags |= BAM_ENTRY_UPGFSMODULE;
7427 			break;
7428 		}
7429 
7430 	}
7431 
7432 	if (ent && entry_num) {
7433 		*entry_num = i;
7434 	}
7435 
7436 	if (ent) {
7437 		BAM_DPRINTF((D_RETURN_RET, fcn, i));
7438 	} else {
7439 		BAM_DPRINTF((D_RETURN_RET, fcn, BAM_ERROR));
7440 	}
7441 	return (ent);
7442 }
7443 
7444 static int
7445 update_boot_entry(menu_t *mp, char *title, char *findroot, char *root,
7446     char *kernel, char *mod_kernel, char *module, int root_opt)
7447 {
7448 	int		i;
7449 	int		change_kernel = 0;
7450 	entry_t		*ent;
7451 	line_t		*lp;
7452 	line_t		*tlp;
7453 	char		linebuf[BAM_MAXLINE];
7454 	const char	*fcn = "update_boot_entry()";
7455 
7456 	/* note: don't match on title, it's updated on upgrade */
7457 	ent = find_boot_entry(mp, NULL, kernel, findroot, root, module,
7458 	    root_opt, &i);
7459 	if ((ent == NULL) && (bam_direct == BAM_DIRECT_DBOOT)) {
7460 		/*
7461 		 * We may be upgrading a kernel from multiboot to
7462 		 * directboot.  Look for a multiboot entry. A multiboot
7463 		 * entry will not have a findroot line.
7464 		 */
7465 		ent = find_boot_entry(mp, NULL, "multiboot", NULL, root,
7466 		    MULTIBOOT_ARCHIVE, root_opt, &i);
7467 		if (ent != NULL) {
7468 			BAM_DPRINTF((D_UPGRADE_FROM_MULTIBOOT, fcn, root));
7469 			change_kernel = 1;
7470 		}
7471 	} else if (ent) {
7472 		BAM_DPRINTF((D_FOUND_FINDROOT, fcn, findroot));
7473 	}
7474 
7475 	if (ent == NULL) {
7476 		BAM_DPRINTF((D_ENTRY_NOT_FOUND_CREATING, fcn, findroot));
7477 		return (add_boot_entry(mp, title, findroot,
7478 		    kernel, mod_kernel, module));
7479 	}
7480 
7481 	/* replace title of existing entry and update findroot line */
7482 	lp = ent->start;
7483 	lp = lp->next;	/* title line */
7484 	(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
7485 	    menu_cmds[TITLE_CMD], menu_cmds[SEP_CMD], title);
7486 	free(lp->arg);
7487 	free(lp->line);
7488 	lp->arg = s_strdup(title);
7489 	lp->line = s_strdup(linebuf);
7490 	BAM_DPRINTF((D_CHANGING_TITLE, fcn, title));
7491 
7492 	tlp = lp;	/* title line */
7493 	lp = lp->next;	/* root line */
7494 
7495 	/* if no root or findroot command, create a new line_t */
7496 	if (strcmp(lp->cmd, menu_cmds[ROOT_CMD]) != 0 &&
7497 	    strcmp(lp->cmd, menu_cmds[FINDROOT_CMD]) != 0) {
7498 		lp = s_calloc(1, sizeof (line_t));
7499 		bam_add_line(mp, ent, tlp, lp);
7500 	} else {
7501 		free(lp->cmd);
7502 		free(lp->sep);
7503 		free(lp->arg);
7504 		free(lp->line);
7505 	}
7506 
7507 	lp->cmd = s_strdup(menu_cmds[FINDROOT_CMD]);
7508 	lp->sep = s_strdup(menu_cmds[SEP_CMD]);
7509 	lp->arg = s_strdup(findroot);
7510 	(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
7511 	    menu_cmds[FINDROOT_CMD], menu_cmds[SEP_CMD], findroot);
7512 	lp->line = s_strdup(linebuf);
7513 	BAM_DPRINTF((D_ADDING_FINDROOT_LINE, fcn, findroot));
7514 
7515 	/* kernel line */
7516 	lp = lp->next;
7517 
7518 	if (ent->flags & BAM_ENTRY_UPGFSKERNEL) {
7519 		char		*params = NULL;
7520 
7521 		params = strstr(lp->line, "-s");
7522 		if (params != NULL)
7523 			(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s%s",
7524 			    menu_cmds[KERNEL_DOLLAR_CMD], menu_cmds[SEP_CMD],
7525 			    kernel, params+2);
7526 		else
7527 			(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
7528 			    menu_cmds[KERNEL_DOLLAR_CMD], menu_cmds[SEP_CMD],
7529 			    kernel);
7530 
7531 		free(lp->cmd);
7532 		free(lp->arg);
7533 		free(lp->line);
7534 		lp->cmd = s_strdup(menu_cmds[KERNEL_DOLLAR_CMD]);
7535 		lp->arg = s_strdup(strstr(linebuf, "/"));
7536 		lp->line = s_strdup(linebuf);
7537 		ent->flags &= ~BAM_ENTRY_UPGFSKERNEL;
7538 		BAM_DPRINTF((D_ADDING_KERNEL_DOLLAR, fcn, lp->prev->cmd));
7539 	}
7540 
7541 	if (change_kernel) {
7542 		/*
7543 		 * We're upgrading from multiboot to directboot.
7544 		 */
7545 		if (strcmp(lp->cmd, menu_cmds[KERNEL_CMD]) == 0) {
7546 			(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
7547 			    menu_cmds[KERNEL_DOLLAR_CMD], menu_cmds[SEP_CMD],
7548 			    kernel);
7549 			free(lp->cmd);
7550 			free(lp->arg);
7551 			free(lp->line);
7552 			lp->cmd = s_strdup(menu_cmds[KERNEL_DOLLAR_CMD]);
7553 			lp->arg = s_strdup(kernel);
7554 			lp->line = s_strdup(linebuf);
7555 			lp = lp->next;
7556 			BAM_DPRINTF((D_ADDING_KERNEL_DOLLAR, fcn, kernel));
7557 		}
7558 		if (strcmp(lp->cmd, menu_cmds[MODULE_CMD]) == 0) {
7559 			(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
7560 			    menu_cmds[MODULE_DOLLAR_CMD], menu_cmds[SEP_CMD],
7561 			    module);
7562 			free(lp->cmd);
7563 			free(lp->arg);
7564 			free(lp->line);
7565 			lp->cmd = s_strdup(menu_cmds[MODULE_DOLLAR_CMD]);
7566 			lp->arg = s_strdup(module);
7567 			lp->line = s_strdup(linebuf);
7568 			lp = lp->next;
7569 			BAM_DPRINTF((D_ADDING_MODULE_DOLLAR, fcn, module));
7570 		}
7571 	}
7572 
7573 	/* module line */
7574 	lp = lp->next;
7575 
7576 	if (ent->flags & BAM_ENTRY_UPGFSMODULE) {
7577 		if (strcmp(lp->cmd, menu_cmds[MODULE_CMD]) == 0) {
7578 			(void) snprintf(linebuf, sizeof (linebuf), "%s%s%s",
7579 			    menu_cmds[MODULE_DOLLAR_CMD], menu_cmds[SEP_CMD],
7580 			    module);
7581 			free(lp->cmd);
7582 			free(lp->arg);
7583 			free(lp->line);
7584 			lp->cmd = s_strdup(menu_cmds[MODULE_DOLLAR_CMD]);
7585 			lp->arg = s_strdup(module);
7586 			lp->line = s_strdup(linebuf);
7587 			lp = lp->next;
7588 			ent->flags &= ~BAM_ENTRY_UPGFSMODULE;
7589 			BAM_DPRINTF((D_ADDING_MODULE_DOLLAR, fcn, module));
7590 		}
7591 	}
7592 
7593 	BAM_DPRINTF((D_RETURN_RET, fcn, i));
7594 	return (i);
7595 }
7596 
7597 int
7598 root_optional(char *osroot, char *menu_root)
7599 {
7600 	char			*ospecial;
7601 	char			*mspecial;
7602 	char			*slash;
7603 	int			root_opt;
7604 	int			ret1;
7605 	int			ret2;
7606 	const char		*fcn = "root_optional()";
7607 
7608 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, osroot, menu_root));
7609 
7610 	/*
7611 	 * For all filesystems except ZFS, a straight compare of osroot
7612 	 * and menu_root will tell us if root is optional.
7613 	 * For ZFS, the situation is complicated by the fact that
7614 	 * menu_root and osroot are always different
7615 	 */
7616 	ret1 = is_zfs(osroot);
7617 	ret2 = is_zfs(menu_root);
7618 	INJECT_ERROR1("ROOT_OPT_NOT_ZFS", ret1 = 0);
7619 	if (!ret1 || !ret2) {
7620 		BAM_DPRINTF((D_ROOT_OPT_NOT_ZFS, fcn, osroot, menu_root));
7621 		root_opt = (strcmp(osroot, menu_root) == 0);
7622 		goto out;
7623 	}
7624 
7625 	ospecial = get_special(osroot);
7626 	INJECT_ERROR1("ROOT_OPTIONAL_OSPECIAL", ospecial = NULL);
7627 	if (ospecial == NULL) {
7628 		bam_error(GET_OSROOT_SPECIAL_ERR, osroot);
7629 		return (0);
7630 	}
7631 	BAM_DPRINTF((D_ROOT_OPTIONAL_OSPECIAL, fcn, ospecial, osroot));
7632 
7633 	mspecial = get_special(menu_root);
7634 	INJECT_ERROR1("ROOT_OPTIONAL_MSPECIAL", mspecial = NULL);
7635 	if (mspecial == NULL) {
7636 		bam_error(GET_MENU_ROOT_SPECIAL_ERR, menu_root);
7637 		free(ospecial);
7638 		return (0);
7639 	}
7640 	BAM_DPRINTF((D_ROOT_OPTIONAL_MSPECIAL, fcn, mspecial, menu_root));
7641 
7642 	slash = strchr(ospecial, '/');
7643 	if (slash)
7644 		*slash = '\0';
7645 	BAM_DPRINTF((D_ROOT_OPTIONAL_FIXED_OSPECIAL, fcn, ospecial, osroot));
7646 
7647 	root_opt = (strcmp(ospecial, mspecial) == 0);
7648 
7649 	free(ospecial);
7650 	free(mspecial);
7651 
7652 out:
7653 	INJECT_ERROR1("ROOT_OPTIONAL_NO", root_opt = 0);
7654 	INJECT_ERROR1("ROOT_OPTIONAL_YES", root_opt = 1);
7655 	if (root_opt) {
7656 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
7657 	} else {
7658 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
7659 	}
7660 
7661 	return (root_opt);
7662 }
7663 
7664 /*ARGSUSED*/
7665 static error_t
7666 update_entry(menu_t *mp, char *menu_root, char *osdev)
7667 {
7668 	int		entry;
7669 	char		*grubsign;
7670 	char		*grubroot;
7671 	char		*title;
7672 	char		osroot[PATH_MAX];
7673 	char		*failsafe_kernel = NULL;
7674 	struct stat	sbuf;
7675 	char		failsafe[256];
7676 	char		failsafe_64[256];
7677 	int		ret;
7678 	const char	*fcn = "update_entry()";
7679 
7680 	assert(mp);
7681 	assert(menu_root);
7682 	assert(osdev);
7683 	assert(bam_root);
7684 
7685 	BAM_DPRINTF((D_FUNC_ENTRY3, fcn, menu_root, osdev, bam_root));
7686 
7687 	(void) strlcpy(osroot, bam_root, sizeof (osroot));
7688 
7689 	title = get_title(osroot);
7690 	assert(title);
7691 
7692 	grubsign = get_grubsign(osroot, osdev);
7693 	INJECT_ERROR1("GET_GRUBSIGN_FAIL", grubsign = NULL);
7694 	if (grubsign == NULL) {
7695 		bam_error(GET_GRUBSIGN_ERROR, osroot, osdev);
7696 		return (BAM_ERROR);
7697 	}
7698 
7699 	/*
7700 	 * It is not a fatal error if get_grubroot() fails
7701 	 * We no longer rely on biosdev to populate the
7702 	 * menu
7703 	 */
7704 	grubroot = get_grubroot(osroot, osdev, menu_root);
7705 	INJECT_ERROR1("GET_GRUBROOT_FAIL", grubroot = NULL);
7706 	if (grubroot) {
7707 		BAM_DPRINTF((D_GET_GRUBROOT_SUCCESS,
7708 		    fcn, osroot, osdev, menu_root));
7709 	} else {
7710 		BAM_DPRINTF((D_GET_GRUBROOT_FAILURE,
7711 		    fcn, osroot, osdev, menu_root));
7712 	}
7713 
7714 	/* add the entry for normal Solaris */
7715 	INJECT_ERROR1("UPDATE_ENTRY_MULTIBOOT",
7716 	    bam_direct = BAM_DIRECT_MULTIBOOT);
7717 	if (bam_direct == BAM_DIRECT_DBOOT) {
7718 		entry = update_boot_entry(mp, title, grubsign, grubroot,
7719 		    (bam_zfs ? DIRECT_BOOT_KERNEL_ZFS : DIRECT_BOOT_KERNEL),
7720 		    NULL, DIRECT_BOOT_ARCHIVE,
7721 		    root_optional(osroot, menu_root));
7722 		BAM_DPRINTF((D_UPDATED_BOOT_ENTRY, fcn, bam_zfs, grubsign));
7723 		if ((entry != BAM_ERROR) && (bam_is_hv == BAM_HV_PRESENT)) {
7724 			(void) update_boot_entry(mp, NEW_HV_ENTRY, grubsign,
7725 			    grubroot, XEN_MENU, bam_zfs ?
7726 			    XEN_KERNEL_MODULE_LINE_ZFS : XEN_KERNEL_MODULE_LINE,
7727 			    DIRECT_BOOT_ARCHIVE,
7728 			    root_optional(osroot, menu_root));
7729 			BAM_DPRINTF((D_UPDATED_HV_ENTRY,
7730 			    fcn, bam_zfs, grubsign));
7731 		}
7732 	} else {
7733 		entry = update_boot_entry(mp, title, grubsign, grubroot,
7734 		    MULTI_BOOT, NULL, MULTIBOOT_ARCHIVE,
7735 		    root_optional(osroot, menu_root));
7736 
7737 		BAM_DPRINTF((D_UPDATED_MULTIBOOT_ENTRY, fcn, grubsign));
7738 	}
7739 
7740 	/*
7741 	 * Add the entry for failsafe archive.  On a bfu'd system, the
7742 	 * failsafe may be different than the installed kernel.
7743 	 */
7744 	(void) snprintf(failsafe, sizeof (failsafe), "%s%s",
7745 	    osroot, FAILSAFE_ARCHIVE_32);
7746 	(void) snprintf(failsafe_64, sizeof (failsafe_64), "%s%s",
7747 	    osroot, FAILSAFE_ARCHIVE_64);
7748 
7749 	/*
7750 	 * Check if at least one of the two archives exists
7751 	 * Using $ISADIR as the default line, we have an entry which works
7752 	 * for both the cases.
7753 	 */
7754 
7755 	if (stat(failsafe, &sbuf) == 0 || stat(failsafe_64, &sbuf) == 0) {
7756 
7757 		/* Figure out where the kernel line should point */
7758 		(void) snprintf(failsafe, sizeof (failsafe), "%s%s", osroot,
7759 		    DIRECT_BOOT_FAILSAFE_32);
7760 		(void) snprintf(failsafe_64, sizeof (failsafe_64), "%s%s",
7761 		    osroot, DIRECT_BOOT_FAILSAFE_64);
7762 		if (stat(failsafe, &sbuf) == 0 ||
7763 		    stat(failsafe_64, &sbuf) == 0) {
7764 			failsafe_kernel = DIRECT_BOOT_FAILSAFE_LINE;
7765 		} else {
7766 			(void) snprintf(failsafe, sizeof (failsafe), "%s%s",
7767 			    osroot, MULTI_BOOT_FAILSAFE);
7768 			if (stat(failsafe, &sbuf) == 0) {
7769 				failsafe_kernel = MULTI_BOOT_FAILSAFE_LINE;
7770 			}
7771 		}
7772 		if (failsafe_kernel != NULL) {
7773 			(void) update_boot_entry(mp, FAILSAFE_TITLE, grubsign,
7774 			    grubroot, failsafe_kernel, NULL, FAILSAFE_ARCHIVE,
7775 			    root_optional(osroot, menu_root));
7776 			BAM_DPRINTF((D_UPDATED_FAILSAFE_ENTRY, fcn,
7777 			    failsafe_kernel));
7778 		}
7779 	}
7780 	free(grubroot);
7781 
7782 	INJECT_ERROR1("UPDATE_ENTRY_ERROR", entry = BAM_ERROR);
7783 	if (entry == BAM_ERROR) {
7784 		bam_error(FAILED_TO_ADD_BOOT_ENTRY, title, grubsign);
7785 		free(grubsign);
7786 		return (BAM_ERROR);
7787 	}
7788 	free(grubsign);
7789 
7790 	update_numbering(mp);
7791 	ret = set_global(mp, menu_cmds[DEFAULT_CMD], entry);
7792 	INJECT_ERROR1("SET_DEFAULT_ERROR", ret = BAM_ERROR);
7793 	if (ret == BAM_ERROR) {
7794 		bam_error(SET_DEFAULT_FAILED, entry);
7795 	}
7796 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
7797 	return (BAM_WRITE);
7798 }
7799 
7800 static void
7801 save_default_entry(menu_t *mp, const char *which)
7802 {
7803 	int		lineNum;
7804 	int		entryNum;
7805 	int		entry = 0;	/* default is 0 */
7806 	char		linebuf[BAM_MAXLINE];
7807 	line_t		*lp = mp->curdefault;
7808 	const char	*fcn = "save_default_entry()";
7809 
7810 	if (mp->start) {
7811 		lineNum = mp->end->lineNum;
7812 		entryNum = mp->end->entryNum;
7813 	} else {
7814 		lineNum = LINE_INIT;
7815 		entryNum = ENTRY_INIT;
7816 	}
7817 
7818 	if (lp)
7819 		entry = s_strtol(lp->arg);
7820 
7821 	(void) snprintf(linebuf, sizeof (linebuf), "#%s%d", which, entry);
7822 	BAM_DPRINTF((D_SAVING_DEFAULT_TO, fcn, linebuf));
7823 	line_parser(mp, linebuf, &lineNum, &entryNum);
7824 	BAM_DPRINTF((D_SAVED_DEFAULT_TO, fcn, lineNum, entryNum));
7825 }
7826 
7827 static void
7828 restore_default_entry(menu_t *mp, const char *which, line_t *lp)
7829 {
7830 	int		entry;
7831 	char		*str;
7832 	const char	*fcn = "restore_default_entry()";
7833 
7834 	if (lp == NULL) {
7835 		BAM_DPRINTF((D_RESTORE_DEFAULT_NULL, fcn));
7836 		return;		/* nothing to restore */
7837 	}
7838 
7839 	BAM_DPRINTF((D_RESTORE_DEFAULT_STR, fcn, which));
7840 
7841 	str = lp->arg + strlen(which);
7842 	entry = s_strtol(str);
7843 	(void) set_global(mp, menu_cmds[DEFAULT_CMD], entry);
7844 
7845 	BAM_DPRINTF((D_RESTORED_DEFAULT_TO, fcn, entry));
7846 
7847 	/* delete saved old default line */
7848 	unlink_line(mp, lp);
7849 	line_free(lp);
7850 }
7851 
7852 /*
7853  * This function is for supporting reboot with args.
7854  * The opt value can be:
7855  * NULL		delete temp entry, if present
7856  * entry=<n>	switches default entry to <n>
7857  * else		treated as boot-args and setup a temperary menu entry
7858  *		and make it the default
7859  * Note that we are always rebooting the current OS instance
7860  * so osroot == / always.
7861  */
7862 #define	REBOOT_TITLE	"Solaris_reboot_transient"
7863 
7864 /*ARGSUSED*/
7865 static error_t
7866 update_temp(menu_t *mp, char *dummy, char *opt)
7867 {
7868 	int		entry;
7869 	char		*osdev;
7870 	char		*fstype;
7871 	char		*sign;
7872 	char		*opt_ptr;
7873 	char		*path;
7874 	char		kernbuf[BUFSIZ];
7875 	char		args_buf[BUFSIZ];
7876 	char		signbuf[PATH_MAX];
7877 	int		ret;
7878 	const char	*fcn = "update_temp()";
7879 
7880 	assert(mp);
7881 	assert(dummy == NULL);
7882 
7883 	/* opt can be NULL */
7884 	BAM_DPRINTF((D_FUNC_ENTRY1, fcn, opt ? opt : "<NULL>"));
7885 	BAM_DPRINTF((D_BAM_ROOT, fcn, bam_alt_root, bam_root));
7886 
7887 	if (bam_alt_root || bam_rootlen != 1 ||
7888 	    strcmp(bam_root, "/") != 0 ||
7889 	    strcmp(rootbuf, "/") != 0) {
7890 		bam_error(ALT_ROOT_INVALID, bam_root);
7891 		return (BAM_ERROR);
7892 	}
7893 
7894 	/* If no option, delete exiting reboot menu entry */
7895 	if (opt == NULL) {
7896 		entry_t		*ent;
7897 		BAM_DPRINTF((D_OPT_NULL, fcn));
7898 		ent = find_boot_entry(mp, REBOOT_TITLE, NULL, NULL,
7899 		    NULL, NULL, 0, &entry);
7900 		if (ent == NULL) {	/* not found is ok */
7901 			BAM_DPRINTF((D_TRANSIENT_NOTFOUND, fcn));
7902 			return (BAM_SUCCESS);
7903 		}
7904 		(void) do_delete(mp, entry);
7905 		restore_default_entry(mp, BAM_OLDDEF, mp->olddefault);
7906 		mp->olddefault = NULL;
7907 		BAM_DPRINTF((D_RESTORED_DEFAULT, fcn));
7908 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
7909 		return (BAM_WRITE);
7910 	}
7911 
7912 	/* if entry= is specified, set the default entry */
7913 	if (strncmp(opt, "entry=", strlen("entry=")) == 0) {
7914 		int entryNum = s_strtol(opt + strlen("entry="));
7915 		BAM_DPRINTF((D_ENTRY_EQUALS, fcn, opt));
7916 		if (selector(mp, opt, &entry, NULL) == BAM_SUCCESS) {
7917 			/* this is entry=# option */
7918 			ret = set_global(mp, menu_cmds[DEFAULT_CMD], entry);
7919 			BAM_DPRINTF((D_ENTRY_SET_IS, fcn, entry, ret));
7920 			return (ret);
7921 		} else {
7922 			bam_error(SET_DEFAULT_FAILED, entryNum);
7923 			return (BAM_ERROR);
7924 		}
7925 	}
7926 
7927 	/*
7928 	 * add a new menu entry based on opt and make it the default
7929 	 */
7930 
7931 	fstype = get_fstype("/");
7932 	INJECT_ERROR1("REBOOT_FSTYPE_NULL", fstype = NULL);
7933 	if (fstype == NULL) {
7934 		bam_error(REBOOT_FSTYPE_FAILED);
7935 		return (BAM_ERROR);
7936 	}
7937 
7938 	osdev = get_special("/");
7939 	INJECT_ERROR1("REBOOT_SPECIAL_NULL", osdev = NULL);
7940 	if (osdev == NULL) {
7941 		free(fstype);
7942 		bam_error(REBOOT_SPECIAL_FAILED);
7943 		return (BAM_ERROR);
7944 	}
7945 
7946 	sign = find_existing_sign("/", osdev, fstype);
7947 	INJECT_ERROR1("REBOOT_SIGN_NULL", sign = NULL);
7948 	if (sign == NULL) {
7949 		free(fstype);
7950 		free(osdev);
7951 		bam_error(REBOOT_SIGN_FAILED);
7952 		return (BAM_ERROR);
7953 	}
7954 
7955 	free(fstype);
7956 	free(osdev);
7957 	(void) strlcpy(signbuf, sign, sizeof (signbuf));
7958 	free(sign);
7959 
7960 	assert(strchr(signbuf, '(') == NULL && strchr(signbuf, ',') == NULL &&
7961 	    strchr(signbuf, ')') == NULL);
7962 
7963 	/*
7964 	 * There is no alternate root while doing reboot with args
7965 	 * This version of bootadm is only delivered with a DBOOT
7966 	 * version of Solaris.
7967 	 */
7968 	INJECT_ERROR1("REBOOT_NOT_DBOOT", bam_direct = BAM_DIRECT_MULTIBOOT);
7969 	if (bam_direct != BAM_DIRECT_DBOOT) {
7970 		bam_error(REBOOT_DIRECT_FAILED);
7971 		return (BAM_ERROR);
7972 	}
7973 
7974 	/* add an entry for Solaris reboot */
7975 	if (opt[0] == '-') {
7976 		/* It's an option - first see if boot-file is set */
7977 		ret = get_kernel(mp, KERNEL_CMD, kernbuf, sizeof (kernbuf));
7978 		INJECT_ERROR1("REBOOT_GET_KERNEL", ret = BAM_ERROR);
7979 		if (ret != BAM_SUCCESS) {
7980 			bam_error(REBOOT_GET_KERNEL_FAILED);
7981 			return (BAM_ERROR);
7982 		}
7983 		if (kernbuf[0] == '\0')
7984 			(void) strlcpy(kernbuf, DIRECT_BOOT_KERNEL,
7985 			    sizeof (kernbuf));
7986 		(void) strlcat(kernbuf, " ", sizeof (kernbuf));
7987 		(void) strlcat(kernbuf, opt, sizeof (kernbuf));
7988 		BAM_DPRINTF((D_REBOOT_OPTION, fcn, kernbuf));
7989 	} else if (opt[0] == '/') {
7990 		/* It's a full path, so write it out. */
7991 		(void) strlcpy(kernbuf, opt, sizeof (kernbuf));
7992 
7993 		/*
7994 		 * If someone runs:
7995 		 *
7996 		 *	# eeprom boot-args='-kd'
7997 		 *	# reboot /platform/i86pc/kernel/unix
7998 		 *
7999 		 * we want to use the boot-args as part of the boot
8000 		 * line.  On the other hand, if someone runs:
8001 		 *
8002 		 *	# reboot "/platform/i86pc/kernel/unix -kd"
8003 		 *
8004 		 * we don't need to mess with boot-args.  If there's
8005 		 * no space in the options string, assume we're in the
8006 		 * first case.
8007 		 */
8008 		if (strchr(opt, ' ') == NULL) {
8009 			ret = get_kernel(mp, ARGS_CMD, args_buf,
8010 			    sizeof (args_buf));
8011 			INJECT_ERROR1("REBOOT_GET_ARGS", ret = BAM_ERROR);
8012 			if (ret != BAM_SUCCESS) {
8013 				bam_error(REBOOT_GET_ARGS_FAILED);
8014 				return (BAM_ERROR);
8015 			}
8016 
8017 			if (args_buf[0] != '\0') {
8018 				(void) strlcat(kernbuf, " ", sizeof (kernbuf));
8019 				(void) strlcat(kernbuf, args_buf,
8020 				    sizeof (kernbuf));
8021 			}
8022 		}
8023 		BAM_DPRINTF((D_REBOOT_ABSPATH, fcn, kernbuf));
8024 	} else {
8025 		/*
8026 		 * It may be a partial path, or it may be a partial
8027 		 * path followed by options.  Assume that only options
8028 		 * follow a space.  If someone sends us a kernel path
8029 		 * that includes a space, they deserve to be broken.
8030 		 */
8031 		opt_ptr = strchr(opt, ' ');
8032 		if (opt_ptr != NULL) {
8033 			*opt_ptr = '\0';
8034 		}
8035 
8036 		path = expand_path(opt);
8037 		if (path != NULL) {
8038 			(void) strlcpy(kernbuf, path, sizeof (kernbuf));
8039 			free(path);
8040 
8041 			/*
8042 			 * If there were options given, use those.
8043 			 * Otherwise, copy over the default options.
8044 			 */
8045 			if (opt_ptr != NULL) {
8046 				/* Restore the space in opt string */
8047 				*opt_ptr = ' ';
8048 				(void) strlcat(kernbuf, opt_ptr,
8049 				    sizeof (kernbuf));
8050 			} else {
8051 				ret = get_kernel(mp, ARGS_CMD, args_buf,
8052 				    sizeof (args_buf));
8053 				INJECT_ERROR1("UPDATE_TEMP_PARTIAL_ARGS",
8054 				    ret = BAM_ERROR);
8055 				if (ret != BAM_SUCCESS) {
8056 					bam_error(REBOOT_GET_ARGS_FAILED);
8057 					return (BAM_ERROR);
8058 				}
8059 
8060 				if (args_buf[0] != '\0') {
8061 					(void) strlcat(kernbuf, " ",
8062 					    sizeof (kernbuf));
8063 					(void) strlcat(kernbuf,
8064 					    args_buf, sizeof (kernbuf));
8065 				}
8066 			}
8067 			BAM_DPRINTF((D_REBOOT_RESOLVED_PARTIAL, fcn, kernbuf));
8068 		} else {
8069 			bam_error(UNKNOWN_KERNEL, opt);
8070 			bam_print_stderr(UNKNOWN_KERNEL_REBOOT);
8071 			return (BAM_ERROR);
8072 		}
8073 	}
8074 	entry = add_boot_entry(mp, REBOOT_TITLE, signbuf, kernbuf,
8075 	    NULL, NULL);
8076 	INJECT_ERROR1("REBOOT_ADD_BOOT_ENTRY", entry = BAM_ERROR);
8077 	if (entry == BAM_ERROR) {
8078 		bam_error(REBOOT_WITH_ARGS_ADD_ENTRY_FAILED);
8079 		return (BAM_ERROR);
8080 	}
8081 
8082 	save_default_entry(mp, BAM_OLDDEF);
8083 	ret = set_global(mp, menu_cmds[DEFAULT_CMD], entry);
8084 	INJECT_ERROR1("REBOOT_SET_GLOBAL", ret = BAM_ERROR);
8085 	if (ret == BAM_ERROR) {
8086 		bam_error(REBOOT_SET_DEFAULT_FAILED, entry);
8087 	}
8088 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
8089 	return (BAM_WRITE);
8090 }
8091 
8092 static error_t
8093 set_global(menu_t *mp, char *globalcmd, int val)
8094 {
8095 	line_t		*lp;
8096 	line_t		*found;
8097 	line_t		*last;
8098 	char		*cp;
8099 	char		*str;
8100 	char		prefix[BAM_MAXLINE];
8101 	size_t		len;
8102 	const char	*fcn = "set_global()";
8103 
8104 	assert(mp);
8105 	assert(globalcmd);
8106 
8107 	if (strcmp(globalcmd, menu_cmds[DEFAULT_CMD]) == 0) {
8108 		INJECT_ERROR1("SET_GLOBAL_VAL_NEG", val = -1);
8109 		INJECT_ERROR1("SET_GLOBAL_MENU_EMPTY", mp->end = NULL);
8110 		INJECT_ERROR1("SET_GLOBAL_VAL_TOO_BIG", val = 100);
8111 		if (val < 0 || mp->end == NULL || val > mp->end->entryNum) {
8112 			(void) snprintf(prefix, sizeof (prefix), "%d", val);
8113 			bam_error(INVALID_ENTRY, prefix);
8114 			return (BAM_ERROR);
8115 		}
8116 	}
8117 
8118 	found = last = NULL;
8119 	for (lp = mp->start; lp; lp = lp->next) {
8120 		if (lp->flags != BAM_GLOBAL)
8121 			continue;
8122 
8123 		last = lp; /* track the last global found */
8124 
8125 		INJECT_ERROR1("SET_GLOBAL_NULL_CMD", lp->cmd = NULL);
8126 		if (lp->cmd == NULL) {
8127 			bam_error(NO_CMD, lp->lineNum);
8128 			continue;
8129 		}
8130 		if (strcmp(globalcmd, lp->cmd) != 0)
8131 			continue;
8132 
8133 		BAM_DPRINTF((D_FOUND_GLOBAL, fcn, globalcmd));
8134 
8135 		if (found) {
8136 			bam_error(DUP_CMD, globalcmd, lp->lineNum, bam_root);
8137 		}
8138 		found = lp;
8139 	}
8140 
8141 	if (found == NULL) {
8142 		lp = s_calloc(1, sizeof (line_t));
8143 		if (last == NULL) {
8144 			lp->next = mp->start;
8145 			mp->start = lp;
8146 			mp->end = (mp->end) ? mp->end : lp;
8147 		} else {
8148 			lp->next = last->next;
8149 			last->next = lp;
8150 			if (lp->next == NULL)
8151 				mp->end = lp;
8152 		}
8153 		lp->flags = BAM_GLOBAL; /* other fields not needed for writes */
8154 		len = strlen(globalcmd) + strlen(menu_cmds[SEP_CMD]);
8155 		len += 10;	/* val < 10 digits */
8156 		lp->line = s_calloc(1, len);
8157 		(void) snprintf(lp->line, len, "%s%s%d",
8158 		    globalcmd, menu_cmds[SEP_CMD], val);
8159 		BAM_DPRINTF((D_SET_GLOBAL_WROTE_NEW, fcn, lp->line));
8160 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
8161 		return (BAM_WRITE);
8162 	}
8163 
8164 	/*
8165 	 * We are changing an existing entry. Retain any prefix whitespace,
8166 	 * but overwrite everything else. This preserves tabs added for
8167 	 * readability.
8168 	 */
8169 	str = found->line;
8170 	cp = prefix;
8171 	while (*str == ' ' || *str == '\t')
8172 		*(cp++) = *(str++);
8173 	*cp = '\0'; /* Terminate prefix */
8174 	len = strlen(prefix) + strlen(globalcmd);
8175 	len += strlen(menu_cmds[SEP_CMD]) + 10;
8176 
8177 	free(found->line);
8178 	found->line = s_calloc(1, len);
8179 	(void) snprintf(found->line, len,
8180 	    "%s%s%s%d", prefix, globalcmd, menu_cmds[SEP_CMD], val);
8181 
8182 	BAM_DPRINTF((D_SET_GLOBAL_REPLACED, fcn, found->line));
8183 	BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
8184 	return (BAM_WRITE); /* need a write to menu */
8185 }
8186 
8187 /*
8188  * partial_path may be anything like "kernel/unix" or "kmdb".  Try to
8189  * expand it to a full unix path.  The calling function is expected to
8190  * output a message if an error occurs and NULL is returned.
8191  */
8192 static char *
8193 expand_path(const char *partial_path)
8194 {
8195 	int		new_path_len;
8196 	char		*new_path;
8197 	char		new_path2[PATH_MAX];
8198 	struct stat	sb;
8199 	const char	*fcn = "expand_path()";
8200 
8201 	new_path_len = strlen(partial_path) + 64;
8202 	new_path = s_calloc(1, new_path_len);
8203 
8204 	/* First, try the simplest case - something like "kernel/unix" */
8205 	(void) snprintf(new_path, new_path_len, "/platform/i86pc/%s",
8206 	    partial_path);
8207 	if (stat(new_path, &sb) == 0) {
8208 		BAM_DPRINTF((D_EXPAND_PATH, fcn, new_path));
8209 		return (new_path);
8210 	}
8211 
8212 	if (strcmp(partial_path, "kmdb") == 0) {
8213 		(void) snprintf(new_path, new_path_len, "%s -k",
8214 		    DIRECT_BOOT_KERNEL);
8215 		BAM_DPRINTF((D_EXPAND_PATH, fcn, new_path));
8216 		return (new_path);
8217 	}
8218 
8219 	/*
8220 	 * We've quickly reached unsupported usage.  Try once more to
8221 	 * see if we were just given a glom name.
8222 	 */
8223 	(void) snprintf(new_path, new_path_len, "/platform/i86pc/%s/unix",
8224 	    partial_path);
8225 	(void) snprintf(new_path2, PATH_MAX, "/platform/i86pc/%s/amd64/unix",
8226 	    partial_path);
8227 	if (stat(new_path, &sb) == 0) {
8228 		if (stat(new_path2, &sb) == 0) {
8229 			/*
8230 			 * We matched both, so we actually
8231 			 * want to write the $ISADIR version.
8232 			 */
8233 			(void) snprintf(new_path, new_path_len,
8234 			    "/platform/i86pc/kernel/%s/$ISADIR/unix",
8235 			    partial_path);
8236 		}
8237 		BAM_DPRINTF((D_EXPAND_PATH, fcn, new_path));
8238 		return (new_path);
8239 	}
8240 
8241 	free(new_path);
8242 	BAM_DPRINTF((D_RETURN_FAILURE, fcn));
8243 	return (NULL);
8244 }
8245 
8246 /*
8247  * The kernel cmd and arg have been changed, so
8248  * check whether the archive line needs to change.
8249  */
8250 static void
8251 set_archive_line(entry_t *entryp, line_t *kernelp)
8252 {
8253 	line_t		*lp = entryp->start;
8254 	char		*new_archive;
8255 	menu_cmd_t	m_cmd;
8256 	const char	*fcn = "set_archive_line()";
8257 
8258 	for (; lp != NULL; lp = lp->next) {
8259 		if (strncmp(lp->cmd, menu_cmds[MODULE_CMD],
8260 		    sizeof (menu_cmds[MODULE_CMD]) - 1) == 0) {
8261 			break;
8262 		}
8263 
8264 		INJECT_ERROR1("SET_ARCHIVE_LINE_END_ENTRY", lp = entryp->end);
8265 		if (lp == entryp->end) {
8266 			BAM_DPRINTF((D_ARCHIVE_LINE_NONE, fcn,
8267 			    entryp->entryNum));
8268 			return;
8269 		}
8270 	}
8271 	INJECT_ERROR1("SET_ARCHIVE_LINE_END_MENU", lp = NULL);
8272 	if (lp == NULL) {
8273 		BAM_DPRINTF((D_ARCHIVE_LINE_NONE, fcn, entryp->entryNum));
8274 		return;
8275 	}
8276 
8277 	if (strstr(kernelp->arg, "$ISADIR") != NULL) {
8278 		new_archive = DIRECT_BOOT_ARCHIVE;
8279 		m_cmd = MODULE_DOLLAR_CMD;
8280 	} else if (strstr(kernelp->arg, "amd64") != NULL) {
8281 		new_archive = DIRECT_BOOT_ARCHIVE_64;
8282 		m_cmd = MODULE_CMD;
8283 	} else {
8284 		new_archive = DIRECT_BOOT_ARCHIVE_32;
8285 		m_cmd = MODULE_CMD;
8286 	}
8287 
8288 	if (strcmp(lp->arg, new_archive) == 0) {
8289 		BAM_DPRINTF((D_ARCHIVE_LINE_NOCHANGE, fcn, lp->arg));
8290 		return;
8291 	}
8292 
8293 	if (strcmp(lp->cmd, menu_cmds[m_cmd]) != 0) {
8294 		free(lp->cmd);
8295 		lp->cmd = s_strdup(menu_cmds[m_cmd]);
8296 	}
8297 
8298 	free(lp->arg);
8299 	lp->arg = s_strdup(new_archive);
8300 	update_line(lp);
8301 	BAM_DPRINTF((D_ARCHIVE_LINE_REPLACED, fcn, lp->line));
8302 }
8303 
8304 /*
8305  * Title for an entry to set properties that once went in bootenv.rc.
8306  */
8307 #define	BOOTENV_RC_TITLE	"Solaris bootenv rc"
8308 
8309 /*
8310  * If path is NULL, return the kernel (optnum == KERNEL_CMD) or arguments
8311  * (optnum == ARGS_CMD) in the argument buf.  If path is a zero-length
8312  * string, reset the value to the default.  If path is a non-zero-length
8313  * string, set the kernel or arguments.
8314  */
8315 static error_t
8316 get_set_kernel(
8317 	menu_t *mp,
8318 	menu_cmd_t optnum,
8319 	char *path,
8320 	char *buf,
8321 	size_t bufsize)
8322 {
8323 	int		entryNum;
8324 	int		rv = BAM_SUCCESS;
8325 	int		free_new_path = 0;
8326 	entry_t		*entryp;
8327 	line_t		*ptr;
8328 	line_t		*kernelp;
8329 	char		*new_arg;
8330 	char		*old_args;
8331 	char		*space;
8332 	char		*new_path;
8333 	char		old_space;
8334 	size_t		old_kernel_len;
8335 	size_t		new_str_len;
8336 	char		*fstype;
8337 	char		*osdev;
8338 	char		*sign;
8339 	char		signbuf[PATH_MAX];
8340 	int		ret;
8341 	const char	*fcn = "get_set_kernel()";
8342 
8343 	assert(bufsize > 0);
8344 
8345 	ptr = kernelp = NULL;
8346 	new_arg = old_args = space = NULL;
8347 	new_path = NULL;
8348 	buf[0] = '\0';
8349 
8350 	INJECT_ERROR1("GET_SET_KERNEL_NOT_DBOOT",
8351 	    bam_direct = BAM_DIRECT_MULTIBOOT);
8352 	if (bam_direct != BAM_DIRECT_DBOOT) {
8353 		bam_error(NOT_DBOOT, optnum == KERNEL_CMD ? "kernel" : "args");
8354 		return (BAM_ERROR);
8355 	}
8356 
8357 	/*
8358 	 * If a user changed the default entry to a non-bootadm controlled
8359 	 * one, we don't want to mess with it.  Just print an error and
8360 	 * return.
8361 	 */
8362 	if (mp->curdefault) {
8363 		entryNum = s_strtol(mp->curdefault->arg);
8364 		for (entryp = mp->entries; entryp; entryp = entryp->next) {
8365 			if (entryp->entryNum == entryNum)
8366 				break;
8367 		}
8368 		if ((entryp != NULL) &&
8369 		    ((entryp->flags & (BAM_ENTRY_BOOTADM|BAM_ENTRY_LU)) == 0)) {
8370 			bam_error(DEFAULT_NOT_BAM);
8371 			return (BAM_ERROR);
8372 		}
8373 	}
8374 
8375 	entryp = find_boot_entry(mp, BOOTENV_RC_TITLE, NULL, NULL, NULL, NULL,
8376 	    0, &entryNum);
8377 
8378 	if (entryp != NULL) {
8379 		for (ptr = entryp->start; ptr && ptr != entryp->end;
8380 		    ptr = ptr->next) {
8381 			if (strncmp(ptr->cmd, menu_cmds[KERNEL_CMD],
8382 			    sizeof (menu_cmds[KERNEL_CMD]) - 1) == 0) {
8383 				kernelp = ptr;
8384 				break;
8385 			}
8386 		}
8387 		if (kernelp == NULL) {
8388 			bam_error(NO_KERNEL, entryNum);
8389 			return (BAM_ERROR);
8390 		}
8391 
8392 		old_kernel_len = strcspn(kernelp->arg, " \t");
8393 		space = old_args = kernelp->arg + old_kernel_len;
8394 		while ((*old_args == ' ') || (*old_args == '\t'))
8395 			old_args++;
8396 	}
8397 
8398 	if (path == NULL) {
8399 		if (entryp == NULL) {
8400 			BAM_DPRINTF((D_GET_SET_KERNEL_NO_RC, fcn));
8401 			BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
8402 			return (BAM_SUCCESS);
8403 		}
8404 		assert(kernelp);
8405 		if (optnum == ARGS_CMD) {
8406 			if (old_args[0] != '\0') {
8407 				(void) strlcpy(buf, old_args, bufsize);
8408 				BAM_DPRINTF((D_GET_SET_KERNEL_ARGS, fcn, buf));
8409 			}
8410 		} else {
8411 			/*
8412 			 * We need to print the kernel, so we just turn the
8413 			 * first space into a '\0' and print the beginning.
8414 			 * We don't print anything if it's the default kernel.
8415 			 */
8416 			old_space = *space;
8417 			*space = '\0';
8418 			if (strcmp(kernelp->arg, DIRECT_BOOT_KERNEL) != 0) {
8419 				(void) strlcpy(buf, kernelp->arg, bufsize);
8420 				BAM_DPRINTF((D_GET_SET_KERNEL_KERN, fcn, buf));
8421 			}
8422 			*space = old_space;
8423 		}
8424 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
8425 		return (BAM_SUCCESS);
8426 	}
8427 
8428 	/*
8429 	 * First, check if we're resetting an entry to the default.
8430 	 */
8431 	if ((path[0] == '\0') ||
8432 	    ((optnum == KERNEL_CMD) &&
8433 	    (strcmp(path, DIRECT_BOOT_KERNEL) == 0))) {
8434 		if ((entryp == NULL) || (kernelp == NULL)) {
8435 			/* No previous entry, it's already the default */
8436 			BAM_DPRINTF((D_GET_SET_KERNEL_ALREADY, fcn));
8437 			return (BAM_SUCCESS);
8438 		}
8439 
8440 		/*
8441 		 * Check if we can delete the entry.  If we're resetting the
8442 		 * kernel command, and the args is already empty, or if we're
8443 		 * resetting the args command, and the kernel is already the
8444 		 * default, we can restore the old default and delete the entry.
8445 		 */
8446 		if (((optnum == KERNEL_CMD) &&
8447 		    ((old_args == NULL) || (old_args[0] == '\0'))) ||
8448 		    ((optnum == ARGS_CMD) &&
8449 		    (strncmp(kernelp->arg, DIRECT_BOOT_KERNEL,
8450 		    sizeof (DIRECT_BOOT_KERNEL) - 1) == 0))) {
8451 			kernelp = NULL;
8452 			(void) do_delete(mp, entryNum);
8453 			restore_default_entry(mp, BAM_OLD_RC_DEF,
8454 			    mp->old_rc_default);
8455 			mp->old_rc_default = NULL;
8456 			rv = BAM_WRITE;
8457 			BAM_DPRINTF((D_GET_SET_KERNEL_RESTORE_DEFAULT, fcn));
8458 			goto done;
8459 		}
8460 
8461 		if (optnum == KERNEL_CMD) {
8462 			/*
8463 			 * At this point, we've already checked that old_args
8464 			 * and entryp are valid pointers.  The "+ 2" is for
8465 			 * a space a the string termination character.
8466 			 */
8467 			new_str_len = (sizeof (DIRECT_BOOT_KERNEL) - 1) +
8468 			    strlen(old_args) + 2;
8469 			new_arg = s_calloc(1, new_str_len);
8470 			(void) snprintf(new_arg, new_str_len, "%s %s",
8471 			    DIRECT_BOOT_KERNEL, old_args);
8472 			free(kernelp->arg);
8473 			kernelp->arg = new_arg;
8474 
8475 			/*
8476 			 * We have changed the kernel line, so we may need
8477 			 * to update the archive line as well.
8478 			 */
8479 			set_archive_line(entryp, kernelp);
8480 			BAM_DPRINTF((D_GET_SET_KERNEL_RESET_KERNEL_SET_ARG,
8481 			    fcn, kernelp->arg));
8482 		} else {
8483 			/*
8484 			 * We're resetting the boot args to nothing, so
8485 			 * we only need to copy the kernel.  We've already
8486 			 * checked that the kernel is not the default.
8487 			 */
8488 			new_arg = s_calloc(1, old_kernel_len + 1);
8489 			(void) snprintf(new_arg, old_kernel_len + 1, "%s",
8490 			    kernelp->arg);
8491 			free(kernelp->arg);
8492 			kernelp->arg = new_arg;
8493 			BAM_DPRINTF((D_GET_SET_KERNEL_RESET_ARG_SET_KERNEL,
8494 			    fcn, kernelp->arg));
8495 		}
8496 		rv = BAM_WRITE;
8497 		goto done;
8498 	}
8499 
8500 	/*
8501 	 * Expand the kernel file to a full path, if necessary
8502 	 */
8503 	if ((optnum == KERNEL_CMD) && (path[0] != '/')) {
8504 		new_path = expand_path(path);
8505 		if (new_path == NULL) {
8506 			bam_error(UNKNOWN_KERNEL, path);
8507 			BAM_DPRINTF((D_RETURN_FAILURE, fcn));
8508 			return (BAM_ERROR);
8509 		}
8510 		free_new_path = 1;
8511 	} else {
8512 		new_path = path;
8513 		free_new_path = 0;
8514 	}
8515 
8516 	/*
8517 	 * At this point, we know we're setting a new value.  First, take care
8518 	 * of the case where there was no previous entry.
8519 	 */
8520 	if (entryp == NULL) {
8521 
8522 		/* Similar to code in update_temp */
8523 		fstype = get_fstype("/");
8524 		INJECT_ERROR1("GET_SET_KERNEL_FSTYPE", fstype = NULL);
8525 		if (fstype == NULL) {
8526 			bam_error(BOOTENV_FSTYPE_FAILED);
8527 			rv = BAM_ERROR;
8528 			goto done;
8529 		}
8530 
8531 		osdev = get_special("/");
8532 		INJECT_ERROR1("GET_SET_KERNEL_SPECIAL", osdev = NULL);
8533 		if (osdev == NULL) {
8534 			free(fstype);
8535 			bam_error(BOOTENV_SPECIAL_FAILED);
8536 			rv = BAM_ERROR;
8537 			goto done;
8538 		}
8539 
8540 		sign = find_existing_sign("/", osdev, fstype);
8541 		INJECT_ERROR1("GET_SET_KERNEL_SIGN", sign = NULL);
8542 		if (sign == NULL) {
8543 			free(fstype);
8544 			free(osdev);
8545 			bam_error(BOOTENV_SIGN_FAILED);
8546 			rv = BAM_ERROR;
8547 			goto done;
8548 		}
8549 
8550 		free(fstype);
8551 		free(osdev);
8552 		(void) strlcpy(signbuf, sign, sizeof (signbuf));
8553 		free(sign);
8554 		assert(strchr(signbuf, '(') == NULL &&
8555 		    strchr(signbuf, ',') == NULL &&
8556 		    strchr(signbuf, ')') == NULL);
8557 
8558 		if (optnum == KERNEL_CMD) {
8559 			BAM_DPRINTF((D_GET_SET_KERNEL_NEW_KERN, fcn, new_path));
8560 			entryNum = add_boot_entry(mp, BOOTENV_RC_TITLE,
8561 			    signbuf, new_path, NULL, NULL);
8562 		} else {
8563 			new_str_len = strlen(DIRECT_BOOT_KERNEL) +
8564 			    strlen(path) + 8;
8565 			new_arg = s_calloc(1, new_str_len);
8566 
8567 			(void) snprintf(new_arg, new_str_len, "%s %s",
8568 			    DIRECT_BOOT_KERNEL, path);
8569 			BAM_DPRINTF((D_GET_SET_KERNEL_NEW_ARG, fcn, new_arg));
8570 			entryNum = add_boot_entry(mp, BOOTENV_RC_TITLE,
8571 			    signbuf, new_arg, NULL, DIRECT_BOOT_ARCHIVE);
8572 			free(new_arg);
8573 		}
8574 		INJECT_ERROR1("GET_SET_KERNEL_ADD_BOOT_ENTRY",
8575 		    entryNum = BAM_ERROR);
8576 		if (entryNum == BAM_ERROR) {
8577 			bam_error(GET_SET_KERNEL_ADD_BOOT_ENTRY,
8578 			    BOOTENV_RC_TITLE);
8579 			rv = BAM_ERROR;
8580 			goto done;
8581 		}
8582 		save_default_entry(mp, BAM_OLD_RC_DEF);
8583 		ret = set_global(mp, menu_cmds[DEFAULT_CMD], entryNum);
8584 		INJECT_ERROR1("GET_SET_KERNEL_SET_GLOBAL", ret = BAM_ERROR);
8585 		if (ret == BAM_ERROR) {
8586 			bam_error(GET_SET_KERNEL_SET_GLOBAL, entryNum);
8587 		}
8588 		rv = BAM_WRITE;
8589 		goto done;
8590 	}
8591 
8592 	/*
8593 	 * There was already an bootenv entry which we need to edit.
8594 	 */
8595 	if (optnum == KERNEL_CMD) {
8596 		new_str_len = strlen(new_path) + strlen(old_args) + 2;
8597 		new_arg = s_calloc(1, new_str_len);
8598 		(void) snprintf(new_arg, new_str_len, "%s %s", new_path,
8599 		    old_args);
8600 		free(kernelp->arg);
8601 		kernelp->arg = new_arg;
8602 
8603 		/*
8604 		 * If we have changed the kernel line, we may need to update
8605 		 * the archive line as well.
8606 		 */
8607 		set_archive_line(entryp, kernelp);
8608 		BAM_DPRINTF((D_GET_SET_KERNEL_REPLACED_KERNEL_SAME_ARG, fcn,
8609 		    kernelp->arg));
8610 	} else {
8611 		new_str_len = old_kernel_len + strlen(path) + 8;
8612 		new_arg = s_calloc(1, new_str_len);
8613 		(void) strncpy(new_arg, kernelp->arg, old_kernel_len);
8614 		(void) strlcat(new_arg, " ", new_str_len);
8615 		(void) strlcat(new_arg, path, new_str_len);
8616 		free(kernelp->arg);
8617 		kernelp->arg = new_arg;
8618 		BAM_DPRINTF((D_GET_SET_KERNEL_SAME_KERNEL_REPLACED_ARG, fcn,
8619 		    kernelp->arg));
8620 	}
8621 	rv = BAM_WRITE;
8622 
8623 done:
8624 	if ((rv == BAM_WRITE) && kernelp)
8625 		update_line(kernelp);
8626 	if (free_new_path)
8627 		free(new_path);
8628 	if (rv == BAM_WRITE) {
8629 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
8630 	} else {
8631 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
8632 	}
8633 	return (rv);
8634 }
8635 
8636 static error_t
8637 get_kernel(menu_t *mp, menu_cmd_t optnum, char *buf, size_t bufsize)
8638 {
8639 	const char	*fcn = "get_kernel()";
8640 	BAM_DPRINTF((D_FUNC_ENTRY1, fcn, menu_cmds[optnum]));
8641 	return (get_set_kernel(mp, optnum, NULL, buf, bufsize));
8642 }
8643 
8644 static error_t
8645 set_kernel(menu_t *mp, menu_cmd_t optnum, char *path, char *buf, size_t bufsize)
8646 {
8647 	const char	*fcn = "set_kernel()";
8648 	assert(path != NULL);
8649 	BAM_DPRINTF((D_FUNC_ENTRY2, fcn, menu_cmds[optnum], path));
8650 	return (get_set_kernel(mp, optnum, path, buf, bufsize));
8651 }
8652 
8653 /*ARGSUSED*/
8654 static error_t
8655 set_option(menu_t *mp, char *dummy, char *opt)
8656 {
8657 	int		optnum;
8658 	int		optval;
8659 	char		*val;
8660 	char		buf[BUFSIZ] = "";
8661 	error_t		rv;
8662 	const char	*fcn = "set_option()";
8663 
8664 	assert(mp);
8665 	assert(opt);
8666 	assert(dummy == NULL);
8667 
8668 	/* opt is set from bam_argv[0] and is always non-NULL */
8669 	BAM_DPRINTF((D_FUNC_ENTRY1, fcn, opt));
8670 
8671 	val = strchr(opt, '=');
8672 	if (val != NULL) {
8673 		*val = '\0';
8674 	}
8675 
8676 	if (strcmp(opt, "default") == 0) {
8677 		optnum = DEFAULT_CMD;
8678 	} else if (strcmp(opt, "timeout") == 0) {
8679 		optnum = TIMEOUT_CMD;
8680 	} else if (strcmp(opt, menu_cmds[KERNEL_CMD]) == 0) {
8681 		optnum = KERNEL_CMD;
8682 	} else if (strcmp(opt, menu_cmds[ARGS_CMD]) == 0) {
8683 		optnum = ARGS_CMD;
8684 	} else {
8685 		bam_error(INVALID_OPTION, opt);
8686 		return (BAM_ERROR);
8687 	}
8688 
8689 	/*
8690 	 * kernel and args are allowed without "=new_value" strings.  All
8691 	 * others cause errors
8692 	 */
8693 	if ((val == NULL) && (optnum != KERNEL_CMD) && (optnum != ARGS_CMD)) {
8694 		bam_error(NO_OPTION_ARG, opt);
8695 		return (BAM_ERROR);
8696 	} else if (val != NULL) {
8697 		*val = '=';
8698 	}
8699 
8700 	if ((optnum == KERNEL_CMD) || (optnum == ARGS_CMD)) {
8701 		BAM_DPRINTF((D_SET_OPTION, fcn, menu_cmds[optnum],
8702 		    val ? val + 1 : "NULL"));
8703 
8704 		if (val)
8705 			rv = set_kernel(mp, optnum, val + 1, buf, sizeof (buf));
8706 		else
8707 			rv = get_kernel(mp, optnum, buf, sizeof (buf));
8708 		if ((rv == BAM_SUCCESS) && (buf[0] != '\0'))
8709 			(void) printf("%s\n", buf);
8710 	} else {
8711 		optval = s_strtol(val + 1);
8712 		BAM_DPRINTF((D_SET_OPTION, fcn, menu_cmds[optnum], val + 1));
8713 		rv = set_global(mp, menu_cmds[optnum], optval);
8714 	}
8715 
8716 	if (rv == BAM_WRITE || rv == BAM_SUCCESS) {
8717 		BAM_DPRINTF((D_RETURN_SUCCESS, fcn));
8718 	} else {
8719 		BAM_DPRINTF((D_RETURN_FAILURE, fcn));
8720 	}
8721 
8722 	return (rv);
8723 }
8724 
8725 /*
8726  * The quiet argument suppresses messages. This is used
8727  * when invoked in the context of other commands (e.g. list_entry)
8728  */
8729 static error_t
8730 read_globals(menu_t *mp, char *menu_path, char *globalcmd, int quiet)
8731 {
8732 	line_t *lp;
8733 	char *arg;
8734 	int done, ret = BAM_SUCCESS;
8735 
8736 	assert(mp);
8737 	assert(menu_path);
8738 	assert(globalcmd);
8739 
8740 	if (mp->start == NULL) {
8741 		if (!quiet)
8742 			bam_error(NO_MENU, menu_path);
8743 		return (BAM_ERROR);
8744 	}
8745 
8746 	done = 0;
8747 	for (lp = mp->start; lp; lp = lp->next) {
8748 		if (lp->flags != BAM_GLOBAL)
8749 			continue;
8750 
8751 		if (lp->cmd == NULL) {
8752 			if (!quiet)
8753 				bam_error(NO_CMD, lp->lineNum);
8754 			continue;
8755 		}
8756 
8757 		if (strcmp(globalcmd, lp->cmd) != 0)
8758 			continue;
8759 
8760 		/* Found global. Check for duplicates */
8761 		if (done && !quiet) {
8762 			bam_error(DUP_CMD, globalcmd, lp->lineNum, bam_root);
8763 			ret = BAM_ERROR;
8764 		}
8765 
8766 		arg = lp->arg ? lp->arg : "";
8767 		bam_print(GLOBAL_CMD, globalcmd, arg);
8768 		done = 1;
8769 	}
8770 
8771 	if (!done && bam_verbose)
8772 		bam_print(NO_ENTRY, globalcmd);
8773 
8774 	return (ret);
8775 }
8776 
8777 static error_t
8778 menu_write(char *root, menu_t *mp)
8779 {
8780 	const char *fcn = "menu_write()";
8781 
8782 	BAM_DPRINTF((D_MENU_WRITE_ENTER, fcn, root));
8783 	return (list2file(root, MENU_TMP, GRUB_MENU, mp->start));
8784 }
8785 
8786 void
8787 line_free(line_t *lp)
8788 {
8789 	if (lp == NULL)
8790 		return;
8791 
8792 	if (lp->cmd)
8793 		free(lp->cmd);
8794 	if (lp->sep)
8795 		free(lp->sep);
8796 	if (lp->arg)
8797 		free(lp->arg);
8798 	if (lp->line)
8799 		free(lp->line);
8800 	free(lp);
8801 }
8802 
8803 static void
8804 linelist_free(line_t *start)
8805 {
8806 	line_t *lp;
8807 
8808 	while (start) {
8809 		lp = start;
8810 		start = start->next;
8811 		line_free(lp);
8812 	}
8813 }
8814 
8815 static void
8816 filelist_free(filelist_t *flistp)
8817 {
8818 	linelist_free(flistp->head);
8819 	flistp->head = NULL;
8820 	flistp->tail = NULL;
8821 }
8822 
8823 static void
8824 menu_free(menu_t *mp)
8825 {
8826 	entry_t *ent, *tmp;
8827 	assert(mp);
8828 
8829 	if (mp->start)
8830 		linelist_free(mp->start);
8831 	ent = mp->entries;
8832 	while (ent) {
8833 		tmp = ent;
8834 		ent = tmp->next;
8835 		free(tmp);
8836 	}
8837 
8838 	free(mp);
8839 }
8840 
8841 /*
8842  * Utility routines
8843  */
8844 
8845 
8846 /*
8847  * Returns 0 on success
8848  * Any other value indicates an error
8849  */
8850 static int
8851 exec_cmd(char *cmdline, filelist_t *flistp)
8852 {
8853 	char buf[BUFSIZ];
8854 	int ret;
8855 	FILE *ptr;
8856 	sigset_t set;
8857 	void (*disp)(int);
8858 
8859 	/*
8860 	 * For security
8861 	 * - only absolute paths are allowed
8862 	 * - set IFS to space and tab
8863 	 */
8864 	if (*cmdline != '/') {
8865 		bam_error(ABS_PATH_REQ, cmdline);
8866 		return (-1);
8867 	}
8868 	(void) putenv("IFS= \t");
8869 
8870 	/*
8871 	 * We may have been exec'ed with SIGCHLD blocked
8872 	 * unblock it here
8873 	 */
8874 	(void) sigemptyset(&set);
8875 	(void) sigaddset(&set, SIGCHLD);
8876 	if (sigprocmask(SIG_UNBLOCK, &set, NULL) != 0) {
8877 		bam_error(CANT_UNBLOCK_SIGCHLD, strerror(errno));
8878 		return (-1);
8879 	}
8880 
8881 	/*
8882 	 * Set SIGCHLD disposition to SIG_DFL for popen/pclose
8883 	 */
8884 	disp = sigset(SIGCHLD, SIG_DFL);
8885 	if (disp == SIG_ERR) {
8886 		bam_error(FAILED_SIG, strerror(errno));
8887 		return (-1);
8888 	}
8889 	if (disp == SIG_HOLD) {
8890 		bam_error(BLOCKED_SIG, cmdline);
8891 		return (-1);
8892 	}
8893 
8894 	ptr = popen(cmdline, "r");
8895 	if (ptr == NULL) {
8896 		bam_error(POPEN_FAIL, cmdline, strerror(errno));
8897 		return (-1);
8898 	}
8899 
8900 	/*
8901 	 * If we simply do a pclose() following a popen(), pclose()
8902 	 * will close the reader end of the pipe immediately even
8903 	 * if the child process has not started/exited. pclose()
8904 	 * does wait for cmd to terminate before returning though.
8905 	 * When the executed command writes its output to the pipe
8906 	 * there is no reader process and the command dies with
8907 	 * SIGPIPE. To avoid this we read repeatedly until read
8908 	 * terminates with EOF. This indicates that the command
8909 	 * (writer) has closed the pipe and we can safely do a
8910 	 * pclose().
8911 	 *
8912 	 * Since pclose() does wait for the command to exit,
8913 	 * we can safely reap the exit status of the command
8914 	 * from the value returned by pclose()
8915 	 */
8916 	while (s_fgets(buf, sizeof (buf), ptr) != NULL) {
8917 		if (flistp == NULL) {
8918 			/* s_fgets strips newlines, so insert them at the end */
8919 			bam_print(PRINT, buf);
8920 		} else {
8921 			append_to_flist(flistp, buf);
8922 		}
8923 	}
8924 
8925 	ret = pclose(ptr);
8926 	if (ret == -1) {
8927 		bam_error(PCLOSE_FAIL, cmdline, strerror(errno));
8928 		return (-1);
8929 	}
8930 
8931 	if (WIFEXITED(ret)) {
8932 		return (WEXITSTATUS(ret));
8933 	} else {
8934 		bam_error(EXEC_FAIL, cmdline, ret);
8935 		return (-1);
8936 	}
8937 }
8938 
8939 /*
8940  * Since this function returns -1 on error
8941  * it cannot be used to convert -1. However,
8942  * that is sufficient for what we need.
8943  */
8944 static long
8945 s_strtol(char *str)
8946 {
8947 	long l;
8948 	char *res = NULL;
8949 
8950 	if (str == NULL) {
8951 		return (-1);
8952 	}
8953 
8954 	errno = 0;
8955 	l = strtol(str, &res, 10);
8956 	if (errno || *res != '\0') {
8957 		return (-1);
8958 	}
8959 
8960 	return (l);
8961 }
8962 
8963 /*
8964  * Wrapper around fputs, that adds a newline (since fputs doesn't)
8965  */
8966 static int
8967 s_fputs(char *str, FILE *fp)
8968 {
8969 	char linebuf[BAM_MAXLINE];
8970 
8971 	(void) snprintf(linebuf, sizeof (linebuf), "%s\n", str);
8972 	return (fputs(linebuf, fp));
8973 }
8974 
8975 /*
8976  * Wrapper around fgets, that strips newlines returned by fgets
8977  */
8978 char *
8979 s_fgets(char *buf, int buflen, FILE *fp)
8980 {
8981 	int n;
8982 
8983 	buf = fgets(buf, buflen, fp);
8984 	if (buf) {
8985 		n = strlen(buf);
8986 		if (n == buflen - 1 && buf[n-1] != '\n')
8987 			bam_error(TOO_LONG, buflen - 1, buf);
8988 		buf[n-1] = (buf[n-1] == '\n') ? '\0' : buf[n-1];
8989 	}
8990 
8991 	return (buf);
8992 }
8993 
8994 void *
8995 s_calloc(size_t nelem, size_t sz)
8996 {
8997 	void *ptr;
8998 
8999 	ptr = calloc(nelem, sz);
9000 	if (ptr == NULL) {
9001 		bam_error(NO_MEM, nelem*sz);
9002 		bam_exit(1);
9003 	}
9004 	return (ptr);
9005 }
9006 
9007 void *
9008 s_realloc(void *ptr, size_t sz)
9009 {
9010 	ptr = realloc(ptr, sz);
9011 	if (ptr == NULL) {
9012 		bam_error(NO_MEM, sz);
9013 		bam_exit(1);
9014 	}
9015 	return (ptr);
9016 }
9017 
9018 char *
9019 s_strdup(char *str)
9020 {
9021 	char *ptr;
9022 
9023 	if (str == NULL)
9024 		return (NULL);
9025 
9026 	ptr = strdup(str);
9027 	if (ptr == NULL) {
9028 		bam_error(NO_MEM, strlen(str) + 1);
9029 		bam_exit(1);
9030 	}
9031 	return (ptr);
9032 }
9033 
9034 /*
9035  * Returns 1 if amd64 (or sparc, for syncing x86 diskless clients)
9036  * Returns 0 otherwise
9037  */
9038 static int
9039 is_amd64(void)
9040 {
9041 	static int amd64 = -1;
9042 	char isabuf[257];	/* from sysinfo(2) manpage */
9043 
9044 	if (amd64 != -1)
9045 		return (amd64);
9046 
9047 	if (bam_alt_platform) {
9048 		if (strcmp(bam_platform, "i86pc") == 0) {
9049 			amd64 = 1;		/* diskless server */
9050 		}
9051 	} else {
9052 		if (sysinfo(SI_ISALIST, isabuf, sizeof (isabuf)) > 0 &&
9053 		    strncmp(isabuf, "amd64 ", strlen("amd64 ")) == 0) {
9054 			amd64 = 1;
9055 		} else if (strstr(isabuf, "i386") == NULL) {
9056 			amd64 = 1;		/* diskless server */
9057 		}
9058 	}
9059 	if (amd64 == -1)
9060 		amd64 = 0;
9061 
9062 	return (amd64);
9063 }
9064 
9065 static char *
9066 get_machine(void)
9067 {
9068 	static int cached = -1;
9069 	static char mbuf[257];	/* from sysinfo(2) manpage */
9070 
9071 	if (cached == 0)
9072 		return (mbuf);
9073 
9074 	if (bam_alt_platform) {
9075 		return (bam_platform);
9076 	} else {
9077 		if (sysinfo(SI_MACHINE, mbuf, sizeof (mbuf)) > 0) {
9078 			cached = 1;
9079 		}
9080 	}
9081 	if (cached == -1) {
9082 		mbuf[0] = '\0';
9083 		cached = 0;
9084 	}
9085 
9086 	return (mbuf);
9087 }
9088 
9089 int
9090 is_sparc(void)
9091 {
9092 	static int issparc = -1;
9093 	char mbuf[257];	/* from sysinfo(2) manpage */
9094 
9095 	if (issparc != -1)
9096 		return (issparc);
9097 
9098 	if (bam_alt_platform) {
9099 		if (strncmp(bam_platform, "sun4", 4) == 0) {
9100 			issparc = 1;
9101 		}
9102 	} else {
9103 		if (sysinfo(SI_ARCHITECTURE, mbuf, sizeof (mbuf)) > 0 &&
9104 		    strcmp(mbuf, "sparc") == 0) {
9105 			issparc = 1;
9106 		}
9107 	}
9108 	if (issparc == -1)
9109 		issparc = 0;
9110 
9111 	return (issparc);
9112 }
9113 
9114 static void
9115 append_to_flist(filelist_t *flistp, char *s)
9116 {
9117 	line_t *lp;
9118 
9119 	lp = s_calloc(1, sizeof (line_t));
9120 	lp->line = s_strdup(s);
9121 	if (flistp->head == NULL)
9122 		flistp->head = lp;
9123 	else
9124 		flistp->tail->next = lp;
9125 	flistp->tail = lp;
9126 }
9127 
9128 #if !defined(_OPB)
9129 
9130 UCODE_VENDORS;
9131 
9132 /*ARGSUSED*/
9133 static void
9134 ucode_install(char *root)
9135 {
9136 	int i;
9137 
9138 	for (i = 0; ucode_vendors[i].filestr != NULL; i++) {
9139 		int cmd_len = PATH_MAX + 256;
9140 		char cmd[PATH_MAX + 256];
9141 		char file[PATH_MAX];
9142 		char timestamp[PATH_MAX];
9143 		struct stat fstatus, tstatus;
9144 		struct utimbuf u_times;
9145 
9146 		(void) snprintf(file, PATH_MAX, "%s/%s/%s-ucode.%s",
9147 		    bam_root, UCODE_INSTALL_PATH, ucode_vendors[i].filestr,
9148 		    ucode_vendors[i].extstr);
9149 
9150 		if (stat(file, &fstatus) != 0 || !(S_ISREG(fstatus.st_mode)))
9151 			continue;
9152 
9153 		(void) snprintf(timestamp, PATH_MAX, "%s.ts", file);
9154 
9155 		if (stat(timestamp, &tstatus) == 0 &&
9156 		    fstatus.st_mtime <= tstatus.st_mtime)
9157 			continue;
9158 
9159 		(void) snprintf(cmd, cmd_len, "/usr/sbin/ucodeadm -i -R "
9160 		    "%s/%s/%s %s > /dev/null 2>&1", bam_root,
9161 		    UCODE_INSTALL_PATH, ucode_vendors[i].vendorstr, file);
9162 		if (system(cmd) != 0)
9163 			return;
9164 
9165 		if (creat(timestamp, S_IRUSR | S_IWUSR) == -1)
9166 			return;
9167 
9168 		u_times.actime = fstatus.st_atime;
9169 		u_times.modtime = fstatus.st_mtime;
9170 		(void) utime(timestamp, &u_times);
9171 	}
9172 }
9173 #endif
9174