mirror of
https://github.com/flatcar/scripts.git
synced 2025-08-07 04:56:58 +02:00
Specifically, detect if the umount failed due to current access, if so, give it up to 9 more runs (w/ 1s pauses) continuing only if it's still failing due to currently open files. Via this, it should suppress the race of gvfs/trashd looking at quick mounted/umounted pathways. This CL is a two parter; this adds the script, and converts common.sh consumers over to using the override. The next CL will modify the chroot itself to ensure our script gets picked up/used. BUG=chromium-os:23443 TEST=trybot, manul validation. Change-Id: I92dedd91d6133c2063b1e5dbbc1a68366844801d Reviewed-on: https://gerrit.chromium.org/gerrit/32087 Commit-Ready: Brian Harring <ferringb@chromium.org> Reviewed-by: Brian Harring <ferringb@chromium.org> Tested-by: Brian Harring <ferringb@chromium.org>
231 lines
8.5 KiB
Bash
Executable File
231 lines
8.5 KiB
Bash
Executable File
#!/bin/bash
|
|
|
|
# Copyright (c) 2012 The Chromium OS Authors. All rights reserved.
|
|
# Use of this source code is governed by a BSD-style license that can be
|
|
# found in the LICENSE file.
|
|
|
|
# Helper script that generates the legacy/efi bootloader partitions.
|
|
# It does not populate the templates, but can update a loop device.
|
|
|
|
SCRIPT_ROOT=$(dirname $(readlink -f "$0"))
|
|
. "${SCRIPT_ROOT}/common.sh" || exit 1
|
|
|
|
# Need to be inside the chroot to load chromeos-common.sh
|
|
assert_inside_chroot
|
|
|
|
# Load functions and constants for chromeos-install
|
|
. /usr/lib/installer/chromeos-common.sh || exit 1
|
|
|
|
# Flags.
|
|
DEFINE_string arch "x86" \
|
|
"The boot architecture: arm or x86. (Default: x86)"
|
|
# TODO(wad) once extlinux is dead, we can remove this.
|
|
DEFINE_boolean install_syslinux ${FLAGS_FALSE} \
|
|
"Controls whether syslinux is run on 'to'. (Default: false)"
|
|
DEFINE_string from "/tmp/boot" \
|
|
"Path the legacy bootloader templates are copied from. (Default /tmp/boot)"
|
|
DEFINE_string to "/tmp/esp.img" \
|
|
"Path to esp image (Default: /tmp/esp.img)"
|
|
DEFINE_integer to_offset 0 \
|
|
"Offset in bytes into 'to' if it is a file (Default: 0)"
|
|
DEFINE_integer to_size -1 \
|
|
"Size in bytes of 'to' to use if it is a file. -1 is ignored. (Default: -1)"
|
|
DEFINE_string vmlinuz "/tmp/vmlinuz" \
|
|
"Path to the vmlinuz file to use (Default: /tmp/vmlinuz)"
|
|
# The kernel_partition and the kernel_cmdline each are used to supply
|
|
# verified boot configuration: dm="".
|
|
DEFINE_string kernel_partition "/tmp/vmlinuz.image" \
|
|
"Path to the signed kernel image. (Default: /tmp/vmlinuz.image)"
|
|
DEFINE_string kernel_cmdline "" \
|
|
"Kernel commandline if no kernel_partition given. (Default: '')"
|
|
DEFINE_string kernel_partition_offset "0" \
|
|
"Offset to the kernel partition [KERN-A] (Default: 0)"
|
|
DEFINE_string kernel_partition_sectors "0" \
|
|
"Kernel partition sectors (Default: 0)"
|
|
DEFINE_string usb_disk /dev/sdb3 \
|
|
"Path syslinux should use to do a usb boot. Default: /dev/sdb3"
|
|
|
|
# Parse flags
|
|
FLAGS "$@" || exit 1
|
|
eval set -- "${FLAGS_ARGV}"
|
|
switch_to_strict_mode
|
|
|
|
part_index_to_uuid() {
|
|
local image="$1"
|
|
local index="$2"
|
|
cgpt show -i "$index" -u "$image"
|
|
}
|
|
|
|
# If not provided by chromeos-common.sh, this will update all of the
|
|
# boot loader files (both A and B) with the data pulled
|
|
# from the kernel_partition. The default boot target should
|
|
# be set when the rootfs is stuffed.
|
|
if ! type -p update_x86_bootloaders; then
|
|
update_x86_bootloaders() {
|
|
local old_root="$1" # e.g., /dev/sd%D%P or %U+1
|
|
local kernel_cmdline="$2"
|
|
local esp_fs_dir="$3"
|
|
local template_dir="$4"
|
|
local to="$5"
|
|
|
|
# Pull out the dm="" values
|
|
dm_table=
|
|
if echo "$kernel_cmdline" | grep -q 'dm="'; then
|
|
dm_table=$(echo "$kernel_cmdline" | sed -s 's/.*dm="\([^"]*\)".*/\1/')
|
|
fi
|
|
|
|
# Maintain cros_debug flag in developer and test images.
|
|
cros_flags="cros_legacy"
|
|
if echo "$kernel_cmdline" | grep -q 'cros_debug'; then
|
|
cros_flags="cros_legacy cros_debug"
|
|
fi
|
|
|
|
root_a_uuid="PARTUUID=$(part_index_to_uuid "$to" 3)"
|
|
root_b_uuid="PARTUUID=$(part_index_to_uuid "$to" 5)"
|
|
|
|
# Rewrite grub table
|
|
grub_dm_table_a=${dm_table//${old_root}/${root_a_uuid}}
|
|
grub_dm_table_b=${dm_table//${old_root}/${root_b_uuid}}
|
|
sed -e "s|DMTABLEA|${grub_dm_table_a}|g" \
|
|
-e "s|DMTABLEB|${grub_dm_table_b}|g" \
|
|
-e "s|cros_legacy|${cros_flags}|g" \
|
|
"${template_dir}"/efi/boot/grub.cfg |
|
|
sudo dd of="${esp_fs_dir}"/efi/boot/grub.cfg
|
|
sed -e "s|/dev/\\\$linuxpartA|${root_a_uuid}|g" \
|
|
-e "s|/dev/\\\$linuxpartB|${root_b_uuid}|g" \
|
|
"${template_dir}"/efi/boot/grub.cfg |
|
|
sudo dd of="${esp_fs_dir}"/efi/boot/grub.cfg
|
|
|
|
# Rewrite syslinux DM_TABLE
|
|
syslinux_dm_table_usb=${dm_table//${old_root}/${root_a_uuid}}
|
|
sed -e "s|DMTABLEA|${syslinux_dm_table_usb}|g" \
|
|
-e "s|cros_legacy|${cros_flags}|g" \
|
|
"${template_dir}"/syslinux/usb.A.cfg |
|
|
sudo dd of="${esp_fs_dir}"/syslinux/usb.A.cfg
|
|
|
|
syslinux_dm_table_a=${dm_table//${old_root}/HDROOTA}
|
|
sed -e "s|DMTABLEA|${syslinux_dm_table_a}|g" \
|
|
-e "s|cros_legacy|${cros_flags}|g" \
|
|
"${template_dir}"/syslinux/root.A.cfg |
|
|
sudo dd of="${esp_fs_dir}"/syslinux/root.A.cfg
|
|
|
|
syslinux_dm_table_b=${dm_table//${old_root}/HDROOTB}
|
|
sed -e "s|DMTABLEB|${syslinux_dm_table_b}|g" \
|
|
-e "s|cros_legacy|${cros_flags}|g" \
|
|
"${template_dir}"/syslinux/root.B.cfg |
|
|
sudo dd of="${esp_fs_dir}"/syslinux/root.B.cfg
|
|
|
|
# Copy the vmlinuz's into place for syslinux
|
|
sudo cp -f "${template_dir}"/vmlinuz "${esp_fs_dir}"/syslinux/vmlinuz.A
|
|
sudo cp -f "${template_dir}"/vmlinuz "${esp_fs_dir}"/syslinux/vmlinuz.B
|
|
|
|
# The only work left for the installer is to pick the correct defaults
|
|
# and replace HDROOTA and HDROOTB with the correct /dev/sd%D%P/%U+1
|
|
}
|
|
fi
|
|
|
|
ESP_DEV=
|
|
if [[ ! -e "${FLAGS_to}" ]]; then
|
|
error "The ESP doesn't exist"
|
|
# This shouldn't happen.
|
|
info "Creating a new esp image at ${FLAGS_to}" anyway.
|
|
# Create EFI System Partition to boot stock EFI BIOS (but not ChromeOS EFI
|
|
# BIOS). ARM uses this space to determine which partition is bootable.
|
|
# NOTE: The size argument for mkfs.vfat is in 1024-byte blocks.
|
|
# We'll hard-code it to 16M for now.
|
|
ESP_BLOCKS=16384
|
|
/usr/sbin/mkfs.vfat -C "${FLAGS_to}" ${ESP_BLOCKS}
|
|
ESP_DEV=$(sudo losetup --show -f "${FLAGS_to}")
|
|
if [ -z "${ESP_DEV}" ]; then
|
|
die "No free loop devices."
|
|
fi
|
|
else
|
|
if [[ -f "${FLAGS_to}" ]]; then
|
|
esp_offset="--offset ${FLAGS_to_offset}"
|
|
esp_size="--sizelimit ${FLAGS_to_size}"
|
|
if [ ${FLAGS_to_size} -lt 0 ]; then
|
|
esp_size=
|
|
fi
|
|
ESP_DEV=$(sudo losetup --show -f ${esp_offset} ${esp_size} "${FLAGS_to}")
|
|
if [ -z "${ESP_DEV}" ]; then
|
|
die "No free loop devices."
|
|
fi
|
|
else
|
|
# If it is a block device or something else, try to mount it anyway.
|
|
ESP_DEV="${FLAGS_to}"
|
|
fi
|
|
fi
|
|
|
|
ESP_FS_DIR=$(mktemp -d /tmp/esp.XXXXXX)
|
|
cleanup() {
|
|
set +e
|
|
if ! safe_umount "${ESP_FS_DIR}"; then
|
|
# There is a race condition possible on some ubuntu setups
|
|
# with mounting and unmounting a device very quickly
|
|
# Doing a quick sleep/retry as a temporary workaround
|
|
warn "Initial unmount failed. Possibly crosbug.com/23443. Retrying"
|
|
sleep 5
|
|
safe_umount "${ESP_FS_DIR}"
|
|
fi
|
|
if [[ -n "${ESP_DEV}" && -z "${ESP_DEV//\/dev\/loop*}" ]]; then
|
|
sudo losetup -d "${ESP_DEV}"
|
|
fi
|
|
rm -rf "${ESP_FS_DIR}"
|
|
}
|
|
trap cleanup EXIT
|
|
sudo mount "${ESP_DEV}" "${ESP_FS_DIR}"
|
|
|
|
if [[ "${FLAGS_arch}" = "x86" || "${FLAGS_arch}" = "amd64" ]]; then
|
|
# Populate the EFI bootloader configuration
|
|
sudo mkdir -p "${ESP_FS_DIR}/efi/boot"
|
|
sudo cp "${FLAGS_from}"/efi/boot/bootx64.efi \
|
|
"${ESP_FS_DIR}/efi/boot/bootx64.efi"
|
|
sudo cp "${FLAGS_from}/efi/boot/grub.cfg" \
|
|
"${ESP_FS_DIR}/efi/boot/grub.cfg"
|
|
|
|
# Prepopulate the syslinux directories too and update for verified boot values
|
|
# after the rootfs work is done.
|
|
sudo mkdir -p "${ESP_FS_DIR}"/syslinux
|
|
sudo cp -r "${FLAGS_from}"/syslinux/. "${ESP_FS_DIR}"/syslinux
|
|
|
|
# Stage both kernels with the only one we built.
|
|
sudo cp -f "${FLAGS_vmlinuz}" "${ESP_FS_DIR}"/syslinux/vmlinuz.A
|
|
sudo cp -f "${FLAGS_vmlinuz}" "${ESP_FS_DIR}"/syslinux/vmlinuz.B
|
|
|
|
# Extract kernel flags
|
|
kernel_cfg=
|
|
old_root="%U+1"
|
|
if [[ -n "${FLAGS_kernel_cmdline}" ]]; then
|
|
info "Using supplied kernel_cmdline to update templates."
|
|
kernel_cfg="${FLAGS_kernel_cmdline}"
|
|
elif [[ -n "${FLAGS_kernel_partition}" ]]; then
|
|
info "Extracting the kernel command line from ${FLAGS_kernel_partition}"
|
|
kernel_cfg=$(dump_kernel_config "${FLAGS_kernel_partition}")
|
|
fi
|
|
update_x86_bootloaders "${old_root}" \
|
|
"${kernel_cfg}" \
|
|
"${ESP_FS_DIR}" \
|
|
"${FLAGS_from}" \
|
|
"${FLAGS_to}"
|
|
|
|
# Install the syslinux loader on the ESP image (part 12) so it is ready when
|
|
# we cut over from rootfs booting (extlinux).
|
|
if [[ ${FLAGS_install_syslinux} -eq ${FLAGS_TRUE} ]]; then
|
|
safe_umount "${ESP_FS_DIR}"
|
|
sudo syslinux -d /syslinux "${ESP_DEV}"
|
|
# mount again for cleanup to free resource gracefully
|
|
sudo mount -o ro "${ESP_DEV}" "${ESP_FS_DIR}"
|
|
fi
|
|
elif [[ "${FLAGS_arch}" = "arm" ]]; then
|
|
# Copy u-boot script to ESP partition
|
|
if [ -r "${FLAGS_from}/boot-A.scr.uimg" ]; then
|
|
sudo mkdir -p "${ESP_FS_DIR}/u-boot"
|
|
sudo cp "${FLAGS_from}/boot-A.scr.uimg" \
|
|
"${ESP_FS_DIR}/u-boot/boot.scr.uimg"
|
|
sudo cp -f "${FLAGS_from}"/vmlinuz "${ESP_FS_DIR}"/vmlinuz.uimg.A
|
|
sudo cp -f "${FLAGS_from}"/zImage "${ESP_FS_DIR}"/vmlinuz.A
|
|
fi
|
|
fi
|
|
|
|
set +e
|