|
|
|
#!/bin/sh
|
|
|
|
|
|
|
|
. /etc/initrd.defaults
|
|
|
|
. /etc/initrd.d/00-splash.sh
|
|
|
|
. /etc/initrd.d/00-common.sh
|
|
|
|
. /etc/initrd.d/00-fsdev.sh
|
|
|
|
. /etc/initrd.d/00-devmgr.sh
|
|
|
|
. /etc/initrd.d/00-zfs.sh
|
|
|
|
. /etc/initrd.d/00-modules.sh
|
|
|
|
. /etc/initrd.d/00-livecd.sh
|
|
|
|
. /etc/initrd.d/00-nfs.sh
|
|
|
|
. /etc/initrd.d/00-keymaps.sh
|
|
|
|
. /etc/initrd.d/00-crypt.sh
|
|
|
|
|
|
|
|
|
|
|
|
real_root_init() {
|
|
|
|
if [ -z "${REAL_ROOT}" ] && [ "${FAKE_ROOT}" != "/dev/ram0" ]; then
|
|
|
|
REAL_ROOT="${FAKE_ROOT}"
|
|
|
|
fi
|
|
|
|
if [ -z "${REAL_INIT}" ] && [ "${FAKE_INIT}" != "/linuxrc" ]; then
|
|
|
|
REAL_INIT="${FAKE_INIT}"
|
|
|
|
fi
|
|
|
|
if [ -z "${REAL_ROOTFLAGS}" ]; then
|
|
|
|
REAL_ROOTFLAGS="${FAKE_ROOTFLAGS}"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
start_iscsi() {
|
|
|
|
if [ ! -e /usr/sbin/iscsistart ]; then
|
|
|
|
return 0 # disabled
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ ! -n "${ISCSI_NOIBFT}" ]
|
|
|
|
then
|
|
|
|
good_msg "Activating iSCSI via iBFT"
|
|
|
|
iscsistart -b
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ -n "${ISCSI_INITIATORNAME}" ] && [ -n "${ISCSI_TARGET}" ] && [ -n "${ISCSI_ADDRESS}" ]; then
|
|
|
|
good_msg "Activating iSCSI via cmdline"
|
|
|
|
|
|
|
|
if [ "${ISCSI_TGPT}" ]
|
|
|
|
then
|
|
|
|
ADDITIONAL="${ADDITIONAL} -g ${ISCSI_TGPT}"
|
|
|
|
else
|
|
|
|
ADDITIONAL="${ADDITIONAL} -g 1"
|
|
|
|
fi
|
|
|
|
if [ "${ISCSI_PORT}" ]
|
|
|
|
then
|
|
|
|
ADDITIONAL="${ADDITIONAL} -p ${ISCSI_PORT}"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ "${ISCSI_USERNAME}" ]
|
|
|
|
then
|
|
|
|
ADDITIONAL="${ADDITIONAL} -u ${ISCSI_USERNAME}"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ "${ISCSI_PASSWORD}" ]
|
|
|
|
then
|
|
|
|
ADDITIONAL="${ADDITIONAL} -w ${ISCSI_PASSWORD}"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ "${ISCSI_USERNAME_IN}" ]
|
|
|
|
then
|
|
|
|
ADDITIONAL="${ADDITIONAL} -U ${ISCSI_USERNAME_IN}"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ "${ISCSI_PASSWORD_IN}" ]
|
|
|
|
then
|
|
|
|
ADDITIONAL="${ADDITIONAL} -W ${ISCSI_PASSWORD_IN}"
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ "${ISCSI_DEBUG}" ]
|
|
|
|
then
|
|
|
|
ADDITIONAL="${ADDITIONAL} -d ${ISCSI_DEBUG}"
|
|
|
|
fi
|
|
|
|
|
|
|
|
iscsistart -i "${ISCSI_INITIATORNAME}" -t "${ISCSI_TARGET}" \
|
|
|
|
-a "${ISCSI_ADDRESS}" ${ADDITIONAL}
|
|
|
|
|
|
|
|
# let iscsid settle - otherwise mounting the iSCSI-disk
|
|
|
|
# will fail (very rarely, though)
|
|
|
|
sleep 1
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
resume_init() {
|
|
|
|
if [ -z "${REAL_RESUME}" ]; then
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
if [ "${NORESUME}" = "1" ]; then
|
|
|
|
return 0
|
|
|
|
fi
|
|
|
|
|
|
|
|
local resume_dev=$(find_real_device "${REAL_RESUME}")
|
|
|
|
if [ -n "${resume_dev}" ]; then
|
|
|
|
REAL_RESUME="${resume_dev}"
|
|
|
|
good_msg "Detected real_resume=${resume_dev}"
|
|
|
|
else
|
|
|
|
bad_msg "Cannot resolve real_resume=${REAL_RESUME}"
|
|
|
|
bad_msg "Something bad may happen, crossing fingers"
|
|
|
|
fi
|
|
|
|
|
|
|
|
swsusp_resume
|
|
|
|
}
|
|
|
|
|
|
|
|
swsusp_resume() {
|
|
|
|
# determine swap resume partition
|
|
|
|
local device=$(ls -lL "${REAL_RESUME}" | sed 's/\ */ /g' | \
|
|
|
|
cut -d \ -f 5-6 | sed 's/,\ */:/')
|
|
|
|
[ -f /sys/power/resume -a -n "${device}" ] && \
|
|
|
|
echo "${device}" > /sys/power/resume
|
|
|
|
}
|
|
|
|
|
|
|
|
getdvhoff() {
|
|
|
|
echo $(( $(hexdump -n 4 -s $((316 + 12 * $2)) -e '"%i"' $1) * 512))
|
|
|
|
}
|
|
|
|
|
|
|
|
get_mounts_list() {
|
|
|
|
awk '
|
|
|
|
/^[[:blank:]]*#/ { next }
|
|
|
|
{ print $1 }
|
|
|
|
' ${NEW_ROOT}/etc/initramfs.mounts
|
|
|
|
}
|
|
|
|
|
|
|
|
get_mount_fstype() {
|
|
|
|
[ -e "${NEW_ROOT}"/etc/fstab ] || return 1
|
|
|
|
awk -v fs="$1" '
|
|
|
|
/^[[:blank:]]*#/ { next }
|
|
|
|
$2 == fs { print $3 }
|
|
|
|
' ${NEW_ROOT}/etc/fstab
|
|
|
|
}
|
|
|
|
|
|
|
|
get_mount_options() {
|
|
|
|
[ -e "${NEW_ROOT}"/etc/fstab ] || return 1
|
|
|
|
awk -v fs="$1" '
|
|
|
|
/^[[:blank:]]*#/ { next }
|
|
|
|
$2 == fs { print $4 }
|
|
|
|
' ${NEW_ROOT}/etc/fstab
|
|
|
|
}
|
|
|
|
|
|
|
|
get_mount_device() {
|
|
|
|
[ -e "${NEW_ROOT}"/etc/fstab ] || return 1
|
|
|
|
awk -v fs="$1" '
|
|
|
|
/^[[:blank:]]*#/ { next }
|
|
|
|
$2 == fs { print $1 }
|
|
|
|
' ${NEW_ROOT}/etc/fstab
|
|
|
|
}
|
|
|
|
|
|
|
|
# If the kernel is handed a mount option is does not recognize, it
|
|
|
|
# WILL fail to mount. util-linux handles auto/noauto, but busybox
|
|
|
|
# passes it straight to the kernel which then rejects the mount. To
|
|
|
|
# make like a little easier, busybox mount does not care about
|
|
|
|
# leading, trailing or duplicate commas.
|
|
|
|
strip_mount_options() {
|
|
|
|
sed -r 's/(,|^)(no)?auto(,|$)/,/g'
|
|
|
|
}
|
|
|
|
|
|
|
|
# Read /etc/initramfs.mounts from ${NEW_ROOT} and mount the
|
|
|
|
# listed filesystem mountpoints. For instance, /usr, which is
|
|
|
|
# required by udev & systemd.
|
|
|
|
ensure_initramfs_mounts() {
|
|
|
|
local fslist=
|
|
|
|
|
|
|
|
if [ -f "${NEW_ROOT}/etc/initramfs.mounts" ]; then
|
|
|
|
fslist="$(get_mounts_list)"
|
|
|
|
else
|
|
|
|
fslist="/usr"
|
|
|
|
fi
|
|
|
|
|
|
|
|
local dev= fstype= opts= mnt= cmd=
|
|
|
|
for fs in ${fslist}; do
|
|
|
|
|
|
|
|
mnt="${NEW_ROOT}${fs}"
|
|
|
|
if mountpoint -q "${mnt}"; then
|
|
|
|
good_msg "${fs} already mounted, skipping..."
|
|
|
|
continue
|
|
|
|
fi
|
|
|
|
|
|
|
|
dev=$(get_mount_device "${fs}")
|
|
|
|
[ -z "${dev}" ] && continue
|
|
|
|
# Resolve it like util-linux mount does
|
|
|
|
[ -L "${dev}" ] && dev=$(readlink "${dev}")
|
|
|
|
# In this case, it's probably part of the filesystem
|
|
|
|
# and not a mountpoint
|
|
|
|
[ -z "${dev}" ] && continue
|
|
|
|
|
|
|
|
fstype=$(get_mount_fstype "${fs}")
|
|
|
|
if get_mount_options "${fs}" | fgrep -q bind; then
|
|
|
|
opts="bind"
|
|
|
|
dev="${NEW_ROOT}${dev}"
|
|
|
|
else
|
|
|
|
# ro must be trailing, and the options will always
|
|
|
|
# contain at least 'defaults'
|
|
|
|
opts="$(get_mount_options ${fs} | strip_mount_options)"
|
|
|
|
opts="${opts},ro"
|
|
|
|
fi
|
|
|
|
|
|
|
|
cmd="mount -t ${fstype} -o ${opts} ${dev} ${mnt}"
|
|
|
|
good_msg "Mounting ${dev} as ${fs}: ${cmd}"
|
|
|
|
if ! ${cmd}; then
|
|
|
|
bad_msg "Unable to mount ${dev} for ${fs}"
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
fstype_init() {
|
|
|
|
local fstype="${1}"
|
|
|
|
if [ "${fstype}" = "btrfs" ]; then
|
|
|
|
# start BTRFS volume detection, if available
|
|
|
|
[ -x /sbin/btrfs ] && /sbin/btrfs device scan
|
|
|
|
elif [ -z "${fstype}" ]; then
|
|
|
|
warn_msg "Unable to detect the filesystem type (empty variable)"
|
|
|
|
fi
|
|
|
|
}
|
|
|
|
|
|
|
|
_rootdev_detect() {
|
|
|
|
local got_good_root=0
|
|
|
|
while [ "${got_good_root}" != "1" ]; do
|
|
|
|
|
|
|
|
case "${REAL_ROOT}" in
|
|
|
|
LABEL=*|UUID=*)
|
|
|
|
local root_dev=$(find_real_device "${REAL_ROOT}")
|
|
|
|
if [ -n "${root_dev}" ]; then
|
|
|
|
REAL_ROOT="${root_dev}"
|
|
|
|
good_msg "Detected root: ${REAL_ROOT}"
|
|
|
|
else
|
|
|
|
bad_msg "Unable to resolve root: ${REAL_ROOT}"
|
|
|
|
|
|
|
|
got_good_root=0
|
|
|
|
prompt_user "REAL_ROOT" "root block device"
|
|
|
|
continue
|
|
|
|
fi
|
|
|
|
;;
|
|
|
|
|
|
|
|
ZFS*)
|
|
|
|
# zfs_rootdev_init will tweak ${REAL_ROOT}
|
|
|
|
zfs_rootdev_init
|
|
|
|
if [ "${?}" = "0" ]; then
|
|
|
|
got_good_root=1
|
|
|
|
else
|
|
|
|
got_good_root=0
|
|
|
|
prompt_user "REAL_ROOT" "root block device"
|
|
|
|
continue
|
|
|
|
fi
|
|
|
|
;;
|
|
|
|
esac
|
|
|
|
|
|
|
|
if [ -z "${REAL_ROOT}" ]; then
|
|
|
|
# No REAL_ROOT determined/specified.
|
|
|
|
# Prompt user for root block device.
|
|
|
|
prompt_user "REAL_ROOT" "root block device"
|
|
|
|
got_good_root=0
|
|
|
|
|
|
|
|
# Check for a block device or NFS
|
|
|
|
elif [ -b "${REAL_ROOT}" ] || is_nfs; then
|
|
|
|
got_good_root=1
|
|
|
|
else
|
|
|
|
bad_msg "${REAL_ROOT} is an invalid root device..."
|
|
|
|
REAL_ROOT=""
|
|
|
|
got_good_root=0
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
_rootdev_mount() {
|
|
|
|
local mount_opts=ro
|
|
|
|
local mount_fstype="${ROOTFSTYPE}"
|
|
|
|
local fstype=$(get_device_fstype "${REAL_ROOT}")
|
|
|
|
if is_zfs_fstype "${fstype}"; then
|
|
|
|
mount_fstype=zfs
|
|
|
|
mount_opts=$(zfs_get_real_root_mount_flags)
|
|
|
|
fi
|
|
|
|
|
|
|
|
good_msg "Detected fstype: ${fstype}"
|
|
|
|
good_msg "Using mount fstype: ${mount_fstype}"
|
|
|
|
fstype_init "${fstype}"
|
|
|
|
|
|
|
|
local mopts="${mount_opts}"
|
|
|
|
[ -n "${REAL_ROOTFLAGS}" ] && \
|
|
|
|
mopts="${mopts},${REAL_ROOTFLAGS}"
|
|
|
|
good_msg "Using mount opts: -o ${mopts}"
|
|
|
|
|
|
|
|
mount -t "${mount_fstype}" -o "${mopts}" \
|
|
|
|
"${REAL_ROOT}" "${NEW_ROOT}" || return 1
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
|
|
|
rootdev_init() {
|
|
|
|
good_msg "Initializing root device..."
|
|
|
|
|
|
|
|
while true; do
|
|
|
|
|
|
|
|
if ! _rootdev_detect; then
|
|
|
|
bad_msg "Could not mount specified ROOT, try again"
|
|
|
|
prompt_user "REAL_ROOT" "root block device"
|
|
|
|
continue
|
|
|
|
fi
|
|
|
|
|
|
|
|
if is_livecd && ! is_nfs; then
|
|
|
|
# CD already mounted; no further checks necessary
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
if [ "${LOOPTYPE}" = "sgimips" ]; then
|
|
|
|
# sgimips mounts the livecd root partition directly
|
|
|
|
# there is no isofs filesystem to worry about
|
|
|
|
break
|
|
|
|
fi
|
|
|
|
|
|
|
|
good_msg "Mounting ${REAL_ROOT} as root..."
|
|
|
|
|
|
|
|
# Try to mount the device as ${NEW_ROOT}
|
|
|
|
local out=1
|
|
|
|
if is_nfs; then
|
|
|
|
find_nfs && out=0
|
|
|
|
else
|
|
|
|
_rootdev_mount && out=0
|
|
|
|
fi
|
|
|
|
|
|
|
|
if [ "${out}" != "0" ]; then
|
|
|
|
bad_msg "Could not mount specified ROOT, try again"
|
|
|
|
prompt_user "REAL_ROOT" "root block device"
|
|
|
|
continue
|
|
|
|
fi
|
|
|
|
|
|
|
|
# now that the root filesystem is mounted, before
|
|
|
|
# checking the validity of ${NEW_ROOT} and ${REAL_INIT},
|
|
|
|
# ensure that ${NEW_ROOT}/etc/initramfs.mounts entries
|
|
|
|
# are mounted.
|
|
|
|
ensure_initramfs_mounts
|
|
|
|
|
|
|
|
# NFS does not need further checks here.
|
|
|
|
is_nfs && break
|
|
|
|
|
|
|
|
if [ ! -d "${NEW_ROOT}/dev" ]; then
|
|
|
|
_msg="The filesystem ${REAL_ROOT},"
|
|
|
|
_msg="${_msg} mounted at ${NEW_ROOT}"
|
|
|
|
_msg="${_msg} does not contain /dev"
|
|
|
|
_msg="${_msg}, init will likely fail..."
|
|
|
|
bad_msg "${_msg}"
|
|
|
|
prompt_user "REAL_ROOT" "root block device"
|
|
|
|
continue
|
|
|
|
fi
|
|
|
|
if [ ! -x "${NEW_ROOT}${REAL_INIT}" ]; then
|
|
|
|
_msg="The filesystem ${REAL_ROOT},"
|
|
|
|
_msg="${_msg} mounted at ${NEW_ROOT}"
|
|
|
|
_msg="${_msg} does not contain a valid"
|
|
|
|
_msg="${_msg} init=${REAL_INIT}"
|
|
|
|
bad_msg "${_msg}"
|
|
|
|
prompt_user "REAL_ROOT" "root block device"
|
|
|
|
continue
|
|
|
|
fi
|
|
|
|
break
|
|
|
|
done
|
|
|
|
}
|
|
|
|
|
|
|
|
# If devtmpfs is mounted, try move it to the new root
|
|
|
|
# If that fails, try to unmount all possible mounts of devtmpfs as
|
|
|
|
# stuff breaks otherwise
|
|
|
|
move_mounts_to_chroot() {
|
|
|
|
for fs in /run /dev /sys /proc; do
|
|
|
|
if grep -qs "$fs" /proc/mounts; then
|
|
|
|
local chroot_dir="${CHROOT}${fs}"
|
|
|
|
mkdir -p "${chroot_dir}"
|
|
|
|
if ! mount --move $fs "${chroot_dir}"
|
|
|
|
then
|
|
|
|
umount $fs || \
|
|
|
|
bad_msg "Failed to move and umount $fs!"
|
|
|
|
fi
|
|
|
|
fi
|
|
|
|
done
|
|
|
|
}
|