調(diào)試Android上的c/c++程序一直是個(gè)難題,以前我經(jīng)常靠輸出log來解決問題,對于稍復(fù)雜一些的工程,這幾乎是個(gè)不可能完成的任務(wù),尤其有些錯(cuò)誤,在wincewindows下都沒事,只在android上出現(xiàn),就更難找了。后來看了些資料,知道可以用gdbserver來調(diào)試,今天決定必須把這個(gè)先弄清楚,不然以后干活效率實(shí)在是太低了,找了很多網(wǎng)站,終于成功了。這里把整個(gè)過程整理一下,以備以后查閱。
1. 準(zhǔn)備gdbserver。
android 1.0 代碼剛開放到時(shí)候,里面并沒有帶gdbserver,有些強(qiáng)人就自己編譯了gdbserver來使用。不過現(xiàn)在好了,android的新源碼里已經(jīng)包含了gdbserver,就在prebuilt目錄下。如果想在android 1.0里使用,可以到如下地址下載:
http://android.git.kernel.org/?p=platform/prebuilt.git;a=tree。gdbserver的二進(jìn)制文件就在android-arm/gdbserver/gdbserver,我們只需要把gdbserver這個(gè)可執(zhí)行文件放到模擬器上即可。
準(zhǔn)備把它放在/system/bin,這樣它就在默認(rèn)的PATH里了。啟動模擬器后:
adb push gdbserver /system/bin
如果提示文件系統(tǒng)不可寫的話,執(zhí)行一下“adb remount”,因?yàn)槟J(rèn)是用只讀模式掛載的。
2. 編譯程序。
默認(rèn)情況下,android的編譯系統(tǒng)在編譯程序時(shí)已經(jīng)使用了“-g”選項(xiàng),即已經(jīng)生成了調(diào)試信息。但是在生成最終的文件時(shí),是經(jīng)過strip的,去除了所有到調(diào)試信息。所以最終我們到調(diào)試目標(biāo)要使用strip之前的文件。用make showcommands查看詳細(xì)的命令信息,下面是其中某次的輸出:
target Executable: libomstts (out/target/product/generic/obj/EXECUTABLES/libomstts_intermediates/LINKED/libomstts)
prebuilt/linux-x86/toolchain/arm-eabi-4.2.1/bin/arm-eabi-g++ -nostdlib -Bdynamic -Wl,-T,build/core/armelf.x -Wl,-dynamic-linker,/system/bin/linker -Wl,--gc-sections -Wl,-z,nocopyreloc -o out/target/product/generic/obj/EXECUTABLES/libomstts_intermediates/LINKED/libomstts -Lout/target/product/generic/obj/lib -Wl,-rpath-link=out/target/product/generic/obj/lib -llog -lcutils -lutils -landroid_runtime -lnativehelper -lstdc++ -lz -lc -lstdc++ -lm out/target/product/generic/obj/lib/crtbegin_dynamic.o out/target/product/generic/obj/EXECUTABLES/libomstts_intermediates/src/tts.o out/target/product/generic/obj/EXECUTABLES/libomstts_intermediates/src/TTSWrapper.o out/target/product/generic/obj/lib/crtend_android.o
target Non-prelinked: libomstts (out/target/product/generic/symbols/system/bin/libomstts)
out/host/linux-x86/bin/acp -fpt out/target/product/generic/obj/EXECUTABLES/libomstts_intermediates/LINKED/libomstts out/target/product/generic/symbols/system/bin/libomstts
target Strip: libomstts (out/target/product/generic/obj/EXECUTABLES/libomstts_intermediates/libomstts)
out/host/linux-x86/bin/soslim --strip --shady --quiet out/target/product/generic/symbols/system/bin/libomstts --outfile out/target/product/generic/obj/EXECUTABLES/libomstts_intermediates/libomstts
Install: out/target/product/generic/system/bin/libomstts
out/host/linux-x86/bin/acp -fpt out/target/product/generic/obj/EXECUTABLES/libomstts_intermediates/libomstts out/target/product/generic/system/bin/libomstts
生成的可執(zhí)行文件是libomstts,可以看到,初次鏈接的目標(biāo)文件是“out/target/product/generic/obj/EXECUTABLES/libomstts_intermediates/LINKED/libomstts”,然后拷貝到“out/target/product/generic/symbols/system/bin/libomstts”,strip后的文件是“out/target/product/generic/obj/EXECUTABLES/libomstts_intermediates/libomstts”和“out/target/product/generic/system/bin/libomstts”。調(diào)試只能使用前兩個(gè)文件。
把帶調(diào)試信息到可執(zhí)行文件放到模擬器上,我用到是“out/target/product/generic/symbols/system/bin/libomstts”:
adb push out/target/product/generic/symbols/system/bin/libomstts /system/bin
3. 啟動調(diào)試器
首先在模擬器上啟動gdbserver:
adb shell
進(jìn)入模擬器的控制臺后
gdbserver 10.0.2.2:1234 /system/bin/libomstts
10.0.2.2是模擬器的默認(rèn)ip地址,讓gdbserver在模擬器上監(jiān)聽1234端口。如果啟動成功會顯示以下信息:
Process /system/bin/libomstts created; pid = 1025
Listening on port 1234
為來讓gdb能連接到模擬器上到gdbserver,必須進(jìn)行數(shù)據(jù)轉(zhuǎn)發(fā):
telnet localhost 5554
Trying 127.0.0.1...
Connected to localhost.
Escape character is '^]'.
Android Console: type 'help' for a list of commands
OK
redir add tcp:1234:1234
OK
exit
上面的telnet localhost 5554,redir add tcp:1234:1234,exit是自己輸入的命令,其他的是輸出信息。5554是模擬器控制臺的監(jiān)聽端口,這些命令是將所有到localhost:1234的數(shù)據(jù)轉(zhuǎn)發(fā)到模擬器的1234端口。
最后在本機(jī)啟動gdb:
arm-eabi-gdb out/target/product/generic/symbols/system/bin/libomstts
arm-eabi-gdb是android自帶的toolchain里的,注意后面的可執(zhí)行文件是strip之前的。
gdb啟動后,在gdb里輸入命令連接gdbserver:
target remote localhost:1234
連接到gdbserver成功后,就可以使用所有的gdb調(diào)試命令啦
現(xiàn)在的這個(gè)gdbserver還不能調(diào)試動態(tài)鏈接庫,只能先編譯成可執(zhí)行文件調(diào)試。