• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>
            posts - 9, comments - 4, trackbacks - 0, articles - 9

            gdb調試過程

            Posted on 2009-03-11 17:49 xlz 閱讀(219) 評論(0)  編輯 收藏 引用
             
            GCC版本

              使用下面命令,可以知道它的版本:

            gcc -v

            GCC安裝后目錄結構

              /usr/lib/gcc-lib/target/version/ (及子目錄) 編譯器就在這個目錄下。

              /usr/bin/gcc可以從命令行執行的二進制程序在這個目錄下。

              /usr/target/(bin|lib|include)/ 庫和頭文件在這個目錄下。

              /lib/,/usr/lib和其他目錄,系統的庫在這些目錄下。

            符號定義

              使用-V開關,就能看到GCC定義的符號。參見下列實例:

              $ echo 'main(){printf("hello world");}' | gcc -E -v -

               Reading specs from /usr/lib/gcc-lib/i486-box-linux/2.7.2/specs

               gcc version 2.7.2

              /usr/lib/gcc-lib/i486-box-linux/2.7.2/cpp -lang-c -v -undef

               -D__GNUC__=2 -D__GNUC_MINOR__=7 -D__ELF__ -Dunix -Di386 -Dlinux

              -D__ELF__ -D__unix__ -D__i386__ -D__linux__ -D__unix -D__i386

              -D__linux -Asystem(unix) -Asystem(posix) -Acpu(i386)

              -Amachine(i386) -D__i486__ -

            GCC編譯器使用簡介

              通常后跟一些選項和文件名來使用 GCC 編譯器。gcc 命令的基本用法如下:

              gcc [options] [filenames]

              選項指定編譯器怎樣進行編譯。

            GCC選項

              GCC 有100個編譯選項。這些選項中的許多可能永遠都不會用到,但一些主要的選項會經常遇到。很多的 GCC

            選項包括一個以上的字符,因此必須為每個選項指定各自的連字符。例如, 下面的兩個命令是不同的:

              gcc -p -g test.c

              gcc -pg test.c

              第一條命令告訴 GCC 編譯 test.c 時為 prof 命令建立剖析(profile)信息并且把調試信息加入到可執行的文件里。 第二條命令只告訴 GCC

            為 gprof 命令建立剖析信息。

              沒有選項時,GCC 會生成一個名為 a.out 的可執行文件。

              用 -o 編譯選項來為將產生的可執行文件用指定的文件名來命名。例如, 將一個叫 count.c 的 C 程序編譯為名叫 count 的可執行文件,

            要這樣輸入命令:

              gcc -o count count.c

              -c 選項告訴 GCC 僅把源代碼編譯為目標代碼。缺省時 GCC 建立的目標代碼文件有一個 .o 的擴展名。

              -S 編譯選項告訴 GCC 在為 C 代碼產生了匯編語言文件后停止編譯。 GCC 產生的匯編語言文件的缺省擴展名是 .s 。

              -E 選項指示編譯器僅對輸入文件進行預處理。當這個選項被使用時, 預處理器的輸出被送到標準輸出而不是儲存在文件里.

              用 GCC 編譯 C 代碼時, 它會試著用最少的時間完成編譯并且使編譯后的代碼易于調試。

            易于調試意味著編譯后的代碼沒有經過優化。必要時,需要讓編譯器對代碼進行優化。

              -O 選項告訴 GCC 對源代碼進行基本優化。這些優化在大多數情況下都會使程序執行的更快。 -O2 選項告訴 GCC 產生盡可能小和盡可能快的代碼。 -O2

            選項將使編譯的速度比使用 -O 時慢, 但通常產生的代碼執行速度會更快。

              GCC 支持數種調試和剖析選項,常用到的是 -g 和 -pg 。

              -g 選項告訴 GCC 產生能被 GNU 調試器使用的調試信息以便調試你的程序。GCC 提供了一個很多其他 C 編譯器里沒有的特性, 在 GCC 里你能使

            -g 和 -O (產生優化代碼)聯用。

              -pg 選項告訴 GCC 在編譯好的程序里加入額外的代碼。運行程序時, 產生 gprof 用的剖析信息以顯示你的程序的耗時情況。

            用 gdb 調試 GCC 程序

              Linux 包含了一個叫 gdb 的 GNU 調試程序。在程序運行時能觀察程序的內部結構和內存的使用情況。 以下是 gdb 所提供的一些功能:

              監視程序中變量的值

              設置斷點,使程序在指定的代碼行上停止執行。

              一行行的執行代碼

              為了用GDB調試程序,在編譯是必須指定調試選項。在命令行上鍵入 gdb 并按回車鍵就可以運行 gdb 了。如果一切正常的話, gdb

            將被啟動并在屏幕上顯示:

              GDB is free software and you are welcome to distribute copies of it under

            certain conditions; type "show copying" to see the conditions.

              There is absolutely no warranty for GDB; type "show warranty" for details.

              GDB 4.14 (i486-slakware-linux), Copyright 1995 Free Software Foundation, Inc.

              (gdb)

              可以在啟動GDB時,加入許多選項。也可以在這個命令后面直接指定要調試的程序。

            gdb < fname>

            gdb 基本命令

            gdb 支持很多的命令,這些命令從簡單的文件裝入到允許檢查所調用的堆棧內容的復雜命令。下表列出了你在用 gdb 調試時會用到的一些命令。

            命令 描 述

            file 裝入想要調試的可執行文件

            kill 終止正在調試的程序

            list 列出產生執行文件的源代碼的一部分

            next 執行一行源代碼但不進入函數內部

            step 執行一行源代碼而且進入函數內部

            run 執行當前被調試的程序

            quit 終止 gdb

            watch 使你能監視一個變量的值而不管它何時被改變

            break 在代碼里設置斷點, 這將使程序執行到這里時被掛起

            make 使你能不退出 gdb 就可以重新產生可執行文件

            shell 使你能不離開 gdb 就執行 UNIX shell 命令

            gdb 應用舉例

              下面列出了將被調試的程序,這個程序被稱為 greeting ,顯示一個簡單的問候, 再用反序將它列出。

            #include < stdio.h>

            main ()

            {

            char my_string[] = "hello there";

            my_print (my_string);

            my_print2 (my_string);

            }

            void my_print (char *string)

            {

            printf ("The string is %s", string);

            }

            void my_print2 (char *string)

            {

            char *string2;

            int size, i;

            size = strlen (string);

            string2 = (char *) malloc (size + 1);

            for (i = 0; i < size; i++)

            string2[size - i] = string[i];

            string2[size+1] = `0';

            printf ("The string printed backward is %s", string2);

            }

              用下面的命令編譯這個程序:

                gcc -o -g test test.c

              運行編譯好的程序,顯示如下:

                The string is hello there

                The string printed backward is

              輸出的第一行是正確的, 但第二行打印出的東西并不是我們所期望的。我們所設想的輸出應該是:

                The string printed backward is ereht olleh

              由于某些原因, my_print2 函數沒有正常工作。用 gdb 看看問題究竟出在哪兒, 先鍵入如下命令:

                gdb greeting

              如果在輸入命令時忘了把要調試的程序作為參數傳給 gdb ,可以在 gdb 提示符下用 file 命令來載入它:

                (gdb) file greeting

              這個命令載入 greeting 可執行文件,就象在 gdb 命令行里指定啟動gdb裝入它一樣。

              這時就能用 gdb 的 run 命令來運行 greeting 了。 當它在 gdb 里被運行后結果大約會象這樣:

              (gdb) run

              Starting program: /root/greeting

              The string is hello there

              The string printed backward is

              Program exited with code 041

              

              這個輸出和在 gdb 外面運行的結果一樣。問題是, 為什么反序打印沒有工作? 為了找出癥結所在, 我們可以在 my_print2 函數的 for

            語句后設一個斷點, 具體的做法是在 gdb 提示符下鍵入 list 命令三次, 列出源代碼:

              (gdb) list

              (gdb) list

              (gdb) list

              第一次鍵入 list 命令的輸出如下:   1 #include < stdio.h>

              2

              3 main ()

              4 {

              5 char my_string[] = "hello there";

              6

              7 my_print (my_string);

              8 my_print2 (my_string);

              9 }

              10

              如果按下回車, gdb 將再執行一次 list 命令, 給出下列輸出:

              11 my_print (char *string)

              12 {

              13 printf ("The string is %s", string);

              14 }

              15

              16 my_print2 (char *string)

              17 {

              18 char *string2;

              19 int size, i;

              20

              再按一次回車將列出 greeting 程序的剩余部分:

              21 size = strlen (string);

              22 string2 = (char *) malloc (size + 1);

              23 for (i = 0; i < size; i++)

              24 string2[size - i] = string[i];

              25 string2[size+1] = `0';

              26 printf ("The string printed backward is %s", string2);

              27 }

              根據列出的源程序, 你能看到要設斷點的地方在第24行, 在 gdb 命令行提示符下鍵入如下命令設置斷點:

                (gdb) break 24

              該命令的執行結果如下:

                Breakpoint 1 at 0x139: file greeting.c, line 24

                (gdb)

              現在再鍵入 run 命令, 將產生如下的輸出:

                Starting program: /root/greeting

                The string is hello there

                Breakpoint 1, my_print2 (string = 0xbfffdc4 "hello there") at greeting.c :24

            24 string2[size-i]=string[i]

              你能通過設置一個觀察 string2[size - i] 變量的值的觀察點來看出錯誤是怎樣產生的, 做法是鍵入:

                (gdb) watch string2[size - i]

              執行結果如下:

                Watchpoint 2: string2[size - i]

              現在可以用 next 命令來一步步的執行 for 循環了:

                (gdb) next

              經過第一次循環后, gdb 告訴我們 string2[size - i] 的值是 `h`。這是執行next命令后的結果:

              Watchpoint 2, string2[size - i]

              Old value = 0 `000'

              New value = 104 `h'

              my_print2(string = 0xbfffdc4 "hello there") at greeting.c:23

              23 for (i=0; i< size; i++)

              這個值正是期望的。后來的數次循環的結果都是正確的。當 i=10 時, 表達式 string2[size - i] 的值等于 `e`, size - i

            的值等于 1, 最后一個字符已經拷到新串里了。

              如果再把循環執行下去,會看到已經沒有值分配給 string2[0] 了, 而它是新串的第一個字符, 因為 malloc

            函數在分配內存時把它們初始化為空(null)字符。所以 string2 的第一個字符是空字符。于是就發現了為什么在打印 string2 時沒有任何輸出了.

              找出了問題出在哪里后, 修正這個錯誤是很容易的。把代碼里寫入 string2 的第一個字符的的偏移量改為 size - 1 而不是 size。這是因為

            string2 的大小為 12, 但起始偏移量是 0, 串內的字符從偏移量 0 到 偏移量 10, 偏移量 11 為空字符保留。

              為了使代碼正常工作有很多種修改辦法. 。一種是另設一個比串的實際大小小 1 的變量,下面是這種辦法的程序。

            #include < stdio.h>

            main ()

            {

            char my_string[] = "hello there";

            my_print (my_string);

            my_print2 (my_string);

            }

            my_print (char *string)

            {

            printf ("The string is %s", string);

            }

            my_print2 (char *string)

            {

            char *string2;

            int size, size2, i;

            size = strlen (string);

            size2 = size -1;

            string2 = (char *) malloc (size + 1);

            for (i = 0; i < size; i++)

            string2[size2 - i] = string[i];

            久久婷婷激情综合色综合俺也去| 久久青青草原亚洲av无码| 久久综合精品国产二区无码| 久久精品夜夜夜夜夜久久| 91久久九九无码成人网站 | 伊人久久大香线蕉AV色婷婷色| 欧美熟妇另类久久久久久不卡| 久久噜噜电影你懂的| 久久无码专区国产精品发布| 国产亚洲婷婷香蕉久久精品| 久久久久亚洲精品中文字幕| 色偷偷88888欧美精品久久久| 久久综合欧美成人| 久久精品国产2020| 国产精品热久久无码av| 久久无码人妻一区二区三区午夜 | 99久久精品免费| 7777久久久国产精品消防器材| 国产99久久久国产精品~~牛| 亚洲伊人久久成综合人影院| 欧美激情精品久久久久| 久久天天躁狠狠躁夜夜网站 | 欧美久久久久久午夜精品| 精品久久香蕉国产线看观看亚洲 | AA级片免费看视频久久| 久久久久亚洲AV片无码下载蜜桃 | 亚洲国产精品嫩草影院久久| 国产一区二区三区久久精品| 无码人妻久久一区二区三区免费丨 | 精品久久久久久久国产潘金莲| 日本精品久久久中文字幕 | 色妞色综合久久夜夜| 亚洲精品第一综合99久久| 色99久久久久高潮综合影院| 精品久久久久一区二区三区| 91精品国产综合久久久久久| 国产精品久久自在自线观看| 99久久精品费精品国产一区二区| 久久久精品国产sm调教网站| 久久精品中文闷骚内射| 久久国产精品-国产精品|