Linux内核的编译

本文最后更新于:November 28, 2021 am

前言:

常见 Linux 内核编译有两种方式,一是直接在 Linux 系统上编译得到二进制文件,并对原有 Linux 内核进行替换,即更换 Linux 内核,此方法可能因新内核有 bug 导致系统奔溃,且难以返回原版本内核而不得不重装;第二种方法则是在模拟器中运行新的 Linux 内核,以避免对系统内核的修改。

BusyBox 是一个集成了三百多个最常用 Linux 命令和工具的软件,因为单独的 Linux 内核无任何用于用户交互的 UI,所以需要通过其它工具与新编译的Linux 内核交互。

QEMU 是以 GPL 许可证分发源码的模拟处理器,可用于模拟常见的硬件平台,常用于在 Linux 系统中建立虚拟机。

本文在阿里云 Ubuntu 18.04 64 位操作系统环境下编译 ARM Linux 内核。过程中主要是用交叉编译工具链 gcc-arm-linux-gnueabi 编译系统源码,并使用 QEMU 软件仿真硬件平台测试对象系统。

**建议使用 root 用户操作 **

本文所使用的环境:

操作系统:4.15.0-96-generic #97-Ubuntu SMP Wed Apr 1 03:25:46 UTC 2020 x86_64 x86_64 x86_64 GNU/Linux

gcc: 7.5.0

qemu: 2.11.1(Debian 1:2.11+dfsg-1ubuntu7.26)

make:GNU Make 4.1

  1. 工具准备

    Busybox 需手动下载安装,QEMU 等其他工具可在线安装。

    Linux内核下载:https://www.kernel.org/

    本文使用5.4.45版本的,并使用清华大学镜像

    1
    wget https://mirrors.tuna.tsinghua.edu.cn/kernel/v5.x/linux-5.4.5.tar.gz

    Busybox :

    1
    wget https://busybox.net/downloads/busybox-1.28.4.tar.bz2
  1. 环境配置

    Linux 内核编译环境需要大量软件包,可提前直接在线安装,或在内核编译的过程中安装,若缺少安装包,内核编译过程中会提示缺失错误。以下是部分需要的软件包,其中部分相同功能的软件包在不同的 Linux 版本下会以不同的名字存在。

    1
    apt-get install gcc qemu qemu-system-arm gcc-arm-linux-gnueabi libncurses5-dev build-essential flex bison bc  

    image-20200607225302378

  2. 编译内核

    解压 Linux 内核文件包:

    1
    tar -xzvf linux-5.4.5.tar.gz

    编译最小文件系统:

    解压 busybox,进入目录并编译:

    1
    2
    3
    4
    5
    tar -jxvf busybox-1.28.4.tar.bz2
    cd busybox-1.28.4
    export ARCH=arm
    export CROSS_COMPILE=arm-linux-gnueabi-
    make menuconfig

    以上内容中,“export”后是指定交叉编译工具链,指定芯片框架为 ARM。如下图所示是图形化界面进行内核配置。

    image-20200607231737776

按照以下路径配置成静态编译(回车进入,空格选中) :

Settings —->

  Build Options
       [*]Build static binary(no shared libs)

image-20200607232057964

配置完毕退出后继续完成编译:

1
make install  

完成后会在目录中生成“_install”目录,本目录存放了编译好的文件系统需要的命令集合,如下图所示:

image-20200607233324632

将上一步骤中生成的“_install”目录拷贝至之前解压后的内核目录,进入“_install”目录,分别创建 etc、dev、mnt、etc/init.d 等目录。

1
2
3
4
5
6
cp -r ./busybox-1.28.4/_install ./linux-5.4.5/_install
cd ./linux-5.4.5/_install/
mkdir etc
mkdir dev
mkdir mnt
mkdir -p etc/init.d

image-20200607234014872

在“_install/etc/init.d”目录下新建“rcS”文件,并写入以下内容:

1
2
3
4
5
6
7
8
9
mkdir -p /proc
mkdir -p /tmp
mkdir -p /sys
mkdir -p /mnt
/bin/mount -a
mkdir -p /dev/pts
mount -t devpts devpts /dev/pts
echo /sbin/mdev > /proc/sys/kernel/hotplug
mdev –s

image-20200607234356567

在“_install/etc”目录创建“fstab”文件,并写入以下内容:

1
2
3
4
5
proc /proc proc defaults 0 0
tmpfs /tmp tmpfs defaults 0 0
sysfs /sys sysfs defaults 0 0
tempfs /dev tmpfs defaults 0 0
debugfs /sys/kernel/debug debugfs defaults 0 0

image-20200607234653102

在“_install/etc”目录创建“inittab”文件,并写入以下内容:

1
2
3
4
::sysinit:/etc/init.d/rcS
::respawn:-/bin/sh
::askfirst:-/bin/sh
::ctrlaltdel:/bin/umount -a -r

image-20200607234926195

在“_install/dev”目录中创建如下设备节点。 :

1
2
mknod console c 5 1
mknod null c 1 3

image-20200607235200236

完成上述设置后,在内核目录中编译内核 :

1
2
3
4
export ARCH=arm
export CROSS_COMPILE=arm-linux-gnueabi-
make vexpress_defconfig
make menuconfig

image-20200607235445654

make menuconfig 设置中,按照以下路径,在 initramfs source file 中填入“_install”(按回车填入) :
General setup —->
       [*]Initial RAM filesystem and RAM disk (initramfs/initrd) support
           (_install)Initramfs source file(s)

image-20200607235912353

返回上一层,在

Boot option —->
    ()Default kernel command string

回车进入 Default kernel command string 并清 空 。 下 图 中 删 除 原 有 内 容 用
Ctl+Backspace 键。

image-20200608000442154

配置 memory split 为“3G/1G user/kernel split”,并打开High Memory Support:

Kernel features —->
    Memory split(3G/1G user/kernel split) -
    [*] High Memory Support

image-20200608000716572

在内核目录下编译内核(此步骤时间较长)

1
2
make bzImage ARCH=arm CROSS_COMPILE=arm-linux-gnueabi-
make dtbs

可能会出现“./usr/gen_initramfs_list.sh: 131: local: 1: bad variable name”的错误,原因是以前用的bash执行而现在使用sh。

解决办法:131行改为 :

1
local dev="`LC_ALL=C ls -l "${location}"`"

image-20200608093024179

编译完成后会有如下提示,并显示编译后内核的存储路径。

image-20200608003409578

  1. 运行 QEMU

    如下所示,输入 QEMU 启动命令,成功启动 QEMU,注意需指定 bzImage路径,并注意使用当前命令与 bzImage 路径的关系。

    1
    qemu-system-arm -M vexpress-a9 -m 256M -kernel arch/arm/boot/zImage -append "rdinit=/linuxrc console=ttyAMA0 loglevel=8" -dtb arch/arm/boot/dts/vexpress-v2p-ca9.dtb -nographic  

    以上命令中参数含义如下;
    -M:指定硬件芯片框架
    -m:指定运行内存大小
    -kernel:指定运行的内核镜像
    -dtb:指定具体芯片的配置信息
    -nographic:指定不使用图形界面

  2. 完成

    如下图可以看到,成功运行了我们刚刚编译的新内核

    image-20200608004409477