Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

plugins: kernel: split kernel plugin into kernel and initrd plugins #4302

Open
wants to merge 24 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
99ee260
plugins: split kernel plugin into kernel and initrd plugins
kubiko Aug 2, 2023
480291f
tests: fix kernel, initrd tests after kernel plugin split
kubiko Jul 31, 2023
3b6c30d
tests: fix legacy(v2) kernel, initrd tests after kernel plugin split
kubiko Aug 1, 2023
f91d378
plugins: kernel: remove unused kernel-compiler-paths option
kubiko Aug 2, 2023
ec1a143
plugins: kernel: use new CRAFT_ARCH_TRIPLET_BUILD_FOR variable
kubiko Oct 26, 2023
2ab8709
plugins: initrd: fix efi image build
kubiko Nov 8, 2023
68455aa
plugins: inird: download debs to source dir
kubiko Nov 8, 2023
f044f80
plugins: kernel: removing obsolete plugin options
kubiko Nov 9, 2023
a19f638
plugins: initrd: auto remove initrd.img if building kernel.efi
kubiko Nov 9, 2023
0584406
plugins: kernel: fix kernel build corner case error
kubiko Nov 15, 2023
cc09e45
plugins: initrd: if there are no modules defined, use default set def…
kubiko Nov 21, 2023
56f1322
plugins: kernel: update build dependencies
kubiko Jul 18, 2024
f8d8b10
plugins: kernel: fix and add more tests for kdefconfig case
kubiko Nov 22, 2023
f32a7f4
plugins: initrd: remove redundant kernel-image option
kubiko Jan 31, 2024
565bc45
plugins: kernel: only install kernel.img as kernel binary
kubiko Jan 31, 2024
7f7213d
plugins: kernel: fix various pytests
kubiko Jan 31, 2024
0ef81d8
plugins: initrd: fix various pytests
kubiko Apr 22, 2024
1d2483e
plugins: kernel: set KERNELRELEASE name if building with ubuntu flavour
kubiko Apr 19, 2024
8ce878b
plugins: initrd: rework of initrd pluging with chroot based build env
kubiko Jul 17, 2024
c1d2d47
plugins: initrd: fix tests after rework to chroot build env
kubiko Jul 24, 2024
9ca78ac
plugins: kernel: refactor kernel plugin for better readability
kubiko Aug 5, 2024
7ad6103
plugins: kernel: fix test after readability refactor
kubiko Aug 15, 2024
9eb551f
plugins: initrd: error out with message if stage contains more than o…
kubiko Nov 19, 2024
6ea70b7
plugins: initrd: do not copy os-release to the chroot
kubiko Jan 16, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions snapcraft/parts/plugins/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@
from .colcon_plugin import ColconPlugin
from .conda_plugin import CondaPlugin
from .flutter_plugin import FlutterPlugin
from .initrd_plugin import InitrdPlugin
from .kernel_plugin import KernelPlugin
from .matter_sdk_plugin import MatterSdkPlugin
from .poetry_plugin import PoetryPlugin
Expand All @@ -30,6 +31,7 @@
"ColconPlugin",
"CondaPlugin",
"FlutterPlugin",
"InitrdPlugin",
"MatterSdkPlugin",
"KernelPlugin",
"PoetryPlugin",
Expand Down
236 changes: 236 additions & 0 deletions snapcraft/parts/plugins/initrd_plugin.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,236 @@
# -*- Mode:Python; indent-tabs-mode:nil; tab-width:4 -*-
# pylint: disable=line-too-long,too-many-lines,attribute-defined-outside-init
#
# Copyright 2020-2022,2024 Canonical Ltd.
#
# This program is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License version 3 as
# published by the Free Software Foundation.
#
# This program 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 this program. If not, see <http://www.gnu.org/licenses/>.

"""The initrd plugin for building kernel snaps.

The following initramfs-specific options are provided by this plugin:
- initrd-build-efi-image
Optional, true if we want to create an EFI image.
Default: false

- initrd-efi-image-key
Optional string; default: snake oil key (/usr/lib/ubuntu-core-initramfs/snakeoil/PkKek-1-snakeoil.key)
Key to be used to create efi image
Requires initrd-build-efi-image to be true.

- initrd-efi-image-cert
Optional string; default: snake oil certificate.
Certificate to be used to create efi image.
Requires initrd-build-efi-image to be true.

- initrd-modules:
(array of string; default: none)
list of modules to include in initrd.
Note that kernel snaps do not provide the core boot logic which
comes from the Ubuntu Core base snap. Include all modules you need
for mounting the rootfs here. If installed module(s) have any
dependencies, they are automatically installed.

- initrd-configured-modules:
(array of string; default: none)
list of modules to be added to the initrd
/lib/modules-load.d/ubuntu-core-initramfs.conf config
to be automatically loaded.
Configured modules are automatically added to initrd-modules.
If module in question is not supported by the kernel, it is ignored.

- initrd-firmware:
(array of string; default: none)
list of firmware files to be included in the initrd; these need to be
relative paths to stage directory.
<stage/part install dir>/firmware/* -> initrd:/lib/firmware/*

- initrd-compression:
(string; default: as defined in ubuntu-core-initrd(zstd)
initrd compression to use; the only supported values now are
'lz4', 'xz', 'gz', 'zstd'.

- initrd-compression-options:
Optional list of parameters to be passed to compressor used for initrd
(array of string): defaults are
gz: -7
lz4: -9 -l
xz: -7
zstd: -1 -T0

- initrd-overlay
(string; default: none)
Optional overlay to be applied to built initrd.
This option is designed to provide easy way to apply initrd overlay for
cases modifies initrd scripts for pre uc20 initrds.
Value is relative path, in stage directory. and related part needs to be
built before initrd part. During build it will be expanded to
${CRAFT_STAGE}/{initrd-overlay}
Default: none

- initrd-addons
(array of string; default: none)
Optional list of files to be added to the initrd.
Function is similar to initrd-overlay, only it works on per file
selection without a need to have overlay in dedicated directory.
This option is designed to provide easy way to add additional content
to initrd for cases like full disk encryption support, when device
specific hook needs to be added to the initrd.
Values are relative path from stage directory, so related part(s)
need to be built before kernel part.
During build it will be expanded to ${CRAFT_STAGE}/{initrd-addon}
Default: none

This plugin support cross compilation, for which plugin expects
the build-environment is comfigured accordingly and has foreign architectures
setup accordingly.
"""

