多进程是Linux编程中一个很重要的内容,典型的例子就是守护进程(daemon)。有关守护进程的定义和编程规范,请参考:
http://blog.youkuaiyun.com/tuzhutuzhu/article/details/19092211
最常见的多进程的形式如下:
pid = fork();
if (pid < 0) { // fork failed
printf("fork error\n");
exit(1);
} else if (pid > 0) { // parent process
// command
} else { // child process
// command
}
对于这种类型的多进程程序的调试,在gdb中使用选项follow-fork-mode即可。
使用:set follow-fork-mode child,即可追踪子进程。而set follow-fork-mode parent可调试父进程。
还有一种多进程程序的形式为:
pid = fork();
if (pid < 0) { // fork failed
printf("fork error\n");
exit(1);
} else if (pid > 0) { // parent process
// command
} else { // child process
execv("a.out", NULL);
}
这种程序的调试则要困难一些,下面我们通过一个实际的例子来看一下它的调试方法。
当前有两个代码test.c和child.c,其中test.c中的子进程通过execv调用child.c。
test.c
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
int main()
{
int ret = 0;
ret = fork();
if (ret == 0) {
execv("a.out", NULL); //a.out是child.c编译成的可执行文件
}
return 0;
}
child.c
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
int main()
{
my_print();
return 0;
}
int my_print()
{
printf("hello world\n");
return 0;
}
使用gdb调试的详细过程如下所示:
上面的例子中,最重要的操作时catch exec这个事件。捕获到exec这个事件之后再往子进程的程序中打一个断点,然后执行continue操作。可以看到,此时程序就会进入到exec调用的子进程中了。
网上介绍gdb调试fork+exec创建的子进程的方法有不少,实际使用之后觉得,还是这种方法操作起来较为简洁明了。大家如果有什么更好的方法,请一定告诉我。
后记:
本文主要介绍的是使用gdb调试fork+exec创建的子进程的方法。虽然方法是知道了,但是在实际的工作中,这种方法的实用性并不是十分的高。因为上述方法有很大的局限性:这种方法只能用在父进程中仅有一个exec的程序中。
当程序中存在多个fork+exec,使用上述的方法只能进入到第一个子进程中!!因此,想要调试其他的子进程就不行了。然而,对于任意一个子进程都可以自由的进行调试,才是我的真正目标。我也会继续调查gdb对于fork+exec创建的多个子进程的调试方法,一旦有进展将会及时和大家分享。