GDB 调试多进程

这位IBMer提出三种调试方式,很是靠谱。http://www.ibm.com/developerworks/cn/linux/l-cn-gdbmp/。鉴于本人的嵌入式背景,多数不太适合采用VNC,于是乎,就举例说明前两种方法吧。待调试代码

  1 #include <stdlib.h>
  2 #include <unistd.h>
  3 
  4 int main(){
  5     pid_t pid;
  6     int status , i = 9;
  7 
  8     printf("Now going to fork\n");
  9 
 10     pid = fork();
 11     if(pid<0){
 12         printf("fork error\n");
 13         return -1;
 14     }
 15     else if( pid>0 ){
 16         wait(&status);
 17         printf("return %d\n",WEXITSTATUS(status));
 18         exit(0);
 19     }
 20     else {
 21         while(i-->0){
 22             printf("loop %d\n",i);
 23             sleep(2);
 24         }
 25         exit(3);
 26         }
 27 
 28  
 29 }

方法1:
   编译 gcc -g multi_proc.c
   加载 gdb a.out

(gdb) set follow-fork-mode child 
(gdb) b 23
(gdb) r
Starting program: /home/XXX/test/a.out 
Now going to fork
[New process 30427]
[Switching to process 30427]

Breakpoint 1, main () at multi_proc.c:22
22	            printf("loop %d\n",i);
Missing separate debuginfos, use: debuginfo-install glibc-2.13-2.i686
(gdb) c
Continuing.
loop 8

于是就可以正常调试子进程了。


方法2 :
对于方案1,如果进程继承级别比较多,调试起来比较麻烦,而且还限定了GDB必须为高版本。当然现在的GDB版本都是满足的。方案2的精髓就是attach子进程,如果需要让子进程在被attach之前不做任何操作,可以采用条件触发的方法。其中一个方案可以采用下文。

while(1){
	if(file exist){
		some thing realy do;
		break;	
	}
	else{
		sleep(1);
	}
}

在attach进程之后,touch file,进程才真正执行。

调试步骤,打开一个console

[XXX@localhost test]$ ./a.out 
Now going to fork
child pid = 31803

再打开一个cnsole

gdb a.out 31803
(gdb) bt
#0  0x00544416 in __kernel_vsyscall ()
#1  0x00bf1060 in __nanosleep_nocancel () from /lib/libc.so.6
#2  0x00bf0ea4 in sleep () from /lib/libc.so.6
#3  0x08048563 in main () at multi_proc.c:23
(gdb) l 23
19	        exit(0);
20	    }
21	    else {
22	        while(i-->0){
23	            printf("loop %d\n",i);
24	            sleep(4);
25	        }
26	        exit(3);
27	        }
28	
(gdb) 

结合bt命令和第24行的源代码,说明进程睡在这里呢


<think>好的,用户想了解如何使用GDB调试多进程程序,特别是设置调试进程的方法。首先,我需要回忆GDB处理多进程调试的相关命令和策略。记得GDB默认是跟踪父进程,子进程会继续运行不受控制,所以需要配置才能调试子进程。 接下来,应该分步骤说明不同的方法。首先想到的是`follow-fork-mode`,这个选项可以设置为父进程或子进程。需要详细解释如何设置,比如在启动GDB后使用`set follow-fork-mode child`来跟踪子进程,或者用`parent`保持默认。还要提到`show follow-fork-mode`来查看当前模式。 然后,另一个相关的是`detach-on-fork`,它决定在fork之后GDB是否保持对其他进程的控制。如果设置为off,GDB会同时控制父进程和子进程,这时候可能需要配合`inferior`命令来切换不同的进程。这部分需要举例说明,比如切换inferior编号,以及如何查看当前进程。 另外,用户可能需要附加到已运行的进程上,所以需要提到`attach`命令的用法,结合进程ID。还有在代码中插入断点或者`sleep`的方法,以便有时间附加到子进程。 还需要注意不同GDB版本的差异,比如某些命令可能在旧版本中不可用。此外,多线程和多进程的区别可能让用户混淆,需要明确说明这里是多进程的情况。 最后,总结常用方法,提醒用户根据具体情况选择跟踪模式或手动附加,并给出示例命令。确保步骤清晰,让用户能逐步操作,并验证是否有效。可能还要提醒检查GDB版本,或者如何处理常见问题,比如无法附加进程时的权限问题。</think>GDB调试多进程程序时,可通过以下方法设置调试目标进程: ===== 1. 基础设置 ===== 在GDB中通过命令控制fork后的跟踪目标: ```gdb (gdb) set follow-fork-mode parent|child # 设置跟踪父/子进程 (gdb) set detach-on-fork on|off # 控制是否分离未跟踪进程 ``` 示例场景: ```c // 示例代码片段 pid_t pid = fork(); if (pid == 0) { child_process(); // 子进程代码 } else { parent_process(); // 父进程代码 } ``` ===== 2. 多进程调试模式 ===== ◆ 模式1:跟踪单个进程 ```gdb # 默认跟踪父进程 (gdb) set follow-fork-mode parent (gdb) set detach-on-fork on # 或跟踪子进程 (gdb) set follow-fork-mode child ``` ◆ 模式2:同时调试多个进程 ```gdb (gdb) set detach-on-fork off # 保留所有fork的进程 (gdb) info inferiors # 查看进程列表 (gdb) inferior 2 # 切换到第二个进程 ``` ===== 3. 高级调试技巧 ===== ▶ 附加调试已有进程 ```bash # 在终端获取子进程PID ps -ef | grep [程序名] # 在GDB中附加 (gdb) attach <pid> ``` ▶ 代码层控制 ```c // 在子进程代码开始处添加调试桩 void child_process() { volatile int wait_gdb = 1; while(wait_gdb); // 在此处设置断点 //...实际代码... } ``` ===== 4. 验证设置 ===== 调试时使用以下命令确认状态: ```gdb (gdb) show follow-fork-mode # 显示当前跟踪模式 (gdb) info inferiors # 列出所有被控进程 (gdb) inferior # 显示当前调试的进程 ``` ===== 注意事项 ===== 1. Linux内核≥3.9需设置: ```bash echo 0 > /proc/sys/kernel/yama/ptrace_scope ``` 2. 建议使用GDB≥7.0版本 3. 多进程调试时注意断点设置位置(不同进程地址空间独立) 典型调试流程示例: ```gdb $ gdb ./multi_process_prog (gdb) b main (gdb) set detach-on-fork off (gdb) r # 程序fork后暂停 (gdb) info inferiors Num Description Executable 1 process 1234 multi_process_prog 2 process 5678 multi_process_prog (gdb) inferior 2 (gdb) b child_func (gdb) c ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值