您的位置:首页 > 运维架构 > Linux

linux基础之gdb调试多进程,多线程

2017-06-07 12:37 711 查看

一.gdb常用目录表

操作描述
l命令相当于list,从第一行开始列出源码
回车重复上一次命令
break +行号设置断点
break+函数名在函数的入口点设置断点
r运行程序running的简写
into break查看断点信息
n单语句执行 ,next的简写
c继续运行程序,continue的简写
bt查看函数的堆栈
finish退出函数
q退出gdb

二.多进程调试

测试代码:

#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>

int main()
{
pid_t id = fork();
if(id == 0)//child
{
printf("I am child ,my id is:%d;my father id is:%d \n",getpid(),getppid());
}
else if(id > 0)//parent
{
sleep(2);
printf("I am father:%d \n",getpid());
}
else
{
perror("fork error");
return -1;
}
return 0;
}


默认设置下,在调试多进程程序时GDB只会调试主进程。但是GDB(>V7.0)支持多进程的分别以及同时调试,换句话说,GDB可以同时调试多个程序。只需要设置follow-fork-mode(默认值:parent)和detach-on-fork(默认值:on)即可。

parenton只调试主进程(此程序中系统默认只调试parent)
childon只调试子进程
parentoff同时调试两个进程,gdb跟主进程。子进程block(阻塞)在fork位置
childoff同时调试两个进程,gdb跟子进程,主进程block在fork位置
设置方法如下图:show follow-fork-mode show detach-on-fork



查询正在调试的进程:info inferiors

显示GDB调试的所有inferior,GDB会为他们分配ID。其中带有*的进程是正在调试的inferior.

( GDB将每一个被调试程序的执行状态记录在一个名为inferior的结构中。一般情况下一个inferior对应一个进程,每个不同的inferior有不同的地址空间。inferior有时候会在进程没有启动的时候就存在。)

切换调试的进程: inferior < infer number >

切换到ID是num的inferior进行调试。



添加新的调试进程: add-inferior [-copies n] [-exec executable] ,可以用file executable来分配给inferior可执行文件。

增加n个inferior并执行程序为executable。如果不指定n只增加一个inferior。如果不指定executable,则执行程序留空,增加后可使用file命令重新指定执行程序。这时候创建的inferior其关联的进程并没启动。

remove-inferiors infno:

删除一个infno号的inferior。如果inferior正在运行,则不能删除,所以删除前需要先kill或者detach这个inferior。

detach inferior:

detach掉编号是infno的inferior。注意这个inferior还存在,可以再次用run命令执行它

kill inferior infno:

kill掉infno号inferior。注意这个inferior仍然存在,可以再次用run等命令执行它。

三.多线程调试

多线程代码

#include<stdio.h>
#include<pthread.h>

void* start_routine1()
{
printf("I am a thread1 ,my tid is:%u\n",pthread_self());
return NULL;
}
void* start_routine2()
{
printf("I am a thread1 ,my tid is:%u\n",pthread_self());
return NULL;
}
int main()
{
pthread_t tid1;
pthread_t tid2;
pthread_create(&tid1,NULL, start_routine1, NULL);
pthread_create(&tid2,NULL, start_routine2, NULL);

pthread_join(tid1,NULL);
pthread_join(tid2,NULL);
return 0;
}


注意:在创建线程时候,在编译链接的时候一定要加-lpthread!!!一定要加-lpthread!!!一定要加-lpthread!!!! 重要的事情说三遍,惨痛的教训

运行图



在多线程编程时,当我们需要调试时,有时需要控制某些线程停在断点,有些线程继续执行。有时需要控制线程的运行顺序。有时需要中断某个线程,切换到其他线程。这些都可以通过gdb实现。

GDB默认支持调试多线程,跟主线程,子线程block在create thread。

多线程编程常用目录表

操作描述
info threads显示当前可调试的所有线程,每个线程会有一个GDB为其分配的ID,后面操作线程的时候会用到这个ID。 前面有*的是当前调试的线程。
thread ID切换当前调试的线程为指定ID的线程
break FileName.cpp:LinuNum thread all所有线程都在文件FileName.cpp的第LineNum行有断点
thread apply ID1 ID2 command让一个或者多个线程执行GDB命令command
thread apply all command让所有被调试线程执行GDB命令command
set scheduler-locking off/on/step在调式某一个线程时,其他线程是否执行。off,不锁定任何线程,默认值。on,锁定其他线程,只有当前线程执行。step,在step(单步)时,只有被调试线程运行。
set non-stop on/off当调式一个线程时,其他线程是否运行
set pagination on/off在使用backtrace时,在分页时是否停止
set target-async on/ff同步和异步。同步,gdb在输出提示符之前等待程序报告一些线程已经终止的信息。而异步的则是直接返回

四.关于core

core的意思是核心,dumped的意思就是抛出,转储,core dumped就是核心转储的意思。当一个进程异常退出前,该进程会抛出当时该程序进程的内存详细情况存储在硬盘上,文件名通常是core,这就叫core dump。

进程异常终止通常是因为代码存在BUG,比如非法内存访问导致段错误,事后可以用调试器检查core文件以查清错误原因,这叫做事后调试.

关于core的指令:

指令描述
uname -a查看机器参数
ulimit -a查看默认参数
ulimit -c 1024设置core文件大小为1024
ulimit -c unlimit设置core文件大小为无限
修改ulimit的设置,让它产生。

指令描述
ulimit -c 1024设置core文件大小为1024。要是core文件大于1024个块,就产生不出来了。
ulimit -c unlimit设置core文件大小为无限
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息