linux 内核启动调试,学会利用Gdb调试内核的启动

学会利用Gdb调试内核的启动

“调试是通过某种办法发现BUG进行修正的过程” – 《Debug Hacks》

在做Kdump时,发现crashkernel区域位于某些位置时,vmcore初始化总是失败。原 因是Dump-Capture

kernel启动后已经不在crashkernel的区域内了,这显然是不对 的。原来是Dump-Capture

kernel的映像zImage启动解压时重定向,将内核解压到非

crashkernel的区域,使用Image替代zImage来做Dump-Capture kernel就可以解决这个 问题了。

分享一下自己Gdb调试zImage/Image启动的一些方法,以更方便理解了内核启动中一

些魔法。以下方法可在Qemu下的vexpress-a9目标演示。

Gdb调试zImage的启动

像日志Qemu下调试内核一样启动Qemu.Gdb连接到Qemu.

(gdb) target remote tcp:localhost:1234

vexpress-a9的”BIOS”会在RAM的起始地址0x60000000处开始执行,初始化atags和 processor

id,并跳转到0x60010000

(gdb) b * 0x60010000

(gdb) c

手工加载内核调试映像。zImage对应的ELF文件是

$output/arch/arm/boot/compressed/vmlinux,由于这个映像地址无关,手工添加符号表

到指定位置。

(gdb) load vmlinux 0x60010000

(gdb) add-symbol-file vmlinux 0x60010000

(gdb) b start

(gdb) c

至此,可以在源代码下单步调试zImage了。如果代码重定位了,再 add-symbol-file到指定位置就可以了。

Gdb调试Image的启动

Image是自解压之后的内核映像,刚启动时,MMU还没有将内存映射到vmlinux的链接

地址。Image对应的ELF文件是$output/vmlinux, 需要加载映像到 0x60008000的内存位置。

(gdb) target remote tcp:localhost:1234

(gdb) b * 0x60010000

(gdb) c

(gdb) load vmlinux -0x60000000

通过$ arm-linux-readelf -S vmlinux得到符号表段的偏移后,加载符号表到相应的 内存物理地址。

(gdb) add-symbol-file vmlinux 0x60008240 -s .head.text

0x60008000 -s .rodata 0x60479000

修改pc地址以修正load指令更新的默认pc地址。

(gdb) set $pc=0x60008000

(gdb) b stext

(gdb) c

至些,像调试普通应用一样,可以一直调试到打开MMU __enable_mmu为止。此后,就

要重新add-symbol-file到相应的虚拟地址,即vmlinux的链接地址。

在真实的目标机中,如使用Trace32调试内核的启动代码,方法也是一样的。

