您如何从 AOSP 为 Android 构建 Linux 内核?
How do you build a Linux Kernel for Android from the AOSP?
我发现 AOSP build.sh dist
创建:
XML 个文件:
./out/soong/.intermediates/kernel/configs/q/android-4.19/kernel_config_q_4.19/gen/conditional.xml
./out/soong/.intermediates/kernel/configs/q/android-4.19/kernel_config_q_4.19/matrix.xml
来自内核配置文件:
kernel/configs/q/
├── android-4.14
│ ├── android-base-conditional.xml
│ ├── android-base.config
│ ├── Android.bp
│ ├── android-recommended-arm64.config
│ ├── android-recommended-arm.config
│ ├── android-recommended.config
│ └── android-recommended-x86.config
├── android-4.19
│ ├── android-base-conditional.xml
... ...
Linux 内核目录的完整副本:
out/target/product/sdm660_64/obj/kernel/msm-4.14
这个新创建的 msm-4.14
包含一个看起来像正常内核配置的 .config
文件。
如何为 aarch64
编译这个内核?
当我使用 make
时,它提示重新生成一个 x86_64
配置文件,因为 .config
正确包含 Aarch64 设置。
也许我尝试了错误的事情,但是 kernel/msm-4.14
中的 mm
忽略了对内核配置的更改,并且 build.sh dist
花了几个小时。
我希望能够在短短几分钟内更改内核配置和源代码并构建新内核,就像在台式机上一样。
我应该如何为 Android 执行此操作?
这是为您的硬件编译内核并创建引导映像的方法:
$ cd <aosp_root_dir>
$ source ./build/envsetup.sh
$ lunch <product_name>-<build_variant>
# Example: lunch sdm660_64-userdebug
$ make bootimage -j4
# This compiles the kernel and copies it to
# <aosp_root_dir>/out/target/product/<product_name>/kernel,
# and creates boot image at
# <aosp_root_dir>/out/target/product/<product_name>/boot.img
如果您仍然需要减少花费的时间,那么您必须找到 AOSP 构建系统执行编译内核和创建引导映像的各个命令。
您可以从 <aosp_root_dir>/out/verbose.log.gz
中找到命令。它是包含上次构建的详细日志的压缩包。
因此,首先使用 make bootimage
命令构建启动映像,然后解压缩 verbose.log.gz
包,您将获得 verbose.log
文件。
在该文件中,找到包含文本的日志行:defconfig
,这很可能是 AOSP 构建系统执行编译内核的命令。
在我的例子中,这 2 个是我从 verbose log
中找到的用于编译内核的命令:
# make sdm660_defconfig
${ANDROID_BUILD_TOP}/prebuilts/build-tools/linux-x86/bin/make -j1 \
-C kernel/msm-4.4 \
O=${ANDROID_PRODUCT_OUT}/obj/kernel/msm-4.4 \
HOSTCC=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-gcc \
HOSTAR=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-ar \
HOSTLD=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-ld \
HOSTCFLAGS="-I/usr/include -I/usr/include/x86_64-linux-gnu -L/usr/lib -L/usr/lib/x86_64-linux-gnu" \
HOSTLDFLAGS="-L/usr/lib -L/usr/lib/x86_64-linux-gnu" \
ARCH=arm64 \
CROSS_COMPILE=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/aarch64/aarch64-linux-android-4.9/bin/aarch64-linux-android- \
sdm660_defconfig
# make
${ANDROID_BUILD_TOP}/prebuilts/build-tools/linux-x86/bin/make -j4 \
-C kernel/msm-4.4 \
O=${ANDROID_PRODUCT_OUT}/obj/kernel/msm-4.4 \
HOSTCC=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-gcc \
HOSTAR=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-ar \
HOSTLD=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-ld \
HOSTCFLAGS="-I/usr/include -I/usr/include/x86_64-linux-gnu -L/usr/lib -L/usr/lib/x86_64-linux-gnu" \
HOSTLDFLAGS="-L/usr/lib -L/usr/lib/x86_64-linux-gnu" \
ARCH=arm64 \
CROSS_COMPILE=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/aarch64/aarch64-linux-android-4.9/bin/aarch64-linux-android-
使用上述命令编译内核后,根据verbose log
将编译输出复制到<aosp_root_dir>/out/target/product/<product_name>/kernel
。那是用于创建引导映像的内核文件。
cp "${ANDROID_PRODUCT_OUT}/obj/kernel/msm-4.4/arch/arm64/boot/Image.gz" \
"${ANDROID_PRODUCT_OUT}/kernel"
最后,您可以找到创建启动映像的命令。根据我的 verbose log
,以下是命令:
${ANDROID_BUILD_TOP}/out/host/linux-x86/bin/mkbootimg \
--kernel ${ANDROID_PRODUCT_OUT}/kernel \
--ramdisk ${ANDROID_PRODUCT_OUT}/ramdisk-recovery.img \
--cmdline "console=ttyMSM0,115200,n8 androidboot.console=ttyMSM0 earlycon=msm_serial_dm,0xc170000 androidboot.hardware=qcom user_debug=31 msm_rtb.filter=0x37 ehci-hcd.park=3 lpm_levels.sleep_disabled=1 sched_enable_hmp=1 sched_enable_power_aware=1 service_locator.enable=1 swiotlb=1 loop.max_part=7 buildvariant=eng veritykeyid=id:`openssl x509 -in build/target/product/security/verity.x509.pem -text | grep keyid | sed 's/://g' | tr -d '[:space:]' | tr '[:upper:]' '[:lower:]' | sed 's/keyid//g'`" \
--base 0x00000000 \
--pagesize 4096 \
--os_version 10 \
--os_patch_level yyyy-mm-dd \
--header_version 1 \
--output ${ANDROID_PRODUCT_OUT}/boot.img
像这样,您可以找到为您的硬件编译内核并创建引导映像的命令。
我发现 AOSP build.sh dist
创建:
XML 个文件:
./out/soong/.intermediates/kernel/configs/q/android-4.19/kernel_config_q_4.19/gen/conditional.xml ./out/soong/.intermediates/kernel/configs/q/android-4.19/kernel_config_q_4.19/matrix.xml
来自内核配置文件:
kernel/configs/q/ ├── android-4.14 │ ├── android-base-conditional.xml │ ├── android-base.config │ ├── Android.bp │ ├── android-recommended-arm64.config │ ├── android-recommended-arm.config │ ├── android-recommended.config │ └── android-recommended-x86.config ├── android-4.19 │ ├── android-base-conditional.xml ... ...
Linux 内核目录的完整副本:
out/target/product/sdm660_64/obj/kernel/msm-4.14
这个新创建的 msm-4.14
包含一个看起来像正常内核配置的 .config
文件。
如何为 aarch64
编译这个内核?
当我使用 make
时,它提示重新生成一个 x86_64
配置文件,因为 .config
正确包含 Aarch64 设置。
也许我尝试了错误的事情,但是 kernel/msm-4.14
中的 mm
忽略了对内核配置的更改,并且 build.sh dist
花了几个小时。
我希望能够在短短几分钟内更改内核配置和源代码并构建新内核,就像在台式机上一样。
我应该如何为 Android 执行此操作?
这是为您的硬件编译内核并创建引导映像的方法:
$ cd <aosp_root_dir>
$ source ./build/envsetup.sh
$ lunch <product_name>-<build_variant>
# Example: lunch sdm660_64-userdebug
$ make bootimage -j4
# This compiles the kernel and copies it to
# <aosp_root_dir>/out/target/product/<product_name>/kernel,
# and creates boot image at
# <aosp_root_dir>/out/target/product/<product_name>/boot.img
如果您仍然需要减少花费的时间,那么您必须找到 AOSP 构建系统执行编译内核和创建引导映像的各个命令。
您可以从 <aosp_root_dir>/out/verbose.log.gz
中找到命令。它是包含上次构建的详细日志的压缩包。
因此,首先使用 make bootimage
命令构建启动映像,然后解压缩 verbose.log.gz
包,您将获得 verbose.log
文件。
在该文件中,找到包含文本的日志行:defconfig
,这很可能是 AOSP 构建系统执行编译内核的命令。
在我的例子中,这 2 个是我从 verbose log
中找到的用于编译内核的命令:
# make sdm660_defconfig
${ANDROID_BUILD_TOP}/prebuilts/build-tools/linux-x86/bin/make -j1 \
-C kernel/msm-4.4 \
O=${ANDROID_PRODUCT_OUT}/obj/kernel/msm-4.4 \
HOSTCC=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-gcc \
HOSTAR=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-ar \
HOSTLD=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-ld \
HOSTCFLAGS="-I/usr/include -I/usr/include/x86_64-linux-gnu -L/usr/lib -L/usr/lib/x86_64-linux-gnu" \
HOSTLDFLAGS="-L/usr/lib -L/usr/lib/x86_64-linux-gnu" \
ARCH=arm64 \
CROSS_COMPILE=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/aarch64/aarch64-linux-android-4.9/bin/aarch64-linux-android- \
sdm660_defconfig
# make
${ANDROID_BUILD_TOP}/prebuilts/build-tools/linux-x86/bin/make -j4 \
-C kernel/msm-4.4 \
O=${ANDROID_PRODUCT_OUT}/obj/kernel/msm-4.4 \
HOSTCC=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-gcc \
HOSTAR=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-ar \
HOSTLD=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/host/x86_64-linux-glibc2.17-4.8/bin/x86_64-linux-ld \
HOSTCFLAGS="-I/usr/include -I/usr/include/x86_64-linux-gnu -L/usr/lib -L/usr/lib/x86_64-linux-gnu" \
HOSTLDFLAGS="-L/usr/lib -L/usr/lib/x86_64-linux-gnu" \
ARCH=arm64 \
CROSS_COMPILE=${ANDROID_BUILD_TOP}/prebuilts/gcc/linux-x86/aarch64/aarch64-linux-android-4.9/bin/aarch64-linux-android-
使用上述命令编译内核后,根据verbose log
将编译输出复制到<aosp_root_dir>/out/target/product/<product_name>/kernel
。那是用于创建引导映像的内核文件。
cp "${ANDROID_PRODUCT_OUT}/obj/kernel/msm-4.4/arch/arm64/boot/Image.gz" \
"${ANDROID_PRODUCT_OUT}/kernel"
最后,您可以找到创建启动映像的命令。根据我的 verbose log
,以下是命令:
${ANDROID_BUILD_TOP}/out/host/linux-x86/bin/mkbootimg \
--kernel ${ANDROID_PRODUCT_OUT}/kernel \
--ramdisk ${ANDROID_PRODUCT_OUT}/ramdisk-recovery.img \
--cmdline "console=ttyMSM0,115200,n8 androidboot.console=ttyMSM0 earlycon=msm_serial_dm,0xc170000 androidboot.hardware=qcom user_debug=31 msm_rtb.filter=0x37 ehci-hcd.park=3 lpm_levels.sleep_disabled=1 sched_enable_hmp=1 sched_enable_power_aware=1 service_locator.enable=1 swiotlb=1 loop.max_part=7 buildvariant=eng veritykeyid=id:`openssl x509 -in build/target/product/security/verity.x509.pem -text | grep keyid | sed 's/://g' | tr -d '[:space:]' | tr '[:upper:]' '[:lower:]' | sed 's/keyid//g'`" \
--base 0x00000000 \
--pagesize 4096 \
--os_version 10 \
--os_patch_level yyyy-mm-dd \
--header_version 1 \
--output ${ANDROID_PRODUCT_OUT}/boot.img
像这样,您可以找到为您的硬件编译内核并创建引导映像的命令。