xref: /freebsd/usr.sbin/bsdinstall/scripts/zfsboot (revision 695a612f671ceca4ac442e40c624fda299a114b7)
1cd88b886SDevin Teske#!/bin/sh
2cd88b886SDevin Teske#-
3cd88b886SDevin Teske# Copyright (c) 2013 Allan Jude
4cd88b886SDevin Teske# Copyright (c) 2013 Devin Teske
5cd88b886SDevin Teske# All rights reserved.
6cd88b886SDevin Teske#
7cd88b886SDevin Teske# Redistribution and use in source and binary forms, with or without
8cd88b886SDevin Teske# modification, are permitted provided that the following conditions
9cd88b886SDevin Teske# are met:
10cd88b886SDevin Teske# 1. Redistributions of source code must retain the above copyright
11cd88b886SDevin Teske#    notice, this list of conditions and the following disclaimer.
12cd88b886SDevin Teske# 2. Redistributions in binary form must reproduce the above copyright
13cd88b886SDevin Teske#    notice, this list of conditions and the following disclaimer in the
14cd88b886SDevin Teske#    documentation and/or other materials provided with the distribution.
15cd88b886SDevin Teske#
16cd88b886SDevin Teske# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
17cd88b886SDevin Teske# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18cd88b886SDevin Teske# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19cd88b886SDevin Teske# ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
20cd88b886SDevin Teske# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21cd88b886SDevin Teske# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
22cd88b886SDevin Teske# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
23cd88b886SDevin Teske# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
24cd88b886SDevin Teske# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
25cd88b886SDevin Teske# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26cd88b886SDevin Teske# SUCH DAMAGE.
27cd88b886SDevin Teske#
28cd88b886SDevin Teske# $FreeBSD$
29cd88b886SDevin Teske#
30cd88b886SDevin Teske############################################################ INCLUDES
31cd88b886SDevin Teske
32cd88b886SDevin TeskeBSDCFG_SHARE="/usr/share/bsdconfig"
33cd88b886SDevin Teske. $BSDCFG_SHARE/common.subr || exit 1
34cd88b886SDevin Teskef_dprintf "%s: loading includes..." "$0"
35cd88b886SDevin Teskef_include $BSDCFG_SHARE/device.subr
36cd88b886SDevin Teskef_include $BSDCFG_SHARE/dialog.subr
37cd88b886SDevin Teskef_include $BSDCFG_SHARE/password/password.subr
38cd88b886SDevin Teskef_include $BSDCFG_SHARE/variable.subr
39cd88b886SDevin Teske
40cd88b886SDevin Teske############################################################ CONFIGURATION
41cd88b886SDevin Teske
42cd88b886SDevin Teske#
43cd88b886SDevin Teske# Default name of the boot-pool
44cd88b886SDevin Teske#
45cd88b886SDevin Teske: ${ZFSBOOT_POOL_NAME:=zroot}
46cd88b886SDevin Teske
47cd88b886SDevin Teske#
48cd88b886SDevin Teske# Default name for the boot environment parent dataset
49cd88b886SDevin Teske#
5067635c19SDevin Teske: ${ZFSBOOT_BEROOT_NAME:=ROOT}
51cd88b886SDevin Teske
52cd88b886SDevin Teske#
53cd88b886SDevin Teske# Default name for the primany boot environment
54cd88b886SDevin Teske#
55cd88b886SDevin Teske: ${ZFSBOOT_BOOTFS_NAME:=default}
56cd88b886SDevin Teske
57cd88b886SDevin Teske#
58cd88b886SDevin Teske# Default Virtual Device (vdev) type to create
59cd88b886SDevin Teske#
60cd88b886SDevin Teske: ${ZFSBOOT_VDEV_TYPE:=stripe}
61cd88b886SDevin Teske
62cd88b886SDevin Teske#
63cd88b886SDevin Teske# Should we use gnop(8) to configure a transparent mapping to 4K sectors?
64cd88b886SDevin Teske#
65cd88b886SDevin Teske: ${ZFSBOOT_GNOP_4K_FORCE_ALIGN:=1}
66cd88b886SDevin Teske
67cd88b886SDevin Teske#
68cd88b886SDevin Teske# Should we use geli(8) to encrypt the drives?
697cae6aabSDevin Teske# NB: Automatically enables ZFSBOOT_BOOT_POOL
70cd88b886SDevin Teske#
71bc4a673fSDevin Teske: ${ZFSBOOT_GELI_ENCRYPTION=}
72cd88b886SDevin Teske
73cd88b886SDevin Teske#
74cd88b886SDevin Teske# Default path to the geli(8) keyfile used in drive encryption
75cd88b886SDevin Teske#
76cd88b886SDevin Teske: ${ZFSBOOT_GELI_KEY_FILE:=/boot/encryption.key}
77cd88b886SDevin Teske
78cd88b886SDevin Teske#
797cae6aabSDevin Teske# Create a separate boot pool?
807cae6aabSDevin Teske# NB: Automatically set when using geli(8) or MBR
817cae6aabSDevin Teske#
827cae6aabSDevin Teske: ${ZFSBOOT_BOOT_POOL=}
837cae6aabSDevin Teske
847cae6aabSDevin Teske#
857cae6aabSDevin Teske# Default name for boot pool when enabled (e.g., geli(8) or MBR)
867cae6aabSDevin Teske#
877cae6aabSDevin Teske: ${ZFSBOOT_BOOT_POOL_NAME:=bootpool}
887cae6aabSDevin Teske
897cae6aabSDevin Teske#
907cae6aabSDevin Teske# Default size for boot pool when enabled (e.g., geli(8) or MBR)
917cae6aabSDevin Teske#
927cae6aabSDevin Teske: ${ZFSBOOT_BOOT_POOL_SIZE:=2g}
937cae6aabSDevin Teske
947cae6aabSDevin Teske#
95cd88b886SDevin Teske# Default disks to use (always empty unless being scripted)
96cd88b886SDevin Teske#
97cd88b886SDevin Teske: ${ZFSBOOT_DISKS:=}
98cd88b886SDevin Teske
99cd88b886SDevin Teske#
100cd88b886SDevin Teske# Default partitioning scheme to use on disks
101cd88b886SDevin Teske#
102cd88b886SDevin Teske: ${ZFSBOOT_PARTITION_SCHEME:=GPT}
103cd88b886SDevin Teske
104cd88b886SDevin Teske#
105cd88b886SDevin Teske# How much swap to put on each block device in the boot zpool
106cd88b886SDevin Teske# NOTE: Value passed to gpart(8); which supports SI unit suffixes.
107cd88b886SDevin Teske#
108cd88b886SDevin Teske: ${ZFSBOOT_SWAP_SIZE:=2g}
109cd88b886SDevin Teske
110cd88b886SDevin Teske#
1112875e59fSOllivier Robert# Should we use geli(8) to encrypt the swap?
1122875e59fSOllivier Robert#
1132875e59fSOllivier Robert: ${ZFSBOOT_SWAP_ENCRYPTION=}
1142875e59fSOllivier Robert
1152875e59fSOllivier Robert#
1162875e59fSOllivier Robert# Should we use gmirror(8) to mirror the swap?
1172875e59fSOllivier Robert#
1182875e59fSOllivier Robert: ${ZFSBOOT_SWAP_MIRROR=}
1192875e59fSOllivier Robert
1202875e59fSOllivier Robert#
121bc4a673fSDevin Teske# Default ZFS datasets for root zpool
122cd88b886SDevin Teske#
123cd88b886SDevin Teske# NOTE: Requires /tmp, /var/tmp, /$ZFSBOOT_BOOTFS_NAME/$ZFSBOOT_BOOTFS_NAME
124cd88b886SDevin Teske# NOTE: Anything after pound/hash character [#] is ignored as a comment.
125cd88b886SDevin Teske#
126cd88b886SDevin Teskef_isset ZFSBOOT_DATASETS || ZFSBOOT_DATASETS="
127cd88b886SDevin Teske	# DATASET	OPTIONS (comma or space separated; or both)
128cd88b886SDevin Teske
129cd88b886SDevin Teske	# Boot Environment [BE] root and default boot dataset
130cd88b886SDevin Teske	/$ZFSBOOT_BEROOT_NAME				mountpoint=none
131cd88b886SDevin Teske	/$ZFSBOOT_BEROOT_NAME/$ZFSBOOT_BOOTFS_NAME	mountpoint=/
132cd88b886SDevin Teske
133cd88b886SDevin Teske	# Compress /tmp, allow exec but not setuid
134ed8690e3SOllivier Robert	/tmp		mountpoint=/tmp,exec=on,setuid=off
135cd88b886SDevin Teske
136cd88b886SDevin Teske	# Don't mount /usr so that 'base' files go to the BEROOT
137cd88b886SDevin Teske	/usr		mountpoint=/usr,canmount=off
138cd88b886SDevin Teske
139cd88b886SDevin Teske	# Home directories separated so they are common to all BEs
140bc4a673fSDevin Teske	/usr/home	# NB: /home is a symlink to /usr/home
141cd88b886SDevin Teske
142cd88b886SDevin Teske	# Ports tree
143ed8690e3SOllivier Robert	/usr/ports	setuid=off
144cd88b886SDevin Teske
145cd88b886SDevin Teske	# Source tree (compressed)
146ed8690e3SOllivier Robert	/usr/src
147cd88b886SDevin Teske
148cd88b886SDevin Teske	# Create /var and friends
149cd88b886SDevin Teske	/var		mountpoint=/var
150ed8690e3SOllivier Robert	/var/crash	exec=off,setuid=off
151ed8690e3SOllivier Robert	/var/log	exec=off,setuid=off
152ed8690e3SOllivier Robert	/var/mail	atime=on
153ed8690e3SOllivier Robert	/var/tmp	setuid=off
154cd88b886SDevin Teske" # END-QUOTE
155cd88b886SDevin Teske
156bc4a673fSDevin Teske#
157bc4a673fSDevin Teske# If interactive and the user has not explicitly chosen a vdev type or disks,
158bc4a673fSDevin Teske# make the user confirm scripted/default choices when proceeding to install.
159bc4a673fSDevin Teske#
160bc4a673fSDevin Teske: ${ZFSBOOT_CONFIRM_LAYOUT:=1}
161bc4a673fSDevin Teske
162cd88b886SDevin Teske############################################################ GLOBALS
163cd88b886SDevin Teske
164cd88b886SDevin Teske#
165bc4a673fSDevin Teske# Format of a line in printf(1) syntax to add to fstab(5)
166bc4a673fSDevin Teske#
167bc4a673fSDevin TeskeFSTAB_FMT="%s\t\t%s\t%s\t%s\t\t%s\t%s\n"
168bc4a673fSDevin Teske
169bc4a673fSDevin Teske#
170bc4a673fSDevin Teske# Command strings for various tasks
171bc4a673fSDevin Teske#
172bc4a673fSDevin TeskeCHMOD_MODE='chmod %s "%s"'
173bc4a673fSDevin TeskeDD_WITH_OPTIONS='dd if="%s" of="%s" %s'
174bc4a673fSDevin TeskeECHO_APPEND='echo "%s" >> "%s"'
175bc4a673fSDevin TeskeGELI_ATTACH='geli attach -j - -k "%s" "%s"'
176a622223fSDevin TeskeGELI_DETACH_F='geli detach -f "%s"'
177bc4a673fSDevin TeskeGELI_PASSWORD_INIT='geli init -b -B "%s" -e %s -J - -K "%s" -l 256 -s 4096 "%s"'
178bc4a673fSDevin TeskeGNOP_CREATE='gnop create -S 4096 "%s"'
179a622223fSDevin TeskeGNOP_DESTROY='gnop destroy "%s"'
180bc4a673fSDevin TeskeGPART_ADD='gpart add -t %s "%s"'
181bc4a673fSDevin TeskeGPART_ADD_INDEX='gpart add -i %s -t %s "%s"'
182bc4a673fSDevin TeskeGPART_ADD_INDEX_WITH_SIZE='gpart add -i %s -t %s -s %s "%s"'
183bc4a673fSDevin TeskeGPART_ADD_LABEL='gpart add -l %s -t %s "%s"'
184bc4a673fSDevin TeskeGPART_ADD_LABEL_WITH_SIZE='gpart add -l %s -t %s -s %s "%s"'
185bc4a673fSDevin TeskeGPART_BOOTCODE='gpart bootcode -b "%s" "%s"'
186bc4a673fSDevin TeskeGPART_BOOTCODE_PART='gpart bootcode -b "%s" -p "%s" -i %s "%s"'
187bc4a673fSDevin TeskeGPART_CREATE='gpart create -s %s "%s"'
188a622223fSDevin TeskeGPART_DESTROY_F='gpart destroy -F "%s"'
189bc4a673fSDevin TeskeGPART_SET_ACTIVE='gpart set -a active -i %s "%s"'
190a622223fSDevin TeskeGRAID_DELETE='graid delete "%s"'
191bc4a673fSDevin TeskeLN_SF='ln -sf "%s" "%s"'
192bc4a673fSDevin TeskeMKDIR_P='mkdir -p "%s"'
193bc4a673fSDevin TeskeMOUNT_TYPE='mount -t %s "%s" "%s"'
194bc4a673fSDevin TeskePRINTF_CONF="printf '%s=\"%%s\"\\\n' %s >> \"%s\""
195bc4a673fSDevin TeskePRINTF_FSTAB='printf "$FSTAB_FMT" "%s" "%s" "%s" "%s" "%s" "%s" >> "%s"'
196bc4a673fSDevin TeskeSHELL_TRUNCATE=':> "%s"'
1972875e59fSOllivier RobertSWAP_GMIRROR_LABEL='gmirror label swap %s'
198a622223fSDevin TeskeUMOUNT='umount "%s"'
199bc4a673fSDevin TeskeZFS_CREATE_WITH_OPTIONS='zfs create %s "%s"'
200bc4a673fSDevin TeskeZFS_SET='zfs set "%s" "%s"'
201bc4a673fSDevin TeskeZFS_UNMOUNT='zfs unmount "%s"'
202bc4a673fSDevin TeskeZPOOL_CREATE_WITH_OPTIONS='zpool create %s "%s" %s %s'
203a88393ceSDevin TeskeZPOOL_DESTROY='zpool destroy "%s"'
204bc4a673fSDevin TeskeZPOOL_EXPORT='zpool export "%s"'
205bc4a673fSDevin TeskeZPOOL_IMPORT_WITH_OPTIONS='zpool import %s "%s"'
206a622223fSDevin TeskeZPOOL_LABELCLEAR_F='zpool labelclear -f "%s"'
207bc4a673fSDevin TeskeZPOOL_SET='zpool set %s "%s"'
208bc4a673fSDevin Teske
209bc4a673fSDevin Teske#
210cd88b886SDevin Teske# Strings that should be moved to an i18n file and loaded with f_include_lang()
211cd88b886SDevin Teske#
212cd88b886SDevin Teskehline_alnum_arrows_punc_tab_enter="Use alnum, arrows, punctuation, TAB or ENTER"
213cd88b886SDevin Teskehline_arrows_space_tab_enter="Use arrows, SPACE, TAB or ENTER"
214cd88b886SDevin Teskehline_arrows_tab_enter="Press arrows, TAB or ENTER"
215bc4a673fSDevin Teskemsg_an_unknown_error_occurred="An unknown error occurred"
216cd88b886SDevin Teskemsg_back="Back"
217cd88b886SDevin Teskemsg_cancel="Cancel"
218bc4a673fSDevin Teskemsg_change_selection="Change Selection"
219cd88b886SDevin Teskemsg_configure_options="Configure Options:"
220cd88b886SDevin Teskemsg_detailed_disk_info="gpart(8) show %s:\n%s\n\ncamcontrol(8) inquiry %s:\n%s\n\n\ncamcontrol(8) identify %s:\n%s\n"
221cd88b886SDevin Teskemsg_disk_info="Disk Info"
222cd88b886SDevin Teskemsg_disk_info_help="Get detailed information on disk device(s)"
223*695a612fSOllivier Robertmsg_disk_singular="disk"
224*695a612fSOllivier Robertmsg_disk_plural="disks"
225bc4a673fSDevin Teskemsg_encrypt_disks="Encrypt Disks?"
226bc4a673fSDevin Teskemsg_encrypt_disks_help="Use geli(8) to encrypt all data partitions"
227bc4a673fSDevin Teskemsg_error="Error"
228cd88b886SDevin Teskemsg_force_4k_sectors="Force 4K Sectors?"
229cd88b886SDevin Teskemsg_force_4k_sectors_help="Use gnop(8) to configure forced 4K sector alignment"
230cd88b886SDevin Teskemsg_freebsd_installer="FreeBSD Installer"
231cd88b886SDevin Teskemsg_geli_password="Enter a strong passphrase, used to protect your encryption keys. You will be required to enter this passphrase each time the system is booted"
2327a434c5cSDevin Teskemsg_geli_setup="Initializing encryption on selected disks,\n this will take several seconds per disk"
233bc4a673fSDevin Teskemsg_install="Install"
234bc4a673fSDevin Teskemsg_install_desc="Proceed with Installation"
235bc4a673fSDevin Teskemsg_install_help="Create ZFS boot pool with displayed options"
2367cae6aabSDevin Teskemsg_invalid_boot_pool_size="Invalid boot pool size \`%s'"
237bc4a673fSDevin Teskemsg_invalid_disk_argument="Invalid disk argument \`%s'"
238bc4a673fSDevin Teskemsg_invalid_index_argument="Invalid index argument \`%s'"
239bc4a673fSDevin Teskemsg_invalid_swap_size="Invalid swap size \`%s'"
240cd88b886SDevin Teskemsg_invalid_virtual_device_type="Invalid Virtual Device type \`%s'"
241bc4a673fSDevin Teskemsg_last_chance_are_you_sure="Last Chance! Are you sure you want to destroy\nthe current contents of the following disks:\n\n   %s"
242bc4a673fSDevin Teskemsg_last_chance_are_you_sure_color='\\ZrLast Chance!\\ZR Are you \\Z1sure\\Zn you want to \\Zr\\Z1destroy\\Zn\nthe current contents of the following disks:\n\n   %s'
243cd88b886SDevin Teskemsg_mirror_desc="Mirror - n-Way Mirroring"
244cd88b886SDevin Teskemsg_mirror_help="[2+ Disks] Mirroring provides the best performance, but the least storage"
245bc4a673fSDevin Teskemsg_missing_disk_arguments="missing disk arguments"
246bc4a673fSDevin Teskemsg_missing_one_or_more_scripted_disks="Missing one or more scripted disks!"
247cd88b886SDevin Teskemsg_no="NO"
248cd88b886SDevin Teskemsg_no_disks_present_to_configure="No disk(s) present to configure"
249cd88b886SDevin Teskemsg_no_disks_selected="No disks selected."
250bc4a673fSDevin Teskemsg_not_enough_disks_selected="Not enough disks selected. (%u < %u minimum)"
251bc4a673fSDevin Teskemsg_null_disk_argument="NULL disk argument"
252bc4a673fSDevin Teskemsg_null_index_argument="NULL index argument"
253bc4a673fSDevin Teskemsg_null_poolname="NULL poolname"
254cd88b886SDevin Teskemsg_ok="OK"
255cd88b886SDevin Teskemsg_partition_scheme="Partition Scheme"
256cd88b886SDevin Teskemsg_partition_scheme_help="Toggle between GPT and MBR partitioning schemes"
257cd88b886SDevin Teskemsg_please_enter_a_name_for_your_zpool="Please enter a name for your zpool:"
258cd88b886SDevin Teskemsg_please_enter_amount_of_swap_space="Please enter amount of swap space (SI-Unit suffixes\nrecommended; e.g., \`2g' for 2 Gigabytes):"
259cd88b886SDevin Teskemsg_please_select_one_or_more_disks="Please select one or more disks to create a zpool:"
260cd88b886SDevin Teskemsg_pool_name="Pool Name"
261cd88b886SDevin Teskemsg_pool_name_cannot_be_empty="Pool name cannot be empty."
262cd88b886SDevin Teskemsg_pool_name_help="Customize the name of the zpool to be created (Required)"
263bc4a673fSDevin Teskemsg_pool_type_disks="Pool Type/Disks:"
264bc4a673fSDevin Teskemsg_pool_type_disks_help="Choose type of ZFS Virtual Device and disks to use (Required)"
265cd88b886SDevin Teskemsg_processing_selection="Processing selection..."
266cd88b886SDevin Teskemsg_raidz1_desc="RAID-Z1 - Single Redundant RAID"
267cd88b886SDevin Teskemsg_raidz1_help="[3+ Disks] Withstand failure of 1 disk. Recommended for: 3, 5 or 9 disks"
268cd88b886SDevin Teskemsg_raidz2_desc="RAID-Z2 - Double Redundant RAID"
269cd88b886SDevin Teskemsg_raidz2_help="[4+ Disks] Withstand failure of 2 disks. Recommended for: 4, 6 or 10 disks"
270cd88b886SDevin Teskemsg_raidz3_desc="RAID-Z3 - Triple Redundant RAID"
271cd88b886SDevin Teskemsg_raidz3_help="[5+ Disks] Withstand failure of 3 disks. Recommended for: 5, 7 or 11 disks"
272cd88b886SDevin Teskemsg_rescan_devices="Rescan Devices"
273cd88b886SDevin Teskemsg_rescan_devices_help="Scan for device changes"
274cd88b886SDevin Teskemsg_select="Select"
275cd88b886SDevin Teskemsg_select_a_disk_device="Select a disk device"
276cd88b886SDevin Teskemsg_select_virtual_device_type="Select Virtual Device type:"
277cd88b886SDevin Teskemsg_stripe_desc="Stripe - No Redundancy"
278cd88b886SDevin Teskemsg_stripe_help="[1+ Disks] Striping provides maximum storage but no redundancy"
2792875e59fSOllivier Robertmsg_swap_encrypt="Encrypt Swap?"
2802875e59fSOllivier Robertmsg_swap_encrypt_help="Encrypt swap partitions with temporary keys, discarded on reboot"
2812875e59fSOllivier Robertmsg_swap_mirror="Mirror Swap?"
2822875e59fSOllivier Robertmsg_swap_mirror_help="Mirror swap partitions for redundancy, breaks crash dumps"
283cd88b886SDevin Teskemsg_swap_size="Swap Size"
284cd88b886SDevin Teskemsg_swap_size_help="Customize how much swap space is allocated to each selected disk"
285bc4a673fSDevin Teskemsg_these_disks_are_too_small="These disks are too small given the amount of requested\nswap (%s) and/or geli(8) (%s) partitions, which would\ntake 50%% or more of each of the following selected disk\ndevices (not recommended):\n\n  %s\n\nRecommend changing partition size(s) and/or selecting a\ndifferent set of devices."
286bc4a673fSDevin Teskemsg_unable_to_get_disk_capacity="Unable to get disk capacity of \`%s'"
287bc4a673fSDevin Teskemsg_unsupported_partition_scheme="%s is an unsupported partition scheme"
288bc4a673fSDevin Teskemsg_user_cancelled="User Cancelled."
289cd88b886SDevin Teskemsg_yes="YES"
290cd88b886SDevin Teskemsg_zfs_configuration="ZFS Configuration"
291cd88b886SDevin Teske
292cd88b886SDevin Teske############################################################ FUNCTIONS
293cd88b886SDevin Teske
294cd88b886SDevin Teske# dialog_menu_main
295cd88b886SDevin Teske#
296cd88b886SDevin Teske# Display the dialog(1)-based application main menu.
297cd88b886SDevin Teske#
298cd88b886SDevin Teskedialog_menu_main()
299cd88b886SDevin Teske{
300cd88b886SDevin Teske	local title="$DIALOG_TITLE"
301cd88b886SDevin Teske	local btitle="$DIALOG_BACKTITLE"
302cd88b886SDevin Teske	local prompt="$msg_configure_options"
303cd88b886SDevin Teske	local force4k="$msg_no"
304cd88b886SDevin Teske	local usegeli="$msg_no"
3052875e59fSOllivier Robert	local swapgeli="$msg_no"
3062875e59fSOllivier Robert	local swapmirror="$msg_no"
307cd88b886SDevin Teske	[ "$ZFSBOOT_GNOP_4K_FORCE_ALIGN" ] && force4k="$msg_yes"
308cd88b886SDevin Teske	[ "$ZFSBOOT_GELI_ENCRYPTION" ] && usegeli="$msg_yes"
3092875e59fSOllivier Robert	[ "$ZFSBOOT_SWAP_ENCRYPTION" ] && swapgeli="$msg_yes"
3102875e59fSOllivier Robert	[ "$ZFSBOOT_SWAP_MIRROR" ] && swapmirror="$msg_yes"
311*695a612fSOllivier Robert	local disks n disks_grammar
312a88393ceSDevin Teske	f_count n $ZFSBOOT_DISKS
313*695a612fSOllivier Robert	{ [ $n -eq 1 ] && disks_grammar=$msg_disk_singular; } ||
314*695a612fSOllivier Robert		disks_grammar=$msg_disk_plural # grammar
315cd88b886SDevin Teske	local menu_list="
316bc4a673fSDevin Teske		'>>> $msg_install'        '$msg_install_desc'
317bc4a673fSDevin Teske		                          '$msg_install_help'
318*695a612fSOllivier Robert		'T $msg_pool_type_disks'  '$ZFSBOOT_VDEV_TYPE: $n $disks_grammar'
319bc4a673fSDevin Teske                                          '$msg_pool_type_disks_help'
320cd88b886SDevin Teske		'- $msg_rescan_devices'   '*'
321cd88b886SDevin Teske		                          '$msg_rescan_devices_help'
322cd88b886SDevin Teske		'- $msg_disk_info'        '*'
323cd88b886SDevin Teske		                          '$msg_disk_info_help'
324bc4a673fSDevin Teske		'N $msg_pool_name'        '$ZFSBOOT_POOL_NAME'
325cd88b886SDevin Teske		                          '$msg_pool_name_help'
326cd88b886SDevin Teske		'4 $msg_force_4k_sectors' '$force4k'
327cd88b886SDevin Teske		                          '$msg_force_4k_sectors_help'
328bc4a673fSDevin Teske		'E $msg_encrypt_disks'    '$usegeli'
329bc4a673fSDevin Teske		                          '$msg_encrypt_disks_help'
330bc4a673fSDevin Teske		'P $msg_partition_scheme' '$ZFSBOOT_PARTITION_SCHEME'
331cd88b886SDevin Teske		                          '$msg_partition_scheme_help'
332bc4a673fSDevin Teske		'S $msg_swap_size'        '$ZFSBOOT_SWAP_SIZE'
333cd88b886SDevin Teske		                          '$msg_swap_size_help'
3342875e59fSOllivier Robert		'M $msg_swap_mirror'      '$swapmirror'
3352875e59fSOllivier Robert		                          '$msg_swap_mirror_help'
3362875e59fSOllivier Robert		'W $msg_swap_encrypt'     '$swapgeli'
3372875e59fSOllivier Robert		                          '$msg_swap_encrypt_help'
338cd88b886SDevin Teske	" # END-QUOTE
339cd88b886SDevin Teske	local defaultitem= # Calculated below
340cd88b886SDevin Teske	local hline="$hline_alnum_arrows_punc_tab_enter"
341cd88b886SDevin Teske
342cd88b886SDevin Teske	local height width rows
343cd88b886SDevin Teske	eval f_dialog_menu_with_help_size height width rows \
344cd88b886SDevin Teske		\"\$title\" \"\$btitle\" \"\$prompt\" \"\$hline\" $menu_list
345cd88b886SDevin Teske
346cd88b886SDevin Teske	# Obtain default-item from previously stored selection
347cd88b886SDevin Teske	f_dialog_default_fetch defaultitem
348cd88b886SDevin Teske
349cd88b886SDevin Teske	local menu_choice
350cd88b886SDevin Teske	menu_choice=$( eval $DIALOG \
351cd88b886SDevin Teske		--title \"\$title\"              \
352cd88b886SDevin Teske		--backtitle \"\$btitle\"         \
353cd88b886SDevin Teske		--hline \"\$hline\"              \
354cd88b886SDevin Teske		--item-help                      \
355cd88b886SDevin Teske		--ok-label \"\$msg_select\"      \
356cd88b886SDevin Teske		--cancel-label \"\$msg_cancel\"  \
357cd88b886SDevin Teske		--default-item \"\$defaultitem\" \
358cd88b886SDevin Teske		--menu \"\$prompt\"              \
359cd88b886SDevin Teske		$height $width $rows             \
360cd88b886SDevin Teske		$menu_list                       \
361cd88b886SDevin Teske		2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
362cd88b886SDevin Teske	)
363cd88b886SDevin Teske	local retval=$?
364cd88b886SDevin Teske	f_dialog_data_sanitize menu_choice
365cd88b886SDevin Teske	f_dialog_menutag_store "$menu_choice"
366cd88b886SDevin Teske
367cd88b886SDevin Teske	# Only update default-item on success
368cd88b886SDevin Teske	[ $retval -eq $DIALOG_OK ] && f_dialog_default_store "$menu_choice"
369cd88b886SDevin Teske
370cd88b886SDevin Teske	return $retval
371cd88b886SDevin Teske}
372cd88b886SDevin Teske
373bc4a673fSDevin Teske# dialog_last_chance $disks ...
374cd88b886SDevin Teske#
375bc4a673fSDevin Teske# Display a list of the disks that the user is about to destroy. The default
376bc4a673fSDevin Teske# action is to return error status unless the user explicitly (non-default)
377bc4a673fSDevin Teske# selects "Yes" from the noyes dialog.
378cd88b886SDevin Teske#
379bc4a673fSDevin Teskedialog_last_chance()
380cd88b886SDevin Teske{
381cd88b886SDevin Teske	local title="$DIALOG_TITLE"
382cd88b886SDevin Teske	local btitle="$DIALOG_BACKTITLE"
383bc4a673fSDevin Teske	local prompt # Calculated below
384bc4a673fSDevin Teske	local hline="$hline_arrows_tab_enter"
385cd88b886SDevin Teske
386bc4a673fSDevin Teske	local height=8 width=50 prefix="   "
387bc4a673fSDevin Teske	local plen=${#prefix} list= line=
388bc4a673fSDevin Teske	local max_width=$(( $width - 3 - $plen ))
389bc4a673fSDevin Teske
390bc4a673fSDevin Teske	local yes no defaultno extra_args format
391bc4a673fSDevin Teske	if [ "$USE_XDIALOG" ]; then
392bc4a673fSDevin Teske		yes=ok no=cancel defaultno=default-no
393bc4a673fSDevin Teske		extra_args="--wrap --left"
394bc4a673fSDevin Teske		format="$msg_last_chance_are_you_sure"
395bc4a673fSDevin Teske	else
396bc4a673fSDevin Teske		yes=yes no=no defaultno=defaultno
397bc4a673fSDevin Teske		extra_args="--colors --cr-wrap"
398bc4a673fSDevin Teske		format="$msg_last_chance_are_you_sure_color"
399bc4a673fSDevin Teske	fi
400bc4a673fSDevin Teske
401bc4a673fSDevin Teske	local disk line_width
402bc4a673fSDevin Teske	for disk in $*; do
403bc4a673fSDevin Teske		if [ "$line" ]; then
404bc4a673fSDevin Teske			line_width=${#line}
405bc4a673fSDevin Teske		else
406bc4a673fSDevin Teske			line_width=$plen
407bc4a673fSDevin Teske		fi
408bc4a673fSDevin Teske		line_width=$(( $line_width + 1 + ${#disk} ))
409bc4a673fSDevin Teske		# Add newline before disk if it would exceed max_width
410bc4a673fSDevin Teske		if [ $line_width -gt $max_width ]; then
411bc4a673fSDevin Teske			list="$list$line\n"
412bc4a673fSDevin Teske			line="$prefix"
413bc4a673fSDevin Teske			height=$(( $height + 1 ))
414bc4a673fSDevin Teske		fi
415bc4a673fSDevin Teske		# Add the disk to the list
416bc4a673fSDevin Teske		line="$line $disk"
417bc4a673fSDevin Teske	done
418bc4a673fSDevin Teske	# Append the left-overs
419bc4a673fSDevin Teske	if [ "${line#$prefix}" ]; then
420bc4a673fSDevin Teske		list="$list$line"
421bc4a673fSDevin Teske		height=$(( $height + 1 ))
422bc4a673fSDevin Teske	fi
423bc4a673fSDevin Teske
424bc4a673fSDevin Teske	# Add height for Xdialog(1)
425bc4a673fSDevin Teske	[ "$USE_XDIALOG" ] && height=$(( $height + $height / 5 + 3 ))
426bc4a673fSDevin Teske
427bc4a673fSDevin Teske	prompt=$( printf "$format" "$list" )
428bc4a673fSDevin Teske	f_dprintf "%s: Last Chance!" "$0"
429bc4a673fSDevin Teske	$DIALOG \
430bc4a673fSDevin Teske		--title "$title"        \
431bc4a673fSDevin Teske		--backtitle "$btitle"   \
432bc4a673fSDevin Teske		--hline "$hline"        \
433bc4a673fSDevin Teske		--$defaultno            \
434bc4a673fSDevin Teske		--$yes-label "$msg_yes" \
435bc4a673fSDevin Teske		--$no-label "$msg_no"   \
436bc4a673fSDevin Teske		$extra_args             \
437bc4a673fSDevin Teske		--yesno "$prompt" $height $width
438bc4a673fSDevin Teske}
439bc4a673fSDevin Teske
440bc4a673fSDevin Teske# dialog_menu_layout
441cd88b886SDevin Teske#
442bc4a673fSDevin Teske# Configure Virtual Device type and disks to use for the ZFS boot pool. User
443bc4a673fSDevin Teske# must select enough disks to satisfy the chosen vdev type.
444cd88b886SDevin Teske#
445bc4a673fSDevin Teskedialog_menu_layout()
446bc4a673fSDevin Teske{
447bc4a673fSDevin Teske	local funcname=dialog_menu_layout
448bc4a673fSDevin Teske	local title="$DIALOG_TITLE"
449bc4a673fSDevin Teske	local btitle="$DIALOG_BACKTITLE"
450bc4a673fSDevin Teske	local vdev_prompt="$msg_select_virtual_device_type"
451bc4a673fSDevin Teske	local disk_prompt="$msg_please_select_one_or_more_disks"
452bc4a673fSDevin Teske	local vdev_menu_list="
453bc4a673fSDevin Teske		'stripe' '$msg_stripe_desc' '$msg_stripe_help'
454bc4a673fSDevin Teske		'mirror' '$msg_mirror_desc' '$msg_mirror_help'
455bc4a673fSDevin Teske		'raidz1' '$msg_raidz1_desc' '$msg_raidz1_help'
456bc4a673fSDevin Teske		'raidz2' '$msg_raidz2_desc' '$msg_raidz2_help'
457bc4a673fSDevin Teske		'raidz3' '$msg_raidz3_desc' '$msg_raidz3_help'
458bc4a673fSDevin Teske	" # END-QUOTE
459bc4a673fSDevin Teske	local disk_check_list= # Calculated below
460bc4a673fSDevin Teske	local vdev_hline="$hline_arrows_tab_enter"
461bc4a673fSDevin Teske	local disk_hline="$hline_arrows_space_tab_enter"
462bc4a673fSDevin Teske
463bc4a673fSDevin Teske	# Warn the user if vdev type is not valid
464bc4a673fSDevin Teske	case "$ZFSBOOT_VDEV_TYPE" in
465bc4a673fSDevin Teske	stripe|mirror|raidz1|raidz2|raidz3) : known good ;;
466bc4a673fSDevin Teske	*)
467bc4a673fSDevin Teske		f_dprintf "%s: Invalid virtual device type \`%s'" \
468bc4a673fSDevin Teske			  $funcname "$ZFSBOOT_VDEV_TYPE"
469bc4a673fSDevin Teske		f_show_err "$msg_invalid_virtual_device_type" \
470bc4a673fSDevin Teske			   "$ZFSBOOT_VDEV_TYPE"
471bc4a673fSDevin Teske		f_interactive || return $FAILURE
472bc4a673fSDevin Teske	esac
473bc4a673fSDevin Teske
474bc4a673fSDevin Teske	# Calculate size of vdev menu once only
475bc4a673fSDevin Teske	local vheight vwidth vrows
476bc4a673fSDevin Teske	eval f_dialog_menu_with_help_size vheight vwidth vrows \
477bc4a673fSDevin Teske		\"\$title\" \"\$btitle\" \"\$vdev_prompt\" \"\$vdev_hline\" \
478bc4a673fSDevin Teske		$vdev_menu_list
479bc4a673fSDevin Teske
480bc4a673fSDevin Teske	# Get a list of probed disk devices
481bc4a673fSDevin Teske	local disks=
482a88393ceSDevin Teske	debug= f_device_find "" $DEVICE_TYPE_DISK disks
483a88393ceSDevin Teske
484a88393ceSDevin Teske	# Prune out mounted md(4) devices that may be part of the boot process
485a88393ceSDevin Teske	local disk name new_list=
486a88393ceSDevin Teske	for disk in $disks; do
487a88393ceSDevin Teske		debug= $disk get name name
488a88393ceSDevin Teske		case "$name" in
489a88393ceSDevin Teske		md[0-9]*) f_mounted -b "/dev/$name" && continue ;;
490a88393ceSDevin Teske		esac
491a88393ceSDevin Teske		new_list="$new_list $disk"
492a88393ceSDevin Teske	done
493a88393ceSDevin Teske	disks="${new_list# }"
494a88393ceSDevin Teske
495a88393ceSDevin Teske	# Debugging
496a88393ceSDevin Teske	if [ "$debug" ]; then
497a88393ceSDevin Teske		local disk_names=
498a88393ceSDevin Teske		for disk in $disks; do
499a88393ceSDevin Teske			debug= $disk get name name
500a88393ceSDevin Teske			disk_names="$disk_names $name"
501a88393ceSDevin Teske		done
502a88393ceSDevin Teske		f_dprintf "$funcname: disks=[%s]" "${disk_names# }"
503a88393ceSDevin Teske	fi
504a88393ceSDevin Teske
505cd88b886SDevin Teske	if [ ! "$disks" ]; then
506bc4a673fSDevin Teske		f_dprintf "No disk(s) present to configure"
507bc4a673fSDevin Teske		f_show_err "$msg_no_disks_present_to_configure"
508cd88b886SDevin Teske		return $FAILURE
509cd88b886SDevin Teske	fi
510cd88b886SDevin Teske
511cd88b886SDevin Teske	# Lets sort the disks array to be more user friendly
512a88393ceSDevin Teske	f_device_sort_by name disks disks
513cd88b886SDevin Teske
514cd88b886SDevin Teske	#
515bc4a673fSDevin Teske	# Operate in a loop so we can (if interactive) repeat if not enough
516bc4a673fSDevin Teske	# disks are selected to satisfy the chosen vdev type or user wants to
517bc4a673fSDevin Teske	# back-up to the previous menu.
518cd88b886SDevin Teske	#
519a88393ceSDevin Teske	local vardisk ndisks onoff selections vdev_choice breakout device
520a88393ceSDevin Teske	local valid_disks all_valid want_disks desc height width rows
521bc4a673fSDevin Teske	while :; do
522cd88b886SDevin Teske		#
523bc4a673fSDevin Teske		# Confirm the vdev type that was selected
524cd88b886SDevin Teske		#
525bc4a673fSDevin Teske		if f_interactive && [ "$ZFSBOOT_CONFIRM_LAYOUT" ]; then
526bc4a673fSDevin Teske			vdev_choice=$( eval $DIALOG \
527cd88b886SDevin Teske				--title \"\$title\"              \
528cd88b886SDevin Teske				--backtitle \"\$btitle\"         \
529bc4a673fSDevin Teske				--hline \"\$vdev_hline\"         \
530cd88b886SDevin Teske				--ok-label \"\$msg_ok\"          \
531cd88b886SDevin Teske				--cancel-label \"\$msg_cancel\"  \
532cd88b886SDevin Teske				--item-help                      \
533bc4a673fSDevin Teske				--default-item \"\$ZFSBOOT_VDEV_TYPE\" \
534bc4a673fSDevin Teske				--menu \"\$vdev_prompt\"         \
535bc4a673fSDevin Teske				$vheight $vwidth $vrows          \
536bc4a673fSDevin Teske				$vdev_menu_list                  \
537cd88b886SDevin Teske				2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
538bc4a673fSDevin Teske			) || return $?
539bc4a673fSDevin Teske				# Exit if user pressed ESC or chose Cancel/No
540bc4a673fSDevin Teske			f_dialog_data_sanitize vdev_choice
541cd88b886SDevin Teske
542bc4a673fSDevin Teske			ZFSBOOT_VDEV_TYPE="$vdev_choice"
543bc4a673fSDevin Teske			f_dprintf "$funcname: ZFSBOOT_VDEV_TYPE=[%s]" \
544bc4a673fSDevin Teske			          "$ZFSBOOT_VDEV_TYPE"
545bc4a673fSDevin Teske		fi
546bc4a673fSDevin Teske
547bc4a673fSDevin Teske		# Determine the number of disks needed for this vdev type
548a88393ceSDevin Teske		want_disks=0
549bc4a673fSDevin Teske		case "$ZFSBOOT_VDEV_TYPE" in
550cd88b886SDevin Teske		stripe) want_disks=1 ;;
551cd88b886SDevin Teske		mirror) want_disks=2 ;;
552cd88b886SDevin Teske		raidz1) want_disks=3 ;;
553cd88b886SDevin Teske		raidz2) want_disks=4 ;;
554cd88b886SDevin Teske		raidz3) want_disks=5 ;;
555cd88b886SDevin Teske		esac
556bc4a673fSDevin Teske
557a88393ceSDevin Teske		#
558bc4a673fSDevin Teske		# Warn the user if any scripted disks are invalid
559a88393ceSDevin Teske		#
560a88393ceSDevin Teske		valid_disks= all_valid=${ZFSBOOT_DISKS:+1} # optimism
561bc4a673fSDevin Teske		for disk in $ZFSBOOT_DISKS; do
562a88393ceSDevin Teske			if debug= f_device_find -1 \
563a88393ceSDevin Teske				$disk $DEVICE_TYPE_DISK device
564a88393ceSDevin Teske			then
565bc4a673fSDevin Teske				valid_disks="$valid_disks $disk"
566bc4a673fSDevin Teske				continue
567cd88b886SDevin Teske			fi
568bc4a673fSDevin Teske			f_dprintf "$funcname: \`%s' is not a real disk" "$disk"
569bc4a673fSDevin Teske			all_valid=
570bc4a673fSDevin Teske		done
571bc4a673fSDevin Teske		if [ ! "$all_valid" ]; then
572bc4a673fSDevin Teske			if [ "$ZFSBOOT_DISKS" ]; then
573bc4a673fSDevin Teske				f_show_err \
574bc4a673fSDevin Teske				    "$msg_missing_one_or_more_scripted_disks"
575bc4a673fSDevin Teske			else
576bc4a673fSDevin Teske				f_dprintf "No disks selected."
577bc4a673fSDevin Teske				f_interactive ||
578bc4a673fSDevin Teske					f_show_err "$msg_no_disks_selected"
579bc4a673fSDevin Teske			fi
580bc4a673fSDevin Teske			f_interactive || return $FAILURE
581bc4a673fSDevin Teske		fi
582bc4a673fSDevin Teske		ZFSBOOT_DISKS="${valid_disks# }"
583bc4a673fSDevin Teske
584bc4a673fSDevin Teske		#
585bc4a673fSDevin Teske		# Short-circuit if we're running non-interactively
586bc4a673fSDevin Teske		#
587bc4a673fSDevin Teske		if ! f_interactive || [ ! "$ZFSBOOT_CONFIRM_LAYOUT" ]; then
588a88393ceSDevin Teske			f_count ndisks $ZFSBOOT_DISKS
589bc4a673fSDevin Teske			[ $ndisks -ge $want_disks ] && break # to success
590bc4a673fSDevin Teske
591bc4a673fSDevin Teske			# Not enough disks selected
592bc4a673fSDevin Teske			f_dprintf "$funcname: %s: %s (%u < %u minimum)" \
593bc4a673fSDevin Teske				  "$ZFSBOOT_VDEV_TYPE" \
594bc4a673fSDevin Teske			          "Not enough disks selected." \
595bc4a673fSDevin Teske				  $ndisks $want_disks
596bc4a673fSDevin Teske			f_interactive || return $FAILURE
597bc4a673fSDevin Teske			msg_yes="$msg_change_selection" msg_no="$msg_cancel" \
598bc4a673fSDevin Teske				f_yesno "%s: $msg_not_enough_disks_selected" \
599bc4a673fSDevin Teske				"$ZFSBOOT_VDEV_TYPE" $ndisks $want_disks ||
600bc4a673fSDevin Teske				return $FAILURE
601bc4a673fSDevin Teske		fi
602bc4a673fSDevin Teske
603bc4a673fSDevin Teske		#
604bc4a673fSDevin Teske		# Confirm the disks that were selected
605bc4a673fSDevin Teske		# Loop until the user cancels or selects enough disks
606bc4a673fSDevin Teske		#
607a88393ceSDevin Teske		breakout=
608bc4a673fSDevin Teske		while :; do
609bc4a673fSDevin Teske			# Loop over list of available disks, resetting state
610a88393ceSDevin Teske			for disk in $disks; do
611a88393ceSDevin Teske				f_isset _${disk}_status && _${disk}_status=
612a88393ceSDevin Teske			done
613bc4a673fSDevin Teske
614bc4a673fSDevin Teske			# Loop over list of selected disks and create temporary
615bc4a673fSDevin Teske			# locals to map statuses onto up-to-date list of disks
616bc4a673fSDevin Teske			for disk in $ZFSBOOT_DISKS; do
617a88393ceSDevin Teske				debug= f_device_find -1 \
618a88393ceSDevin Teske					$disk $DEVICE_TYPE_DISK disk
619a88393ceSDevin Teske				f_isset _${disk}_status ||
620a88393ceSDevin Teske					local _${disk}_status
621a88393ceSDevin Teske				_${disk}_status=on
622bc4a673fSDevin Teske			done
623bc4a673fSDevin Teske
624bc4a673fSDevin Teske			# Create the checklist menu of discovered disk devices
625bc4a673fSDevin Teske			disk_check_list=
626bc4a673fSDevin Teske			for disk in $disks; do
627a88393ceSDevin Teske				desc=
628a88393ceSDevin Teske				$disk get name name
629a88393ceSDevin Teske				$disk get desc desc
630bc4a673fSDevin Teske				f_shell_escape "$desc" desc
631bc4a673fSDevin Teske				f_getvar _${disk}_status:-off onoff
632bc4a673fSDevin Teske				disk_check_list="$disk_check_list
633a88393ceSDevin Teske					$name '$desc' $onoff"
634bc4a673fSDevin Teske			done
635bc4a673fSDevin Teske
636bc4a673fSDevin Teske			eval f_dialog_checklist_size height width rows \
637bc4a673fSDevin Teske				\"\$title\" \"\$btitle\" \"\$prompt\" \
638bc4a673fSDevin Teske				\"\$hline\" $disk_check_list
639bc4a673fSDevin Teske
640bc4a673fSDevin Teske			selections=$( eval $DIALOG \
641bc4a673fSDevin Teske				--title \"\$DIALOG_TITLE\"         \
642bc4a673fSDevin Teske				--backtitle \"\$DIALOG_BACKTITLE\" \
643751952aeSDevin Teske				--separate-output                  \
644bc4a673fSDevin Teske				--hline \"\$hline\"                \
645bc4a673fSDevin Teske				--ok-label \"\$msg_ok\"            \
646bc4a673fSDevin Teske				--cancel-label \"\$msg_back\"      \
647bc4a673fSDevin Teske				--checklist \"\$prompt\"           \
648bc4a673fSDevin Teske				$height $width $rows               \
649bc4a673fSDevin Teske				$disk_check_list                   \
650bc4a673fSDevin Teske				2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
651bc4a673fSDevin Teske			) || break
652bc4a673fSDevin Teske				# Loop if user pressed ESC or chose Cancel/No
653bc4a673fSDevin Teske			f_dialog_data_sanitize selections
654bc4a673fSDevin Teske
655bc4a673fSDevin Teske			ZFSBOOT_DISKS="$selections"
656bc4a673fSDevin Teske			f_dprintf "$funcname: ZFSBOOT_DISKS=[%s]" \
657bc4a673fSDevin Teske			          "$ZFSBOOT_DISKS"
658bc4a673fSDevin Teske
659a88393ceSDevin Teske			f_count ndisks $ZFSBOOT_DISKS
660bc4a673fSDevin Teske			[ $ndisks -ge $want_disks ] &&
661bc4a673fSDevin Teske				breakout=break && break
662bc4a673fSDevin Teske
663bc4a673fSDevin Teske			# Not enough disks selected
664bc4a673fSDevin Teske			f_dprintf "$funcname: %s: %s (%u < %u minimum)" \
665bc4a673fSDevin Teske				  "$ZFSBOOT_VDEV_TYPE" \
666bc4a673fSDevin Teske			          "Not enough disks selected." \
667bc4a673fSDevin Teske			          $ndisks $want_disks
668bc4a673fSDevin Teske			msg_yes="$msg_change_selection" msg_no="$msg_cancel" \
669bc4a673fSDevin Teske				f_yesno "%s: $msg_not_enough_disks_selected" \
670bc4a673fSDevin Teske				"$ZFSBOOT_VDEV_TYPE" $ndisks $want_disks ||
671bc4a673fSDevin Teske				break
672bc4a673fSDevin Teske		done
673bc4a673fSDevin Teske		[ "$breakout" = "break" ] && break
674bc4a673fSDevin Teske		[ "$ZFSBOOT_CONFIRM_LAYOUT" ] || return $FAILURE
675bc4a673fSDevin Teske	done
676bc4a673fSDevin Teske
677bc4a673fSDevin Teske	return $DIALOG_OK
678cd88b886SDevin Teske}
679cd88b886SDevin Teske
680cd88b886SDevin Teske# zfs_create_diskpart $disk $index
681cd88b886SDevin Teske#
682cd88b886SDevin Teske# For each block device to be used in the zpool, rather than just create the
683cd88b886SDevin Teske# zpool with the raw block devices (e.g., da0, da1, etc.) we create partitions
684cd88b886SDevin Teske# so we can have some real swap. This also provides wiggle room incase your
685cd88b886SDevin Teske# replacement drivers do not have the exact same sector counts.
686cd88b886SDevin Teske#
6877cae6aabSDevin Teske# NOTE: $swapsize and $bootsize should be defined by the calling function.
688a622223fSDevin Teske# NOTE: Sets $bootpart and $targetpart for the calling function.
689cd88b886SDevin Teske#
690cd88b886SDevin Teskezfs_create_diskpart()
691cd88b886SDevin Teske{
692cd88b886SDevin Teske	local funcname=zfs_create_diskpart
693bc4a673fSDevin Teske	local disk="$1" index="$2"
694cd88b886SDevin Teske
695cd88b886SDevin Teske	# Check arguments
696bc4a673fSDevin Teske	if [ ! "$disk" ]; then
697bc4a673fSDevin Teske		f_dprintf "$funcname: NULL disk argument"
698bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
699bc4a673fSDevin Teske			f_show_err "$msg_null_disk_argument"
700bc4a673fSDevin Teske		return $FAILURE
701bc4a673fSDevin Teske	fi
702bc4a673fSDevin Teske	if [ "${disk#*[$IFS]}" != "$disk" ]; then
703bc4a673fSDevin Teske		f_dprintf "$funcname: Invalid disk argument \`%s'" "$disk"
704bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
705bc4a673fSDevin Teske			f_show_err "$msg_invalid_disk_argument" "$disk"
706bc4a673fSDevin Teske		return $FAILURE
707bc4a673fSDevin Teske	fi
708bc4a673fSDevin Teske	if [ ! "$index" ]; then
709bc4a673fSDevin Teske		f_dprintf "$funcname: NULL index argument"
710bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
711bc4a673fSDevin Teske			f_show_err "$msg_null_index_argument"
712bc4a673fSDevin Teske		return $FAILURE
713bc4a673fSDevin Teske	fi
714bc4a673fSDevin Teske	if ! f_isinteger "$index"; then
715bc4a673fSDevin Teske		f_dprintf "$funcname: Invalid index argument \`%s'" "$index"
716bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
717bc4a673fSDevin Teske			f_show_err "$msg_invalid_index_argument" "$index"
718bc4a673fSDevin Teske		return $FAILURE
719bc4a673fSDevin Teske	fi
720bc4a673fSDevin Teske	f_dprintf "$funcname: disk=[%s] index=[%s]" "$disk" "$index"
721cd88b886SDevin Teske
722d4d729e4SDevin Teske	# Check for unknown partition scheme before proceeding further
723d4d729e4SDevin Teske	case "$ZFSBOOT_PARTITION_SCHEME" in
724d4d729e4SDevin Teske	""|MBR|GPT) : known good ;;
725d4d729e4SDevin Teske	*)
726d4d729e4SDevin Teske		f_dprintf "$funcname: %s is an unsupported partition scheme" \
727d4d729e4SDevin Teske		          "$ZFSBOOT_PARTITION_SCHEME"
728d4d729e4SDevin Teske		msg_error="$msg_error: $funcname" f_show_err \
729d4d729e4SDevin Teske			"$msg_unsupported_partition_scheme" \
730d4d729e4SDevin Teske			"$ZFSBOOT_PARTITION_SCHEME"
731d4d729e4SDevin Teske		return $FAILURE
732d4d729e4SDevin Teske	esac
733d4d729e4SDevin Teske
734cd88b886SDevin Teske	#
735cd88b886SDevin Teske	# Destroy whatever partition layout is currently on disk.
736cd88b886SDevin Teske	# NOTE: `-F' required to destroy if partitions still exist.
737cd88b886SDevin Teske	# NOTE: Failure is ok here, blank disk will have nothing to destroy.
738cd88b886SDevin Teske	#
739bc4a673fSDevin Teske	f_dprintf "$funcname: Destroying all data/layouts on \`%s'..." "$disk"
740a622223fSDevin Teske	f_eval_catch -d $funcname gpart "$GPART_DESTROY_F" $disk
741a622223fSDevin Teske	f_eval_catch -d $funcname graid "$GRAID_DELETE" $disk
742a622223fSDevin Teske	f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" /dev/$disk
743cd88b886SDevin Teske
744cd88b886SDevin Teske	# Make doubly-sure backup GPT is destroyed
745a622223fSDevin Teske	f_eval_catch -d $funcname gpart "$GPART_CREATE" gpt $disk
746a622223fSDevin Teske	f_eval_catch -d $funcname gpart "$GPART_DESTROY_F" $disk
747cd88b886SDevin Teske
7487cae6aabSDevin Teske	#
7497cae6aabSDevin Teske	# Enable boot pool if encryption is desired
7507cae6aabSDevin Teske	#
7517cae6aabSDevin Teske	[ "$ZFSBOOT_GELI_ENCRYPTION" ] && ZFSBOOT_BOOT_POOL=1
752cd88b886SDevin Teske
753cd88b886SDevin Teske	#
754cd88b886SDevin Teske	# Lay down the desired type of partition scheme
755cd88b886SDevin Teske	#
756cd88b886SDevin Teske	local setsize mbrindex
757cd88b886SDevin Teske	case "$ZFSBOOT_PARTITION_SCHEME" in
758bc4a673fSDevin Teske	""|GPT) f_dprintf "$funcname: Creating GPT layout..."
759cd88b886SDevin Teske		#
760cd88b886SDevin Teske		# 1. Create GPT layout using labels
761cd88b886SDevin Teske		#
762a622223fSDevin Teske		f_eval_catch $funcname gpart "$GPART_CREATE" gpt $disk ||
763bc4a673fSDevin Teske		             return $FAILURE
764cd88b886SDevin Teske
765cd88b886SDevin Teske		#
766cd88b886SDevin Teske		# 2. Add small freebsd-boot partition labeled `boot#'
767cd88b886SDevin Teske		#
768bc4a673fSDevin Teske		f_eval_catch $funcname gpart "$GPART_ADD_LABEL_WITH_SIZE" \
769a622223fSDevin Teske		             gptboot$index freebsd-boot 512k $disk ||
770cd88b886SDevin Teske		             return $FAILURE
771bc4a673fSDevin Teske		f_eval_catch $funcname gpart "$GPART_BOOTCODE_PART" \
772a622223fSDevin Teske		             /boot/pmbr /boot/gptzfsboot 1 $disk ||
773cd88b886SDevin Teske		             return $FAILURE
774cd88b886SDevin Teske
775a622223fSDevin Teske		# NB: zpool will use the `zfs#' GPT labels
7762875e59fSOllivier Robert		bootpart=p2 swappart=p2 targetpart=p2
7777cae6aabSDevin Teske		[ ${swapsize:-0} -gt 0 ] && targetpart=p3
778cd88b886SDevin Teske
7797cae6aabSDevin Teske		#
7807cae6aabSDevin Teske		# Prepare boot pool if enabled (e.g., for geli(8))
7817cae6aabSDevin Teske		#
7827cae6aabSDevin Teske		if [ "$ZFSBOOT_BOOT_POOL" ]; then
7832875e59fSOllivier Robert			bootpart=p2 swappart=p3 targetpart=p3
7847cae6aabSDevin Teske			[ ${swapsize:-0} -gt 0 ] && targetpart=p4
785bc4a673fSDevin Teske			f_eval_catch $funcname gpart \
786a622223fSDevin Teske			             "$GPART_ADD_LABEL_WITH_SIZE" boot$index \
7877cae6aabSDevin Teske			             freebsd-zfs ${bootsize}b $disk ||
788bc4a673fSDevin Teske			             return $FAILURE
7897cae6aabSDevin Teske			# Pedantically nuke any old labels
790a622223fSDevin Teske			f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
791a622223fSDevin Teske			                /dev/$disk$bootpart
7927cae6aabSDevin Teske			if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
7937cae6aabSDevin Teske				# Pedantically detach targetpart for later
7947cae6aabSDevin Teske				f_eval_catch -d $funcname geli \
7957cae6aabSDevin Teske				                "$GELI_DETACH_F" \
796a622223fSDevin Teske				                /dev/$disk$targetpart
797cd88b886SDevin Teske			fi
7987cae6aabSDevin Teske		fi
799cd88b886SDevin Teske
800cd88b886SDevin Teske		#
8017cae6aabSDevin Teske		# 3. Add freebsd-swap partition labeled `swap#'
802cd88b886SDevin Teske		#
803bc4a673fSDevin Teske		if [ ${swapsize:-0} -gt 0 ]; then
804bc4a673fSDevin Teske			f_eval_catch $funcname gpart \
8057cae6aabSDevin Teske			             "$GPART_ADD_LABEL_WITH_SIZE" swap$index \
8067cae6aabSDevin Teske			             freebsd-swap ${swapsize}b $disk ||
807cd88b886SDevin Teske			             return $FAILURE
808cf9c3e56SDevin Teske			# Pedantically nuke any old labels on the swap
809cf9c3e56SDevin Teske			f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
8102875e59fSOllivier Robert			                /dev/$disk$swappart
811cd88b886SDevin Teske		fi
8127cae6aabSDevin Teske
8137cae6aabSDevin Teske		#
8147cae6aabSDevin Teske		# 4. Add freebsd-zfs partition labeled `zfs#' for zroot
8157cae6aabSDevin Teske		#
8167cae6aabSDevin Teske		f_eval_catch $funcname gpart "$GPART_ADD_LABEL" \
8177cae6aabSDevin Teske		             zfs$index freebsd-zfs $disk || return $FAILURE
8187cae6aabSDevin Teske		f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
8197cae6aabSDevin Teske		                /dev/$disk$targetpart
820cd88b886SDevin Teske		;;
821cd88b886SDevin Teske
822bc4a673fSDevin Teske	MBR) f_dprintf "$funcname: Creating MBR layout..."
823cd88b886SDevin Teske		#
824cd88b886SDevin Teske		# 1. Create MBR layout (no labels)
825cd88b886SDevin Teske		#
826a622223fSDevin Teske		f_eval_catch $funcname gpart "$GPART_CREATE" mbr $disk ||
827bc4a673fSDevin Teske		             return $FAILURE
8282925848dSDevin Teske		f_eval_catch $funcname gpart "$GPART_BOOTCODE" /boot/mbr \
829a622223fSDevin Teske		             $disk || return $FAILURE
830cd88b886SDevin Teske
831cd88b886SDevin Teske		#
832cd88b886SDevin Teske		# 2. Add freebsd slice with all available space
833cd88b886SDevin Teske		#
834a622223fSDevin Teske		f_eval_catch $funcname gpart "$GPART_ADD" freebsd $disk ||
835bc4a673fSDevin Teske		             return $FAILURE
836a622223fSDevin Teske		f_eval_catch $funcname gpart "$GPART_SET_ACTIVE" 1 $disk ||
837bc4a673fSDevin Teske		             return $FAILURE
838a622223fSDevin Teske		# Pedantically nuke any old labels
839a622223fSDevin Teske		f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
840a622223fSDevin Teske		                /dev/${disk}s1
841a622223fSDevin Teske		# Pedantically nuke any old scheme
842a622223fSDevin Teske		f_eval_catch -d $funcname gpart "$GPART_DESTROY_F" ${disk}s1
843cd88b886SDevin Teske
844cd88b886SDevin Teske		#
845bc4a673fSDevin Teske		# 3. Write BSD scheme to the freebsd slice
846cd88b886SDevin Teske		#
847a622223fSDevin Teske		f_eval_catch $funcname gpart "$GPART_CREATE" BSD ${disk}s1 ||
848bc4a673fSDevin Teske		             return $FAILURE
849cd88b886SDevin Teske
8507cae6aabSDevin Teske		# NB: zpool will use s1a (no labels)
8512875e59fSOllivier Robert		bootpart=s1a swappart=s1b targetpart=s1d mbrindex=4
852cd88b886SDevin Teske
8537cae6aabSDevin Teske		#
8547cae6aabSDevin Teske		# Always prepare a boot pool on MBR
8557cae6aabSDevin Teske		#
8567cae6aabSDevin Teske		ZFSBOOT_BOOT_POOL=1
857bc4a673fSDevin Teske		f_eval_catch $funcname gpart \
858bc4a673fSDevin Teske		             "$GPART_ADD_INDEX_WITH_SIZE" \
8597cae6aabSDevin Teske		             1 freebsd-zfs ${bootsize}b ${disk}s1 ||
860bc4a673fSDevin Teske		             return $FAILURE
861a622223fSDevin Teske		# Pedantically nuke any old labels
862a622223fSDevin Teske		f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
863a622223fSDevin Teske		                /dev/$disk$bootpart
8647cae6aabSDevin Teske		if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
865a622223fSDevin Teske			# Pedantically detach targetpart for later
866a622223fSDevin Teske			f_eval_catch -d $funcname geli \
867a622223fSDevin Teske			                "$GELI_DETACH_F" \
868a622223fSDevin Teske					/dev/$disk$targetpart
869cd88b886SDevin Teske		fi
870cd88b886SDevin Teske
871cd88b886SDevin Teske		#
8727cae6aabSDevin Teske		# 4. Add freebsd-swap partition
873cd88b886SDevin Teske		#
874bc4a673fSDevin Teske		if [ ${swapsize:-0} -gt 0 ]; then
875bc4a673fSDevin Teske			f_eval_catch $funcname gpart \
8767cae6aabSDevin Teske			             "$GPART_ADD_INDEX_WITH_SIZE" 2 \
8777cae6aabSDevin Teske			             freebsd-swap ${swapsize}b ${disk}s1 ||
878cd88b886SDevin Teske			             return $FAILURE
879cf9c3e56SDevin Teske			# Pedantically nuke any old labels on the swap
880cf9c3e56SDevin Teske			f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
881cf9c3e56SDevin Teske			                /dev/${disk}s1b
882cd88b886SDevin Teske		fi
8837cae6aabSDevin Teske
8847cae6aabSDevin Teske		#
8857cae6aabSDevin Teske		# 5. Add freebsd-zfs partition for zroot
8867cae6aabSDevin Teske		#
8877cae6aabSDevin Teske		f_eval_catch $funcname gpart "$GPART_ADD_INDEX" \
8887cae6aabSDevin Teske			     $mbrindex freebsd-zfs ${disk}s1 || return $FAILURE
8897cae6aabSDevin Teske		f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
8907cae6aabSDevin Teske				/dev/$disk$targetpart # Pedantic
8917cae6aabSDevin Teske		f_eval_catch $funcname dd "$DD_WITH_OPTIONS" \
8927cae6aabSDevin Teske			     /boot/zfsboot /dev/${disk}s1 count=1 ||
8937cae6aabSDevin Teske			     return $FAILURE
894cd88b886SDevin Teske		;;
895cd88b886SDevin Teske
896cd88b886SDevin Teske	esac # $ZFSBOOT_PARTITION_SCHEME
897cd88b886SDevin Teske
8982875e59fSOllivier Robert	# Update fstab(5)
8992875e59fSOllivier Robert	if [ "$isswapmirror" ]; then
9002875e59fSOllivier Robert		# This is not the first disk in the mirror, do nothing
9012875e59fSOllivier Robert	elif [ "$ZFSBOOT_SWAP_ENCRYPTION" -a "$ZFSBOOT_SWAP_MIRROR" ]; then
9022875e59fSOllivier Robert		f_eval_catch $funcname printf "$PRINTF_FSTAB" \
9032875e59fSOllivier Robert		             /dev/mirror/swap.eli none swap sw 0 0 \
9042875e59fSOllivier Robert		             $BSDINSTALL_TMPETC/fstab ||
9052875e59fSOllivier Robert		             return $FAILURE
9062875e59fSOllivier Robert		isswapmirror=1
9072875e59fSOllivier Robert	elif [ "$ZFSBOOT_SWAP_MIRROR" ]; then
9082875e59fSOllivier Robert		f_eval_catch $funcname printf "$PRINTF_FSTAB" \
9092875e59fSOllivier Robert		             /dev/mirror/swap none swap sw 0 0 \
9102875e59fSOllivier Robert		             $BSDINSTALL_TMPETC/fstab ||
9112875e59fSOllivier Robert		             return $FAILURE
9122875e59fSOllivier Robert		isswapmirror=1
9132875e59fSOllivier Robert	elif [ "$ZFSBOOT_SWAP_ENCRYPTION" ]; then
9142875e59fSOllivier Robert		f_eval_catch $funcname printf "$PRINTF_FSTAB" \
9152875e59fSOllivier Robert		             /dev/$disk${swappart}.eli none swap sw 0 0 \
9162875e59fSOllivier Robert		             $BSDINSTALL_TMPETC/fstab ||
9172875e59fSOllivier Robert		             return $FAILURE
9182875e59fSOllivier Robert	else
9192875e59fSOllivier Robert		f_eval_catch $funcname printf "$PRINTF_FSTAB" \
9202875e59fSOllivier Robert		             /dev/$disk$swappart none swap sw 0 0 \
9212875e59fSOllivier Robert		             $BSDINSTALL_TMPETC/fstab ||
9222875e59fSOllivier Robert		             return $FAILURE
9232875e59fSOllivier Robert	fi
9242875e59fSOllivier Robert
925cd88b886SDevin Teske	return $SUCCESS
926cd88b886SDevin Teske}
927cd88b886SDevin Teske
928bc4a673fSDevin Teske# zfs_create_boot $poolname $vdev_type $disks ...
929cd88b886SDevin Teske#
930cd88b886SDevin Teske# Creates boot pool and dataset layout. Returns error if something goes wrong.
931cd88b886SDevin Teske# Errors are printed to stderr for collection and display.
932cd88b886SDevin Teske#
933cd88b886SDevin Teskezfs_create_boot()
934cd88b886SDevin Teske{
935cd88b886SDevin Teske	local funcname=zfs_create_boot
9367cae6aabSDevin Teske	local zroot_name="$1"
9377cae6aabSDevin Teske	local zroot_vdevtype="$2"
9387cae6aabSDevin Teske	local zroot_vdevs= # Calculated below
9392875e59fSOllivier Robert	local swap_devs= # Calculated below
9407cae6aabSDevin Teske	local boot_vdevs= # Used for geli(8) and/or MBR layouts
941a622223fSDevin Teske	shift 2 # poolname vdev_type
942a622223fSDevin Teske	local disks="$*" disk
9432875e59fSOllivier Robert	local isswapmirror
9442875e59fSOllivier Robert	local bootpart targetpart swappart # Set by zfs_create_diskpart() below
945cd88b886SDevin Teske
946a622223fSDevin Teske	#
947cd88b886SDevin Teske	# Pedantic checks; should never be seen
948a622223fSDevin Teske	#
9497cae6aabSDevin Teske	if [ ! "$zroot_name" ]; then
950bc4a673fSDevin Teske		f_dprintf "$funcname: NULL poolname"
951bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
952bc4a673fSDevin Teske			f_show_err "$msg_null_poolname"
953cd88b886SDevin Teske		return $FAILURE
954cd88b886SDevin Teske	fi
955cd88b886SDevin Teske	if [ $# -lt 1 ]; then
956bc4a673fSDevin Teske		f_dprintf "$funcname: missing disk arguments"
957bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
958bc4a673fSDevin Teske			f_show_err "$msg_missing_disk_arguments"
959cd88b886SDevin Teske		return $FAILURE
960cd88b886SDevin Teske	fi
961bc4a673fSDevin Teske	f_dprintf "$funcname: poolname=[%s] vdev_type=[%s]" \
9627cae6aabSDevin Teske	          "$zroot_name" "$zroot_vdevtype"
963cd88b886SDevin Teske
964a622223fSDevin Teske	#
965cd88b886SDevin Teske	# Initialize fstab(5)
966a622223fSDevin Teske	#
967bc4a673fSDevin Teske	f_dprintf "$funcname: Initializing temporary fstab(5) file..."
968a622223fSDevin Teske	f_eval_catch $funcname sh "$SHELL_TRUNCATE" $BSDINSTALL_TMPETC/fstab ||
969bc4a673fSDevin Teske	             return $FAILURE
970bc4a673fSDevin Teske	f_eval_catch $funcname printf "$PRINTF_FSTAB" \
971cd88b886SDevin Teske	             "# Device" Mountpoint FStype Options Dump "Pass#" \
972a622223fSDevin Teske	             $BSDINSTALL_TMPETC/fstab || return $FAILURE
973cd88b886SDevin Teske
974a622223fSDevin Teske	#
975cd88b886SDevin Teske	# Expand SI units in desired sizes
976a622223fSDevin Teske	#
9777cae6aabSDevin Teske	f_dprintf "$funcname: Expanding supplied size values..."
9787cae6aabSDevin Teske	local swapsize bootsize
979bc4a673fSDevin Teske	if ! f_expand_number "$ZFSBOOT_SWAP_SIZE" swapsize; then
980bc4a673fSDevin Teske		f_dprintf "$funcname: Invalid swap size \`%s'" \
981bc4a673fSDevin Teske		          "$ZFSBOOT_SWAP_SIZE"
982bc4a673fSDevin Teske		f_show_err "$msg_invalid_swap_size" "$ZFSBOOT_SWAP_SIZE"
983bc4a673fSDevin Teske		return $FAILURE
984bc4a673fSDevin Teske	fi
9857cae6aabSDevin Teske	if ! f_expand_number "$ZFSBOOT_BOOT_POOL_SIZE" bootsize; then
9867cae6aabSDevin Teske		f_dprintf "$funcname: Invalid boot pool size \`%s'" \
9877cae6aabSDevin Teske		          "$ZFSBOOT_BOOT_POOL_SIZE"
9887cae6aabSDevin Teske		f_show_err "$msg_invalid_boot_pool_size" \
9897cae6aabSDevin Teske		           "$ZFSBOOT_BOOT_POOL_SIZE"
990bc4a673fSDevin Teske		return $FAILURE
991bc4a673fSDevin Teske	fi
992bc4a673fSDevin Teske	f_dprintf "$funcname: ZFSBOOT_SWAP_SIZE=[%s] swapsize=[%s]" \
993bc4a673fSDevin Teske	          "$ZFSBOOT_SWAP_SIZE" "$swapsize"
9947cae6aabSDevin Teske	f_dprintf "$funcname: ZFSBOOT_BOOT_POOL_SIZE=[%s] bootsize=[%s]" \
9957cae6aabSDevin Teske	          "$ZFSBOOT_BOOT_POOL_SIZE" "$bootsize"
996cd88b886SDevin Teske
997a622223fSDevin Teske	#
998a88393ceSDevin Teske	# Destroy the pool in-case this is our second time 'round (case of
999a88393ceSDevin Teske	# failure and installer presented ``Retry'' option to come back).
1000a88393ceSDevin Teske	#
1001a88393ceSDevin Teske	# NB: If we don't destroy the pool, later gpart(8) destroy commands
1002a88393ceSDevin Teske	# that try to clear existing partitions (see zfs_create_diskpart())
1003a88393ceSDevin Teske	# will fail with a `Device Busy' error, leading to `GEOM exists'.
1004a88393ceSDevin Teske	#
1005a88393ceSDevin Teske	f_eval_catch -d $funcname zpool "$ZPOOL_DESTROY" "$zroot_name"
1006a88393ceSDevin Teske
1007a88393ceSDevin Teske	#
10087cae6aabSDevin Teske	# Prepare the disks and build pool device list(s)
1009a622223fSDevin Teske	#
1010bc4a673fSDevin Teske	f_dprintf "$funcname: Preparing disk partitions for ZFS pool..."
10117cae6aabSDevin Teske	[ "$ZFSBOOT_GNOP_4K_FORCE_ALIGN" ] &&
10127cae6aabSDevin Teske		f_dprintf "$funcname: With 4k alignment using gnop(8)..."
1013cd88b886SDevin Teske	local n=0
1014a622223fSDevin Teske	for disk in $disks; do
1015cd88b886SDevin Teske		zfs_create_diskpart $disk $n || return $FAILURE
10162875e59fSOllivier Robert		# Now $bootpart, $targetpart, and $swappart are set (suffix
10172875e59fSOllivier Robert		# for $disk)
1018cd88b886SDevin Teske
10197cae6aabSDevin Teske		# Forced 4k alignment support using Geom NOP (see gnop(8))
1020cd88b886SDevin Teske		if [ "$ZFSBOOT_GNOP_4K_FORCE_ALIGN" ]; then
10217cae6aabSDevin Teske			if [ "$ZFSBOOT_BOOT_POOL" ]; then
10227cae6aabSDevin Teske				boot_vdevs="$boot_vdevs $disk$bootpart.nop"
1023bc4a673fSDevin Teske				f_eval_catch $funcname gnop "$GNOP_CREATE" \
10247cae6aabSDevin Teske				             $disk$bootpart || return $FAILURE
1025cd88b886SDevin Teske			fi
10267cae6aabSDevin Teske			# Don't gnop encrypted partition
10277cae6aabSDevin Teske			if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
10287cae6aabSDevin Teske				zroot_vdevs="$zroot_vdevs $disk$targetpart.eli"
1029cd88b886SDevin Teske			else
10307cae6aabSDevin Teske				zroot_vdevs="$zroot_vdevs $disk$targetpart.nop"
10317cae6aabSDevin Teske				f_eval_catch $funcname gnop "$GNOP_CREATE" \
10327cae6aabSDevin Teske					     $disk$targetpart ||
10337cae6aabSDevin Teske				             return $FAILURE
10347cae6aabSDevin Teske			fi
10357cae6aabSDevin Teske		else
10367cae6aabSDevin Teske			if [ "$ZFSBOOT_BOOT_POOL" ]; then
10377cae6aabSDevin Teske				boot_vdevs="$boot_vdevs $disk$bootpart"
10387cae6aabSDevin Teske			fi
10397cae6aabSDevin Teske			zroot_vdevs="$zroot_vdevs $disk$targetpart"
10407cae6aabSDevin Teske		fi
10417cae6aabSDevin Teske
10427cae6aabSDevin Teske		n=$(( $n + 1 ))
10437cae6aabSDevin Teske	done # disks
10447cae6aabSDevin Teske
10457cae6aabSDevin Teske	#
10467cae6aabSDevin Teske	# If we need/want a boot pool, create it
10477cae6aabSDevin Teske	#
10487cae6aabSDevin Teske	if [ "$ZFSBOOT_BOOT_POOL" ]; then
10497cae6aabSDevin Teske		local bootpool_vdevtype= # Calculated below
10507cae6aabSDevin Teske		local bootpool_options= # Calculated below
10517cae6aabSDevin Teske		local bootpool_name="$ZFSBOOT_BOOT_POOL_NAME"
10527cae6aabSDevin Teske		local bootpool="$BSDINSTALL_CHROOT/$bootpool_name"
10537cae6aabSDevin Teske		local zroot_key="${ZFSBOOT_GELI_KEY_FILE#/}"
10547cae6aabSDevin Teske
10557cae6aabSDevin Teske		f_dprintf "$funcname: Setting up boot pool..."
1056cd88b886SDevin Teske		[ "$ZFSBOOT_GELI_ENCRYPTION" ] &&
10577cae6aabSDevin Teske			f_dprintf "$funcname: For encrypted root disk..."
1058cd88b886SDevin Teske
10597cae6aabSDevin Teske		# Create parent directory for boot pool
1060a622223fSDevin Teske		f_eval_catch -d $funcname umount "$UMOUNT" /mnt
1061bc4a673fSDevin Teske		f_eval_catch $funcname mount "$MOUNT_TYPE" tmpfs none \
1062a622223fSDevin Teske		             $BSDINSTALL_CHROOT || return $FAILURE
1063cd88b886SDevin Teske
10647cae6aabSDevin Teske		# Create mirror across the boot partition on all disks
1065a88393ceSDevin Teske		local nvdevs
1066a88393ceSDevin Teske		f_count nvdevs $boot_vdevs
1067a88393ceSDevin Teske		[ $nvdevs -gt 1 ] && bootpool_vdevtype=mirror
1068cd88b886SDevin Teske
10697cae6aabSDevin Teske		bootpool_options="-o altroot=$BSDINSTALL_CHROOT"
10707cae6aabSDevin Teske		bootpool_options="$bootpool_options -m \"/$bootpool_name\" -f"
1071bc4a673fSDevin Teske		f_eval_catch $funcname zpool "$ZPOOL_CREATE_WITH_OPTIONS" \
10727cae6aabSDevin Teske		             "$bootpool_options" "$bootpool_name" \
10734b4b90d5SDevin Teske		             "$bootpool_vdevtype" "$boot_vdevs" ||
10744b4b90d5SDevin Teske		             return $FAILURE
107545df402eSDevin Teske
10767cae6aabSDevin Teske		f_eval_catch $funcname mkdir "$MKDIR_P" "$bootpool/boot" ||
1077cd88b886SDevin Teske		             return $FAILURE
1078cd88b886SDevin Teske
10797cae6aabSDevin Teske		if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
1080cd88b886SDevin Teske			# Generate an encryption key using random(4)
1081bc4a673fSDevin Teske			f_eval_catch $funcname dd "$DD_WITH_OPTIONS" \
10827cae6aabSDevin Teske				     /dev/random "$bootpool/$zroot_key" \
10837cae6aabSDevin Teske			             "bs=4096 count=1" || return $FAILURE
10847cae6aabSDevin Teske		else
10857cae6aabSDevin Teske			# Clean up
10867cae6aabSDevin Teske			f_eval_catch $funcname zfs "$ZFS_UNMOUNT" \
10877cae6aabSDevin Teske			             "$bootpool_name" || return $FAILURE
10887cae6aabSDevin Teske			f_eval_catch -d $funcname umount "$UMOUNT" /mnt # tmpfs
10897cae6aabSDevin Teske		fi
10907cae6aabSDevin Teske
10917cae6aabSDevin Teske	fi
1092cd88b886SDevin Teske
1093a622223fSDevin Teske	#
1094cd88b886SDevin Teske	# Create the geli(8) GEOMS
1095a622223fSDevin Teske	#
10967cae6aabSDevin Teske	if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
10977cae6aabSDevin Teske		# Prompt user for password (twice)
1098bc4a673fSDevin Teske		if ! msg_enter_new_password="$msg_geli_password" \
1099bc4a673fSDevin Teske			f_dialog_input_password
1100bc4a673fSDevin Teske		then
1101bc4a673fSDevin Teske			f_dprintf "$funcname: User cancelled"
1102bc4a673fSDevin Teske			f_show_err "$msg_user_cancelled"
1103bc4a673fSDevin Teske			return $FAILURE
1104bc4a673fSDevin Teske		fi
1105a622223fSDevin Teske
1106a622223fSDevin Teske		# Initialize geli(8) on each of the target partitions
1107bc4a673fSDevin Teske		for disk in $disks; do
1108cd88b886SDevin Teske			f_dialog_info "$msg_geli_setup" \
1109cd88b886SDevin Teske				2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
1110a622223fSDevin Teske			if ! echo "$pw_password" | f_eval_catch \
1111a622223fSDevin Teske				$funcname geli "$GELI_PASSWORD_INIT" \
11127cae6aabSDevin Teske				"$bootpool/boot/$disk$targetpart.eli" \
11137cae6aabSDevin Teske				AES-XTS "$bootpool/$zroot_key" \
1114a622223fSDevin Teske				$disk$targetpart
1115bc4a673fSDevin Teske			then
1116bc4a673fSDevin Teske				f_interactive || f_die
1117a622223fSDevin Teske				unset pw_password # Sensitive info
1118cd88b886SDevin Teske				return $FAILURE
1119bc4a673fSDevin Teske			fi
1120a622223fSDevin Teske			if ! echo "$pw_password" | f_eval_catch \
1121a622223fSDevin Teske				$funcname geli "$GELI_ATTACH" \
11227cae6aabSDevin Teske				"$bootpool/$zroot_key" $disk$targetpart
1123bc4a673fSDevin Teske			then
1124bc4a673fSDevin Teske				f_interactive || f_die
1125a622223fSDevin Teske				unset pw_password # Sensitive info
1126bc4a673fSDevin Teske				return $FAILURE
1127bc4a673fSDevin Teske			fi
1128cd88b886SDevin Teske		done
1129a622223fSDevin Teske		unset pw_password # Sensitive info
1130a622223fSDevin Teske
1131a622223fSDevin Teske		# Clean up
11327cae6aabSDevin Teske		f_eval_catch $funcname zfs "$ZFS_UNMOUNT" "$bootpool_name" ||
11337cae6aabSDevin Teske			return $FAILURE
1134a622223fSDevin Teske		f_eval_catch -d $funcname umount "$UMOUNT" /mnt # tmpfs
1135cd88b886SDevin Teske	fi
11362875e59fSOllivier Robert	#
11372875e59fSOllivier Robert	# Create the gmirror(8) GEOMS for swap
11382875e59fSOllivier Robert	#
11392875e59fSOllivier Robert	if [ "$ZFSBOOT_SWAP_MIRROR" ]; then
11402875e59fSOllivier Robert		for disk in $disks; do
11412875e59fSOllivier Robert			swap_devs="$swap_devs $disk$swappart"
11422875e59fSOllivier Robert		done
11432875e59fSOllivier Robert		f_eval_catch $funcname gmirror "$SWAP_GMIRROR_LABEL" \
11442875e59fSOllivier Robert			"$swap_devs" || return $FAILURE
11452875e59fSOllivier Robert	fi
1146cd88b886SDevin Teske
1147cd88b886SDevin Teske	#
1148a622223fSDevin Teske	# Create the ZFS root pool with desired type and disk devices
1149cd88b886SDevin Teske	#
11507cae6aabSDevin Teske	f_dprintf "$funcname: Creating root pool..."
1151bc4a673fSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_CREATE_WITH_OPTIONS" \
1152ed8690e3SOllivier Robert	             "-o altroot=$BSDINSTALL_CHROOT -O compress=lz4
1153ed8690e3SOllivier Robert	             -m none -f" \
11544b4b90d5SDevin Teske	             "$zroot_name" "$zroot_vdevtype" "$zroot_vdevs" ||
11554b4b90d5SDevin Teske	             return $FAILURE
1156cd88b886SDevin Teske
11577cae6aabSDevin Teske	# Customize the zroot a bit...
1158bc4a673fSDevin Teske	local option
11597cae6aabSDevin Teske	f_dprintf "$funcname: Setting miscellaneous options on root pool..."
1160ca2d97c2SDevin Teske	for option in atime=off; do
11617cae6aabSDevin Teske		f_eval_catch $funcname zfs "$ZFS_SET" $option "$zroot_name" ||
1162bc4a673fSDevin Teske			return $FAILURE
1163bc4a673fSDevin Teske	done
1164cd88b886SDevin Teske
1165cd88b886SDevin Teske	#
11667cae6aabSDevin Teske	# Create ZFS dataset layout within the new root pool
1167cd88b886SDevin Teske	#
1168bc4a673fSDevin Teske	f_dprintf "$funcname: Creating ZFS datasets..."
1169cd88b886SDevin Teske	echo "$ZFSBOOT_DATASETS" | while read dataset options; do
1170cd88b886SDevin Teske		# Skip blank lines and comments
1171cd88b886SDevin Teske		case "$dataset" in "#"*|"") continue; esac
1172cd88b886SDevin Teske		# Remove potential inline comments in options
1173cd88b886SDevin Teske		options="${options%%#*}"
1174cd88b886SDevin Teske		# Replace tabs with spaces
1175cd88b886SDevin Teske		f_replaceall "$options" "	" " " options
1176cd88b886SDevin Teske		# Reduce contiguous runs of space to one single space
1177cd88b886SDevin Teske		oldoptions=
1178cd88b886SDevin Teske		while [ "$oldoptions" != "$options" ]; do
1179cd88b886SDevin Teske			oldoptions="$options"
1180cd88b886SDevin Teske			f_replaceall "$options" "  " " " options
1181cd88b886SDevin Teske		done
1182cd88b886SDevin Teske		# Replace both commas and spaces with ` -o '
1183cd88b886SDevin Teske		f_replaceall "$options" "[ ,]" " -o " options
1184cd88b886SDevin Teske		# Create the dataset with desired options
1185bc4a673fSDevin Teske		f_eval_catch $funcname zfs "$ZFS_CREATE_WITH_OPTIONS" \
11867cae6aabSDevin Teske		             "${options:+-o $options}" "$zroot_name$dataset" ||
1187cd88b886SDevin Teske		             return $FAILURE
1188cd88b886SDevin Teske	done
1189cd88b886SDevin Teske
1190cd88b886SDevin Teske	# Touch up permissions on the tmp directories
1191bc4a673fSDevin Teske	f_dprintf "$funcname: Modifying directory permissions..."
1192bc4a673fSDevin Teske	local dir
1193bc4a673fSDevin Teske	for dir in /tmp /var/tmp; do
1194bc4a673fSDevin Teske		f_eval_catch $funcname chmod "$CHMOD_MODE" 1777 \
1195a622223fSDevin Teske		             $BSDINSTALL_CHROOTDIR$dir || return $FAILURE
1196bc4a673fSDevin Teske	done
1197cd88b886SDevin Teske
1198cd88b886SDevin Teske	# Create symlink(s)
11997cae6aabSDevin Teske	if [ "$ZFSBOOT_BOOT_POOL" ]; then
12007cae6aabSDevin Teske		f_dprintf "$funcname: Creating /boot symlink for boot pool..."
12017cae6aabSDevin Teske		f_eval_catch $funcname ln "$LN_SF" "$bootpool_name/boot" \
1202a622223fSDevin Teske		             $BSDINSTALL_CHROOT/boot || return $FAILURE
1203bc4a673fSDevin Teske	fi
1204cd88b886SDevin Teske
1205cd88b886SDevin Teske	# Set bootfs property
12067cae6aabSDevin Teske	local zroot_bootfs="$ZFSBOOT_BEROOT_NAME/$ZFSBOOT_BOOTFS_NAME"
1207bc4a673fSDevin Teske	f_dprintf "$funcname: Setting bootfs property..."
1208bc4a673fSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_SET" \
12097cae6aabSDevin Teske		"bootfs=\"$zroot_name/$zroot_bootfs\"" "$zroot_name" ||
12107cae6aabSDevin Teske		return $FAILURE
1211cd88b886SDevin Teske
1212cd88b886SDevin Teske	# Export the pool(s)
1213bc4a673fSDevin Teske	f_dprintf "$funcname: Temporarily exporting ZFS pool(s)..."
12147cae6aabSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_EXPORT" "$zroot_name" ||
1215bc4a673fSDevin Teske	             return $FAILURE
12167cae6aabSDevin Teske	if [ "$ZFSBOOT_BOOT_POOL" ]; then
1217bc4a673fSDevin Teske		f_eval_catch $funcname zpool "$ZPOOL_EXPORT" \
12187cae6aabSDevin Teske		             "$bootpool_name" || return $FAILURE
1219bc4a673fSDevin Teske	fi
1220cd88b886SDevin Teske
1221cd88b886SDevin Teske	# Destroy the gnop devices (if enabled)
1222bc4a673fSDevin Teske	for disk in ${ZFSBOOT_GNOP_4K_FORCE_ALIGN:+$disks}; do
12237cae6aabSDevin Teske		if [ "$ZFSBOOT_BOOT_POOL" ]; then
1224a622223fSDevin Teske			f_eval_catch -d $funcname gnop "$GNOP_DESTROY" \
1225a622223fSDevin Teske			                $disk$bootpart.nop
12267cae6aabSDevin Teske		fi
12277cae6aabSDevin Teske		if [ ! "$ZFSBOOT_GELI_ENCRYPTION" ]; then
1228a622223fSDevin Teske			f_eval_catch -d $funcname gnop "$GNOP_DESTROY" \
1229a622223fSDevin Teske			                $disk$targetpart.nop
1230cd88b886SDevin Teske		fi
1231cd88b886SDevin Teske	done
1232cd88b886SDevin Teske
12337cae6aabSDevin Teske	# MBR boot loader touch-up
1234cd88b886SDevin Teske	if [ "$ZFSBOOT_PARTITION_SCHEME" = "MBR" ]; then
1235bc4a673fSDevin Teske		f_dprintf "$funcname: Updating MBR boot loader on disks..."
1236cd88b886SDevin Teske		# Stick the ZFS boot loader in the "convienient hole" after
1237cd88b886SDevin Teske		# the ZFS internal metadata
1238bc4a673fSDevin Teske		for disk in $disks; do
1239bc4a673fSDevin Teske			f_eval_catch $funcname dd "$DD_WITH_OPTIONS" \
1240a622223fSDevin Teske			             /boot/zfsboot /dev/$disk$bootpart \
1241bc4a673fSDevin Teske			             "skip=1 seek=1024" || return $FAILURE
1242cd88b886SDevin Teske		done
1243cd88b886SDevin Teske	fi
1244cd88b886SDevin Teske
1245cd88b886SDevin Teske	# Re-import the ZFS pool(s)
1246bc4a673fSDevin Teske	f_dprintf "$funcname: Re-importing ZFS pool(s)..."
1247bc4a673fSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_IMPORT_WITH_OPTIONS" \
12487cae6aabSDevin Teske	             "-o altroot=\"$BSDINSTALL_CHROOT\"" "$zroot_name" ||
1249bc4a673fSDevin Teske	             return $FAILURE
12507cae6aabSDevin Teske	if [ "$ZFSBOOT_BOOT_POOL" ]; then
1251bc4a673fSDevin Teske		f_eval_catch $funcname zpool "$ZPOOL_IMPORT_WITH_OPTIONS" \
12527cae6aabSDevin Teske		             "-o altroot=\"$BSDINSTALL_CHROOT\"" \
12537cae6aabSDevin Teske		             "$bootpool_name" || return $FAILURE
1254bc4a673fSDevin Teske	fi
1255cd88b886SDevin Teske
1256cd88b886SDevin Teske	# While this is apparently not needed, it seems to help MBR
12576311cc9eSDevin Teske	f_dprintf "$funcname: Configuring zpool.cache for zroot..."
1258a622223fSDevin Teske	f_eval_catch $funcname mkdir "$MKDIR_P" $BSDINSTALL_CHROOT/boot/zfs ||
1259bc4a673fSDevin Teske	             return $FAILURE
1260bc4a673fSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_SET" \
1261a622223fSDevin Teske	             "cachefile=\"$BSDINSTALL_CHROOT/boot/zfs/zpool.cache\"" \
12627cae6aabSDevin Teske	             "$zroot_name" || return $FAILURE
1263cd88b886SDevin Teske
126413d8e1ceSDevin Teske	# Last, but not least... required lines for rc.conf(5)/loader.conf(5)
1265cd88b886SDevin Teske	# NOTE: We later concatenate these into their destination
1266bc4a673fSDevin Teske	f_dprintf "%s: Configuring rc.conf(5)/loader.conf(5) additions..." \
1267bc4a673fSDevin Teske	          "$funcname"
1268bc4a673fSDevin Teske	f_eval_catch $funcname echo "$ECHO_APPEND" 'zfs_enable=\"YES\"' \
1269a622223fSDevin Teske	             $BSDINSTALL_TMPETC/rc.conf.zfs || return $FAILURE
1270bc4a673fSDevin Teske	f_eval_catch $funcname echo "$ECHO_APPEND" 'zfs_load=\"YES\"' \
1271a622223fSDevin Teske	             $BSDINSTALL_TMPBOOT/loader.conf.zfs || return $FAILURE
1272ee482f2cSDevin Teske	f_eval_catch $funcname echo "$ECHO_APPEND" \
1273ee482f2cSDevin Teske	             'kern.geom.label.disk_ident.enable=\"0\"' \
1274ee482f2cSDevin Teske	             $BSDINSTALL_TMPBOOT/loader.conf.zfs || return $FAILURE
12752875e59fSOllivier Robert	f_eval_catch $funcname echo "$ECHO_APPEND" \
12762875e59fSOllivier Robert	             'kern.geom.label.gptid.enable=\"0\"' \
12772875e59fSOllivier Robert	             $BSDINSTALL_TMPBOOT/loader.conf.zfs || return $FAILURE
12782875e59fSOllivier Robert
12792875e59fSOllivier Robert	if [ "$ZFSBOOT_SWAP_MIRROR" ]; then
12802875e59fSOllivier Robert		f_eval_catch $funcname echo "$ECHO_APPEND" 'geom_mirror_load=\"YES\"' \
12812875e59fSOllivier Robert		             $BSDINSTALL_TMPBOOT/loader.conf.gmirror || return $FAILURE
12822875e59fSOllivier Robert	fi
1283cd88b886SDevin Teske
12847cae6aabSDevin Teske	# We're all done unless we should go on for boot pool
12857cae6aabSDevin Teske	[ "$ZFSBOOT_BOOT_POOL" ] || return $SUCCESS
1286cd88b886SDevin Teske
12876311cc9eSDevin Teske	# Set cachefile for boot pool so it auto-imports at system start
12886311cc9eSDevin Teske	f_dprintf "$funcname: Configuring zpool.cache for boot pool..."
12896311cc9eSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_SET" \
12906311cc9eSDevin Teske	             "cachefile=\"$BSDINSTALL_CHROOT/boot/zfs/zpool.cache\"" \
12916311cc9eSDevin Teske	             "$bootpool_name" || return $FAILURE
12926311cc9eSDevin Teske
1293bc4a673fSDevin Teske	# Some additional geli(8) requirements for loader.conf(5)
1294bc4a673fSDevin Teske	for option in \
1295a622223fSDevin Teske		'zpool_cache_load=\"YES\"' \
1296a622223fSDevin Teske		'zpool_cache_type=\"/boot/zfs/zpool.cache\"' \
1297a622223fSDevin Teske		'zpool_cache_name=\"/boot/zfs/zpool.cache\"' \
1298bc4a673fSDevin Teske	; do
1299a622223fSDevin Teske		f_eval_catch $funcname echo "$ECHO_APPEND" "$option" \
1300a622223fSDevin Teske		             $BSDINSTALL_TMPBOOT/loader.conf.zfs ||
1301bc4a673fSDevin Teske		             return $FAILURE
1302bc4a673fSDevin Teske	done
13037cae6aabSDevin Teske	f_eval_catch $funcname printf "$PRINTF_CONF" vfs.root.mountfrom \
13047cae6aabSDevin Teske	    "\"zfs:$zroot_name/$zroot_bootfs\"" \
13057cae6aabSDevin Teske	    $BSDINSTALL_TMPBOOT/loader.conf.root || return $FAILURE
13067cae6aabSDevin Teske
13077cae6aabSDevin Teske	# We're all done unless we should go on to do encryption
13087cae6aabSDevin Teske	[ "$ZFSBOOT_GELI_ENCRYPTION" ] || return $SUCCESS
130913d8e1ceSDevin Teske
1310cd88b886SDevin Teske	#
1311cd88b886SDevin Teske	# Configure geli(8)-based encryption
1312cd88b886SDevin Teske	#
1313bc4a673fSDevin Teske	f_dprintf "$funcname: Configuring disk encryption..."
1314bc4a673fSDevin Teske	f_eval_catch $funcname echo "$ECHO_APPEND" 'aesni_load=\"YES\"' \
1315a622223fSDevin Teske	    $BSDINSTALL_TMPBOOT/loader.conf.aesni || return $FAILURE
1316bc4a673fSDevin Teske	f_eval_catch $funcname echo "$ECHO_APPEND" 'geom_eli_load=\"YES\"' \
1317a622223fSDevin Teske	    $BSDINSTALL_TMPBOOT/loader.conf.geli || return $FAILURE
1318bc4a673fSDevin Teske	for disk in $disks; do
1319bc4a673fSDevin Teske		f_eval_catch $funcname printf "$PRINTF_CONF" \
1320a622223fSDevin Teske			geli_%s_keyfile0_load "$disk$targetpart YES" \
1321a622223fSDevin Teske			$BSDINSTALL_TMPBOOT/loader.conf.$disk$targetpart ||
1322cd88b886SDevin Teske			return $FAILURE
1323bc4a673fSDevin Teske		f_eval_catch $funcname printf "$PRINTF_CONF" \
1324bc4a673fSDevin Teske			geli_%s_keyfile0_type \
1325a622223fSDevin Teske			"$disk$targetpart $disk$targetpart:geli_keyfile0" \
1326a622223fSDevin Teske			$BSDINSTALL_TMPBOOT/loader.conf.$disk$targetpart ||
1327cd88b886SDevin Teske			return $FAILURE
1328bc4a673fSDevin Teske		f_eval_catch $funcname printf "$PRINTF_CONF" \
1329bc4a673fSDevin Teske			geli_%s_keyfile0_name \
1330a622223fSDevin Teske			"$disk$targetpart \"$ZFSBOOT_GELI_KEY_FILE\"" \
1331a622223fSDevin Teske			$BSDINSTALL_TMPBOOT/loader.conf.$disk$targetpart ||
1332cd88b886SDevin Teske			return $FAILURE
1333cd88b886SDevin Teske	done
1334cd88b886SDevin Teske
1335cd88b886SDevin Teske	return $SUCCESS
1336cd88b886SDevin Teske}
1337cd88b886SDevin Teske
1338cd88b886SDevin Teske# dialog_menu_diskinfo
1339cd88b886SDevin Teske#
1340cd88b886SDevin Teske# Prompt the user to select a disk and then provide detailed info on it.
1341cd88b886SDevin Teske#
1342cd88b886SDevin Teskedialog_menu_diskinfo()
1343cd88b886SDevin Teske{
1344a88393ceSDevin Teske	local device disk
1345cd88b886SDevin Teske
1346cd88b886SDevin Teske	#
1347cd88b886SDevin Teske	# Break from loop when user cancels disk selection
1348cd88b886SDevin Teske	#
1349cd88b886SDevin Teske	while :; do
1350a88393ceSDevin Teske		device=$( msg_cancel="$msg_back" f_device_menu \
1351cd88b886SDevin Teske			"$DIALOG_TITLE" "$msg_select_a_disk_device" "" \
1352cd88b886SDevin Teske			$DEVICE_TYPE_DISK 2>&1 ) || break
1353a88393ceSDevin Teske		$device get name disk
1354cd88b886SDevin Teske
1355cd88b886SDevin Teske		# Show gpart(8) `show' and camcontrol(8) `inquiry' data
1356cd88b886SDevin Teske		f_show_msg "$msg_detailed_disk_info" \
1357cd88b886SDevin Teske			"$disk" "$( gpart show $disk 2> /dev/null )" \
1358cd88b886SDevin Teske			"$disk" "$( camcontrol inquiry $disk 2> /dev/null )" \
1359cd88b886SDevin Teske			"$disk" "$( camcontrol identify $disk 2> /dev/null )"
1360cd88b886SDevin Teske	done
1361cd88b886SDevin Teske
1362cd88b886SDevin Teske	return $SUCCESS
1363cd88b886SDevin Teske}
1364cd88b886SDevin Teske
1365cd88b886SDevin Teske############################################################ MAIN
1366cd88b886SDevin Teske
1367cd88b886SDevin Teske#
1368cd88b886SDevin Teske# Initialize
1369cd88b886SDevin Teske#
1370cd88b886SDevin Teskef_dialog_title "$msg_zfs_configuration"
1371cd88b886SDevin Teskef_dialog_backtitle "$msg_freebsd_installer"
1372cd88b886SDevin Teske
1373cd88b886SDevin Teske# User may have specifically requested ZFS-related operations be interactive
1374cd88b886SDevin Teske! f_interactive && f_zfsinteractive && unset $VAR_NONINTERACTIVE
1375cd88b886SDevin Teske
1376cd88b886SDevin Teske#
1377bc4a673fSDevin Teske# Debugging
1378bc4a673fSDevin Teske#
1379bc4a673fSDevin Teskef_dprintf "BSDINSTALL_CHROOT=[%s]" "$BSDINSTALL_CHROOT"
1380bc4a673fSDevin Teskef_dprintf "BSDINSTALL_TMPETC=[%s]" "$BSDINSTALL_TMPETC"
1381a622223fSDevin Teskef_dprintf "FSTAB_FMT=[%s]" "$FSTAB_FMT"
1382bc4a673fSDevin Teske
1383bc4a673fSDevin Teske#
1384cd88b886SDevin Teske# Loop over the main menu until we've accomplished what we came here to do
1385cd88b886SDevin Teske#
1386cd88b886SDevin Teskewhile :; do
1387cd88b886SDevin Teske	if ! f_interactive; then
1388cd88b886SDevin Teske		retval=$DIALOG_OK
1389bc4a673fSDevin Teske		mtag=">>> $msg_install"
1390cd88b886SDevin Teske	else
1391cd88b886SDevin Teske		dialog_menu_main
1392cd88b886SDevin Teske		retval=$?
1393cd88b886SDevin Teske		f_dialog_menutag_fetch mtag
1394cd88b886SDevin Teske	fi
1395cd88b886SDevin Teske
1396bc4a673fSDevin Teske	f_dprintf "retval=%u mtag=[%s]" $retval "$mtag"
1397cd88b886SDevin Teske	[ $retval -eq $DIALOG_OK ] || f_die
1398cd88b886SDevin Teske
1399cd88b886SDevin Teske	case "$mtag" in
1400bc4a673fSDevin Teske	">>> $msg_install")
1401cd88b886SDevin Teske		#
1402cd88b886SDevin Teske		# First, validate the user's selections
1403cd88b886SDevin Teske		#
1404cd88b886SDevin Teske
1405cd88b886SDevin Teske		# Make sure they gave us a name for the pool
1406cd88b886SDevin Teske		if [ ! "$ZFSBOOT_POOL_NAME" ]; then
1407bc4a673fSDevin Teske			f_dprintf "Pool name cannot be empty."
1408bc4a673fSDevin Teske			f_show_err "$msg_pool_name_cannot_be_empty"
1409cd88b886SDevin Teske			continue
1410cd88b886SDevin Teske		fi
1411bc4a673fSDevin Teske
1412bc4a673fSDevin Teske		# Validate vdev type against number of disks selected/scripted
1413bc4a673fSDevin Teske		# (also validates that ZFSBOOT_DISKS are real [probed] disks)
1414bc4a673fSDevin Teske		# NB: dialog_menu_layout supports running non-interactively
1415bc4a673fSDevin Teske		dialog_menu_layout || continue
1416bc4a673fSDevin Teske
1417cd88b886SDevin Teske		# Make sure each disk will be at least 50% ZFS
1418cd88b886SDevin Teske		if f_expand_number "$ZFSBOOT_SWAP_SIZE" swapsize &&
14197cae6aabSDevin Teske		   f_expand_number "$ZFSBOOT_BOOT_POOL_SIZE" bootsize
1420cd88b886SDevin Teske		then
1421cd88b886SDevin Teske			minsize=$swapsize teeny_disks=
14227cae6aabSDevin Teske			[ "$ZFSBOOT_BOOT_POOL" ] &&
14237cae6aabSDevin Teske				minsize=$(( $minsize + $bootsize ))
1424bc4a673fSDevin Teske			for disk in $ZFSBOOT_DISKS; do
1425a88393ceSDevin Teske				debug= f_device_find -1 \
1426a88393ceSDevin Teske					$disk $DEVICE_TYPE_DISK device
1427a88393ceSDevin Teske				$device get capacity disksize || continue
1428a88393ceSDevin Teske				[ ${disksize:-0} -ge 0 ] || disksize=0
1429cd88b886SDevin Teske				disksize=$(( $disksize - $minsize ))
1430cd88b886SDevin Teske				[ $disksize -lt $minsize ] &&
1431cd88b886SDevin Teske					teeny_disks="$teeny_disks $disk"
1432cd88b886SDevin Teske			done
1433cd88b886SDevin Teske			if [ "$teeny_disks" ]; then
14347cae6aabSDevin Teske				f_dprintf "swapsize=[%s] bootsize[%s] %s" \
1435bc4a673fSDevin Teske				          "$ZFSBOOT_SWAP_SIZE" \
14367cae6aabSDevin Teske				          "$ZFSBOOT_BOOT_POOL_SIZE" \
14377cae6aabSDevin Teske				          "minsize=[$minsize]"
1438bc4a673fSDevin Teske				f_dprintf "These disks are too small: %s" \
1439bc4a673fSDevin Teske				          "$teeny_disks"
1440bc4a673fSDevin Teske				f_show_err "$msg_these_disks_are_too_small" \
1441cd88b886SDevin Teske				           "$ZFSBOOT_SWAP_SIZE" \
14427cae6aabSDevin Teske				           "$ZFSBOOT_BOOT_POOL_SIZE" \
1443cd88b886SDevin Teske				           "$teeny_disks"
1444cd88b886SDevin Teske				continue
1445cd88b886SDevin Teske			fi
1446cd88b886SDevin Teske		fi
1447cd88b886SDevin Teske
1448cd88b886SDevin Teske		#
1449cd88b886SDevin Teske		# Last Chance!
1450cd88b886SDevin Teske		#
1451bc4a673fSDevin Teske		if f_interactive; then
1452bc4a673fSDevin Teske			dialog_last_chance $ZFSBOOT_DISKS || continue
1453cd88b886SDevin Teske		fi
1454cd88b886SDevin Teske
1455cd88b886SDevin Teske		#
1456cd88b886SDevin Teske		# Let's do this
1457cd88b886SDevin Teske		#
1458cd88b886SDevin Teske
1459cd88b886SDevin Teske		vdev_type="$ZFSBOOT_VDEV_TYPE"
1460cd88b886SDevin Teske
1461cd88b886SDevin Teske		# Blank the vdev type for the default layout
1462cd88b886SDevin Teske		[ "$vdev_type" = "stripe" ] && vdev_type=
1463cd88b886SDevin Teske
1464bc4a673fSDevin Teske		zfs_create_boot "$ZFSBOOT_POOL_NAME" \
1465bc4a673fSDevin Teske		                "$vdev_type" $ZFSBOOT_DISKS || continue
1466cd88b886SDevin Teske
1467cd88b886SDevin Teske		break # to success
1468cd88b886SDevin Teske		;;
1469bc4a673fSDevin Teske	?" $msg_pool_type_disks")
1470bc4a673fSDevin Teske		ZFSBOOT_CONFIRM_LAYOUT=1
1471bc4a673fSDevin Teske		dialog_menu_layout
1472bc4a673fSDevin Teske		# User has poked settings, disable later confirmation
1473bc4a673fSDevin Teske		ZFSBOOT_CONFIRM_LAYOUT=
1474bc4a673fSDevin Teske		;;
1475cd88b886SDevin Teske	"- $msg_rescan_devices") f_device_rescan ;;
1476cd88b886SDevin Teske	"- $msg_disk_info") dialog_menu_diskinfo ;;
1477cd88b886SDevin Teske	?" $msg_pool_name")
1478cd88b886SDevin Teske		# Prompt the user to input/change the name for the new pool
1479cd88b886SDevin Teske		f_dialog_input input \
1480cd88b886SDevin Teske			"$msg_please_enter_a_name_for_your_zpool" \
1481cd88b886SDevin Teske			"$ZFSBOOT_POOL_NAME" &&
1482cd88b886SDevin Teske			ZFSBOOT_POOL_NAME="$input"
1483cd88b886SDevin Teske		;;
1484cd88b886SDevin Teske	?" $msg_force_4k_sectors")
1485cd88b886SDevin Teske		# Toggle the variable referenced both by the menu and later
1486cd88b886SDevin Teske		if [ "$ZFSBOOT_GNOP_4K_FORCE_ALIGN" ]; then
1487cd88b886SDevin Teske			ZFSBOOT_GNOP_4K_FORCE_ALIGN=
1488cd88b886SDevin Teske		else
1489cd88b886SDevin Teske			ZFSBOOT_GNOP_4K_FORCE_ALIGN=1
1490cd88b886SDevin Teske		fi
1491cd88b886SDevin Teske		;;
1492bc4a673fSDevin Teske	?" $msg_encrypt_disks")
1493cd88b886SDevin Teske		# Toggle the variable referenced both by the menu and later
1494cd88b886SDevin Teske		if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
1495cd88b886SDevin Teske			ZFSBOOT_GELI_ENCRYPTION=
1496cd88b886SDevin Teske		else
1497926ec73fSDevin Teske			ZFSBOOT_GNOP_4K_FORCE_ALIGN=1
1498cd88b886SDevin Teske			ZFSBOOT_GELI_ENCRYPTION=1
1499cd88b886SDevin Teske		fi
1500cd88b886SDevin Teske		;;
1501cd88b886SDevin Teske	?" $msg_partition_scheme")
1502cd88b886SDevin Teske		# Toggle between GPT and MBR
1503cd88b886SDevin Teske		if [ "$ZFSBOOT_PARTITION_SCHEME" = GPT ]; then
1504cd88b886SDevin Teske			ZFSBOOT_PARTITION_SCHEME=MBR
1505cd88b886SDevin Teske		else
1506cd88b886SDevin Teske			ZFSBOOT_PARTITION_SCHEME=GPT
1507cd88b886SDevin Teske		fi
1508cd88b886SDevin Teske		;;
1509cd88b886SDevin Teske	?" $msg_swap_size")
1510cd88b886SDevin Teske		# Prompt the user to input/change the swap size for each disk
1511cd88b886SDevin Teske		f_dialog_input input \
1512cd88b886SDevin Teske			"$msg_please_enter_amount_of_swap_space" \
1513cd88b886SDevin Teske			"$ZFSBOOT_SWAP_SIZE" &&
151430c8ebe9SDevin Teske			ZFSBOOT_SWAP_SIZE="${input:-0}"
1515cd88b886SDevin Teske		;;
15162875e59fSOllivier Robert	?" $msg_swap_mirror")
15172875e59fSOllivier Robert		# Toggle the variable referenced both by the menu and later
15182875e59fSOllivier Robert		if [ "$ZFSBOOT_SWAP_MIRROR" ]; then
15192875e59fSOllivier Robert			ZFSBOOT_SWAP_MIRROR=
15202875e59fSOllivier Robert		else
15212875e59fSOllivier Robert			ZFSBOOT_SWAP_MIRROR=1
15222875e59fSOllivier Robert		fi
15232875e59fSOllivier Robert		;;
15242875e59fSOllivier Robert	?" $msg_swap_encrypt")
15252875e59fSOllivier Robert		# Toggle the variable referenced both by the menu and later
15262875e59fSOllivier Robert		if [ "$ZFSBOOT_SWAP_ENCRYPTION" ]; then
15272875e59fSOllivier Robert			ZFSBOOT_SWAP_ENCRYPTION=
15282875e59fSOllivier Robert		else
15292875e59fSOllivier Robert			ZFSBOOT_SWAP_ENCRYPTION=1
15302875e59fSOllivier Robert		fi
15312875e59fSOllivier Robert		;;
1532cd88b886SDevin Teske	esac
1533cd88b886SDevin Teskedone
1534cd88b886SDevin Teske
1535cd88b886SDevin Teskereturn $SUCCESS
1536cd88b886SDevin Teske
1537cd88b886SDevin Teske################################################################################
1538cd88b886SDevin Teske# END
1539cd88b886SDevin Teske################################################################################
1540