本文将介绍有关Android Init进程相关的一些知识。
系统版本: Ubuntu 22.04 lts
AOSP分支: android-14.0.0_r28
什么是Init进程
当我们在adb shel中输入ps -A
列出所有进程时,我们可以看到有一个名字为init
,pid为1的进程,
再往下看,可以看到很多ppid为1的进程,这意味着这些进程都是从init
进程中fork
出来的,包括我们最熟悉的zygote进程:
可以说,init
进程是我们整个Android世界的起始点,那么了解它的作用,就变得十分重要了。
谁启动了Init进程
在Android设备启动时,第一个步骤便是bootloader
,这部分Android官网有相关的介绍:
之后,就会开始启动我们的Linux Kernel,当Linux Kernel启动完成,就会启动我们的Init进程了,关于这部分,可以在Android Linux Kernel源码中,查看/init/main.c
的rest_init
方法,这里即是Linux Kernel启动我们的Init进程的地方:
如果不知道如何查看Android Linux Kernel源码的,可以看这篇文章:
Init进程做了什么
Init相关的代码主要集中在/system/core/init
,入口方法是这个目录下的main.cpp
的main
方法,打开之后,我们可以看到如下代码,这就是init进程的初始化逻辑的主干部分:
Init进程执行顺序
从上面的代码我们可以看到,我们的Init貌似有好几个步骤,根据参数的不同,要执行不同的逻辑,那么这些步骤都是干什么的?
其实我们可以打开/system/core/init/README.md
,可以看到Google已经为我们做了大致的解释:
从这里我们可以知道,这个main
方法会被传入不同的参数反复执行,执行的顺序大概是first stage, selinux stage和second stage。
解析RC文件并启动服务
first stage从文档中可以看到,工作和逻辑较为繁杂,对这部分有兴趣的可以查看/system/core/init/first_stage_init.cpp
,这里不做过多赘述:
selinux stage,则是为Android加载selinux,关于Android对SELinux的支持,可以看这里:
最后,就是second stage了,文档里面也提到了,在second stage,我们就会开始解析rc文件并根据rc文件启动一系列进程,打开/system/core/init/init.cpp
,就可以看到我们的second stage的主要处理方法了:
在这个方法中查找,我们可以看到LoadBootScripts
方法被调用,进入此方法,我们就可以看到,在这里,Android完成了对rc文件的读取以及解析工作:
最后,就是开始启动rc文件中的early-init
和init
了:
下面是/system/core/rootdir/init.rc
文件中early-init
和init
两个trigger的定义:
如果不知道如何阅读rc文件,可以看这里:
至此,可以说Android已经开始基于rc文件要启动起来了。