Cross-Compilation of Linux Kernel and Debug with QEMU and GDB
In the following tutorial it uses LLVM=1
because I wanted to use LLVM to compile because I was using rust for linux.
Kernel Build
make LLVM=1 ARCH=x86_64 menuconfig
Kernel hacking → Compile the Kernel with debug info Kernel hacking → Provide GDB scripts for Kernel debugging
make LLVM=1 ARCH=x86_64 CROSS_COMPILE=x86_64-unknown-linux-gnu -j$(nproc)
Buildroot
git clone https://github.com/buildroot/buildroot
cd buildroot
make LLVM=1 ARCH=x86_64 menuconfig
Target Options → Target Architecture → x86_64
Filesystem images → ext2/3/4 root file system → ext4
make LLVM=1 ARCH=x86_64 CROSS_COMPILE=x86_64-unknown-linux-gnu -j$(nproc)
QEMU
Installing
sudo apt install qemu qemu-system
To boot the kernel using the root filesystem generated by Buildroot, run the following command:
BUILDROOT_PATH=../buildroot
qemu-system-x86_64 \
-kernel arch/x86/boot/bzImage \
-boot c \
-m 2049M \
-drive file=$BUILDROOT_PATH/output/images/rootfs.ext4,format=raw \
-append "root=/dev/sda rw console=ttyS0,115200 acpi=off nokaslr" \
-serial stdio \
-display none \
-s -S
-s
: Opens a GDB server on port 1234.-S
: Stops QEMU execution, allowing GDB to connect.
Debugging with gdb-multiarch
Installing:
sudo apt-get install gdb-multiarch
Add the following line to your ~/.gdbinit
file to load the GDB scripts provided by the kernel for enhanced debugging:
add-auto-load-safe-path <replace_kernel_path>/scripts/gdb/vmlinux-gdb.py
This enables GDB helper scripts that are useful for kernel debugging.
Debugging Session:
Start QEMU in Debug Mode
Ensure QEMU is running with the -s -S options so that it waits for GDB to connect.
Lauch GDB
In another terminal, start gdb-multiarch and load the kernel symbols from vmlinux:
gdb-multiarch vmlinux
In GDB, connect to QEMU by running target remote :1234
:
(gdb) target remote :1234
Remote debugging using :1234
0x000000000000fff0 in exception_stacks ()
Now you can debug the kernel:
(gdb) hb start_kernel
Hardware assisted breakpoint 1 at 0xffffffff834174c3: file init/main.c, line 905.
(gdb) c
Continuing.
Breakpoint 1, start_kernel () at init/main.c:905
905 char *command_line;
References
- https://medium.com/@depressedcoder/m1-mac-linux-kernel-development-environment-setup-748637131f92