Module Name:    src
Committed By:   roy
Date:           Sat Feb 22 09:53:48 UTC 2020

Added Files:
        src/distrib/amd64/ramdisks/ramdisk-zfsroot: Makefile list
        src/distrib/common: list.zfsroot mtree.zfsroot zfsroot.rc

Log Message:
zfs: build a ramdisk on amd64 with enough to mount rpool/ROOT on /

Until we get ZFS integrated into our boot loader, this is the next best
thing. The idea is simple - have a small FFS partition with a kernel,
modules and this ramdisk. Once the ramdisk boots it will mount the FFS
partition read only, copy the needed ZFS modules to the ramdisk and then
unmount the partition. Then we import the ZFS root pool, mount the
ZFS root filesystem and then pivot to it.

Because the initial FFS partition is not mounted at this point, we
can mount it in /altroot so we can replace the kernel and modules with
newer ones so it's easily maintainable.

This ZFS boot strapper currently makes the following assumptions:
 * The device NAME=boot is the FFS with kernel, modules and this ramdisk.
 * The ZFS root pool and root filesystem are called rpool/ROOT.

A boot.cfg menu entry can then be added like so:
menu=Boot ZFS root:fs /ramdisk-zfsroot.fs;boot


To generate a diff of this commit:
cvs rdiff -u -r0 -r1.1 src/distrib/amd64/ramdisks/ramdisk-zfsroot/Makefile \
    src/distrib/amd64/ramdisks/ramdisk-zfsroot/list
cvs rdiff -u -r0 -r1.1 src/distrib/common/list.zfsroot \
    src/distrib/common/mtree.zfsroot src/distrib/common/zfsroot.rc

Please note that diffs are not public domain; they are subject to the
copyright notices on the relevant files.

Added files:

Index: src/distrib/amd64/ramdisks/ramdisk-zfsroot/Makefile
diff -u /dev/null src/distrib/amd64/ramdisks/ramdisk-zfsroot/Makefile:1.1
--- /dev/null	Sat Feb 22 09:53:48 2020
+++ src/distrib/amd64/ramdisks/ramdisk-zfsroot/Makefile	Sat Feb 22 09:53:47 2020
@@ -0,0 +1,17 @@
+#	$NetBSD: Makefile,v 1.1 2020/02/22 09:53:47 roy Exp $
+
+IMAGE=			ramdisk-zfsroot.fs
+IMAGESIZE=		5000k
+IMAGEDEPENDS=
+MAKEDEVTARGETS=		all
+CRUNCHENV=		INIT_CHROOT=1
+SMALLPROG_INET6=	1
+
+LISTS+=			${DISTRIBDIR}/common/list.zfsroot
+
+.include "${.CURDIR}/../common/Makefile.ramdisk"
+.include "${DISTRIBDIR}/common/Makefile.makedev"
+
+LISTS+=			${.CURDIR}/list
+
+MTREECONF+=		${DISTRIBDIR}/common/mtree.zfsroot
Index: src/distrib/amd64/ramdisks/ramdisk-zfsroot/list
diff -u /dev/null src/distrib/amd64/ramdisks/ramdisk-zfsroot/list:1.1
--- /dev/null	Sat Feb 22 09:53:48 2020
+++ src/distrib/amd64/ramdisks/ramdisk-zfsroot/list	Sat Feb 22 09:53:47 2020
@@ -0,0 +1,12 @@
+#	$NetBSD: list,v 1.1 2020/02/22 09:53:47 roy Exp $
+
+PROG	bin/sync
+
+PROG	sbin/fdisk
+PROG	sbin/gpt
+PROG	sbin/mbrlabel
+PROG	sbin/shutdown
+
+PROG	usr/bin/less	usr/bin/more
+
+PROG	usr/sbin/installboot

