343 lines
9.9 KiB
Bash
343 lines
9.9 KiB
Bash
|
#!/bin/sh -e
|
||
|
#
|
||
|
# Copyright (c) 2016, Intel Corporation.
|
||
|
# All rights reserved.
|
||
|
#
|
||
|
# install.sh [device_name] [rootfs_name]
|
||
|
#
|
||
|
# This file is a copy of file with same name in OE:
|
||
|
# meta/recipes-core/initrdscripts/files/. We modify
|
||
|
# it for RMC feature to deploy file blobs from RMC
|
||
|
# database file to target.
|
||
|
|
||
|
PATH=/sbin:/bin:/usr/sbin:/usr/bin
|
||
|
|
||
|
# We need 20 Mb for the boot partition
|
||
|
boot_size=20
|
||
|
|
||
|
# 5% for swap
|
||
|
swap_ratio=5
|
||
|
|
||
|
# Get a list of hard drives
|
||
|
hdnamelist=""
|
||
|
live_dev_name=`cat /proc/mounts | grep ${1%/} | awk '{print $1}'`
|
||
|
live_dev_name=${live_dev_name#\/dev/}
|
||
|
# Only strip the digit identifier if the device is not an mmc
|
||
|
case $live_dev_name in
|
||
|
mmcblk*)
|
||
|
;;
|
||
|
nvme*)
|
||
|
;;
|
||
|
*)
|
||
|
live_dev_name=${live_dev_name%%[0-9]*}
|
||
|
;;
|
||
|
esac
|
||
|
|
||
|
echo "Searching for hard drives ..."
|
||
|
|
||
|
for device in `ls /sys/block/`; do
|
||
|
case $device in
|
||
|
loop*)
|
||
|
# skip loop device
|
||
|
;;
|
||
|
sr*)
|
||
|
# skip CDROM device
|
||
|
;;
|
||
|
ram*)
|
||
|
# skip ram device
|
||
|
;;
|
||
|
*)
|
||
|
# skip the device LiveOS is on
|
||
|
# Add valid hard drive name to the list
|
||
|
case $device in
|
||
|
$live_dev_name*)
|
||
|
# skip the device we are running from
|
||
|
;;
|
||
|
*)
|
||
|
hdnamelist="$hdnamelist $device"
|
||
|
;;
|
||
|
esac
|
||
|
;;
|
||
|
esac
|
||
|
done
|
||
|
|
||
|
if [ -z "${hdnamelist}" ]; then
|
||
|
echo "You need another device (besides the live device /dev/${live_dev_name}) to install the image. Installation aborted."
|
||
|
exit 1
|
||
|
fi
|
||
|
|
||
|
TARGET_DEVICE_NAME=""
|
||
|
for hdname in $hdnamelist; do
|
||
|
# Display found hard drives and their basic info
|
||
|
echo "-------------------------------"
|
||
|
echo /dev/$hdname
|
||
|
if [ -r /sys/block/$hdname/device/vendor ]; then
|
||
|
echo -n "VENDOR="
|
||
|
cat /sys/block/$hdname/device/vendor
|
||
|
fi
|
||
|
if [ -r /sys/block/$hdname/device/model ]; then
|
||
|
echo -n "MODEL="
|
||
|
cat /sys/block/$hdname/device/model
|
||
|
fi
|
||
|
if [ -r /sys/block/$hdname/device/uevent ]; then
|
||
|
echo -n "UEVENT="
|
||
|
cat /sys/block/$hdname/device/uevent
|
||
|
fi
|
||
|
echo
|
||
|
done
|
||
|
|
||
|
# Get user choice
|
||
|
while true; do
|
||
|
echo "Please select an install target or press n to exit ($hdnamelist ): "
|
||
|
read answer
|
||
|
if [ "$answer" = "n" ]; then
|
||
|
echo "Installation manually aborted."
|
||
|
exit 1
|
||
|
fi
|
||
|
for hdname in $hdnamelist; do
|
||
|
if [ "$answer" = "$hdname" ]; then
|
||
|
TARGET_DEVICE_NAME=$answer
|
||
|
break
|
||
|
fi
|
||
|
done
|
||
|
if [ -n "$TARGET_DEVICE_NAME" ]; then
|
||
|
break
|
||
|
fi
|
||
|
done
|
||
|
|
||
|
if [ -n "$TARGET_DEVICE_NAME" ]; then
|
||
|
echo "Installing image on /dev/$TARGET_DEVICE_NAME ..."
|
||
|
else
|
||
|
echo "No hard drive selected. Installation aborted."
|
||
|
exit 1
|
||
|
fi
|
||
|
|
||
|
device=/dev/$TARGET_DEVICE_NAME
|
||
|
|
||
|
#
|
||
|
# The udev automounter can cause pain here, kill it
|
||
|
#
|
||
|
rm -f /etc/udev/rules.d/automount.rules
|
||
|
rm -f /etc/udev/scripts/mount*
|
||
|
|
||
|
#
|
||
|
# Unmount anything the automounter had mounted
|
||
|
#
|
||
|
umount ${device}* 2> /dev/null || /bin/true
|
||
|
|
||
|
mkdir -p /tmp
|
||
|
|
||
|
# Create /etc/mtab if not present
|
||
|
if [ ! -e /etc/mtab ]; then
|
||
|
cat /proc/mounts > /etc/mtab
|
||
|
fi
|
||
|
|
||
|
disk_size=$(parted ${device} unit mb print | grep '^Disk .*: .*MB' | cut -d" " -f 3 | sed -e "s/MB//")
|
||
|
|
||
|
swap_size=$((disk_size*swap_ratio/100))
|
||
|
rootfs_size=$((disk_size-boot_size-swap_size))
|
||
|
|
||
|
rootfs_start=$((boot_size))
|
||
|
rootfs_end=$((rootfs_start+rootfs_size))
|
||
|
swap_start=$((rootfs_end))
|
||
|
|
||
|
# MMC devices are special in a couple of ways
|
||
|
# 1) they use a partition prefix character 'p'
|
||
|
# 2) they are detected asynchronously (need rootwait)
|
||
|
rootwait=""
|
||
|
part_prefix=""
|
||
|
if [ ! "${device#/dev/mmcblk}" = "${device}" ] || \
|
||
|
[ ! "${device#/dev/nvme}" = "${device}" ]; then
|
||
|
part_prefix="p"
|
||
|
rootwait="rootwait"
|
||
|
fi
|
||
|
bootfs=${device}${part_prefix}1
|
||
|
rootfs=${device}${part_prefix}2
|
||
|
swap=${device}${part_prefix}3
|
||
|
|
||
|
echo "*****************"
|
||
|
echo "Boot partition size: $boot_size MB ($bootfs)"
|
||
|
echo "Rootfs partition size: $rootfs_size MB ($rootfs)"
|
||
|
echo "Swap partition size: $swap_size MB ($swap)"
|
||
|
echo "*****************"
|
||
|
echo "Deleting partition table on ${device} ..."
|
||
|
dd if=/dev/zero of=${device} bs=512 count=35
|
||
|
|
||
|
echo "Creating new partition table on ${device} ..."
|
||
|
parted ${device} mklabel gpt
|
||
|
|
||
|
echo "Creating boot partition on $bootfs"
|
||
|
parted ${device} mkpart boot fat32 0% $boot_size
|
||
|
parted ${device} set 1 boot on
|
||
|
|
||
|
echo "Creating rootfs partition on $rootfs"
|
||
|
parted ${device} mkpart root ext3 $rootfs_start $rootfs_end
|
||
|
|
||
|
echo "Creating swap partition on $swap"
|
||
|
parted ${device} mkpart swap linux-swap $swap_start 100%
|
||
|
|
||
|
parted ${device} print
|
||
|
|
||
|
echo "Formatting $bootfs to vfat..."
|
||
|
mkfs.vfat $bootfs
|
||
|
|
||
|
echo "Formatting $rootfs to ext3..."
|
||
|
mkfs.ext3 $rootfs
|
||
|
|
||
|
echo "Formatting swap partition...($swap)"
|
||
|
mkswap $swap
|
||
|
|
||
|
mkdir /tgt_root
|
||
|
mkdir /src_root
|
||
|
mkdir -p /boot
|
||
|
|
||
|
# Handling of the target root partition
|
||
|
mount $rootfs /tgt_root
|
||
|
mount -o rw,loop,noatime,nodiratime /run/media/$1/$2 /src_root
|
||
|
echo "Copying rootfs files..."
|
||
|
cp -a /src_root/* /tgt_root
|
||
|
if [ -d /tgt_root/etc/ ] ; then
|
||
|
boot_uuid=$(blkid -o value -s UUID ${bootfs})
|
||
|
swap_part_uuid=$(blkid -o value -s PARTUUID ${swap})
|
||
|
echo "/dev/disk/by-partuuid/$swap_part_uuid swap swap defaults 0 0" >> /tgt_root/etc/fstab
|
||
|
echo "UUID=$boot_uuid /boot vfat defaults 1 2" >> /tgt_root/etc/fstab
|
||
|
# We dont want udev to mount our root device while we're booting...
|
||
|
if [ -d /tgt_root/etc/udev/ ] ; then
|
||
|
echo "${device}" >> /tgt_root/etc/udev/mount.blacklist
|
||
|
fi
|
||
|
fi
|
||
|
|
||
|
# Handling of the target boot partition
|
||
|
mount $bootfs /boot
|
||
|
echo "Preparing boot partition..."
|
||
|
|
||
|
EFIDIR="/boot/EFI/BOOT"
|
||
|
mkdir -p $EFIDIR
|
||
|
# Copy the efi loader
|
||
|
cp /run/media/$1/EFI/BOOT/*.efi $EFIDIR
|
||
|
|
||
|
# RMC deployment
|
||
|
RMC_CMD=/src_root/usr/bin/rmc
|
||
|
RMC_DB=/run/media/$1/rmc.db
|
||
|
|
||
|
# We don't want to quit when a step failed. For example,
|
||
|
# a file system could not support some operations.
|
||
|
set +e
|
||
|
|
||
|
if [ -f "${RMC_DB}" ] && [ -f "${RMC_CMD}" ]; then
|
||
|
echo "Found RMC database and tool, start RMC deployment"
|
||
|
# query INSTALLER.CONFIG from RMC DB
|
||
|
if ${RMC_CMD} -B INSTALLER.CONFIG -d "${RMC_DB}" -o /tmp/installer.config; then
|
||
|
while IFS=':' read -r NAME TGT_UID TGT_GID TGT_MODE TGT_PATH; do
|
||
|
# skip comment
|
||
|
# The regexp in grep works with busybox grep which doesn't
|
||
|
# seem to have a -P to recognize '\t'. But this expression could not
|
||
|
# work with gnu grep...
|
||
|
if echo "$NAME"|grep -q $'^[ \t]*#'; then
|
||
|
continue
|
||
|
fi
|
||
|
# check if we should create a directory (last char in target path is '/')
|
||
|
# or deploy a file
|
||
|
LAST_CHAR=$(echo "${TGT_PATH:$((${#TGT_PATH}-1)):1}")
|
||
|
|
||
|
# Do not bail out for failures but user should get stderr message
|
||
|
if [ ${LAST_CHAR} = "/" ]; then
|
||
|
# name field is skipped for directory
|
||
|
echo "DIR: ${TGT_UID}:${TGT_GID}:${TGT_MODE} => ${TGT_PATH}"
|
||
|
mkdir -p "$TGT_PATH"
|
||
|
chown "${TGT_UID}:${TGT_GID}" "$TGT_PATH"
|
||
|
chmod "${TGT_MODE}" "$TGT_PATH"
|
||
|
else
|
||
|
${RMC_CMD} -B "${NAME}" -d "${RMC_DB}" -o "${TGT_PATH}"
|
||
|
echo "FILE: ${NAME}:${TGT_UID}:${TGT_GID}:${TGT_MODE} => ${TGT_PATH}"
|
||
|
chown "${TGT_UID}:${TGT_GID}" "$TGT_PATH"
|
||
|
chmod "${TGT_MODE}" "$TGT_PATH"
|
||
|
fi
|
||
|
done < /tmp/installer.config
|
||
|
rm -rf /tmp/installer.config
|
||
|
|
||
|
# remove rmc from target since we don't think it is a valid
|
||
|
# case to run rmc after installation.
|
||
|
rm -rf /tgt_root/usr/bin/rmc
|
||
|
echo "RMC deployment finished"
|
||
|
else
|
||
|
echo "INSTALLER.CONFIG is not found, skip RMC deployment"
|
||
|
fi
|
||
|
|
||
|
# Final retouching by calling post-install hook
|
||
|
if ${RMC_CMD} -B POSTINSTALL.sh -d "${RMC_DB}" -o /tmp/POSTINSTALL.sh; then
|
||
|
echo "Found POSTINSTALL.sh execute it..."
|
||
|
chmod 500 /tmp/POSTINSTALL.sh
|
||
|
/tmp/POSTINSTALL.sh
|
||
|
rm -rf /tmp/POSTINSTALL.sh
|
||
|
fi
|
||
|
fi
|
||
|
set -e
|
||
|
|
||
|
if [ -f /run/media/$1/EFI/BOOT/grub.cfg ]; then
|
||
|
root_part_uuid=$(blkid -o value -s PARTUUID ${rootfs})
|
||
|
GRUBCFG="$EFIDIR/grub.cfg"
|
||
|
cp /run/media/$1/EFI/BOOT/grub.cfg $GRUBCFG
|
||
|
# Update grub config for the installed image
|
||
|
# Delete the install entry
|
||
|
sed -i "/menuentry 'install'/,/^}/d" $GRUBCFG
|
||
|
# Delete the initrd lines
|
||
|
sed -i "/initrd /d" $GRUBCFG
|
||
|
# Delete any LABEL= strings
|
||
|
sed -i "s/ LABEL=[^ ]*/ /" $GRUBCFG
|
||
|
# Delete any root= strings
|
||
|
sed -i "s/ root=[^ ]*/ /g" $GRUBCFG
|
||
|
# Add the root= and other standard boot options
|
||
|
sed -i "s@linux /vmlinuz *@linux /vmlinuz root=PARTUUID=$root_part_uuid rw $rootwait quiet @" $GRUBCFG
|
||
|
fi
|
||
|
|
||
|
if [ -d /run/media/$1/loader ]; then
|
||
|
rootuuid=$(blkid -o value -s PARTUUID ${rootfs})
|
||
|
GUMMIBOOT_CFGS="/boot/loader/entries/*.conf"
|
||
|
if [ -d /boot/loader ]; then
|
||
|
# Don't override loader.conf RMC already deployed
|
||
|
if [ ! -f /boot/loader/loader.conf ]; then
|
||
|
cp /run/media/$1/loader/loader.conf /boot/loader/
|
||
|
fi
|
||
|
# only copy built OE entries when RMC entries don't exist.
|
||
|
if [ ! -d /boot/loader/entries ] || [ ! ls /boot/loader/entries/*.conf &>/dev/null ]; then
|
||
|
cp -dr /run/media/$1/loader/entries /boot/loader
|
||
|
fi
|
||
|
else
|
||
|
# copy config files for gummiboot
|
||
|
cp -dr /run/media/$1/loader /boot
|
||
|
# delete the install entry
|
||
|
rm -f /boot/loader/entries/install.conf
|
||
|
fi
|
||
|
# delete the initrd lines
|
||
|
sed -i "/initrd /d" $GUMMIBOOT_CFGS
|
||
|
# delete any LABEL= strings
|
||
|
sed -i "s/ LABEL=[^ ]*/ /" $GUMMIBOOT_CFGS
|
||
|
# delete any root= strings
|
||
|
sed -i "s/ root=[^ ]*/ /" $GUMMIBOOT_CFGS
|
||
|
# add the root= and other standard boot options
|
||
|
sed -i "s@options *@options root=PARTUUID=$rootuuid rw $rootwait quiet @" $GUMMIBOOT_CFGS
|
||
|
# if RMC feature presents, append global kernel command line fragment when it exists.
|
||
|
if [ -f "${RMC_DB}" ] && [ -f "${RMC_CMD}" ]; then
|
||
|
if ${RMC_CMD} -B KBOOTPARAM -d "${RMC_DB}" -o /tmp/kbootparam; then
|
||
|
sed -i "/^[ \t]*options/ s/$/ $(cat /tmp/kbootparam)/" $GUMMIBOOT_CFGS
|
||
|
rm /tmp/kbootparam
|
||
|
fi
|
||
|
fi
|
||
|
fi
|
||
|
|
||
|
cp /run/media/$1/vmlinuz /boot
|
||
|
|
||
|
umount /src_root
|
||
|
umount /tgt_root
|
||
|
umount /boot
|
||
|
|
||
|
sync
|
||
|
|
||
|
echo "Remove your installation media, and press ENTER"
|
||
|
|
||
|
read enter
|
||
|
|
||
|
echo "Rebooting..."
|
||
|
reboot -f
|