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