Android FrameWork学习(二)Android系统源码调试

通过上一篇 Android FrameWork学习(一)Android 7.0系统源码下载\编译 我们了解了如何进行系统源码的下载和编译工作。

为了更进一步地学习跟研究 Android 系统源码,今天我们来讲讲如何进行 Android 系统源码的调试,只有学会了如何进行系统源码的调试,才能帮助我们更高效地阅读跟理解源码。

我们知道,Android Framework 的代码主要由Java、C\C++等代码组成,因此,对于系统源码的调试,我们这里将其分为了两部分

  • Java 相关代码的调试
  • C\C++ Native 相关代码的调试

一、Java 相关代码的调试

对于 Java 相关代码的调试,这里我们主要使用 Android Studio 开发工具来进行。

导入源码到 Android Studio

要在 Android Studio 中调试源码,那第一步自然是导入系统源码到 Android Studio 中了。

1. 编译 idegen

对于 Android 源码的导入, Google 官方给我们提供了一个很方便的工具 idegen

它位于我们所下载的系统源码路径中:

developement/tools/idegen

引用 README 的一句话

IDEGen automatically generates Android IDE configurations for IntelliJ IDEA
and Eclipse.

idegen 工具会自动生成针对 Android 开发工具(Android Studio和Eclipse)的配置文件。

既然如此,那我们就来使用 idegen 工具生成导入源码所需的配置文件。

首先打开命令行工具,cd 进入到源码路径下,

执行如下指令:

#初始化命令工具
soruce build/envsetup.sh 
#编译 idegen 模块,生成idegen.jar
mmm development/tools/idegen/
#生成针对 Android 开发工具的配置文件 
sudo ./development/tools/idegen/idegen.sh

在执行完上述指令后,会在源码路径下生成下面三个文件

Paste_Image.png

android.ipr:工程相关的设置,比如编译器配置、入口,相关的libraries等。

android.iml:描述了modules,比如modules的路径,依赖关系等。

android.iws:包含了一些个人工作区的设置。

2. 导入源码

接下来我们可以开始导入源码了.

如果你是第一次导入源码, Android Studio 可能需要占用大量的内存,我们需要设置下我们的 VM 选项。

Linux 设备的话在 Android Studio 的 bin/studio64.vmoptions 文件中添加-Xms748m -Xmx748m

如果你使用的是 Mac ,那么在 AS 目录的 Contents/Info.plist 目录中进行添加。

由于 Android 的系统源码非常庞大,一次性导入 Android Studio 的话需要加载非常长的时间

因此,在正式开始导入前,我们可以打开 android.iml 文件根据自己需要调整要加载的源码。

Paste_Image.png

这里 <excludeFolder> 表示不需要加载的目录,我们根据自己的需要使用 <excludeFolder> 标签添加对应的目录地址即可。

接着,选择 File -> open 选中 android.ipr 文件,打开

Paste_Image.png

这时 Android Studio 就会开始加载源码了

在没有添加修改 <excludeFolder> 的情况下,这个加载的时间会比较长,经过一段时间的等待后,代码就加载完毕了,如图:

Paste_Image.png

这里红色的目录代表被 exclude 排除了,代码加载 scan index 的时候会过滤掉该目录。

在加载完源码后,我们也可以在 Project Structure 中的 Module 选项中右键 exclude 来排除不需要加载的源码目录,如图:

Paste_Image.png
Paste_Image.png

3. 配置代码依赖,确保代码跳转正确

为了阅读和调试代码的时候能够保证代码跳转正确,我们需要配置下相关依赖。

首先是 AOSP 源码的跳转,我们通过 File -> Project Structure 打开 Module,然后选中 Dependencies, 保留 JDK 跟 Module Source 项,并添加源码的 external 和 frameworks 依赖,如图:

Paste_Image.png

然后是 SDK 的设置,确保关联对应版本的 SDK 于系统版本一直

Paste_Image.png

开始调试源码

调试前要设置 Project 的 SDK , File -> Project 下打开 Project Structure,选中 Project 设置对应版本的 SDK,于系统版本一致:

Paste_Image.png

确保 Android Studio 允许 ADB 调试

Paste_Image.png

接着我们参照上一篇文章中讲的方法打开 Android 模拟器

此时点击 Android Studio 工具栏的 attach debugger to Android process 按钮,会打开 Choose Process 窗口,我们根据自己需要调试的代码选择对应的进程:

Paste_Image.png

这里假设我们要调试 Android 自带浏览器的源码,如图,我们在它的入口文件 WebViewBrowserActivity 中的 loadUrlFromUrlBar 方法中打上断点。

Paste_Image.png

点击 WebViewBrowser 打开 app


Paste_Image.png

打开之后,点击 attach to Android process 按钮打开 choose Process,可以看到 webViewBrowser 运行的进程,选中,ok

Paste_Image.png

然后我们在 app 的 url 输入栏输入 网址进行跳转

Paste_Image.png
Paste_Image.png

如图所示我们可以看到,代码成功进入了断点,然后我们就可以随心所欲地调试我们想要的调试的 Java 代码了。


二、Native C\C++ 相关代码调试

对于 Framework Native 代码,我们这里使用 GDB 工具来进行调试。

什么是 GDB 呢?

它是一款 GNU 项目调试工具,它的功能非常强大,可以用来调试 C 、C++、Object-C、Pascal 等语言编写的项目。

对于使用习惯了可视化 IDE 的同学们来说,它最大的缺点可能就是它不支持图形化了

但是 GDB 提供的指令非常灵活,通过指令我们

  • 可以随心所欲地启动程序,
  • 可以根据自己的需要设置断点,
  • 可以查看断点处的变量,代码信息
  • 可以查看程序运行的调用栈

一旦你熟悉了它,你便可以玩得飞起!

一般情况下,使用 gdb 来调试 Android 源码需要在 Android 设备上安装 gbdserver attach 关联我们需要调试的进程,再使用 gdb 指令去连接 gdbserver 进行调试

不过官方给我们提供了 gdbclient 工具,可以让我们方便地进行 gdb 调试。

开始 GDB 调试

这里我们就基于 gdbclient 来进行实际的 gdb 调试演示:

跟上面 Java 调试一样,我们这里还是以系统自带的浏览器为例。

1. 点击启动图标打开浏览器 app:

Paste_Image.png

2. 打开一个命令行终端,cd 进入到系统源码目录(我的源码路径为 aosp),初始化命令工具:

#进入源码路径
cd aosp
#初始化命令工具
source build/envsetup.sh
#选择编译的源码的版本,参考上一篇文章
lunch
初始化命令工具
Paste_Image.png

3. 通过 adb 指令来查找要调试进程的 PID

# 通过 shell ps 指令查找相关进程,grep 搜索过滤 webview 进程
adb shell ps -A | grep webview
Paste_Image.png

如图,2157 为系统自带浏览器 app 所在进程的 PID

4. 使用 gdbclient <app pid> 命令工具启用 gdb 调试 PID 对应进程

# gdbclient <app pid> 可以启用 gdb 调试对应 PID 进程
gdbclient 2157
Paste_Image.png

等待进入 gdb 调试命令界面

Paste_Image.png

5. 使用 GDB b 命令打断点

在 gdb 指令中,我们使用b <代码文件>:行号 来设置断点.

这里我们选择 frameworks/base/libs/hwui/renderthread/DrawFrameTask.cpp 代码文件的 drawFrame 方法打上断点,位于文件 71 行:

Paste_Image.png

该方法主要用于绘制帧,当浏览器 app 的界面发生变化时会触发该方法。

我们输入设置断点命令:

b frameworks/base/libs/hwui/renderthread/DrawFrameTask.cpp:71
Paste_Image.png

输入指令后显示
Breakpoint 2 at 0x7f69e9892c11: file frameworks/base/libs/hwui/renderthread/DrawFrameTask.cpp, line 71.
说明我们的断点设置成功了。

6. 在命令行输入c 开始监听

Paste_Image.png

c 即 continue,此时界面上出现 Continuing 说明开始监听进程了

我们点开模拟器,随意操作,触发界面变化时,便会进入绘制帧的代码断点了:

Paste_Image.png

如图,显示进入断点,这样代表我们的代码调试成功了。


这里我们只是演示了一个大概流程,

gdb 代码的调试需要你对源码有一定的熟悉,知道哪个进程会调用哪个文件方法。

同时,我们还需要熟悉 gdb 的各种命令,这里给大家推荐一篇不错的入门文章,可以快速入门:

GDB十分钟教程

这里补充一点,如果你希望在某个进程启动时就监听,可以使用下面的指令关联目录,得到 pid,再通过 gdbclient 来进行调试

adb shell gdbserver :5039 /system/bin/my_test_app
Process my_test_app created; pid = 3460
Listening on port 5039
gdbclient <app pid>

如果你希望通过 Android Studio 来调试 Framework 的 C\C++ 代码的话,也可以参考下面的两篇文章,不过个人觉得这种方法有一定的局限性。

如何调试Android Native Framework

用Android Studio调试Framework层代码


结语

正如文章开头所说,只有学会了如何调试 Framework 源码,才能帮助我们更好地学习 Android Framework,希望这篇文章能给大家一些帮助,如果有更好地调试方法,欢迎大家给我留言咯!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,185评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,445评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,684评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,564评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,681评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,874评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,025评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,761评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,217评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,545评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,694评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,351评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,988评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,778评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,007评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,427评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,580评论 2 349

推荐阅读更多精彩内容