| #!/bin/bash |
| # |
| # This file is subject to the terms and conditions of the GNU General Public |
| # License. See the file "COPYING" in the main directory of this archive |
| # for more details. |
| # |
| # Copyright (C) 2017 by Changbin Du <changbin.du@intel.com> |
| # |
| # Adapted from code in arch/x86/boot/Makefile by H. Peter Anvin and others |
| # |
| # "make fdimage/fdimage144/fdimage288/hdimage/isoimage" |
| # script for x86 architecture |
| # |
| # Arguments: |
| # $1 - fdimage format |
| # $2 - target image file |
| # $3 - kernel bzImage file |
| # $4 - mtools configuration file |
| # $5 - kernel cmdline |
| # $6+ - initrd image file(s) |
| # |
| # This script requires: |
| # bash |
| # syslinux |
| # mtools (for fdimage* and hdimage) |
| # edk2/OVMF (for hdimage) |
| # |
| # Otherwise try to stick to POSIX shell commands... |
| # |
| |
| # Use "make V=1" to debug this script |
| case "${KBUILD_VERBOSE}" in |
| *1*) |
| set -x |
| ;; |
| esac |
| |
| # Exit the top-level shell with an error |
| topshell=$$ |
| trap 'exit 1' USR1 |
| die() { |
| echo "" 1>&2 |
| echo " *** $*" 1>&2 |
| echo "" 1>&2 |
| kill -USR1 $topshell |
| } |
| |
| # Verify the existence and readability of a file |
| verify() { |
| if [ ! -f "$1" -o ! -r "$1" ]; then |
| die "Missing file: $1" |
| fi |
| } |
| |
| diskfmt="$1" |
| FIMAGE="$2" |
| FBZIMAGE="$3" |
| MTOOLSRC="$4" |
| KCMDLINE="$5" |
| shift 5 # Remaining arguments = initrd files |
| |
| export MTOOLSRC |
| |
| # common options for dd |
| dd='dd iflag=fullblock' |
| |
| # Make sure the files actually exist |
| verify "$FBZIMAGE" |
| |
| declare -a FDINITRDS |
| irdpfx=' initrd=' |
| initrdopts_syslinux='' |
| initrdopts_efi='' |
| for f in "$@"; do |
| if [ -f "$f" -a -r "$f" ]; then |
| FDINITRDS=("${FDINITRDS[@]}" "$f") |
| fname="$(basename "$f")" |
| initrdopts_syslinux="${initrdopts_syslinux}${irdpfx}${fname}" |
| irdpfx=, |
| initrdopts_efi="${initrdopts_efi} initrd=${fname}" |
| fi |
| done |
| |
| # Read a $3-byte littleendian unsigned value at offset $2 from file $1 |
| le() { |
| local n=0 |
| local m=1 |
| for b in $(od -A n -v -j $2 -N $3 -t u1 "$1"); do |
| n=$((n + b*m)) |
| m=$((m * 256)) |
| done |
| echo $n |
| } |
| |
| # Get the EFI architecture name such that boot{name}.efi is the default |
| # boot file name. Returns false with no output if the file is not an |
| # EFI image or otherwise unknown. |
| efiarch() { |
| [ -f "$1" ] || return |
| [ $(le "$1" 0 2) -eq 23117 ] || return # MZ magic |
| peoffs=$(le "$1" 60 4) # PE header offset |
| [ $peoffs -ge 64 ] || return |
| [ $(le "$1" $peoffs 4) -eq 17744 ] || return # PE magic |
| case $(le "$1" $((peoffs+4+20)) 2) in # PE type |
| 267) ;; # PE32 |
| 523) ;; # PE32+ |
| *) return 1 ;; # Invalid |
| esac |
| [ $(le "$1" $((peoffs+4+20+68)) 2) -eq 10 ] || return # EFI app |
| case $(le "$1" $((peoffs+4)) 2) in # Machine type |
| 332) echo i386 ;; |
| 450) echo arm ;; |
| 512) echo ia64 ;; |
| 20530) echo riscv32 ;; |
| 20580) echo riscv64 ;; |
| 20776) echo riscv128 ;; |
| 34404) echo x64 ;; |
| 43620) echo aa64 ;; |
| esac |
| } |
| |
| # Get the combined sizes in bytes of the files given, counting sparse |
| # files as full length, and padding each file to cluster size |
| cluster=16384 |
| filesizes() { |
| local t=0 |
| local s |
| for s in $(ls -lnL "$@" 2>/dev/null | awk '/^-/{ print $5; }'); do |
| t=$((t + ((s+cluster-1)/cluster)*cluster)) |
| done |
| echo $t |
| } |
| |
| # Expand directory names which should be in /usr/share into a list |
| # of possible alternatives |
| sharedirs() { |
| local dir file |
| for dir in /usr/share /usr/lib64 /usr/lib; do |
| for file; do |
| echo "$dir/$file" |
| echo "$dir/${file^^}" |
| done |
| done |
| } |
| efidirs() { |
| local dir file |
| for dir in /usr/share /boot /usr/lib64 /usr/lib; do |
| for file; do |
| echo "$dir/$file" |
| echo "$dir/${file^^}" |
| done |
| done |
| } |
| |
| findsyslinux() { |
| local f="$(find -L $(sharedirs syslinux isolinux) \ |
| -name "$1" -readable -type f -print -quit 2>/dev/null)" |
| if [ ! -f "$f" ]; then |
| die "Need a $1 file, please install syslinux/isolinux." |
| fi |
| echo "$f" |
| return 0 |
| } |
| |
| findovmf() { |
| local arch="$1" |
| shift |
| local -a names=(-false) |
| local name f |
| for name; do |
| names=("${names[@]}" -or -iname "$name") |
| done |
| for f in $(find -L $(efidirs edk2 ovmf) \ |
| \( "${names[@]}" \) -readable -type f \ |
| -print 2>/dev/null); do |
| if [ "$(efiarch "$f")" = "$arch" ]; then |
| echo "$f" |
| return 0 |
| fi |
| done |
| die "Need a $1 file for $arch, please install EDK2/OVMF." |
| } |
| |
| do_mcopy() { |
| if [ ${#FDINITRDS[@]} -gt 0 ]; then |
| mcopy "${FDINITRDS[@]}" "$1" |
| fi |
| if [ -n "$efishell" ]; then |
| mmd "$1"EFI "$1"EFI/Boot |
| mcopy "$efishell" "$1"EFI/Boot/boot${kefiarch}.efi |
| fi |
| if [ -n "$kefiarch" ]; then |
| echo linux "$KCMDLINE$initrdopts_efi" | \ |
| mcopy - "$1"startup.nsh |
| fi |
| echo default linux "$KCMDLINE$initrdopts_syslinux" | \ |
| mcopy - "$1"syslinux.cfg |
| mcopy "$FBZIMAGE" "$1"linux |
| } |
| |
| genbzdisk() { |
| verify "$MTOOLSRC" |
| mformat -v 'LINUX_BOOT' a: |
| syslinux "$FIMAGE" |
| do_mcopy a: |
| } |
| |
| genfdimage144() { |
| verify "$MTOOLSRC" |
| $dd if=/dev/zero of="$FIMAGE" bs=1024 count=1440 2>/dev/null |
| mformat -v 'LINUX_BOOT' v: |
| syslinux "$FIMAGE" |
| do_mcopy v: |
| } |
| |
| genfdimage288() { |
| verify "$MTOOLSRC" |
| $dd if=/dev/zero of="$FIMAGE" bs=1024 count=2880 2>/dev/null |
| mformat -v 'LINUX_BOOT' w: |
| syslinux "$FIMAGE" |
| do_mcopy w: |
| } |
| |
| genhdimage() { |
| verify "$MTOOLSRC" |
| mbr="$(findsyslinux mbr.bin)" |
| kefiarch="$(efiarch "$FBZIMAGE")" |
| if [ -n "$kefiarch" ]; then |
| # The efishell provides command line handling |
| efishell="$(findovmf $kefiarch shell.efi shell${kefiarch}.efi)" |
| ptype='-T 0xef' # EFI system partition, no GPT |
| fi |
| sizes=$(filesizes "$FBZIMAGE" "${FDINITRDS[@]}" "$efishell") |
| # Allow 1% + 2 MiB for filesystem and partition table overhead, |
| # syslinux, and config files; this is probably excessive... |
| megs=$(((sizes + sizes/100 + 2*1024*1024 - 1)/(1024*1024))) |
| $dd if=/dev/zero of="$FIMAGE" bs=$((1024*1024)) count=$megs 2>/dev/null |
| mpartition -I -c -s 32 -h 64 $ptype -b 64 -a p: |
| $dd if="$mbr" of="$FIMAGE" bs=440 count=1 conv=notrunc 2>/dev/null |
| mformat -v 'LINUX_BOOT' -s 32 -h 64 -c $((cluster/512)) -t $megs h: |
| syslinux --offset $((64*512)) "$FIMAGE" |
| do_mcopy h: |
| } |
| |
| geniso() { |
| tmp_dir="$(dirname "$FIMAGE")/isoimage" |
| rm -rf "$tmp_dir" |
| mkdir "$tmp_dir" |
| isolinux=$(findsyslinux isolinux.bin) |
| ldlinux=$(findsyslinux ldlinux.c32) |
| cp "$isolinux" "$ldlinux" "$tmp_dir" |
| cp "$FBZIMAGE" "$tmp_dir"/linux |
| echo default linux "$KCMDLINE" > "$tmp_dir"/isolinux.cfg |
| cp "${FDINITRDS[@]}" "$tmp_dir"/ |
| genisoimage -J -r -appid 'LINUX_BOOT' -input-charset=utf-8 \ |
| -quiet -o "$FIMAGE" -b isolinux.bin \ |
| -c boot.cat -no-emul-boot -boot-load-size 4 \ |
| -boot-info-table "$tmp_dir" |
| isohybrid "$FIMAGE" 2>/dev/null || true |
| rm -rf "$tmp_dir" |
| } |
| |
| rm -f "$FIMAGE" |
| |
| case "$diskfmt" in |
| bzdisk) genbzdisk;; |
| fdimage144) genfdimage144;; |
| fdimage288) genfdimage288;; |
| hdimage) genhdimage;; |
| isoimage) geniso;; |
| *) die "Unknown image format: $diskfmt";; |
| esac |