calamares/src/modules/bootloader/main.py

247 lines
9.1 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 - <http://github.com/calamares> ===
#
# Copyright 2014, Aurélien Gâteau <agateau@kde.org>
# Copyright 2014, Anke Boersma <demm@kaosx.us>
# Copyright 2014, Daniel Hillenbrand <codeworkx@bbqlinux.org>
# Copyright 2014, Benjamin Vaudour <benjamin.vaudour@yahoo.fr>
# Copyright 2014, Kevin Kofler <kevin.kofler@chello.at>
# Copyright 2015, Philip Mueller <philm@manjaro.org>
#
# Calamares is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# Calamares is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with Calamares. If not, see <http://www.gnu.org/licenses/>.
import libcalamares
import os
import subprocess
from libcalamares.utils import check_chroot_call
2014-10-16 21:08:18 +02:00
def get_uuid():
2015-02-21 10:09:01 +01:00
""" Checks and passes 'uuid' to other routine.
2015-02-20 20:54:25 +01:00
:return:
"""
root_mount_point = libcalamares.globalstorage.value("rootMountPoint")
print("Root mount point: \"{!s}\"".format(root_mount_point))
partitions = libcalamares.globalstorage.value("partitions")
print("Partitions: \"{!s}\"".format(partitions))
2015-06-14 05:08:52 +02:00
for partition in partitions:
if partition["mountPoint"] == "/":
print("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():
2015-02-21 10:09:01 +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"]
def get_kernel_line(kernel_type):
2015-02-21 10:09:01 +01:00
""" Passes 'kernel_line' to other routine based on configuration file.
2015-02-20 20:54:25 +01:00
:param kernel_type:
:return:
"""
if kernel_type == "fallback":
if "fallbackKernelLine" in libcalamares.job.configuration:
return libcalamares.job.configuration["fallbackKernelLine"]
else:
return " (fallback)"
else:
if "kernelLine" in libcalamares.job.configuration:
return libcalamares.job.configuration["kernelLine"]
else:
return ""
2015-02-18 16:03:57 +01:00
2015-06-11 00:32:34 +02:00
def create_systemd_boot_conf(uuid, conf_path, kernel_line):
""" Creates systemd-boot configuration files based on given parameters.
2015-02-20 20:54:25 +01:00
:param uuid:
:param conf_path:
:param kernel_line:
"""
distribution = get_bootloader_entry_name()
kernel = libcalamares.job.configuration["kernel"]
img = libcalamares.job.configuration["img"]
partitions = libcalamares.globalstorage.value("partitions")
2014-12-31 19:33:07 +01:00
swap = ""
2015-06-14 05:08:52 +02:00
for partition in partitions:
if partition["fs"] == "linuxswap":
swap = partition["uuid"]
2015-02-15 00:03:50 +01:00
lines = [
2015-02-19 20:24:07 +01:00
'## This is just an example config file.\n',
'## Please edit the paths and kernel parameters according to your system.\n',
'\n',
2015-06-14 05:08:52 +02:00
"title {!s}{!s}\n".format(distribution, kernel_line),
"linux {!s}\n".format(kernel),
"initrd {!s}\n".format(img),
"options root=UUID={!s} quiet resume=UUID={!s} rw\n".format(uuid, swap),
]
2014-10-16 21:08:18 +02:00
with open(conf_path, 'w') as f:
for l in lines:
f.write(l)
2014-10-16 21:08:18 +02:00
def create_loader(loader_path):
2015-02-21 10:09:01 +01:00
""" Writes configuration for loader.
2015-02-20 20:54:25 +01:00
:param loader_path:
"""
distribution = get_bootloader_entry_name()
timeout = libcalamares.job.configuration["timeout"]
file_name_sanitizer = str.maketrans(" /", "_-")
distribution_translated = distribution.translate(file_name_sanitizer)
lines = [
"timeout {!s}\n".format(timeout),
"default {!s}\n".format(distribution_translated),
]
2014-10-16 21:08:18 +02:00
with open(loader_path, 'w') as f:
for l in lines:
f.write(l)
2014-10-16 21:08:18 +02:00
2015-06-11 00:32:34 +02:00
def install_systemd_boot(efi_directory):
""" Installs systemd-boot as bootloader for EFI setups.
2015-02-20 20:54:25 +01:00
:param efi_directory:
"""
2015-06-11 00:32:34 +02:00
print("Bootloader: systemd-boot")
install_path = libcalamares.globalstorage.value("rootMountPoint")
install_efi_directory = install_path + efi_directory
fallback_kernel_line = libcalamares.job.configuration["fallbackKernelLine"]
uuid = get_uuid()
distribution = get_bootloader_entry_name()
file_name_sanitizer = str.maketrans(" /", "_-")
distribution_translated = distribution.translate(file_name_sanitizer)
2015-06-14 05:08:52 +02:00
conf_path = os.path.join(install_efi_directory, "loader", "entries",
"{!s}.conf".format(distribution_translated))
fallback_path = os.path.join(install_efi_directory, "loader", "entries",
"{!s}-fallback.conf".format(distribution_translated))
loader_path = os.path.join(install_efi_directory, "loader", "loader.conf")
2015-06-11 00:32:34 +02:00
subprocess.call(["bootctl", "--path={!s}".format(install_efi_directory), "install"])
kernel_line = get_kernel_line("default")
print("Configure: \"{!s}\"".format(kernel_line))
2015-06-11 00:32:34 +02:00
create_systemd_boot_conf(uuid, conf_path, kernel_line)
kernel_line = get_kernel_line("fallback")
print("Configure: \"{!s}\"".format(kernel_line))
2015-06-11 00:32:34 +02:00
create_systemd_boot_conf(uuid, fallback_path, kernel_line)
create_loader(loader_path)
def install_grub(efi_directory, fw_type):
2015-02-21 10:09:01 +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":
print("Bootloader: grub (efi)")
efi_directory_firmware = efi_directory + "/EFI"
check_chroot_call(["mkdir", "-p", "{!s}".format(efi_directory)])
2015-06-14 05:08:52 +02:00
if "efiBootloaderId" in libcalamares.job.configuration:
efi_bootloader_id = libcalamares.job.configuration["efiBootloaderId"]
else:
branding = libcalamares.globalstorage.value("branding")
distribution = branding["bootloaderEntryName"]
file_name_sanitizer = str.maketrans(" /", "_-")
efi_bootloader_id = distribution.translate(file_name_sanitizer)
2015-06-14 05:08:52 +02:00
check_chroot_call([libcalamares.job.configuration["grubInstall"], "--target=x86_64-efi",
"--efi-directory={!s}".format(efi_directory),
"--bootloader-id={!s}".format(efi_bootloader_id),
"--force"])
# Workaround for some UEFI firmwares
check_chroot_call(["mkdir", "-p", "{!s}/boot".format(efi_directory_firmware)])
2015-06-14 05:08:52 +02:00
check_chroot_call(["cp", "{!s}/{!s}/grubx64.efi".format(efi_directory_firmware, efi_bootloader_id),
"{!s}/boot/bootx64.efi".format(efi_directory_firmware)])
else:
print("Bootloader: grub (bios)")
boot_loader = libcalamares.globalstorage.value("bootLoader")
2015-06-14 05:08:52 +02:00
check_chroot_call([libcalamares.job.configuration["grubInstall"], "--target=i386-pc",
"--recheck", "--force", boot_loader["installPath"]])
check_chroot_call([libcalamares.job.configuration["grubMkconfig"], "-o",
libcalamares.job.configuration["grubCfg"]])
def prepare_bootloader(fw_type):
2015-02-21 10:09:01 +01:00
""" Prepares bootloader and set proper flags to EFI boot partition (esp,boot).
2015-06-11 00:32:34 +02:00
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 fw_type == "efi":
partitions = libcalamares.globalstorage.value("partitions")
boot_p = ""
device = ""
2015-06-14 05:08:52 +02:00
for partition in partitions:
if partition["mountPoint"] == efi_directory:
boot_device = partition["device"]
boot_p = boot_device[-1:]
device = boot_device[:-1]
2015-06-14 05:08:52 +02:00
2015-06-14 14:22:35 +02:00
if not boot_p or not device:
return ("EFI directory \"{!s}\" not found!",
"Boot partition: \"{!s}\"",
2015-06-14 05:08:52 +02:00
"Boot device: \"{!s}\"".format(efi_directory, boot_p, device))
else:
print("EFI directory: \"{!s}\"".format(efi_directory))
print("Boot partition: \"{!s}\"".format(boot_p))
print("Boot device: \"{!s}\"".format(device))
print("Set 'EF00' flag")
2015-06-14 05:08:52 +02:00
subprocess.call(["sgdisk", "--typecode={!s}:EF00".format(boot_p), "{!s}".format(device)])
if efi_boot_loader == "systemd-boot" and fw_type == "efi":
2015-06-11 00:32:34 +02:00
install_systemd_boot(efi_directory)
else:
2015-02-20 01:00:31 +01:00
install_grub(efi_directory, fw_type)
2014-10-16 21:08:18 +02:00
def run():
2015-02-21 10:09:01 +01:00
""" Starts procedure and passes 'fw_type' to other routine.
2015-02-20 20:54:25 +01:00
:return:
"""
if libcalamares.globalstorage.value("bootLoader") is None:
return None
fw_type = libcalamares.globalstorage.value("firmwareType")
prepare_bootloader(fw_type)
2015-06-14 05:08:52 +02:00
return None