calamares/src/modules/bootloader/main.py

549 lines
21 KiB
Python
Raw Normal View History

#!/usr/bin/env python3
2015-02-18 15:06:10 +01:00
# -*- coding: utf-8 -*-
#
# === This file is part of Calamares - <https://calamares.io> ===
#
# SPDX-FileCopyrightText: 2014 Aurélien Gâteau <agateau@kde.org>
# SPDX-FileCopyrightText: 2014 Anke Boersma <demm@kaosx.us>
# SPDX-FileCopyrightText: 2014 Daniel Hillenbrand <codeworkx@bbqlinux.org>
# SPDX-FileCopyrightText: 2014 Benjamin Vaudour <benjamin.vaudour@yahoo.fr>
# SPDX-FileCopyrightText: 2014-2019 Kevin Kofler <kevin.kofler@chello.at>
# SPDX-FileCopyrightText: 2015-2018 Philip Mueller <philm@manjaro.org>
# SPDX-FileCopyrightText: 2016-2017 Teo Mrnjavac <teo@kde.org>
# SPDX-FileCopyrightText: 2017 Alf Gaida <agaida@siduction.org>
# SPDX-FileCopyrightText: 2017-2019 Adriaan de Groot <groot@kde.org>
# SPDX-FileCopyrightText: 2017 Gabriel Craciunescu <crazy@frugalware.org>
# SPDX-FileCopyrightText: 2017 Ben Green <Bezzy1999@hotmail.com>
# SPDX-FileCopyrightText: 2021 Neal Gompa <ngompa13@gmail.com>
# SPDX-License-Identifier: GPL-3.0-or-later
#
# Calamares is Free Software: see the License-Identifier above.
#
import os
import shutil
import subprocess
import libcalamares
from libcalamares.utils import check_target_env_call
2014-10-16 21:08:18 +02:00
import gettext
_ = gettext.translation("calamares-python",
localedir=libcalamares.utils.gettext_path(),
languages=libcalamares.utils.gettext_languages(),
fallback=True).gettext
# This is the sanitizer used all over to tidy up filenames
# to make identifiers (or to clean up names to make filenames).
file_name_sanitizer = str.maketrans(" /()", "_-__")
def pretty_name():
return _("Install bootloader.")
def get_uuid():
2017-03-24 16:39:25 +01:00
"""
Checks and passes 'uuid' to other routine.
2015-02-20 20:54:25 +01:00
:return:
"""
partitions = libcalamares.globalstorage.value("partitions")
2015-06-14 05:08:52 +02:00
for partition in partitions:
if partition["mountPoint"] == "/":
libcalamares.utils.debug("Root partition uuid: \"{!s}\"".format(partition["uuid"]))
return partition["uuid"]
2015-06-14 05:08:52 +02:00
return ""
2014-10-16 21:08:18 +02:00
def get_bootloader_entry_name():
2017-03-24 16:39:25 +01:00
"""
Passes 'bootloader_entry_name' to other routine based
on configuration file.
2015-02-20 20:54:25 +01:00
:return:
"""
2015-02-19 17:59:52 +01:00
if "bootloaderEntryName" in libcalamares.job.configuration:
return libcalamares.job.configuration["bootloaderEntryName"]
else:
branding = libcalamares.globalstorage.value("branding")
2015-02-19 17:59:52 +01:00
return branding["bootloaderEntryName"]
2021-02-22 00:27:46 +01:00
def create_systemd_boot_conf(install_path, efi_dir, uuid, entry, entry_name, kernel, kernel_type, kernel_version):
2017-03-24 16:39:25 +01:00
"""
Creates systemd-boot configuration files based on given parameters.
2015-02-20 20:54:25 +01:00
:param install_path:
:param efi_dir:
2015-02-20 20:54:25 +01:00
:param uuid:
:param entry:
:param entry_name:
2021-02-22 00:27:46 +01:00
:param kernel:
:param kernel_type:
2021-02-22 00:27:46 +01:00
:param kernel_version:
2015-02-20 20:54:25 +01:00
"""
kernel_params = ["quiet"]
partitions = libcalamares.globalstorage.value("partitions")
swap_uuid = ""
swap_outer_mappername = None
cryptdevice_params = []
2015-06-14 05:08:52 +02:00
# Take over swap settings:
# - unencrypted swap partition sets swap_uuid
# - encrypted root sets cryptdevice_params
for partition in partitions:
if partition["fs"] == "linuxswap" and not partition.get("claimed", None):
continue
has_luks = "luksMapperName" in partition
if partition["fs"] == "linuxswap" and not has_luks:
swap_uuid = partition["uuid"]
if (partition["fs"] == "linuxswap" and has_luks):
swap_outer_mappername = partition["luksMapperName"]
if partition["mountPoint"] == "/" and has_luks:
2017-03-24 16:39:25 +01:00
cryptdevice_params = ["cryptdevice=UUID="
+ partition["luksUuid"]
+ ":"
+ partition["luksMapperName"],
"root=/dev/mapper/"
+ partition["luksMapperName"]]
# systemd-boot with a BTRFS root filesystem needs to be told
# about the root subvolume.
for partition in partitions:
if partition["mountPoint"] == "/" and partition["fs"] == "btrfs":
kernel_params.append("rootflags=subvol=@")
if cryptdevice_params:
kernel_params.extend(cryptdevice_params)
else:
kernel_params.append("root=UUID={!s}".format(uuid))
if swap_uuid:
kernel_params.append("resume=UUID={!s}".format(swap_uuid))
2015-02-15 00:03:50 +01:00
if swap_outer_mappername:
kernel_params.append("resume=/dev/mapper/{!s}".format(
swap_outer_mappername))
2021-02-22 00:27:46 +01:00
kernel_line = entry_name + " " + kernel_version
libcalamares.utils.debug("Configure: \"{!s}\"".format(kernel_line))
if kernel_type == "fallback":
2021-02-22 00:27:46 +01:00
version_string = kernel_version + "-fallback"
initrd = "initrd-fallback"
mkinitcpio_option = "-S autodetect"
else:
2021-02-22 00:27:46 +01:00
version_string = kernel_version
initrd = "initrd"
mkinitcpio_option = ""
2021-02-22 00:27:46 +01:00
# get the machine-id
with open(os.path.join(install_path, "etc", "machine-id"), 'r') as machineid_file:
machine_id = machineid_file.read().rstrip('\n')
2021-02-22 00:27:46 +01:00
# Copy kernel to a subdirectory of /efi partition
machine_dir = os.path.join(install_path + efi_dir, machine_id)
try:
os.mkdir(machine_dir)
except FileExistsError: # We can ignore errors caused by the directory existing already
pass
2021-02-22 00:27:46 +01:00
files_dir = os.path.join(machine_dir, kernel_version)
try:
os.mkdir(files_dir)
except FileExistsError: # We can ignore errors caused by the directory existing already
pass
kernel_path = os.path.join(install_path, kernel)
kernel_name = os.path.basename(kernel_path)
2021-02-22 00:27:46 +01:00
shutil.copyfile(kernel_path, os.path.join(files_dir, "linux"))
2021-02-22 00:27:46 +01:00
# generate the initramfs - this is Arch specific and should be replaced for other distros
try:
subprocess.run(["chroot " +
install_path +
" mkinitcpio -k " + kernel_version + " -g " + os.path.join("/", os.path.relpath(files_dir, install_path), initrd)],
shell=True,
capture_output=True,
check=True)
except subprocess.CalledProcessError as cpe:
libcalamares.utils.debug("mkiniticpio failed")
libcalamares.utils.debug("STDOUT: " + cpe.stdout.decode())
libcalamares.utils.debug("STDERR: " + cpe.stderr.decode())
raise
# write the entry
lines = [
2021-02-22 00:27:46 +01:00
'## Generated by Calamares\n',
'\n',
2021-02-22 00:27:46 +01:00
"title {!s}\n".format(entry),
"version {!s}\n".format(version_string),
"machine-id {!s}\n".format(machine_id),
"linux {!s}\n".format(os.path.join("/", machine_id, kernel_version, "linux")),
]
2014-10-16 21:08:18 +02:00
2021-02-22 00:27:46 +01:00
try:
additional_initrd_files = libcalamares.job.configuration["additionalInitrdFiles"]
for initrd_file in additional_initrd_files.split(','):
libcalamares.utils.debug("Attempting to handle initrd image " + initrd_file)
if os.path.isfile(os.path.join(install_path, initrd_file)):
libcalamares.utils.debug("Found image " + initrd_file)
shutil.copyfile(os.path.join(install_path, initrd_file), os.path.join(files_dir, os.path.basename(initrd_file)))
lines.append("initrd {!s}\n".format(os.path.join("/", machine_id, kernel_version, os.path.basename(initrd_file))))
except KeyError: # If the configuration option isn't set, we can just move on
libcalamares.utils.debug("Failed to find key additionalInitrdFiles")
pass
lines.append("initrd {!s}\n".format(os.path.join("/", machine_id, kernel_version, initrd)))
lines.append("options {!s} rw\n".format(" ".join(kernel_params)))
conf_path = os.path.join(install_path + efi_dir,
"loader",
"entries",
machine_id + "-" + version_string + ".conf")
2016-05-04 13:30:54 +02:00
with open(conf_path, 'w') as conf_file:
for line in lines:
conf_file.write(line)
2014-10-16 21:08:18 +02:00
def create_loader(loader_path, entry):
2017-03-24 16:39:25 +01:00
"""
Writes configuration for loader.
2015-02-20 20:54:25 +01:00
:param loader_path:
:param entry:
2015-02-20 20:54:25 +01:00
"""
timeout = libcalamares.job.configuration["timeout"]
lines = [
"timeout {!s}\n".format(timeout),
"default {!s}\n".format(entry),
]
2014-10-16 21:08:18 +02:00
2016-05-04 13:30:54 +02:00
with open(loader_path, 'w') as loader_file:
for line in lines:
loader_file.write(line)
2014-10-16 21:08:18 +02:00
def efi_label():
if "efiBootloaderId" in libcalamares.job.configuration:
efi_bootloader_id = libcalamares.job.configuration[
"efiBootloaderId"]
else:
branding = libcalamares.globalstorage.value("branding")
efi_bootloader_id = branding["bootloaderEntryName"]
return efi_bootloader_id.translate(file_name_sanitizer)
def efi_word_size():
# get bitness of the underlying UEFI
try:
sysfile = open("/sys/firmware/efi/fw_platform_size", "r")
efi_bitness = sysfile.read(2)
except Exception:
# if the kernel is older than 4.0, the UEFI bitness likely isn't
# exposed to the userspace so we assume a 64 bit UEFI here
efi_bitness = "64"
return efi_bitness
def efi_boot_next():
"""
Tell EFI to definitely boot into the just-installed
system next time.
"""
boot_mgr = libcalamares.job.configuration["efiBootMgr"]
boot_entry = None
efi_bootvars = subprocess.check_output([boot_mgr], text=True)
for line in efi_bootvars.split('\n'):
if not line:
continue
words = line.split()
if len(words) >= 2 and words[0] == "BootOrder:":
boot_entry = words[1].split(',')[0]
break
if boot_entry:
subprocess.call([boot_mgr, "-n", boot_entry])
2015-06-11 00:32:34 +02:00
def install_systemd_boot(efi_directory):
2017-03-24 16:39:25 +01:00
"""
Installs systemd-boot as bootloader for EFI setups.
2015-02-20 20:54:25 +01:00
:param efi_directory:
"""
libcalamares.utils.debug("Bootloader: systemd-boot")
install_path = libcalamares.globalstorage.value("rootMountPoint")
2021-02-22 00:27:46 +01:00
kernel_search_path = libcalamares.job.configuration["kernelSearchPath"]
source_kernel_name = libcalamares.job.configuration["kernelName"]
install_efi_directory = install_path + efi_directory
uuid = get_uuid()
distribution = get_bootloader_entry_name()
distribution_translated = distribution.translate(file_name_sanitizer)
2016-05-04 13:30:54 +02:00
loader_path = os.path.join(install_efi_directory,
"loader",
"loader.conf")
subprocess.call(["bootctl",
"--path={!s}".format(install_efi_directory),
"install"])
2021-02-22 00:27:46 +01:00
# find all the installed kernels and generate default and fallback entries for each
# This is Arch-specific and may need adjustment for other distros
2021-02-24 16:41:42 +01:00
for root, dirs, files in os.walk(os.path.join(install_path, kernel_search_path.lstrip('/'))):
2021-02-22 00:27:46 +01:00
for file in files:
if file == source_kernel_name:
rel_root = os.path.relpath(root, install_path)
create_systemd_boot_conf(install_path,
efi_directory,
uuid,
distribution,
distribution_translated,
os.path.join(rel_root, file),
"default",
os.path.basename(root))
create_systemd_boot_conf(install_path,
efi_directory,
uuid,
distribution,
distribution_translated,
os.path.join(rel_root, file),
"fallback",
os.path.basename(root))
create_loader(loader_path, distribution_translated)
def get_grub_efi_parameters():
"""
Returns a 3-tuple of suitable parameters for GRUB EFI installation,
depending on the host machine architecture. The return is
- target name
- grub.efi name
- boot.efi name
all three are strings. May return None if there is no suitable
set for the current machine. May return unsuitable values if the
host architecture is unknown (e.g. defaults to x86_64).
"""
import platform
efi_bitness = efi_word_size()
cpu_type = platform.machine()
if efi_bitness == "32":
# Assume all 32-bitters are legacy x86
return ("i386-efi", "grubia32.efi", "bootia32.efi")
elif efi_bitness == "64" and cpu_type == "aarch64":
return ("arm64-efi", "grubaa64.efi", "bootaa64.efi")
elif efi_bitness == "64":
# If it's not ARM, must by AMD64
return ("x86_64-efi", "grubx64.efi", "bootx64.efi")
libcalamares.utils.warning("Could not find GRUB parameters for bits {b} and cpu {c}".format(b=repr(efi_bitness), c=repr(cpu_type)))
return None
def install_grub(efi_directory, fw_type):
2017-03-24 16:39:25 +01:00
"""
Installs grub as bootloader, either in pc or efi mode.
2015-02-20 20:54:25 +01:00
:param efi_directory:
:param fw_type:
"""
if fw_type == "efi":
libcalamares.utils.debug("Bootloader: grub (efi)")
install_path = libcalamares.globalstorage.value("rootMountPoint")
install_efi_directory = install_path + efi_directory
if not os.path.isdir(install_efi_directory):
2017-03-24 16:39:25 +01:00
os.makedirs(install_efi_directory)
2015-06-14 05:08:52 +02:00
efi_bootloader_id = efi_label()
efi_target, efi_grub_file, efi_boot_file = get_grub_efi_parameters()
2017-10-24 21:32:15 +02:00
2016-05-04 13:30:54 +02:00
check_target_env_call([libcalamares.job.configuration["grubInstall"],
2017-10-24 21:32:15 +02:00
"--target=" + efi_target,
2017-03-24 16:39:25 +01:00
"--efi-directory=" + efi_directory,
"--bootloader-id=" + efi_bootloader_id,
2016-05-04 13:30:54 +02:00
"--force"])
# VFAT is weird, see issue CAL-385
2017-03-24 16:39:25 +01:00
install_efi_directory_firmware = (vfat_correct_case(
install_efi_directory,
"EFI"))
if not os.path.exists(install_efi_directory_firmware):
2017-03-24 16:39:25 +01:00
os.makedirs(install_efi_directory_firmware)
# there might be several values for the boot directory
# most usual they are boot, Boot, BOOT
2017-03-24 16:39:25 +01:00
install_efi_boot_directory = (vfat_correct_case(
install_efi_directory_firmware,
"boot"))
if not os.path.exists(install_efi_boot_directory):
2017-03-24 16:39:25 +01:00
os.makedirs(install_efi_boot_directory)
# Workaround for some UEFI firmwares
FALLBACK = "installEFIFallback"
libcalamares.utils.debug("UEFI Fallback: " + str(libcalamares.job.configuration.get(FALLBACK, "<unset>")))
if libcalamares.job.configuration.get(FALLBACK, True):
libcalamares.utils.debug(" .. installing '{!s}' fallback firmware".format(efi_boot_file))
efi_file_source = os.path.join(install_efi_directory_firmware,
efi_bootloader_id,
efi_grub_file)
efi_file_target = os.path.join(install_efi_boot_directory,
efi_boot_file)
shutil.copy2(efi_file_source, efi_file_target)
else:
libcalamares.utils.debug("Bootloader: grub (bios)")
2017-01-18 19:10:09 +01:00
if libcalamares.globalstorage.value("bootLoader") is None:
return
boot_loader = libcalamares.globalstorage.value("bootLoader")
2017-01-18 19:10:09 +01:00
if boot_loader["installPath"] is None:
return
2016-05-04 13:30:54 +02:00
check_target_env_call([libcalamares.job.configuration["grubInstall"],
"--target=i386-pc",
"--recheck",
"--force",
boot_loader["installPath"]])
# The input file /etc/default/grub should already be filled out by the
# grubcfg job module.
2016-05-04 13:30:54 +02:00
check_target_env_call([libcalamares.job.configuration["grubMkconfig"],
2017-03-24 16:39:25 +01:00
"-o", libcalamares.job.configuration["grubCfg"]])
def install_secureboot(efi_directory):
"""
Installs the secureboot shim in the system by calling efibootmgr.
"""
efi_bootloader_id = efi_label()
install_path = libcalamares.globalstorage.value("rootMountPoint")
install_efi_directory = install_path + efi_directory
if efi_word_size() == "64":
install_efi_bin = "shimx64.efi"
elif efi_word_size() == "32":
install_efi_bin = "shimia32.efi"
# Copied, roughly, from openSUSE's install script,
# and pythonified. *disk* is something like /dev/sda,
# while *drive* may return "(disk/dev/sda,gpt1)" ..
# we're interested in the numbers in the second part
# of that tuple.
efi_drive = subprocess.check_output([
libcalamares.job.configuration["grubProbe"],
"-t", "drive", "--device-map=", install_efi_directory]).decode("ascii")
efi_disk = subprocess.check_output([
libcalamares.job.configuration["grubProbe"],
"-t", "disk", "--device-map=", install_efi_directory]).decode("ascii")
efi_drive_partition = efi_drive.replace("(","").replace(")","").split(",")[1]
# Get the first run of digits from the partition
efi_partition_number = None
c = 0
start = None
while c < len(efi_drive_partition):
if efi_drive_partition[c].isdigit() and start is None:
start = c
if not efi_drive_partition[c].isdigit() and start is not None:
efi_partition_number = efi_drive_partition[start:c]
break
c += 1
if efi_partition_number is None:
raise ValueError("No partition number found for %s" % install_efi_directory)
subprocess.call([
libcalamares.job.configuration["efiBootMgr"],
"-c",
"-w",
"-L", efi_bootloader_id,
"-d", efi_disk,
"-p", efi_partition_number,
"-l", install_efi_directory + "/" + install_efi_bin])
efi_boot_next()
# The input file /etc/default/grub should already be filled out by the
# grubcfg job module.
check_target_env_call([libcalamares.job.configuration["grubMkconfig"],
"-o", os.path.join(efi_directory, "EFI",
efi_bootloader_id, "grub.cfg")])
def vfat_correct_case(parent, name):
for candidate in os.listdir(parent):
if name.lower() == candidate.lower():
return os.path.join(parent, candidate)
return os.path.join(parent, name)
def prepare_bootloader(fw_type):
2017-03-24 16:39:25 +01:00
"""
Prepares bootloader.
Based on value 'efi_boot_loader', it either calls systemd-boot
or grub to be installed.
2015-02-20 20:54:25 +01:00
:param fw_type:
:return:
"""
efi_boot_loader = libcalamares.job.configuration["efiBootLoader"]
efi_directory = libcalamares.globalstorage.value("efiSystemPartition")
2015-06-14 05:08:52 +02:00
if efi_boot_loader == "systemd-boot" and fw_type == "efi":
2015-06-11 00:32:34 +02:00
install_systemd_boot(efi_directory)
elif efi_boot_loader == "sb-shim" and fw_type == "efi":
install_secureboot(efi_directory)
elif efi_boot_loader == "grub" or fw_type != "efi":
2015-02-20 01:00:31 +01:00
install_grub(efi_directory, fw_type)
else:
libcalamares.utils.debug( "WARNING: the combination of "
"boot-loader '{!s}' and firmware '{!s}' "
"is not supported.".format(efi_boot_loader, fw_type) )
2014-10-16 21:08:18 +02:00
def run():
2017-03-24 16:39:25 +01:00
"""
Starts procedure and passes 'fw_type' to other routine.
2015-02-20 20:54:25 +01:00
:return:
"""
fw_type = libcalamares.globalstorage.value("firmwareType")
2017-01-17 18:13:51 +01:00
if (libcalamares.globalstorage.value("bootLoader") is None and fw_type != "efi"):
libcalamares.utils.warning( "Non-EFI system, and no bootloader is set." )
2017-01-17 18:13:51 +01:00
return None
partitions = libcalamares.globalstorage.value("partitions")
if fw_type == "efi":
2019-04-19 16:43:07 +02:00
efi_system_partition = libcalamares.globalstorage.value("efiSystemPartition")
esp_found = [ p for p in partitions if p["mountPoint"] == efi_system_partition ]
if not esp_found:
libcalamares.utils.warning( "EFI system, but nothing mounted on {!s}".format(efi_system_partition) )
return None
try:
prepare_bootloader(fw_type)
except subprocess.CalledProcessError as e:
libcalamares.utils.warning(str(e))
libcalamares.utils.debug("stdout:" + str(e.stdout))
libcalamares.utils.debug("stderr:" + str(e.stderr))
return (_("Bootloader installation error"),
_("The bootloader could not be installed. The installation command <pre>{!s}</pre> returned error code {!s}.")
.format(e.cmd, e.returncode))
2015-06-14 05:08:52 +02:00
return None