mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2026-06-21 15:43:21 +02:00
9f2aee8f7d
Add distributed ThinLTO build support for the Linux kernel.
This new mode offers several advantages: (1) Increased
flexibility in handling user-specified build options.
(2) Improved user-friendliness for developers. (3) Greater
convenience for integrating with objtool and livepatch.
Note that "distributed" in this context refers to a term
that differentiates in-process ThinLTO builds by invoking
backend compilation through the linker, not necessarily
building in distributed environments.
Distributed ThinLTO is enabled via the
`CONFIG_LTO_CLANG_THIN_DIST` Kconfig option. For example:
> make LLVM=1 defconfig
> scripts/config -e LTO_CLANG_THIN_DIST
> make LLVM=1 oldconfig
> make LLVM=1 vmlinux -j <..>
The build flow proceeds in four stages:
1. Perform FE compilation, mirroring the in-process ThinLTO mode.
2. Thin-link the generated IR files and object files.
3. Find all IR files and perform BE compilation, using the flags
stored in the .*.o.cmd files.
4. Link the BE results to generate the final vmlinux.o.
NOTE: This patch currently implements the build for the main kernel
image (vmlinux) only. Kernel module support is planned for a
subsequent patch.
Tested on the following arch: x86, arm64, loongarch, and
riscv.
The earlier implementation details can be found here:
https://discourse.llvm.org/t/rfc-distributed-thinlto-build-for-kernel/85934
Signed-off-by: Rong Xu <xur@google.com>
Co-developed-by: Masahiro Yamada <masahiroy@kernel.org>
Signed-off-by: Masahiro Yamada <masahiroy@kernel.org>
Tested-by: Piotr Gorski <piotrgorski@cachyos.org>
Tested-by: Nathan Chancellor <nathan@kernel.org>
Link: https://patch.msgid.link/20260529185347.2418373-4-xur@google.com
Signed-off-by: Nathan Chancellor <nathan@kernel.org>
84 lines
2.3 KiB
Makefile
84 lines
2.3 KiB
Makefile
# SPDX-License-Identifier: GPL-2.0-only
|
|
|
|
PHONY := __default
|
|
__default: vmlinux.a
|
|
|
|
include include/config/auto.conf
|
|
include $(srctree)/scripts/Kbuild.include
|
|
include $(srctree)/scripts/Makefile.lib
|
|
|
|
# Link of built-in-fixup.a
|
|
# ---------------------------------------------------------------------------
|
|
|
|
# '$(AR) mPi' needs 'T' to workaround the bug of llvm-ar <= 14
|
|
quiet_cmd_ar_builtin_fixup = AR $@
|
|
cmd_ar_builtin_fixup = \
|
|
rm -f $@; \
|
|
$(AR) cDPrST $@ $(KBUILD_VMLINUX_OBJS); \
|
|
$(AR) mPiT $$($(AR) t $@ | sed -n 1p) $@ $$($(AR) t $@ | grep -F -f $(srctree)/scripts/head-object-list.txt)
|
|
|
|
targets += built-in-fixup.a
|
|
built-in-fixup.a: $(KBUILD_VMLINUX_OBJS) scripts/head-object-list.txt FORCE
|
|
$(call if_changed,ar_builtin_fixup)
|
|
|
|
ifdef CONFIG_LTO_CLANG_THIN_DIST
|
|
|
|
quiet_cmd_builtin.order = GEN $@
|
|
cmd_builtin.order = $(AR) t $< > $@
|
|
|
|
targets += builtin.order
|
|
builtin.order: built-in-fixup.a FORCE
|
|
$(call if_changed,builtin.order)
|
|
|
|
quiet_cmd_ld_thinlto_index = LD $@
|
|
cmd_ld_thinlto_index = \
|
|
$(LD) $(KBUILD_LDFLAGS) -r --thinlto-index-only=$@ @$<
|
|
|
|
targets += vmlinux.thinlto-index
|
|
vmlinux.thinlto-index: builtin.order FORCE
|
|
$(call if_changed,ld_thinlto_index)
|
|
|
|
quiet_cmd_ar_vmlinux.a = GEN $@
|
|
cmd_ar_vmlinux.a = \
|
|
rm -f $@; \
|
|
while read -r obj; do \
|
|
if grep -Fqx $${obj} $(word 2, $^); then \
|
|
echo $${obj%.o}.thinlto-native.o; \
|
|
else \
|
|
echo $${obj}; \
|
|
fi; \
|
|
done < $< | xargs $(AR) cDPrS --thin $@
|
|
|
|
targets += vmlinux.a
|
|
vmlinux.a: builtin.order vmlinux.thinlto-index FORCE
|
|
$(Q)$(MAKE) -f $(srctree)/scripts/Makefile.thinlto
|
|
$(call if_changed,ar_vmlinux.a)
|
|
|
|
else
|
|
|
|
# vmlinux.a
|
|
# ---------------------------------------------------------------------------
|
|
|
|
targets += vmlinux.a
|
|
vmlinux.a: built-in-fixup.a FORCE
|
|
$(call if_changed,copy)
|
|
|
|
endif
|
|
|
|
# Add FORCE to the prerequisites of a target to force it to be always rebuilt.
|
|
# ---------------------------------------------------------------------------
|
|
|
|
PHONY += FORCE
|
|
FORCE:
|
|
|
|
# Read all saved command lines and dependencies for the $(targets) we
|
|
# may be building above, using $(if_changed{,_dep}). As an
|
|
# optimization, we don't need to read them if the target does not
|
|
# exist, we will rebuild anyway in that case.
|
|
|
|
existing-targets := $(wildcard $(sort $(targets)))
|
|
|
|
-include $(foreach f,$(existing-targets),$(dir $(f)).$(notdir $(f)).cmd)
|
|
|
|
.PHONY: $(PHONY)
|