Debug Native c/c++ Application for Android(Step by Step)
2013-05-30 19:32
627 查看
GDB 调试 android 本地 c/c++ 程序
有一段时间没弄Android了,整理一下,把之前 用GDB调试 Android jni程序过程共享一下:Android 应用层为JAVA语言,但有时需要C/C++实现一些特定功能,比如视频编解码,此时需要借助JAVA的JNI技术,用JAVA语言调用c/C++程序,
而C/C++程序库 调试通常用GDB,不例外,Android 平台也提供了调试 JNI的功能,步骤如下:
1. 把
android:debuggable="true" 设置到
AndroidManifest.xml
文件的
<application>标签,
2. NDK-BUILD 执行编译动态库时时 后面带上 NDK_DEBUG=1
3. Application.mk
文件中 添加
APP_OPTIM := debug
4.
启动Android设备上的 gdb server : gdbserver
:5055 --attach PID
参数为监听端口 5055, 要调试进程号 PID,
5.
在 宿主机上 执行 adb forward tcp:5055
tcp:5055
连接到GDB server.
6. 启动客户端 GBD :
arm-linux-androideabi-gdb
7.
设置库路径:
set
solib-search-path obj/local/armeabi/libMySharedLib.so
target
remote :5055
哦了,进入调试
过程吧 :)
---------------------------------------------------------------------------------------------------------------------------------------------------
调试Android上的c/c++程序一直是个难题,以前我经常靠输出log来解决问题,对于稍复杂一些的工程,这几乎是个不可能完成的任务,尤其有些错误,在wincewindows下都没事,只在android上出现,就更难找了。后来看了些资料,知道可以用gdbserver来调试,今天决定必须把这个先弄清楚,不然以后干活效率实在是太低了,找了很多网站,终于成功了。这里把整个过程整理一下,以备以后查阅。
1. 准备gdbserver。
android 1.0 代码刚开放到时候,里面并没有带gdbserver,有些强人就自己编译了gdbserver来使用。不过现在好了,android的新源码里已经包含了gdbserver,就在prebuilt目录下。如果想在android 1.0里使用,可以到如下地址下载:http://android.git.kernel.org/?p=platform/prebuilt.git;a=tree。gdbserver的二进制文件就在android-arm/gdbserver/gdbserver,我们只需要把gdbserver这个可执行文件放到模拟器上即可。
准备把它放在/system/bin,这样它就在默认的PATH里了。启动模拟器后:
adb push gdbserver /system/bin
如果提示文件系统不可写的话,执行一下“adb remount”,因为默认是用只读模式挂载的。
2. 编译程序。
默认情况下,android的编译系统在编译程序时已经使用了“-g”选项,即已经生成了调试信息。但是在生成最终的文件时,是经过strip的,去除了所有到调试信息。所以最终我们到调试目标要使用strip之前的文件。用make showcommands查看详细的命令信息,下面是其中某次的输出:
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
生成的可执行文件是libomstts,可以看到,初次链接的目标文件是“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”。调试只能使用前两个文件。
把带调试信息到可执行文件放到模拟器上,我用到是“out/target/product/generic/symbols/system/bin/libomstts”:
adb push out/target/product/generic/symbols/system/bin/libomstts /system/bin
3. 启动调试器
首先在模拟器上启动gdbserver:
adb shell
进入模拟器的控制台后
gdbserver 10.0.2.2:1234 /system/bin/libomstts
10.0.2.2是模拟器的默认ip地址,让gdbserver在模拟器上监听1234端口。如果启动成功会显示以下信息:
Process /system/bin/libomstts created; pid = 1025
Listening on port 1234
为来让gdb能连接到模拟器上到gdbserver,必须进行数据转发:
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是模拟器控制台的监听端口,这些命令是将所有到localhost:1234的数据转发到模拟器的1234端口。
最后在本机启动gdb:
arm-eabi-gdb out/target/product/generic/symbols/system/bin/libomstts
arm-eabi-gdb是android自带的toolchain里的,注意后面的可执行文件是strip之前的。
gdb启动后,在gdb里输入命令连接gdbserver:
target remote localhost:1234
连接到gdbserver成功后,就可以使用所有的gdb调试命令啦
现在的这个gdbserver还不能调试动态链接库,只能先编译成可执行文件调试。
相关文章推荐
- Debug Native c/c++ Application for Android(Step by Step)
- Debug Native c/c++ Application for Android(Step by Step)
- Debug Native c/c++ Application for Android(Step by Step)
- Debug Native c/c++ Application for Android(Step by Step)
- Step by step for a helloworld android application
- Program for Android in C/C++ with the Native Development Kit (if you dare)
- A step-by-step guide for debugging native code (Sequoyah/ndk guide)
- Android—Step by step - Compile kernel module for Samsung Galaxy S3 (Linux)
- Linux环境下android平台调试native代码, 从java debug 到C++ (NDK DEBUG)
- [Android]Error:Application and test application id cannot be the same: both are '' for debugAndroid
- Linux环境下android平台调试native代码, 从java debug 到C++ (NDK DEBUG)
- android debug:Unable to add window -- token null is not for an application
- [已解决]Android 5.1 上面进行debug的时候 debug 的界面总是不消失 一直显示 Application is waiting for the debugger to attach
- Step by Step - How to create a c++ library with NDK on Android Studio 1.5 (not experimental way)
- Linux环境下android平台调试native代码, 从java debug 到C++ (NDK DEBUG)
- utility for debugging android native application
- Windows环境下android平台native调试,从java debug 到 C++ (NDK DEBUG) .
- Error:Execution failed for task ':myapplication:processDebugResources'. > com.android.ide.common.pro
- Eclipse Debug Android Native Application
- How to debug Android Native Application with eclipse