繁体   English   中英

如何在真实设备上调试Android本机代码

[英]How to debug Android native code on a real device

我在Android中的媒体后端(主要是Stagefrightplayer)遇到了一些麻烦,我想了解它为什么会抛出错误。 错误通常是设备特定的,因此在仿真器上进行调试是不够的。

例:

I/AwesomePlayer(  147): mConnectingDataSource->connect() returned -1004
V/MediaPlayerService(  147): [332] notify (0x272830, 100, 1, -1004)
E/MediaPlayer(24881): error (1, -1004)
E/MediaPlayer(24881): Error (1,-1004)
W/PlayerListener(24881): Received error: what = 1, extra = -1004

例2:

E/MediaPlayer(  941): error (1, -2147483648)

我也让玩家完全开玩笑并吐出了痕迹.txt。

有没有办法调试正在发生的事情,就像我调试Java代码一样? 谢谢。

你可以做很多事情。

如果你认为错误是在框架本身,那么获取源代码并挖掘http://source.android.com/

否则,Android的最佳调试器是DDMS,它可以与仿真器一起使用,也可以与真实设备一起使用。 http://developer.android.com/guide/developing/tools/ddms.html

dumpstate通过adb( http://developer.android.com/guide/developing/tools/adb.html )也会为您提供设备上发生的事情的完整快照,但是您很难获得准确的点当错误发生时。

虽然这仍然不会像GDB那样为您提供源代码级调试(或者我不确定您通常的调试Java代码的方式是什么意思)。

如果你真的把内核称为内核,那么你不再是Android了,而是更多的Linux世界,但我不认为你需要走得那么远。

如果您在使用特定的Android应用程序时遇到问题(这不是您自己的应用程序),我担心您运气不好。

对于MediaPlayer部分,Eclair的文件位于https://android.googlesource.com/platform/frameworks/base/+/eclair-release/media/java/android/media/MediaPlayer.java ,但找不到您放在那里的特定错误消息。

这不是直接回答您的问题,但这些信息可能对您有用。

因此,根据您的-1004错误代码,您尝试流式传输时出现I / O错误。 至于-2147483648错误代码,对你帮不了多少。 您必须查看媒体播放器的所有日志输出,以了解您为何获得该代码,因为它未定义。 我已经看到了解码器阻塞视频编码。

借鉴于:/frameworks / base / include / media / stagefright / MediaErrors.h

MEDIA_ERROR_BASE = -1000,

ERROR_ALREADY_CONNECTED = MEDIA_ERROR_BASE,
ERROR_NOT_CONNECTED     = MEDIA_ERROR_BASE - 1,
ERROR_UNKNOWN_HOST      = MEDIA_ERROR_BASE - 2,
ERROR_CANNOT_CONNECT    = MEDIA_ERROR_BASE - 3,
ERROR_IO                = MEDIA_ERROR_BASE - 4,
ERROR_CONNECTION_LOST   = MEDIA_ERROR_BASE - 5,
ERROR_MALFORMED         = MEDIA_ERROR_BASE - 7,
ERROR_OUT_OF_RANGE      = MEDIA_ERROR_BASE - 8,
ERROR_BUFFER_TOO_SMALL  = MEDIA_ERROR_BASE - 9,
ERROR_UNSUPPORTED       = MEDIA_ERROR_BASE - 10,
ERROR_END_OF_STREAM     = MEDIA_ERROR_BASE - 11,

即使您无法在内核级别进行调试,将隐藏的错误号跟踪到正确的头文件(以及描述性定义)仍然很有用。

-1004表示ERROR_IO ,可以在MediaErrors.h找到:
https://android.googlesource.com/platform/frameworks/base/+/eclair-release/include/media/stagefright/MediaErrors.h#32

-2147483648可能是UNKNOWN_ERROR ,它可以在Errors.h找到:
https://android.googlesource.com/platform/frameworks/base/+/eclair-release/include/utils/Errors.h#49

正如您在Errors.h中看到的Errors.h ,它包含errno.h ,其中包含内核级错误代码/kernel/include/asm-generic/errno.h。

例如,如果connect()返回错误代码-110,您将知道它是因为超时,因为它被定义为:

#define ETIMEDOUT       110     /* Connection timed out */

远程调试(目标上的gdbserver +主机上的gdb)可用于逐步执行在真实硬件上运行的C / C ++用户态代码。 它提供所有“常用”选项,如断点,回溯,视图/设置变量,跟踪点。

有关详细信息,请查看Android构建系统的'gdbclient'shell函数,预构建eabi gdb以及DDD或其他前端。 Eclipse应该没问题。

您可以通过几种不同的方式完成此操作。 首先,您需要找到要调试的服务,如Java框架服务,如system_server或纯本机应用程序,如surfaceflinger。

如果是纯本机服务,请查看调试android平台本机应用程序文章。

如果服务是在system_server进程中托管的Java代码,请查看调试Android Java框架服务文章。

如果要调试的代码是由Java服务通过JNI隐式加载的本机库,请检查调试Android框架本机库文章。

虽然Android确实支持远程GDB会话,但这可能不适用于内核端代码。 最好的办法是使用JTAG连接,该连接可用于执行停止模式调试。 由于停止模式调试有效地停止了CPU的执行,您可能会发现这会导致看门狗定时器出现问题。

或者,将跟踪插入内核代码可能更容易。

除非您想在程序集级别进行调试,否则您可能必须自己构建内核并启用调试+调试符号。 我认为小型设备中的大多数内核都会在默认情况下避免这样做,因为它会使内核变得更大。 那时你可以启用内核调试器......

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM