tinyramfs/init

212 lines
5.1 KiB
Plaintext
Raw Normal View History

2020-04-11 22:31:02 +03:00
#!/bin/sh -ef
2020-01-30 17:40:13 +03:00
#
2020-04-11 22:31:02 +03:00
# tiny init
#
# word splitting is safe by design
# shellcheck disable=2068,2046,2086
#
# false positive
# shellcheck disable=2154,2163,1091
2020-01-05 21:01:39 +03:00
2020-04-21 17:35:55 +03:00
print()
{
printf "%b %s\n" "${2:-"\033[1;37m>>\033[m"}" "$1"
2020-04-21 17:35:55 +03:00
}
panic()
{
print "${1:-unexpected error occurred}" \
"\033[1;31m!!\033[m" >&2; sh
2020-01-30 17:40:13 +03:00
}
2020-01-28 18:13:42 +03:00
2020-04-11 22:31:02 +03:00
findfs()
{
count=0; device=
2020-04-11 22:31:02 +03:00
case "${1%%=*}" in
/dev/*) device="$1" ;;
UUID) device="/dev/disk/by-uuid/${1##*=}" ;;
LABEL) device="/dev/disk/by-label/${1##*=}" ;;
PARTUUID) device="/dev/disk/by-partuuid/${1##*=}" ;;
2020-04-11 22:31:02 +03:00
esac
# prevent race condition
while [ ! -e "$device" ]; do sleep 1
2020-05-26 16:44:20 +03:00
[ "$(( count += 1 ))" != 30 ] || {
panic "failed to lookup partition"
break
}
2020-05-26 16:44:20 +03:00
done
2020-02-21 11:57:07 +03:00
}
2020-01-05 21:01:39 +03:00
2020-04-11 22:31:02 +03:00
prepare_environment()
{
. /etc/tinyramfs/config
2020-04-11 22:31:02 +03:00
export \
PATH=/bin TERM=linux SHELL=/bin/sh \
LANG=C LC_ALL=C PS1="# " HOME=/root \
2020-03-08 06:29:14 +03:00
mount -t proc -o nosuid,noexec,nodev proc /proc
mount -t sysfs -o nosuid,noexec,nodev sys /sys
mount -t tmpfs -o nosuid,nodev,mode=0755 run /run
2020-04-11 22:31:02 +03:00
mount -t devtmpfs -o nosuid,noexec,mode=0755 dev /dev
2020-03-08 06:29:14 +03:00
mkdir -p \
/run/cryptsetup \
/run/lock \
/run/lvm
2020-04-11 22:31:02 +03:00
ln -s /proc/self/fd /dev/fd
ln -s fd/0 /dev/stdin
ln -s fd/1 /dev/stdout
ln -s fd/2 /dev/stderr
2020-04-17 20:41:54 +03:00
trap panic EXIT
[ ! "$modules" ] || modprobe -a "$modules"
2020-04-11 22:31:02 +03:00
}
parse_cmdline()
{
read -r cmdline < /proc/cmdline
for line in $cmdline; do case "$line" in
debug | debug=1) set -x ;;
2020-05-26 16:44:20 +03:00
rootfstype=*) root_type="${line##*=}" ;;
rootflags=*) root_opts="${line##*=}" ;;
ro | rw) rorw="-o $line" ;;
--) init_args="${cmdline##*--}"; break ;;
*=*) command export "$line" ;;
*) command export "${line}=1" ;;
esac; done 2> /dev/null || :
2020-01-30 17:40:13 +03:00
}
2020-01-05 21:01:39 +03:00
2020-04-11 22:31:02 +03:00
setup_devmgr()
{
[ "$break" = devmgr ] && { print "break before setup_devmgr()"; sh; }
2020-04-11 22:31:02 +03:00
2020-02-25 22:43:25 +03:00
case "$devmgr" in
udev)
udevd -N never & devmgr_pid="$!"
2020-02-25 22:43:25 +03:00
udevadm trigger -c add -t subsystems
udevadm trigger -c add -t devices
udevadm settle
2020-04-11 22:31:02 +03:00
;;
2020-02-25 22:43:25 +03:00
mdev)
mdev -s
mdev -df & devmgr_pid="$!"
[ "$monolith" = 1 ] && return 0
2020-04-11 22:31:02 +03:00
2020-05-26 16:44:20 +03:00
find /sys/devices -name uevent |
while read -r uevent; do
printf add > "$uevent"
done
2020-04-11 22:31:02 +03:00
;;
2020-02-25 22:43:25 +03:00
mdevd)
mdevd & devmgr_pid="$!"
2020-02-25 22:43:25 +03:00
mdevd-coldplug
2020-04-11 22:31:02 +03:00
;;
esac 2> /dev/null
2020-01-30 17:40:13 +03:00
}
2020-01-05 21:01:39 +03:00
2020-04-11 22:31:02 +03:00
unlock_luks()
{
[ "$break" = luks ] && { print "break before unlock_luks()"; sh; }
2020-02-25 22:43:25 +03:00
{ IFS=,; set -- $luks_opts; unset IFS; }
2020-02-15 22:33:13 +03:00
for opt; do case "$opt" in
discard | discard=1) luks_discard="--allow-discards" ;;
header=*) luks_header="--${opt}" ;;
name=*) luks_name="${opt##*=}" ;;
key=*) luks_key="-d ${opt##*=}" ;;
esac; done
2020-02-15 22:33:13 +03:00
2020-04-11 22:31:02 +03:00
findfs "$luks_root"
set -- \
2020-05-26 16:44:20 +03:00
"$luks_key" "$luks_header" "$luks_discard" \
"$device" "${luks_name:-luks-${device##*/}}"
2020-04-11 22:31:02 +03:00
cryptsetup open $@ || panic "failed to unlock LUKS"
}
2020-04-11 22:31:02 +03:00
trigger_lvm()
{
[ "$break" = lvm ] && { print "break before trigger_lvm()"; sh; }
{ IFS=,; set -- $lvm_opts; unset IFS; }
2020-04-11 22:31:02 +03:00
for opt; do case "$opt" in
discard | discard=1) lvm_discard="--config=devices{issue_discards=1}" ;;
config=0) : > /etc/lvm/lvm.conf ;;
group=*) lvm_group="${opt##*=}" ;;
name=*) lvm_name="/${opt##*=}" ;;
tag=*) lvm_tag="@${opt##*=}" ;;
esac; done
2020-04-11 22:31:02 +03:00
set -- "--sysinit" "-qq" "-aay" "$lvm_discard"
2020-01-25 14:27:02 +03:00
2020-02-21 11:57:07 +03:00
if [ "$lvm_group" ] && [ "$lvm_name" ]; then
lvm lvchange $@ "${lvm_group}${lvm_name}"
2020-02-21 11:57:07 +03:00
elif [ "$lvm_group" ]; then
lvm vgchange $@ "$lvm_group"
2020-04-11 22:31:02 +03:00
elif [ "$lvm_tag" ]; then
lvm lvchange $@ "$lvm_tag"
2020-02-21 11:57:07 +03:00
else
lvm vgchange $@
2020-02-21 11:57:07 +03:00
fi
2020-05-26 17:00:30 +03:00
[ "$?" = 0 ] || panic "failed to trigger LVM"
2020-01-28 18:13:42 +03:00
}
2020-01-05 21:01:39 +03:00
2020-04-11 22:31:02 +03:00
mount_root()
{
[ "$break" = root ] && { print "break before mount_root()"; sh; }
2020-04-11 22:31:02 +03:00
findfs "$root"
set -- \
"${rorw:--o ro}${root_opts:+,$root_opts}" \
2020-05-26 16:44:20 +03:00
"${root_type:+-t $root_type}" "$device" "/mnt/root"
2020-04-11 22:31:02 +03:00
mount $@ || panic "failed to mount root"
2020-01-30 17:40:13 +03:00
}
2020-01-05 21:01:39 +03:00
2020-05-26 16:44:20 +03:00
boot_system()
2020-04-11 22:31:02 +03:00
{
2020-05-26 16:44:20 +03:00
[ "$break" = boot ] && { print "break before boot_system()"; sh; }
2020-04-11 22:31:02 +03:00
kill "$devmgr_pid"
# temporary workaround until util-linux release a new version
2020-04-11 22:31:02 +03:00
# see https://github.com/karelzak/util-linux/issues/997
for dir in run dev sys proc; do
mount -o move "$dir" "/mnt/root/${dir}" ||
mount --move "$dir" "/mnt/root/${dir}"
2020-04-11 22:31:02 +03:00
done
2020-01-05 21:01:39 +03:00
2020-05-26 16:44:20 +03:00
set -- \
"/mnt/root" "${init:-/sbin/init}" "$init_args"
2020-01-30 17:40:13 +03:00
2020-05-26 16:44:20 +03:00
exec switch_root $@ || panic "failed to boot system"
2020-04-11 22:31:02 +03:00
}
2020-01-30 18:28:03 +03:00
2020-04-11 22:31:02 +03:00
# int main()
{
prepare_environment
parse_cmdline
setup_devmgr
2020-02-25 22:43:25 +03:00
# trigger lvm twice to handle both LUKS on LVM and LVM on LUKS
[ "$lvm" = 1 ] && trigger_lvm
2020-04-11 22:31:02 +03:00
[ "$luks" = 1 ] && unlock_luks
[ "$lvm" = 1 ] && trigger_lvm
2020-02-25 22:43:25 +03:00
2020-04-11 22:31:02 +03:00
mount_root
boot_system
}