tinyramfs/init

147 lines
3.7 KiB
Plaintext
Raw Normal View History

#!/bin/sh
2020-01-30 17:40:13 +03:00
#
2021-05-10 14:35:34 +03:00
# Tiny init
#
2021-05-10 14:35:34 +03:00
# https://www.shellcheck.net/wiki/SC2154
2020-07-03 18:49:09 +03:00
# shellcheck disable=2154
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}" \
2021-05-10 14:35:34 +03:00
"\033[1;31m!!\033[m"
sh
} >&2
2020-01-28 18:13:42 +03:00
2020-06-02 14:26:42 +03:00
resolve_device()
2020-04-11 22:31:02 +03:00
{
count=0; device="$1"
2020-04-11 22:31:02 +03:00
case "${device%%=*}" in
UUID) device="/dev/disk/by-uuid/${device#*=}" ;;
LABEL) device="/dev/disk/by-label/${device#*=}" ;;
PARTUUID) device="/dev/disk/by-partuuid/${device#*=}" ;;
/dev/*) ;;
*) return 0 ;;
2020-04-11 22:31:02 +03:00
esac
2021-05-10 14:35:34 +03:00
# Race condition may occur if device manager is not yet initialized device.
# To fix this, we simply waiting until device is available. If device
# didn't appear in specified time, we panic.
while :; do
if [ -b "$device" ]; then
return 0
elif [ "$((count += 1))" = "${rootdelay:=30}" ]; then
break
2021-05-10 14:35:34 +03:00
else
sleep 1
fi
done
panic "failed to lookup partition"
}
run_hook()
{
2020-09-10 22:53:39 +03:00
type="$1"
2021-05-10 14:35:34 +03:00
# Run hooks if any exist.
#
# https://www.shellcheck.net/wiki/SC1090
2020-07-03 18:49:09 +03:00
# shellcheck disable=1090
for hook in $hooks; do
2020-09-10 22:53:39 +03:00
[ -f "/usr/share/tinyramfs/hooks/${hook}/${hook}.${type}" ] || continue
. "/usr/share/tinyramfs/hooks/${hook}/${hook}.${type}"
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()
{
2021-05-10 14:35:34 +03:00
# https://www.shellcheck.net/wiki/SC1091
2020-07-03 18:49:09 +03:00
# shellcheck disable=1091
. /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
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-11 22:31:02 +03:00
}
parse_cmdline()
{
read -r cmdline < /proc/cmdline
2020-07-05 11:11:39 +03:00
for line in $cmdline; do case "$line" in
rootfstype=*) root_type="${line#*=}" ;;
rootflags=*) root_opts="${line#*=}" ;;
debug=1) set -x ;;
ro | rw) rorw="$line" ;;
2020-07-05 11:11:39 +03:00
--*) init_args="${cmdline#*-- }"; break ;;
*=*) command export "$line" ;;
*) command export "${line}=1" ;;
esac 2> /dev/null || :; 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
mount_root()
{
[ "$break" = root ] && { print "break before mount_root()"; sh; }
2020-04-11 22:31:02 +03:00
2020-06-02 14:26:42 +03:00
resolve_device "$root"
2020-04-11 22:31:02 +03:00
2021-05-10 14:35:34 +03:00
# https://www.shellcheck.net/wiki/SC2086
# shellcheck disable=2086
mount \
-o "${rorw:-ro}${root_opts:+,$root_opts}" ${root_type:+-t $root_type} \
2021-05-10 14:35:34 +03:00
-- "$device" /mnt/root || 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
for dir in run dev sys proc; do
2020-07-26 06:31:20 +03:00
mount -o move "$dir" "/mnt/root/${dir}"
done
2020-01-05 21:01:39 +03:00
2021-05-10 14:35:34 +03:00
# POSIX 'exec' has no '-c' flag to execute command with empty environment.
# Using 'env -i' instead to prevent leaking exported variables.
#
# Some implementations of 'switch_root' doesn't conform to POSIX utility
# guidelines and doesn't support '--'. This means that we can't guarantee
# safety of init_args.
# shellcheck disable=2086
exec \
env -i TERM=linux PATH=/bin:/sbin:/usr/bin:/usr/sbin \
switch_root /mnt/root "${init-/sbin/init}" $init_args ||
panic "failed to boot system"
2020-04-11 22:31:02 +03:00
}
2020-01-30 18:28:03 +03:00
2021-05-10 14:35:34 +03:00
# Exit if command fails and disable globbing.
set -ef
# Run emergency shell if init unexpectedly exiting due to error.
trap panic EXIT
# TODO display fancy colored status info
2020-09-10 22:53:39 +03:00
prepare_environment
parse_cmdline
run_hook init
mount_root
run_hook init.late
boot_system