当前位置: 首页 > article >正文

如何调试Linux内核?

通过创建一个最小的根文件系统,并使用QEMU和GDB进行调试。

1.准备工作环境

确保系统上安装了所有必要的工具和依赖项。

sudo apt-get update    //更新一下软件包
sudo apt-get install build-essential git libncurses-dev bison flex libssl-dev qemu-system-x86 gdb busybox-static
  • sudo apt-get install build-essential:安装编译工具链,其中包含了gcc(GNU编译器集合,用于C/C++程序)、g++(GNU C++编译器)、make(构建工具,用于自动化编译和安装)、libc-dev(C标准库的开发文件)。

  • git:分布式版本控制系统。
  • libncurses-dev:提供字符终端处理库的开发文件。在编译和配置内核的时候,需要依赖此库。
  • bison:GNU解析器生成器。用于生成语法分析器(Parser),在编译内核时可能会使用。
  • flex:快速词法分析器生成器。配合bison使用。
  • libssl-dev:解密一些协议,在内核模块可能使用。
  • qemu-system-x86:QEMU的x86系统模拟器。用于模拟x86架构的虚拟机,支持运行操作系统内核或完整系统。
  • gdb:GNU调试器。用于调试C/C++程序,支持设置断点、单步执行、查看变量和内存,与QEMU配合使用,调试Linux内核。
  • busybox-static:静态编译的BusyBox工具集,提供一组精简的Unix工具,将所有依赖库都包含在可执行文件中,无需外部依赖。

2.获取并配置Linux内核源码

先进入到一个目录下,然后自己克隆就可以了

git clone https://github.com/torvalds/linux.git
cd linux

3.配置内核

为了简化过程,可以使用默认配置并启动调试信息:

make defconfig

在Linux源代码中,找到 .config 文件,然后“Ctrl + f”启动搜索键,搜索调试信息“CONFIG_DEBUG_INFO”,然后可能会出现“CONFIG_DEBUG_INFO_NODE”,你就把他注释了。然后你进行编译“make -j4”,启动四个内核进行编译,提高速度,但是会出现四个调试选项,是因为你把之前个给注释了,在这里你要选择与“CONFIG_DEBUG_INFO”相关的。之后就会成功编译。

编译成功后,你应该能找到两个文件:

  • arch/x86/boot/bzImage:这是内核映像文件。
  • vmlinux:这是未压缩的内核映像文件,用于GDB调试。

4.创建最小的根文件系统

4.1准备目录结构

创建一个目录来存放根文件系统的文件:

mkdir -p ~/Compiler-2/rootfs   //具体路径视情况而定
cd ~/Compiler-2/rootfs
mkdir -p bin sbin etc proc sys usr/bin usr/sbin
ln -s bin sbin
  • bin:存放系统的基本指令(二进制可执行文件),这些命令对所有用户可用。
  • sbin:存放系统管理员使用的管理命令(二进制可执行文件),通常需要root权限才能执行。
  • etc:存放系统的配置文件,包括一些用户信息,网络配置等
  • proc:一个虚拟文件系统,提供内核和进程信息的接口。它不占用磁盘空间,而是由内核动态生成。包括CPU信息、内存信息等
  • sys:一个虚拟文件系统,提供内核和硬件设备的配置接口。与proc类似,它也是由内核动态生成的。用来配置内核参数、管理设备等
  • usr/bin:存放用户安装的命令和应用程序(二进制可执行文件)。gcc、vim等
  • usr/sbin:存放用户安装的系统管理命令(二进制可执行文件),通常需要root权限才能执行、

第三行创建的很多目录是Linux文件系统层次结构标准(FHS)的一部分。

第四行命令的作用是创建一个 符号链接(symbolic link),将sbin目录软链接到bin目录。因为在某些情况下,bin和sbin目录的内容可能会相同或相似,通过将sbin链接到bin,可以节省空间,方便统一管理,之后也就不需要区分这两个目录,也可以简化路径。

4.2复制BuysBox二进制文件

复制BusyBox二进制文件到根文件系统目录:

cp $(which busybox) bin/

将当前目录下 bin/ 目录中的 busybox 可执行文件所支持的所有工具,以符号链接的形式安装到指定目录中(其实就是默认当前目录下的 bin/ 目录)。其中-s是 --install 选项的子选项,表示创建符号链接而不是硬链接(hard link)。

