kwseeker

学编程:找代码,读代码,改代码,写代码;少看书。但少看不是不看,看一本书要限制在一天内。任务是通读书中主要的概念,明白大致是干什么用的,形成一个大概的框架;然后一周内干掉书中全部代码,代码不理解再浏览书中相关章节,注意是浏览。或者刷博客。代码实在刷不懂,可以先刷后面,待日后重新刷;因为书中代码有些部分可能出自后面的章节。代码总是在多次刷过之后,拨开乌云见日月的。。。

导航

GDB调试器

/*this project used for gdb debug c programs*/
/*At first,using compile command turn out the executable file. exp: gcc -g sourcefile.c -o test.exe */        //!!!
/*windows: start gdb debuging(test.exe complete trace) : gdb test.exe */
/*Linux: start gdb debuging(current trace) : gdb ./test  */                    //!!!
/*
    int strcmp(const char *s1,const char *s2);
    char *fgets(char *buf,int n,FILE *fp);  //send data read from file stream to buffer.
        FILE *fp = fopen(argv[1],"r");
*/
/*

补充:

  对含有命令行参数的程序的调试

  (gdb) r arg1 arg2 arg3 …

  


    使可执行程序最大限度的保留调试信息:
        gcc -ggdb3 -o <test> <test.c>
              -Wall
            -gstabs  关闭编译优化

    gdb帮助:    

           (gdb) h
                (gdb) help all
                (gdb) help <command class>
                (gdb) help <command>
                (gdb) apropos <word>    搜索相关word的命令


    编译时注意加上-g,在可执行文件中加入源文件信息(提供我们以后对程序进行调试的信息)        !!!
                -o,要求输出可执行文件文件
                -c,表示我们只要求编译器输出目标代码,而不必输出可执行文件

    调试过程
    1.设置断点           (gdb) b 行号  //可以设置多个断点
                    (gdb) b(reak) <funcname>
      查看断点           (gdb) info b
      禁用断点           (gdb) disable b 断点号
      再次激活断点     (gdb) enable breakpoints 断点号
      删除断点           (gdb) delete breakpoints 断点号
      清除断点        (gdb) clear 断点所在行号
      设置条件断点    (gdb) condition 1 <svalues[counter]->chechsum>700>
      临时断点        (gdb) tbreak <41>
    2.使用gdb单步调试程序  
                     (gdb) r          //在断点处停下
                     (gdb) n          //单步运行,不进入函数内部
                     (gdb) s(tep)     //一次一行
                     (gdb) c           //停止单步调试,继续执行直至断点或者输入
                     (gdb) cont <n>     //执行n次continue
                     (gdb) finish     //退出函数
    3.跟踪程序变量
                    (gdb) p buffer         //打印buffer的值
                    (gdb) p strlen(buffer)-1
                    (gdb) p $ (n)        //打印当前序号(或序号n)的前一序号变量的值
                    (gdb) p $$ (n)        //打印序号n-2的变量的值
      使用print给变量赋值
                      (gdb) p 变量=表达式
      打印内存某个部分开始的一块连续空间的内容
                      (gdb) p 开始表达式@要打印的空间大小  例:(gdb) p *array@len

      显示各种变量的值以及内存地址
                    (gdb) p pStr[0]
                    (gdb) p *(pStr[0]+1)
                    (gdb) p &pStr[0]
      printf查看变量
                      (gdb) printf "%2.2s\n",(char*)<addr>
      set修改变量的值
                      (gdb) set variable svalues[0]->checksum=2000

    4.自动显示指定变量的值
                    (gdb) display pStr_index
                    (gdb) display/<fmt> <expr>
                    (gdb) display/<fmt> <addr>
                    (gdb) info display
                    (gdb) disable d 2
                    (gdb) enable display 2
                    (gdb) delete diaplay 2

    5。查看函数堆栈            //程序调用一个函数时,函数的地址/参数/局部变量都会被压入栈中
                    (gdb) bt (backtrace)    //打印当前的函数调用栈的所有信息
                    (gdb) bt <n>             //打印前n层的栈顶信息
                    (gdb) bt <-n>             //打印栈底的前n层信息
                    (gdb) f(rame) <n>            //查看栈上某层信息(栈的层编号,当前函数名,函数参数值,函数所在文件及行号) frame 0 栈顶
                    (gdb) up <n>
                    (gdb) down <n>
                    (gdb) select-frame <n>     //对应frame,但是不输出栈层信息
                    (gdb) up-silently <n>
                    (gdb) down-silently <n>
                    (gdb) info f(rame)         //显示更为详细的当前栈层的信息
                    (gdb) info args            //显示当前函数的参数名及值
                    (gdb) info locals          //显示当前函数中所有局部变量及值
                    (gdb) info catch         //显示当前函数中的异常处理信息
                    
    6.                 (gdb) make
                    (gdb) shell
                    (gdb) q   //退出gdb调试
    7.代码显示
                    (gdb) where                 //查找错误位置
                    (gdb) l(ist) <n>            //查看当前行后面的代码<显示第n行附近的代码>
                    (gdb) l(ist) -                //查看当前行前面的代码
                    (gdb) l     <function>        //显示函数的源程序
                    (gdb) l     test.c:9         //查看test.c第九行代码
                    (gdb) set     listsize <count>    //设置显示行数
                    (gdb) show     listsize            //查看当前设置的显示的行数
                    (gdb) l     <first> <last>        //显示从first行到last行的源代码
                    (gdb) l     , <last>            //显示从当前行到last行的源代码
                    (gdb) l     +                    //向后显示代码
                    (gdb) forward-search <regexp>    //向前搜索,regexp是正则表达式
                    (gdb) reverse-search <regexp>    //全部搜索

    8.查看内存命令(x命令)
                    (gdb) x/<n/f/u> <addr>  //n、f、u是可选的参数 详见:(gdb) help x
                        u 表示从当前地址往后请求的字节数,如果不指定的话,GDB默认是4个bytes。u参数可以用下面的字符来代替,b表示单字节,h表示双字节,w表示四字 节,g表示八字节。当我们指定了字节长度后,GDB会从指内存定的内存地址开始,读写指定字节,并把其当作一个值取出来。
                        f 表示显示的格式,参见下面。如果地址所指的是字符串,那么格式可以是s,如果地十是指令地址,那么格式可以是i。
                        n 是一个正整数,表示需要显示的内存单元的个数,也就是说从当前地址向后显示几个内存单元的内容,一个内存单元的大小由后面的u定义。
                        例:命令:x/3uh 0x54320 表示,从内存地址0x54320读取内容,h表示以双字节为一个单位,3表示输出三个单位,u表示按十六进制显示。


    9.使用观察窗口
                    (gdb) watch <counter>15>  //满足条件后程序运行终止

    10.指定源文件的路径
                    dir <dirname>            //多个路径使用:(UNIX ) ;(WINDOW)
                    directory                //清除所有的自定义的源文件搜索路径信息
                    show directory             //显示定义了的源文件搜索路径

 11.查看运行时数据
        (gdb) print <expr>
        (gdb) print /<f>     <expr>    //  <f> 输出格式 <expr> 表达式
                    x        @        //与数组有关
                    d        ::        //指定一个在文件或一个函数中的变量
                    u        {<type>} <addr>        //表示一个指向内存地址<addr>的类型为type的一个对象
                    o
                    t
                    a
                    c
                    f

        (gdb) print file::variable            //查看文件中的全局变量
        (gdb) print function::variable        //查看函数中的局部变量 例: (gdb) p 'f2.c'::x
    
    12.设置显示选项   
        (gdb) set print address
        (gdb) set print address on          //打开函数地址参数显示
        (gdb) f                             //显示栈信息
        (gdb) set print address off         //关闭函数地址参数显示
        (gdb) show print address

        (gdb) set print array                 
        (gdb) set print array on            //打开数组显示
        (gdb) show print array

        (gdb) set print elements <num-of-elements>     //对大数组,显示部分长度
        (gdb) show print elements             //查看print elements的选项信息

        (gdb) set print null-stop <on/off>  //当显示字符串时,遇到结束符则停止显示。选项默认为off.

        (gdb) set print pretty on             //结构体完美显示
        (gdb) set print pretty off
        (gdb) show print pretty             //查看gdb如何显示结构体

        (gdb) set print sevenbit-strings <on/off>    //设置字符显示是否按照\nnn的格式显示
        (gdb) show print sevenbit-strings            //查看字符显示开关是否打开
        (gdb) set print union <on/off>                 //设置显示结构体时,是否显示内部的联合体结构
        (gdb) show print union

        (gdb) set print object <on/off>             //当一个对象指针指向其派生类时,如果打开这个选项,gdb自动按照虚方法调用的规则显示输出,关闭后,gdb就不管虚函数表了。默认为off
        (gdb) show print object

        (gdb) set print static-members <on/off>      //显示C++对象的静态数据成员。默认是on
        (gdb) show print static-members                

        (gdb) set print ytbl <on/off>                //使用规整的格式显示虚拟函数表。默认是关闭的
        (gdb) show print vtbl                        //查看虚函数显示格式的选项

    13.gdb环境变量
        可以在gdb的调试环境中定义自己的变量,用来保存一些调试程序中的运行数据。
        (gdb) set $foo=*object_ptr
        (gdb) show convenience                 //查看当前所设置的所有的环境变量
        


    14.查看寄存器
        (gdb) info registers
        (gdb) info all-registers
        (gdb) info registers <regname>         //查看指定寄存器的情况
        (gdb) p $<regname>                    //打印寄存器的内容


    15.改变程序的执行
        修改变量的值
        (gdb) p x=4
        (gdb) whatis width                    //gdb中的参数width与程序中的变量width重名的话,使用var指定为程序的 变量名
        (gdb) p width        
        (gdb) set var width=47                

        跳转执行(最好在同一个函数中跳转)
        jump命令的实质是改变了pc寄存器的值 set $pc= 0x...
        (gdb) jump <linespec>                //<linespec>可以是行号/file:line/+num
        (gdb) jump <address>                //代码行的内存地址

        产生信号量
        (gdb) signal <signal>                //linux的系统信号量通常是1到15

        强制函数返回(详见《Linux环境下C编程指南》P(82+12))

        强制调用函数


    16.在不同语言中使用gdb
        以下命令主要用于C以外的程序语言
        (gdb) show language             //查看当前语言环境
        (gdb) info frame                 //查看当前函数的程序语言
        (gdb) info source                 //查看当前文件的程序语言
        (gdb) set language <languagename>


    17.core dump分析
        程序崩溃原因分析
        $ulimit -c unlimited             //获得core dump使用权限

    $ gcc -ggdb3 ...

    $ ./a.out

    $ gdb a.out core

 

 

case12:

	typedef enum (Tree,Bug) Species;
	typedef enum (Big_tree, Acorn, Seedling) Tree_forms;
	typedef enum (Caterpillar, Cocoon, Butterfly) Bug_forms;
	struct thing 
	{
		Species it;
		union
		{
			Tree_forms tree;
			Bug_forms bug;
		}form;
	};
	
	struct thing foo = {Tree, {Acorn}}
	
	(gdb) set print union on
	(gdb) p foo
	$1={it=Tree, form={tree=Acorn, bug=Cocoon}}
	(gdb) set print union off
	(gdb) p foo
	$1={it=Tree, form={...}}

 case13:

        (gdb) set $i=0
	(gdb) print bar[$i++]->contents     

 

posted on 2015-03-25 19:46  kwseeker  阅读(307)  评论(0编辑  收藏  举报