GDB调试多进程程序或同时调试多个程序
转自:http://hellogcc.blogbus.com/logs/71170939.html
在以前,如果在GDB中想调试多进程的程序,需要在fork以前“set follow-fork-mode parent|child”来确定fork后是调试父进程还是调试子进程,而另一个进程在fork就会被detach。而在GDB 7.1之后你只要用"set detach-on-fork off"关掉fork之后的detach就可以很方便的同时调试2个进程。
同时GDB现在还可以支持同时调试多个程序(符号信息不同的几个进程)。
首先进行一个名词解释inferior,GDB将每个被调试程序的执行状态的记录结构称为一个inferior。
一般情况下inferior会对应一个进程,当然嵌入式平台可能有不同情况。inferior有时候会在进程没有启动的情况下就存在(这在命令中会有所体现,后面会详细介绍),每个inferior会有不同的地址空间,并且一个inferior里面可以包含多个线程。
下面就详细介绍一下多进程调试的命令:
info inferiors
显示GDB调试的所有inferior。
前面有*的是当前inferior,直接发GDB命令控制的就是当前inferior。
inferior infno
设置infno号inferior为当前inferior。
add-inferior [ -copies n ] [ -exec executable ]
增加n个inferior并且其执行程序为executable;如果不指定n则只增加一个inferior;如果不指定executable,则执行程序留空,增加后可使用file命令重新指定执行程序。
注意:这时候创建的inferior其关联的进程就并没有启动。
clone-inferior [ -copies n ] [ infno ]
增加n个inferior并且其执行程序指定为和infno号inferior一样;如果不指定n则只增加一个inferior;如果不指定infno,则指定跟当前inferior一样的执行程序。
注意:这个命令只是克隆了另一个infno的执行文件名称,和clone这个系统调用并没什么相关性(个人觉得用copy更不容易产生歧义)。
remove-inferior infno
删除一个infno号inferior。如果inferior在运行,则不能删除inferior,所以在删除以前需要先kill或者detach这个inferior。
detach inferior infno
detach掉infno号inferior。注意这个inferior仍然存在,可以再次用run等命令执行它,如果想删除结构需要用remove-inferior命令。
kill inferior infno
kill掉infno号inferior。注意这个inferior仍然存在,可以再次用run等命令执行它,如果想删除结构需要用remove-inferior命令。
set print inferior-events on|off
show print inferior-events
这个选项用来打开和关闭inferior状态的提示信息。
set detach-on-fork on|off
show detach-on-fork
这个选项用来控制当fork或者vfork发生的时候,是否detach掉父进程或者子进程(至于具体哪个就由下面的命令决定),默认值是打开,所以想同时调试父进程和子进程的就需要关闭这个选项。
set follow-fork-mode parent|child
show follow-fork-mode
这个选项用来设置当发生fork或者vfork的时候,GDB将继续调试父进程或者子进程,上面的命令也介绍了,如果detach-on-fork选项打开,而不被调试的哪个进程将被detach。
set follow-exec-mode new|same
show follow-exec-mode
当发生exec的时候,如果这个选项是same(默认值),因为父进程已经退出,所以自动在执行exec的inferior上控制子进程。
如果选项是new,则新建一个inferior给执行起来的子进程,而父进程的inferior仍然保留,当前保留的inferior的程序状态是没有执行。
set schedule-multiple on|off
show schedule-multiple
这个选项类似于多线程调试里的set scheduler-locking选项(参见 http://hellogcc.blogbus.com/logs/69150201.html)。
当选项是off(默认值)的时候,GDB发出执行指令的时候,只有当前inferior会执行。
而当选项是on的时候,GDB发出执行命令后,全部状态是执行状态的inferior都会执行。
注意,如果scheduler-locking选项设置为lock的时候,即使schedule-multiple设置为on,也只有当前进程的当前线程会执行。
maint info program-spaces
用来显示当前GDB一共管理了多少地址空间。