tinyramfs/generate

332 lines
7.9 KiB
Plaintext
Raw Normal View History

2020-01-05 21:50:12 +03:00
#!/bin/sh
2020-01-05 21:01:39 +03:00
#
# tiny initramfs generation tool
2020-01-30 16:53:17 +03:00
panic() {
printf "panic >> %s\n" "$@"
2020-01-09 18:46:50 +03:00
exit 1
2020-01-30 16:53:17 +03:00
}
2020-01-05 21:01:39 +03:00
2020-01-30 16:53:17 +03:00
info() {
printf "info >> %s\n" "$@"
}
2020-01-05 21:01:39 +03:00
2020-02-02 15:19:20 +03:00
# remove tmpdir
2020-01-30 16:53:17 +03:00
remove_tmpdir() {
rm -rf "$tmpdir"
}
2020-02-02 15:19:20 +03:00
# change current directory to script directory if user haven't do it
check_currentdir() {
script_dir=$(dirname $(readlink -f -- "$0"))
[ "$PWD" = "$script_dir" ] || {
cd "$script_dir" || panic "failed to change directory"
}
}
# check needed files
2020-01-30 16:53:17 +03:00
check_requirements() {
2020-02-02 15:19:20 +03:00
# TODO use system busybox
2020-02-02 16:00:23 +03:00
for f in ./init ./busybox; do
2020-01-30 16:53:17 +03:00
[ -e "$f" ] || panic "$f doesn't exists"
done
# TODO handle busybox requirements ( busybox --list | grep ash )
}
2020-02-02 15:19:20 +03:00
# create FHS directory structure
2020-01-30 16:53:17 +03:00
create_structure() {
for d in dev tmp var run etc usr/lib usr/bin mnt/root proc root sys; do
mkdir -p "${tmpdir}/${d}"
done
}
2020-02-02 15:19:20 +03:00
# some dynamically linked libraries and binaries compiled with hardcoded
# dependencies path. to make it worked we need create symlinks for them.
# also POSIX ln doesn't have --relative flag like in GNU ln. as workaround
# we change directory to tmpdir and make needed symlinks.
2020-01-30 16:53:17 +03:00
create_symlinks() {
( cd "$tmpdir" && {
ln -s usr/lib lib
ln -s usr/lib lib64
ln -s usr/bin bin
ln -s usr/bin sbin
ln -s ../run var/run
2020-02-02 16:00:23 +03:00
cd "${tmpdir}/usr"
2020-01-30 16:53:17 +03:00
ln -s bin sbin
ln -s lib lib64
} )
}
#parse_fstab() {
# TODO parse fstab
2020-01-05 21:01:39 +03:00
#while [ "$use_fstab" -eq 1 ] && read fs dir type opts; do thing; done < /etc/fstab
2020-01-30 16:53:17 +03:00
#}
#parse_crypttab() {
# TODO parse crypttab
#}
2020-01-05 21:01:39 +03:00
2020-02-02 15:19:20 +03:00
# install mdev
2020-02-02 16:10:43 +03:00
install_mdev() {
2020-02-02 16:00:23 +03:00
install -m644 mdev.conf -t "${tmpdir}/etc"
install -Dm755 storage-device -t "${tmpdir}/lib/mdev"
2020-01-30 16:53:17 +03:00
}
2020-01-19 00:01:21 +03:00
2020-02-02 15:19:20 +03:00
# install mdevd
2020-02-02 16:10:43 +03:00
install_mdevd() {
2020-01-30 16:53:17 +03:00
install_binaries mdevd mdevd-coldplug
2020-02-02 16:00:23 +03:00
install -m644 mdev.conf -t "${tmpdir}/etc"
install -Dm755 storage-device -t "${tmpdir}/lib/mdev"
2020-01-30 16:53:17 +03:00
}
2020-01-25 14:27:02 +03:00
2020-02-02 15:19:20 +03:00
# install udev
2020-02-02 16:10:43 +03:00
install_udev() {
2020-01-30 16:53:17 +03:00
install_binaries udevd udevadm dmsetup
# FIXME rewrite this piece of crap
find /usr/lib/udev -type f | grep -v "rc_keymaps\|hwdb.d" | cpio -pd "$tmpdir" >/dev/null 2>&1
}
2020-01-25 14:27:02 +03:00
2020-01-28 18:13:42 +03:00
# handle lvm
2020-02-02 16:10:43 +03:00
install_lvm() {
2020-01-30 16:53:17 +03:00
install_binaries lvm
# FIXME this code doesn't working with udev
2020-01-28 18:13:42 +03:00
#mkdir "$tmpdir/etc/lvm"
# use_lvmetad = 0 - avoid lvmetad missing warning message
#cat <<EOF > "$tmpdir/etc/lvm/lvmlocal.conf"
#local {
# issue_discards = ${lvm_discard:-0}
# use_lvmetad = 0
#}
#EOF
2020-01-25 14:27:02 +03:00
# TODO implement use_lvmconf
2020-01-30 16:53:17 +03:00
}
2020-01-19 00:01:21 +03:00
2020-01-25 14:27:02 +03:00
# handle luks
2020-02-02 16:10:43 +03:00
install_luks() {
2020-01-30 16:53:17 +03:00
install_binaries cryptsetup
2020-01-25 14:27:02 +03:00
# avoid locking directory missing warning message
2020-01-30 16:53:17 +03:00
mkdir "${tmpdir}/run/cryptsetup"
2020-01-25 14:27:02 +03:00
# TODO get rid of this workaround
# workaround for luks2
2020-01-30 16:53:17 +03:00
install -s -m755 /usr/lib/libgcc_s.so.1 -t "${tmpdir}/usr/lib"
2020-01-25 14:27:02 +03:00
2020-01-28 18:13:42 +03:00
[ "$luks_discard" = 1 ] && luks_args="--allow-discards $luks_args"
2020-01-25 14:27:02 +03:00
# TODO detached header
# TODO keyfile
2020-01-30 16:53:17 +03:00
}
2020-01-19 00:01:21 +03:00
2020-01-05 21:01:39 +03:00
# install drivers
2020-01-30 16:53:17 +03:00
install_drivers() {
modker="${moddir}${kernel}"
2020-02-06 01:02:46 +03:00
# TODO reimplement using functions
2020-02-05 16:35:17 +03:00
if [ "$hostonly" = 1 ]; then
2020-02-05 16:58:58 +03:00
[ -n "$root_type" ] || panic "hostonly mode required root_type option to be configured"
2020-02-06 01:02:46 +03:00
for driver in $(find /sys/devices -name modalias -exec sort -u "{}" "+"); do
for driver_dep in $(modprobe -D "$driver" 2>/dev/null | cut -d " " -f 2); do
install -Dm644 "$driver_dep" "${tmpdir}${driver_dep}"
2020-02-05 16:35:17 +03:00
done
done
2020-02-05 16:58:58 +03:00
2020-02-06 01:02:46 +03:00
# TODO separate root type option
# install root fs driver
2020-02-05 21:03:23 +03:00
for root_driver in $(modprobe -D "$root_type" 2>/dev/null | cut -d " " -f 2); do
2020-02-05 16:58:58 +03:00
install -Dm644 "$root_driver" "${tmpdir}${root_driver}"
done
2020-02-06 01:02:46 +03:00
# TODO move to install_luks function
# install LUKS drivers
[ "$use_luks" = 1 ] && {
for luks_driver in aes dm-crypt sha256 sha512 wp512 ecb lrw xts twofish serpent; do
for luks_driver_dep in $(modprobe -D "$luks_driver" 2>/dev/null | cut -d " " -f 2); do
install -Dm644 "${luks_driver_dep}" "${tmpdir}${luks_driver_dep}"
done
done
}
# TODO move to install_lvm function
# install LVM drivers
[ "$use_lvm" = 1 ] && {
for lvm_driver in dm-thin-pool dm-multipath dm-snapshot dm-cache dm-log dm-mirror; do
for lvm_driver_dep in $(modprobe -D "$lvm_driver" 2>/dev/null | cut -d " " -f 2); do
install -Dm644 "$lvm_driver_dep" "${tmpdir}${lvm_driver_dep}"
done
done
}
2020-02-05 16:35:17 +03:00
else
find \
"${modker}/kernel/drivers/virtio" \
"${modker}/kernel/arch" \
"${modker}/kernel/crypto" \
"${modker}/kernel/fs" \
"${modker}/kernel/lib" \
"${modker}/kernel/drivers/block" \
"${modker}/kernel/drivers/ata" \
"${modker}/kernel/drivers/md" \
"${modker}/kernel/drivers/scsi" \
"${modker}/kernel/drivers/usb/storage" \
"${modker}/kernel/drivers/usb/host" \
-type f | cpio -pd "$tmpdir" >/dev/null 2>&1
fi
2020-01-30 16:53:17 +03:00
# install list of drivers
cp "${modker}/modules.softdep" "${modker}/modules.builtin" "${modker}/modules.order" "${tmpdir}/${modker}"
# generate dependencies list of drivers
depmod -b "$tmpdir" "$kernel"
}
# TODO make strip optional
# handle binaries
install_binaries() {
for b in $(printf "%s\n" "$@" | tr " " "\n"); do
# check binary existence
command -v "$b" >/dev/null 2>&1 || panic "$b doesn't exists"
# install and strip binary
install -s -m755 "$(command -v $b)" -t "${tmpdir}/usr/bin"
# check statically linking
ldd "$(command -v $b)" >/dev/null 2>&1 || continue
# install libraries
install_libraries $b
done
}
2020-01-05 21:01:39 +03:00
2020-01-13 05:22:45 +03:00
# TODO make strip optional
2020-01-30 16:53:17 +03:00
# handle libraries
install_libraries() {
for l in $(ldd "$(command -v $1)" | sed -nre 's,.* (/.*lib.*/.*.so.*) .*,\1,p' -e 's,.*(/lib.*/ld.*.so.*) .*,\1,p'); do
2020-01-12 00:12:24 +03:00
# check symlink
2020-01-07 00:39:15 +03:00
if [ -h "$l" ]; then
2020-01-13 05:22:45 +03:00
# check lib already existence
2020-01-30 16:53:17 +03:00
if [ ! -e "${tmpdir}/usr/lib/${l##*/}" ] && [ ! -e "${tmpdir}/$(readlink -f $l)" ]; then
2020-01-12 00:12:24 +03:00
# regular
2020-01-30 16:53:17 +03:00
install -s -m755 "$(readlink -f $l)" -t "${tmpdir}/usr/lib"
# FIXME handle all symlinks
2020-01-28 18:13:42 +03:00
# symlink may link to symlink
2020-01-30 16:53:17 +03:00
[ -h "/usr/lib/$(readlink $l)" ] && cp -a "/usr/lib/$(readlink $l)" "${tmpdir}/usr/lib"
2020-01-12 00:12:24 +03:00
# symlink
2020-01-30 16:53:17 +03:00
cp -a "$l" "${tmpdir}/usr/lib"
2020-01-12 00:12:24 +03:00
fi
2020-01-07 00:39:15 +03:00
else
2020-01-30 16:53:17 +03:00
if [ ! -e "${tmpdir}/usr/lib/${l##*/}" ]; then
install -s -m755 "$l" -t "${tmpdir}/usr/lib"
2020-01-12 00:12:24 +03:00
fi
2020-01-07 00:39:15 +03:00
fi
done
2020-01-30 16:53:17 +03:00
}
2020-01-05 21:01:39 +03:00
2020-01-30 16:53:17 +03:00
install_files() {
2020-01-30 18:28:03 +03:00
# FIXME eof broken
2020-01-07 00:39:15 +03:00
# initialize config
2020-01-30 16:53:17 +03:00
cat <<EOF > "${tmpdir}/config"
2020-01-30 17:47:23 +03:00
debug="$debug"
2020-01-05 21:01:39 +03:00
root="$root"
2020-01-25 14:27:02 +03:00
root_type="$root_type"
root_args="$root_args"
2020-01-30 16:53:17 +03:00
devmgr="$devmgr"
2020-01-25 14:27:02 +03:00
#drivers="$drivers"
use_lvm="$use_lvm"
use_luks="$use_luks"
luks_root="$luks_root"
luks_header="$luks_header"
luks_keyfile="$luks_keyfile"
luks_args="$luks_args"
2020-01-05 21:01:39 +03:00
EOF
2020-01-30 16:53:17 +03:00
# needed for devmgr
cat <<EOF > "${tmpdir}/etc/group"
root:x:0:
tty:x:5:
dialout:x:11:
uucp:x:14:
kmem:x:3:
input:x:25:
video:x:13:
audio:x:12:
lp:x:10:
disk:x:9:
cdrom:x:16:
tape:x:6:
kvm:x:24:
floppy:x:8:
EOF
# needed for devmgr
cat <<EOF > "${tmpdir}/etc/passwd"
root:x:0:0::/root:/bin/sh
nobody:x:99:99::/:/bin/false
EOF
# install init script
install -m755 ./init -t "$tmpdir"
}
# TODO add more compession tools
2020-01-30 18:34:33 +03:00
# create and compress cpio archive
2020-01-30 16:53:17 +03:00
create_initramfs() {
{
( cd "$tmpdir" && {
find . | cpio -oH newc | gzip -9
2020-02-02 16:00:23 +03:00
} ) > "${script_dir}/initramfs-${kernel}.img.gz"
2020-01-30 16:53:17 +03:00
} >/dev/null 2>&1
[ "$?" = 0 ] || panic "failed to generate initramfs image"
}
2020-02-02 15:19:20 +03:00
# check root
2020-01-30 16:53:17 +03:00
[ "$(id -u)" = 0 ] || panic "must be run as root"
2020-02-05 16:35:17 +03:00
# remove tmpdir on exit or unexpected error
trap remove_tmpdir EXIT INT
2020-02-02 15:19:20 +03:00
check_currentdir
2020-01-30 16:53:17 +03:00
2020-02-02 15:19:20 +03:00
# source config
. ./config || panic "./config doesn't exists"
2020-01-30 18:28:03 +03:00
# handle debug mode
[ "$debug" = 1 ] && {
2020-01-30 18:34:33 +03:00
# debug shell commands
2020-01-30 18:28:03 +03:00
set -x
2020-01-30 18:34:33 +03:00
# don't remove anything
2020-01-30 18:28:03 +03:00
trap : EXIT INT
}
2020-02-02 15:19:20 +03:00
# variables
tmpdir="$(mktemp -d /tmp/initramfs.XXXXXXXX)" || panic "failed to create working directory"
kernel="$(uname -r)"
moddir="/lib/modules/"
check_requirements
2020-01-30 16:53:17 +03:00
create_structure
create_symlinks
#parse_fstab
#parse_crypttab
install_binaries $binaries
install_drivers
install_files
2020-02-02 15:19:20 +03:00
# handle device manager
2020-01-30 16:53:17 +03:00
case "$devmgr" in
2020-02-02 16:10:43 +03:00
mdev) install_mdev ;;
mdevd) install_mdevd ;;
udev) install_udev ;;
2020-01-30 16:53:17 +03:00
*) panic "devmgr option broken" ;;
esac
2020-02-02 16:10:43 +03:00
[ "$use_luks" = 1 ] && install_luks
[ "$use_lvm" = 1 ] && install_lvm
2020-01-05 21:01:39 +03:00
2020-01-30 16:53:17 +03:00
create_initramfs
2020-01-05 21:01:39 +03:00
2020-01-30 16:53:17 +03:00
info "done! check out initramfs-${kernel}.img.gz"