另外,使用调试器查看汇编代码,似乎也是学习汇编代码一个好途径。​

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: gdb是一个强大的调试工具,可以用于调试Linux内核。使用gdb调试Linux内核需要以下步骤: 1. 编译内核时添加调试信息 在编译内核时需要添加调试信息,可以通过在Makefile中添加以下选项来实现: CONFIG_DEBUG_INFO=y CONFIG_DEBUG_KERNEL=y 2. 启动内核调试启动内核时需要添加调试选项,可以通过在grub或者lilo中添加以下选项来实现: debug debug earlyprintk=serial,ttyS0,115200 3. 连接gdb 使用gdb连接内核需要使用kgdb插件,可以通过以下命令加载kgdb插件: modprobe kgdb 然后使用以下命令连接gdb: gdb vmlinux (gdb) target remote /dev/ttyS0 4. 使用gdb调试 连接成功后,可以使用gdb进行调试,例如: (gdb) b start_kernel (gdb) c 这将在内核启动时设置断点,并继续执行内核。当内核执行到断点时,gdb将停止执行并等待命令。 以上是使用gdb调试Linux内核的基本步骤,具体调试方法和命令可以参考gdb文档和Linux内核调试文档。 ### 回答2: gdb是一款功能强大的调试器,在日常的编程开发中得到了广泛应用。然而,gdb调试Linux内核时与调试用户态应用程序时有所不同。调试内核需要使用gdb的特殊功能来处理调试内核的问题。在下面的几个方面中,我将解释如何使用gdb来调试Linux内核。 1. 准备gdb环境 首先需要将gdb环境设置为可以使用内核符号。在编译内核时,需要在Makefile中添加CONFIG_DEBUG_INFO和CONFIG_DEBUG_KERNEL选项,以支持调试信息。此外,还需要安装所需的内核符号,然后通过"sudo sysctl -w kernel.yama.ptrace_scope=0"以解决防止调试器附加的安全机制问题。 2. 加载内核映像 通过gdb来加载内核映像。使用gdb命令"file vmlinux"来加载内核映像,其中vmlinux是含有调试符号的内核镜像文件。 3. 内核断点调试 可以使用gdb设置内核断点,以调试内核时确定内核程序执行过程中的问题。使用gdb命令"b <function>"设置函数断点,而使用"b * <address>"设置指定地址的断点。 4. 调试内核panic 当内核执行时发生错误时,系统会进入panic状态。如果需要调试内核panic,可以使用gdb命令"handle SIGTRAP noprint pass"来设置中断处理。使用"monitor halt"或直接ctrl+c可以停止内核,查看是什么出问题了,并且使用"cont"命令让内核继续运行。 5. 查看内核堆栈 可以使用gdb命令"bt"来查看内核的堆栈,以确定调试内核时的问题。在通过gdb调试内核处理内核问题时,内核堆栈非常有用。 总的来说,使用gdb调试Linux内核需要更多的操作方式和技巧,但是如果熟练掌握gdb的某些功能和命令,并且了解内核基本结构和运行机制,就可以高效地调试内核出现的问题。 ### 回答3: GDB(GNU调试器)是一个强大的调试工具,也可以用来调试Linux内核。但是,与调试应用程序或用户空间程序相比,内核调试可能会更加复杂。下面是关于如何使用GDB调试Linux内核的一些指南和步骤。 1.编译内核 为了调试内核,首先需要编译内核并安装它。在编译内核时,需要启用符号表。使用如下命令: $ make menuconfig 在Kernel hacking中设置Debug kernel,这将启用符号表的编译。然后使用命令“make”编译内核并安装它。 2.配置调试环境 在内核启动时,启用调试合适的调试器非常重要。例如,可以使用串线调试器而不是控制台输出,因为调试信息可能无法立即打印到控制台。可以通过UART控制台或JTAG调试器进行调试,这通常比控制台输出更可靠。 3.使用GDB连接到内核 使用GDB的第一步是启动GDB,同时指定内核映像文件作为第一个参数。例如,如果内核映像文件为”vmlinuz”,可以使用如下命令连接GDB到内核: $ gdb vmlinuz 然后,需要设置GDB的默认连接地址: add-symbol-file /path/to/kernel/vmlinux 0xADDRESS 这里的ADDRESS 应该是编译内核时已知的地址。可以在内核映像文件的/System.map文件中找到地址信息。 4.使用GDB调试 在连接GDB到内核后,可以使用GDB来单步执行内核代码或设置断点。可以使用GDB的“step”命令来进入下一个函数调用,并使用“next”命令来执行下一行代码。还可以使用“break”命令设置断点,以捕获特定的行动或事件。 也可以在调试内核时使用一些GDB的调试命令,例如:“watch”命令用于设置监视点,以监视变量的值,而不必停止调试器。可以使用“info” 命令,以获取关于调试目标状态的信息。 总之,调试Linux内核需要仔细的计划和准备。在内核源代码中随时插入代码并编译并不是一个高效的方法。使用GDB可以更轻松地捕获和解决内核的问题。通过使用GDB,可以提高软件开发的效率,并确保Linux内核的稳定性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值