xref: /freebsd/usr.sbin/bsdinstall/scripts/zfsboot (revision 2875e59f52f95d59ab7bf94ea67a98292f4f7775)
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#
111*2875e59fSOllivier Robert# Should we use geli(8) to encrypt the swap?
112*2875e59fSOllivier Robert#
113*2875e59fSOllivier Robert: ${ZFSBOOT_SWAP_ENCRYPTION=}
114*2875e59fSOllivier Robert
115*2875e59fSOllivier Robert#
116*2875e59fSOllivier Robert# Should we use gmirror(8) to mirror the swap?
117*2875e59fSOllivier Robert#
118*2875e59fSOllivier Robert: ${ZFSBOOT_SWAP_MIRROR=}
119*2875e59fSOllivier Robert
120*2875e59fSOllivier 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
134cd88b886SDevin Teske	/tmp		mountpoint=/tmp,compression=lz4,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
143cd88b886SDevin Teske	/usr/ports		compression=lz4,setuid=off
144cd88b886SDevin Teske
145cd88b886SDevin Teske	# Source tree (compressed)
146cd88b886SDevin Teske	/usr/src	compression=lz4,exec=off,setuid=off
147cd88b886SDevin Teske
148cd88b886SDevin Teske	# Create /var and friends
149cd88b886SDevin Teske	/var		mountpoint=/var
150cd88b886SDevin Teske	/var/crash	compression=lz4,exec=off,setuid=off
151cd88b886SDevin Teske	/var/log	compression=lz4,exec=off,setuid=off
152a6c8532cSDevin Teske	/var/mail	compression=lz4,atime=on
153cd88b886SDevin Teske	/var/tmp	compression=lz4,exec=on,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"'
197*2875e59fSOllivier 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)"
223bc4a673fSDevin Teskemsg_encrypt_disks="Encrypt Disks?"
224bc4a673fSDevin Teskemsg_encrypt_disks_help="Use geli(8) to encrypt all data partitions"
225bc4a673fSDevin Teskemsg_error="Error"
226cd88b886SDevin Teskemsg_force_4k_sectors="Force 4K Sectors?"
227cd88b886SDevin Teskemsg_force_4k_sectors_help="Use gnop(8) to configure forced 4K sector alignment"
228cd88b886SDevin Teskemsg_freebsd_installer="FreeBSD Installer"
229cd88b886SDevin 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"
2307a434c5cSDevin Teskemsg_geli_setup="Initializing encryption on selected disks,\n this will take several seconds per disk"
231bc4a673fSDevin Teskemsg_install="Install"
232bc4a673fSDevin Teskemsg_install_desc="Proceed with Installation"
233bc4a673fSDevin Teskemsg_install_help="Create ZFS boot pool with displayed options"
2347cae6aabSDevin Teskemsg_invalid_boot_pool_size="Invalid boot pool size \`%s'"
235bc4a673fSDevin Teskemsg_invalid_disk_argument="Invalid disk argument \`%s'"
236bc4a673fSDevin Teskemsg_invalid_index_argument="Invalid index argument \`%s'"
237bc4a673fSDevin Teskemsg_invalid_swap_size="Invalid swap size \`%s'"
238cd88b886SDevin Teskemsg_invalid_virtual_device_type="Invalid Virtual Device type \`%s'"
239bc4a673fSDevin 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"
240bc4a673fSDevin 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'
241cd88b886SDevin Teskemsg_mirror_desc="Mirror - n-Way Mirroring"
242cd88b886SDevin Teskemsg_mirror_help="[2+ Disks] Mirroring provides the best performance, but the least storage"
243bc4a673fSDevin Teskemsg_missing_disk_arguments="missing disk arguments"
244bc4a673fSDevin Teskemsg_missing_one_or_more_scripted_disks="Missing one or more scripted disks!"
245cd88b886SDevin Teskemsg_no="NO"
246cd88b886SDevin Teskemsg_no_disks_present_to_configure="No disk(s) present to configure"
247cd88b886SDevin Teskemsg_no_disks_selected="No disks selected."
248bc4a673fSDevin Teskemsg_not_enough_disks_selected="Not enough disks selected. (%u < %u minimum)"
249bc4a673fSDevin Teskemsg_null_disk_argument="NULL disk argument"
250bc4a673fSDevin Teskemsg_null_index_argument="NULL index argument"
251bc4a673fSDevin Teskemsg_null_poolname="NULL poolname"
252cd88b886SDevin Teskemsg_ok="OK"
253cd88b886SDevin Teskemsg_partition_scheme="Partition Scheme"
254cd88b886SDevin Teskemsg_partition_scheme_help="Toggle between GPT and MBR partitioning schemes"
255cd88b886SDevin Teskemsg_please_enter_a_name_for_your_zpool="Please enter a name for your zpool:"
256cd88b886SDevin Teskemsg_please_enter_amount_of_swap_space="Please enter amount of swap space (SI-Unit suffixes\nrecommended; e.g., \`2g' for 2 Gigabytes):"
257cd88b886SDevin Teskemsg_please_select_one_or_more_disks="Please select one or more disks to create a zpool:"
258cd88b886SDevin Teskemsg_pool_name="Pool Name"
259cd88b886SDevin Teskemsg_pool_name_cannot_be_empty="Pool name cannot be empty."
260cd88b886SDevin Teskemsg_pool_name_help="Customize the name of the zpool to be created (Required)"
261bc4a673fSDevin Teskemsg_pool_type_disks="Pool Type/Disks:"
262bc4a673fSDevin Teskemsg_pool_type_disks_help="Choose type of ZFS Virtual Device and disks to use (Required)"
263cd88b886SDevin Teskemsg_processing_selection="Processing selection..."
264cd88b886SDevin Teskemsg_raidz1_desc="RAID-Z1 - Single Redundant RAID"
265cd88b886SDevin Teskemsg_raidz1_help="[3+ Disks] Withstand failure of 1 disk. Recommended for: 3, 5 or 9 disks"
266cd88b886SDevin Teskemsg_raidz2_desc="RAID-Z2 - Double Redundant RAID"
267cd88b886SDevin Teskemsg_raidz2_help="[4+ Disks] Withstand failure of 2 disks. Recommended for: 4, 6 or 10 disks"
268cd88b886SDevin Teskemsg_raidz3_desc="RAID-Z3 - Triple Redundant RAID"
269cd88b886SDevin Teskemsg_raidz3_help="[5+ Disks] Withstand failure of 3 disks. Recommended for: 5, 7 or 11 disks"
270cd88b886SDevin Teskemsg_rescan_devices="Rescan Devices"
271cd88b886SDevin Teskemsg_rescan_devices_help="Scan for device changes"
272cd88b886SDevin Teskemsg_select="Select"
273cd88b886SDevin Teskemsg_select_a_disk_device="Select a disk device"
274cd88b886SDevin Teskemsg_select_virtual_device_type="Select Virtual Device type:"
275cd88b886SDevin Teskemsg_stripe_desc="Stripe - No Redundancy"
276cd88b886SDevin Teskemsg_stripe_help="[1+ Disks] Striping provides maximum storage but no redundancy"
277*2875e59fSOllivier Robertmsg_swap_encrypt="Encrypt Swap?"
278*2875e59fSOllivier Robertmsg_swap_encrypt_help="Encrypt swap partitions with temporary keys, discarded on reboot"
279*2875e59fSOllivier Robertmsg_swap_mirror="Mirror Swap?"
280*2875e59fSOllivier Robertmsg_swap_mirror_help="Mirror swap partitions for redundancy, breaks crash dumps"
281cd88b886SDevin Teskemsg_swap_size="Swap Size"
282cd88b886SDevin Teskemsg_swap_size_help="Customize how much swap space is allocated to each selected disk"
283bc4a673fSDevin 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."
284bc4a673fSDevin Teskemsg_unable_to_get_disk_capacity="Unable to get disk capacity of \`%s'"
285bc4a673fSDevin Teskemsg_unsupported_partition_scheme="%s is an unsupported partition scheme"
286bc4a673fSDevin Teskemsg_user_cancelled="User Cancelled."
287cd88b886SDevin Teskemsg_yes="YES"
288cd88b886SDevin Teskemsg_zfs_configuration="ZFS Configuration"
289cd88b886SDevin Teske
290cd88b886SDevin Teske############################################################ FUNCTIONS
291cd88b886SDevin Teske
292cd88b886SDevin Teske# dialog_menu_main
293cd88b886SDevin Teske#
294cd88b886SDevin Teske# Display the dialog(1)-based application main menu.
295cd88b886SDevin Teske#
296cd88b886SDevin Teskedialog_menu_main()
297cd88b886SDevin Teske{
298cd88b886SDevin Teske	local title="$DIALOG_TITLE"
299cd88b886SDevin Teske	local btitle="$DIALOG_BACKTITLE"
300cd88b886SDevin Teske	local prompt="$msg_configure_options"
301cd88b886SDevin Teske	local force4k="$msg_no"
302cd88b886SDevin Teske	local usegeli="$msg_no"
303*2875e59fSOllivier Robert	local swapgeli="$msg_no"
304*2875e59fSOllivier Robert	local swapmirror="$msg_no"
305cd88b886SDevin Teske	[ "$ZFSBOOT_GNOP_4K_FORCE_ALIGN" ] && force4k="$msg_yes"
306cd88b886SDevin Teske	[ "$ZFSBOOT_GELI_ENCRYPTION" ] && usegeli="$msg_yes"
307*2875e59fSOllivier Robert	[ "$ZFSBOOT_SWAP_ENCRYPTION" ] && swapgeli="$msg_yes"
308*2875e59fSOllivier Robert	[ "$ZFSBOOT_SWAP_MIRROR" ] && swapmirror="$msg_yes"
309a88393ceSDevin Teske	local disks n
310a88393ceSDevin Teske	f_count n $ZFSBOOT_DISKS
311bc4a673fSDevin Teske	{ [ $n -eq 1 ] && disks=disk; } || disks=disks # grammar
312cd88b886SDevin Teske	local menu_list="
313bc4a673fSDevin Teske		'>>> $msg_install'        '$msg_install_desc'
314bc4a673fSDevin Teske		                          '$msg_install_help'
315bc4a673fSDevin Teske		'T $msg_pool_type_disks'  '$ZFSBOOT_VDEV_TYPE: $n $disks'
316bc4a673fSDevin Teske                                          '$msg_pool_type_disks_help'
317cd88b886SDevin Teske		'- $msg_rescan_devices'   '*'
318cd88b886SDevin Teske		                          '$msg_rescan_devices_help'
319cd88b886SDevin Teske		'- $msg_disk_info'        '*'
320cd88b886SDevin Teske		                          '$msg_disk_info_help'
321bc4a673fSDevin Teske		'N $msg_pool_name'        '$ZFSBOOT_POOL_NAME'
322cd88b886SDevin Teske		                          '$msg_pool_name_help'
323cd88b886SDevin Teske		'4 $msg_force_4k_sectors' '$force4k'
324cd88b886SDevin Teske		                          '$msg_force_4k_sectors_help'
325bc4a673fSDevin Teske		'E $msg_encrypt_disks'    '$usegeli'
326bc4a673fSDevin Teske		                          '$msg_encrypt_disks_help'
327bc4a673fSDevin Teske		'P $msg_partition_scheme' '$ZFSBOOT_PARTITION_SCHEME'
328cd88b886SDevin Teske		                          '$msg_partition_scheme_help'
329bc4a673fSDevin Teske		'S $msg_swap_size'        '$ZFSBOOT_SWAP_SIZE'
330cd88b886SDevin Teske		                          '$msg_swap_size_help'
331*2875e59fSOllivier Robert		'M $msg_swap_mirror'      '$swapmirror'
332*2875e59fSOllivier Robert		                          '$msg_swap_mirror_help'
333*2875e59fSOllivier Robert		'W $msg_swap_encrypt'     '$swapgeli'
334*2875e59fSOllivier Robert		                          '$msg_swap_encrypt_help'
335cd88b886SDevin Teske	" # END-QUOTE
336cd88b886SDevin Teske	local defaultitem= # Calculated below
337cd88b886SDevin Teske	local hline="$hline_alnum_arrows_punc_tab_enter"
338cd88b886SDevin Teske
339cd88b886SDevin Teske	local height width rows
340cd88b886SDevin Teske	eval f_dialog_menu_with_help_size height width rows \
341cd88b886SDevin Teske		\"\$title\" \"\$btitle\" \"\$prompt\" \"\$hline\" $menu_list
342cd88b886SDevin Teske
343cd88b886SDevin Teske	# Obtain default-item from previously stored selection
344cd88b886SDevin Teske	f_dialog_default_fetch defaultitem
345cd88b886SDevin Teske
346cd88b886SDevin Teske	local menu_choice
347cd88b886SDevin Teske	menu_choice=$( eval $DIALOG \
348cd88b886SDevin Teske		--title \"\$title\"              \
349cd88b886SDevin Teske		--backtitle \"\$btitle\"         \
350cd88b886SDevin Teske		--hline \"\$hline\"              \
351cd88b886SDevin Teske		--item-help                      \
352cd88b886SDevin Teske		--ok-label \"\$msg_select\"      \
353cd88b886SDevin Teske		--cancel-label \"\$msg_cancel\"  \
354cd88b886SDevin Teske		--default-item \"\$defaultitem\" \
355cd88b886SDevin Teske		--menu \"\$prompt\"              \
356cd88b886SDevin Teske		$height $width $rows             \
357cd88b886SDevin Teske		$menu_list                       \
358cd88b886SDevin Teske		2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
359cd88b886SDevin Teske	)
360cd88b886SDevin Teske	local retval=$?
361cd88b886SDevin Teske	f_dialog_data_sanitize menu_choice
362cd88b886SDevin Teske	f_dialog_menutag_store "$menu_choice"
363cd88b886SDevin Teske
364cd88b886SDevin Teske	# Only update default-item on success
365cd88b886SDevin Teske	[ $retval -eq $DIALOG_OK ] && f_dialog_default_store "$menu_choice"
366cd88b886SDevin Teske
367cd88b886SDevin Teske	return $retval
368cd88b886SDevin Teske}
369cd88b886SDevin Teske
370bc4a673fSDevin Teske# dialog_last_chance $disks ...
371cd88b886SDevin Teske#
372bc4a673fSDevin Teske# Display a list of the disks that the user is about to destroy. The default
373bc4a673fSDevin Teske# action is to return error status unless the user explicitly (non-default)
374bc4a673fSDevin Teske# selects "Yes" from the noyes dialog.
375cd88b886SDevin Teske#
376bc4a673fSDevin Teskedialog_last_chance()
377cd88b886SDevin Teske{
378cd88b886SDevin Teske	local title="$DIALOG_TITLE"
379cd88b886SDevin Teske	local btitle="$DIALOG_BACKTITLE"
380bc4a673fSDevin Teske	local prompt # Calculated below
381bc4a673fSDevin Teske	local hline="$hline_arrows_tab_enter"
382cd88b886SDevin Teske
383bc4a673fSDevin Teske	local height=8 width=50 prefix="   "
384bc4a673fSDevin Teske	local plen=${#prefix} list= line=
385bc4a673fSDevin Teske	local max_width=$(( $width - 3 - $plen ))
386bc4a673fSDevin Teske
387bc4a673fSDevin Teske	local yes no defaultno extra_args format
388bc4a673fSDevin Teske	if [ "$USE_XDIALOG" ]; then
389bc4a673fSDevin Teske		yes=ok no=cancel defaultno=default-no
390bc4a673fSDevin Teske		extra_args="--wrap --left"
391bc4a673fSDevin Teske		format="$msg_last_chance_are_you_sure"
392bc4a673fSDevin Teske	else
393bc4a673fSDevin Teske		yes=yes no=no defaultno=defaultno
394bc4a673fSDevin Teske		extra_args="--colors --cr-wrap"
395bc4a673fSDevin Teske		format="$msg_last_chance_are_you_sure_color"
396bc4a673fSDevin Teske	fi
397bc4a673fSDevin Teske
398bc4a673fSDevin Teske	local disk line_width
399bc4a673fSDevin Teske	for disk in $*; do
400bc4a673fSDevin Teske		if [ "$line" ]; then
401bc4a673fSDevin Teske			line_width=${#line}
402bc4a673fSDevin Teske		else
403bc4a673fSDevin Teske			line_width=$plen
404bc4a673fSDevin Teske		fi
405bc4a673fSDevin Teske		line_width=$(( $line_width + 1 + ${#disk} ))
406bc4a673fSDevin Teske		# Add newline before disk if it would exceed max_width
407bc4a673fSDevin Teske		if [ $line_width -gt $max_width ]; then
408bc4a673fSDevin Teske			list="$list$line\n"
409bc4a673fSDevin Teske			line="$prefix"
410bc4a673fSDevin Teske			height=$(( $height + 1 ))
411bc4a673fSDevin Teske		fi
412bc4a673fSDevin Teske		# Add the disk to the list
413bc4a673fSDevin Teske		line="$line $disk"
414bc4a673fSDevin Teske	done
415bc4a673fSDevin Teske	# Append the left-overs
416bc4a673fSDevin Teske	if [ "${line#$prefix}" ]; then
417bc4a673fSDevin Teske		list="$list$line"
418bc4a673fSDevin Teske		height=$(( $height + 1 ))
419bc4a673fSDevin Teske	fi
420bc4a673fSDevin Teske
421bc4a673fSDevin Teske	# Add height for Xdialog(1)
422bc4a673fSDevin Teske	[ "$USE_XDIALOG" ] && height=$(( $height + $height / 5 + 3 ))
423bc4a673fSDevin Teske
424bc4a673fSDevin Teske	prompt=$( printf "$format" "$list" )
425bc4a673fSDevin Teske	f_dprintf "%s: Last Chance!" "$0"
426bc4a673fSDevin Teske	$DIALOG \
427bc4a673fSDevin Teske		--title "$title"        \
428bc4a673fSDevin Teske		--backtitle "$btitle"   \
429bc4a673fSDevin Teske		--hline "$hline"        \
430bc4a673fSDevin Teske		--$defaultno            \
431bc4a673fSDevin Teske		--$yes-label "$msg_yes" \
432bc4a673fSDevin Teske		--$no-label "$msg_no"   \
433bc4a673fSDevin Teske		$extra_args             \
434bc4a673fSDevin Teske		--yesno "$prompt" $height $width
435bc4a673fSDevin Teske}
436bc4a673fSDevin Teske
437bc4a673fSDevin Teske# dialog_menu_layout
438cd88b886SDevin Teske#
439bc4a673fSDevin Teske# Configure Virtual Device type and disks to use for the ZFS boot pool. User
440bc4a673fSDevin Teske# must select enough disks to satisfy the chosen vdev type.
441cd88b886SDevin Teske#
442bc4a673fSDevin Teskedialog_menu_layout()
443bc4a673fSDevin Teske{
444bc4a673fSDevin Teske	local funcname=dialog_menu_layout
445bc4a673fSDevin Teske	local title="$DIALOG_TITLE"
446bc4a673fSDevin Teske	local btitle="$DIALOG_BACKTITLE"
447bc4a673fSDevin Teske	local vdev_prompt="$msg_select_virtual_device_type"
448bc4a673fSDevin Teske	local disk_prompt="$msg_please_select_one_or_more_disks"
449bc4a673fSDevin Teske	local vdev_menu_list="
450bc4a673fSDevin Teske		'stripe' '$msg_stripe_desc' '$msg_stripe_help'
451bc4a673fSDevin Teske		'mirror' '$msg_mirror_desc' '$msg_mirror_help'
452bc4a673fSDevin Teske		'raidz1' '$msg_raidz1_desc' '$msg_raidz1_help'
453bc4a673fSDevin Teske		'raidz2' '$msg_raidz2_desc' '$msg_raidz2_help'
454bc4a673fSDevin Teske		'raidz3' '$msg_raidz3_desc' '$msg_raidz3_help'
455bc4a673fSDevin Teske	" # END-QUOTE
456bc4a673fSDevin Teske	local disk_check_list= # Calculated below
457bc4a673fSDevin Teske	local vdev_hline="$hline_arrows_tab_enter"
458bc4a673fSDevin Teske	local disk_hline="$hline_arrows_space_tab_enter"
459bc4a673fSDevin Teske
460bc4a673fSDevin Teske	# Warn the user if vdev type is not valid
461bc4a673fSDevin Teske	case "$ZFSBOOT_VDEV_TYPE" in
462bc4a673fSDevin Teske	stripe|mirror|raidz1|raidz2|raidz3) : known good ;;
463bc4a673fSDevin Teske	*)
464bc4a673fSDevin Teske		f_dprintf "%s: Invalid virtual device type \`%s'" \
465bc4a673fSDevin Teske			  $funcname "$ZFSBOOT_VDEV_TYPE"
466bc4a673fSDevin Teske		f_show_err "$msg_invalid_virtual_device_type" \
467bc4a673fSDevin Teske			   "$ZFSBOOT_VDEV_TYPE"
468bc4a673fSDevin Teske		f_interactive || return $FAILURE
469bc4a673fSDevin Teske	esac
470bc4a673fSDevin Teske
471bc4a673fSDevin Teske	# Calculate size of vdev menu once only
472bc4a673fSDevin Teske	local vheight vwidth vrows
473bc4a673fSDevin Teske	eval f_dialog_menu_with_help_size vheight vwidth vrows \
474bc4a673fSDevin Teske		\"\$title\" \"\$btitle\" \"\$vdev_prompt\" \"\$vdev_hline\" \
475bc4a673fSDevin Teske		$vdev_menu_list
476bc4a673fSDevin Teske
477bc4a673fSDevin Teske	# Get a list of probed disk devices
478bc4a673fSDevin Teske	local disks=
479a88393ceSDevin Teske	debug= f_device_find "" $DEVICE_TYPE_DISK disks
480a88393ceSDevin Teske
481a88393ceSDevin Teske	# Prune out mounted md(4) devices that may be part of the boot process
482a88393ceSDevin Teske	local disk name new_list=
483a88393ceSDevin Teske	for disk in $disks; do
484a88393ceSDevin Teske		debug= $disk get name name
485a88393ceSDevin Teske		case "$name" in
486a88393ceSDevin Teske		md[0-9]*) f_mounted -b "/dev/$name" && continue ;;
487a88393ceSDevin Teske		esac
488a88393ceSDevin Teske		new_list="$new_list $disk"
489a88393ceSDevin Teske	done
490a88393ceSDevin Teske	disks="${new_list# }"
491a88393ceSDevin Teske
492a88393ceSDevin Teske	# Debugging
493a88393ceSDevin Teske	if [ "$debug" ]; then
494a88393ceSDevin Teske		local disk_names=
495a88393ceSDevin Teske		for disk in $disks; do
496a88393ceSDevin Teske			debug= $disk get name name
497a88393ceSDevin Teske			disk_names="$disk_names $name"
498a88393ceSDevin Teske		done
499a88393ceSDevin Teske		f_dprintf "$funcname: disks=[%s]" "${disk_names# }"
500a88393ceSDevin Teske	fi
501a88393ceSDevin Teske
502cd88b886SDevin Teske	if [ ! "$disks" ]; then
503bc4a673fSDevin Teske		f_dprintf "No disk(s) present to configure"
504bc4a673fSDevin Teske		f_show_err "$msg_no_disks_present_to_configure"
505cd88b886SDevin Teske		return $FAILURE
506cd88b886SDevin Teske	fi
507cd88b886SDevin Teske
508cd88b886SDevin Teske	# Lets sort the disks array to be more user friendly
509a88393ceSDevin Teske	f_device_sort_by name disks disks
510cd88b886SDevin Teske
511cd88b886SDevin Teske	#
512bc4a673fSDevin Teske	# Operate in a loop so we can (if interactive) repeat if not enough
513bc4a673fSDevin Teske	# disks are selected to satisfy the chosen vdev type or user wants to
514bc4a673fSDevin Teske	# back-up to the previous menu.
515cd88b886SDevin Teske	#
516a88393ceSDevin Teske	local vardisk ndisks onoff selections vdev_choice breakout device
517a88393ceSDevin Teske	local valid_disks all_valid want_disks desc height width rows
518bc4a673fSDevin Teske	while :; do
519cd88b886SDevin Teske		#
520bc4a673fSDevin Teske		# Confirm the vdev type that was selected
521cd88b886SDevin Teske		#
522bc4a673fSDevin Teske		if f_interactive && [ "$ZFSBOOT_CONFIRM_LAYOUT" ]; then
523bc4a673fSDevin Teske			vdev_choice=$( eval $DIALOG \
524cd88b886SDevin Teske				--title \"\$title\"              \
525cd88b886SDevin Teske				--backtitle \"\$btitle\"         \
526bc4a673fSDevin Teske				--hline \"\$vdev_hline\"         \
527cd88b886SDevin Teske				--ok-label \"\$msg_ok\"          \
528cd88b886SDevin Teske				--cancel-label \"\$msg_cancel\"  \
529cd88b886SDevin Teske				--item-help                      \
530bc4a673fSDevin Teske				--default-item \"\$ZFSBOOT_VDEV_TYPE\" \
531bc4a673fSDevin Teske				--menu \"\$vdev_prompt\"         \
532bc4a673fSDevin Teske				$vheight $vwidth $vrows          \
533bc4a673fSDevin Teske				$vdev_menu_list                  \
534cd88b886SDevin Teske				2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
535bc4a673fSDevin Teske			) || return $?
536bc4a673fSDevin Teske				# Exit if user pressed ESC or chose Cancel/No
537bc4a673fSDevin Teske			f_dialog_data_sanitize vdev_choice
538cd88b886SDevin Teske
539bc4a673fSDevin Teske			ZFSBOOT_VDEV_TYPE="$vdev_choice"
540bc4a673fSDevin Teske			f_dprintf "$funcname: ZFSBOOT_VDEV_TYPE=[%s]" \
541bc4a673fSDevin Teske			          "$ZFSBOOT_VDEV_TYPE"
542bc4a673fSDevin Teske		fi
543bc4a673fSDevin Teske
544bc4a673fSDevin Teske		# Determine the number of disks needed for this vdev type
545a88393ceSDevin Teske		want_disks=0
546bc4a673fSDevin Teske		case "$ZFSBOOT_VDEV_TYPE" in
547cd88b886SDevin Teske		stripe) want_disks=1 ;;
548cd88b886SDevin Teske		mirror) want_disks=2 ;;
549cd88b886SDevin Teske		raidz1) want_disks=3 ;;
550cd88b886SDevin Teske		raidz2) want_disks=4 ;;
551cd88b886SDevin Teske		raidz3) want_disks=5 ;;
552cd88b886SDevin Teske		esac
553bc4a673fSDevin Teske
554a88393ceSDevin Teske		#
555bc4a673fSDevin Teske		# Warn the user if any scripted disks are invalid
556a88393ceSDevin Teske		#
557a88393ceSDevin Teske		valid_disks= all_valid=${ZFSBOOT_DISKS:+1} # optimism
558bc4a673fSDevin Teske		for disk in $ZFSBOOT_DISKS; do
559a88393ceSDevin Teske			if debug= f_device_find -1 \
560a88393ceSDevin Teske				$disk $DEVICE_TYPE_DISK device
561a88393ceSDevin Teske			then
562bc4a673fSDevin Teske				valid_disks="$valid_disks $disk"
563bc4a673fSDevin Teske				continue
564cd88b886SDevin Teske			fi
565bc4a673fSDevin Teske			f_dprintf "$funcname: \`%s' is not a real disk" "$disk"
566bc4a673fSDevin Teske			all_valid=
567bc4a673fSDevin Teske		done
568bc4a673fSDevin Teske		if [ ! "$all_valid" ]; then
569bc4a673fSDevin Teske			if [ "$ZFSBOOT_DISKS" ]; then
570bc4a673fSDevin Teske				f_show_err \
571bc4a673fSDevin Teske				    "$msg_missing_one_or_more_scripted_disks"
572bc4a673fSDevin Teske			else
573bc4a673fSDevin Teske				f_dprintf "No disks selected."
574bc4a673fSDevin Teske				f_interactive ||
575bc4a673fSDevin Teske					f_show_err "$msg_no_disks_selected"
576bc4a673fSDevin Teske			fi
577bc4a673fSDevin Teske			f_interactive || return $FAILURE
578bc4a673fSDevin Teske		fi
579bc4a673fSDevin Teske		ZFSBOOT_DISKS="${valid_disks# }"
580bc4a673fSDevin Teske
581bc4a673fSDevin Teske		#
582bc4a673fSDevin Teske		# Short-circuit if we're running non-interactively
583bc4a673fSDevin Teske		#
584bc4a673fSDevin Teske		if ! f_interactive || [ ! "$ZFSBOOT_CONFIRM_LAYOUT" ]; then
585a88393ceSDevin Teske			f_count ndisks $ZFSBOOT_DISKS
586bc4a673fSDevin Teske			[ $ndisks -ge $want_disks ] && break # to success
587bc4a673fSDevin Teske
588bc4a673fSDevin Teske			# Not enough disks selected
589bc4a673fSDevin Teske			f_dprintf "$funcname: %s: %s (%u < %u minimum)" \
590bc4a673fSDevin Teske				  "$ZFSBOOT_VDEV_TYPE" \
591bc4a673fSDevin Teske			          "Not enough disks selected." \
592bc4a673fSDevin Teske				  $ndisks $want_disks
593bc4a673fSDevin Teske			f_interactive || return $FAILURE
594bc4a673fSDevin Teske			msg_yes="$msg_change_selection" msg_no="$msg_cancel" \
595bc4a673fSDevin Teske				f_yesno "%s: $msg_not_enough_disks_selected" \
596bc4a673fSDevin Teske				"$ZFSBOOT_VDEV_TYPE" $ndisks $want_disks ||
597bc4a673fSDevin Teske				return $FAILURE
598bc4a673fSDevin Teske		fi
599bc4a673fSDevin Teske
600bc4a673fSDevin Teske		#
601bc4a673fSDevin Teske		# Confirm the disks that were selected
602bc4a673fSDevin Teske		# Loop until the user cancels or selects enough disks
603bc4a673fSDevin Teske		#
604a88393ceSDevin Teske		breakout=
605bc4a673fSDevin Teske		while :; do
606bc4a673fSDevin Teske			# Loop over list of available disks, resetting state
607a88393ceSDevin Teske			for disk in $disks; do
608a88393ceSDevin Teske				f_isset _${disk}_status && _${disk}_status=
609a88393ceSDevin Teske			done
610bc4a673fSDevin Teske
611bc4a673fSDevin Teske			# Loop over list of selected disks and create temporary
612bc4a673fSDevin Teske			# locals to map statuses onto up-to-date list of disks
613bc4a673fSDevin Teske			for disk in $ZFSBOOT_DISKS; do
614a88393ceSDevin Teske				debug= f_device_find -1 \
615a88393ceSDevin Teske					$disk $DEVICE_TYPE_DISK disk
616a88393ceSDevin Teske				f_isset _${disk}_status ||
617a88393ceSDevin Teske					local _${disk}_status
618a88393ceSDevin Teske				_${disk}_status=on
619bc4a673fSDevin Teske			done
620bc4a673fSDevin Teske
621bc4a673fSDevin Teske			# Create the checklist menu of discovered disk devices
622bc4a673fSDevin Teske			disk_check_list=
623bc4a673fSDevin Teske			for disk in $disks; do
624a88393ceSDevin Teske				desc=
625a88393ceSDevin Teske				$disk get name name
626a88393ceSDevin Teske				$disk get desc desc
627bc4a673fSDevin Teske				f_shell_escape "$desc" desc
628bc4a673fSDevin Teske				f_getvar _${disk}_status:-off onoff
629bc4a673fSDevin Teske				disk_check_list="$disk_check_list
630a88393ceSDevin Teske					$name '$desc' $onoff"
631bc4a673fSDevin Teske			done
632bc4a673fSDevin Teske
633bc4a673fSDevin Teske			eval f_dialog_checklist_size height width rows \
634bc4a673fSDevin Teske				\"\$title\" \"\$btitle\" \"\$prompt\" \
635bc4a673fSDevin Teske				\"\$hline\" $disk_check_list
636bc4a673fSDevin Teske
637bc4a673fSDevin Teske			selections=$( eval $DIALOG \
638bc4a673fSDevin Teske				--title \"\$DIALOG_TITLE\"         \
639bc4a673fSDevin Teske				--backtitle \"\$DIALOG_BACKTITLE\" \
640751952aeSDevin Teske				--separate-output                  \
641bc4a673fSDevin Teske				--hline \"\$hline\"                \
642bc4a673fSDevin Teske				--ok-label \"\$msg_ok\"            \
643bc4a673fSDevin Teske				--cancel-label \"\$msg_back\"      \
644bc4a673fSDevin Teske				--checklist \"\$prompt\"           \
645bc4a673fSDevin Teske				$height $width $rows               \
646bc4a673fSDevin Teske				$disk_check_list                   \
647bc4a673fSDevin Teske				2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
648bc4a673fSDevin Teske			) || break
649bc4a673fSDevin Teske				# Loop if user pressed ESC or chose Cancel/No
650bc4a673fSDevin Teske			f_dialog_data_sanitize selections
651bc4a673fSDevin Teske
652bc4a673fSDevin Teske			ZFSBOOT_DISKS="$selections"
653bc4a673fSDevin Teske			f_dprintf "$funcname: ZFSBOOT_DISKS=[%s]" \
654bc4a673fSDevin Teske			          "$ZFSBOOT_DISKS"
655bc4a673fSDevin Teske
656a88393ceSDevin Teske			f_count ndisks $ZFSBOOT_DISKS
657bc4a673fSDevin Teske			[ $ndisks -ge $want_disks ] &&
658bc4a673fSDevin Teske				breakout=break && break
659bc4a673fSDevin Teske
660bc4a673fSDevin Teske			# Not enough disks selected
661bc4a673fSDevin Teske			f_dprintf "$funcname: %s: %s (%u < %u minimum)" \
662bc4a673fSDevin Teske				  "$ZFSBOOT_VDEV_TYPE" \
663bc4a673fSDevin Teske			          "Not enough disks selected." \
664bc4a673fSDevin Teske			          $ndisks $want_disks
665bc4a673fSDevin Teske			msg_yes="$msg_change_selection" msg_no="$msg_cancel" \
666bc4a673fSDevin Teske				f_yesno "%s: $msg_not_enough_disks_selected" \
667bc4a673fSDevin Teske				"$ZFSBOOT_VDEV_TYPE" $ndisks $want_disks ||
668bc4a673fSDevin Teske				break
669bc4a673fSDevin Teske		done
670bc4a673fSDevin Teske		[ "$breakout" = "break" ] && break
671bc4a673fSDevin Teske		[ "$ZFSBOOT_CONFIRM_LAYOUT" ] || return $FAILURE
672bc4a673fSDevin Teske	done
673bc4a673fSDevin Teske
674bc4a673fSDevin Teske	return $DIALOG_OK
675cd88b886SDevin Teske}
676cd88b886SDevin Teske
677cd88b886SDevin Teske# zfs_create_diskpart $disk $index
678cd88b886SDevin Teske#
679cd88b886SDevin Teske# For each block device to be used in the zpool, rather than just create the
680cd88b886SDevin Teske# zpool with the raw block devices (e.g., da0, da1, etc.) we create partitions
681cd88b886SDevin Teske# so we can have some real swap. This also provides wiggle room incase your
682cd88b886SDevin Teske# replacement drivers do not have the exact same sector counts.
683cd88b886SDevin Teske#
6847cae6aabSDevin Teske# NOTE: $swapsize and $bootsize should be defined by the calling function.
685a622223fSDevin Teske# NOTE: Sets $bootpart and $targetpart for the calling function.
686cd88b886SDevin Teske#
687cd88b886SDevin Teskezfs_create_diskpart()
688cd88b886SDevin Teske{
689cd88b886SDevin Teske	local funcname=zfs_create_diskpart
690bc4a673fSDevin Teske	local disk="$1" index="$2"
691cd88b886SDevin Teske
692cd88b886SDevin Teske	# Check arguments
693bc4a673fSDevin Teske	if [ ! "$disk" ]; then
694bc4a673fSDevin Teske		f_dprintf "$funcname: NULL disk argument"
695bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
696bc4a673fSDevin Teske			f_show_err "$msg_null_disk_argument"
697bc4a673fSDevin Teske		return $FAILURE
698bc4a673fSDevin Teske	fi
699bc4a673fSDevin Teske	if [ "${disk#*[$IFS]}" != "$disk" ]; then
700bc4a673fSDevin Teske		f_dprintf "$funcname: Invalid disk argument \`%s'" "$disk"
701bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
702bc4a673fSDevin Teske			f_show_err "$msg_invalid_disk_argument" "$disk"
703bc4a673fSDevin Teske		return $FAILURE
704bc4a673fSDevin Teske	fi
705bc4a673fSDevin Teske	if [ ! "$index" ]; then
706bc4a673fSDevin Teske		f_dprintf "$funcname: NULL index argument"
707bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
708bc4a673fSDevin Teske			f_show_err "$msg_null_index_argument"
709bc4a673fSDevin Teske		return $FAILURE
710bc4a673fSDevin Teske	fi
711bc4a673fSDevin Teske	if ! f_isinteger "$index"; then
712bc4a673fSDevin Teske		f_dprintf "$funcname: Invalid index argument \`%s'" "$index"
713bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
714bc4a673fSDevin Teske			f_show_err "$msg_invalid_index_argument" "$index"
715bc4a673fSDevin Teske		return $FAILURE
716bc4a673fSDevin Teske	fi
717bc4a673fSDevin Teske	f_dprintf "$funcname: disk=[%s] index=[%s]" "$disk" "$index"
718cd88b886SDevin Teske
719d4d729e4SDevin Teske	# Check for unknown partition scheme before proceeding further
720d4d729e4SDevin Teske	case "$ZFSBOOT_PARTITION_SCHEME" in
721d4d729e4SDevin Teske	""|MBR|GPT) : known good ;;
722d4d729e4SDevin Teske	*)
723d4d729e4SDevin Teske		f_dprintf "$funcname: %s is an unsupported partition scheme" \
724d4d729e4SDevin Teske		          "$ZFSBOOT_PARTITION_SCHEME"
725d4d729e4SDevin Teske		msg_error="$msg_error: $funcname" f_show_err \
726d4d729e4SDevin Teske			"$msg_unsupported_partition_scheme" \
727d4d729e4SDevin Teske			"$ZFSBOOT_PARTITION_SCHEME"
728d4d729e4SDevin Teske		return $FAILURE
729d4d729e4SDevin Teske	esac
730d4d729e4SDevin Teske
731cd88b886SDevin Teske	#
732cd88b886SDevin Teske	# Destroy whatever partition layout is currently on disk.
733cd88b886SDevin Teske	# NOTE: `-F' required to destroy if partitions still exist.
734cd88b886SDevin Teske	# NOTE: Failure is ok here, blank disk will have nothing to destroy.
735cd88b886SDevin Teske	#
736bc4a673fSDevin Teske	f_dprintf "$funcname: Destroying all data/layouts on \`%s'..." "$disk"
737a622223fSDevin Teske	f_eval_catch -d $funcname gpart "$GPART_DESTROY_F" $disk
738a622223fSDevin Teske	f_eval_catch -d $funcname graid "$GRAID_DELETE" $disk
739a622223fSDevin Teske	f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" /dev/$disk
740cd88b886SDevin Teske
741cd88b886SDevin Teske	# Make doubly-sure backup GPT is destroyed
742a622223fSDevin Teske	f_eval_catch -d $funcname gpart "$GPART_CREATE" gpt $disk
743a622223fSDevin Teske	f_eval_catch -d $funcname gpart "$GPART_DESTROY_F" $disk
744cd88b886SDevin Teske
7457cae6aabSDevin Teske	#
7467cae6aabSDevin Teske	# Enable boot pool if encryption is desired
7477cae6aabSDevin Teske	#
7487cae6aabSDevin Teske	[ "$ZFSBOOT_GELI_ENCRYPTION" ] && ZFSBOOT_BOOT_POOL=1
749cd88b886SDevin Teske
750cd88b886SDevin Teske	#
751cd88b886SDevin Teske	# Lay down the desired type of partition scheme
752cd88b886SDevin Teske	#
753cd88b886SDevin Teske	local setsize mbrindex
754cd88b886SDevin Teske	case "$ZFSBOOT_PARTITION_SCHEME" in
755bc4a673fSDevin Teske	""|GPT) f_dprintf "$funcname: Creating GPT layout..."
756cd88b886SDevin Teske		#
757cd88b886SDevin Teske		# 1. Create GPT layout using labels
758cd88b886SDevin Teske		#
759a622223fSDevin Teske		f_eval_catch $funcname gpart "$GPART_CREATE" gpt $disk ||
760bc4a673fSDevin Teske		             return $FAILURE
761cd88b886SDevin Teske
762cd88b886SDevin Teske		#
763cd88b886SDevin Teske		# 2. Add small freebsd-boot partition labeled `boot#'
764cd88b886SDevin Teske		#
765bc4a673fSDevin Teske		f_eval_catch $funcname gpart "$GPART_ADD_LABEL_WITH_SIZE" \
766a622223fSDevin Teske		             gptboot$index freebsd-boot 512k $disk ||
767cd88b886SDevin Teske		             return $FAILURE
768bc4a673fSDevin Teske		f_eval_catch $funcname gpart "$GPART_BOOTCODE_PART" \
769a622223fSDevin Teske		             /boot/pmbr /boot/gptzfsboot 1 $disk ||
770cd88b886SDevin Teske		             return $FAILURE
771cd88b886SDevin Teske
772a622223fSDevin Teske		# NB: zpool will use the `zfs#' GPT labels
773*2875e59fSOllivier Robert		bootpart=p2 swappart=p2 targetpart=p2
7747cae6aabSDevin Teske		[ ${swapsize:-0} -gt 0 ] && targetpart=p3
775cd88b886SDevin Teske
7767cae6aabSDevin Teske		#
7777cae6aabSDevin Teske		# Prepare boot pool if enabled (e.g., for geli(8))
7787cae6aabSDevin Teske		#
7797cae6aabSDevin Teske		if [ "$ZFSBOOT_BOOT_POOL" ]; then
780*2875e59fSOllivier Robert			bootpart=p2 swappart=p3 targetpart=p3
7817cae6aabSDevin Teske			[ ${swapsize:-0} -gt 0 ] && targetpart=p4
782bc4a673fSDevin Teske			f_eval_catch $funcname gpart \
783a622223fSDevin Teske			             "$GPART_ADD_LABEL_WITH_SIZE" boot$index \
7847cae6aabSDevin Teske			             freebsd-zfs ${bootsize}b $disk ||
785bc4a673fSDevin Teske			             return $FAILURE
7867cae6aabSDevin Teske			# Pedantically nuke any old labels
787a622223fSDevin Teske			f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
788a622223fSDevin Teske			                /dev/$disk$bootpart
7897cae6aabSDevin Teske			if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
7907cae6aabSDevin Teske				# Pedantically detach targetpart for later
7917cae6aabSDevin Teske				f_eval_catch -d $funcname geli \
7927cae6aabSDevin Teske				                "$GELI_DETACH_F" \
793a622223fSDevin Teske				                /dev/$disk$targetpart
794cd88b886SDevin Teske			fi
7957cae6aabSDevin Teske		fi
796cd88b886SDevin Teske
797cd88b886SDevin Teske		#
7987cae6aabSDevin Teske		# 3. Add freebsd-swap partition labeled `swap#'
799cd88b886SDevin Teske		#
800bc4a673fSDevin Teske		if [ ${swapsize:-0} -gt 0 ]; then
801bc4a673fSDevin Teske			f_eval_catch $funcname gpart \
8027cae6aabSDevin Teske			             "$GPART_ADD_LABEL_WITH_SIZE" swap$index \
8037cae6aabSDevin Teske			             freebsd-swap ${swapsize}b $disk ||
804cd88b886SDevin Teske			             return $FAILURE
805cf9c3e56SDevin Teske			# Pedantically nuke any old labels on the swap
806cf9c3e56SDevin Teske			f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
807*2875e59fSOllivier Robert			                /dev/$disk$swappart
808cd88b886SDevin Teske		fi
8097cae6aabSDevin Teske
8107cae6aabSDevin Teske		#
8117cae6aabSDevin Teske		# 4. Add freebsd-zfs partition labeled `zfs#' for zroot
8127cae6aabSDevin Teske		#
8137cae6aabSDevin Teske		f_eval_catch $funcname gpart "$GPART_ADD_LABEL" \
8147cae6aabSDevin Teske		             zfs$index freebsd-zfs $disk || return $FAILURE
8157cae6aabSDevin Teske		f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
8167cae6aabSDevin Teske		                /dev/$disk$targetpart
817cd88b886SDevin Teske		;;
818cd88b886SDevin Teske
819bc4a673fSDevin Teske	MBR) f_dprintf "$funcname: Creating MBR layout..."
820cd88b886SDevin Teske		#
821cd88b886SDevin Teske		# 1. Create MBR layout (no labels)
822cd88b886SDevin Teske		#
823a622223fSDevin Teske		f_eval_catch $funcname gpart "$GPART_CREATE" mbr $disk ||
824bc4a673fSDevin Teske		             return $FAILURE
8252925848dSDevin Teske		f_eval_catch $funcname gpart "$GPART_BOOTCODE" /boot/mbr \
826a622223fSDevin Teske		             $disk || return $FAILURE
827cd88b886SDevin Teske
828cd88b886SDevin Teske		#
829cd88b886SDevin Teske		# 2. Add freebsd slice with all available space
830cd88b886SDevin Teske		#
831a622223fSDevin Teske		f_eval_catch $funcname gpart "$GPART_ADD" freebsd $disk ||
832bc4a673fSDevin Teske		             return $FAILURE
833a622223fSDevin Teske		f_eval_catch $funcname gpart "$GPART_SET_ACTIVE" 1 $disk ||
834bc4a673fSDevin Teske		             return $FAILURE
835a622223fSDevin Teske		# Pedantically nuke any old labels
836a622223fSDevin Teske		f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
837a622223fSDevin Teske		                /dev/${disk}s1
838a622223fSDevin Teske		# Pedantically nuke any old scheme
839a622223fSDevin Teske		f_eval_catch -d $funcname gpart "$GPART_DESTROY_F" ${disk}s1
840cd88b886SDevin Teske
841cd88b886SDevin Teske		#
842bc4a673fSDevin Teske		# 3. Write BSD scheme to the freebsd slice
843cd88b886SDevin Teske		#
844a622223fSDevin Teske		f_eval_catch $funcname gpart "$GPART_CREATE" BSD ${disk}s1 ||
845bc4a673fSDevin Teske		             return $FAILURE
846cd88b886SDevin Teske
8477cae6aabSDevin Teske		# NB: zpool will use s1a (no labels)
848*2875e59fSOllivier Robert		bootpart=s1a swappart=s1b targetpart=s1d mbrindex=4
849cd88b886SDevin Teske
8507cae6aabSDevin Teske		#
8517cae6aabSDevin Teske		# Always prepare a boot pool on MBR
8527cae6aabSDevin Teske		#
8537cae6aabSDevin Teske		ZFSBOOT_BOOT_POOL=1
854bc4a673fSDevin Teske		f_eval_catch $funcname gpart \
855bc4a673fSDevin Teske		             "$GPART_ADD_INDEX_WITH_SIZE" \
8567cae6aabSDevin Teske		             1 freebsd-zfs ${bootsize}b ${disk}s1 ||
857bc4a673fSDevin Teske		             return $FAILURE
858a622223fSDevin Teske		# Pedantically nuke any old labels
859a622223fSDevin Teske		f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
860a622223fSDevin Teske		                /dev/$disk$bootpart
8617cae6aabSDevin Teske		if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
862a622223fSDevin Teske			# Pedantically detach targetpart for later
863a622223fSDevin Teske			f_eval_catch -d $funcname geli \
864a622223fSDevin Teske			                "$GELI_DETACH_F" \
865a622223fSDevin Teske					/dev/$disk$targetpart
866cd88b886SDevin Teske		fi
867cd88b886SDevin Teske
868cd88b886SDevin Teske		#
8697cae6aabSDevin Teske		# 4. Add freebsd-swap partition
870cd88b886SDevin Teske		#
871bc4a673fSDevin Teske		if [ ${swapsize:-0} -gt 0 ]; then
872bc4a673fSDevin Teske			f_eval_catch $funcname gpart \
8737cae6aabSDevin Teske			             "$GPART_ADD_INDEX_WITH_SIZE" 2 \
8747cae6aabSDevin Teske			             freebsd-swap ${swapsize}b ${disk}s1 ||
875cd88b886SDevin Teske			             return $FAILURE
876cf9c3e56SDevin Teske			# Pedantically nuke any old labels on the swap
877cf9c3e56SDevin Teske			f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
878cf9c3e56SDevin Teske			                /dev/${disk}s1b
879cd88b886SDevin Teske		fi
8807cae6aabSDevin Teske
8817cae6aabSDevin Teske		#
8827cae6aabSDevin Teske		# 5. Add freebsd-zfs partition for zroot
8837cae6aabSDevin Teske		#
8847cae6aabSDevin Teske		f_eval_catch $funcname gpart "$GPART_ADD_INDEX" \
8857cae6aabSDevin Teske			     $mbrindex freebsd-zfs ${disk}s1 || return $FAILURE
8867cae6aabSDevin Teske		f_eval_catch -d $funcname zpool "$ZPOOL_LABELCLEAR_F" \
8877cae6aabSDevin Teske				/dev/$disk$targetpart # Pedantic
8887cae6aabSDevin Teske		f_eval_catch $funcname dd "$DD_WITH_OPTIONS" \
8897cae6aabSDevin Teske			     /boot/zfsboot /dev/${disk}s1 count=1 ||
8907cae6aabSDevin Teske			     return $FAILURE
891cd88b886SDevin Teske		;;
892cd88b886SDevin Teske
893cd88b886SDevin Teske	esac # $ZFSBOOT_PARTITION_SCHEME
894cd88b886SDevin Teske
895*2875e59fSOllivier Robert	# Update fstab(5)
896*2875e59fSOllivier Robert	if [ "$isswapmirror" ]; then
897*2875e59fSOllivier Robert		# This is not the first disk in the mirror, do nothing
898*2875e59fSOllivier Robert	elif [ "$ZFSBOOT_SWAP_ENCRYPTION" -a "$ZFSBOOT_SWAP_MIRROR" ]; then
899*2875e59fSOllivier Robert		f_eval_catch $funcname printf "$PRINTF_FSTAB" \
900*2875e59fSOllivier Robert		             /dev/mirror/swap.eli none swap sw 0 0 \
901*2875e59fSOllivier Robert		             $BSDINSTALL_TMPETC/fstab ||
902*2875e59fSOllivier Robert		             return $FAILURE
903*2875e59fSOllivier Robert		isswapmirror=1
904*2875e59fSOllivier Robert	elif [ "$ZFSBOOT_SWAP_MIRROR" ]; then
905*2875e59fSOllivier Robert		f_eval_catch $funcname printf "$PRINTF_FSTAB" \
906*2875e59fSOllivier Robert		             /dev/mirror/swap none swap sw 0 0 \
907*2875e59fSOllivier Robert		             $BSDINSTALL_TMPETC/fstab ||
908*2875e59fSOllivier Robert		             return $FAILURE
909*2875e59fSOllivier Robert		isswapmirror=1
910*2875e59fSOllivier Robert	elif [ "$ZFSBOOT_SWAP_ENCRYPTION" ]; then
911*2875e59fSOllivier Robert		f_eval_catch $funcname printf "$PRINTF_FSTAB" \
912*2875e59fSOllivier Robert		             /dev/$disk${swappart}.eli none swap sw 0 0 \
913*2875e59fSOllivier Robert		             $BSDINSTALL_TMPETC/fstab ||
914*2875e59fSOllivier Robert		             return $FAILURE
915*2875e59fSOllivier Robert	else
916*2875e59fSOllivier Robert		f_eval_catch $funcname printf "$PRINTF_FSTAB" \
917*2875e59fSOllivier Robert		             /dev/$disk$swappart none swap sw 0 0 \
918*2875e59fSOllivier Robert		             $BSDINSTALL_TMPETC/fstab ||
919*2875e59fSOllivier Robert		             return $FAILURE
920*2875e59fSOllivier Robert	fi
921*2875e59fSOllivier Robert
922cd88b886SDevin Teske	return $SUCCESS
923cd88b886SDevin Teske}
924cd88b886SDevin Teske
925bc4a673fSDevin Teske# zfs_create_boot $poolname $vdev_type $disks ...
926cd88b886SDevin Teske#
927cd88b886SDevin Teske# Creates boot pool and dataset layout. Returns error if something goes wrong.
928cd88b886SDevin Teske# Errors are printed to stderr for collection and display.
929cd88b886SDevin Teske#
930cd88b886SDevin Teskezfs_create_boot()
931cd88b886SDevin Teske{
932cd88b886SDevin Teske	local funcname=zfs_create_boot
9337cae6aabSDevin Teske	local zroot_name="$1"
9347cae6aabSDevin Teske	local zroot_vdevtype="$2"
9357cae6aabSDevin Teske	local zroot_vdevs= # Calculated below
936*2875e59fSOllivier Robert	local swap_devs= # Calculated below
9377cae6aabSDevin Teske	local boot_vdevs= # Used for geli(8) and/or MBR layouts
938a622223fSDevin Teske	shift 2 # poolname vdev_type
939a622223fSDevin Teske	local disks="$*" disk
940*2875e59fSOllivier Robert	local isswapmirror
941*2875e59fSOllivier Robert	local bootpart targetpart swappart # Set by zfs_create_diskpart() below
942cd88b886SDevin Teske
943a622223fSDevin Teske	#
944cd88b886SDevin Teske	# Pedantic checks; should never be seen
945a622223fSDevin Teske	#
9467cae6aabSDevin Teske	if [ ! "$zroot_name" ]; then
947bc4a673fSDevin Teske		f_dprintf "$funcname: NULL poolname"
948bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
949bc4a673fSDevin Teske			f_show_err "$msg_null_poolname"
950cd88b886SDevin Teske		return $FAILURE
951cd88b886SDevin Teske	fi
952cd88b886SDevin Teske	if [ $# -lt 1 ]; then
953bc4a673fSDevin Teske		f_dprintf "$funcname: missing disk arguments"
954bc4a673fSDevin Teske		msg_error="$msg_error: $funcname" \
955bc4a673fSDevin Teske			f_show_err "$msg_missing_disk_arguments"
956cd88b886SDevin Teske		return $FAILURE
957cd88b886SDevin Teske	fi
958bc4a673fSDevin Teske	f_dprintf "$funcname: poolname=[%s] vdev_type=[%s]" \
9597cae6aabSDevin Teske	          "$zroot_name" "$zroot_vdevtype"
960cd88b886SDevin Teske
961a622223fSDevin Teske	#
962cd88b886SDevin Teske	# Initialize fstab(5)
963a622223fSDevin Teske	#
964bc4a673fSDevin Teske	f_dprintf "$funcname: Initializing temporary fstab(5) file..."
965a622223fSDevin Teske	f_eval_catch $funcname sh "$SHELL_TRUNCATE" $BSDINSTALL_TMPETC/fstab ||
966bc4a673fSDevin Teske	             return $FAILURE
967bc4a673fSDevin Teske	f_eval_catch $funcname printf "$PRINTF_FSTAB" \
968cd88b886SDevin Teske	             "# Device" Mountpoint FStype Options Dump "Pass#" \
969a622223fSDevin Teske	             $BSDINSTALL_TMPETC/fstab || return $FAILURE
970cd88b886SDevin Teske
971a622223fSDevin Teske	#
972cd88b886SDevin Teske	# Expand SI units in desired sizes
973a622223fSDevin Teske	#
9747cae6aabSDevin Teske	f_dprintf "$funcname: Expanding supplied size values..."
9757cae6aabSDevin Teske	local swapsize bootsize
976bc4a673fSDevin Teske	if ! f_expand_number "$ZFSBOOT_SWAP_SIZE" swapsize; then
977bc4a673fSDevin Teske		f_dprintf "$funcname: Invalid swap size \`%s'" \
978bc4a673fSDevin Teske		          "$ZFSBOOT_SWAP_SIZE"
979bc4a673fSDevin Teske		f_show_err "$msg_invalid_swap_size" "$ZFSBOOT_SWAP_SIZE"
980bc4a673fSDevin Teske		return $FAILURE
981bc4a673fSDevin Teske	fi
9827cae6aabSDevin Teske	if ! f_expand_number "$ZFSBOOT_BOOT_POOL_SIZE" bootsize; then
9837cae6aabSDevin Teske		f_dprintf "$funcname: Invalid boot pool size \`%s'" \
9847cae6aabSDevin Teske		          "$ZFSBOOT_BOOT_POOL_SIZE"
9857cae6aabSDevin Teske		f_show_err "$msg_invalid_boot_pool_size" \
9867cae6aabSDevin Teske		           "$ZFSBOOT_BOOT_POOL_SIZE"
987bc4a673fSDevin Teske		return $FAILURE
988bc4a673fSDevin Teske	fi
989bc4a673fSDevin Teske	f_dprintf "$funcname: ZFSBOOT_SWAP_SIZE=[%s] swapsize=[%s]" \
990bc4a673fSDevin Teske	          "$ZFSBOOT_SWAP_SIZE" "$swapsize"
9917cae6aabSDevin Teske	f_dprintf "$funcname: ZFSBOOT_BOOT_POOL_SIZE=[%s] bootsize=[%s]" \
9927cae6aabSDevin Teske	          "$ZFSBOOT_BOOT_POOL_SIZE" "$bootsize"
993cd88b886SDevin Teske
994a622223fSDevin Teske	#
995a88393ceSDevin Teske	# Destroy the pool in-case this is our second time 'round (case of
996a88393ceSDevin Teske	# failure and installer presented ``Retry'' option to come back).
997a88393ceSDevin Teske	#
998a88393ceSDevin Teske	# NB: If we don't destroy the pool, later gpart(8) destroy commands
999a88393ceSDevin Teske	# that try to clear existing partitions (see zfs_create_diskpart())
1000a88393ceSDevin Teske	# will fail with a `Device Busy' error, leading to `GEOM exists'.
1001a88393ceSDevin Teske	#
1002a88393ceSDevin Teske	f_eval_catch -d $funcname zpool "$ZPOOL_DESTROY" "$zroot_name"
1003a88393ceSDevin Teske
1004a88393ceSDevin Teske	#
10057cae6aabSDevin Teske	# Prepare the disks and build pool device list(s)
1006a622223fSDevin Teske	#
1007bc4a673fSDevin Teske	f_dprintf "$funcname: Preparing disk partitions for ZFS pool..."
10087cae6aabSDevin Teske	[ "$ZFSBOOT_GNOP_4K_FORCE_ALIGN" ] &&
10097cae6aabSDevin Teske		f_dprintf "$funcname: With 4k alignment using gnop(8)..."
1010cd88b886SDevin Teske	local n=0
1011a622223fSDevin Teske	for disk in $disks; do
1012cd88b886SDevin Teske		zfs_create_diskpart $disk $n || return $FAILURE
1013*2875e59fSOllivier Robert		# Now $bootpart, $targetpart, and $swappart are set (suffix
1014*2875e59fSOllivier Robert		# for $disk)
1015cd88b886SDevin Teske
10167cae6aabSDevin Teske		# Forced 4k alignment support using Geom NOP (see gnop(8))
1017cd88b886SDevin Teske		if [ "$ZFSBOOT_GNOP_4K_FORCE_ALIGN" ]; then
10187cae6aabSDevin Teske			if [ "$ZFSBOOT_BOOT_POOL" ]; then
10197cae6aabSDevin Teske				boot_vdevs="$boot_vdevs $disk$bootpart.nop"
1020bc4a673fSDevin Teske				f_eval_catch $funcname gnop "$GNOP_CREATE" \
10217cae6aabSDevin Teske				             $disk$bootpart || return $FAILURE
1022cd88b886SDevin Teske			fi
10237cae6aabSDevin Teske			# Don't gnop encrypted partition
10247cae6aabSDevin Teske			if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
10257cae6aabSDevin Teske				zroot_vdevs="$zroot_vdevs $disk$targetpart.eli"
1026cd88b886SDevin Teske			else
10277cae6aabSDevin Teske				zroot_vdevs="$zroot_vdevs $disk$targetpart.nop"
10287cae6aabSDevin Teske				f_eval_catch $funcname gnop "$GNOP_CREATE" \
10297cae6aabSDevin Teske					     $disk$targetpart ||
10307cae6aabSDevin Teske				             return $FAILURE
10317cae6aabSDevin Teske			fi
10327cae6aabSDevin Teske		else
10337cae6aabSDevin Teske			if [ "$ZFSBOOT_BOOT_POOL" ]; then
10347cae6aabSDevin Teske				boot_vdevs="$boot_vdevs $disk$bootpart"
10357cae6aabSDevin Teske			fi
10367cae6aabSDevin Teske			zroot_vdevs="$zroot_vdevs $disk$targetpart"
10377cae6aabSDevin Teske		fi
10387cae6aabSDevin Teske
10397cae6aabSDevin Teske		n=$(( $n + 1 ))
10407cae6aabSDevin Teske	done # disks
10417cae6aabSDevin Teske
10427cae6aabSDevin Teske	#
10437cae6aabSDevin Teske	# If we need/want a boot pool, create it
10447cae6aabSDevin Teske	#
10457cae6aabSDevin Teske	if [ "$ZFSBOOT_BOOT_POOL" ]; then
10467cae6aabSDevin Teske		local bootpool_vdevtype= # Calculated below
10477cae6aabSDevin Teske		local bootpool_options= # Calculated below
10487cae6aabSDevin Teske		local bootpool_name="$ZFSBOOT_BOOT_POOL_NAME"
10497cae6aabSDevin Teske		local bootpool="$BSDINSTALL_CHROOT/$bootpool_name"
10507cae6aabSDevin Teske		local zroot_key="${ZFSBOOT_GELI_KEY_FILE#/}"
10517cae6aabSDevin Teske
10527cae6aabSDevin Teske		f_dprintf "$funcname: Setting up boot pool..."
1053cd88b886SDevin Teske		[ "$ZFSBOOT_GELI_ENCRYPTION" ] &&
10547cae6aabSDevin Teske			f_dprintf "$funcname: For encrypted root disk..."
1055cd88b886SDevin Teske
10567cae6aabSDevin Teske		# Create parent directory for boot pool
1057a622223fSDevin Teske		f_eval_catch -d $funcname umount "$UMOUNT" /mnt
1058bc4a673fSDevin Teske		f_eval_catch $funcname mount "$MOUNT_TYPE" tmpfs none \
1059a622223fSDevin Teske		             $BSDINSTALL_CHROOT || return $FAILURE
1060cd88b886SDevin Teske
10617cae6aabSDevin Teske		# Create mirror across the boot partition on all disks
1062a88393ceSDevin Teske		local nvdevs
1063a88393ceSDevin Teske		f_count nvdevs $boot_vdevs
1064a88393ceSDevin Teske		[ $nvdevs -gt 1 ] && bootpool_vdevtype=mirror
1065cd88b886SDevin Teske
10667cae6aabSDevin Teske		bootpool_options="-o altroot=$BSDINSTALL_CHROOT"
10677cae6aabSDevin Teske		bootpool_options="$bootpool_options -m \"/$bootpool_name\" -f"
1068bc4a673fSDevin Teske		f_eval_catch $funcname zpool "$ZPOOL_CREATE_WITH_OPTIONS" \
10697cae6aabSDevin Teske		             "$bootpool_options" "$bootpool_name" \
10704b4b90d5SDevin Teske		             "$bootpool_vdevtype" "$boot_vdevs" ||
10714b4b90d5SDevin Teske		             return $FAILURE
107245df402eSDevin Teske
10737cae6aabSDevin Teske		f_eval_catch $funcname mkdir "$MKDIR_P" "$bootpool/boot" ||
1074cd88b886SDevin Teske		             return $FAILURE
1075cd88b886SDevin Teske
10767cae6aabSDevin Teske		if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
1077cd88b886SDevin Teske			# Generate an encryption key using random(4)
1078bc4a673fSDevin Teske			f_eval_catch $funcname dd "$DD_WITH_OPTIONS" \
10797cae6aabSDevin Teske				     /dev/random "$bootpool/$zroot_key" \
10807cae6aabSDevin Teske			             "bs=4096 count=1" || return $FAILURE
10817cae6aabSDevin Teske		else
10827cae6aabSDevin Teske			# Clean up
10837cae6aabSDevin Teske			f_eval_catch $funcname zfs "$ZFS_UNMOUNT" \
10847cae6aabSDevin Teske			             "$bootpool_name" || return $FAILURE
10857cae6aabSDevin Teske			f_eval_catch -d $funcname umount "$UMOUNT" /mnt # tmpfs
10867cae6aabSDevin Teske		fi
10877cae6aabSDevin Teske
10887cae6aabSDevin Teske	fi
1089cd88b886SDevin Teske
1090a622223fSDevin Teske	#
1091cd88b886SDevin Teske	# Create the geli(8) GEOMS
1092a622223fSDevin Teske	#
10937cae6aabSDevin Teske	if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
10947cae6aabSDevin Teske		# Prompt user for password (twice)
1095bc4a673fSDevin Teske		if ! msg_enter_new_password="$msg_geli_password" \
1096bc4a673fSDevin Teske			f_dialog_input_password
1097bc4a673fSDevin Teske		then
1098bc4a673fSDevin Teske			f_dprintf "$funcname: User cancelled"
1099bc4a673fSDevin Teske			f_show_err "$msg_user_cancelled"
1100bc4a673fSDevin Teske			return $FAILURE
1101bc4a673fSDevin Teske		fi
1102a622223fSDevin Teske
1103a622223fSDevin Teske		# Initialize geli(8) on each of the target partitions
1104bc4a673fSDevin Teske		for disk in $disks; do
1105cd88b886SDevin Teske			f_dialog_info "$msg_geli_setup" \
1106cd88b886SDevin Teske				2>&1 >&$DIALOG_TERMINAL_PASSTHRU_FD
1107a622223fSDevin Teske			if ! echo "$pw_password" | f_eval_catch \
1108a622223fSDevin Teske				$funcname geli "$GELI_PASSWORD_INIT" \
11097cae6aabSDevin Teske				"$bootpool/boot/$disk$targetpart.eli" \
11107cae6aabSDevin Teske				AES-XTS "$bootpool/$zroot_key" \
1111a622223fSDevin Teske				$disk$targetpart
1112bc4a673fSDevin Teske			then
1113bc4a673fSDevin Teske				f_interactive || f_die
1114a622223fSDevin Teske				unset pw_password # Sensitive info
1115cd88b886SDevin Teske				return $FAILURE
1116bc4a673fSDevin Teske			fi
1117a622223fSDevin Teske			if ! echo "$pw_password" | f_eval_catch \
1118a622223fSDevin Teske				$funcname geli "$GELI_ATTACH" \
11197cae6aabSDevin Teske				"$bootpool/$zroot_key" $disk$targetpart
1120bc4a673fSDevin Teske			then
1121bc4a673fSDevin Teske				f_interactive || f_die
1122a622223fSDevin Teske				unset pw_password # Sensitive info
1123bc4a673fSDevin Teske				return $FAILURE
1124bc4a673fSDevin Teske			fi
1125cd88b886SDevin Teske		done
1126a622223fSDevin Teske		unset pw_password # Sensitive info
1127a622223fSDevin Teske
1128a622223fSDevin Teske		# Clean up
11297cae6aabSDevin Teske		f_eval_catch $funcname zfs "$ZFS_UNMOUNT" "$bootpool_name" ||
11307cae6aabSDevin Teske			return $FAILURE
1131a622223fSDevin Teske		f_eval_catch -d $funcname umount "$UMOUNT" /mnt # tmpfs
1132cd88b886SDevin Teske	fi
1133*2875e59fSOllivier Robert	#
1134*2875e59fSOllivier Robert	# Create the gmirror(8) GEOMS for swap
1135*2875e59fSOllivier Robert	#
1136*2875e59fSOllivier Robert	if [ "$ZFSBOOT_SWAP_MIRROR" ]; then
1137*2875e59fSOllivier Robert		for disk in $disks; do
1138*2875e59fSOllivier Robert			swap_devs="$swap_devs $disk$swappart"
1139*2875e59fSOllivier Robert		done
1140*2875e59fSOllivier Robert		f_eval_catch $funcname gmirror "$SWAP_GMIRROR_LABEL" \
1141*2875e59fSOllivier Robert			"$swap_devs" || return $FAILURE
1142*2875e59fSOllivier Robert	fi
1143cd88b886SDevin Teske
1144cd88b886SDevin Teske	#
1145a622223fSDevin Teske	# Create the ZFS root pool with desired type and disk devices
1146cd88b886SDevin Teske	#
11477cae6aabSDevin Teske	f_dprintf "$funcname: Creating root pool..."
1148bc4a673fSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_CREATE_WITH_OPTIONS" \
1149a622223fSDevin Teske	             "-o altroot=$BSDINSTALL_CHROOT -m none -f" \
11504b4b90d5SDevin Teske	             "$zroot_name" "$zroot_vdevtype" "$zroot_vdevs" ||
11514b4b90d5SDevin Teske	             return $FAILURE
1152cd88b886SDevin Teske
11537cae6aabSDevin Teske	# Customize the zroot a bit...
1154bc4a673fSDevin Teske	local option
11557cae6aabSDevin Teske	f_dprintf "$funcname: Setting miscellaneous options on root pool..."
1156ca2d97c2SDevin Teske	for option in atime=off; do
11577cae6aabSDevin Teske		f_eval_catch $funcname zfs "$ZFS_SET" $option "$zroot_name" ||
1158bc4a673fSDevin Teske			return $FAILURE
1159bc4a673fSDevin Teske	done
1160cd88b886SDevin Teske
1161cd88b886SDevin Teske	#
11627cae6aabSDevin Teske	# Create ZFS dataset layout within the new root pool
1163cd88b886SDevin Teske	#
1164bc4a673fSDevin Teske	f_dprintf "$funcname: Creating ZFS datasets..."
1165cd88b886SDevin Teske	echo "$ZFSBOOT_DATASETS" | while read dataset options; do
1166cd88b886SDevin Teske		# Skip blank lines and comments
1167cd88b886SDevin Teske		case "$dataset" in "#"*|"") continue; esac
1168cd88b886SDevin Teske		# Remove potential inline comments in options
1169cd88b886SDevin Teske		options="${options%%#*}"
1170cd88b886SDevin Teske		# Replace tabs with spaces
1171cd88b886SDevin Teske		f_replaceall "$options" "	" " " options
1172cd88b886SDevin Teske		# Reduce contiguous runs of space to one single space
1173cd88b886SDevin Teske		oldoptions=
1174cd88b886SDevin Teske		while [ "$oldoptions" != "$options" ]; do
1175cd88b886SDevin Teske			oldoptions="$options"
1176cd88b886SDevin Teske			f_replaceall "$options" "  " " " options
1177cd88b886SDevin Teske		done
1178cd88b886SDevin Teske		# Replace both commas and spaces with ` -o '
1179cd88b886SDevin Teske		f_replaceall "$options" "[ ,]" " -o " options
1180cd88b886SDevin Teske		# Create the dataset with desired options
1181bc4a673fSDevin Teske		f_eval_catch $funcname zfs "$ZFS_CREATE_WITH_OPTIONS" \
11827cae6aabSDevin Teske		             "${options:+-o $options}" "$zroot_name$dataset" ||
1183cd88b886SDevin Teske		             return $FAILURE
1184cd88b886SDevin Teske	done
1185cd88b886SDevin Teske
1186cd88b886SDevin Teske	# Touch up permissions on the tmp directories
1187bc4a673fSDevin Teske	f_dprintf "$funcname: Modifying directory permissions..."
1188bc4a673fSDevin Teske	local dir
1189bc4a673fSDevin Teske	for dir in /tmp /var/tmp; do
1190bc4a673fSDevin Teske		f_eval_catch $funcname chmod "$CHMOD_MODE" 1777 \
1191a622223fSDevin Teske		             $BSDINSTALL_CHROOTDIR$dir || return $FAILURE
1192bc4a673fSDevin Teske	done
1193cd88b886SDevin Teske
1194cd88b886SDevin Teske	# Create symlink(s)
11957cae6aabSDevin Teske	if [ "$ZFSBOOT_BOOT_POOL" ]; then
11967cae6aabSDevin Teske		f_dprintf "$funcname: Creating /boot symlink for boot pool..."
11977cae6aabSDevin Teske		f_eval_catch $funcname ln "$LN_SF" "$bootpool_name/boot" \
1198a622223fSDevin Teske		             $BSDINSTALL_CHROOT/boot || return $FAILURE
1199bc4a673fSDevin Teske	fi
1200cd88b886SDevin Teske
1201cd88b886SDevin Teske	# Set bootfs property
12027cae6aabSDevin Teske	local zroot_bootfs="$ZFSBOOT_BEROOT_NAME/$ZFSBOOT_BOOTFS_NAME"
1203bc4a673fSDevin Teske	f_dprintf "$funcname: Setting bootfs property..."
1204bc4a673fSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_SET" \
12057cae6aabSDevin Teske		"bootfs=\"$zroot_name/$zroot_bootfs\"" "$zroot_name" ||
12067cae6aabSDevin Teske		return $FAILURE
1207cd88b886SDevin Teske
1208cd88b886SDevin Teske	# Export the pool(s)
1209bc4a673fSDevin Teske	f_dprintf "$funcname: Temporarily exporting ZFS pool(s)..."
12107cae6aabSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_EXPORT" "$zroot_name" ||
1211bc4a673fSDevin Teske	             return $FAILURE
12127cae6aabSDevin Teske	if [ "$ZFSBOOT_BOOT_POOL" ]; then
1213bc4a673fSDevin Teske		f_eval_catch $funcname zpool "$ZPOOL_EXPORT" \
12147cae6aabSDevin Teske		             "$bootpool_name" || return $FAILURE
1215bc4a673fSDevin Teske	fi
1216cd88b886SDevin Teske
1217cd88b886SDevin Teske	# Destroy the gnop devices (if enabled)
1218bc4a673fSDevin Teske	for disk in ${ZFSBOOT_GNOP_4K_FORCE_ALIGN:+$disks}; do
12197cae6aabSDevin Teske		if [ "$ZFSBOOT_BOOT_POOL" ]; then
1220a622223fSDevin Teske			f_eval_catch -d $funcname gnop "$GNOP_DESTROY" \
1221a622223fSDevin Teske			                $disk$bootpart.nop
12227cae6aabSDevin Teske		fi
12237cae6aabSDevin Teske		if [ ! "$ZFSBOOT_GELI_ENCRYPTION" ]; then
1224a622223fSDevin Teske			f_eval_catch -d $funcname gnop "$GNOP_DESTROY" \
1225a622223fSDevin Teske			                $disk$targetpart.nop
1226cd88b886SDevin Teske		fi
1227cd88b886SDevin Teske	done
1228cd88b886SDevin Teske
12297cae6aabSDevin Teske	# MBR boot loader touch-up
1230cd88b886SDevin Teske	if [ "$ZFSBOOT_PARTITION_SCHEME" = "MBR" ]; then
1231bc4a673fSDevin Teske		f_dprintf "$funcname: Updating MBR boot loader on disks..."
1232cd88b886SDevin Teske		# Stick the ZFS boot loader in the "convienient hole" after
1233cd88b886SDevin Teske		# the ZFS internal metadata
1234bc4a673fSDevin Teske		for disk in $disks; do
1235bc4a673fSDevin Teske			f_eval_catch $funcname dd "$DD_WITH_OPTIONS" \
1236a622223fSDevin Teske			             /boot/zfsboot /dev/$disk$bootpart \
1237bc4a673fSDevin Teske			             "skip=1 seek=1024" || return $FAILURE
1238cd88b886SDevin Teske		done
1239cd88b886SDevin Teske	fi
1240cd88b886SDevin Teske
1241cd88b886SDevin Teske	# Re-import the ZFS pool(s)
1242bc4a673fSDevin Teske	f_dprintf "$funcname: Re-importing ZFS pool(s)..."
1243bc4a673fSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_IMPORT_WITH_OPTIONS" \
12447cae6aabSDevin Teske	             "-o altroot=\"$BSDINSTALL_CHROOT\"" "$zroot_name" ||
1245bc4a673fSDevin Teske	             return $FAILURE
12467cae6aabSDevin Teske	if [ "$ZFSBOOT_BOOT_POOL" ]; then
1247bc4a673fSDevin Teske		f_eval_catch $funcname zpool "$ZPOOL_IMPORT_WITH_OPTIONS" \
12487cae6aabSDevin Teske		             "-o altroot=\"$BSDINSTALL_CHROOT\"" \
12497cae6aabSDevin Teske		             "$bootpool_name" || return $FAILURE
1250bc4a673fSDevin Teske	fi
1251cd88b886SDevin Teske
1252cd88b886SDevin Teske	# While this is apparently not needed, it seems to help MBR
12536311cc9eSDevin Teske	f_dprintf "$funcname: Configuring zpool.cache for zroot..."
1254a622223fSDevin Teske	f_eval_catch $funcname mkdir "$MKDIR_P" $BSDINSTALL_CHROOT/boot/zfs ||
1255bc4a673fSDevin Teske	             return $FAILURE
1256bc4a673fSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_SET" \
1257a622223fSDevin Teske	             "cachefile=\"$BSDINSTALL_CHROOT/boot/zfs/zpool.cache\"" \
12587cae6aabSDevin Teske	             "$zroot_name" || return $FAILURE
1259cd88b886SDevin Teske
126013d8e1ceSDevin Teske	# Last, but not least... required lines for rc.conf(5)/loader.conf(5)
1261cd88b886SDevin Teske	# NOTE: We later concatenate these into their destination
1262bc4a673fSDevin Teske	f_dprintf "%s: Configuring rc.conf(5)/loader.conf(5) additions..." \
1263bc4a673fSDevin Teske	          "$funcname"
1264bc4a673fSDevin Teske	f_eval_catch $funcname echo "$ECHO_APPEND" 'zfs_enable=\"YES\"' \
1265a622223fSDevin Teske	             $BSDINSTALL_TMPETC/rc.conf.zfs || return $FAILURE
1266bc4a673fSDevin Teske	f_eval_catch $funcname echo "$ECHO_APPEND" 'zfs_load=\"YES\"' \
1267a622223fSDevin Teske	             $BSDINSTALL_TMPBOOT/loader.conf.zfs || return $FAILURE
1268ee482f2cSDevin Teske	f_eval_catch $funcname echo "$ECHO_APPEND" \
1269ee482f2cSDevin Teske	             'kern.geom.label.disk_ident.enable=\"0\"' \
1270ee482f2cSDevin Teske	             $BSDINSTALL_TMPBOOT/loader.conf.zfs || return $FAILURE
1271*2875e59fSOllivier Robert	f_eval_catch $funcname echo "$ECHO_APPEND" \
1272*2875e59fSOllivier Robert	             'kern.geom.label.gptid.enable=\"0\"' \
1273*2875e59fSOllivier Robert	             $BSDINSTALL_TMPBOOT/loader.conf.zfs || return $FAILURE
1274*2875e59fSOllivier Robert
1275*2875e59fSOllivier Robert	if [ "$ZFSBOOT_SWAP_MIRROR" ]; then
1276*2875e59fSOllivier Robert		f_eval_catch $funcname echo "$ECHO_APPEND" 'geom_mirror_load=\"YES\"' \
1277*2875e59fSOllivier Robert		             $BSDINSTALL_TMPBOOT/loader.conf.gmirror || return $FAILURE
1278*2875e59fSOllivier Robert	fi
1279cd88b886SDevin Teske
12807cae6aabSDevin Teske	# We're all done unless we should go on for boot pool
12817cae6aabSDevin Teske	[ "$ZFSBOOT_BOOT_POOL" ] || return $SUCCESS
1282cd88b886SDevin Teske
12836311cc9eSDevin Teske	# Set cachefile for boot pool so it auto-imports at system start
12846311cc9eSDevin Teske	f_dprintf "$funcname: Configuring zpool.cache for boot pool..."
12856311cc9eSDevin Teske	f_eval_catch $funcname zpool "$ZPOOL_SET" \
12866311cc9eSDevin Teske	             "cachefile=\"$BSDINSTALL_CHROOT/boot/zfs/zpool.cache\"" \
12876311cc9eSDevin Teske	             "$bootpool_name" || return $FAILURE
12886311cc9eSDevin Teske
1289bc4a673fSDevin Teske	# Some additional geli(8) requirements for loader.conf(5)
1290bc4a673fSDevin Teske	for option in \
1291a622223fSDevin Teske		'zpool_cache_load=\"YES\"' \
1292a622223fSDevin Teske		'zpool_cache_type=\"/boot/zfs/zpool.cache\"' \
1293a622223fSDevin Teske		'zpool_cache_name=\"/boot/zfs/zpool.cache\"' \
1294bc4a673fSDevin Teske	; do
1295a622223fSDevin Teske		f_eval_catch $funcname echo "$ECHO_APPEND" "$option" \
1296a622223fSDevin Teske		             $BSDINSTALL_TMPBOOT/loader.conf.zfs ||
1297bc4a673fSDevin Teske		             return $FAILURE
1298bc4a673fSDevin Teske	done
12997cae6aabSDevin Teske	f_eval_catch $funcname printf "$PRINTF_CONF" vfs.root.mountfrom \
13007cae6aabSDevin Teske	    "\"zfs:$zroot_name/$zroot_bootfs\"" \
13017cae6aabSDevin Teske	    $BSDINSTALL_TMPBOOT/loader.conf.root || return $FAILURE
13027cae6aabSDevin Teske
13037cae6aabSDevin Teske	# We're all done unless we should go on to do encryption
13047cae6aabSDevin Teske	[ "$ZFSBOOT_GELI_ENCRYPTION" ] || return $SUCCESS
130513d8e1ceSDevin Teske
1306cd88b886SDevin Teske	#
1307cd88b886SDevin Teske	# Configure geli(8)-based encryption
1308cd88b886SDevin Teske	#
1309bc4a673fSDevin Teske	f_dprintf "$funcname: Configuring disk encryption..."
1310bc4a673fSDevin Teske	f_eval_catch $funcname echo "$ECHO_APPEND" 'aesni_load=\"YES\"' \
1311a622223fSDevin Teske	    $BSDINSTALL_TMPBOOT/loader.conf.aesni || return $FAILURE
1312bc4a673fSDevin Teske	f_eval_catch $funcname echo "$ECHO_APPEND" 'geom_eli_load=\"YES\"' \
1313a622223fSDevin Teske	    $BSDINSTALL_TMPBOOT/loader.conf.geli || return $FAILURE
1314bc4a673fSDevin Teske	for disk in $disks; do
1315bc4a673fSDevin Teske		f_eval_catch $funcname printf "$PRINTF_CONF" \
1316a622223fSDevin Teske			geli_%s_keyfile0_load "$disk$targetpart YES" \
1317a622223fSDevin Teske			$BSDINSTALL_TMPBOOT/loader.conf.$disk$targetpart ||
1318cd88b886SDevin Teske			return $FAILURE
1319bc4a673fSDevin Teske		f_eval_catch $funcname printf "$PRINTF_CONF" \
1320bc4a673fSDevin Teske			geli_%s_keyfile0_type \
1321a622223fSDevin Teske			"$disk$targetpart $disk$targetpart:geli_keyfile0" \
1322a622223fSDevin Teske			$BSDINSTALL_TMPBOOT/loader.conf.$disk$targetpart ||
1323cd88b886SDevin Teske			return $FAILURE
1324bc4a673fSDevin Teske		f_eval_catch $funcname printf "$PRINTF_CONF" \
1325bc4a673fSDevin Teske			geli_%s_keyfile0_name \
1326a622223fSDevin Teske			"$disk$targetpart \"$ZFSBOOT_GELI_KEY_FILE\"" \
1327a622223fSDevin Teske			$BSDINSTALL_TMPBOOT/loader.conf.$disk$targetpart ||
1328cd88b886SDevin Teske			return $FAILURE
1329cd88b886SDevin Teske	done
1330cd88b886SDevin Teske
1331cd88b886SDevin Teske	return $SUCCESS
1332cd88b886SDevin Teske}
1333cd88b886SDevin Teske
1334cd88b886SDevin Teske# dialog_menu_diskinfo
1335cd88b886SDevin Teske#
1336cd88b886SDevin Teske# Prompt the user to select a disk and then provide detailed info on it.
1337cd88b886SDevin Teske#
1338cd88b886SDevin Teskedialog_menu_diskinfo()
1339cd88b886SDevin Teske{
1340a88393ceSDevin Teske	local device disk
1341cd88b886SDevin Teske
1342cd88b886SDevin Teske	#
1343cd88b886SDevin Teske	# Break from loop when user cancels disk selection
1344cd88b886SDevin Teske	#
1345cd88b886SDevin Teske	while :; do
1346a88393ceSDevin Teske		device=$( msg_cancel="$msg_back" f_device_menu \
1347cd88b886SDevin Teske			"$DIALOG_TITLE" "$msg_select_a_disk_device" "" \
1348cd88b886SDevin Teske			$DEVICE_TYPE_DISK 2>&1 ) || break
1349a88393ceSDevin Teske		$device get name disk
1350cd88b886SDevin Teske
1351cd88b886SDevin Teske		# Show gpart(8) `show' and camcontrol(8) `inquiry' data
1352cd88b886SDevin Teske		f_show_msg "$msg_detailed_disk_info" \
1353cd88b886SDevin Teske			"$disk" "$( gpart show $disk 2> /dev/null )" \
1354cd88b886SDevin Teske			"$disk" "$( camcontrol inquiry $disk 2> /dev/null )" \
1355cd88b886SDevin Teske			"$disk" "$( camcontrol identify $disk 2> /dev/null )"
1356cd88b886SDevin Teske	done
1357cd88b886SDevin Teske
1358cd88b886SDevin Teske	return $SUCCESS
1359cd88b886SDevin Teske}
1360cd88b886SDevin Teske
1361cd88b886SDevin Teske############################################################ MAIN
1362cd88b886SDevin Teske
1363cd88b886SDevin Teske#
1364cd88b886SDevin Teske# Initialize
1365cd88b886SDevin Teske#
1366cd88b886SDevin Teskef_dialog_title "$msg_zfs_configuration"
1367cd88b886SDevin Teskef_dialog_backtitle "$msg_freebsd_installer"
1368cd88b886SDevin Teske
1369cd88b886SDevin Teske# User may have specifically requested ZFS-related operations be interactive
1370cd88b886SDevin Teske! f_interactive && f_zfsinteractive && unset $VAR_NONINTERACTIVE
1371cd88b886SDevin Teske
1372cd88b886SDevin Teske#
1373bc4a673fSDevin Teske# Debugging
1374bc4a673fSDevin Teske#
1375bc4a673fSDevin Teskef_dprintf "BSDINSTALL_CHROOT=[%s]" "$BSDINSTALL_CHROOT"
1376bc4a673fSDevin Teskef_dprintf "BSDINSTALL_TMPETC=[%s]" "$BSDINSTALL_TMPETC"
1377a622223fSDevin Teskef_dprintf "FSTAB_FMT=[%s]" "$FSTAB_FMT"
1378bc4a673fSDevin Teske
1379bc4a673fSDevin Teske#
1380cd88b886SDevin Teske# Loop over the main menu until we've accomplished what we came here to do
1381cd88b886SDevin Teske#
1382cd88b886SDevin Teskewhile :; do
1383cd88b886SDevin Teske	if ! f_interactive; then
1384cd88b886SDevin Teske		retval=$DIALOG_OK
1385bc4a673fSDevin Teske		mtag=">>> $msg_install"
1386cd88b886SDevin Teske	else
1387cd88b886SDevin Teske		dialog_menu_main
1388cd88b886SDevin Teske		retval=$?
1389cd88b886SDevin Teske		f_dialog_menutag_fetch mtag
1390cd88b886SDevin Teske	fi
1391cd88b886SDevin Teske
1392bc4a673fSDevin Teske	f_dprintf "retval=%u mtag=[%s]" $retval "$mtag"
1393cd88b886SDevin Teske	[ $retval -eq $DIALOG_OK ] || f_die
1394cd88b886SDevin Teske
1395cd88b886SDevin Teske	case "$mtag" in
1396bc4a673fSDevin Teske	">>> $msg_install")
1397cd88b886SDevin Teske		#
1398cd88b886SDevin Teske		# First, validate the user's selections
1399cd88b886SDevin Teske		#
1400cd88b886SDevin Teske
1401cd88b886SDevin Teske		# Make sure they gave us a name for the pool
1402cd88b886SDevin Teske		if [ ! "$ZFSBOOT_POOL_NAME" ]; then
1403bc4a673fSDevin Teske			f_dprintf "Pool name cannot be empty."
1404bc4a673fSDevin Teske			f_show_err "$msg_pool_name_cannot_be_empty"
1405cd88b886SDevin Teske			continue
1406cd88b886SDevin Teske		fi
1407bc4a673fSDevin Teske
1408bc4a673fSDevin Teske		# Validate vdev type against number of disks selected/scripted
1409bc4a673fSDevin Teske		# (also validates that ZFSBOOT_DISKS are real [probed] disks)
1410bc4a673fSDevin Teske		# NB: dialog_menu_layout supports running non-interactively
1411bc4a673fSDevin Teske		dialog_menu_layout || continue
1412bc4a673fSDevin Teske
1413cd88b886SDevin Teske		# Make sure each disk will be at least 50% ZFS
1414cd88b886SDevin Teske		if f_expand_number "$ZFSBOOT_SWAP_SIZE" swapsize &&
14157cae6aabSDevin Teske		   f_expand_number "$ZFSBOOT_BOOT_POOL_SIZE" bootsize
1416cd88b886SDevin Teske		then
1417cd88b886SDevin Teske			minsize=$swapsize teeny_disks=
14187cae6aabSDevin Teske			[ "$ZFSBOOT_BOOT_POOL" ] &&
14197cae6aabSDevin Teske				minsize=$(( $minsize + $bootsize ))
1420bc4a673fSDevin Teske			for disk in $ZFSBOOT_DISKS; do
1421a88393ceSDevin Teske				debug= f_device_find -1 \
1422a88393ceSDevin Teske					$disk $DEVICE_TYPE_DISK device
1423a88393ceSDevin Teske				$device get capacity disksize || continue
1424a88393ceSDevin Teske				[ ${disksize:-0} -ge 0 ] || disksize=0
1425cd88b886SDevin Teske				disksize=$(( $disksize - $minsize ))
1426cd88b886SDevin Teske				[ $disksize -lt $minsize ] &&
1427cd88b886SDevin Teske					teeny_disks="$teeny_disks $disk"
1428cd88b886SDevin Teske			done
1429cd88b886SDevin Teske			if [ "$teeny_disks" ]; then
14307cae6aabSDevin Teske				f_dprintf "swapsize=[%s] bootsize[%s] %s" \
1431bc4a673fSDevin Teske				          "$ZFSBOOT_SWAP_SIZE" \
14327cae6aabSDevin Teske				          "$ZFSBOOT_BOOT_POOL_SIZE" \
14337cae6aabSDevin Teske				          "minsize=[$minsize]"
1434bc4a673fSDevin Teske				f_dprintf "These disks are too small: %s" \
1435bc4a673fSDevin Teske				          "$teeny_disks"
1436bc4a673fSDevin Teske				f_show_err "$msg_these_disks_are_too_small" \
1437cd88b886SDevin Teske				           "$ZFSBOOT_SWAP_SIZE" \
14387cae6aabSDevin Teske				           "$ZFSBOOT_BOOT_POOL_SIZE" \
1439cd88b886SDevin Teske				           "$teeny_disks"
1440cd88b886SDevin Teske				continue
1441cd88b886SDevin Teske			fi
1442cd88b886SDevin Teske		fi
1443cd88b886SDevin Teske
1444cd88b886SDevin Teske		#
1445cd88b886SDevin Teske		# Last Chance!
1446cd88b886SDevin Teske		#
1447bc4a673fSDevin Teske		if f_interactive; then
1448bc4a673fSDevin Teske			dialog_last_chance $ZFSBOOT_DISKS || continue
1449cd88b886SDevin Teske		fi
1450cd88b886SDevin Teske
1451cd88b886SDevin Teske		#
1452cd88b886SDevin Teske		# Let's do this
1453cd88b886SDevin Teske		#
1454cd88b886SDevin Teske
1455cd88b886SDevin Teske		vdev_type="$ZFSBOOT_VDEV_TYPE"
1456cd88b886SDevin Teske
1457cd88b886SDevin Teske		# Blank the vdev type for the default layout
1458cd88b886SDevin Teske		[ "$vdev_type" = "stripe" ] && vdev_type=
1459cd88b886SDevin Teske
1460bc4a673fSDevin Teske		zfs_create_boot "$ZFSBOOT_POOL_NAME" \
1461bc4a673fSDevin Teske		                "$vdev_type" $ZFSBOOT_DISKS || continue
1462cd88b886SDevin Teske
1463cd88b886SDevin Teske		break # to success
1464cd88b886SDevin Teske		;;
1465bc4a673fSDevin Teske	?" $msg_pool_type_disks")
1466bc4a673fSDevin Teske		ZFSBOOT_CONFIRM_LAYOUT=1
1467bc4a673fSDevin Teske		dialog_menu_layout
1468bc4a673fSDevin Teske		# User has poked settings, disable later confirmation
1469bc4a673fSDevin Teske		ZFSBOOT_CONFIRM_LAYOUT=
1470bc4a673fSDevin Teske		;;
1471cd88b886SDevin Teske	"- $msg_rescan_devices") f_device_rescan ;;
1472cd88b886SDevin Teske	"- $msg_disk_info") dialog_menu_diskinfo ;;
1473cd88b886SDevin Teske	?" $msg_pool_name")
1474cd88b886SDevin Teske		# Prompt the user to input/change the name for the new pool
1475cd88b886SDevin Teske		f_dialog_input input \
1476cd88b886SDevin Teske			"$msg_please_enter_a_name_for_your_zpool" \
1477cd88b886SDevin Teske			"$ZFSBOOT_POOL_NAME" &&
1478cd88b886SDevin Teske			ZFSBOOT_POOL_NAME="$input"
1479cd88b886SDevin Teske		;;
1480cd88b886SDevin Teske	?" $msg_force_4k_sectors")
1481cd88b886SDevin Teske		# Toggle the variable referenced both by the menu and later
1482cd88b886SDevin Teske		if [ "$ZFSBOOT_GNOP_4K_FORCE_ALIGN" ]; then
1483cd88b886SDevin Teske			ZFSBOOT_GNOP_4K_FORCE_ALIGN=
1484cd88b886SDevin Teske		else
1485cd88b886SDevin Teske			ZFSBOOT_GNOP_4K_FORCE_ALIGN=1
1486cd88b886SDevin Teske		fi
1487cd88b886SDevin Teske		;;
1488bc4a673fSDevin Teske	?" $msg_encrypt_disks")
1489cd88b886SDevin Teske		# Toggle the variable referenced both by the menu and later
1490cd88b886SDevin Teske		if [ "$ZFSBOOT_GELI_ENCRYPTION" ]; then
1491cd88b886SDevin Teske			ZFSBOOT_GELI_ENCRYPTION=
1492cd88b886SDevin Teske		else
1493926ec73fSDevin Teske			ZFSBOOT_GNOP_4K_FORCE_ALIGN=1
1494cd88b886SDevin Teske			ZFSBOOT_GELI_ENCRYPTION=1
1495cd88b886SDevin Teske		fi
1496cd88b886SDevin Teske		;;
1497cd88b886SDevin Teske	?" $msg_partition_scheme")
1498cd88b886SDevin Teske		# Toggle between GPT and MBR
1499cd88b886SDevin Teske		if [ "$ZFSBOOT_PARTITION_SCHEME" = GPT ]; then
1500cd88b886SDevin Teske			ZFSBOOT_PARTITION_SCHEME=MBR
1501cd88b886SDevin Teske		else
1502cd88b886SDevin Teske			ZFSBOOT_PARTITION_SCHEME=GPT
1503cd88b886SDevin Teske		fi
1504cd88b886SDevin Teske		;;
1505cd88b886SDevin Teske	?" $msg_swap_size")
1506cd88b886SDevin Teske		# Prompt the user to input/change the swap size for each disk
1507cd88b886SDevin Teske		f_dialog_input input \
1508cd88b886SDevin Teske			"$msg_please_enter_amount_of_swap_space" \
1509cd88b886SDevin Teske			"$ZFSBOOT_SWAP_SIZE" &&
151030c8ebe9SDevin Teske			ZFSBOOT_SWAP_SIZE="${input:-0}"
1511cd88b886SDevin Teske		;;
1512*2875e59fSOllivier Robert	?" $msg_swap_mirror")
1513*2875e59fSOllivier Robert		# Toggle the variable referenced both by the menu and later
1514*2875e59fSOllivier Robert		if [ "$ZFSBOOT_SWAP_MIRROR" ]; then
1515*2875e59fSOllivier Robert			ZFSBOOT_SWAP_MIRROR=
1516*2875e59fSOllivier Robert		else
1517*2875e59fSOllivier Robert			ZFSBOOT_SWAP_MIRROR=1
1518*2875e59fSOllivier Robert		fi
1519*2875e59fSOllivier Robert		;;
1520*2875e59fSOllivier Robert	?" $msg_swap_encrypt")
1521*2875e59fSOllivier Robert		# Toggle the variable referenced both by the menu and later
1522*2875e59fSOllivier Robert		if [ "$ZFSBOOT_SWAP_ENCRYPTION" ]; then
1523*2875e59fSOllivier Robert			ZFSBOOT_SWAP_ENCRYPTION=
1524*2875e59fSOllivier Robert		else
1525*2875e59fSOllivier Robert			ZFSBOOT_SWAP_ENCRYPTION=1
1526*2875e59fSOllivier Robert		fi
1527*2875e59fSOllivier Robert		;;
1528cd88b886SDevin Teske	esac
1529cd88b886SDevin Teskedone
1530cd88b886SDevin Teske
1531cd88b886SDevin Teskereturn $SUCCESS
1532cd88b886SDevin Teske
1533cd88b886SDevin Teske################################################################################
1534cd88b886SDevin Teske# END
1535cd88b886SDevin Teske################################################################################
1536