# 准备
sudo apt install build-essential | |
sudo apt install qemu | |
sudo apt install libncurses5-dev bison flex libssl-dev libelf-dev |
# 下载内核源码
sudo apt install axel | |
axel -n 20 https://mirrors.edge.kernel.org/pub/linux/kernel/v5.x/linux-5.4.34.tar.xz | |
xz -d linux-5.4.34.tar.xz | |
tar -xvf linux-5.4.34.tar | |
cd linux-5.4.34 |
# 配置内核选项
make defconfig # Default configuration is based on 'x86_64_defconfig' | |
make menuconfig # 打开 debug 相关选项 |
进行如下设置:
Kernel hacking —> | |
Compile-time checks and compiler options —> | |
[*] Compile the kernel with debug info | |
[*] Provide GDB scripts for kernel debugging | |
[*] Kernel debugging | |
关闭KASLR(随机地址),否则会导致打断点失败。 | |
Processor type and features ----> | |
[ ] Randomize the address of the kernel image (KASLR) |
# 编译运行内核
make -j$(nproc) | |
qemu-system-x86_64 -kernel arch/x86/boot/bzImage |
# 制作内存根文件系统
#首先从 https://www.busybox.net 下载 busybox 源代码解压,解压完成后,跟内核一样先配置编译,并安装。 | |
axel -n 20 https://busybox.net/downloads/busybox-1.31.1.tar.bz2 | |
tar -jxvf busybox-1.31.1.tar.bz2 | |
cd busybox-1.31.1 | |
make menuconfig #Setting 里的 Build static binary (no shared libs) 选中 |
然后编译安装,默认会安装到源码目录下的 _install 目录中。
make -j$(nproc) && sudo make install |
然后制作内存根文件系统镜像:
mkdir rootfs # 在/linux-5.4.34文件夹下新建rootfs文件夹
cd rootfs
cp ../busybox-1.31.1/_install/* ./ -rf
mkdir dev proc sys home
sudo cp -a /dev/{null,console,tty,tty1,tty2,tty3,tty4} dev/
准备 init 脚本文件放在根文件系统跟目录下(rootfs/init),添加如下内容到 init 文件。
#!/bin/sh | |
mount -t proc none /proc | |
mount -t sysfs none /sys | |
echo “Wellcome qjliOS!” | |
echo “--------------------” | |
cd home | |
/bin/sh |
给 init 脚本添加可执行权限:
chmod +x init |
打包成内存根文件系统镜像:
find . -print0 | cpio --null -ov --format=newc | gzip -9 > ../rootfs.cpio.gz |
测试挂载根文件系统,看内核启动完成后是否执行 init 脚本。
qemu-system-x86_64 -kernel ./arch/x86/boot/bzImage -initrd rootfs.cpio.gz |
出现 “Wellcome qjliOS!” 则 init 脚本执行成功。
# VSCode 配置 Linux 内核调试环境
由于 Linux 内核高度定制化,所以没有办法直接通过配置 includePath 等让 Intellisense 正常提示,这里借助一个 Python 脚本来生成 compile_commands.json 文件帮助 Intellisense 正常提示(包括头文件和宏定义等)。在 Linux 源代码目录下直接运行如下命令就可以生成 compile_commands.json 了。
python ./scripts/gen_compile_commands.py |
新建一个.vscode 文件夹,将配置文件放入该文件夹内,配置如下五个文件:c_cpp_properties.json,init,launch.json,settings.json,tasks.json。
c_cpp_properties.json
{ | |
"configurations": [ | |
{ | |
"name": "Linux", | |
"includePath": [ | |
"${workspaceFolder}/arch/x86/include/**", | |
"${workspaceFolder}/include/**", | |
"${workspaceFolder}/include/linux/**", | |
"${workspaceFolder}/arch/x86/**", | |
"${workspaceFolder}/**" | |
], | |
"cStandard": "c11", | |
"intelliSenseMode": "gcc-x64", | |
"compileCommands": "${workspaceFolder}/compile_commands.json" | |
} | |
], | |
"version": 4 | |
} |
init
#!/bin/sh | |
mount -t proc none /proc | |
mount -t sysfs none /sys | |
echo "Wellcome qjlOS!" | |
echo "--------------------" | |
cd home | |
/bin/sh |
launch.json
{ | |
// Use IntelliSense to learn about possible attributes. | |
// Hover to view descriptions of existing attributes. | |
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387 | |
"version": "0.2.0", | |
"configurations": [ | |
{ | |
"name": "(gdb) linux", | |
"type": "cppdbg", | |
"request": "launch", | |
"preLaunchTask": "vm", | |
"program": "${workspaceRoot}/vmlinux", | |
"miDebuggerServerAddress": "localhost:1234", | |
"args": [], | |
"stopAtEntry": true, | |
"cwd": "${workspaceFolder}", | |
"environment": [], | |
"externalConsole": false, | |
"MIMode": "gdb", | |
"miDebuggerArgs": "-n", | |
"targetArchitecture": "x64", | |
"setupCommands": [ | |
{ | |
"text": "set arch i386:x86-64:intel", | |
"ignoreFailures": false | |
}, | |
{ | |
"text": "dir .", | |
"ignoreFailures": false | |
}, | |
{ | |
"text": "add-auto-load-safe-path ./", | |
"ignoreFailures": false | |
}, | |
{ | |
"text": "-enable-pretty-printing", | |
"ignoreFailures": true | |
} | |
] | |
} | |
] | |
} |
settings.json
{ | |
"search.exclude": { | |
"**/.git": true, | |
"**/.svn": true, | |
"**/.DS_Store": true, | |
"**/drivers": true, | |
"**/sound": true, | |
"**/tools": true, | |
"**/arch/alpha": true, | |
"**/arch/arc": true, | |
"**/arch/c6x": true, | |
"**/arch/h8300": true, | |
"**/arch/hexagon": true, | |
"**/arch/ia64": true, | |
"**/arch/m32r": true, | |
"**/arch/m68k": true, | |
"**/arch/microblaze": true, | |
"**/arch/mn10300": true, | |
"**/arch/nds32": true, | |
"**/arch/nios2": true, | |
"**/arch/parisc": true, | |
"**/arch/powerpc": true, | |
"**/arch/s390": true, | |
"**/arch/sparc": true, | |
"**/arch/score": true, | |
"**/arch/sh": true, | |
"**/arch/um": true, | |
"**/arch/unicore32": true, | |
"**/arch/xtensa": true | |
}, | |
"files.exclude": { | |
"**/.*.*.cmd": true, | |
"**/.*.d": true, | |
"**/.*.o": true, | |
"**/.*.S": true, | |
"**/.git": true, | |
"**/.svn": true, | |
"**/.DS_Store": true, | |
"**/drivers": true, | |
"**/sound": true, | |
"**/tools": true, | |
"**/arch/alpha": true, | |
"**/arch/arc": true, | |
"**/arch/c6x": true, | |
"**/arch/h8300": true, | |
"**/arch/hexagon": true, | |
"**/arch/ia64": true, | |
"**/arch/m32r": true, | |
"**/arch/m68k": true, | |
"**/arch/microblaze": true, | |
"**/arch/mn10300": true, | |
"**/arch/nds32": true, | |
"**/arch/nios2": true, | |
"**/arch/parisc": true, | |
"**/arch/powerpc": true, | |
"**/arch/s390": true, | |
"**/arch/sparc": true, | |
"**/arch/score": true, | |
"**/arch/sh": true, | |
"**/arch/um": true, | |
"**/arch/unicore32": true, | |
"**/arch/xtensa": true | |
}, | |
"[c]": { | |
"editor.detectIndentation": false, | |
"editor.tabSize": 8, | |
"editor.insertSpaces": false | |
}, | |
"C_Cpp.errorSquiggles": "Disabled" | |
} |
tasks.json
{ | |
// See https://go.microsoft.com/fwlink/?LinkId=733558 | |
// for the documentation about the tasks.json format | |
"version": "2.0.0", | |
"tasks": [ | |
{ | |
"label": "vm", | |
"type": "shell", | |
"command": "qemu-system-x86_64 -kernel ${workspaceFolder}/arch/x86/boot/bzImage -initrd ../rootfs.cpio.gz -S -s -nographic -append \"console=ttyS0\"", | |
"presentation": { | |
"echo": true, | |
"clear": true, | |
"group": "vm" | |
}, | |
"isBackground": true, | |
"problemMatcher": [ | |
{ | |
"pattern": [ | |
{ | |
"regexp": ".", | |
"file": 1, | |
"location": 2, | |
"message": 3 | |
} | |
], | |
"background": { | |
"activeOnStart": true, | |
"beginsPattern": ".", | |
"endsPattern": ".", | |
} | |
} | |
] | |
}, | |
{ | |
"label": "build linux", | |
"type": "shell", | |
"command": "make", | |
"group": { | |
"kind": "build", | |
"isDefault": true | |
}, | |
"presentation": { | |
"echo": false, | |
"group": "build" | |
} | |
} | |
] | |
} |
# start_kernel 分析
Linux 内核的起点是 start_kernel 函数,因此先在 start_kernel 处打断点,启动调试,程序在断点处暂停,从 start_kernel 开始进行跟踪分析。单步跳过进行跟踪分析,发现 0 号进程 init_task 被设为整个系统的初始进程,即 0 号进程是手工创建的,其他进程都是 0 号进程创建的。在内核引导时,init_task 会被创建并启动,是所有其他进程的起点。
继续单步跳过,start_kernel 执行各种初始化操作。
在 start_kernel () 函数末尾,arch_call_rest_init () 函数体内为 rest_init () 函数,因此设置一个 rest_init 函数断点,进入 rest_init 函数体内,该函数由 0 号进程执行。
kernel_thread 函数创建 kernel_init,对应 1 号进程,是所有用户进程的祖先。
接着 kernel_thread 函数创建 kthreadd,对应 2 号进程,是所有内核进程的祖先。
进入 kernel_thread 函数查看,该函数通过 do_fork 函数创建进程。通过 kernel_thread 函数代码可以看到 1 号进程和 2 号进程最终都是通过 do_fork 创建的,用户态通过系统调用 fork 创建一个进程最终也是通过 do_fork 来完成的。
进程的创建过程大致是父进程通过 fork 系统调用进入内核_do_fork 函数,复制进程描述符及相关进程资源(采用写时复制技术)、分配子进程的内核堆栈并对内核堆栈和 thread 等进程关键上下文进行初始化,最后将子进程放入就绪队列,fork 系统调用返回;而子进程则在被调度执行时根据设置的内核堆栈和 thread 等进程关键上下文开始执行。
查看 kernel_init 函数定义,如下图的代码实现部分会调用 run_init_process 函数。
run_init_process 函数如下,其中 do_execve 用于加载可执行文件、运行 init 程序并执行 exec 系统调用。1 号进程完成用户态初始化。
查看 kthreadd 函数定义。2 号进程创建并完成初始化。