GDB-2——GDB调试多线程

 

一、简介

前一博文实际上已经介绍了多线程的调试方法,这节专门进行一下总结。

 

二、调试多线程

1. 使用 gdb 将程序跑起来,然后按 Ctrl + C 将程序中断下来,使用 info threads 命令查看当前进程有多少线程。

2. 使用 thread <线程编号> 可以切换到对应的线程,然后使用 bt 命令可以查看对应线程从顶到底层的函数调用,以及上层调用下层对应的源码中的位置;可以使用 frame <栈函数编号> 切换到当前函数调用堆栈的任何一层函数调用中去,然后分析该函数执行逻辑;使用 print 等命令输出各种变量和表达式值。

标号为1的线程是主线程,可以切到标号为1的线程上,然后 bt 打印出的调用栈就可以看到main()函数。对每个线程都进行这样的分析之后,基本上就可以搞清楚整个程序运行中的执行逻辑了。对不熟悉的C/C++项目搞一轮比较有帮助。

3. 调试时控制线程切换

在调试多线程程序时,有时候我们希望执行流一直在某个线程执行,而不是切换到其他线程。gdb 提供了一个在调试时将程序执行流锁定在当前调试线程的命令选项 scheduler-locking 选项,它有三个值,分别是 on、step 和 off,使用方法是 set scheduler-locking on/step/off。

(1) set scheduler-locking on 可以用来锁定当前线程,只观察这个线程的运行情况。当锁定这个线程时,其他线程就处于了暂停状态,也就是说你在当前线程执行 next、step、until、finish、return 命令时,其他线程是不会运行的。需要注意的是,你在使用 set scheduler-locking on/step 选项时要确认下当前线程是否是你期望锁定的线程,如果不是,可以使用 thread <线程编号> 切换到你需要的线程,然后再进行锁定。

(2) set scheduler-locking step 也是用来锁定当前线程,当且仅当使用 next 或 step 命令做单步调试时会锁定当前线程,如果使用 until、finish、return 等线程内的非单步调试命令,其他线程还是有机会运行的。相比较 on 选项值,step 选项值给为单步调试提供了更加精细化的控制,因为通常我们只希望在单步调试时,不希望其他线程对当前调试的各个变量值造成影响。

(3) set scheduler-locking off 用于关闭锁定当前线程。

4. 断点是全局的,无论哪个线程执行到断点位置,所有线程都会停住。

5. 实验

(1) 实验程序

 1 #include <stdio.h>
 2 #include <pthread.h>
 3 #include <unistd.h>
 4 
 5 int g = 0;
 6 
 7 void* worker_thread_1(void* p)
 8 {
 9     while (true) {
10         g = 100;
11         printf("worker_thread_1\n");
12         usleep(300000);
13     }
14     return NULL;
15 }
16 
17 void* worker_thread_2(void* p)
18 {
19     while (true)
20     {
21         g = -100;
22         printf("worker_thread_2\n");
23         usleep(500000);
24     }
25     return NULL;
26 }
27 
28 int main()
29 {
30     pthread_t thread_id_1;
31     pthread_create(&thread_id_1, NULL, worker_thread_1, NULL);
32     pthread_t thread_id_2;
33     pthread_create(&thread_id_2, NULL, worker_thread_2, NULL);
34     while (true) {
35         g = -1;
36         printf("g=%d\n", g);
37         g = -2;
38         printf("g=%d\n", g);
39         g = -3;
40         printf("g=%d\n", g);
41         g = -4;
42         printf("g=%d\n", g);
43         usleep(1000000);
44     }
45     return 0;
46 }

(2) 实验示范

lvm:~/origin_tmp/3.gdb$ g++ -g -o thread thread.cpp -lpthread
lvm:~/origin_tmp/3.gdb$ gdb thread
...
Reading symbols from thread...done.
(gdb) b 10
Breakpoint 1 at 0x4006d2: file thread.cpp, line 10.
(gdb) b 35
Breakpoint 2 at 0x40076b: file thread.cpp, line 35.
(gdb) r
Starting program: /origin_tmp/3.gdb/thread 
[Thread debugging using libthread_db enabled]
Using host libthread_db library "/lib/x86_64-linux-gnu/libthread_db.so.1".
[New Thread 0x7ffff77ef700 (LWP 219338)]
[New Thread 0x7ffff6fee700 (LWP 219339)]
worker_thread_2
[Switching to Thread 0x7ffff77ef700 (LWP 219338)]

