mirror of
https://git.proxmox.com/git/mirror_zfs.git
synced 2024-12-26 03:09:34 +03:00
contrib: dracut: fix race with root=zfs:dset when necessities required
This had always worked in my testing, but a user on hardware reported this to happen 100%, and I reproduced it once with cold VM host caches. dracut-zfs-generator runs as a systemd generator, i.e. at Some Relatively Early Time; if root= is a fixed dataset, it tries to "solve [necessities] statically at generation time". If by that point zfs-import.target hasn't popped (because the import is taking a non-negligible amount of time for whatever reason), it'll see no children for the root datase, and as such generate no mounts. This has never had any right to work. No-one caught this earlier because it's just that much more convenient to have root=zfs:AUTO, which orders itself properly. To fix this, always run zfs-nonroot-necessities.service; this additionally simplifies the implementation by: * making BOOTFS from zfs-env-bootfs.service be the real, canonical, root dataset name, not just "whatever the first bootfs is", and only set it if we're ZFS-booting * zfs-{rollback,snapshot}-bootfs.service can use this instead of re-implementing it * having zfs-env-bootfs.service also set BOOTFSFLAGS * this means the sysroot.mount drop-in can be fixed text * zfs-nonroot-necessities.service can also be constant and always enabled, because it's conditioned on BOOTFS being set There is no longer any code generated at run-time (the sysroot.mount drop-in is an unavoidable gratuitous cp). The flow of BOOTFS{,FLAGS} from zfs-env-bootfs.service to sysroot.mount is not noted explicitly in dracut.zfs(7), because (a) at some point it's just visual noise and (b) it's already ordered via d-p-m.s from z-i.t. Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Signed-off-by: Ahelenia Ziemiańska <nabijaczleweli@nabijaczleweli.xyz> Closes #14690
This commit is contained in:
parent
c5431f1465
commit
3399a30ee0
@ -81,6 +81,9 @@ install() {
|
|||||||
inst_simple "${moddir}/zfs-env-bootfs.service" "${systemdsystemunitdir}/zfs-env-bootfs.service"
|
inst_simple "${moddir}/zfs-env-bootfs.service" "${systemdsystemunitdir}/zfs-env-bootfs.service"
|
||||||
systemctl -q --root "${initdir}" add-wants zfs-import.target zfs-env-bootfs.service
|
systemctl -q --root "${initdir}" add-wants zfs-import.target zfs-env-bootfs.service
|
||||||
|
|
||||||
|
inst_simple "${moddir}/zfs-nonroot-necessities.service" "${systemdsystemunitdir}/zfs-nonroot-necessities.service"
|
||||||
|
systemctl -q --root "${initdir}" add-requires initrd-root-fs.target zfs-nonroot-necessities.service
|
||||||
|
|
||||||
# Add user-provided unit overrides:
|
# Add user-provided unit overrides:
|
||||||
# - /etc/systemd/system/${_service}
|
# - /etc/systemd/system/${_service}
|
||||||
# - /etc/systemd/system/${_service}.d/overrides.conf
|
# - /etc/systemd/system/${_service}.d/overrides.conf
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
[Unit]
|
[Unit]
|
||||||
Description=Set BOOTFS environment for dracut
|
Description=Set BOOTFS and BOOTFSFLAGS environment variables for dracut
|
||||||
Documentation=man:zpool(8)
|
|
||||||
DefaultDependencies=no
|
DefaultDependencies=no
|
||||||
After=zfs-import-cache.service
|
After=zfs-import-cache.service
|
||||||
After=zfs-import-scan.service
|
After=zfs-import-scan.service
|
||||||
@ -8,7 +7,17 @@ Before=zfs-import.target
|
|||||||
|
|
||||||
[Service]
|
[Service]
|
||||||
Type=oneshot
|
Type=oneshot
|
||||||
ExecStart=/bin/sh -c "exec systemctl set-environment BOOTFS=$(@sbindir@/zpool list -H -o bootfs | grep -m1 -vFx -)"
|
ExecStart=/bin/sh -c ' \
|
||||||
|
. /lib/dracut-zfs-lib.sh; \
|
||||||
|
decode_root_args || exit 0; \
|
||||||
|
[ "$root" = "zfs:AUTO" ] && root="$(@sbindir@/zpool list -H -o bootfs | grep -m1 -vFx -)"; \
|
||||||
|
rootflags="$(getarg rootflags=)"; \
|
||||||
|
case ",$rootflags," in \
|
||||||
|
*,zfsutil,*) ;; \
|
||||||
|
,,) rootflags=zfsutil ;; \
|
||||||
|
*) rootflags="zfsutil,$rootflags" ;; \
|
||||||
|
esac; \
|
||||||
|
exec systemctl set-environment BOOTFS="$root" BOOTFSFLAGS="$rootflags"'
|
||||||
|
|
||||||
[Install]
|
[Install]
|
||||||
WantedBy=zfs-import.target
|
WantedBy=zfs-import.target
|
||||||
|
@ -14,81 +14,24 @@ GENERATOR_DIR="$1"
|
|||||||
. /lib/dracut-zfs-lib.sh
|
. /lib/dracut-zfs-lib.sh
|
||||||
decode_root_args || exit 0
|
decode_root_args || exit 0
|
||||||
|
|
||||||
[ -z "${rootflags}" ] && rootflags=$(getarg rootflags=)
|
|
||||||
case ",${rootflags}," in
|
|
||||||
*,zfsutil,*) ;;
|
|
||||||
,,) rootflags=zfsutil ;;
|
|
||||||
*) rootflags="zfsutil,${rootflags}" ;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
[ -n "$debug" ] && echo "zfs-generator: writing extension for sysroot.mount to $GENERATOR_DIR/sysroot.mount.d/zfs-enhancement.conf" >> /dev/kmsg
|
[ -n "$debug" ] && echo "zfs-generator: writing extension for sysroot.mount to $GENERATOR_DIR/sysroot.mount.d/zfs-enhancement.conf" >> /dev/kmsg
|
||||||
|
|
||||||
|
|
||||||
mkdir -p "$GENERATOR_DIR"/sysroot.mount.d "$GENERATOR_DIR"/initrd-root-fs.target.requires "$GENERATOR_DIR"/dracut-pre-mount.service.d
|
mkdir -p "$GENERATOR_DIR"/sysroot.mount.d "$GENERATOR_DIR"/dracut-pre-mount.service.d
|
||||||
|
|
||||||
{
|
{
|
||||||
echo "[Unit]"
|
echo "[Unit]"
|
||||||
echo "Before=initrd-root-fs.target"
|
echo "Before=initrd-root-fs.target"
|
||||||
echo "After=zfs-import.target"
|
echo "After=zfs-import.target"
|
||||||
echo
|
echo
|
||||||
echo "[Mount]"
|
echo "[Mount]"
|
||||||
if [ "${root}" = "zfs:AUTO" ]; then
|
echo "PassEnvironment=BOOTFS BOOTFSFLAGS"
|
||||||
echo "PassEnvironment=BOOTFS"
|
echo 'What=${BOOTFS}'
|
||||||
echo 'What=${BOOTFS}'
|
|
||||||
else
|
|
||||||
echo "What=${root}"
|
|
||||||
fi
|
|
||||||
echo "Type=zfs"
|
echo "Type=zfs"
|
||||||
echo "Options=${rootflags}"
|
echo 'Options=${BOOTFSFLAGS}'
|
||||||
} > "$GENERATOR_DIR"/sysroot.mount.d/zfs-enhancement.conf
|
} > "$GENERATOR_DIR"/sysroot.mount.d/zfs-enhancement.conf
|
||||||
ln -fs ../sysroot.mount "$GENERATOR_DIR"/initrd-root-fs.target.requires/sysroot.mount
|
ln -fs ../sysroot.mount "$GENERATOR_DIR"/initrd-root-fs.target.requires/sysroot.mount
|
||||||
|
|
||||||
|
|
||||||
if [ "${root}" = "zfs:AUTO" ]; then
|
|
||||||
{
|
|
||||||
echo "[Unit]"
|
|
||||||
echo "Before=initrd-root-fs.target"
|
|
||||||
echo "After=sysroot.mount"
|
|
||||||
echo "DefaultDependencies=no"
|
|
||||||
echo
|
|
||||||
echo "[Service]"
|
|
||||||
echo "Type=oneshot"
|
|
||||||
echo "PassEnvironment=BOOTFS"
|
|
||||||
echo "ExecStart=/bin/sh -c '" ' \
|
|
||||||
. /lib/dracut-zfs-lib.sh; \
|
|
||||||
_zfs_nonroot_necessities_cb() { \
|
|
||||||
zfs mount | grep -m1 -q "^$1 " && return 0; \
|
|
||||||
echo "Mounting $1 on /sysroot$2"; \
|
|
||||||
mount -o zfsutil -t zfs "$1" "/sysroot$2"; \
|
|
||||||
}; \
|
|
||||||
for_relevant_root_children "${BOOTFS}" _zfs_nonroot_necessities_cb;' \
|
|
||||||
"'"
|
|
||||||
} > "$GENERATOR_DIR"/zfs-nonroot-necessities.service
|
|
||||||
ln -fs ../zfs-nonroot-necessities.service "$GENERATOR_DIR"/initrd-root-fs.target.requires/zfs-nonroot-necessities.service
|
|
||||||
else
|
|
||||||
# We can solve this statically at generation time, so do!
|
|
||||||
_zfs_generator_cb() {
|
|
||||||
dset="${1}"
|
|
||||||
mpnt="${2}"
|
|
||||||
unit="$(systemd-escape --suffix=mount -p "/sysroot${mpnt}")"
|
|
||||||
|
|
||||||
{
|
|
||||||
echo "[Unit]"
|
|
||||||
echo "Before=initrd-root-fs.target"
|
|
||||||
echo "After=sysroot.mount"
|
|
||||||
echo
|
|
||||||
echo "[Mount]"
|
|
||||||
echo "Where=/sysroot${mpnt}"
|
|
||||||
echo "What=${dset}"
|
|
||||||
echo "Type=zfs"
|
|
||||||
echo "Options=zfsutil"
|
|
||||||
} > "$GENERATOR_DIR/${unit}"
|
|
||||||
ln -fs ../"${unit}" "$GENERATOR_DIR"/initrd-root-fs.target.requires/"${unit}"
|
|
||||||
}
|
|
||||||
|
|
||||||
for_relevant_root_children "${root}" _zfs_generator_cb
|
|
||||||
fi
|
|
||||||
|
|
||||||
|
|
||||||
{
|
{
|
||||||
echo "[Unit]"
|
echo "[Unit]"
|
||||||
echo "After=zfs-import.target"
|
echo "After=zfs-import.target"
|
||||||
|
@ -39,7 +39,7 @@ mount_dataset() {
|
|||||||
|
|
||||||
# for_relevant_root_children DATASET EXEC
|
# for_relevant_root_children DATASET EXEC
|
||||||
# Runs "EXEC dataset mountpoint" for all children of DATASET that are needed for system bringup
|
# Runs "EXEC dataset mountpoint" for all children of DATASET that are needed for system bringup
|
||||||
# Used by zfs-generator.sh and friends, too!
|
# Used by zfs-nonroot-necessities.service and friends, too!
|
||||||
for_relevant_root_children() {
|
for_relevant_root_children() {
|
||||||
dataset="${1}"
|
dataset="${1}"
|
||||||
exec="${2}"
|
exec="${2}"
|
||||||
|
20
contrib/dracut/90zfs/zfs-nonroot-necessities.service.in
Normal file
20
contrib/dracut/90zfs/zfs-nonroot-necessities.service.in
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
[Unit]
|
||||||
|
Before=initrd-root-fs.target
|
||||||
|
After=sysroot.mount
|
||||||
|
DefaultDependencies=no
|
||||||
|
ConditionEnvironment=BOOTFS
|
||||||
|
|
||||||
|
[Service]
|
||||||
|
Type=oneshot
|
||||||
|
PassEnvironment=BOOTFS
|
||||||
|
ExecStart=/bin/sh -c ' \
|
||||||
|
. /lib/dracut-zfs-lib.sh; \
|
||||||
|
_zfs_nonroot_necessities_cb() { \
|
||||||
|
@sbindir@/zfs mount | grep -m1 -q "^$1 " && return 0; \
|
||||||
|
echo "Mounting $1 on /sysroot$2"; \
|
||||||
|
mount -o zfsutil -t zfs "$1" "/sysroot$2"; \
|
||||||
|
}; \
|
||||||
|
for_relevant_root_children "${BOOTFS}" _zfs_nonroot_necessities_cb'
|
||||||
|
|
||||||
|
[Install]
|
||||||
|
RequiredBy=initrd-root-fs.target
|
@ -5,8 +5,9 @@ After=zfs-import.target dracut-pre-mount.service zfs-snapshot-bootfs.service
|
|||||||
Before=dracut-mount.service
|
Before=dracut-mount.service
|
||||||
DefaultDependencies=no
|
DefaultDependencies=no
|
||||||
ConditionKernelCommandLine=bootfs.rollback
|
ConditionKernelCommandLine=bootfs.rollback
|
||||||
|
ConditionEnvironment=BOOTFS
|
||||||
|
|
||||||
[Service]
|
[Service]
|
||||||
Type=oneshot
|
Type=oneshot
|
||||||
ExecStart=/bin/sh -c '. /lib/dracut-zfs-lib.sh; decode_root_args || exit; [ "$root" = "zfs:AUTO" ] && root="$BOOTFS"; SNAPNAME="$(getarg bootfs.rollback)"; exec @sbindir@/zfs rollback -Rf "$root@${SNAPNAME:-%v}"'
|
ExecStart=/bin/sh -c '. /lib/dracut-lib.sh; SNAPNAME="$(getarg bootfs.rollback)"; exec @sbindir@/zfs rollback -Rf "$BOOTFS@${SNAPNAME:-%v}"'
|
||||||
RemainAfterExit=yes
|
RemainAfterExit=yes
|
||||||
|
@ -5,8 +5,9 @@ After=zfs-import.target dracut-pre-mount.service
|
|||||||
Before=dracut-mount.service
|
Before=dracut-mount.service
|
||||||
DefaultDependencies=no
|
DefaultDependencies=no
|
||||||
ConditionKernelCommandLine=bootfs.snapshot
|
ConditionKernelCommandLine=bootfs.snapshot
|
||||||
|
ConditionEnvironment=BOOTFS
|
||||||
|
|
||||||
[Service]
|
[Service]
|
||||||
Type=oneshot
|
Type=oneshot
|
||||||
ExecStart=-/bin/sh -c '. /lib/dracut-zfs-lib.sh; decode_root_args || exit; [ "$root" = "zfs:AUTO" ] && root="$BOOTFS"; SNAPNAME="$(getarg bootfs.snapshot)"; exec @sbindir@/zfs snapshot "$root@${SNAPNAME:-%v}"'
|
ExecStart=-/bin/sh -c '. /lib/dracut-lib.sh; SNAPNAME="$(getarg bootfs.snapshot)"; exec @sbindir@/zfs snapshot "$BOOTFS@${SNAPNAME:-%v}"'
|
||||||
RemainAfterExit=yes
|
RemainAfterExit=yes
|
||||||
|
@ -16,6 +16,7 @@ pkgdracut_90_SCRIPTS = \
|
|||||||
|
|
||||||
pkgdracut_90_DATA = \
|
pkgdracut_90_DATA = \
|
||||||
%D%/90zfs/zfs-env-bootfs.service \
|
%D%/90zfs/zfs-env-bootfs.service \
|
||||||
|
%D%/90zfs/zfs-nonroot-necessities.service \
|
||||||
%D%/90zfs/zfs-rollback-bootfs.service \
|
%D%/90zfs/zfs-rollback-bootfs.service \
|
||||||
%D%/90zfs/zfs-snapshot-bootfs.service
|
%D%/90zfs/zfs-snapshot-bootfs.service
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
.\" SPDX-License-Identifier: 0BSD
|
.\" SPDX-License-Identifier: 0BSD
|
||||||
.\"
|
.\"
|
||||||
.Dd April 4, 2022
|
.Dd March 28, 2023
|
||||||
.Dt DRACUT.ZFS 7
|
.Dt DRACUT.ZFS 7
|
||||||
.Os
|
.Os
|
||||||
.
|
.
|
||||||
@ -28,13 +28,13 @@ zfs-import-scan.service \(da \(da | zfs-import-c
|
|||||||
zfs-import.target \(-> dracut-pre-mount.service
|
zfs-import.target \(-> dracut-pre-mount.service
|
||||||
| \(ua |
|
| \(ua |
|
||||||
| dracut-zfs-generator |
|
| dracut-zfs-generator |
|
||||||
| ____________________/|
|
| _____________________/|
|
||||||
|/ \(da
|
|/ \(da
|
||||||
| sysroot.mount \(<-\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em dracut-zfs-generator
|
| sysroot.mount \(<-\(em\(em\(em dracut-zfs-generator
|
||||||
| | \(da |
|
| |
|
||||||
| \(da sysroot-{usr,etc,lib,&c.}.mount |
|
| \(da
|
||||||
| initrd-root-fs.target \(<-\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em or \(da
|
| initrd-root-fs.target \(<-\(em zfs-nonroot-necessities.service
|
||||||
| | zfs-nonroot-necessities.service
|
| | |
|
||||||
| \(da |
|
| \(da |
|
||||||
\(da dracut-mount.service |
|
\(da dracut-mount.service |
|
||||||
zfs-snapshot-bootfs.service | |
|
zfs-snapshot-bootfs.service | |
|
||||||
@ -42,7 +42,7 @@ zfs-import-scan.service \(da \(da | zfs-import-c
|
|||||||
\(da … |
|
\(da … |
|
||||||
zfs-rollback-bootfs.service | |
|
zfs-rollback-bootfs.service | |
|
||||||
| \(da |
|
| \(da |
|
||||||
| sysroot-usr.mount \(<-\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em/
|
| /sysroot/{usr,etc,lib,&c.} \(<-\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em\(em/
|
||||||
| |
|
| |
|
||||||
| \(da
|
| \(da
|
||||||
| initrd-fs.target
|
| initrd-fs.target
|
||||||
|
Loading…
Reference in New Issue
Block a user