./bin/busybox --install -s

4.3创建 init 脚本

在根文件系统中创建一个简单的初始化脚本(init script)来启动shell:

创建一个名为 init 的文件,并将 #!/bin/sh 写入文件的第一行。其中 #!/bin/sh 是脚本的 shenbang,指定脚本使用 /bin/sh(Bourne Shell)作为解释器。

echo '#!/bin/sh' > init

将 mount -t proc none /proc 追加到 init 文件中。mount -t proc none /proc 挂载 proc 文件系统打破 /proc 目录。proc文件系统提供了内核和进程信息的接口,是Linux系统的重要组成部分。>> init将输出追加到init文件中,不会覆盖原有内容。

echo 'mount -t proc none /proc' >> init

sysfs文件系统提供了内核和硬件设备的配置接口,通常用于管理设备和内核模块。

echo 'mount -t sysfs none /sys' >> init

exec sh 是启动一个交互式的Shell。exec 用新的进程替换当前进程。这行代码的作用是让系统在初始化完成后进入一个交互式的 Shell 环境,方便用户操作或调试。

echo 'exec sh' >> init

为 init 文件添加可执行权限。这行代码的作用是让 init 文件可以被执行。因为在Linux系统中,脚本文件必须具有可执行权限才能直接运行。

chmod +x init

4.4 打包成 cpio 归档

将 rootfs 目录打包成一个 cpio 归档文件,并使用 gzip 压缩,最终生成一个压缩的根文件系统映像文件 rootfs.cpio.gz。

cd ~/Compiler-2
find rootfs | cpio -o --format=newc | gzip > rootfs.cpio.gz

最后在我的电脑上,输出 1 block,表示 cpio 归档文件中只包含一个块的数据(512字节)。

5.使用QEMU启动内核

在第一个终端窗口中运行一下命令启动QEMU:

qemu-system-x86_64 \
    -kernel arch/x86/boot/bzImage \
    -append nokaslr\
    -s -S \
    -m 2024
  • qemu-system-x86_64:这是QEMU的可执行文件,用于模拟x86_64架构的虚拟机
  • -kernel linux/arch/x86/boot/bzImage:指定要启动的Linux内核镜像文件(在上面编译的时候也提到了)
  • -s:启用 GDB 调试服务器,默认监听端口 1234
  • -S:启动时暂停CPU,等待GDB连接。
  • -m 2024 :给虚拟机分配2024MB   //可加可不加

6.启动GDB连接QEMU进行调试

在第二个终端窗口中启动GDB并连接到QEMU:

cd ~/Compiler-2/linux-6.1
gdb vmlinux
(gdb) target remote localhost:1234
(gdb) continue

之后你就可以使用gdb打断点进行调试了。(*^▽^*)


http://www.kler.cn/a/569081.html

相关文章:

  • VS Code Python调试执行代码时出现“ConnectionRefusedError: [WinError 10061] 由于目标计算机积极拒绝,无法连接”的问题解决
  • Llama 2中的Margin Loss:为何更高的Margin导致更大的Loss和梯度?
  • 三七互娱,蓝禾,顺丰,oppo,游卡,汤臣倍健,康冠科技,作业帮,高途教育25届春招内推
  • 深入浅出理解编译器:前端视角
  • Kotlin 扩展函数
  • 【无人机】无人机飞行日志下载及分析,飞行日志分析软件的使用
  • 蓝桥杯(握手问题)
  • Express + MongoDB 实现用户登录
  • 蓝桥杯好题推荐-----高精度减法
  • PyQt5入门教程和简单使用
  • 阿里云 Qwen2.5-Max:超大规模 MoE 模型架构和性能评估
  • [AIGC]Agent的ReAct原理基于LangChain框架的Agent构建详解
  • Windows系统安装GPU驱动/CUDA/cuDNN/PyTorch
  • PHP环境安装达梦数据库驱动实操
  • 迷你世界脚本玩家接口:Player
  • Excel 豆知识 - XLOOKUP 为啥会出 #N/A 错误
  • 【音视频】RGG、YUV基础
  • Docker 部署 Graylog 日志管理系统
  • 通往 AI 之路:Python 机器学习入门-语法基础
  • Mysql的基础命令有哪些?