Thread 2 "thread" hit Breakpoint 1, worker_thread_1 (p=0x0) at thread.cpp:10
10                      g = 100;
(gdb) until 43
worker_thread_2 //线程2在运行
[Switching to Thread 0x7ffff7fd9700 (LWP 219334)]

Thread 1 "thread" hit Breakpoint 2, main () at thread.cpp:35
35                      g = -1;
(gdb) until 43
worker_thread_2  //线程2在运行
worker_thread_1  //线程1在运行
g=-1             //主线程在运行
g=-2
g=-3
g=-4
main () at thread.cpp:43
43                      usleep(1000000);
(gdb) set scheduler-locking on  //设置 locking on
(gdb) until 43

Thread 1 "thread" hit Breakpoint 2, main () at thread.cpp:35
35                      g = -1;
(gdb) until 43  //可以看到locking on后只有主线程运行
g=-1
g=-2
g=-3
g=-4
main () at thread.cpp:43
43                      usleep(1000000);
(gdb) until 43  //可以看到locking on后只有主线程运行

Thread 1 "thread" hit Breakpoint 2, main () at thread.cpp:35
35                      g = -1;
(gdb) set scheduler-locking step  //设置locking step
(gdb) until 43
worker_thread_2
[Switching to Thread 0x7ffff77ef700 (LWP 219338)]

Thread 2 "thread" hit Breakpoint 1, worker_thread_1 (p=0x0) at thread.cpp:10
10                      g = 100;
(gdb) until 43  //locking step下,until还是多线程可以运行
g=100
g=-2
g=-3
g=-4
worker_thread_2
worker_thread_1

Thread 2 "thread" hit Breakpoint 1, worker_thread_1 (p=0x0) at thread.cpp:10
10                      g = 100;
(gdb) info threads  //切到主线程
  Id   Target Id         Frame 
  1    Thread 0x7ffff7fd9700 (LWP 219334) "thread" 0x00007ffff78bc38d in nanosleep () at ../sysdeps/unix/syscall-template.S:84
* 2    Thread 0x7ffff77ef700 (LWP 219338) "thread" worker_thread_1 (p=0x0) at thread.cpp:10
  3    Thread 0x7ffff6fee700 (LWP 219339) "thread" 0x00007ffff78bc38d in nanosleep () at ../sysdeps/unix/syscall-template.S:84
(gdb) thread 1
[Switching to thread 1 (Thread 0x7ffff7fd9700 (LWP 219334))]
#0  0x00007ffff78bc38d in nanosleep () at ../sysdeps/unix/syscall-template.S:84
84      ../sysdeps/unix/syscall-template.S: No such file or directory.
(gdb) n  //在locking step下单步运行情况下只有主线运行
main () at thread.cpp:34
34              while (true) {
(gdb) n

Thread 1 "thread" hit Breakpoint 2, main () at thread.cpp:35
35                      g = -1;
(gdb) n
36                      printf("g=%d\n", g);
(gdb) n
g=-1
37                      g = -2;
(gdb) n
38                      printf("g=%d\n", g);
(gdb) n
g=-2
39                      g = -3;
(gdb) n
40                      printf("g=%d\n", g);
(gdb) n
g=-3
41                      g = -4;
(gdb) n
42                      printf("g=%d\n", g);
(gdb) n
g=-4
43                      usleep(1000000);
(gdb) n
n
34              while (true) {

 

三、调试多进程

在实际的应用中,有一类应用会通过 Linux 函数 fork 出新的子进程。例如网络传输中的创建一个新的进程来处理该连接上的客户端,新产生的进程与原进程互为父子关系。那么如何用 gdb 调试这样父子进程呢,有两种方法:

(1) 方法1: 开两个 shell 终端分别对父子进程进行调试。用 gdb 先调试父进程,等子进程被 fork 出来后,重新开启一个 Shell 窗口,使用 gdb attach 到子进程上去。

(2) set follow-fork <parent/child> 来设置是当一个进程 fork 出新的子进程时,gdb 是继续调试父进程还是子进程,默认是父进程。使用 how follow-fork mode 来查看当前的调试模式。
默认是fork之后attach到父进程的,但是如果想在 fork 之后 gdb 去 attach 子进程,我们可以在程序 run 之前在 gdb 中设置 set follow-fork child 来实现。可以通过 Ctrl +C 将程序中断下来,然后使用 bt 命令查看当前线程调用堆栈来确定是主进程还是子进程。

 

posted on 2022-11-08 11:46  Hello-World3  阅读(3716)  评论(0编辑  收藏  举报

导航