| # Copyright (C) 2019 The Android Open Source Project |
| # |
| # Licensed under the Apache License, Version 2.0 (the "License"); |
| # you may not use this file except in compliance with the License. |
| # You may obtain a copy of the License at |
| # |
| # http://www.apache.org/licenses/LICENSE-2.0 |
| # |
| # Unless required by applicable law or agreed to in writing, software |
| # distributed under the License is distributed on an "AS IS" BASIS, |
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| # See the License for the specific language governing permissions and |
| # limitations under the License. |
| |
| # This is an implementation detail of Kleaf. Do not source directly as it will |
| # spoil your shell. You have been warned! If you have a good reason to source |
| # the result of this file into a shell, please let kernel-team@android.com know |
| # and we will be happy to help with your use case. |
| |
| [ -n "$_SETUP_ENV_SH_INCLUDED" ] && return || export _SETUP_ENV_SH_INCLUDED=1 |
| |
| # TODO: Use a $(gettop) style method. |
| export ROOT_DIR=$(readlink -f $PWD) |
| |
| if [ "$KLEAF_INTERNAL_NO_BUILD_CONFIG" != "1" ]; then |
| export BUILD_CONFIG=${BUILD_CONFIG:-build.config} |
| fi |
| |
| # Helper function to let build.config files add command to PRE_DEFCONFIG_CMDS, EXTRA_CMDS, etc. |
| # Usage: append_cmd PRE_DEFCONFIG_CMDS 'the_cmd' |
| function append_cmd() { |
| if [ ! -z "${!1}" ]; then |
| eval "$1=\"${!1} && \$2\"" |
| else |
| eval "$1=\"\$2\"" |
| fi |
| } |
| export -f append_cmd |
| |
| export KERNEL_DIR |
| if [ -n "${KLEAF_INTERNAL_PREFERRED_KERNEL_DIR}" ]; then |
| KERNEL_DIR="${KLEAF_INTERNAL_PREFERRED_KERNEL_DIR}" |
| fi |
| # for case that KERNEL_DIR is not specified in environment |
| if [ -z "${KERNEL_DIR}" ] && [ -n "${BUILD_CONFIG}" ]; then |
| # for the case that KERNEL_DIR is not specified in the BUILD_CONFIG file |
| # use the directory of the build config file as KERNEL_DIR |
| # for the case that KERNEL_DIR is specified in the BUILD_CONFIG file, |
| # or via the config files sourced, the value of KERNEL_DIR |
| # set here would be overwritten, and the specified value would be used. |
| build_config_path=$(readlink -f ${ROOT_DIR}/${BUILD_CONFIG}) |
| real_root_dir=${build_config_path%%${BUILD_CONFIG}} |
| build_config_dir=$(dirname ${build_config_path}) |
| build_config_dir=${build_config_dir##${ROOT_DIR}/} |
| build_config_dir=${build_config_dir##${real_root_dir}} |
| KERNEL_DIR="${build_config_dir}" |
| fi |
| if [ -z "${KERNEL_DIR}" ]; then |
| echo "ERROR: If kernel_build.build_config is not provided, kernel_build.makefile must be set." >&2 |
| exit 1 |
| fi |
| |
| set -a |
| if [ -n "${BUILD_CONFIG}" ]; then |
| . ${ROOT_DIR}/${BUILD_CONFIG} |
| fi |
| for fragment in ${BUILD_CONFIG_FRAGMENTS}; do |
| . ${ROOT_DIR}/${fragment} |
| done |
| set +a |
| |
| if [ -n "${KLEAF_INTERNAL_PREFERRED_KERNEL_DIR}" ]; then |
| if [ "${KERNEL_DIR}" != "${KLEAF_INTERNAL_PREFERRED_KERNEL_DIR}" ]; then |
| # If kernel_build.makefile is set and inconsistent with the value in build config, print a error. |
| echo "ERROR: kernel_build.makefile is set to be below ${KLEAF_INTERNAL_PREFERRED_KERNEL_DIR}," >&2 |
| echo " But it is not the same as KERNEL_DIR=${KERNEL_DIR}." >&2 |
| echo " Please delete KERNEL_DIR=${KERNEL_DIR} from ${BUILD_CONFIG}." >&2 |
| exit 1 |
| fi |
| unset KLEAF_INTERNAL_PREFERRED_KERNEL_DIR |
| else |
| # If kernel_build.makefile is not set, print a warning |
| echo "WARNING: kernel_build.makefile is not set, and KERNEL_DIR=${KERNEL_DIR}. " >&2 |
| echo " To suppress this warning, set:" >&2 |
| echo ' kernel_build(makefile = "//'"${KERNEL_DIR}"':Makefile")' >&2 |
| echo " and delete KERNEL_DIR from build config if it is set explicitly." >&2 |
| fi |
| |
| # For incremental kernel development, it is beneficial to trade certain |
| # optimizations for faster builds. |
| if [[ -n "${FAST_BUILD}" ]]; then |
| # Decrease lz4 compression level to significantly speed up ramdisk compression. |
| : ${LZ4_RAMDISK_COMPRESS_ARGS:="--fast"} |
| # Use ThinLTO for fast incremental compiles |
| : ${LTO:="thin"} |
| # skip installing kernel headers |
| : ${SKIP_CP_KERNEL_HDR:="1"} |
| fi |
| |
| export COMMON_OUT_DIR=$(readlink -m ${OUT_DIR:-${ROOT_DIR}/out${OUT_DIR_SUFFIX}/${BRANCH}}) |
| export OUT_DIR=$(readlink -m ${COMMON_OUT_DIR}/${KERNEL_DIR}) |
| export DIST_DIR=$(readlink -m ${DIST_DIR:-${COMMON_OUT_DIR}/dist}) |
| export UNSTRIPPED_DIR=${DIST_DIR}/unstripped |
| export UNSTRIPPED_MODULES_ARCHIVE=unstripped_modules.tar.gz |
| export MODULES_ARCHIVE=modules.tar.gz |
| |
| export TZ=UTC |
| export LC_ALL=C |
| if [ -z "${SOURCE_DATE_EPOCH}" ]; then |
| if [[ -n "${KLEAF_SOURCE_DATE_EPOCHS}" ]]; then |
| export SOURCE_DATE_EPOCH=$(extract_git_metadata "${KLEAF_SOURCE_DATE_EPOCHS}" "${KERNEL_DIR}" SOURCE_DATE_EPOCH) |
| # Unset KLEAF_SOURCE_DATE_EPOCHS to avoid polluting {kernel_build}_env.sh |
| # with unnecessary information (git metadata of unrelated projects) |
| unset KLEAF_SOURCE_DATE_EPOCHS |
| else |
| export SOURCE_DATE_EPOCH=$(git -C ${ROOT_DIR}/${KERNEL_DIR} log -1 --pretty=%ct) |
| fi |
| fi |
| if [ -z "${SOURCE_DATE_EPOCH}" ]; then |
| echo "WARNING: Unable to determine SOURCE_DATE_EPOCH for ${ROOT_DIR}/${KERNEL_DIR}, fallback to 0" >&2 |
| export SOURCE_DATE_EPOCH=0 |
| fi |
| export KBUILD_BUILD_TIMESTAMP="$(date -d @${SOURCE_DATE_EPOCH})" |
| export KBUILD_BUILD_HOST=build-host |
| export KBUILD_BUILD_USER=build-user |
| export KBUILD_BUILD_VERSION=1 |
| export KBUILD_GENDWARFKSYMS_STABLE=1 |
| |
| # List of dreprecated prebuilt directories that should not be used anywhere. |
| deprecated_prebuilts_paths=( |
| CLANG_PREBUILT_BIN |
| CLANGTOOLS_PREBUILT_BIN |
| RUST_PREBUILT_BIN |
| LZ4_PREBUILTS_BIN |
| DTC_PREBUILTS_BIN |
| LIBUFDT_PREBUILTS_BIN |
| BUILDTOOLS_PREBUILT_BIN |
| ) |
| |
| # List of prebuilt directories shell variables to incorporate into PATH |
| # TODO(b/164420327): Remove these once uboot usage are cleaned up. |
| prebuilts_paths=( |
| LINUX_GCC_CROSS_COMPILE_PREBUILTS_BIN |
| LINUX_GCC_CROSS_COMPILE_ARM32_PREBUILTS_BIN |
| LINUX_GCC_CROSS_COMPILE_COMPAT_PREBUILTS_BIN |
| ) |
| prebuilts_paths+=("${deprecated_prebuilts_paths[@]}") |
| |
| unset LD_LIBRARY_PATH |
| |
| for prebuilt_bin in "${deprecated_prebuilts_paths[@]}"; do |
| prebuilt_bin_value=\${${prebuilt_bin}} |
| eval prebuilt_bin_value="${prebuilt_bin_value}" |
| if [ -n "${prebuilt_bin_value}" ]; then |
| echo "WARNING: ${prebuilt_bin} should not be set (value: ${prebuilt_bin_value}). This will be an error in the future." >&2 |
| fi |
| done |
| |
| for prebuilt_bin in "${prebuilts_paths[@]}"; do |
| prebuilt_bin=\${${prebuilt_bin}} |
| eval prebuilt_bin="${prebuilt_bin}" |
| if [ -n "${prebuilt_bin}" ]; then |
| # Mitigate dup paths |
| PATH=${PATH//"${ROOT_DIR}\/${prebuilt_bin}:"} |
| PATH=${ROOT_DIR}/${prebuilt_bin}:${PATH} |
| fi |
| done |
| export PATH |
| |
| unset PYTHONPATH |
| unset PYTHONHOME |
| unset PYTHONSTARTUP |
| |
| export HOSTCC HOSTCXX CC LD AR NM OBJCOPY OBJDUMP OBJSIZE READELF STRIP AS |
| |
| tool_args=() |
| |
| # LLVM=1 implies what is otherwise set below; it is a more concise way of |
| # specifying CC=clang LD=ld.lld NM=llvm-nm OBJCOPY=llvm-objcopy <etc>, for |
| # newer kernel versions. |
| if [[ -n "${LLVM}" ]]; then |
| tool_args+=("LLVM=1") |
| # Reset a bunch of variables that the kernel's top level Makefile does, just |
| # in case someone tries to use these binaries in this script such as in |
| # initramfs generation below. |
| HOSTCC=clang |
| HOSTCXX=clang++ |
| CC=clang |
| LD=ld.lld |
| AR=llvm-ar |
| NM=llvm-nm |
| OBJCOPY=llvm-objcopy |
| OBJDUMP=llvm-objdump |
| OBJSIZE=llvm-size |
| READELF=llvm-readelf |
| STRIP=llvm-strip |
| else |
| if [ -n "${HOSTCC}" ]; then |
| tool_args+=("HOSTCC=${HOSTCC}") |
| fi |
| |
| if [ -n "${CC}" ]; then |
| tool_args+=("CC=${CC}") |
| if [ -z "${HOSTCC}" ]; then |
| tool_args+=("HOSTCC=${CC}") |
| fi |
| fi |
| |
| if [ -n "${LD}" ]; then |
| tool_args+=("LD=${LD}" "HOSTLD=${LD}") |
| fi |
| |
| if [ -n "${NM}" ]; then |
| tool_args+=("NM=${NM}") |
| fi |
| |
| if [ -n "${OBJCOPY}" ]; then |
| tool_args+=("OBJCOPY=${OBJCOPY}") |
| fi |
| fi |
| |
| if [ -n "${LLVM_IAS}" ]; then |
| tool_args+=("LLVM_IAS=${LLVM_IAS}") |
| # Reset $AS for the same reason that we reset $CC etc above. |
| AS=clang |
| fi |
| |
| if [ -n "${DEPMOD}" ]; then |
| tool_args+=("DEPMOD=${DEPMOD}") |
| fi |
| |
| if [ -n "${DTC}" ]; then |
| tool_args+=("DTC=${DTC}") |
| fi |
| |
| export TOOL_ARGS="${tool_args[@]}" |
| |
| export DECOMPRESS_GZIP DECOMPRESS_LZ4 RAMDISK_COMPRESS RAMDISK_DECOMPRESS RAMDISK_EXT |
| |
| DECOMPRESS_GZIP="gzip -c -d" |
| DECOMPRESS_LZ4="lz4 -c -d -l" |
| if [ -z "${LZ4_RAMDISK}" ] ; then |
| RAMDISK_COMPRESS="gzip -c -f" |
| RAMDISK_DECOMPRESS="${DECOMPRESS_GZIP}" |
| RAMDISK_EXT="gz" |
| else |
| RAMDISK_COMPRESS="lz4 -c -l ${LZ4_RAMDISK_COMPRESS_ARGS:--12 --favor-decSpeed}" |
| RAMDISK_DECOMPRESS="${DECOMPRESS_LZ4}" |
| RAMDISK_EXT="lz4" |
| fi |
| |
| # Checks .config against the result of savedefconfig. |
| # $1: source defconfig file |
| function kleaf_internal_check_defconfig_minimized() { |
| local source_config="$1" |
| |
| (cd ${OUT_DIR} && \ |
| make ${TOOL_ARGS} O=${OUT_DIR} savedefconfig) |
| RES=0 |
| diff -u "${source_config}" ${OUT_DIR}/defconfig >&2 || RES=$? |
| if [ ${RES} -ne 0 ]; then |
| echo ERROR: savedefconfig does not match "${source_config}" >&2 |
| fi |
| return ${RES} |
| } |
| export -f kleaf_internal_check_defconfig_minimized |
| |
| # verifies that defconfig matches the DEFCONFIG |
| function check_defconfig() { |
| [ "$ARCH" = "x86_64" -o "$ARCH" = "i386" ] && local ARCH=x86 |
| if [[ -f "${KERNEL_DIR}/arch/${ARCH}/configs/${DEFCONFIG}" ]]; then |
| kleaf_internal_check_defconfig_minimized "${KERNEL_DIR}/arch/${ARCH}/configs/${DEFCONFIG}" |
| else |
| kleaf_internal_check_defconfig_minimized "${OUT_DIR}/arch/${ARCH}/configs/${DEFCONFIG}" |
| fi |
| } |
| export -f check_defconfig |