123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428 |
- #
- # Copyright (c) 2014, Intel Corporation.
- #
- # SPDX-License-Identifier: GPL-2.0-only
- #
- # DESCRIPTION
- # This implements the 'bootimg-efi' source plugin class for 'wic'
- #
- # AUTHORS
- # Tom Zanussi <tom.zanussi (at] linux.intel.com>
- #
- import logging
- import os
- import tempfile
- import shutil
- import re
- from glob import glob
- from wic import WicError
- from wic.engine import get_custom_config
- from wic.pluginbase import SourcePlugin
- from wic.misc import (exec_cmd, exec_native_cmd,
- get_bitbake_var, BOOTDD_EXTRA_SPACE)
- logger = logging.getLogger('wic')
- class BootimgEFIPlugin(SourcePlugin):
- """
- Create EFI boot partition.
- This plugin supports GRUB 2 and systemd-boot bootloaders.
- """
- name = 'bootimg-efi'
- @classmethod
- def _copy_additional_files(cls, hdddir, initrd, dtb):
- bootimg_dir = get_bitbake_var("DEPLOY_DIR_IMAGE")
- if not bootimg_dir:
- raise WicError("Couldn't find DEPLOY_DIR_IMAGE, exiting")
- if initrd:
- initrds = initrd.split(';')
- for rd in initrds:
- cp_cmd = "cp -v -p %s/%s %s" % (bootimg_dir, rd, hdddir)
- out = exec_cmd(cp_cmd, True)
- logger.debug("initrd files:\n%s" % (out))
- else:
- logger.debug("Ignoring missing initrd")
- if dtb:
- if ';' in dtb:
- raise WicError("Only one DTB supported, exiting")
- cp_cmd = "cp -v -p %s/%s %s" % (bootimg_dir, dtb, hdddir)
- out = exec_cmd(cp_cmd, True)
- logger.debug("dtb files:\n%s" % (out))
- @classmethod
- def do_configure_grubefi(cls, hdddir, creator, cr_workdir, source_params):
- """
- Create loader-specific (grub-efi) config
- """
- configfile = creator.ks.bootloader.configfile
- custom_cfg = None
- if configfile:
- custom_cfg = get_custom_config(configfile)
- if custom_cfg:
- # Use a custom configuration for grub
- grubefi_conf = custom_cfg
- logger.debug("Using custom configuration file "
- "%s for grub.cfg", configfile)
- else:
- raise WicError("configfile is specified but failed to "
- "get it from %s." % configfile)
- initrd = source_params.get('initrd')
- dtb = source_params.get('dtb')
- cls._copy_additional_files(hdddir, initrd, dtb)
- if not custom_cfg:
- # Create grub configuration using parameters from wks file
- bootloader = creator.ks.bootloader
- title = source_params.get('title')
- grubefi_conf = ""
- grubefi_conf += "serial --unit=0 --speed=115200 --word=8 --parity=no --stop=1\n"
- grubefi_conf += "default=boot\n"
- grubefi_conf += "timeout=%s\n" % bootloader.timeout
- grubefi_conf += "menuentry '%s'{\n" % (title if title else "boot")
- kernel = get_bitbake_var("KERNEL_IMAGETYPE")
- if get_bitbake_var("INITRAMFS_IMAGE_BUNDLE") == "1":
- if get_bitbake_var("INITRAMFS_IMAGE"):
- kernel = "%s-%s.bin" % \
- (get_bitbake_var("KERNEL_IMAGETYPE"), get_bitbake_var("INITRAMFS_LINK_NAME"))
- label = source_params.get('label')
- label_conf = "root=%s" % creator.rootdev
- if label:
- label_conf = "LABEL=%s" % label
- grubefi_conf += "linux /%s %s rootwait %s\n" \
- % (kernel, label_conf, bootloader.append)
- if initrd:
- initrds = initrd.split(';')
- grubefi_conf += "initrd"
- for rd in initrds:
- grubefi_conf += " /%s" % rd
- grubefi_conf += "\n"
- if dtb:
- grubefi_conf += "devicetree /%s\n" % dtb
- grubefi_conf += "}\n"
- logger.debug("Writing grubefi config %s/hdd/boot/EFI/BOOT/grub.cfg",
- cr_workdir)
- cfg = open("%s/hdd/boot/EFI/BOOT/grub.cfg" % cr_workdir, "w")
- cfg.write(grubefi_conf)
- cfg.close()
- @classmethod
- def do_configure_systemdboot(cls, hdddir, creator, cr_workdir, source_params):
- """
- Create loader-specific systemd-boot/gummiboot config. Unified Kernel Image (uki)
- support is done in image recipe with uki.bbclass and only systemd-boot loader config
- and ESP partition structure is created here.
- """
- # detect uki.bbclass usage
- image_classes = get_bitbake_var("IMAGE_CLASSES").split()
- unified_image = False
- if "uki" in image_classes:
- unified_image = True
- install_cmd = "install -d %s/loader" % hdddir
- exec_cmd(install_cmd)
- install_cmd = "install -d %s/loader/entries" % hdddir
- exec_cmd(install_cmd)
- bootloader = creator.ks.bootloader
- loader_conf = ""
- # 5 seconds is a sensible default timeout
- loader_conf += "timeout %d\n" % (bootloader.timeout or 5)
- logger.debug("Writing systemd-boot config "
- "%s/hdd/boot/loader/loader.conf", cr_workdir)
- cfg = open("%s/hdd/boot/loader/loader.conf" % cr_workdir, "w")
- cfg.write(loader_conf)
- logger.debug("loader.conf:\n%s" % (loader_conf))
- cfg.close()
- initrd = source_params.get('initrd')
- dtb = source_params.get('dtb')
- if not unified_image:
- cls._copy_additional_files(hdddir, initrd, dtb)
- configfile = creator.ks.bootloader.configfile
- custom_cfg = None
- boot_conf = ""
- if configfile:
- custom_cfg = get_custom_config(configfile)
- if custom_cfg:
- # Use a custom configuration for systemd-boot
- boot_conf = custom_cfg
- logger.debug("Using custom configuration file "
- "%s for systemd-boots's boot.conf", configfile)
- else:
- raise WicError("configfile is specified but failed to "
- "get it from %s.", configfile)
- else:
- # Create systemd-boot configuration using parameters from wks file
- kernel = get_bitbake_var("KERNEL_IMAGETYPE")
- if get_bitbake_var("INITRAMFS_IMAGE_BUNDLE") == "1":
- if get_bitbake_var("INITRAMFS_IMAGE"):
- kernel = "%s-%s.bin" % \
- (get_bitbake_var("KERNEL_IMAGETYPE"), get_bitbake_var("INITRAMFS_LINK_NAME"))
- title = source_params.get('title')
- boot_conf += "title %s\n" % (title if title else "boot")
- boot_conf += "linux /%s\n" % kernel
- label = source_params.get('label')
- label_conf = "LABEL=Boot root=%s" % creator.rootdev
- if label:
- label_conf = "LABEL=%s" % label
- boot_conf += "options %s %s\n" % \
- (label_conf, bootloader.append)
- if initrd:
- initrds = initrd.split(';')
- for rd in initrds:
- boot_conf += "initrd /%s\n" % rd
- if dtb:
- boot_conf += "devicetree /%s\n" % dtb
- if not unified_image:
- logger.debug("Writing systemd-boot config "
- "%s/hdd/boot/loader/entries/boot.conf", cr_workdir)
- cfg = open("%s/hdd/boot/loader/entries/boot.conf" % cr_workdir, "w")
- cfg.write(boot_conf)
- logger.debug("boot.conf:\n%s" % (boot_conf))
- cfg.close()
- @classmethod
- def do_configure_partition(cls, part, source_params, creator, cr_workdir,
- oe_builddir, bootimg_dir, kernel_dir,
- native_sysroot):
- """
- Called before do_prepare_partition(), creates loader-specific config
- """
- hdddir = "%s/hdd/boot" % cr_workdir
- install_cmd = "install -d %s/EFI/BOOT" % hdddir
- exec_cmd(install_cmd)
- try:
- if source_params['loader'] == 'grub-efi':
- cls.do_configure_grubefi(hdddir, creator, cr_workdir, source_params)
- elif source_params['loader'] == 'systemd-boot':
- cls.do_configure_systemdboot(hdddir, creator, cr_workdir, source_params)
- elif source_params['loader'] == 'uefi-kernel':
- pass
- else:
- raise WicError("unrecognized bootimg-efi loader: %s" % source_params['loader'])
- except KeyError:
- raise WicError("bootimg-efi requires a loader, none specified")
- if get_bitbake_var("IMAGE_EFI_BOOT_FILES") is None:
- logger.debug('No boot files defined in IMAGE_EFI_BOOT_FILES')
- else:
- boot_files = None
- for (fmt, id) in (("_uuid-%s", part.uuid), ("_label-%s", part.label), (None, None)):
- if fmt:
- var = fmt % id
- else:
- var = ""
- boot_files = get_bitbake_var("IMAGE_EFI_BOOT_FILES" + var)
- if boot_files:
- break
- logger.debug('Boot files: %s', boot_files)
- # list of tuples (src_name, dst_name)
- deploy_files = []
- for src_entry in re.findall(r'[\w;\-\.\+/\*]+', boot_files):
- if ';' in src_entry:
- dst_entry = tuple(src_entry.split(';'))
- if not dst_entry[0] or not dst_entry[1]:
- raise WicError('Malformed boot file entry: %s' % src_entry)
- else:
- dst_entry = (src_entry, src_entry)
- logger.debug('Destination entry: %r', dst_entry)
- deploy_files.append(dst_entry)
- cls.install_task = [];
- for deploy_entry in deploy_files:
- src, dst = deploy_entry
- if '*' in src:
- # by default install files under their basename
- entry_name_fn = os.path.basename
- if dst != src:
- # unless a target name was given, then treat name
- # as a directory and append a basename
- entry_name_fn = lambda name: \
- os.path.join(dst,
- os.path.basename(name))
- srcs = glob(os.path.join(kernel_dir, src))
- logger.debug('Globbed sources: %s', ', '.join(srcs))
- for entry in srcs:
- src = os.path.relpath(entry, kernel_dir)
- entry_dst_name = entry_name_fn(entry)
- cls.install_task.append((src, entry_dst_name))
- else:
- cls.install_task.append((src, dst))
- @classmethod
- def do_prepare_partition(cls, part, source_params, creator, cr_workdir,
- oe_builddir, bootimg_dir, kernel_dir,
- rootfs_dir, native_sysroot):
- """
- Called to do the actual content population for a partition i.e. it
- 'prepares' the partition to be incorporated into the image.
- In this case, prepare content for an EFI (grub) boot partition.
- """
- if not kernel_dir:
- kernel_dir = get_bitbake_var("DEPLOY_DIR_IMAGE")
- if not kernel_dir:
- raise WicError("Couldn't find DEPLOY_DIR_IMAGE, exiting")
- staging_kernel_dir = kernel_dir
- hdddir = "%s/hdd/boot" % cr_workdir
- kernel = get_bitbake_var("KERNEL_IMAGETYPE")
- if get_bitbake_var("INITRAMFS_IMAGE_BUNDLE") == "1":
- if get_bitbake_var("INITRAMFS_IMAGE"):
- kernel = "%s-%s.bin" % \
- (get_bitbake_var("KERNEL_IMAGETYPE"), get_bitbake_var("INITRAMFS_LINK_NAME"))
- if source_params.get('create-unified-kernel-image') == "true":
- raise WicError("create-unified-kernel-image is no longer supported. Please use uki.bbclass.")
- if source_params.get('install-kernel-into-boot-dir') != 'false':
- install_cmd = "install -v -p -m 0644 %s/%s %s/%s" % \
- (staging_kernel_dir, kernel, hdddir, kernel)
- out = exec_cmd(install_cmd)
- logger.debug("Installed kernel files:\n%s" % out)
- if get_bitbake_var("IMAGE_EFI_BOOT_FILES"):
- for src_path, dst_path in cls.install_task:
- install_cmd = "install -v -p -m 0644 -D %s %s" \
- % (os.path.join(kernel_dir, src_path),
- os.path.join(hdddir, dst_path))
- out = exec_cmd(install_cmd)
- logger.debug("Installed IMAGE_EFI_BOOT_FILES:\n%s" % out)
- try:
- if source_params['loader'] == 'grub-efi':
- shutil.copyfile("%s/hdd/boot/EFI/BOOT/grub.cfg" % cr_workdir,
- "%s/grub.cfg" % cr_workdir)
- for mod in [x for x in os.listdir(kernel_dir) if x.startswith("grub-efi-")]:
- cp_cmd = "cp -v -p %s/%s %s/EFI/BOOT/%s" % (kernel_dir, mod, hdddir, mod[9:])
- exec_cmd(cp_cmd, True)
- shutil.move("%s/grub.cfg" % cr_workdir,
- "%s/hdd/boot/EFI/BOOT/grub.cfg" % cr_workdir)
- elif source_params['loader'] == 'systemd-boot':
- for mod in [x for x in os.listdir(kernel_dir) if x.startswith("systemd-")]:
- cp_cmd = "cp -v -p %s/%s %s/EFI/BOOT/%s" % (kernel_dir, mod, hdddir, mod[8:])
- out = exec_cmd(cp_cmd, True)
- logger.debug("systemd-boot files:\n%s" % out)
- elif source_params['loader'] == 'uefi-kernel':
- kernel = get_bitbake_var("KERNEL_IMAGETYPE")
- if not kernel:
- raise WicError("Empty KERNEL_IMAGETYPE")
- target = get_bitbake_var("TARGET_SYS")
- if not target:
- raise WicError("Empty TARGET_SYS")
- if re.match("x86_64", target):
- kernel_efi_image = "bootx64.efi"
- elif re.match('i.86', target):
- kernel_efi_image = "bootia32.efi"
- elif re.match('aarch64', target):
- kernel_efi_image = "bootaa64.efi"
- elif re.match('arm', target):
- kernel_efi_image = "bootarm.efi"
- else:
- raise WicError("UEFI stub kernel is incompatible with target %s" % target)
- for mod in [x for x in os.listdir(kernel_dir) if x.startswith(kernel)]:
- cp_cmd = "cp -v -p %s/%s %s/EFI/BOOT/%s" % (kernel_dir, mod, hdddir, kernel_efi_image)
- out = exec_cmd(cp_cmd, True)
- logger.debug("uefi-kernel files:\n%s" % out)
- else:
- raise WicError("unrecognized bootimg-efi loader: %s" %
- source_params['loader'])
- except KeyError:
- raise WicError("bootimg-efi requires a loader, none specified")
- startup = os.path.join(kernel_dir, "startup.nsh")
- if os.path.exists(startup):
- cp_cmd = "cp -v -p %s %s/" % (startup, hdddir)
- out = exec_cmd(cp_cmd, True)
- logger.debug("startup files:\n%s" % out)
- for paths in part.include_path or []:
- for path in paths:
- cp_cmd = "cp -v -p -r %s %s/" % (path, hdddir)
- exec_cmd(cp_cmd, True)
- logger.debug("include_path files:\n%s" % out)
- du_cmd = "du -bks %s" % hdddir
- out = exec_cmd(du_cmd)
- blocks = int(out.split()[0])
- extra_blocks = part.get_extra_block_count(blocks)
- if extra_blocks < BOOTDD_EXTRA_SPACE:
- extra_blocks = BOOTDD_EXTRA_SPACE
- blocks += extra_blocks
- logger.debug("Added %d extra blocks to %s to get to %d total blocks",
- extra_blocks, part.mountpoint, blocks)
- # required for compatibility with certain devices expecting file system
- # block count to be equal to partition block count
- if blocks < part.fixed_size:
- blocks = part.fixed_size
- logger.debug("Overriding %s to %d total blocks for compatibility",
- part.mountpoint, blocks)
- # dosfs image, created by mkdosfs
- bootimg = "%s/boot.img" % cr_workdir
- label = part.label if part.label else "ESP"
- dosfs_cmd = "mkdosfs -v -n %s -i %s -C %s %d" % \
- (label, part.fsuuid, bootimg, blocks)
- exec_native_cmd(dosfs_cmd, native_sysroot)
- logger.debug("mkdosfs:\n%s" % (str(out)))
- mcopy_cmd = "mcopy -v -p -i %s -s %s/* ::/" % (bootimg, hdddir)
- out = exec_native_cmd(mcopy_cmd, native_sysroot)
- logger.debug("mcopy:\n%s" % (str(out)))
- chmod_cmd = "chmod 644 %s" % bootimg
- exec_cmd(chmod_cmd)
- du_cmd = "du -Lbks %s" % bootimg
- out = exec_cmd(du_cmd)
- bootimg_size = out.split()[0]
- part.size = int(bootimg_size)
- part.source_file = bootimg
|