我们都知道GDB是Linux下面一款强大程序调试的工具,以前我们都是用GDB来调试单线程程序,今天我们来谈一谈如何用GDB来调试多线程程序把。
有关GDB的基本指令在这里就不叙述了,如果感兴趣的可以看看下面这篇博客:
http://blog.csdn.net/bit_clearoff/article/details/53966718
info threads
这条指令显示可以调试的所有线程,gdb会为每个线程分配一个ID,这个ID和线程ID不同,ID号一般从1开始。
为什么要切换线程呢?因为gdb默认在调试多线程程序时只跟踪主线程,而新创建的线程都阻塞在线程创建函数pthread_create处。如果我们不切换,那么我们就无法调试我们在主线程中所创建的线程。
thread ID
这条指令用来切换当前调试的线程为指定的ID号的线程,ID为GDB为线程所分配的ID.
//指定多个线程执行command指令,其中command为gdb中的指令
thread apply ID1 ID2 ...IDn command
//指定所有线程执行command指令,其中command为gdb中的指令
thread apply all command
//线程锁
set scheduler-locking off|on
off:不锁定任何线程,当程序继续运行的时候如果有断点,那么就把所有的线程都暂停下来,直到你指定某个线程继续执行,如果在当前线程下使用continue的话那么会启动所有线程(GDB默认);
on:打开线程锁,锁定其他线程,只有当前线程执行。
为了对上面的这种一个线程中断在一个断点上,其他所有的线程都会被freeze. gdb v7.0引入了non-stop模式,在这个模式下:
1. 当某个或多个线程在一个断点上,其他线程仍会并行运行
2. 你可以选择某个被中断的线程,只让他运行。
3.
non-stop模式表示不停止模式,除了断点有关的进程会被停下来,其他线程会继续执行。
设置non-stop模式,打开gdb后,在开始r之前,首先连续输入下面的指令
set target-async 1
set pagination off
set non-stop on
首先,我们插入多线程调试的代码:
#include
#include
#include
#include
void* PthreadRun1(void* arg)
{
int count=5;
while(count--){
printf("Thread1::It's %ld pthread running...\n",pthread_self());
sleep(1);
}
pthread_exit((void*)0);
}
void* PthreadRun2(void* arg)
{
int count=5;
while(count--){
printf("Thread2::It's %ld pthread running...\n",pthread_self());
sleep(1);
}
pthread_exit((void*)0);
}
int main()
{
pthread_t id1,id2;
if(0!=pthread_create(&id1,NULL,PthreadRun1,NULL))
{
printf("create id1 pthread error...\n");
exit(1);
}
if(0!=pthread_create(&id2,NULL,PthreadRun2,NULL))
{
printf("create id2 pthread error...\n");
exit(1);
}
pthread_join(id1,NULL);
pthread_join(id2,NULL);
return 0;
}
我们发现在默认情况下,gdb只跟踪主线程,而新创建的线程都被阻塞在pthread_create函数处,如果我们运行到pthread_join函数或者主线程被终止了,此时我们在主线程中创建的线程才会被终止,否则在主线程中创建的线程会被一直阻塞。
下面,我们来使用上面提到的gdb指令来调试多线程程序,首先我们在默认模式下调试。
我们发现当我们设置了线程锁后再次continue时,这次只有一个线程在运行了,而且是当前线程。