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