#!/bin/bash show_help() { cat << EOF Tool adding GRUB specific files to Mender compliant image file. Usage: $0 [options] Options: [-m|--mender-disk-image | -b|--bootloader-toolchain | -k|--keep | -d|--device-type] --mender-disk-image - Mender raw disk image --bootloader-toolchain - GNU Arm Embedded Toolchain --device-type - target device type identification --keep - prevent deleting GRUB workspace Note: supported device types are: beaglebone, raspberrypi3 Examples: ./mender-convert install-bootloader-to-mender-disk-image --mender-disk-image --device-type --bootloader-toolchain arm-linux-gnueabihf Note: toolchain naming convention is arch-vendor-(os-)abi arch is for architecture: arm, mips, x86, i686... vendor is tool chain supplier: apple, Codesourcery, Linux, os is for operating system: linux, none (bare metal) abi is for application binary interface convention: eabi, gnueabi, gnueabihf EOF } tool_dir="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )" output_dir=${tool_dir}/output integration_dir=${tool_dir}/integration grub_dir=$output_dir/grub grubenv_dir=$output_dir/grubenv mender_disk_image= bootloader_toolchain= device_type= keep=0 efi_boot=EFI/BOOT EFI_STUB_VER="4.12.0" build_log=$output_dir/build.log declare -a mender_disk_mappings version() { echo "$@" | awk -F. '{ printf("%d%03d%03d%03d\n", $1,$2,$3,$4); }' } get_kernel_version() { local search_path=$1/boot local resultvar=$2 [ ! -d "$search_path" ] && { return 1; } kernel_image=$(ls -1 $search_path | grep -E "^vmlinuz") local myresult=${kernel_image#*-} eval $resultvar="'$myresult'" return 0 } build_env_lock_boot_files() { log "\tBuilding boot scripts and tools." local grubenv_repo_vc_dir=$grubenv_dir/.git local grubenv_build_dir=$grubenv_dir/build mkdir -p $grubenv_dir if [ ! -d $grubenv_repo_vc_dir ]; then git clone https://github.com/mendersoftware/grub-mender-grubenv.git $grubenv_dir >> "$build_log" 2>&1 fi cd $grubenv_dir mkdir -p $grubenv_build_dir # Remove old defines & settings. make --quiet distclean >> "$build_log" 2>&1 # Prepare configuration file. cp mender_grubenv_defines.example mender_grubenv_defines local kernel_imagetype=kernel local kernel_devicetree=dtb sed -i '/^kernel_imagetype/s/=.*$/='${kernel_imagetype}'/' mender_grubenv_defines sed -i '/^kernel_devicetree/s/=.*$/='${kernel_devicetree//\//\\/}'/' mender_grubenv_defines if [ "$device_type" == "qemux86_64" ]; then local root_base=/dev/hda sed -i '/^mender_kernel_root_base/s/=.*$/='${root_base//\//\\/}'/' mender_grubenv_defines fi make --quiet >> "$build_log" 2>&1 rc=$? [[ $rc -eq 0 ]] && { make --quiet DESTDIR=$grubenv_build_dir install >> "$build_log" 2>&1; } rc=$? [[ $rc -ne 0 ]] && { log "\tError: building process failed. Aborting."; } cd ${output_dir} return $rc } # Takes following arguments: # # $1 - linux kernel version build_grub_efi() { if [ "$device_type" == "rockpro64" ]; then return 0 fi log "\tBuilding GRUB efi file." local grub_build_dir=$grub_dir/build local grub_arm_dir=$grub_build_dir/arm local host=$(uname -m) local grub_host_dir=$grub_build_dir/$host local grub_repo_vc_dir=$grub_dir/.git local version=$(echo $1 | sed 's/[^0-9.]*\([0-9.]*\).*/\1/') # Build grub modules for arm platform and executables for the host. if [ ! -d $grub_repo_vc_dir ]; then git clone git://git.savannah.gnu.org/grub.git $grub_dir >> "$build_log" 2>&1 fi cd $grub_dir make --quiet distclean >> "$build_log" 2>&1 if [ $(version $version) -lt $(version $EFI_STUB_VER) ]; then # To avoid error message: "plain image kernel not supported - rebuild # with CONFIG_(U)EFI_STUB enabled" - use a specific commit. git checkout 9b37229f0 >> "$build_log" 2>&1 else git checkout 72e80c025 >> "$build_log" 2>&1 fi mkdir -p $grub_arm_dir mkdir -p $grub_host_dir local cores=$(nproc) # First build host tools. ./autogen.sh >> "$build_log" 2>&1 ./configure --quiet CC=gcc --target=${host} --with-platform=efi --prefix=$grub_host_dir >> "$build_log" 2>&1 make --quiet -j$cores >> "$build_log" 2>&1 make --quiet install >> "$build_log" 2>&1 local format=${host}-efi grub_name=bootx64.efi local modules_path=$grub_host_dir/lib/grub/$format/ if [ "$device_type" == "beaglebone" ]; then # Clean workspace. make --quiet clean >> "$build_log" 2>&1 make --quiet distclean >> "$build_log" 2>&1 # Now build ARM modules. ./configure --quiet --host=$bootloader_toolchain --with-platform=efi \ --prefix=$grub_arm_dir CFLAGS="-Os -march=armv7-a" \ CCASFLAGS="-march=armv7-a" --disable-werror >> "$build_log" 2>&1 make --quiet -j$cores >> "$build_log" 2>&1 make --quiet install >> "$build_log" 2>&1 format=arm-efi grub_name=grub-arm.efi modules_path=$grub_arm_dir/lib/grub/$format/ fi # Build GRUB EFI image. ${grub_host_dir}/bin/grub-mkimage -v -p /$efi_boot -o $grub_name --format=$format \ -d $modules_path boot linux ext2 fat serial part_msdos part_gpt normal \ efi_gop iso9660 configfile search loadenv test cat echo gcry_sha256 halt \ hashsum loadenv reboot >> "$build_log" 2>&1 rc=$? [[ $rc -ne 0 ]] && { log "\tBuilding grub.efi failed. Aborting."; } \ || { log "\tBuilding grub.efi succeeded."; } cd ${output_dir} return $rc } # Takes following arguments: # # $1 - boot partition mountpoint set_uenv() { local boot_dir=$1 # Erase/create uEnv.txt file. sudo install -b -m 644 /dev/null $boot_dir/uEnv.txt # Fill uEnv.txt file. cat <<- 'EOF' | sudo tee $boot_dir/uEnv.txt 2>&1 >/dev/null bootdir= grubfile=EFI/BOOT/grub-arm.efi grubaddr=0x80007fc0 loadgrub=fatload mmc 0:1 ${grubaddr} ${grubfile} grubstart=bootefi ${grubaddr} uenvcmd=mmc rescan; run loadgrub; run grubstart; EOF } # Takes following arguments: # # $1 - boot partition mountpoint # $2 - primary partition mountpoint # $3 - linux kernel version install_files() { log "\tInstalling GRUB files." local boot_dir=$1 local rootfs_dir=$2 local linux_version=$3 if [ "$device_type" == "rockpro64" ]; then cp $integration_dir/rockpro64/boot.scr ${boot_dir}/ # It is not possible to resize rootfs part when using Mender. so disable # the service touch ${rootfs_dir}/root/.no_rootfs_resize return 0 fi local grub_build_dir=$grub_dir/build local grub_arm_dir=$grub_build_dir/arm local grub_host_dir=$grub_build_dir/$(uname -m) local grubenv_build_dir=$grubenv_dir/build local grubenv_efi_boot_dir=$grubenv_build_dir/boot/efi/EFI/BOOT/ local efi_boot_dir=$boot_dir/$efi_boot # Make sure env, lock, lock.sha256sum files exists in working directory. [[ ! -d $grubenv_efi_boot_dir/mender_grubenv1 || \ ! -d $grubenv_efi_boot_dir/mender_grubenv2 ]] && \ { log "Error: cannot find mender grub related files."; return 1; } sudo install -d -m 755 $efi_boot_dir cd $grubenv_efi_boot_dir && find . -type f -exec sudo install -Dm 644 "{}" "$efi_boot_dir/{}" \; cd ${output_dir} if [ "$device_type" == "qemux86_64" ]; then sudo install -m 0755 ${grub_host_dir}/bin/grub-editenv $rootfs_dir/usr/bin else sudo install -m 0755 ${grub_arm_dir}/bin/grub-editenv $rootfs_dir/usr/bin fi sudo install -m 0644 ${grub_dir}/${grub_name} $efi_boot_dir sudo install -m 0755 $grubenv_build_dir/usr/bin/fw_printenv $rootfs_dir/sbin/fw_printenv sudo install -m 0755 $grubenv_build_dir/usr/bin/fw_setenv $rootfs_dir/sbin/fw_setenv # Replace U-Boot default printenv/setenv commands. sudo ln -fs /sbin/fw_printenv $rootfs_dir/usr/bin/fw_printenv sudo ln -fs /sbin/fw_setenv $rootfs_dir/usr/bin/fw_setenv #Create links for grub if [ "$device_type" == "qemux86_64" ]; then # Copy kernel image to fit the grubenv defines. sudo cp $boot_dir/bzImage $rootfs_dir/boot/kernel elif [ "$device_type" == "beaglebone" ]; then #Replace U-Boot default images for Debian 9.5 if [ `grep -s '9.5' $rootfs_dir/etc/debian_version | wc -l` -eq 1 ]; then sudo cp ${tool_dir}/files/uboot_debian_9.4/MLO ${boot_dir}/MLO sudo cp ${tool_dir}/files/uboot_debian_9.4/u-boot.img ${boot_dir}/u-boot.img fi # Make links to kernel and device tree files. sudo ln -sf /boot/dtbs/$linux_version/am335x-boneblack.dtb $rootfs_dir/boot/dtb sudo ln -sf /boot/vmlinuz-$linux_version $rootfs_dir/boot/kernel set_uenv $boot_dir fi } do_install_bootloader() { if [ -z "${mender_disk_image}" ]; then log "Mender raw disk image not set. Aborting." exit 1 fi if [ -z "${bootloader_toolchain}" ]; then log "ARM GCC toolchain not set. Aborting." exit 1 fi if [ -z "${device_type}" ]; then log "Target device type name not set. Aborting." exit 1 fi if ! [ -x "$(command -v ${bootloader_toolchain}-gcc)" ]; then log "Error: ARM GCC not found in PATH. Aborting." exit 1 fi [ ! -f $mender_disk_image ] && \ { log "$mender_disk_image - file not found. Aborting."; exit 1; } # Map & mount Mender compliant image. create_device_maps $mender_disk_image mender_disk_mappings # Change current directory to 'output' directory. cd $output_dir boot=${mender_disk_mappings[0]} primary=${mender_disk_mappings[1]} map_boot=/dev/mapper/"$boot" map_primary=/dev/mapper/"$primary" path_boot=$output_dir/sdimg/boot path_primary=$output_dir/sdimg/primary mkdir -p ${path_boot} ${path_primary} sudo mount ${map_boot} ${path_boot} sudo mount ${map_primary} ${path_primary} get_kernel_version ${path_primary} kernel_version log "\tFound kernel version: $kernel_version" build_env_lock_boot_files rc=$? if [ "$device_type" == "qemux86_64" ]; then [[ $rc -eq 0 ]] && { build_grub_efi ${EFI_STUB_VER}; } else [[ $rc -eq 0 ]] && { build_grub_efi ${kernel_version}; } fi rc=$? [[ $rc -eq 0 ]] && { install_files ${path_boot} ${path_primary} ${kernel_version}; } rc=$? # Back to working directory. cd $tool_dir && sync detach_device_maps ${mender_disk_mappings[@]} # Clean files. rm -rf $output_dir/sdimg [[ $keep -eq 0 ]] && { rm -rf $grubenv_dir $grub_dir; } [[ $rc -ne 0 ]] && { exit 1; } || { log "\tDone."; } } PARAMS="" while (( "$#" )); do case "$1" in -m | --mender-disk-image) mender_disk_image=$2 shift 2 ;; -b | --bootloader-toolchain) bootloader_toolchain=$2 shift 2 ;; -d | --device-type) device_type=$2 shift 2 ;; -k | --keep) keep=1 shift 1 ;; -h | --help) show_help exit 0 ;; --) shift break ;; -*) log "Error: unsupported option $1" exit 1 ;; *) PARAMS="$PARAMS $1" shift ;; esac done eval set -- "$PARAMS" # Some commands expect elevated privileges. sudo true do_install_bootloader