tinyramfs/init

281 lines
6.0 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
2020-04-21 17:35:55 +03:00
shell
}
2020-02-24 11:19:38 +03:00
shell()
{
2020-04-11 22:31:02 +03:00
# see https://busybox.net/FAQ.html#job_control
2020-04-18 18:12:24 +03:00
setsid sh -c "exec sh <> /dev/${console:-tty1} 2>&1" || 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##*=}"
;;
esac
# avoid race condition
while [ ! -e "$device" ]; do
sleep 1; : $(( count += 1 ))
[ "$count" = 30 ] && {
panic "failed to lookup partition"
break
}
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 \
LANG=C \
2020-04-11 22:31:02 +03:00
PS1="# " \
LC_ALL=C \
TERM=linux \
HOME=/root \
SHELL=/bin/sh \
OLD_IFS="$IFS" \
PATH=/bin:/sbin:/usr/bin:/usr/sbin
2020-03-08 06:29:14 +03:00
2020-04-11 22:31:02 +03:00
# fix for ubase mount
: > /etc/fstab
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" ] || return 0
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-04-21 17:35:55 +03:00
rootfstype=*)
root_type="$line"
;;
rootflags=*)
root_opts="$line"
;;
ro | rw)
rorw="-o $line"
;;
*.*)
# TODO implement backward compatibilty with dracut, mkinitcpio, etc
: no operation
;;
*=*)
export "$line" || :
;;
*)
export "${line}=1" || :
;;
esac; done
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()"; shell; }
2020-04-11 22:31:02 +03:00
2020-02-25 22:43:25 +03:00
case "$devmgr" in
udev)
udevd -dN never
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)
2020-04-11 22:31:02 +03:00
mdev -df 2> /dev/null & mdev_pid="$!"
mdev -s
[ "$monolith" = 1 ] && return 0
2020-04-11 22:31:02 +03:00
# TODO maybe replace sort using while-read loop
# while-read can cause slowdown, so why i'm unsure
set -- $(find /sys -name modalias -type f -exec sort -u {} +)
modprobe -a "$@" 2> /dev/null || :
2020-04-11 22:31:02 +03:00
;;
2020-02-25 22:43:25 +03:00
mdevd)
2020-04-11 22:31:02 +03:00
mdevd 2> /dev/null & mdevd_pid="$!"
2020-02-25 22:43:25 +03:00
mdevd-coldplug
2020-04-11 22:31:02 +03:00
;;
2020-02-25 22:43:25 +03:00
esac
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()"; shell; }
2020-02-25 22:43:25 +03:00
2020-04-11 22:31:02 +03:00
{ IFS=,; set -- $luks_opts; IFS="$OLD_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-04-17 20:41:54 +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()"; shell; }
2020-04-11 22:31:02 +03:00
{ IFS=,; set -- $lvm_opts; IFS="$OLD_IFS"; }
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-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()"; shell; }
2020-04-11 22:31:02 +03:00
findfs "$root"
set -- \
"${root_type:+-t $root_type}" \
"${rorw:--o ro}${root_opts:+,$root_opts}" \
"$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-04-11 22:31:02 +03:00
cleanup()
{
[ "$break" = cleanup ] && { print "break before cleanup()"; shell; }
2020-04-11 22:31:02 +03:00
case "$devmgr" in
2020-02-25 22:43:25 +03:00
udev) udevadm control -e ;;
2020-04-11 22:31:02 +03:00
mdev) kill "$mdev_pid" ;;
mdevd) kill "$mdevd_pid" ;;
esac
# 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-30 17:40:13 +03:00
}
2020-01-05 21:01:39 +03:00
2020-04-11 22:31:02 +03:00
boot_system()
{
[ "$break" = boot ] && { print "break before boot_system()"; shell; }
2020-01-30 17:40:13 +03:00
2020-04-11 22:31:02 +03:00
set -- "/mnt/root" "${init:-/sbin/init}"
exec switch_root $@ 2> /dev/null || panic "failed to boot system"
}
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
cleanup
boot_system
}