第一个用法,set args作为程序运行时参数:
测试代码
#include <stdio.h>
int main(int argc,char**argv){
printf("argv[0] is %s\n",argv[0]);
printf("argv[1] is %s\n",argv[1]);
printf("argv[2] is %s\n",argv[2]);
}
gdb调试,先设置后运行。
(gdb) set args hello world
(gdb) r
Starting program: /home/huqw/test/a.out hello world
argv[0] is /home/huqw/test/a.out
argv[1] is hello
argv[2] is world
等同于
# ./a.out hello world
argv[0] is ./a.out
argv[1] is hello
argv[2] is world
第一个参数都没法变了,是程序名,第二三个参数都是手动设置。
关于设置运行参数,其实这也不是唯一途径,另一个方法同样能解决问题:
(gdb) shell ./a.out 1 2222
argv[0] is ./a.out
argv[1] is 1
argv[2] is 2222
没错,就是shell命令,一直觉得gdb界面像shell,甚至以为有些命令就是shell命令,但是记不太清了,所以不好说。虽然原则上讲这不是shell,是gdb自己的命令输入界面,但是gdb还是提供了命令供你使用系统的shell命令的。这样,就和没用gdb一样,直接运行程序输入参数,也是可以的。
但是这时候就来不及打断点调试了,等于直接运行了,其实如果你gdb是先打开(装载)的别的文件,这时候断点什么的也都是原来那个程序。如下。
(gdb) shell ./a.out 1 2
a is 5
a is 5
//两次之间在外部重新编译了一个a.out,所以下边就不一样了
(gdb) shell ./a.out 1 2222
argv[0] is ./a.out
argv[1] is 1
argv[2] is 2222
(gdb) list
2 int main(){
3 int a = 5;
4 printf("a is %d\n",a);
5 printf("a is %d\n",a);
6 return 0;
7 }
(gdb)
注意:
即使加载的是同一个目标文件,这种方式也不能调试,断点也不能停下,原理你们懂得,毕竟运行的不是一个“副本”吧,我只能这样解释了。所以,其实是代替不了上边的那种运行方式的。。
(gdb) list
1 #include <stdio.h>
2
3 int main(int argc,char**argv){
4 printf("argv[0] is %s\n",argv[0]);
5 printf("argv[1] is %s\n",argv[1]);
6 printf("argv[2] is %s\n",argv[2]);
7 }
(gdb) b 4
Breakpoint 1 at 0x80483cd: file set_args.c, line 4.
(gdb) shell ./a.out
argv[0] is ./a.out
argv[1] is (null)
argv[2] is HOSTNAME=jiaxun
(gdb) shell ./a.out 1 22
argv[0] is ./a.out
argv[1] is 1
argv[2] is 22
第二个用法,改变运行时函数的参数
#include<stdio.h>
int main(){
int a = 5;
printf("a is %d\n",a);
printf("a is %d\n",a);
return 0;
}
gdb调试
Breakpoint 1, main () at set_param.c:4
4 printf("a is %d\n",a);
(gdb) print a
$2 = 5
(gdb) n
a is 5
5 printf("a is %d\n",a);
(gdb) print a
$3 = 5
(gdb) set var a=3
(gdb) print a
$4 = 3
(gdb) n
a is 3
6 return 0;
第二次打印a is 3,说明a的值直接被篡改。通过gdb的print也可测。