import logging
import os
from typing import Literal, cast

import pydantic
from craft_parts import infos, plugins
from overrides import overrides
from typing_extensions import Self

from snapcraft_legacy.plugins.v2 import _initrd_build, _kernel_build

logger = logging.getLogger(__name__)


class InitrdPluginProperties(plugins.PluginProperties, frozen=True):
"""The part properties used by the Initrd plugin."""

plugin: Literal["initrd"] = "initrd"

initrd_build_efi_image: bool = False
initrd_efi_image_key: str | None = None
initrd_efi_image_cert: str | None = None
initrd_modules: list[str] | None = None
initrd_configured_modules: list[str] | None = None
initrd_firmware: list[str] | None = None
initrd_compression: str | None = None
initrd_compression_options: list[str] | None = None
initrd_overlay: str | None = None
initrd_addons: list[str] | None = None

# part properties required by the plugin
@pydantic.model_validator(mode="after")
def validate_plugin_options(self) -> Self:
"""Validate use of initrd-compression-options."""
# If initrd-compression-options is defined, so has to be initrd-compression.
if self.initrd_compression_options and not self.initrd_compression:
raise ValueError(

Check warning on line 134 in snapcraft/parts/plugins/initrd_plugin.py

View check run for this annotation

Codecov / codecov/patch

snapcraft/parts/plugins/initrd_plugin.py#L134

Added line #L134 was not covered by tests
"initrd-compression-options requires also initrd-compression to be defined."
)

# validate if initrd-efi-image-key or initrd-efi-image-cert is set
# initrd-build-efi-image is also set
if (self.initrd_efi_image_key or self.initrd_efi_image_cert) and not (
self.initrd_build_efi_image
and self.initrd_efi_image_key
and self.initrd_efi_image_cert
):
raise ValueError(
"initrd-efi-image-key and initrd-efi-image-cert must be both set if any is set"
"used when initrd-build-efi-image is enabled."
)
return self


class InitrdPlugin(plugins.Plugin):
"""Plugin for the initrd snap build."""

properties_class = InitrdPluginProperties

def __init__(
self, *, properties: plugins.PluginProperties, part_info: infos.PartInfo
) -> None:
super().__init__(properties=properties, part_info=part_info)
self.options = cast(InitrdPluginProperties, self._options)
self._target_arch = self._part_info.target_arch
target_arch = self._part_info.target_arch
self._deb_arch = _kernel_build.get_deb_architecture(target_arch)
# check if we are cross building
self._cross_building = False
if (
self._part_info.project_info.host_arch
!= self._part_info.project_info.target_arch
):
self._cross_building = True
base = self._part_info.base
self._ubuntu_series = "noble"
if base == "core22":
self._ubuntu_series = "jammy"

@overrides
def get_build_snaps(self) -> set[str]: # pylint: disable=missing-function-docstring
return set()

@overrides
def get_build_packages(
self,
) -> set[str]: # pylint: disable=missing-function-docstring
build_packages = {
"curl",
"dracut-core",
"fakechroot",
"fakeroot",
}
# if runnning as non-root and cross-building
# we need libfake{ch}root for the target arch
if self._cross_building and (os.getuid() != 0):
build_packages |= {
f"libfakechroot:{self._target_arch}",
f"libfakeroot:{self._target_arch}",
}
return build_packages

@overrides
def get_build_environment(
self,
) -> dict[str, str]: # pylint: disable=missing-function-docstring
return {
"UC_INITRD_ROOT_NAME": "uc-initramfs-build-root",
"UC_INITRD_ROOT": "${CRAFT_PART_SRC}/${UC_INITRD_ROOT_NAME}",
"KERNEL_MODULES": "${CRAFT_STAGE}/modules",
"KERNEL_FIRMWARE": "${CRAFT_STAGE}/firmware",
"UBUNTU_SERIES": self._ubuntu_series,
"UBUNTU_CORE_BASE": self._part_info.base,
}

@overrides
def get_build_commands(
self,
) -> list[str]: # pylint: disable=missing-function-docstring
logger.info("Getting build commands...")
return _initrd_build.get_build_commands(
initrd_modules=self.options.initrd_modules,
initrd_configured_modules=self.options.initrd_configured_modules,
initrd_compression=self.options.initrd_compression,
initrd_compression_options=self.options.initrd_compression_options,
initrd_firmware=self.options.initrd_firmware,
initrd_addons=self.options.initrd_addons,
initrd_overlay=self.options.initrd_overlay,
initrd_ko_use_workaround=False,
initrd_default_compression="zstd -1 -T0",
build_efi_image=self.options.initrd_build_efi_image,
efi_image_key=self.options.initrd_efi_image_key,
efi_image_cert=self.options.initrd_efi_image_cert,
)

@classmethod
def get_out_of_source_build(cls) -> bool:
"""Return whether the plugin performs out-of-source-tree builds."""
return True
Loading
Loading