1. 使用gdbserver調(diào)試
使用JDWP只能調(diào)試java層面的程序,如果想調(diào)試C層面的代碼,需要使用gdbserver方式,gdbserver的服務(wù)端和客戶端都包含在android的源碼中。
server端是out/target/product/xxxxxx/system/bin/gdbserver。
client端是prebuild/linux-x86/toolchain/xxxxxx/bin/arm-eabi-gdb),不需另外安裝。
(請(qǐng)看完本文再開始調(diào)試,尤其是“注意”部分)
本文依據(jù)張博的調(diào)試文檔, 加以擴(kuò)充說明,感謝原創(chuàng)者。
2. 調(diào)試前的準(zhǔn)備:編譯DEBUG版本的程序和庫
1) 新建(或修改)ANDROID源碼根目錄的buildspec.mk,加入以下內(nèi)容
DEBUG_MODULE_lidvm:=true # 虛擬機(jī)模塊設(shè)為debug
TARGET_CUSTOM_DEBUG_CFLAGS:=-O0 -mlong-calls
(請(qǐng)修改具體模塊名,我調(diào)試的是虛擬機(jī)的libdvm.so庫)
2) 重編dalvik模塊
$ make clean-libdvm
$ make dalvik snod
3) 重?zé)?/span>system.img或替換手機(jī)中的相應(yīng)模塊
3. gdb server端配置
1) 端口映射
$ adb forward tcp:5039 tcp:5039 把設(shè)備的5039端口映射到PC的5039
設(shè)定之后用netstat -na命令可看到PC的5039端口已處于listen狀態(tài)
注意每次斷開手機(jī)再連接時(shí),都要重新執(zhí)行該命令
2) 調(diào)試進(jìn)程號(hào)為2014進(jìn)程
$ adb shell
# ps 找進(jìn)程號(hào)
# gdbserver :5039 --attach 2014 # 指明tcp端口號(hào)和進(jìn)程號(hào)
注意:用此方法只適用于對(duì)已運(yùn)行的程序debug(不能使用直接在gdbserver后跟程序名的方式運(yùn)行)
此時(shí)2014進(jìn)程被掛起,等待調(diào)試
4. gdb client端配置
1) 用命令行工具調(diào)試
$ $ANDROID_DIR/prebuilt/linux-x86/toolchain/xxxxxx/bin/arm-eabi-gdb $ANDROID_DIR/out/target/product/xxxxxx/system/bin/app_process
注意可執(zhí)行程序名必須是app_process,不是你所調(diào)試的程序名
(gdb) set solib-absolute-prefix $ANDROID_SRC)/out/target/product/xxxxxx/symbols/system/lib/
(gdb) set solib-search-path $ANDROID_SRC)/out/target/product/xxxxxx/symbols/system/lib/
以上路徑為GDB默認(rèn)庫的搜索路徑,即交叉編譯器庫路徑,若不設(shè)定,則找不到符號(hào)表,(帶符號(hào)表的庫在symbols/system/lib/*,手機(jī)里strip后無符號(hào)表的庫在system/lib/*,它們必須配套使用)
(gdb) target remote :5039 指明TCP端口號(hào)
此時(shí)連接gdb server,可設(shè)斷點(diǎn)調(diào)試,按c繼續(xù)執(zhí)行程序
2) 用eclipse調(diào)試
a) 安裝cdt,使eclipse支持c/c++程序的開發(fā)
i. 下載
從http://www.eclipse.org/cdt/downloads.php下載cdt-master-4.0.0.zip
ii. 解壓
$ mkdir cdt; cd cdt; unzip ../cdt-master-4.0.0.zip
iii. 將解壓縮后的features、plugins兩個(gè)文件夾的內(nèi)容復(fù)制到Eclipse安裝目錄中
$ cp plugins/* ../../eclipse/plugins/
$ cp features/* ../../eclipse/features/
iv. 重新開啟Eclipse即可
$ eclipse -clean
在新建project中即可看到c/c++相關(guān)選項(xiàng),說明已安裝成功
b) 加入要調(diào)試的代碼
i. 新建C++ project (菜單File->New->Project…)
不使用default location,把Location指定成代碼所在目錄
ii. 取消自動(dòng)編譯選項(xiàng)(菜單Project->Build Automatically)
c) 配置gdb環(huán)境
配置Debug Configurations(菜單Run->Debug Configurations…)
i. 新建一個(gè)C/C++ Local Application的debug configuration
ii. Main選項(xiàng)卡中
指定Project為新建的C++工程,
C/C++ Applications為:
$ANDROID_DIR/out/target/product/xxxxxx/system/bin/app_process
iii. Debugger選項(xiàng)卡中
指定Debugger為gdbserver Debugger,
Main子選項(xiàng)卡的Gdb debuger設(shè)為:
$ANDROID_DIR/prebuilt/linux-x86/toolchain/xxxxxx/bin/arm-eabi-gdb
GDB command file設(shè)為一個(gè)文件名,文件內(nèi)容如下:
file $ANDROID_DIR/out/target/product/xxxxxx/system/bin/app_process
set solib-absolute-prefix $ANDROID_SRC)/out/target/product/xxxxxx/symbols/system/lib/
set solib-search-path $ANDROID_SRC)/out/target/product/xxxxxx/symbols/system/lib/
Connection子選項(xiàng)卡:
Type設(shè)為TCP,Port number設(shè)為5039
iv. 點(diǎn)擊Debug按鈕進(jìn)入調(diào)試,之前掛起的程序此時(shí)繼續(xù)運(yùn)行
d) 設(shè)置斷點(diǎn)及調(diào)試
i. 找開某一C程序(菜單->Open file)
ii. 在程序中雙擊可設(shè)置斷點(diǎn),設(shè)置后斷點(diǎn)出現(xiàn)在右上的Breakpoints中
iii. Debug選項(xiàng)卡提供了工具調(diào)試(suspend, resume等)
5. 加打印語句
如果需要在C程序中加打印語句,有兩種方法
1) 直接在代碼中使用printf,此方法只能應(yīng)用于從命令行啟動(dòng)程序的情況,運(yùn)行時(shí)可以adb shell中看到打印信息
2) 使用程序中提供的重定項(xiàng)后的打印語句,并在logcat中看到它
例如在libdvm.so中使用dvmFprintf(stderr, “xieyan log\n”);
6. 可能出現(xiàn)的問題及解決方法
1) 在找不到原因時(shí),可以寫一個(gè)在android可以運(yùn)行的簡(jiǎn)單c語言程序用gdbserver調(diào)試,以簡(jiǎn)化問題,android中c程序做法見:
http://www.top-e.org/jiaoshi/html/?157.html
2) 我的是在arm-eabi-2.4.1的編譯器編出來的,你的可能不是,編譯時(shí)用make showcommands 確定你的系統(tǒng)使用的編譯工具鏈,否則如果你debug時(shí)用的和編譯時(shí)用的版本不一致,會(huì)導(dǎo)致讀符號(hào)表時(shí)出錯(cuò)(注意看提示)
3) 有時(shí)編譯會(huì)引起源碼目錄的變化,請(qǐng)?jiān)谧髠?cè)Project explorer中刷新相關(guān)項(xiàng)目
(轉(zhuǎn)載請(qǐng)注明出處:http://xy0811.spaces.live.com)