bt
和 info stack
是 GDB(GNU Debugger)中的两个不同命令,用于查看调试信息的不同方面:
bt
(backtrace
)命令:
bt
命令用于获取当前线程的函数调用堆栈信息。info stack
命令:
info stack
命令用于显示当前线程的调用堆栈的更详细信息,包括局部变量和寄存器值。bt
更详细的堆栈信息,包括局部变量、函数参数以及堆栈帧的更多细节。bt
的使用示例在 GDB 中,我们可以使用 backtrace 或 bt 命令来查看当前线程的函数调用堆栈,也可以使用 info stack 命令查看。
首先在终端中启动 GDB,并将需要调试的程序作为参数传入。
$ gdb my_program
设置断点或者直接运行程序。当程序暂停时,可以使用 bt
或 backtrace
命令来查看堆栈情况。
(gdb) bt
#0 my_function (arg1=42, arg2=1337) at my_file.c:123
#1 0x00005555555552b5 in main () at main.c:12
上面的输出结果包括了当前线程的每个函数调用以及它们所在的文件、行号和参数值等信息。其中第一行是最近一次调用的函数,而最后一行是整个堆栈的起始位置(即 main 函数)。
如果想要查看某个特定帧(frame)的详细信息,可以使用 frame 或者简写为 f 命令加上帧编号进行切换。
(gdb) f 0 // 切换到第一个帧
#0 my_function (arg1=42, arg2=1337) at my_file.c:123
123 return arg1 + arg2;
(gdb) f 1 // 切换到第二个帧
#1 0x00005555555552b5 in main () at main.c:12
12 int result = my_function(42, 1337);
除了 bt
命令,还可以使用 info stack
命令来查看当前线程的堆栈信息。
info stack
命令通常用于在调试器中查看当前线程的堆栈信息,我们一般按照以下步骤来查看当前线程的堆栈信息:
info statck
的示例会话:$ gdb my_program
(gdb) run
[Program is running...]
^C
(gdb) info threads
1 Thread 0x7ffff7fc1a40 (LWP 15931) 0x00007ffff7bc8723 in poll () from /lib/x86_64-linux-gnu/libc.so.6
* 2 Thread 0x7ffff776e700 (LWP 15935) main () at my_program.c:10
(gdb) thread 2
[Switching to thread 2 (Thread 0x7ffff776e700 (LWP 15935))]
(gdb) info stack
#0 main () at my_program.c:10
在这个示例中,我们先使用 info threads 命令列出了所有线程,然后选择了线程 ID 为 2 的线程,最后使用 info stack 查看了这个线程的堆栈信息。堆栈信息中列出了函数调用的序列,以及每个函数的位置。
最后再来个真实示例,以 make 后的 Redis 为例:
(gdb) gdb redis-server
(gdb) r
... 此处输出信息很多,省略
# 按 Ctrl+C
^C
(gdb) info threads
Id Target Id Frame
* 1 Thread 0x7ffff7c53680 (LWP 876498) "redis-server" 0x00007ffff7d7346e in epoll_wait (epfd=5, events=0x7ffff78fd000, maxevents=10128, timeout=100) at ../sysdeps/unix/sysv/linux/epoll_wait.c:30
2 Thread 0x7ffff7119700 (LWP 876502) "bio_close_file" futex_wait_cancelable (private=<optimized out>, expected=0, futex_word=0x5555558c22c8 <bio_newjob_cond.lto_priv+40>) at ../sysdeps/nptl/futex-internal.h:183
3 Thread 0x7ffff6918700 (LWP 876504) "bio_aof" futex_wait_cancelable (private=<optimized out>, expected=0, futex_word=0x5555558c22f8 <bio_newjob_cond.lto_priv+88>) at ../sysdeps/nptl/futex-internal.h:183
4 Thread 0x7ffff6117700 (LWP 876505) "bio_lazy_free" futex_wait_cancelable (private=<optimized out>, expected=0, futex_word=0x5555558c2328 <bio_newjob_cond.lto_priv+136>) at ../sysdeps/nptl/futex-internal.h:183
5 Thread 0x7ffff5916700 (LWP 876506) "jemalloc_bg_thd" futex_wait_cancelable (private=<optimized out>, expected=0, futex_word=0x7ffff7a168b0) at ../sysdeps/nptl/futex-internal.h:183
(gdb) thread 2
[Switching to thread 2 (Thread 0x7ffff7119700 (LWP 876502))]
#0 futex_wait_cancelable (private=, expected=0, futex_word=0x5555558c22c8 ) at ../sysdeps/nptl/futex-internal.h:183
183 ../sysdeps/nptl/futex-internal.h: No such file or directory.
(gdb) info stack
#0 futex_wait_cancelable (private=, expected=0, futex_word=0x5555558c22c8 ) at ../sysdeps/nptl/futex-internal.h:183
#1 __pthread_cond_wait_common (abstime=0x0, clockid=0, mutex=0x5555558c2380 , cond=0x5555558c22a0 ) at pthread_cond_wait.c:508
#2 __pthread_cond_wait (cond=cond@entry=0x5555558c22a0 , mutex=mutex@entry=0x5555558c2380 ) at pthread_cond_wait.c:647
#3 0x000055555566d1e8 in bioProcessBackgroundJobs (arg=0x0) at bio.c:233
#4 0x00007ffff7e4e609 in start_thread (arg=) at pthread_create.c:477
#5 0x00007ffff7d73133 in clone () at ../sysdeps/unix/sysv/linux/x86_64/clone.S:95
OK,动动小手开始实践吧。( 别忘了点赞、收藏。感谢~)