Index: src/distrib/common/list.zfsroot
diff -u /dev/null src/distrib/common/list.zfsroot:1.1
--- /dev/null	Sat Feb 22 09:53:48 2020
+++ src/distrib/common/list.zfsroot	Sat Feb 22 09:53:47 2020
@@ -0,0 +1,29 @@
+#	$NetBSD: list.zfsroot,v 1.1 2020/02/22 09:53:47 roy Exp $
+#
+# list file (c.f. parselist.awk) for ZFS on root.
+#
+
+SRCDIRS	external/cddl/osnet/sbin
+PROG	sbin/zfs
+PROG	sbin/zpool
+LINK	sbin/zfs sbin/mount_zfs
+ARGVLN	zfs mount_zfs
+
+# We need sysctl to set init.root=/altroot
+PROG	sbin/sysctl
+
+LIBS	-lnvpair
+LIBS	-luutil
+LIBS	-lzfs
+LIBS	-lavl
+LIBS	-lm
+LIBS	-lpthread
+LIBS	-lumem
+#LIBS	-lutil		# replaced by libhack
+LIBS	-lz
+LIBS	-lzfs_core
+
+COPY	${NETBSDSRCDIR}/distrib/common/zfsroot.rc etc/rc
+
+# Make firmware images available.
+SYMLINK	altroot/libdata libdata
Index: src/distrib/common/mtree.zfsroot
diff -u /dev/null src/distrib/common/mtree.zfsroot:1.1
--- /dev/null	Sat Feb 22 09:53:48 2020
+++ src/distrib/common/mtree.zfsroot	Sat Feb 22 09:53:47 2020
@@ -0,0 +1,8 @@
+#	$NetBSD: mtree.zfsroot,v 1.1 2020/02/22 09:53:47 roy Exp $
+
+.
+./altroot
+./etc
+./etc/zfs
+./rpool
+./stand
Index: src/distrib/common/zfsroot.rc
diff -u /dev/null src/distrib/common/zfsroot.rc:1.1
--- /dev/null	Sat Feb 22 09:53:48 2020
+++ src/distrib/common/zfsroot.rc	Sat Feb 22 09:53:47 2020
@@ -0,0 +1,64 @@
+#/bin/sh
+#
+#	$NetBSD: zfsroot.rc,v 1.1 2020/02/22 09:53:47 roy Exp $
+#	ZFS on Root boot strapper
+
+# Configurable - define the ZFS root pool and ROOT.
+# XXX Can these be set in boot.cfg?
+# Assumption - the root pool is set to legacy mount.
+rpool=rpool
+rroot=ROOT
+
+# Assumption - the boot device is named boot.
+# Could use dk0, wd0a, etc instead.
+# XXX Can be exposed by sysctl kern.boot_device?
+bootdev="NAME=boot"
+
+# Setup some stuff incase things go south and we drop to the shell
+export HOME=/
+export PATH=/sbin:/bin:/usr/sbin:/usr/bin
+umask 022
+
+echo
+echo "Starting ZFS on root boot strapper"
+
+# Avoid having the solaris and zfs modules in ramdisk directly.
+# Means we don't need to update the ramdisk with the kernel modules
+# or load them from boot.cfg so it's less pain for the user.
+#bootdev="$(/sbin/sysctl -n kern.boot_device)"
+modpath="$(/sbin/sysctl -n kern.module.path)"
+modmnt=/mnt
+echo "Copying needed kernel modules from $bootdev:$modpath"
+case "$bootdev" in
+*=*|"/"*)	;;
+*)		bootdev="/dev/$bootdev";;
+esac
+/sbin/mount -o ro "$bootdev" "$modmnt"
+/sbin/mount -t tmpfs none /stand
+for m in solaris zfs; do
+	p="$modpath/$m"
+	if [ ! -e "$modmnt/$p/$m.kmod" ]; then
+		echo "$modmnt/$p/$m.kmod not found!" >&2
+		continue
+	fi
+	echo "	$m.kmod"
+	/bin/mkdir -p "$p"
+	/bin/cp "$modmnt/$p/$m.kmod" "$p"
+done
+# zpool list will load the needed modules, then we can dispose of the mounts.
+/sbin/zpool list >/dev/null 2>&1
+/sbin/umount /stand
+/sbin/umount "$modmnt"
+echo
+
+echo "Importing $rpool, mounting and pivoting"
+# If we can mount the ZFS root partition to /altroot
+# then chroot to it and start /etc/rc
+if /sbin/zpool import -f -N "$rpool" && \
+   /sbin/mount -t zfs "$rpool/$rroot" /altroot; then
+	# This ensures that mountall mounts all ZFS mounts set to automount.
+	if [ ! -e /altroot/etc/zfs/zpool.cache ]; then
+		echo >/altroot/etc/zfs/zpool.cache
+	fi
+	/sbin/sysctl -w init.root=/altroot
+fi

Reply via email to