419 lines
13 KiB
Bash
419 lines
13 KiB
Bash
|
#!/bin/sh
|
|||
|
#
|
|||
|
# Purpose {{{
|
|||
|
# This script will create an Unified Kernel + UEFI entry for ArchLinux
|
|||
|
# 1. Verify dependancies (efibootmgr, objcopy…).
|
|||
|
# 2. Clean older unified kernels.
|
|||
|
# 3. Backup last unified kernel.
|
|||
|
# 4. Create a new unified kernel from vmlinuz, initramfs, UUID of root,…
|
|||
|
# 5. Ensure an UEFI entry exists for this unified kernel.
|
|||
|
#
|
|||
|
# 2023-08-17
|
|||
|
# }}}
|
|||
|
# TODO {{{
|
|||
|
# 1. Encrypted LVM is not yet supported. See crypt_part_uuid relative comments.
|
|||
|
# }}}
|
|||
|
# Flags {{{
|
|||
|
## Exit on error {{{
|
|||
|
set -o errexit
|
|||
|
## }}}
|
|||
|
## Exit on unset var {{{
|
|||
|
### Use "${VARNAME-}" to test a var that may not have been set
|
|||
|
set -o nounset
|
|||
|
## }}}
|
|||
|
## Pipeline command is treated as failed {{{
|
|||
|
### Not available in POSIX sh − https://github.com/koalaman/shellcheck/wiki/SC3040
|
|||
|
#set -o pipefail
|
|||
|
## }}}
|
|||
|
## Help with debugging {{{
|
|||
|
### Call the script by prefixing it with "TRACE=1 ./script.sh"
|
|||
|
if [ "${TRACE-0}" -eq 1 ]; then set -o xtrace; fi
|
|||
|
## }}}
|
|||
|
# }}}
|
|||
|
# Vars {{{
|
|||
|
PROGNAME=$(basename "${0}"); readonly PROGNAME
|
|||
|
PROGDIR=$(readlink --canonicalize-missing $(dirname "${0}")); readonly PROGDIR
|
|||
|
ARGS="${*}"; readonly ARGS
|
|||
|
readonly NBARGS="${#}"
|
|||
|
[ -z "${DEBUG-}" ] && DEBUG=1
|
|||
|
## Export DEBUG for sub-script
|
|||
|
export DEBUG
|
|||
|
|
|||
|
## Default values for Unified kernel
|
|||
|
readonly ROOT_UUID_DEFAULT=$(findmnt --kernel --noheadings --output UUID -- /)
|
|||
|
readonly ROOT_FSTYPE_DEFAULT=$(findmnt --kernel --noheadings --output FSTYPE -- /)
|
|||
|
readonly CRYPT_PART_UUID_DEFAULT=$(blkid | sed --silent 's;/dev/.*_crypt.*UUID="\(.*\)".*TYPE=.*;\1;p')
|
|||
|
readonly UEFI_BOOT_STUB_FILE_DEFAULT="/usr/lib/systemd/boot/efi/linuxx64.efi.stub"
|
|||
|
|
|||
|
## Default values for UEFI entry
|
|||
|
readonly EFI_BASE_LABEL_DEFAULT="Arch Linux unified"
|
|||
|
readonly EFI_MOUNT_PATH_DEFAULT="/boot/efi"
|
|||
|
|
|||
|
## Temp files
|
|||
|
readonly temp_kernel_command_file="/tmp/kernel.command.temp"
|
|||
|
|
|||
|
## Colors
|
|||
|
readonly PURPLE='\033[1;35m'
|
|||
|
readonly RED='\033[0;31m'
|
|||
|
readonly RESET='\033[0m'
|
|||
|
readonly COLOR_DEBUG="${PURPLE}"
|
|||
|
# }}}
|
|||
|
usage() { # {{{
|
|||
|
|
|||
|
cat <<- HELP
|
|||
|
usage: $PROGNAME [-d|-h|-b|-f|-k|-u|-l|-m]
|
|||
|
|
|||
|
Script to build unified kernel for Arch Linux will all required informations
|
|||
|
(vmlinuz, initramfs, UUID of root's partition,…) and create an UEFI entry.
|
|||
|
|
|||
|
EXAMPLES :
|
|||
|
- Build unified kernel with default :
|
|||
|
${PROGNAME}
|
|||
|
|
|||
|
- Build unified kernel with current command line :
|
|||
|
${PROGNAME} --kernel "\$(< /proc/cmdline)"
|
|||
|
|
|||
|
OPTIONS :
|
|||
|
-d,--debug
|
|||
|
Enable debug messages.
|
|||
|
|
|||
|
-h,--help
|
|||
|
Print this help message.
|
|||
|
|
|||
|
Unified kernel OPTIONS :
|
|||
|
|
|||
|
-b,--boot-stub,--uefi-boot
|
|||
|
Define UEFI boot stub file to use.
|
|||
|
(default: ${UEFI_BOOT_STUB_FILE_DEFAULT})
|
|||
|
For more infos, see: https://man.archlinux.org/man/linuxx64.efi.stub.7.en
|
|||
|
|
|||
|
-f,--fs,--root-fs
|
|||
|
Define a different root partition's filesystem.
|
|||
|
(default: ${ROOT_FSTYPE_DEFAULT})
|
|||
|
Thanks to command : findmnt --kernel --noheadings --output FSTYPE -- /
|
|||
|
|
|||
|
-u,--uuid,--root-uuid
|
|||
|
Define a different root partition's UUID.
|
|||
|
(default: ${ROOT_UUID_DEFAULT})
|
|||
|
Thanks to command : findmnt --kernel --noheadings --output UUID -- /
|
|||
|
|
|||
|
-k,--kernel,--kernel-cmd
|
|||
|
Define the kernel command line to use.
|
|||
|
(default: build with root-fs|root-uuid and add_efi_memmap + ro options)
|
|||
|
|
|||
|
UEFI entry OPTIONS :
|
|||
|
|
|||
|
-l,--label
|
|||
|
Set a new label for UEFI entry.
|
|||
|
(default: ${EFI_BASE_LABEL_DEFAULT})
|
|||
|
|
|||
|
-m,--mount-path,--esp
|
|||
|
Define a different mountpoint for the EFI partition.
|
|||
|
(default: ${EFI_MOUNT_PATH_DEFAULT})
|
|||
|
HELP
|
|||
|
}
|
|||
|
# }}}
|
|||
|
debug_message() { # {{{
|
|||
|
|
|||
|
local_debug_message="${1}"
|
|||
|
|
|||
|
## Print message if DEBUG is enable (=0)
|
|||
|
[ "${DEBUG}" -eq "0" ] && printf '\e[1;35m%-6b\e[m\n' "DEBUG − ${PROGNAME} : ${local_debug_message}"
|
|||
|
|
|||
|
unset local_debug_message
|
|||
|
|
|||
|
return 0
|
|||
|
}
|
|||
|
# }}}
|
|||
|
error_message() { # {{{
|
|||
|
|
|||
|
local_error_message="${1}"
|
|||
|
local_error_code="${2}"
|
|||
|
|
|||
|
## Print message
|
|||
|
printf '%b\n' "ERROR − ${PROGNAME} : ${RED}${local_error_message}${RESET}" >&2
|
|||
|
|
|||
|
unset local_error_message
|
|||
|
|
|||
|
exit "${local_error_code:=66}"
|
|||
|
}
|
|||
|
# }}}
|
|||
|
define_vars() { # {{{
|
|||
|
|
|||
|
# If root_uuid wasn't defined (argument) {{{
|
|||
|
if [ -z "${root_uuid-}" ]; then
|
|||
|
## Use default value
|
|||
|
readonly root_uuid="${ROOT_UUID_DEFAULT}"
|
|||
|
fi
|
|||
|
# }}}
|
|||
|
# If root_fstype wasn't defined (argument) {{{
|
|||
|
if [ -z "${root_fstype-}" ]; then
|
|||
|
## Use default value
|
|||
|
readonly root_fstype="${ROOT_FSTYPE_DEFAULT}"
|
|||
|
fi
|
|||
|
# }}}
|
|||
|
# If kernel_command_line wasn't defined (argument) {{{
|
|||
|
if [ -z "${kernel_command_line-}" ]; then
|
|||
|
## Use default value
|
|||
|
readonly kernel_command_line="root=UUID=${root_uuid} rootfstype=${root_fstype} add_efi_memmap ro"
|
|||
|
## For encrypted LVM
|
|||
|
#readonly kernel_command_line="root=UUID=${root_uuid} rootfstype=${root_fstype} add_efi_memmap ro cryptdevice=UUID=${crypt_part_uuid}:lvm"
|
|||
|
fi
|
|||
|
# }}}
|
|||
|
# If uefi_boot_stub_file wasn't defined (argument) {{{
|
|||
|
if [ -z "${uefi_boot_stub_file-}" ]; then
|
|||
|
## Use default value
|
|||
|
readonly uefi_boot_stub_file="${UEFI_BOOT_STUB_FILE_DEFAULT}"
|
|||
|
fi
|
|||
|
# }}}
|
|||
|
|
|||
|
# If efi_base_label wasn't defined (argument) {{{
|
|||
|
if [ -z "${efi_base_label-}" ]; then
|
|||
|
## Use default value
|
|||
|
readonly efi_base_label="${EFI_BASE_LABEL_DEFAULT}"
|
|||
|
fi
|
|||
|
# }}}
|
|||
|
# If efi_mount_path wasn't defined (argument) {{{
|
|||
|
if [ -z "${efi_mount_path-}" ]; then
|
|||
|
## Use default value
|
|||
|
readonly efi_mount_path="${EFI_MOUNT_PATH_DEFAULT}"
|
|||
|
fi
|
|||
|
# }}}
|
|||
|
|
|||
|
# If esp_device wasn't defined (argument) {{{
|
|||
|
if [ -z "${esp_device-}" ]; then
|
|||
|
## Use default value
|
|||
|
readonly esp_device=$(findmnt -kno SOURCE "${efi_mount_path}" | grep --invert-match -- systemd | sed s-/dev/--)
|
|||
|
fi
|
|||
|
# }}}
|
|||
|
# If esp_disk wasn't defined (argument) {{{
|
|||
|
if [ -z "${esp_disk-}" ]; then
|
|||
|
## Use default value
|
|||
|
readonly esp_disk=$(lsblk /dev/"${esp_device}" -no pkname)
|
|||
|
fi
|
|||
|
# }}}
|
|||
|
# If esp_part wasn't defined (argument) {{{
|
|||
|
if [ -z "${esp_part-}" ]; then
|
|||
|
## Use default value
|
|||
|
readonly esp_part=$(cat /sys/class/block/"${esp_device}"/partition)
|
|||
|
fi
|
|||
|
# }}}
|
|||
|
|
|||
|
}
|
|||
|
# }}}
|
|||
|
|
|||
|
is_command_available() { # {{{
|
|||
|
|
|||
|
local_command_available_cmd="${1}"
|
|||
|
debug_prefix="${2:-}"
|
|||
|
|
|||
|
## Return False by default
|
|||
|
return_command_available="1"
|
|||
|
|
|||
|
if [ "$(command -v ${local_command_available_cmd})" ]; then
|
|||
|
debug_message "${debug_prefix}is_command_available − \
|
|||
|
${RED}${local_command_available_cmd}${COLOR_DEBUG} seems present on this host."
|
|||
|
return_command_available="0"
|
|||
|
else
|
|||
|
debug_message "${debug_prefix}is_command_available − \
|
|||
|
${RED}${local_command_available_cmd}${COLOR_DEBUG} is not available on this host."
|
|||
|
return_command_available="1"
|
|||
|
fi
|
|||
|
|
|||
|
unset local_command_available_cmd
|
|||
|
unset debug_prefix
|
|||
|
|
|||
|
return "${return_command_available}"
|
|||
|
}
|
|||
|
# }}}
|
|||
|
|
|||
|
main() { # {{{
|
|||
|
|
|||
|
debug_message "--- MAIN BEGIN"
|
|||
|
|
|||
|
## If a efibootmgr is missing {{{
|
|||
|
### Exit with error message
|
|||
|
is_command_available "efibootmgr" "| " \
|
|||
|
|| error_message "No efibootmgr command available. Please install efibootmgr package with your package manager (pacman -S efibootmgr)." 01
|
|||
|
## }}}
|
|||
|
## If a objcopy is missing {{{
|
|||
|
### Exit with error message
|
|||
|
is_command_available "objcopy" "| " \
|
|||
|
|| error_message "No objcopy command available. Please install binutils package with your package manager (pacman -S binutils)." 02
|
|||
|
## }}}
|
|||
|
|
|||
|
## Define all vars
|
|||
|
define_vars
|
|||
|
debug_message "| Define vars"
|
|||
|
|
|||
|
## If UEFI boot stub is missing {{{
|
|||
|
### Exit with error message
|
|||
|
test -f "${uefi_boot_stub_file}" \
|
|||
|
|| error_message "No UEFI boot stub file available (${uefi_boot_stub_file}). Please verify your systemd installation (systemd or systemd-boot-efi packages)." 03
|
|||
|
## }}}
|
|||
|
## Ensure EFI device is mounted {{{
|
|||
|
if ! mountpoint --quiet "${efi_mount_path}"; then
|
|||
|
mount "${efi_mount_path}" \
|
|||
|
|| error_message "Can't mount EFI device" 04
|
|||
|
fi
|
|||
|
## }}}
|
|||
|
|
|||
|
## Put Kernel command line in temp file {{{
|
|||
|
rm --force -- "${temp_kernel_command_file}" ; touch "${temp_kernel_command_file}"
|
|||
|
printf "%s" "${kernel_command_line}" >> "${temp_kernel_command_file}"
|
|||
|
## }}}
|
|||
|
|
|||
|
## Calculate address values to use for each section {{{
|
|||
|
osrel_offs=$(objdump --section-headers -- "${uefi_boot_stub_file}" | awk 'NF==7 {size=strtonum("0x"$3); offset=strtonum("0x"$4)} END {print size + offset}')
|
|||
|
cmdline_offs=$((osrel_offs + $(stat --dereference --format=%s -- "/usr/lib/os-release")))
|
|||
|
linux_offs=$((cmdline_offs + $(stat --dereference --format=%s -- "${temp_kernel_command_file}")))
|
|||
|
#linux_offs=$((cmdline_offs + $(stat --dereference --format=%s -- "/proc/cmdline")))
|
|||
|
initrd_offs=$((linux_offs + $(stat --dereference --format=%s -- "/boot/vmlinuz-linux")))
|
|||
|
## }}}
|
|||
|
|
|||
|
## Debug message {{{
|
|||
|
debug_message "Try to build Unified kernel with from this informations :\r
|
|||
|
* UEFI boot stub file: ${RED}${uefi_boot_stub_file}${COLOR_DEBUG}\r
|
|||
|
* Root's UUID: ${RED}${root_uuid}${COLOR_DEBUG}\r
|
|||
|
* Root's filesystem: ${RED}${root_fstype}${COLOR_DEBUG}\r
|
|||
|
* Kernel command line: ${RED}${kernel_command_line}${COLOR_DEBUG}\r
|
|||
|
* EFI device: ${RED}${esp_device}${COLOR_DEBUG}\r
|
|||
|
* EFI disk: ${RED}${esp_disk}${COLOR_DEBUG}\r
|
|||
|
* EFI partition number: ${RED}${esp_part}${COLOR_DEBUG}\r
|
|||
|
Addresses values :\r
|
|||
|
* osrel : ${RED}${osrel_offs}${COLOR_DEBUG}\r
|
|||
|
* cmdline : ${RED}${cmdline_offs}${COLOR_DEBUG}\r
|
|||
|
* linux : ${RED}${linux_offs}${COLOR_DEBUG}\r
|
|||
|
* initrd : ${RED}${initrd_offs}${COLOR_DEBUG}\r
|
|||
|
And generate UEFI entry :\r
|
|||
|
* Label: ${RED}${efi_base_label}${COLOR_DEBUG}"
|
|||
|
|
|||
|
## }}}
|
|||
|
|
|||
|
## Ensure EFI mount path subdirectories exists
|
|||
|
mkdir --parents -- "${efi_mount_path}/EFI/arch/"
|
|||
|
## Create unified kernel blob {{{
|
|||
|
objcopy \
|
|||
|
--add-section .osrel="/usr/lib/os-release" --change-section-vma .osrel=$(printf 0x%x $osrel_offs) \
|
|||
|
--add-section .cmdline="${temp_kernel_command_file}" --change-section-vma .cmdline=$(printf 0x%x $cmdline_offs) \
|
|||
|
--add-section .linux="/boot/vmlinuz-linux" --change-section-vma .linux=$(printf 0x%x $linux_offs) \
|
|||
|
--add-section .initrd="/boot/initramfs-linux.img" --change-section-vma .initrd=$(printf 0x%x $initrd_offs) \
|
|||
|
"${uefi_boot_stub_file}" "${efi_mount_path}/EFI/arch/linux.arch.efi"
|
|||
|
## }}}
|
|||
|
## If no UEFI entry for this label + kernel {{{
|
|||
|
## Create one
|
|||
|
efibootmgr | grep --quiet -- "${efi_base_label}.*\\\EFI\\\arch\\\linux.arch.efi" \
|
|||
|
|| efibootmgr --disk /dev/"${esp_disk}" --part "${esp_part}" --create --label "${efi_base_label}" --loader "\\EFI\\arch\\linux.arch.efi"
|
|||
|
## }}}
|
|||
|
|
|||
|
## Remove temp files
|
|||
|
rm --force -- "${temp_kernel_command_file}"
|
|||
|
|
|||
|
debug_message "--- MAIN END"
|
|||
|
}
|
|||
|
# }}}
|
|||
|
|
|||
|
# Manage arguments # {{{
|
|||
|
# This code can't be in a function due to argument management
|
|||
|
|
|||
|
if [ ! "${NBARGS}" -eq "0" ]; then
|
|||
|
|
|||
|
manage_arg="0"
|
|||
|
|
|||
|
## If the first argument ask for help (h|help|-h|-help|-*h|-*help) {{{
|
|||
|
if printf -- '%s' "${1-}" | grep --quiet --extended-regexp -- "^-*h(elp)?$"; then
|
|||
|
usage
|
|||
|
exit 0
|
|||
|
fi
|
|||
|
## }}}
|
|||
|
|
|||
|
## If the first argument is not an option
|
|||
|
if ! printf -- '%s' "${1}" | grep --quiet --extended-regexp -- "^-+";
|
|||
|
then
|
|||
|
## Print help message and exit
|
|||
|
printf '%b\n' "${RED}Invalid option: ${1}${RESET}"
|
|||
|
printf '%b\n' "---"
|
|||
|
usage
|
|||
|
|
|||
|
exit 1
|
|||
|
fi
|
|||
|
|
|||
|
# Parse all options (start with a "-") one by one
|
|||
|
while printf -- '%s' "${1-}" | grep --quiet --extended-regexp -- "^-+"; do
|
|||
|
|
|||
|
case "${1}" in
|
|||
|
## OPTIONS
|
|||
|
-d|--debug ) ## debug
|
|||
|
DEBUG=0
|
|||
|
debug_message "--- Manage argument BEGIN"
|
|||
|
;;
|
|||
|
-h|--help ) ## display help message
|
|||
|
usage
|
|||
|
exit 0
|
|||
|
;;
|
|||
|
## Unified kernel OPTIONS
|
|||
|
-u|--boot-stub|--uefi-boot ) ## Define uefi_boot_stub_file with given arg
|
|||
|
## Move to the next argument
|
|||
|
shift
|
|||
|
## Define var
|
|||
|
readonly uefi_boot_stub_file="${1}"
|
|||
|
;;
|
|||
|
-k|--kernel|--kernel-cmd ) ## Define kernel_command_line with given arg
|
|||
|
## Move to the next argument
|
|||
|
shift
|
|||
|
## Define var
|
|||
|
readonly kernel_command_line="${1}"
|
|||
|
;;
|
|||
|
-f|--fs|--root-fs ) ## Define root_fstype with given arg
|
|||
|
## Move to the next argument
|
|||
|
shift
|
|||
|
## Define var
|
|||
|
readonly root_fstype="${1}"
|
|||
|
;;
|
|||
|
-u|--uuid|--root-uuid ) ## Define root_uuid with given arg
|
|||
|
## Move to the next argument
|
|||
|
shift
|
|||
|
## Define var
|
|||
|
readonly root_uuid="${1}"
|
|||
|
;;
|
|||
|
## UEFI entry OPTIONS
|
|||
|
-l|--label ) ## Define efi_base_label with given arg
|
|||
|
## Move to the next argument
|
|||
|
shift
|
|||
|
## Define var
|
|||
|
readonly efi_base_label="${1}"
|
|||
|
;;
|
|||
|
-m|--mount-path|--esp ) ## Define efi_mount_path with given arg
|
|||
|
## Move to the next argument
|
|||
|
shift
|
|||
|
## Define var
|
|||
|
readonly efi_mount_path="${1}"
|
|||
|
;;
|
|||
|
* ) ## unknow option
|
|||
|
printf '%b\n' "${RED}Invalid option: ${1}${RESET}"
|
|||
|
printf '%b\n' "---"
|
|||
|
usage
|
|||
|
exit 1
|
|||
|
;;
|
|||
|
esac
|
|||
|
|
|||
|
debug_message "| ${RED}${1}${COLOR_DEBUG} option managed."
|
|||
|
|
|||
|
## Move to the next argument
|
|||
|
shift
|
|||
|
manage_arg=$((manage_arg+1))
|
|||
|
|
|||
|
done
|
|||
|
|
|||
|
debug_message "| ${RED}${manage_arg}${COLOR_DEBUG} argument(s) successfully managed."
|
|||
|
else
|
|||
|
debug_message "| No arguments/options to manage."
|
|||
|
fi
|
|||
|
|
|||
|
debug_message "--- Manage argument END"
|
|||
|
# }}}
|
|||
|
|
|||
|
main
|
|||
|
|
|||
|
exit 0
|