新聞中心

        EEPW首頁 > 嵌入式系統 > 學習方法與實踐 > gdbserver調試共享庫

        gdbserver調試共享庫

        ——
        作者:absurd 時間:2008-01-11 來源:電子產品世界 收藏

          在開發嵌入式系統時,調試往往是一大難題。面試過不少嵌入式linux工程師,當問及調試手段時,他們的調試手段一般是兩種:首先是在PC上的模擬環境中運行,若有問題,可以很方便的調試。其次,若在板子上運行時才出錯,就用printf輸出log信息,根據log信息定位錯誤。有少部分人用gdbserver調試板子上的程序,但問到如何在共享庫里設置斷點時,都說沒有辦法。

        本文引用地址:http://www.104case.com/article/76986.htm

          去年,Tinyx的一個內存越界BUG,花了我2天時間。gcc的一個浮點數BUG讓我查了3天時間。這類BUG在PC上根本重現不了,在板子上用printf要花費大量的時間才能把錯誤的范圍縮小一點。后來想了想,與其花時間去加printf,還不如把gdbserver調試共享庫的問題解決了,可以為以后的調試節省不少時間。

          在網上找了半天資料,沒有什么收獲,看來只好自己動手研究。花了一個周末的時間去研究gdbserver的運行方式。辦法是找到了,不過仍然有點麻煩,等有時間了,修改一下gdb的代碼,把這個過程自動化了。

          先調試運行gdbserver,對gdbserver有了一些感性認識,然后研究linux-low.c中的代碼。原來,設置斷點只是在對應的內存中寫入斷點指令(x86上為0xcc)。

          gdbserver為什么不能在共享庫中設置斷點呢?設置斷點只是寫內存,調試時,所有的代碼段都是可寫的,在exe中可以設置斷點,沒有理由不讓在共享庫中設置啊。所以這應該與是否是共享庫關系不大。

          猜測可能是符號與地址對應關系有誤,如果你的本意為function1設置斷點,結果gdb搞錯了,設置一個毫不相干的地方,可能永遠都不會執行到那里,這個斷點自然沒什么效果。

          如果是這樣,有兩種方法可以解決:要么手動計算符號的地址,再設置斷點,當然這樣太累。另外就讓gdb自動對應起來。經過反得嘗試,用下列方法可以在共享庫中設置斷點,雖然有點麻煩,還是可行的。

        1.         準備工作,編寫下面幾個文件:

        test.c:

        #include

        int test(int a, int b)

        {
            int s = a + b;

            printf("%dn", s);

            return s;
        }

        main.c:

        #include

        extern int test(int a, int b);

        int main(int argc, char* argv[])

        {
            int s = test(10, 20);
            return s;
        }

        Makefile:

        all: so main

        so:

            gcc -g test.c -shared -o libtest.so

        main:

            gcc -g main.c -L./ -ltest -o test.exe

        clean:

        rm -f *.exe *.so

        (為了便于演示,整個過程在PC上測試,后來證實在實驗板上能夠正常工作)

        2.         編譯并設置環境變量

        make

        export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:./

        3.         運行gdbserver

        gdbserver localhost:2000 ./test.exe

        4.         運行gdb客戶端

        gdb

        symbol-file test.exe

        target remote localhost:2000

        b main

        c

        5.         查看libtest.so的代碼在內存中的位置。

        (從gdbserver的輸出或者用ps可以得到test.exe的進程ID,這里假設PID是11547)

        cat /proc/11547/maps

        輸出:

        00624000-0063e000 r-xp 00000000 03:01 718192     /lib/ld-2.3.5.so

        0063e000-0063f000 r-xp 00019000 03:01 718192     /lib/ld-2.3.5.so

        0063f000-00640000 rwxp 0001a000 03:01 718192     /lib/ld-2.3.5.so

        00642000-00766000 r-xp 00000000 03:01 718193     /lib/libc-2.3.5.so

        00766000-00768000 r-xp 00124000 03:01 718193     /lib/libc-2.3.5.so

        00768000-0076a000 rwxp 00126000 03:01 718193     /lib/libc-2.3.5.so

        0076a000-0076c000 rwxp 0076a000 00:00 0

        00bbe000-00bbf000 r-xp 00bbe000 00:00 0

        00fcc000-00fcd000 r-xp 00000000 03:01 1238761    /root/test/gdbservertest/libtest.so

        00fcd000-00fce000 rwxp 00000000 03:01 1238761    /root/test/gdbservertest/libtest.so

        08048000-08049000 r-xp 00000000 03:01 1238765    /root/test/gdbservertest/test.exe

        08049000-0804a000 rw-p 00000000 03:01 1238765    /root/test/gdbservertest/test.exe

        b7f8a000-b7f8b000 rw-p b7f8a000 00:00 0

        b7f99000-b7f9a000 rw-p b7f99000 00:00 0

        bfd85000-bfd9a000 rw-p bfd85000 00:00 0          [stack]

        由此可以知道:libtest.so的代碼在00fcc000-00fcd000之間。

        6.         查看libtest.so的.text段在內存中的偏移位置:

        objdump -h libtest.so |grep .text

        輸出:

        9 .text         00000130  00000450  00000450  00000450  2**4

        即偏移位置為0x00000450

        7.         回到gdb窗口,加載libtest.so的符號表。

        add-symbol-file libtest.so 0x00fcc450

        (這里0x00fcc450 = 0x00fcc000 + 0x00000450)

        8.         在共享庫的函數中設置斷點。

        b test

        9.         繼續調試,可以發現在共享庫中設置的斷點,能夠正常工作。

         



        關鍵詞:

        評論


        相關推薦

        技術專區

        關閉
        主站蜘蛛池模板: 林甸县| 乌审旗| 英山县| 南开区| 永川市| 政和县| 泰来县| 涟水县| 宣汉县| 兴文县| 陕西省| 沐川县| 青冈县| 呼伦贝尔市| 平谷区| 鄂伦春自治旗| 台州市| 灵石县| 依兰县| 阿拉善右旗| 商都县| 贵州省| 长汀县| 平山县| 比如县| 东乡县| 新平| 会昌县| 曲水县| 闵行区| 定结县| 固安县| 阳新县| 平凉市| 金门县| 汉中市| 洛南县| 石景山区| 阿拉善右旗| 寻乌县| 漳州市|