用GDB调试多进程程序
如果一个进程通过fork系统调用创建了子进程,gdb会继续调试原来的进程,子进程则正常运行。那么该如何调试子进程呢?
单独调试子进程:
子进程从本质上说也是一个进程,因此我们可以用通用的gdb调试方法来调试他。
举例来说如果要调试一个服务器中的某一个子进程,我们可以先运行服务器,然后找到目标子进程的PID,再将其附加(attach)到gdb调试器上,具体操作为:
$ps -ef | grep 进程名 //通过该命令得到待调试进程的PID
$(gdb) attach "PID" //上面的“PID”即待调试进程的pid
使用调试器选项follow-fork-mode
gdb调试器的选项follow-fork-mode允许我们选择程序在执行fork系统调用后是继续调试父进程还是调试子进程。
其用法如下:
$(gdb)set follow-fork-mode mode
上面的mode可选parent或child,分别表示调试父进程和子进程。
举例:
$(gdb) set follow-fork-mode child
$(gdb) b process.h:264 //在process.h 的264行下断点
用gdb调试多线程程序
gdb有一组命令可辅助多线程程序的调试。
infothreads
显示当前可调试的所有线程。gdb会为每个线程分配一个ID,我们可以使用这个ID来操作对应的线程。ID前面有“*”的线程是当前被调试的线程。
threadID
调试目标ID指定的线程。
set scheduler-locking[off|on|step]
调试多线程程序时,默认除了被调试的线程在执行外,其他线程也在继续执行,但有的时候我们希望只让被调试的线程运行。这可以通过这个命令来实现。
该命令设置scheduler-locking的值:
off表示不锁定任何线程,即所有线程都可以继续执行,这是默认值。
on表示只有当前被调试的线程会继续执行。
step表示在单步执行的时候,只有当前线程会执行。
以下是使用示例
$(gdb) info threads //查看线程信息,当前被调试的是那个线程
$(gdb) set scheduler-locking on //不执行其他线程,锁定调试对象
$(gdb)thread 2 //将调试切换到子线程,其ID为2
关于调试进程池或线程池程序的一个不错的方法:
先将池中的进程个数或线程个数减少至一,以观察程序的逻辑是否正确,然后逐步增加进程或线程的数量,以调试进程或线程的同步是否正确。